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