diff --git a/packages/wyatt_crud_bloc/example/.gitignore b/packages/wyatt_crud_bloc/example/.gitignore
new file mode 100644
index 00000000..c3b6d4c7
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/.gitignore
@@ -0,0 +1,48 @@
+# Miscellaneous
+*.class
+*.log
+*.pyc
+*.swp
+.DS_Store
+.atom/
+.buildlog/
+.history
+.svn/
+
+# IntelliJ related
+*.iml
+*.ipr
+*.iws
+.idea/
+
+# The .vscode folder contains launch configuration and tasks you configure in
+# VS Code which you may wish to be included in version control, so this line
+# is commented out by default.
+#.vscode/
+
+# Flutter/Dart/Pub related
+**/doc/api/
+**/ios/Flutter/.last_build_id
+.dart_tool/
+.flutter-plugins
+.flutter-plugins-dependencies
+.packages
+.pub-cache/
+.pub/
+/build/
+
+# Web related
+lib/generated_plugin_registrant.dart
+
+# Symbolication related
+app.*.symbols
+
+# Obfuscation related
+app.*.map.json
+
+# Android Studio will place build artifacts here
+/android/app/debug
+/android/app/profile
+/android/app/release
+
+/lib/firebase_options.dart
\ No newline at end of file
diff --git a/packages/wyatt_crud_bloc/example/.metadata b/packages/wyatt_crud_bloc/example/.metadata
new file mode 100644
index 00000000..3c3e4b52
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/.metadata
@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5464c5bac742001448fe4fc0597be939379f88ea
+  channel: stable
+
+project_type: app
diff --git a/packages/wyatt_crud_bloc/example/README.md b/packages/wyatt_crud_bloc/example/README.md
new file mode 100644
index 00000000..3b20831a
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/README.md
@@ -0,0 +1,16 @@
+# crud_bloc_example
+
+A new Flutter project.
+
+## Getting Started
+
+This project is a starting point for a Flutter application.
+
+A few resources to get you started if this is your first Flutter project:
+
+- [Lab: Write your first Flutter app](https://flutter.dev/docs/get-started/codelab)
+- [Cookbook: Useful Flutter samples](https://flutter.dev/docs/cookbook)
+
+For help getting started with Flutter, view our
+[online documentation](https://flutter.dev/docs), which offers tutorials,
+samples, guidance on mobile development, and a full API reference.
diff --git a/packages/wyatt_crud_bloc/example/analysis_options.yaml b/packages/wyatt_crud_bloc/example/analysis_options.yaml
new file mode 100644
index 00000000..61b6c4de
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/analysis_options.yaml
@@ -0,0 +1,29 @@
+# This file configures the analyzer, which statically analyzes Dart code to
+# check for errors, warnings, and lints.
+#
+# The issues identified by the analyzer are surfaced in the UI of Dart-enabled
+# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be
+# invoked from the command line by running `flutter analyze`.
+
+# The following line activates a set of recommended lints for Flutter apps,
+# packages, and plugins designed to encourage good coding practices.
+include: package:flutter_lints/flutter.yaml
+
+linter:
+  # The lint rules applied to this project can be customized in the
+  # section below to disable rules from the `package:flutter_lints/flutter.yaml`
+  # included above or to enable additional rules. A list of all available lints
+  # and their documentation is published at
+  # https://dart-lang.github.io/linter/lints/index.html.
+  #
+  # Instead of disabling a lint rule for the entire project in the
+  # section below, it can also be suppressed for a single line of code
+  # or a specific dart file by using the `// ignore: name_of_lint` and
+  # `// ignore_for_file: name_of_lint` syntax on the line or in the file
+  # producing the lint.
+  rules:
+    # avoid_print: false  # Uncomment to disable the `avoid_print` rule
+    # prefer_single_quotes: true  # Uncomment to enable the `prefer_single_quotes` rule
+
+# Additional information about this file can be found at
+# https://dart.dev/guides/language/analysis-options
diff --git a/packages/wyatt_crud_bloc/example/android/.gitignore b/packages/wyatt_crud_bloc/example/android/.gitignore
new file mode 100644
index 00000000..6f568019
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/.gitignore
@@ -0,0 +1,13 @@
+gradle-wrapper.jar
+/.gradle
+/captures/
+/gradlew
+/gradlew.bat
+/local.properties
+GeneratedPluginRegistrant.java
+
+# Remember to never publicly share your keystore.
+# See https://flutter.dev/docs/deployment/android#reference-the-keystore-from-the-app
+key.properties
+**/*.keystore
+**/*.jks
diff --git a/packages/wyatt_crud_bloc/example/android/app/build.gradle b/packages/wyatt_crud_bloc/example/android/app/build.gradle
new file mode 100644
index 00000000..8ddaae13
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/build.gradle
@@ -0,0 +1,71 @@
+def localProperties = new Properties()
+def localPropertiesFile = rootProject.file('local.properties')
+if (localPropertiesFile.exists()) {
+    localPropertiesFile.withReader('UTF-8') { reader ->
+        localProperties.load(reader)
+    }
+}
+
+def flutterRoot = localProperties.getProperty('flutter.sdk')
+if (flutterRoot == null) {
+    throw new GradleException("Flutter SDK not found. Define location with flutter.sdk in the local.properties file.")
+}
+
+def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
+if (flutterVersionCode == null) {
+    flutterVersionCode = '1'
+}
+
+def flutterVersionName = localProperties.getProperty('flutter.versionName')
+if (flutterVersionName == null) {
+    flutterVersionName = '1.0'
+}
+
+apply plugin: 'com.android.application'
+// START: FlutterFire Configuration
+apply plugin: 'com.google.gms.google-services'
+// END: FlutterFire Configuration
+apply plugin: 'kotlin-android'
+apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
+
+android {
+    compileSdkVersion flutter.compileSdkVersion
+
+    compileOptions {
+        sourceCompatibility JavaVersion.VERSION_1_8
+        targetCompatibility JavaVersion.VERSION_1_8
+    }
+
+    kotlinOptions {
+        jvmTarget = '1.8'
+    }
+
+    sourceSets {
+        main.java.srcDirs += 'src/main/kotlin'
+    }
+
+    defaultConfig {
+        // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
+        applicationId "com.example.crud_bloc_example"
+        minSdkVersion 21
+        targetSdkVersion flutter.targetSdkVersion
+        versionCode flutterVersionCode.toInteger()
+        versionName flutterVersionName
+    }
+
+    buildTypes {
+        release {
+            // TODO: Add your own signing config for the release build.
+            // Signing with the debug keys for now, so `flutter run --release` works.
+            signingConfig signingConfigs.debug
+        }
+    }
+}
+
+flutter {
+    source '../..'
+}
+
+dependencies {
+    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
+}
diff --git a/packages/wyatt_crud_bloc/example/android/app/google-services.json b/packages/wyatt_crud_bloc/example/android/app/google-services.json
new file mode 100644
index 00000000..cd75a682
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/google-services.json
@@ -0,0 +1,135 @@
+{
+  "project_info": {
+    "project_number": "136771801992",
+    "firebase_url": "https://tchat-beta.firebaseio.com",
+    "project_id": "tchat-beta",
+    "storage_bucket": "tchat-beta.appspot.com"
+  },
+  "client": [
+    {
+      "client_info": {
+        "mobilesdk_app_id": "1:136771801992:android:4ff419f07afdad6097203d",
+        "android_client_info": {
+          "package_name": "com.example.authentication_bloc_example"
+        }
+      },
+      "oauth_client": [
+        {
+          "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+          "client_type": 3
+        }
+      ],
+      "api_key": [
+        {
+          "current_key": "AIzaSyAYS14uXupkS158Q5QAFP1864UrUN_yDSk"
+        }
+      ],
+      "services": {
+        "appinvite_service": {
+          "other_platform_oauth_client": [
+            {
+              "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+              "client_type": 3
+            },
+            {
+              "client_id": "136771801992-e585bm1n9b3lv89t4phrl9u0glsg52ua.apps.googleusercontent.com",
+              "client_type": 2,
+              "ios_info": {
+                "bundle_id": "com.example.example"
+              }
+            }
+          ]
+        }
+      }
+    },
+    {
+      "client_info": {
+        "mobilesdk_app_id": "1:136771801992:android:8482c9b90bc29de697203d",
+        "android_client_info": {
+          "package_name": "com.example.crud_bloc_example"
+        }
+      },
+      "oauth_client": [
+        {
+          "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+          "client_type": 3
+        }
+      ],
+      "api_key": [
+        {
+          "current_key": "AIzaSyAYS14uXupkS158Q5QAFP1864UrUN_yDSk"
+        }
+      ],
+      "services": {
+        "appinvite_service": {
+          "other_platform_oauth_client": [
+            {
+              "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+              "client_type": 3
+            },
+            {
+              "client_id": "136771801992-e585bm1n9b3lv89t4phrl9u0glsg52ua.apps.googleusercontent.com",
+              "client_type": 2,
+              "ios_info": {
+                "bundle_id": "com.example.example"
+              }
+            }
+          ]
+        }
+      }
+    },
+    {
+      "client_info": {
+        "mobilesdk_app_id": "1:136771801992:android:d20e0361057e815197203d",
+        "android_client_info": {
+          "package_name": "com.example.example"
+        }
+      },
+      "oauth_client": [
+        {
+          "client_id": "136771801992-n2pq8oqutvrqj58e05hbavvc7n1jdfjb.apps.googleusercontent.com",
+          "client_type": 1,
+          "android_info": {
+            "package_name": "com.example.example",
+            "certificate_hash": "5d8790309e13b68c35e5d4d8437c35c6d15e6131"
+          }
+        },
+        {
+          "client_id": "136771801992-pj9s88ao7d99pnp4ace75nj1abtmoq7e.apps.googleusercontent.com",
+          "client_type": 1,
+          "android_info": {
+            "package_name": "com.example.example",
+            "certificate_hash": "bc8faf31c0b0085dc358b2319684e10c93f42719"
+          }
+        },
+        {
+          "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+          "client_type": 3
+        }
+      ],
+      "api_key": [
+        {
+          "current_key": "AIzaSyAYS14uXupkS158Q5QAFP1864UrUN_yDSk"
+        }
+      ],
+      "services": {
+        "appinvite_service": {
+          "other_platform_oauth_client": [
+            {
+              "client_id": "136771801992-ncuib3rbu7p4ro4eo5su4vaudn2u4qrv.apps.googleusercontent.com",
+              "client_type": 3
+            },
+            {
+              "client_id": "136771801992-e585bm1n9b3lv89t4phrl9u0glsg52ua.apps.googleusercontent.com",
+              "client_type": 2,
+              "ios_info": {
+                "bundle_id": "com.example.example"
+              }
+            }
+          ]
+        }
+      }
+    }
+  ],
+  "configuration_version": "1"
+}
\ No newline at end of file
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/debug/AndroidManifest.xml b/packages/wyatt_crud_bloc/example/android/app/src/debug/AndroidManifest.xml
new file mode 100644
index 00000000..e9dd957b
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/debug/AndroidManifest.xml
@@ -0,0 +1,7 @@
+
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/AndroidManifest.xml b/packages/wyatt_crud_bloc/example/android/app/src/main/AndroidManifest.xml
new file mode 100644
index 00000000..082c9cb4
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/AndroidManifest.xml
@@ -0,0 +1,34 @@
+
+   
+        
+            
+            
+            
+                
+                
+            
+        
+        
+        
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/kotlin/com/example/crud_bloc_example/MainActivity.kt b/packages/wyatt_crud_bloc/example/android/app/src/main/kotlin/com/example/crud_bloc_example/MainActivity.kt
new file mode 100644
index 00000000..cb1df98b
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/kotlin/com/example/crud_bloc_example/MainActivity.kt
@@ -0,0 +1,6 @@
+package com.example.crud_bloc_example
+
+import io.flutter.embedding.android.FlutterActivity
+
+class MainActivity: FlutterActivity() {
+}
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable-v21/launch_background.xml b/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable-v21/launch_background.xml
new file mode 100644
index 00000000..f74085f3
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable-v21/launch_background.xml
@@ -0,0 +1,12 @@
+
+
+
+     
+
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable/launch_background.xml b/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable/launch_background.xml
new file mode 100644
index 00000000..304732f8
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/res/drawable/launch_background.xml
@@ -0,0 +1,12 @@
+
+
+
+     
+
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 00000000..db77bb4b
Binary files /dev/null and b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
new file mode 100644
index 00000000..17987b79
Binary files /dev/null and b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 00000000..09d43914
Binary files /dev/null and b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 00000000..d5f1c8d3
Binary files /dev/null and b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 00000000..4d6372ee
Binary files /dev/null and b/packages/wyatt_crud_bloc/example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/values-night/styles.xml b/packages/wyatt_crud_bloc/example/android/app/src/main/res/values-night/styles.xml
new file mode 100644
index 00000000..3db14bb5
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/res/values-night/styles.xml
@@ -0,0 +1,18 @@
+
+
+    
+    
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/main/res/values/styles.xml b/packages/wyatt_crud_bloc/example/android/app/src/main/res/values/styles.xml
new file mode 100644
index 00000000..d460d1e9
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/main/res/values/styles.xml
@@ -0,0 +1,18 @@
+
+
+    
+    
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/app/src/profile/AndroidManifest.xml b/packages/wyatt_crud_bloc/example/android/app/src/profile/AndroidManifest.xml
new file mode 100644
index 00000000..e9dd957b
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/app/src/profile/AndroidManifest.xml
@@ -0,0 +1,7 @@
+
+    
+    
+
diff --git a/packages/wyatt_crud_bloc/example/android/build.gradle b/packages/wyatt_crud_bloc/example/android/build.gradle
new file mode 100644
index 00000000..111a0724
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/build.gradle
@@ -0,0 +1,34 @@
+buildscript {
+    ext.kotlin_version = '1.6.10'
+    repositories {
+        google()
+        mavenCentral()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:4.1.0'
+        // START: FlutterFire Configuration
+        classpath 'com.google.gms:google-services:4.3.10'
+        // END: FlutterFire Configuration
+        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
+    }
+}
+
+allprojects {
+    repositories {
+        google()
+        mavenCentral()
+    }
+}
+
+rootProject.buildDir = '../build'
+subprojects {
+    project.buildDir = "${rootProject.buildDir}/${project.name}"
+}
+subprojects {
+    project.evaluationDependsOn(':app')
+}
+
+task clean(type: Delete) {
+    delete rootProject.buildDir
+}
diff --git a/packages/wyatt_crud_bloc/example/android/gradle.properties b/packages/wyatt_crud_bloc/example/android/gradle.properties
new file mode 100644
index 00000000..94adc3a3
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/gradle.properties
@@ -0,0 +1,3 @@
+org.gradle.jvmargs=-Xmx1536M
+android.useAndroidX=true
+android.enableJetifier=true
diff --git a/packages/wyatt_crud_bloc/example/android/gradle/wrapper/gradle-wrapper.properties b/packages/wyatt_crud_bloc/example/android/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 00000000..bc6a58af
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,6 @@
+#Fri Jun 23 08:50:38 CEST 2017
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-6.7-all.zip
diff --git a/packages/wyatt_crud_bloc/example/android/settings.gradle b/packages/wyatt_crud_bloc/example/android/settings.gradle
new file mode 100644
index 00000000..44e62bcf
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/android/settings.gradle
@@ -0,0 +1,11 @@
+include ':app'
+
+def localPropertiesFile = new File(rootProject.projectDir, "local.properties")
+def properties = new Properties()
+
+assert localPropertiesFile.exists()
+localPropertiesFile.withReader("UTF-8") { reader -> properties.load(reader) }
+
+def flutterSdkPath = properties.getProperty("flutter.sdk")
+assert flutterSdkPath != null, "flutter.sdk not set in local.properties"
+apply from: "$flutterSdkPath/packages/flutter_tools/gradle/app_plugin_loader.gradle"
diff --git a/packages/wyatt_crud_bloc/example/lib/app.dart b/packages/wyatt_crud_bloc/example/lib/app.dart
new file mode 100644
index 00000000..4916f9d5
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/lib/app.dart
@@ -0,0 +1,168 @@
+// Copyright (C) 2022 WYATT GROUP
+// Please see the AUTHORS file for details.
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see .
+
+import 'dart:math';
+
+import 'package:crud_bloc_example/models.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_bloc/flutter_bloc.dart';
+import 'package:wyatt_crud_bloc/wyatt_crud_bloc.dart';
+
+class MyApp extends StatelessWidget {
+  const MyApp({Key? key}) : super(key: key);
+
+  // This widget is the root of your application.
+  @override
+  Widget build(BuildContext context) {
+    final _userRepository = CrudRepositoryFirestore(
+      'users_crud',
+      UserFirestore.parser(),
+    );
+
+    final _userCubit = CrudCubit(_userRepository);
+
+    return RepositoryProvider>(
+      create: (context) => _userRepository,
+      child: BlocProvider>(
+        create: (context) => _userCubit,
+        child: MaterialApp(
+          title: 'Flutter Demo',
+          theme: ThemeData(
+            primarySwatch: Colors.blue,
+          ),
+          home: const MyHomePage(),
+        ),
+      ),
+    );
+  }
+}
+
+class MyHomePage extends StatelessWidget {
+  const MyHomePage({Key? key}) : super(key: key);
+
+  @override
+  Widget build(BuildContext context) {
+    return Scaffold(
+      appBar: AppBar(
+        title: const Text('Flutter Demo Home Page'),
+      ),
+      body: SingleChildScrollView(
+        child: Column(
+          children: [
+            const SizedBox(height: 20),
+            const Text("Data:"),
+            CrudStreamBuilder(
+              onError: (context, state) => const SizedBox.shrink(),
+              onLoading: (context, state) => const Text("Loading..."),
+              onStream: (context, data) {
+                return ListView.builder(
+                  shrinkWrap: true,
+                  itemCount: data.length,
+                  itemBuilder: (context, index) {
+                    final user = data.elementAt(index);
+                    return ListTile(
+                      title: Text(user?.name ?? 'Error'),
+                      subtitle: Text(user?.id ?? 'Error'),
+                      onTap: () {
+                        context.read>().get(
+                              (user?.id)!,
+                            );
+                      },
+                      onLongPress: () {
+                        context.read>().delete(
+                              (user?.id)!,
+                            );
+                      },
+                    );
+                  },
+                );
+              },
+            ),
+            const SizedBox(height: 20),
+            BlocBuilder, CrudState>(
+              builder: (context, state) {
+                return Center(
+                  child: Text(
+                    state.toString(),
+                    textAlign: TextAlign.center,
+                    style: const TextStyle(
+                      fontSize: 20,
+                      color: Colors.grey,
+                    ),
+                  ),
+                );
+              },
+            ),
+            const SizedBox(height: 20),
+            ElevatedButton(
+              onPressed: () {
+                final r = Random().nextInt(1000);
+                context.read>().create(
+                      UserFirestore(
+                          name: 'Wyatt $r',
+                          email: '$r@wyattapp.io',
+                          phone: '06$r'),
+                    );
+              },
+              child: const Text("Create"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context.read>().deleteAll();
+              },
+              child: const Text("DeleteAll"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context.read>().getAll();
+              },
+              child: const Text("GetAll"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context
+                    .read>()
+                    .query([LimitQueryFirestore(1)]);
+              },
+              child: const Text("Query"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context.read>().streamOf();
+              },
+              child: const Text("StreamOf"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context
+                    .read>()
+                    .updateAll({'updated': DateTime.now()});
+              },
+              child: const Text("UpdateAll"),
+            ),
+            ElevatedButton(
+              onPressed: () {
+                context.read>().reset();
+              },
+              child: const Text("Reset"),
+            ),
+            const SizedBox(height: 20),
+          ],
+        ),
+      ),
+    );
+  }
+}
diff --git a/packages/wyatt_crud_bloc/example/lib/main.dart b/packages/wyatt_crud_bloc/example/lib/main.dart
new file mode 100644
index 00000000..b3523faa
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/lib/main.dart
@@ -0,0 +1,29 @@
+// Copyright (C) 2022 WYATT GROUP
+// Please see the AUTHORS file for details.
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see .
+
+import 'package:crud_bloc_example/app.dart';
+import 'package:crud_bloc_example/firebase_options.dart';
+import 'package:firebase_core/firebase_core.dart';
+import 'package:flutter/material.dart';
+
+Future main() async {
+  WidgetsFlutterBinding.ensureInitialized();
+  await Firebase.initializeApp(
+    options: DefaultFirebaseOptions.currentPlatform,
+  );
+
+  runApp(const MyApp());
+}
diff --git a/packages/wyatt_crud_bloc/example/lib/models.dart b/packages/wyatt_crud_bloc/example/lib/models.dart
new file mode 100644
index 00000000..e3654bb9
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/lib/models.dart
@@ -0,0 +1,66 @@
+// Copyright (C) 2022 WYATT GROUP
+// Please see the AUTHORS file for details.
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see .
+
+import 'package:cloud_firestore/cloud_firestore.dart';
+import 'package:wyatt_crud_bloc/wyatt_crud_bloc.dart';
+
+class UserFirestore extends Model {
+  @override
+  String? id;
+
+  String? name;
+  String? email;
+  String? phone;
+
+  UserFirestore({
+    required this.name,
+    required this.email,
+    required this.phone,
+    this.id,
+  });
+  UserFirestore._();
+
+  factory UserFirestore.parser() {
+    return UserFirestore._();
+  }
+
+  @override
+  UserFirestore? from(DocumentSnapshot? object) {
+    if (object == null) return null;
+    if (object.exists) {
+      return UserFirestore(
+        id: object.id,
+        name: (object.data() as Map?)!['name'] as String,
+        email: (object.data() as Map?)!['email'] as String,
+        phone: (object.data() as Map?)!['phone'] as String,
+      );
+    }
+    return null;
+  }
+
+  @override
+  Map toMap() {
+    return {
+      'name': name ?? '',
+      'email': email ?? '',
+      'phone': phone ?? '',
+    };
+  }
+
+  @override
+  String toString() =>
+      'UserFirestore(id: $id, name: $name, email: $email, phone: $phone)';
+}
diff --git a/packages/wyatt_crud_bloc/example/pubspec.yaml b/packages/wyatt_crud_bloc/example/pubspec.yaml
new file mode 100644
index 00000000..1f4747e8
--- /dev/null
+++ b/packages/wyatt_crud_bloc/example/pubspec.yaml
@@ -0,0 +1,94 @@
+name: crud_bloc_example
+description: A new Flutter project.
+
+# The following line prevents the package from being accidentally published to
+# pub.dev using `flutter pub publish`. This is preferred for private packages.
+publish_to: 'none' # Remove this line if you wish to publish to pub.dev
+
+# The following defines the version and build number for your application.
+# A version number is three numbers separated by dots, like 1.2.43
+# followed by an optional build number separated by a +.
+# Both the version and the builder number may be overridden in flutter
+# build by specifying --build-name and --build-number, respectively.
+# In Android, build-name is used as versionName while build-number used as versionCode.
+# Read more about Android versioning at https://developer.android.com/studio/publish/versioning
+# In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion.
+# Read more about iOS versioning at
+# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
+version: 1.0.0+1
+
+environment:
+  sdk: ">=2.16.2 <3.0.0"
+
+# Dependencies specify other packages that your package needs in order to work.
+# To automatically upgrade your package dependencies to the latest versions
+# consider running `flutter pub upgrade --major-versions`. Alternatively,
+# dependencies can be manually updated by changing the version numbers below to
+# the latest version available on pub.dev. To see which dependencies have newer
+# versions available, run `flutter pub outdated`.
+dependencies:
+  flutter:
+    sdk: flutter
+
+  cloud_firestore: ^3.1.12
+  firebase_core: ^1.14.1
+  flutter_bloc: ^8.0.1
+  wyatt_crud_bloc:
+    path: "../"
+
+  # The following adds the Cupertino Icons font to your application.
+  # Use with the CupertinoIcons class for iOS style icons.
+  cupertino_icons: ^1.0.2
+
+dev_dependencies:
+  flutter_test:
+    sdk: flutter
+
+  # The "flutter_lints" package below contains a set of recommended lints to
+  # encourage good coding practices. The lint set provided by the package is
+  # activated in the `analysis_options.yaml` file located at the root of your
+  # package. See that file for information about deactivating specific lint
+  # rules and activating additional ones.
+  flutter_lints: ^1.0.0
+
+# For information on the generic Dart part of this file, see the
+# following page: https://dart.dev/tools/pub/pubspec
+
+# The following section is specific to Flutter.
+flutter:
+
+  # The following line ensures that the Material Icons font is
+  # included with your application, so that you can use the icons in
+  # the material Icons class.
+  uses-material-design: true
+
+  # To add assets to your application, add an assets section, like this:
+  # assets:
+  #   - images/a_dot_burr.jpeg
+  #   - images/a_dot_ham.jpeg
+
+  # An image asset can refer to one or more resolution-specific "variants", see
+  # https://flutter.dev/assets-and-images/#resolution-aware.
+
+  # For details regarding adding assets from package dependencies, see
+  # https://flutter.dev/assets-and-images/#from-packages
+
+  # To add custom fonts to your application, add a fonts section here,
+  # in this "flutter" section. Each entry in this list should have a
+  # "family" key with the font family name, and a "fonts" key with a
+  # list giving the asset and other descriptors for the font. For
+  # example:
+  # fonts:
+  #   - family: Schyler
+  #     fonts:
+  #       - asset: fonts/Schyler-Regular.ttf
+  #       - asset: fonts/Schyler-Italic.ttf
+  #         style: italic
+  #   - family: Trajan Pro
+  #     fonts:
+  #       - asset: fonts/TrajanPro.ttf
+  #       - asset: fonts/TrajanPro_Bold.ttf
+  #         weight: 700
+  #
+  # For details regarding fonts from package dependencies,
+  # see https://flutter.dev/custom-fonts/#from-packages
diff --git a/packages/wyatt_crud_bloc/example/test/widget_test.dart b/packages/wyatt_crud_bloc/example/test/widget_test.dart
new file mode 100644
index 00000000..e69de29b