From 4dc992892f6662e7c7c8eec10d4c3aa6a1069671 Mon Sep 17 00:00:00 2001 From: Klemen Tusar Date: Fri, 28 Jun 2024 20:30:06 +0100 Subject: [PATCH] feat: packages/alice_objectbox (#205) * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: examples/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * docs: packages/alice_objectbox * docs: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox * feat: packages/alice_objectbox --- docs/usage.md | 55 ++ examples/alice_objectbox/.gitignore | 43 ++ examples/alice_objectbox/.metadata | 33 + examples/alice_objectbox/README.md | 3 + .../alice_objectbox/analysis_options.yaml | 10 + examples/alice_objectbox/android/.gitignore | 13 + .../alice_objectbox/android/app/build.gradle | 58 ++ .../android/app/src/debug/AndroidManifest.xml | 7 + .../android/app/src/main/AndroidManifest.xml | 45 ++ .../jhomlala/alice_objectbox/MainActivity.kt | 5 + .../res/drawable-v21/launch_background.xml | 12 + .../main/res/drawable/launch_background.xml | 12 + .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin 0 -> 544 bytes .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin 0 -> 442 bytes .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin 0 -> 721 bytes .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin 0 -> 1031 bytes .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin 0 -> 1443 bytes .../app/src/main/res/values-night/styles.xml | 18 + .../app/src/main/res/values/styles.xml | 18 + .../app/src/profile/AndroidManifest.xml | 7 + examples/alice_objectbox/android/build.gradle | 18 + .../alice_objectbox/android/gradle.properties | 3 + .../gradle/wrapper/gradle-wrapper.properties | 5 + .../alice_objectbox/android/settings.gradle | 25 + examples/alice_objectbox/ios/.gitignore | 34 + .../ios/Flutter/AppFrameworkInfo.plist | 26 + .../ios/Flutter/Debug.xcconfig | 2 + .../ios/Flutter/Release.xcconfig | 2 + examples/alice_objectbox/ios/Podfile | 44 ++ examples/alice_objectbox/ios/Podfile.lock | 78 ++ .../ios/Runner.xcodeproj/project.pbxproj | 634 ++++++++++++++++ .../contents.xcworkspacedata | 7 + .../xcshareddata/IDEWorkspaceChecks.plist | 8 + .../xcshareddata/WorkspaceSettings.xcsettings | 8 + .../xcshareddata/xcschemes/Runner.xcscheme | 98 +++ .../contents.xcworkspacedata | 7 + .../xcshareddata/IDEWorkspaceChecks.plist | 8 + .../xcshareddata/WorkspaceSettings.xcsettings | 8 + .../ios/Runner/AppDelegate.swift | 13 + .../AppIcon.appiconset/Contents.json | 122 ++++ .../Icon-App-1024x1024@1x.png | Bin 0 -> 10932 bytes .../AppIcon.appiconset/Icon-App-20x20@1x.png | Bin 0 -> 295 bytes .../AppIcon.appiconset/Icon-App-20x20@2x.png | Bin 0 -> 406 bytes .../AppIcon.appiconset/Icon-App-20x20@3x.png | Bin 0 -> 450 bytes .../AppIcon.appiconset/Icon-App-29x29@1x.png | Bin 0 -> 282 bytes .../AppIcon.appiconset/Icon-App-29x29@2x.png | Bin 0 -> 462 bytes .../AppIcon.appiconset/Icon-App-29x29@3x.png | Bin 0 -> 704 bytes .../AppIcon.appiconset/Icon-App-40x40@1x.png | Bin 0 -> 406 bytes .../AppIcon.appiconset/Icon-App-40x40@2x.png | Bin 0 -> 586 bytes .../AppIcon.appiconset/Icon-App-40x40@3x.png | Bin 0 -> 862 bytes .../AppIcon.appiconset/Icon-App-60x60@2x.png | Bin 0 -> 862 bytes .../AppIcon.appiconset/Icon-App-60x60@3x.png | Bin 0 -> 1674 bytes .../AppIcon.appiconset/Icon-App-76x76@1x.png | Bin 0 -> 762 bytes .../AppIcon.appiconset/Icon-App-76x76@2x.png | Bin 0 -> 1226 bytes .../Icon-App-83.5x83.5@2x.png | Bin 0 -> 1418 bytes .../LaunchImage.imageset/Contents.json | 23 + .../LaunchImage.imageset/LaunchImage.png | Bin 0 -> 68 bytes .../LaunchImage.imageset/LaunchImage@2x.png | Bin 0 -> 68 bytes .../LaunchImage.imageset/LaunchImage@3x.png | Bin 0 -> 68 bytes .../LaunchImage.imageset/README.md | 5 + .../Runner/Base.lproj/LaunchScreen.storyboard | 37 + .../ios/Runner/Base.lproj/Main.storyboard | 26 + .../alice_objectbox/ios/Runner/Info.plist | 49 ++ .../ios/Runner/Runner-Bridging-Header.h | 1 + .../ios/RunnerTests/RunnerTests.swift | 12 + examples/alice_objectbox/lib/main.dart | 197 +++++ examples/alice_objectbox/pubspec.yaml | 32 + packages/alice/lib/alice.dart | 41 +- packages/alice/lib/core/alice_core.dart | 253 +++---- .../alice/lib/core/alice_memory_storage.dart | 121 ++++ packages/alice/lib/core/alice_storage.dart | 36 + packages/alice/lib/core/alice_store.dart | 3 + .../alice/lib/helper/alice_save_helper.dart | 3 +- packages/alice/lib/model/alice_http_call.dart | 59 -- .../page/alice_call_details_page.dart | 2 +- .../page/alice_calls_list_page.dart | 30 +- .../widget/alice_inspector_screen.dart | 59 ++ .../alice/lib/ui/stats/alice_stats_page.dart | 2 +- packages/alice/lib/utils/curl.dart | 60 ++ packages/alice_objectbox/.gitignore | 29 + packages/alice_objectbox/CHANGELOG.md | 3 + packages/alice_objectbox/LICENSE | 201 +++++ packages/alice_objectbox/README.md | 4 + .../alice_objectbox/analysis_options.yaml | 10 + .../alice_objectbox/lib/alice_objectbox.dart | 137 ++++ .../lib/alice_objectbox_store.dart | 64 ++ .../alice_form_data_field_converter.dart | 22 + .../alice_form_data_file_converter.dart | 24 + .../lib/model/cached_alice_http_call.dart | 130 ++++ .../lib/model/cached_alice_http_error.dart | 47 ++ .../lib/model/cached_alice_http_request.dart | 133 ++++ .../lib/model/cached_alice_http_response.dart | 69 ++ .../alice_objectbox/lib/objectbox-model.json | 230 ++++++ packages/alice_objectbox/lib/objectbox.g.dart | 685 ++++++++++++++++++ packages/alice_objectbox/pubspec.yaml | 26 + 95 files changed, 4077 insertions(+), 277 deletions(-) create mode 100644 examples/alice_objectbox/.gitignore create mode 100644 examples/alice_objectbox/.metadata create mode 100644 examples/alice_objectbox/README.md create mode 100644 examples/alice_objectbox/analysis_options.yaml create mode 100644 examples/alice_objectbox/android/.gitignore create mode 100644 examples/alice_objectbox/android/app/build.gradle create mode 100644 examples/alice_objectbox/android/app/src/debug/AndroidManifest.xml create mode 100644 examples/alice_objectbox/android/app/src/main/AndroidManifest.xml create mode 100644 examples/alice_objectbox/android/app/src/main/kotlin/com/jhomlala/alice_objectbox/MainActivity.kt create mode 100644 examples/alice_objectbox/android/app/src/main/res/drawable-v21/launch_background.xml create mode 100644 examples/alice_objectbox/android/app/src/main/res/drawable/launch_background.xml create mode 100644 examples/alice_objectbox/android/app/src/main/res/mipmap-hdpi/ic_launcher.png create mode 100644 examples/alice_objectbox/android/app/src/main/res/mipmap-mdpi/ic_launcher.png create mode 100644 examples/alice_objectbox/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png create mode 100644 examples/alice_objectbox/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png create mode 100644 examples/alice_objectbox/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png create mode 100644 examples/alice_objectbox/android/app/src/main/res/values-night/styles.xml create mode 100644 examples/alice_objectbox/android/app/src/main/res/values/styles.xml create mode 100644 examples/alice_objectbox/android/app/src/profile/AndroidManifest.xml create mode 100644 examples/alice_objectbox/android/build.gradle create mode 100644 examples/alice_objectbox/android/gradle.properties create mode 100644 examples/alice_objectbox/android/gradle/wrapper/gradle-wrapper.properties create mode 100644 examples/alice_objectbox/android/settings.gradle create mode 100644 examples/alice_objectbox/ios/.gitignore create mode 100644 examples/alice_objectbox/ios/Flutter/AppFrameworkInfo.plist create mode 100644 examples/alice_objectbox/ios/Flutter/Debug.xcconfig create mode 100644 examples/alice_objectbox/ios/Flutter/Release.xcconfig create mode 100644 examples/alice_objectbox/ios/Podfile create mode 100644 examples/alice_objectbox/ios/Podfile.lock create mode 100644 examples/alice_objectbox/ios/Runner.xcodeproj/project.pbxproj create mode 100644 examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata create mode 100644 examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist create mode 100644 examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings create mode 100644 examples/alice_objectbox/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme create mode 100644 examples/alice_objectbox/ios/Runner.xcworkspace/contents.xcworkspacedata create mode 100644 examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist create mode 100644 examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings create mode 100644 examples/alice_objectbox/ios/Runner/AppDelegate.swift create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png create mode 100644 examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md create mode 100644 examples/alice_objectbox/ios/Runner/Base.lproj/LaunchScreen.storyboard create mode 100644 examples/alice_objectbox/ios/Runner/Base.lproj/Main.storyboard create mode 100644 examples/alice_objectbox/ios/Runner/Info.plist create mode 100644 examples/alice_objectbox/ios/Runner/Runner-Bridging-Header.h create mode 100644 examples/alice_objectbox/ios/RunnerTests/RunnerTests.swift create mode 100644 examples/alice_objectbox/lib/main.dart create mode 100644 examples/alice_objectbox/pubspec.yaml create mode 100644 packages/alice/lib/core/alice_memory_storage.dart create mode 100644 packages/alice/lib/core/alice_storage.dart create mode 100644 packages/alice/lib/core/alice_store.dart create mode 100644 packages/alice/lib/ui/calls_list/widget/alice_inspector_screen.dart create mode 100644 packages/alice/lib/utils/curl.dart create mode 100644 packages/alice_objectbox/.gitignore create mode 100644 packages/alice_objectbox/CHANGELOG.md create mode 100644 packages/alice_objectbox/LICENSE create mode 100644 packages/alice_objectbox/README.md create mode 100644 packages/alice_objectbox/analysis_options.yaml create mode 100644 packages/alice_objectbox/lib/alice_objectbox.dart create mode 100644 packages/alice_objectbox/lib/alice_objectbox_store.dart create mode 100644 packages/alice_objectbox/lib/json_converter/alice_form_data_field_converter.dart create mode 100644 packages/alice_objectbox/lib/json_converter/alice_form_data_file_converter.dart create mode 100644 packages/alice_objectbox/lib/model/cached_alice_http_call.dart create mode 100644 packages/alice_objectbox/lib/model/cached_alice_http_error.dart create mode 100644 packages/alice_objectbox/lib/model/cached_alice_http_request.dart create mode 100644 packages/alice_objectbox/lib/model/cached_alice_http_response.dart create mode 100644 packages/alice_objectbox/lib/objectbox-model.json create mode 100644 packages/alice_objectbox/lib/objectbox.g.dart create mode 100644 packages/alice_objectbox/pubspec.yaml diff --git a/docs/usage.md b/docs/usage.md index b4f970a5..e9c10743 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -100,3 +100,58 @@ httpClient aliceHttpClientAdapter.onResponse(httpResponse, request, body: responseBody); }); ``` + +## ObjectBox + +Setting up ObjectBox with Alice is simple, however, there are a few crucial steps which need to be followed. + +1. Add it to your dependencies + +```yaml +dependencies: + alice_objectbox: ^1.0.0 +``` + +2. Follow the ObjectBox [example](https://github.com/objectbox/objectbox-dart/blob/main/objectbox/example/flutter/objectbox_demo/lib/main.dart) + +```dart +Future main() async { + /// This is required so ObjectBox can get the application directory + /// to store the database in. + WidgetsFlutterBinding.ensureInitialized(); + + /// Initialize [AliceObjectBoxStore] before running the app. + final AliceObjectBoxStore store = + await AliceObjectBoxStore.create(persistent: false); + + /// Pass [AliceObjectBoxStore] to the app + runApp(MyApp(store: store)); +} + +class MyApp extends StatefulWidget { + const MyApp({ + super.key, + required this.store, + }); + + final AliceObjectBoxStore store; + + @override + State createState() => _MyAppState(); +} + +class _MyAppState extends State { + late final Alice _alice = Alice( + showNotification: true, + showInspectorOnShake: true, + maxCallsCount: 1000, + /// Pass [AliceObjectBox] to the [Alice] constructor + aliceStorage: AliceObjectBox( + store: widget.store, + maxCallsCount: 1000, + ), + ); + + // your custom stuff... +} +``` diff --git a/examples/alice_objectbox/.gitignore b/examples/alice_objectbox/.gitignore new file mode 100644 index 00000000..29a3a501 --- /dev/null +++ b/examples/alice_objectbox/.gitignore @@ -0,0 +1,43 @@ +# Miscellaneous +*.class +*.log +*.pyc +*.swp +.DS_Store +.atom/ +.buildlog/ +.history +.svn/ +migrate_working_dir/ + +# 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 +.pub-cache/ +.pub/ +/build/ + +# 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 diff --git a/examples/alice_objectbox/.metadata b/examples/alice_objectbox/.metadata new file mode 100644 index 00000000..28c13bdf --- /dev/null +++ b/examples/alice_objectbox/.metadata @@ -0,0 +1,33 @@ +# 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: "761747bfc538b5af34aa0d3fac380f1bc331ec49" + channel: "stable" + +project_type: app + +# Tracks metadata for the flutter migrate command +migration: + platforms: + - platform: root + create_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + base_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + - platform: android + create_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + base_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + - platform: ios + create_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + base_revision: 761747bfc538b5af34aa0d3fac380f1bc331ec49 + + # User provided section + + # List of Local paths (relative to this file) that should be + # ignored by the migrate tool. + # + # Files that are not part of the templates will be ignored by default. + unmanaged_files: + - 'lib/main.dart' + - 'ios/Runner.xcodeproj/project.pbxproj' diff --git a/examples/alice_objectbox/README.md b/examples/alice_objectbox/README.md new file mode 100644 index 00000000..3b997b6f --- /dev/null +++ b/examples/alice_objectbox/README.md @@ -0,0 +1,3 @@ +# example + +Alice + ObjectBox Example diff --git a/examples/alice_objectbox/analysis_options.yaml b/examples/alice_objectbox/analysis_options.yaml new file mode 100644 index 00000000..abad9922 --- /dev/null +++ b/examples/alice_objectbox/analysis_options.yaml @@ -0,0 +1,10 @@ +analyzer: + exclude: + - "lib/generated_plugin_registrant.dart" + - "**.g.dart" + +include: package:flutter_lints/flutter.yaml + +linter: + rules: + #library_annotations: false diff --git a/examples/alice_objectbox/android/.gitignore b/examples/alice_objectbox/android/.gitignore new file mode 100644 index 00000000..6f568019 --- /dev/null +++ b/examples/alice_objectbox/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/examples/alice_objectbox/android/app/build.gradle b/examples/alice_objectbox/android/app/build.gradle new file mode 100644 index 00000000..d307f3e5 --- /dev/null +++ b/examples/alice_objectbox/android/app/build.gradle @@ -0,0 +1,58 @@ +plugins { + id "com.android.application" + id "kotlin-android" + // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. + id "dev.flutter.flutter-gradle-plugin" +} + +def localProperties = new Properties() +def localPropertiesFile = rootProject.file("local.properties") +if (localPropertiesFile.exists()) { + localPropertiesFile.withReader("UTF-8") { reader -> + localProperties.load(reader) + } +} + +def flutterVersionCode = localProperties.getProperty("flutter.versionCode") +if (flutterVersionCode == null) { + flutterVersionCode = "1" +} + +def flutterVersionName = localProperties.getProperty("flutter.versionName") +if (flutterVersionName == null) { + flutterVersionName = "1.0" +} + +android { + namespace = "com.jhomlala.alice_objectbox_example" + compileSdk = flutter.compileSdkVersion + ndkVersion = flutter.ndkVersion + + compileOptions { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 + } + + defaultConfig { + // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). + applicationId = "com.jhomlala.alice_objectbox_example" + // You can update the following values to match your application needs. + // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration. + minSdk = flutter.minSdkVersion + targetSdk = 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 = "../.." +} diff --git a/examples/alice_objectbox/android/app/src/debug/AndroidManifest.xml b/examples/alice_objectbox/android/app/src/debug/AndroidManifest.xml new file mode 100644 index 00000000..399f6981 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/debug/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/examples/alice_objectbox/android/app/src/main/AndroidManifest.xml b/examples/alice_objectbox/android/app/src/main/AndroidManifest.xml new file mode 100644 index 00000000..68f9bd6c --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/AndroidManifest.xml @@ -0,0 +1,45 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/alice_objectbox/android/app/src/main/kotlin/com/jhomlala/alice_objectbox/MainActivity.kt b/examples/alice_objectbox/android/app/src/main/kotlin/com/jhomlala/alice_objectbox/MainActivity.kt new file mode 100644 index 00000000..86a47b47 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/kotlin/com/jhomlala/alice_objectbox/MainActivity.kt @@ -0,0 +1,5 @@ +package com.jhomlala.alice_objectbox_example + +import io.flutter.embedding.android.FlutterActivity + +class MainActivity: FlutterActivity() diff --git a/examples/alice_objectbox/android/app/src/main/res/drawable-v21/launch_background.xml b/examples/alice_objectbox/android/app/src/main/res/drawable-v21/launch_background.xml new file mode 100644 index 00000000..f74085f3 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/res/drawable-v21/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/examples/alice_objectbox/android/app/src/main/res/drawable/launch_background.xml b/examples/alice_objectbox/android/app/src/main/res/drawable/launch_background.xml new file mode 100644 index 00000000..304732f8 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/res/drawable/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/examples/alice_objectbox/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/examples/alice_objectbox/android/app/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..db77bb4b7b0906d62b1847e87f15cdcacf6a4f29 GIT binary patch literal 544 zcmeAS@N?(olHy`uVBq!ia0vp^9w5xY3?!3`olAj~WQl7;NpOBzNqJ&XDuZK6ep0G} zXKrG8YEWuoN@d~6R2!h8bpbvhu0Wd6uZuB!w&u2PAxD2eNXD>P5D~Wn-+_Wa#27Xc zC?Zj|6r#X(-D3u$NCt}(Ms06KgJ4FxJVv{GM)!I~&n8Bnc94O7-Hd)cjDZswgC;Qs zO=b+9!WcT8F?0rF7!Uys2bs@gozCP?z~o%U|N3vA*22NaGQG zlg@K`O_XuxvZ&Ks^m&R!`&1=spLvfx7oGDKDwpwW`#iqdw@AL`7MR}m`rwr|mZgU`8P7SBkL78fFf!WnuYWm$5Z0 zNXhDbCv&49sM544K|?c)WrFfiZvCi9h0O)B3Pgg&ebxsLQ05GG~ AQ2+n{ literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/examples/alice_objectbox/android/app/src/main/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..17987b79bb8a35cc66c3c1fd44f5a5526c1b78be GIT binary patch literal 442 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA3?vioaBc-sk|nMYCBgY=CFO}lsSJ)O`AMk? zp1FzXsX?iUDV2pMQ*D5Xx&nMcT!A!W`0S9QKQy;}1Cl^CgaH=;G9cpY;r$Q>i*pfB zP2drbID<_#qf;rPZx^FqH)F_D#*k@@q03KywUtLX8Ua?`H+NMzkczFPK3lFz@i_kW%1NOn0|D2I9n9wzH8m|-tHjsw|9>@K=iMBhxvkv6m8Y-l zytQ?X=U+MF$@3 zt`~i=@j|6y)RWMK--}M|=T`o&^Ni>IoWKHEbBXz7?A@mgWoL>!*SXo`SZH-*HSdS+ yn*9;$7;m`l>wYBC5bq;=U}IMqLzqbYCidGC!)_gkIk_C@Uy!y&wkt5C($~2D>~)O*cj@FGjOCM)M>_ixfudOh)?xMu#Fs z#}Y=@YDTwOM)x{K_j*Q;dPdJ?Mz0n|pLRx{4n|)f>SXlmV)XB04CrSJn#dS5nK2lM zrZ9#~WelCp7&e13Y$jvaEXHskn$2V!!DN-nWS__6T*l;H&Fopn?A6HZ-6WRLFP=R` zqG+CE#d4|IbyAI+rJJ`&x9*T`+a=p|0O(+s{UBcyZdkhj=yS1>AirP+0R;mf2uMgM zC}@~JfByORAh4SyRgi&!(cja>F(l*O+nd+@4m$|6K6KDn_&uvCpV23&>G9HJp{xgg zoq1^2_p9@|WEo z*X_Uko@K)qYYv~>43eQGMdbiGbo>E~Q& zrYBH{QP^@Sti!`2)uG{irBBq@y*$B zi#&(U-*=fp74j)RyIw49+0MRPMRU)+a2r*PJ$L5roHt2$UjExCTZSbq%V!HeS7J$N zdG@vOZB4v_lF7Plrx+hxo7(fCV&}fHq)$ literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/examples/alice_objectbox/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..d5f1c8d34e7a88e3f88bea192c3a370d44689c3c GIT binary patch literal 1031 zcmeAS@N?(olHy`uVBq!ia0vp^6F``Q8Ax83A=Cw=BuiW)N`mv#O3D+9QW+dm@{>{( zJaZG%Q-e|yQz{EjrrIztFa`(sgt!6~Yi|1%a`XoT0ojZ}lNrNjb9xjc(B0U1_% zz5^97Xt*%oq$rQy4?0GKNfJ44uvxI)gC`h-NZ|&0-7(qS@?b!5r36oQ}zyZrNO3 zMO=Or+<~>+A&uN&E!^Sl+>xE!QC-|oJv`ApDhqC^EWD|@=#J`=d#Xzxs4ah}w&Jnc z$|q_opQ^2TrnVZ0o~wh<3t%W&flvYGe#$xqda2bR_R zvPYgMcHgjZ5nSA^lJr%;<&0do;O^tDDh~=pIxA#coaCY>&N%M2^tq^U%3DB@ynvKo}b?yu-bFc-u0JHzced$sg7S3zqI(2 z#Km{dPr7I=pQ5>FuK#)QwK?Y`E`B?nP+}U)I#c1+FM*1kNvWG|a(TpksZQ3B@sD~b zpQ2)*V*TdwjFOtHvV|;OsiDqHi=6%)o4b!)x$)%9pGTsE z-JL={-Ffv+T87W(Xpooq<`r*VzWQcgBN$$`u}f>-ZQI1BB8ykN*=e4rIsJx9>z}*o zo~|9I;xof literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/examples/alice_objectbox/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000000000000000000000000000000000000..4d6372eebdb28e45604e46eeda8dd24651419bc0 GIT binary patch literal 1443 zcmb`G{WsKk6vsdJTdFg%tJav9_E4vzrOaqkWF|A724Nly!y+?N9`YV6wZ}5(X(D_N(?!*n3`|_r0Hc?=PQw&*vnU?QTFY zB_MsH|!j$PP;I}?dppoE_gA(4uc!jV&0!l7_;&p2^pxNo>PEcNJv za5_RT$o2Mf!<+r?&EbHH6nMoTsDOa;mN(wv8RNsHpG)`^ymG-S5By8=l9iVXzN_eG%Xg2@Xeq76tTZ*dGh~Lo9vl;Zfs+W#BydUw zCkZ$o1LqWQO$FC9aKlLl*7x9^0q%0}$OMlp@Kk_jHXOjofdePND+j!A{q!8~Jn+s3 z?~~w@4?egS02}8NuulUA=L~QQfm;MzCGd)XhiftT;+zFO&JVyp2mBww?;QByS_1w! zrQlx%{^cMj0|Bo1FjwY@Q8?Hx0cIPF*@-ZRFpPc#bBw{5@tD(5%sClzIfl8WU~V#u zm5Q;_F!wa$BSpqhN>W@2De?TKWR*!ujY;Yylk_X5#~V!L*Gw~;$%4Q8~Mad z@`-kG?yb$a9cHIApZDVZ^U6Xkp<*4rU82O7%}0jjHlK{id@?-wpN*fCHXyXh(bLt* zPc}H-x0e4E&nQ>y%B-(EL=9}RyC%MyX=upHuFhAk&MLbsF0LP-q`XnH78@fT+pKPW zu72MW`|?8ht^tz$iC}ZwLp4tB;Q49K!QCF3@!iB1qOI=?w z7In!}F~ij(18UYUjnbmC!qKhPo%24?8U1x{7o(+?^Zu0Hx81|FuS?bJ0jgBhEMzf< zCgUq7r2OCB(`XkKcN-TL>u5y#dD6D!)5W?`O5)V^>jb)P)GBdy%t$uUMpf$SNV31$ zb||OojAbvMP?T@$h_ZiFLFVHDmbyMhJF|-_)HX3%m=CDI+ID$0^C>kzxprBW)hw(v zr!Gmda);ICoQyhV_oP5+C%?jcG8v+D@9f?Dk*!BxY}dazmrT@64UrP3hlslANK)bq z$67n83eh}OeW&SV@HG95P|bjfqJ7gw$e+`Hxo!4cx`jdK1bJ>YDSpGKLPZ^1cv$ek zIB?0S<#tX?SJCLWdMd{-ME?$hc7A$zBOdIJ)4!KcAwb=VMov)nK;9z>x~rfT1>dS+ zZ6#`2v@`jgbqq)P22H)Tx2CpmM^o1$B+xT6`(v%5xJ(?j#>Q$+rx_R|7TzDZe{J6q zG1*EcU%tE?!kO%^M;3aM6JN*LAKUVb^xz8-Pxo#jR5(-KBeLJvA@-gxNHx0M-ZJLl z;#JwQoh~9V?`UVo#}{6ka@II>++D@%KqGpMdlQ}?9E*wFcf5(#XQnP$Dk5~%iX^>f z%$y;?M0BLp{O3a(-4A?ewryHrrD%cx#Q^%KY1H zNre$ve+vceSLZcNY4U(RBX&)oZn*Py()h)XkE?PL$!bNb{N5FVI2Y%LKEm%yvpyTP z(1P?z~7YxD~Rf<(a@_y` literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/android/app/src/main/res/values-night/styles.xml b/examples/alice_objectbox/android/app/src/main/res/values-night/styles.xml new file mode 100644 index 00000000..06952be7 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/res/values-night/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/examples/alice_objectbox/android/app/src/main/res/values/styles.xml b/examples/alice_objectbox/android/app/src/main/res/values/styles.xml new file mode 100644 index 00000000..cb1ef880 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/main/res/values/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/examples/alice_objectbox/android/app/src/profile/AndroidManifest.xml b/examples/alice_objectbox/android/app/src/profile/AndroidManifest.xml new file mode 100644 index 00000000..399f6981 --- /dev/null +++ b/examples/alice_objectbox/android/app/src/profile/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/examples/alice_objectbox/android/build.gradle b/examples/alice_objectbox/android/build.gradle new file mode 100644 index 00000000..d2ffbffa --- /dev/null +++ b/examples/alice_objectbox/android/build.gradle @@ -0,0 +1,18 @@ +allprojects { + repositories { + google() + mavenCentral() + } +} + +rootProject.buildDir = "../build" +subprojects { + project.buildDir = "${rootProject.buildDir}/${project.name}" +} +subprojects { + project.evaluationDependsOn(":app") +} + +tasks.register("clean", Delete) { + delete rootProject.buildDir +} diff --git a/examples/alice_objectbox/android/gradle.properties b/examples/alice_objectbox/android/gradle.properties new file mode 100644 index 00000000..3b5b324f --- /dev/null +++ b/examples/alice_objectbox/android/gradle.properties @@ -0,0 +1,3 @@ +org.gradle.jvmargs=-Xmx4G -XX:+HeapDumpOnOutOfMemoryError +android.useAndroidX=true +android.enableJetifier=true diff --git a/examples/alice_objectbox/android/gradle/wrapper/gradle-wrapper.properties b/examples/alice_objectbox/android/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 00000000..e1ca574e --- /dev/null +++ b/examples/alice_objectbox/android/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,5 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-7.6.3-all.zip diff --git a/examples/alice_objectbox/android/settings.gradle b/examples/alice_objectbox/android/settings.gradle new file mode 100644 index 00000000..536165d3 --- /dev/null +++ b/examples/alice_objectbox/android/settings.gradle @@ -0,0 +1,25 @@ +pluginManagement { + def flutterSdkPath = { + def properties = new Properties() + file("local.properties").withInputStream { properties.load(it) } + def flutterSdkPath = properties.getProperty("flutter.sdk") + assert flutterSdkPath != null, "flutter.sdk not set in local.properties" + return flutterSdkPath + }() + + includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") + + repositories { + google() + mavenCentral() + gradlePluginPortal() + } +} + +plugins { + id "dev.flutter.flutter-plugin-loader" version "1.0.0" + id "com.android.application" version "7.3.0" apply false + id "org.jetbrains.kotlin.android" version "1.7.10" apply false +} + +include ":app" diff --git a/examples/alice_objectbox/ios/.gitignore b/examples/alice_objectbox/ios/.gitignore new file mode 100644 index 00000000..7a7f9873 --- /dev/null +++ b/examples/alice_objectbox/ios/.gitignore @@ -0,0 +1,34 @@ +**/dgph +*.mode1v3 +*.mode2v3 +*.moved-aside +*.pbxuser +*.perspectivev3 +**/*sync/ +.sconsign.dblite +.tags* +**/.vagrant/ +**/DerivedData/ +Icon? +**/Pods/ +**/.symlinks/ +profile +xcuserdata +**/.generated/ +Flutter/App.framework +Flutter/Flutter.framework +Flutter/Flutter.podspec +Flutter/Generated.xcconfig +Flutter/ephemeral/ +Flutter/app.flx +Flutter/app.zip +Flutter/flutter_assets/ +Flutter/flutter_export_environment.sh +ServiceDefinitions.json +Runner/GeneratedPluginRegistrant.* + +# Exceptions to above rules. +!default.mode1v3 +!default.mode2v3 +!default.pbxuser +!default.perspectivev3 diff --git a/examples/alice_objectbox/ios/Flutter/AppFrameworkInfo.plist b/examples/alice_objectbox/ios/Flutter/AppFrameworkInfo.plist new file mode 100644 index 00000000..7c569640 --- /dev/null +++ b/examples/alice_objectbox/ios/Flutter/AppFrameworkInfo.plist @@ -0,0 +1,26 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + App + CFBundleIdentifier + io.flutter.flutter.app + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + App + CFBundlePackageType + FMWK + CFBundleShortVersionString + 1.0 + CFBundleSignature + ???? + CFBundleVersion + 1.0 + MinimumOSVersion + 12.0 + + diff --git a/examples/alice_objectbox/ios/Flutter/Debug.xcconfig b/examples/alice_objectbox/ios/Flutter/Debug.xcconfig new file mode 100644 index 00000000..ec97fc6f --- /dev/null +++ b/examples/alice_objectbox/ios/Flutter/Debug.xcconfig @@ -0,0 +1,2 @@ +#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" +#include "Generated.xcconfig" diff --git a/examples/alice_objectbox/ios/Flutter/Release.xcconfig b/examples/alice_objectbox/ios/Flutter/Release.xcconfig new file mode 100644 index 00000000..c4855bfe --- /dev/null +++ b/examples/alice_objectbox/ios/Flutter/Release.xcconfig @@ -0,0 +1,2 @@ +#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" +#include "Generated.xcconfig" diff --git a/examples/alice_objectbox/ios/Podfile b/examples/alice_objectbox/ios/Podfile new file mode 100644 index 00000000..d97f17e2 --- /dev/null +++ b/examples/alice_objectbox/ios/Podfile @@ -0,0 +1,44 @@ +# Uncomment this line to define a global platform for your project +# platform :ios, '12.0' + +# CocoaPods analytics sends network stats synchronously affecting flutter build latency. +ENV['COCOAPODS_DISABLE_STATS'] = 'true' + +project 'Runner', { + 'Debug' => :debug, + 'Profile' => :release, + 'Release' => :release, +} + +def flutter_root + generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'Generated.xcconfig'), __FILE__) + unless File.exist?(generated_xcode_build_settings_path) + raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure flutter pub get is executed first" + end + + File.foreach(generated_xcode_build_settings_path) do |line| + matches = line.match(/FLUTTER_ROOT\=(.*)/) + return matches[1].strip if matches + end + raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Generated.xcconfig, then run flutter pub get" +end + +require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) + +flutter_ios_podfile_setup + +target 'Runner' do + use_frameworks! + use_modular_headers! + + flutter_install_all_ios_pods File.dirname(File.realpath(__FILE__)) + target 'RunnerTests' do + inherit! :search_paths + end +end + +post_install do |installer| + installer.pods_project.targets.each do |target| + flutter_additional_ios_build_settings(target) + end +end diff --git a/examples/alice_objectbox/ios/Podfile.lock b/examples/alice_objectbox/ios/Podfile.lock new file mode 100644 index 00000000..d6911a69 --- /dev/null +++ b/examples/alice_objectbox/ios/Podfile.lock @@ -0,0 +1,78 @@ +PODS: + - Flutter (1.0.0) + - flutter_local_notifications (0.0.1): + - Flutter + - ObjectBox (2.0.0) + - objectbox_flutter_libs (0.0.1): + - Flutter + - ObjectBox (= 2.0.0) + - open_filex (0.0.2): + - Flutter + - package_info_plus (0.4.5): + - Flutter + - path_provider_foundation (0.0.1): + - Flutter + - FlutterMacOS + - permission_handler_apple (9.3.0): + - Flutter + - sensors_plus (0.0.1): + - Flutter + - share_plus (0.0.1): + - Flutter + - url_launcher_ios (0.0.1): + - Flutter + +DEPENDENCIES: + - Flutter (from `Flutter`) + - flutter_local_notifications (from `.symlinks/plugins/flutter_local_notifications/ios`) + - objectbox_flutter_libs (from `.symlinks/plugins/objectbox_flutter_libs/ios`) + - open_filex (from `.symlinks/plugins/open_filex/ios`) + - package_info_plus (from `.symlinks/plugins/package_info_plus/ios`) + - path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`) + - permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`) + - sensors_plus (from `.symlinks/plugins/sensors_plus/ios`) + - share_plus (from `.symlinks/plugins/share_plus/ios`) + - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) + +SPEC REPOS: + trunk: + - ObjectBox + +EXTERNAL SOURCES: + Flutter: + :path: Flutter + flutter_local_notifications: + :path: ".symlinks/plugins/flutter_local_notifications/ios" + objectbox_flutter_libs: + :path: ".symlinks/plugins/objectbox_flutter_libs/ios" + open_filex: + :path: ".symlinks/plugins/open_filex/ios" + package_info_plus: + :path: ".symlinks/plugins/package_info_plus/ios" + path_provider_foundation: + :path: ".symlinks/plugins/path_provider_foundation/darwin" + permission_handler_apple: + :path: ".symlinks/plugins/permission_handler_apple/ios" + sensors_plus: + :path: ".symlinks/plugins/sensors_plus/ios" + share_plus: + :path: ".symlinks/plugins/share_plus/ios" + url_launcher_ios: + :path: ".symlinks/plugins/url_launcher_ios/ios" + +SPEC CHECKSUMS: + Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 + flutter_local_notifications: 4cde75091f6327eb8517fa068a0a5950212d2086 + ObjectBox: f5319bd9ad2ea960796eff7227e86471867e9ef0 + objectbox_flutter_libs: c7748f6c6fda47d22f15c8062fb8208063fd948a + open_filex: 6e26e659846ec990262224a12ef1c528bb4edbe4 + package_info_plus: 58f0028419748fad15bf008b270aaa8e54380b1c + path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46 + permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2 + sensors_plus: 18a9b346c43e157da17d2c8e99def703f9efb9d8 + share_plus: 8875f4f2500512ea181eef553c3e27dba5135aad + url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe + +PODFILE CHECKSUM: 819463e6a0290f5a72f145ba7cde16e8b6ef0796 + +COCOAPODS: 1.15.2 diff --git a/examples/alice_objectbox/ios/Runner.xcodeproj/project.pbxproj b/examples/alice_objectbox/ios/Runner.xcodeproj/project.pbxproj new file mode 100644 index 00000000..dccc3393 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcodeproj/project.pbxproj @@ -0,0 +1,634 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 54; + objects = { + +/* Begin PBXBuildFile section */ + 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; }; + 331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; + 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; + 74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; + 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; + 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; + 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 331C8085294A63A400263BE5 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 97C146E61CF9000F007C117D /* Project object */; + proxyType = 1; + remoteGlobalIDString = 97C146ED1CF9000F007C117D; + remoteInfo = Runner; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 9705A1C41CF9048500538489 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = ""; }; + 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = ""; }; + 331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = ""; }; + 331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = ""; }; + 74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = ""; }; + 74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = ""; }; + 9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = ""; }; + 9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = ""; }; + 97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; + 97C146FB1CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = ""; }; + 97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; + 97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; + 97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 97C146EB1CF9000F007C117D /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 331C8082294A63A400263BE5 /* RunnerTests */ = { + isa = PBXGroup; + children = ( + 331C807B294A618700263BE5 /* RunnerTests.swift */, + ); + path = RunnerTests; + sourceTree = ""; + }; + 9740EEB11CF90186004384FC /* Flutter */ = { + isa = PBXGroup; + children = ( + 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */, + 9740EEB21CF90195004384FC /* Debug.xcconfig */, + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */, + 9740EEB31CF90195004384FC /* Generated.xcconfig */, + ); + name = Flutter; + sourceTree = ""; + }; + 97C146E51CF9000F007C117D = { + isa = PBXGroup; + children = ( + 9740EEB11CF90186004384FC /* Flutter */, + 97C146F01CF9000F007C117D /* Runner */, + 97C146EF1CF9000F007C117D /* Products */, + 331C8082294A63A400263BE5 /* RunnerTests */, + ); + sourceTree = ""; + }; + 97C146EF1CF9000F007C117D /* Products */ = { + isa = PBXGroup; + children = ( + 97C146EE1CF9000F007C117D /* Runner.app */, + 331C8081294A63A400263BE5 /* RunnerTests.xctest */, + ); + name = Products; + sourceTree = ""; + }; + 97C146F01CF9000F007C117D /* Runner */ = { + isa = PBXGroup; + children = ( + 97C146FA1CF9000F007C117D /* Main.storyboard */, + 97C146FD1CF9000F007C117D /* Assets.xcassets */, + 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */, + 97C147021CF9000F007C117D /* Info.plist */, + 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */, + 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */, + 74858FAE1ED2DC5600515810 /* AppDelegate.swift */, + 74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */, + ); + path = Runner; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + 331C8080294A63A400263BE5 /* RunnerTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; + buildPhases = ( + 331C807D294A63A400263BE5 /* Sources */, + 331C807F294A63A400263BE5 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + 331C8086294A63A400263BE5 /* PBXTargetDependency */, + ); + name = RunnerTests; + productName = RunnerTests; + productReference = 331C8081294A63A400263BE5 /* RunnerTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + 97C146ED1CF9000F007C117D /* Runner */ = { + isa = PBXNativeTarget; + buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; + buildPhases = ( + 9740EEB61CF901F6004384FC /* Run Script */, + 97C146EA1CF9000F007C117D /* Sources */, + 97C146EB1CF9000F007C117D /* Frameworks */, + 97C146EC1CF9000F007C117D /* Resources */, + 9705A1C41CF9048500538489 /* Embed Frameworks */, + 3B06AD1E1E4923F5004D2608 /* Thin Binary */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = Runner; + productName = Runner; + productReference = 97C146EE1CF9000F007C117D /* Runner.app */; + productType = "com.apple.product-type.application"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 97C146E61CF9000F007C117D /* Project object */ = { + isa = PBXProject; + attributes = { + BuildIndependentTargetsInParallel = YES; + LastUpgradeCheck = 1510; + ORGANIZATIONNAME = ""; + TargetAttributes = { + 331C8080294A63A400263BE5 = { + CreatedOnToolsVersion = 14.0; + TestTargetID = 97C146ED1CF9000F007C117D; + }; + 97C146ED1CF9000F007C117D = { + CreatedOnToolsVersion = 7.3.1; + LastSwiftMigration = 1100; + }; + }; + }; + buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */; + compatibilityVersion = "Xcode 9.3"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = 97C146E51CF9000F007C117D; + productRefGroup = 97C146EF1CF9000F007C117D /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 97C146ED1CF9000F007C117D /* Runner */, + 331C8080294A63A400263BE5 /* RunnerTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 331C807F294A63A400263BE5 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 97C146EC1CF9000F007C117D /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */, + 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */, + 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */, + 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXShellScriptBuildPhase section */ + 3B06AD1E1E4923F5004D2608 /* Thin Binary */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputPaths = ( + "${TARGET_BUILD_DIR}/${INFOPLIST_PATH}", + ); + name = "Thin Binary"; + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" embed_and_thin"; + }; + 9740EEB61CF901F6004384FC /* Run Script */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputPaths = ( + ); + name = "Run Script"; + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; + }; +/* End PBXShellScriptBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 331C807D294A63A400263BE5 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 97C146EA1CF9000F007C117D /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */, + 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 331C8086294A63A400263BE5 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 97C146ED1CF9000F007C117D /* Runner */; + targetProxy = 331C8085294A63A400263BE5 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + 97C146FA1CF9000F007C117D /* Main.storyboard */ = { + isa = PBXVariantGroup; + children = ( + 97C146FB1CF9000F007C117D /* Base */, + ); + name = Main.storyboard; + sourceTree = ""; + }; + 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = { + isa = PBXVariantGroup; + children = ( + 97C147001CF9000F007C117D /* Base */, + ); + name = LaunchScreen.storyboard; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 249021D3217E4FDB00AE95B9 /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = iphoneos; + SUPPORTED_PLATFORMS = iphoneos; + TARGETED_DEVICE_FAMILY = "1,2"; + VALIDATE_PRODUCT = YES; + }; + name = Profile; + }; + 249021D4217E4FDB00AE95B9 /* Profile */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; + DEVELOPMENT_TEAM = ""; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox; + PRODUCT_NAME = "$(TARGET_NAME)"; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + VERSIONING_SYSTEM = "apple-generic"; + }; + name = Profile; + }; + 331C8088294A63A400263BE5 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner"; + }; + name = Debug; + }; + 331C8089294A63A400263BE5 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner"; + }; + name = Release; + }; + 331C808A294A63A400263BE5 /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner"; + }; + name = Profile; + }; + 97C147031CF9000F007C117D /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = dwarf; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_DYNAMIC_NO_PIC = NO; + GCC_NO_COMMON_BLOCKS = YES; + GCC_OPTIMIZATION_LEVEL = 0; + GCC_PREPROCESSOR_DEFINITIONS = ( + "DEBUG=1", + "$(inherited)", + ); + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = iphoneos; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Debug; + }; + 97C147041CF9000F007C117D /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = iphoneos; + SUPPORTED_PLATFORMS = iphoneos; + SWIFT_COMPILATION_MODE = wholemodule; + SWIFT_OPTIMIZATION_LEVEL = "-O"; + TARGETED_DEVICE_FAMILY = "1,2"; + VALIDATE_PRODUCT = YES; + }; + name = Release; + }; + 97C147061CF9000F007C117D /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; + DEVELOPMENT_TEAM = ""; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox; + PRODUCT_NAME = "$(TARGET_NAME)"; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + VERSIONING_SYSTEM = "apple-generic"; + }; + name = Debug; + }; + 97C147071CF9000F007C117D /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; + DEVELOPMENT_TEAM = ""; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.jhomlala.aliceObjectbox; + PRODUCT_NAME = "$(TARGET_NAME)"; + SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; + SUPPORTS_MACCATALYST = NO; + SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; + SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + VERSIONING_SYSTEM = "apple-generic"; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 331C8088294A63A400263BE5 /* Debug */, + 331C8089294A63A400263BE5 /* Release */, + 331C808A294A63A400263BE5 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 97C147031CF9000F007C117D /* Debug */, + 97C147041CF9000F007C117D /* Release */, + 249021D3217E4FDB00AE95B9 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 97C147061CF9000F007C117D /* Debug */, + 97C147071CF9000F007C117D /* Release */, + 249021D4217E4FDB00AE95B9 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + }; + rootObject = 97C146E61CF9000F007C117D /* Project object */; +} diff --git a/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..919434a6 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 00000000..18d98100 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 00000000..f9b0d7c5 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/examples/alice_objectbox/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/examples/alice_objectbox/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme new file mode 100644 index 00000000..8e3ca5df --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -0,0 +1,98 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/alice_objectbox/ios/Runner.xcworkspace/contents.xcworkspacedata b/examples/alice_objectbox/ios/Runner.xcworkspace/contents.xcworkspacedata new file mode 100644 index 00000000..1d526a16 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 00000000..18d98100 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 00000000..f9b0d7c5 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/examples/alice_objectbox/ios/Runner/AppDelegate.swift b/examples/alice_objectbox/ios/Runner/AppDelegate.swift new file mode 100644 index 00000000..9074fee9 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/AppDelegate.swift @@ -0,0 +1,13 @@ +import Flutter +import UIKit + +@UIApplicationMain +@objc class AppDelegate: FlutterAppDelegate { + override func application( + _ application: UIApplication, + didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? + ) -> Bool { + GeneratedPluginRegistrant.register(with: self) + return super.application(application, didFinishLaunchingWithOptions: launchOptions) + } +} diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 00000000..d36b1fab --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,122 @@ +{ + "images" : [ + { + "size" : "20x20", + "idiom" : "iphone", + "filename" : "Icon-App-20x20@2x.png", + "scale" : "2x" + }, + { + "size" : "20x20", + "idiom" : "iphone", + "filename" : "Icon-App-20x20@3x.png", + "scale" : "3x" + }, + { + "size" : "29x29", + "idiom" : "iphone", + "filename" : "Icon-App-29x29@1x.png", + "scale" : "1x" + }, + { + "size" : "29x29", + "idiom" : "iphone", + "filename" : "Icon-App-29x29@2x.png", + "scale" : "2x" + }, + { + "size" : "29x29", + "idiom" : "iphone", + "filename" : "Icon-App-29x29@3x.png", + "scale" : "3x" + }, + { + "size" : "40x40", + "idiom" : "iphone", + "filename" : "Icon-App-40x40@2x.png", + "scale" : "2x" + }, + { + "size" : "40x40", + "idiom" : "iphone", + "filename" : "Icon-App-40x40@3x.png", + "scale" : "3x" + }, + { + "size" : "60x60", + "idiom" : "iphone", + "filename" : "Icon-App-60x60@2x.png", + "scale" : "2x" + }, + { + "size" : "60x60", + "idiom" : "iphone", + "filename" : "Icon-App-60x60@3x.png", + "scale" : "3x" + }, + { + "size" : "20x20", + "idiom" : "ipad", + "filename" : "Icon-App-20x20@1x.png", + "scale" : "1x" + }, + { + "size" : "20x20", + "idiom" : "ipad", + "filename" : "Icon-App-20x20@2x.png", + "scale" : "2x" + }, + { + "size" : "29x29", + "idiom" : "ipad", + "filename" : "Icon-App-29x29@1x.png", + "scale" : "1x" + }, + { + "size" : "29x29", + "idiom" : "ipad", + "filename" : "Icon-App-29x29@2x.png", + "scale" : "2x" + }, + { + "size" : "40x40", + "idiom" : "ipad", + "filename" : "Icon-App-40x40@1x.png", + "scale" : "1x" + }, + { + "size" : "40x40", + "idiom" : "ipad", + "filename" : "Icon-App-40x40@2x.png", + "scale" : "2x" + }, + { + "size" : "76x76", + "idiom" : "ipad", + "filename" : "Icon-App-76x76@1x.png", + "scale" : "1x" + }, + { + "size" : "76x76", + "idiom" : "ipad", + "filename" : "Icon-App-76x76@2x.png", + "scale" : "2x" + }, + { + "size" : "83.5x83.5", + "idiom" : "ipad", + "filename" : "Icon-App-83.5x83.5@2x.png", + "scale" : "2x" + }, + { + "size" : "1024x1024", + "idiom" : "ios-marketing", + "filename" : "Icon-App-1024x1024@1x.png", + "scale" : "1x" + } + ], + "info" : { + "version" : 1, + "author" : "xcode" + } +} diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png new file mode 100644 index 0000000000000000000000000000000000000000..dc9ada4725e9b0ddb1deab583e5b5102493aa332 GIT binary patch literal 10932 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 literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..797d452e458972bab9d994556c8305db4c827017 GIT binary patch literal 406 zcmV;H0crk;P))>cdjpWt&rLJgVp-t?DREyuq1A%0Z4)6_WsQ7{nzjN zo!X zGXV)2i3kcZIL~_j>uIKPK_zib+3T+Nt3Mb&Br)s)UIaA}@p{wDda>7=Q|mGRp7pqY zkJ!7E{MNz$9nOwoVqpFb)}$IP24Wn2JJ=Cw(!`OXJBr45rP>>AQr$6c7slJWvbpNW z@KTwna6d?PP>hvXCcp=4F;=GR@R4E7{4VU^0p4F>v^#A|>07*qoM6N<$f*5nx ACIA2c literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..6ed2d933e1120817fe9182483a228007b18ab6ae GIT binary patch literal 450 zcmV;z0X_bSP)iGWQ_5NJQ_~rNh*z)}eT%KUb z`7gNk0#AwF^#0T0?hIa^`~Ck;!}#m+_uT050aTR(J!bU#|IzRL%^UsMS#KsYnTF*!YeDOytlP4VhV?b} z%rz_<=#CPc)tU1MZTq~*2=8~iZ!lSa<{9b@2Jl;?IEV8)=fG217*|@)CCYgFze-x? zIFODUIA>nWKpE+bn~n7;-89sa>#DR>TSlqWk*!2hSN6D~Qb#VqbP~4Fk&m`@1$JGr zXPIdeRE&b2Thd#{MtDK$px*d3-Wx``>!oimf%|A-&-q*6KAH)e$3|6JV%HX{Hig)k suLT-RhftRq8b9;(V=235Wa|I=027H2wCDra;{X5v07*qoM6N<$f;9x^2LJ#7 literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png new file mode 100644 index 0000000000000000000000000000000000000000..4cd7b0099ca80c806f8fe495613e8d6c69460d76 GIT binary patch literal 282 zcmV+#0p(^bcu7P-R4C8Q z&e;xxFbF_Vrezo%_kH*OKhshZ6BFpG-Y1e10`QXJKbND7AMQ&cMj60B5TNObaZxYybcN07*qoM6N<$g3m;S%K!iX literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..fe730945a01f64a61e2235dbe3f45b08f7729182 GIT binary patch literal 462 zcmV;<0WtoGP)-}iV`2<;=$?g5M=KQbZ{F&YRNy7Nn@%_*5{gvDM0aKI4?ESmw z{NnZg)A0R`+4?NF_RZexyVB&^^ZvN!{I28tr{Vje;QNTz`dG&Jz0~Ek&f2;*Z7>B|cg}xYpxEFY+0YrKLF;^Q+-HreN0P{&i zK~zY`?b7ECf-n?@;d<&orQ*Q7KoR%4|C>{W^h6@&01>0SKS`dn{Q}GT%Qj_{PLZ_& zs`MFI#j-(>?bvdZ!8^xTwlY{qA)T4QLbY@j(!YJ7aXJervHy6HaG_2SB`6CC{He}f zHVw(fJWApwPq!6VY7r1w-Fs)@ox~N+q|w~e;JI~C4Vf^@d>Wvj=fl`^u9x9wd9 zR%3*Q+)t%S!MU_`id^@&Y{y7-r98lZX0?YrHlfmwb?#}^1b{8g&KzmkE(L>Z&)179 zp<)v6Y}pRl100G2FL_t(o!|l{-Q-VMg#&MKg7c{O0 z2wJImOS3Gy*Z2Qifdv~JYOp;v+U)a|nLoc7hNH;I$;lzDt$}rkaFw1mYK5_0Q(Sut zvbEloxON7$+HSOgC9Z8ltuC&0OSF!-mXv5caV>#bc3@hBPX@I$58-z}(ZZE!t-aOG zpjNkbau@>yEzH(5Yj4kZiMH32XI!4~gVXNnjAvRx;Sdg^`>2DpUEwoMhTs_st8pKG z(%SHyHdU&v%f36~uERh!bd`!T2dw;z6PrOTQ7Vt*#9F2uHlUVnb#ev_o^fh}Dzmq} zWtlk35}k=?xj28uO|5>>$yXadTUE@@IPpgH`gJ~Ro4>jd1IF|(+IX>8M4Ps{PNvmI zNj4D+XgN83gPt_Gm}`Ybv{;+&yu-C(Grdiahmo~BjG-l&mWM+{e5M1sm&=xduwgM9 z`8OEh`=F3r`^E{n_;%9weN{cf2%7=VzC@cYj+lg>+3|D|_1C@{hcU(DyQG_BvBWe? zvTv``=%b1zrol#=R`JB)>cdjpWt&rLJgVp-t?DREyuq1A%0Z4)6_WsQ7{nzjN zo!X zGXV)2i3kcZIL~_j>uIKPK_zib+3T+Nt3Mb&Br)s)UIaA}@p{wDda>7=Q|mGRp7pqY zkJ!7E{MNz$9nOwoVqpFb)}$IP24Wn2JJ=Cw(!`OXJBr45rP>>AQr$6c7slJWvbpNW z@KTwna6d?PP>hvXCcp=4F;=GR@R4E7{4VU^0p4F>v^#A|>07*qoM6N<$f*5nx ACIA2c literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..502f463a9bc882b461c96aadf492d1729e49e725 GIT binary patch literal 586 zcmV-Q0=4~#P)+}#`wDE{8-2Mebf5<{{PqV{TgVcv*r8?UZ3{-|G?_}T*&y;@cqf{ z{Q*~+qr%%p!1pS*_Uicl#q9lc(D`!D`LN62sNwq{oYw(Wmhk)k<@f$!$@ng~_5)Ru z0Z)trIA5^j{DIW^c+vT2%lW+2<(RtE2wR;4O@)Tm`Xr*?A(qYoM}7i5Yxw>D(&6ou zxz!_Xr~yNF+waPe00049Nkl*;a!v6h%{rlvIH#gW3s8p;bFr=l}mRqpW2h zw=OA%hdyL~z+UHOzl0eKhEr$YYOL-c-%Y<)=j?(bzDweB7{b+%_ypvm_cG{SvM=DK zhv{K@m>#Bw>2W$eUI#iU)Wdgs8Y3U+A$Gd&{+j)d)BmGKx+43U_!tik_YlN)>$7G! zhkE!s;%oku3;IwG3U^2kw?z+HM)jB{@zFhK8P#KMSytSthr+4!c(5c%+^UBn`0X*2 zy3(k600_CSZj?O$Qu%&$;|TGUJrptR(HzyIx>5E(2r{eA(<6t3e3I0B)7d6s7?Z5J zZ!rtKvA{MiEBm&KFtoifx>5P^Z=vl)95XJn()aS5%ad(s?4-=Tkis9IGu{`Fy8r+H07*qoM6N<$f20Z)wqMt%V?S?~D#06};F zA3KcL`Wb+>5ObvgQIG&ig8(;V04hz?@cqy3{mSh8o!|U|)cI!1_+!fWH@o*8vh^CU z^ws0;(c$gI+2~q^tO#GDHf@=;DncUw00J^eL_t(&-tE|HQ`%4vfZ;WsBqu-$0nu1R zq^Vj;p$clf^?twn|KHO+IGt^q#a3X?w9dXC@*yxhv&l}F322(8Y1&=P&I}~G@#h6; z1CV9ecD9ZEe87{{NtI*)_aJ<`kJa z?5=RBtFF50s;jQLFil-`)m2wrb=6h(&brpj%nG_U&ut~$?8Rokzxi8zJoWr#2dto5 zOX_URcc<1`Iky+jc;A%Vzx}1QU{2$|cKPom2Vf1{8m`vja4{F>HS?^Nc^rp}xo+Nh zxd}eOm`fm3@MQC1< zIk&aCjb~Yh%5+Yq0`)D;q{#-Uqlv*o+Oor zE!I71Z@ASH3grl8&P^L0WpavHoP|UX4e?!igT`4?AZk$hu*@%6WJ;zDOGlw7kj@ zY5!B-0ft0f?Lgb>C;$Ke07*qoM6N<$f~t1N9smFU literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..0ec303439225b78712f49115768196d8d76f6790 GIT binary patch literal 862 zcmV-k1EKthP)20Z)wqMt%V?S?~D#06};F zA3KcL`Wb+>5ObvgQIG&ig8(;V04hz?@cqy3{mSh8o!|U|)cI!1_+!fWH@o*8vh^CU z^ws0;(c$gI+2~q^tO#GDHf@=;DncUw00J^eL_t(&-tE|HQ`%4vfZ;WsBqu-$0nu1R zq^Vj;p$clf^?twn|KHO+IGt^q#a3X?w9dXC@*yxhv&l}F322(8Y1&=P&I}~G@#h6; z1CV9ecD9ZEe87{{NtI*)_aJ<`kJa z?5=RBtFF50s;jQLFil-`)m2wrb=6h(&brpj%nG_U&ut~$?8Rokzxi8zJoWr#2dto5 zOX_URcc<1`Iky+jc;A%Vzx}1QU{2$|cKPom2Vf1{8m`vja4{F>HS?^Nc^rp}xo+Nh zxd}eOm`fm3@MQC1< zIk&aCjb~Yh%5+Yq0`)D;q{#-Uqlv*o+Oor zE!I71Z@ASH3grl8&P^L0WpavHoP|UX4e?!igT`4?AZk$hu*@%6WJ;zDOGlw7kj@ zY5!B-0ft0f?Lgb>C;$Ke07*qoM6N<$f~t1N9smFU literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..e9f5fea27c705180eb716271f41b582e76dcbd90 GIT binary patch literal 1674 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 literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..0467bf12aa4d28f374bb26596605a46dcbb3e7c8 GIT binary patch literal 1418 zcmV;51$Fv~P)q zKfU)WzW*n(@|xWGCA9ScMt*e9`2kdxPQ&&>|-UCa7_51w+ zLUsW@ZzZSW0y$)Hp~e9%PvP|a03ks1`~K?q{u;6NC8*{AOqIUq{CL&;p56Lf$oQGq z^={4hPQv)y=I|4n+?>7Fim=dxt1 z2H+Dm+1+fh+IF>G0SjJMkQQre1x4|G*Z==(Ot&kCnUrL4I(rf(ucITwmuHf^hXiJT zkdTm&kdTm&kdTm&kdP`esgWG0BcWCVkVZ&2dUwN`cgM8QJb`Z7Z~e<&Yj2(}>Tmf` zm1{eLgw!b{bXkjWbF%dTkTZEJWyWOb##Lfw4EK2}<0d6%>AGS{po>WCOy&f$Tay_> z?NBlkpo@s-O;0V%Y_Xa-G#_O08q5LR*~F%&)}{}r&L%Sbs8AS4t7Y0NEx*{soY=0MZExqA5XHQkqi#4gW3 zqODM^iyZl;dvf)-bOXtOru(s)Uc7~BFx{w-FK;2{`VA?(g&@3z&bfLFyctOH!cVsF z7IL=fo-qBndRUm;kAdXR4e6>k-z|21AaN%ubeVrHl*<|s&Ax@W-t?LR(P-24A5=>a z*R9#QvjzF8n%@1Nw@?CG@6(%>+-0ASK~jEmCV|&a*7-GKT72W<(TbSjf)&Eme6nGE z>Gkj4Sq&2e+-G%|+NM8OOm5zVl9{Z8Dd8A5z3y8mZ=4Bv4%>as_{9cN#bm~;h>62( zdqY93Zy}v&c4n($Vv!UybR8ocs7#zbfX1IY-*w~)p}XyZ-SFC~4w>BvMVr`dFbelV{lLL0bx7@*ZZdebr3`sP;? zVImji)kG)(6Juv0lz@q`F!k1FE;CQ(D0iG$wchPbKZQELlsZ#~rt8#90Y_Xh&3U-< z{s<&cCV_1`^TD^ia9!*mQDq& zn2{r`j};V|uV%_wsP!zB?m%;FeaRe+X47K0e+KE!8C{gAWF8)lCd1u1%~|M!XNRvw zvtqy3iz0WSpWdhn6$hP8PaRBmp)q`#PCA`Vd#Tc$@f1tAcM>f_I@bC)hkI9|o(Iqv zo}Piadq!j76}004RBio<`)70k^`K1NK)q>w?p^C6J2ZC!+UppiK6&y3Kmbv&O!oYF z34$0Z;QO!JOY#!`qyGH<3Pd}Pt@q*A0V=3SVtWKRR8d8Z&@)3qLPA19LPA19LPEUC YUoZo%k(ykuW&i*H07*qoM6N<$f+CH{y8r+H literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json new file mode 100644 index 00000000..0bedcf2f --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json @@ -0,0 +1,23 @@ +{ + "images" : [ + { + "idiom" : "universal", + "filename" : "LaunchImage.png", + "scale" : "1x" + }, + { + "idiom" : "universal", + "filename" : "LaunchImage@2x.png", + "scale" : "2x" + }, + { + "idiom" : "universal", + "filename" : "LaunchImage@3x.png", + "scale" : "3x" + } + ], + "info" : { + "version" : 1, + "author" : "xcode" + } +} diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png new file mode 100644 index 0000000000000000000000000000000000000000..9da19eacad3b03bb08bbddbbf4ac48dd78b3d838 GIT binary patch literal 68 zcmeAS@N?(olHy`uVBq!ia0vp^j3CUx0wlM}@Gt=>Zci7-kcv6Uzs@r-FtIZ-&5|)J Q1PU{Fy85}Sb4q9e0B4a5jsO4v literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..9da19eacad3b03bb08bbddbbf4ac48dd78b3d838 GIT binary patch literal 68 zcmeAS@N?(olHy`uVBq!ia0vp^j3CUx0wlM}@Gt=>Zci7-kcv6Uzs@r-FtIZ-&5|)J Q1PU{Fy85}Sb4q9e0B4a5jsO4v literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..9da19eacad3b03bb08bbddbbf4ac48dd78b3d838 GIT binary patch literal 68 zcmeAS@N?(olHy`uVBq!ia0vp^j3CUx0wlM}@Gt=>Zci7-kcv6Uzs@r-FtIZ-&5|)J Q1PU{Fy85}Sb4q9e0B4a5jsO4v literal 0 HcmV?d00001 diff --git a/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md new file mode 100644 index 00000000..89c2725b --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md @@ -0,0 +1,5 @@ +# Launch Screen Assets + +You can customize the launch screen with your own desired assets by replacing the image files in this directory. + +You can also do it by opening your Flutter project's Xcode project with `open ios/Runner.xcworkspace`, selecting `Runner/Assets.xcassets` in the Project Navigator and dropping in the desired images. \ No newline at end of file diff --git a/examples/alice_objectbox/ios/Runner/Base.lproj/LaunchScreen.storyboard b/examples/alice_objectbox/ios/Runner/Base.lproj/LaunchScreen.storyboard new file mode 100644 index 00000000..f2e259c7 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Base.lproj/LaunchScreen.storyboard @@ -0,0 +1,37 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/alice_objectbox/ios/Runner/Base.lproj/Main.storyboard b/examples/alice_objectbox/ios/Runner/Base.lproj/Main.storyboard new file mode 100644 index 00000000..f3c28516 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Base.lproj/Main.storyboard @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/examples/alice_objectbox/ios/Runner/Info.plist b/examples/alice_objectbox/ios/Runner/Info.plist new file mode 100644 index 00000000..53398a50 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Info.plist @@ -0,0 +1,49 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleDisplayName + Alice Objectbox + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + alice_objectbox_example + CFBundlePackageType + APPL + CFBundleShortVersionString + $(FLUTTER_BUILD_NAME) + CFBundleSignature + ???? + CFBundleVersion + $(FLUTTER_BUILD_NUMBER) + LSRequiresIPhoneOS + + UILaunchStoryboardName + LaunchScreen + UIMainStoryboardFile + Main + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + CADisableMinimumFrameDurationOnPhone + + UIApplicationSupportsIndirectInputEvents + + + diff --git a/examples/alice_objectbox/ios/Runner/Runner-Bridging-Header.h b/examples/alice_objectbox/ios/Runner/Runner-Bridging-Header.h new file mode 100644 index 00000000..308a2a56 --- /dev/null +++ b/examples/alice_objectbox/ios/Runner/Runner-Bridging-Header.h @@ -0,0 +1 @@ +#import "GeneratedPluginRegistrant.h" diff --git a/examples/alice_objectbox/ios/RunnerTests/RunnerTests.swift b/examples/alice_objectbox/ios/RunnerTests/RunnerTests.swift new file mode 100644 index 00000000..86a7c3b1 --- /dev/null +++ b/examples/alice_objectbox/ios/RunnerTests/RunnerTests.swift @@ -0,0 +1,12 @@ +import Flutter +import UIKit +import XCTest + +class RunnerTests: XCTestCase { + + func testExample() { + // If you add code to the Runner application, consider adding tests here. + // See https://developer.apple.com/documentation/xctest for more information about using XCTest. + } + +} diff --git a/examples/alice_objectbox/lib/main.dart b/examples/alice_objectbox/lib/main.dart new file mode 100644 index 00000000..b3662888 --- /dev/null +++ b/examples/alice_objectbox/lib/main.dart @@ -0,0 +1,197 @@ +import 'package:alice/alice.dart'; +import 'package:alice_http/alice_http_adapter.dart'; +import 'package:alice_http/alice_http_extensions.dart'; +import 'package:alice_objectbox/alice_objectbox.dart'; +import 'package:alice_objectbox/alice_objectbox_store.dart'; +import 'package:flutter/material.dart'; +import 'package:http/http.dart' as http; + +Future main() async { + /// This is required so ObjectBox can get the application directory + /// to store the database in. + WidgetsFlutterBinding.ensureInitialized(); + + /// Initialize [AliceObjectBoxStore] before running the app. + final AliceObjectBoxStore store = + await AliceObjectBoxStore.create(persistent: false); + + /// Pass [AliceObjectBoxStore] to the app. + runApp(MyApp(store: store)); +} + +class MyApp extends StatefulWidget { + const MyApp({ + super.key, + required this.store, + }); + + final AliceObjectBoxStore store; + + @override + State createState() => _MyAppState(); +} + +class _MyAppState extends State { + late final AliceHttpAdapter _aliceHttpAdapter = AliceHttpAdapter(); + + late final Alice _alice = Alice( + showNotification: true, + showInspectorOnShake: true, + maxCallsCount: 1000, + aliceStorage: AliceObjectBox( + store: widget.store, + maxCallsCount: 1000, + ), + )..addAdapter(_aliceHttpAdapter); + + @override + Widget build(BuildContext context) { + return MaterialApp( + navigatorKey: _alice.getNavigatorKey(), + debugShowCheckedModeBanner: false, + home: Scaffold( + appBar: AppBar( + title: const Text('Alice + ObjectBox + HTTP - Example'), + ), + body: Container( + padding: const EdgeInsets.all(16), + child: ListView( + children: [ + const SizedBox(height: 8), + const Text( + style: TextStyle(fontSize: 14), + 'Welcome to example of Alice Http Inspector. ' + 'Click buttons below to generate sample data.', + ), + ElevatedButton( + onPressed: _runHttpHttpRequests, + child: const Text( + 'Run http/http HTTP Requests', + ), + ), + const SizedBox(height: 8), + const Text( + style: TextStyle(fontSize: 14), + 'After clicking on buttons above, you should receive notification.' + ' Click on it to show inspector. ' + 'You can also shake your device or click button below.', + ), + ElevatedButton( + onPressed: _runHttpInspector, + child: const Text( + 'Run HTTP Inspector', + ), + ) + ], + ), + ), + ), + ); + } + + void _runHttpHttpRequests() async { + final Map body = { + 'title': 'foo', + 'body': 'bar', + 'userId': '1' + }; + + http + .post(Uri.https('jsonplaceholder.typicode.com', '/posts'), body: body) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .get(Uri.https('jsonplaceholder.typicode.com', '/posts')) + .interceptWithAlice(_aliceHttpAdapter); + + http + .put(Uri.https('jsonplaceholder.typicode.com', '/posts/1'), body: body) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .patch( + Uri.https('jsonplaceholder.typicode.com', '/posts/1'), + body: body, + ) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .delete(Uri.https('jsonplaceholder.typicode.com', '/posts/1')) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .get(Uri.https('jsonplaceholder.typicode.com', '/test/test')) + .interceptWithAlice(_aliceHttpAdapter); + + http + .post(Uri.https('jsonplaceholder.typicode.com', '/posts'), body: body) + .then((response) => _aliceHttpAdapter.onResponse(response, body: body)); + + http + .get(Uri.https('jsonplaceholder.typicode.com', '/posts')) + .then((response) => _aliceHttpAdapter.onResponse(response)); + + http + .put(Uri.https('jsonplaceholder.typicode.com', '/posts/1'), body: body) + .then((response) => _aliceHttpAdapter.onResponse(response, body: body)); + + http + .patch( + Uri.https('jsonplaceholder.typicode.com', '/posts/1'), + body: body, + ) + .then((response) => _aliceHttpAdapter.onResponse(response, body: body)); + + http + .delete(Uri.https('jsonplaceholder.typicode.com', '/posts/1')) + .then((response) => _aliceHttpAdapter.onResponse(response)); + + http + .get(Uri.https('jsonplaceholder.typicode.com', '/test/test')) + .then((response) => _aliceHttpAdapter.onResponse(response)); + + http + .post( + Uri.https( + 'jsonplaceholder.typicode.com', + '/posts', + {'key1': 'value1'}, + ), + body: body, + ) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .post( + Uri.https( + 'jsonplaceholder.typicode.com', + '/posts', + { + 'key1': 'value1', + 'key2': 'value2', + 'key3': 'value3', + }, + ), + body: body, + ) + .interceptWithAlice(_aliceHttpAdapter, body: body); + + http + .get( + Uri.https( + 'jsonplaceholder.typicode.com', + '/test/test', + { + 'key1': 'value1', + 'key2': 'value2', + 'key3': 'value3', + }, + ), + ) + .then((response) => _aliceHttpAdapter.onResponse(response)); + } + + void _runHttpInspector() { + _alice.showInspector(); + } +} diff --git a/examples/alice_objectbox/pubspec.yaml b/examples/alice_objectbox/pubspec.yaml new file mode 100644 index 00000000..a0e9af87 --- /dev/null +++ b/examples/alice_objectbox/pubspec.yaml @@ -0,0 +1,32 @@ +name: alice_objectbox_example +description: "Alice + ObjectBox + HTTP package - Example" +publish_to: 'none' # Remove this line if you wish to publish to pub.dev +version: 1.0.0+1 + +environment: + sdk: ^3.0.0 + +dependencies: + flutter: + sdk: flutter + alice: + alice_http: + alice_objectbox: + cupertino_icons: ^1.0.8 + http: ^1.2.1 + +dev_dependencies: + flutter_test: + sdk: flutter + flutter_lints: ^4.0.0 + +dependency_overrides: + alice: + path: ../../packages/alice + alice_http: + path: ../../packages/alice_http + alice_objectbox: + path: ../../packages/alice_objectbox + +flutter: + uses-material-design: true diff --git a/packages/alice/lib/alice.dart b/packages/alice/lib/alice.dart index cb9881ce..43c8cc37 100644 --- a/packages/alice/lib/alice.dart +++ b/packages/alice/lib/alice.dart @@ -1,10 +1,13 @@ import 'package:alice/core/alice_adapter.dart'; import 'package:alice/core/alice_core.dart'; +import 'package:alice/core/alice_memory_storage.dart'; +import 'package:alice/core/alice_storage.dart'; import 'package:alice/model/alice_http_call.dart'; import 'package:alice/model/alice_log.dart'; import 'package:flutter/material.dart'; import 'package:flutter/widgets.dart'; +export 'package:alice/core/alice_store.dart'; export 'package:alice/model/alice_log.dart'; class Alice { @@ -31,7 +34,9 @@ class Alice { final bool? showShareButton; GlobalKey? _navigatorKey; - late AliceCore _aliceCore; + late final AliceCore _aliceCore; + + final AliceStorage? _aliceStorage; /// Creates alice instance. Alice({ @@ -42,8 +47,9 @@ class Alice { this.maxCallsCount = 1000, this.directionality, this.showShareButton = true, - }) { - _navigatorKey = navigatorKey ?? GlobalKey(); + AliceStorage? aliceStorage, + }) : _navigatorKey = navigatorKey ?? GlobalKey(), + _aliceStorage = aliceStorage { _aliceCore = AliceCore( _navigatorKey, showNotification: showNotification, @@ -52,6 +58,10 @@ class Alice { maxCallsCount: maxCallsCount, directionality: directionality, showShareButton: showShareButton, + aliceStorage: _aliceStorage ?? + AliceMemoryStorage( + maxCallsCount: maxCallsCount, + ), ); } @@ -62,40 +72,29 @@ class Alice { } /// Get currently used navigation key - GlobalKey? getNavigatorKey() { - return _navigatorKey; - } + GlobalKey? getNavigatorKey() => _navigatorKey; /// Opens Http calls inspector. This will navigate user to the new fullscreen /// page where all listened http calls can be viewed. - void showInspector() { - _aliceCore.navigateToCallListScreen(); - } + void showInspector() => _aliceCore.navigateToCallListScreen(); /// Handle generic http call. Can be used to any http client. void addHttpCall(AliceHttpCall aliceHttpCall) { assert(aliceHttpCall.request != null, "Http call request can't be null"); assert(aliceHttpCall.response != null, "Http call response can't be null"); + _aliceCore.addCall(aliceHttpCall); } /// Adds new log to Alice logger. - void addLog(AliceLog log) { - _aliceCore.addLog(log); - } + void addLog(AliceLog log) => _aliceCore.addLog(log); /// Adds list of logs to Alice logger - void addLogs(List logs) { - _aliceCore.addLogs(logs); - } + void addLogs(List logs) => _aliceCore.addLogs(logs); /// Returns flag which determines whether inspector is opened - bool isInspectorOpened() { - return _aliceCore.isInspectorOpened(); - } + bool get isInspectorOpened => _aliceCore.isInspectorOpened; /// Adds new adapter to Alice. - void addAdapter(AliceAdapter adapter) { - adapter.injectCore(_aliceCore); - } + void addAdapter(AliceAdapter adapter) => adapter.injectCore(_aliceCore); } diff --git a/packages/alice/lib/core/alice_core.dart b/packages/alice/lib/core/alice_core.dart index 12c88eee..f7ac926e 100644 --- a/packages/alice/lib/core/alice_core.dart +++ b/packages/alice/lib/core/alice_core.dart @@ -2,6 +2,7 @@ import 'dart:async' show StreamSubscription; import 'dart:io' show Platform; import 'package:alice/core/alice_logger.dart'; +import 'package:alice/core/alice_storage.dart'; import 'package:alice/core/alice_utils.dart'; import 'package:alice/helper/alice_save_helper.dart'; import 'package:alice/model/alice_http_call.dart'; @@ -9,12 +10,11 @@ import 'package:alice/model/alice_http_error.dart'; import 'package:alice/model/alice_http_response.dart'; import 'package:alice/model/alice_log.dart'; import 'package:alice/ui/common/alice_navigation.dart'; -import 'package:alice/utils/num_comparison.dart'; import 'package:alice/utils/shake_detector.dart'; -import 'package:collection/collection.dart' show IterableExtension; import 'package:flutter/material.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; -import 'package:rxdart/rxdart.dart'; + +typedef AliceOnCallsChanged = Future Function(List? calls); class AliceCore { /// Should user be notified with notification if there's new request caught @@ -25,13 +25,23 @@ class AliceCore { /// with sensors) final bool showInspectorOnShake; - /// Rx subject which contains all intercepted http calls - final BehaviorSubject> callsSubject = - BehaviorSubject.seeded([]); - /// Icon url for notification final String notificationIcon; + final AliceStorage _aliceStorage; + + late final NotificationDetails _notificationDetails = NotificationDetails( + android: AndroidNotificationDetails( + 'Alice', + 'Alice', + channelDescription: 'Alice', + enableVibration: false, + playSound: false, + largeIcon: DrawableResourceAndroidBitmap(notificationIcon), + ), + iOS: const DarwinNotificationDetails(presentSound: false), + ); + ///Max number of calls that are stored in memory. When count is reached, FIFO ///method queue will be used to remove elements. final int maxCallsCount; @@ -44,13 +54,20 @@ class AliceCore { final AliceLogger _aliceLogger = AliceLogger(); - late final FlutterLocalNotificationsPlugin _flutterLocalNotificationsPlugin; + FlutterLocalNotificationsPlugin? _flutterLocalNotificationsPlugin; + GlobalKey? navigatorKey; + bool _isInspectorOpened = false; + ShakeDetector? _shakeDetector; - StreamSubscription? _callsSubscription; + + StreamSubscription>? _callsSubscription; + String? _notificationMessage; + String? _notificationMessageShown; + bool _notificationProcessing = false; /// Creates alice core instance @@ -60,13 +77,14 @@ class AliceCore { required this.showInspectorOnShake, required this.notificationIcon, required this.maxCallsCount, + required AliceStorage aliceStorage, this.directionality, this.showShareButton, - }) { + }) : _aliceStorage = aliceStorage { if (showNotification) { _initializeNotificationsPlugin(); _requestNotificationPermissions(); - _callsSubscription = callsSubject.listen((_) => _onCallsChanged()); + _aliceStorage.subscribeToCallChanges(onCallsChanged); } if (showInspectorOnShake) { if (Platform.isAndroid || Platform.isIOS) { @@ -80,11 +98,11 @@ class AliceCore { /// Dispose subjects and subscriptions void dispose() { - callsSubject.close(); _shakeDetector?.stopListening(); _callsSubscription?.cancel(); } + @protected void _initializeNotificationsPlugin() { _flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); final AndroidInitializationSettings initializationSettingsAndroid = @@ -99,19 +117,21 @@ class AliceCore { iOS: initializationSettingsIOS, macOS: initializationSettingsMacOS, ); - _flutterLocalNotificationsPlugin.initialize( + _flutterLocalNotificationsPlugin?.initialize( initializationSettings, onDidReceiveNotificationResponse: _onDidReceiveNotificationResponse, ); } - Future _onCallsChanged() async { - if (callsSubject.value.isNotEmpty) { - _notificationMessage = _getNotificationMessage(); + @protected + Future onCallsChanged(List? calls) async { + if (calls != null && calls.isNotEmpty) { + final AliceStats stats = _aliceStorage.getStats(); + + _notificationMessage = _getNotificationMessage(stats); if (_notificationMessage != _notificationMessageShown && !_notificationProcessing) { - await _showLocalNotification(); - await _onCallsChanged(); + await _showLocalNotification(stats); } } } @@ -144,73 +164,17 @@ class AliceCore { /// Get context from navigator key. Used to open inspector route. BuildContext? getContext() => navigatorKey?.currentState?.overlay?.context; - String _getNotificationMessage() { - final List calls = callsSubject.value; - final int successCalls = calls - .where( - (AliceHttpCall call) => - (call.response?.status.gte(200) ?? false) && - (call.response?.status.lt(300) ?? false), - ) - .toList() - .length; - - final int redirectCalls = calls - .where((AliceHttpCall call) => - (call.response?.status.gte(300) ?? false) && - (call.response?.status.lt(400) ?? false)) - .toList() - .length; - - final int errorCalls = calls - .where( - (AliceHttpCall call) => - (call.response?.status.gte(400) ?? false) && - (call.response?.status.lt(600) ?? false), - ) - .toList() - .length; - - final int loadingCalls = - calls.where((call) => call.loading).toList().length; - - final StringBuffer notificationsMessage = StringBuffer(); - if (loadingCalls > 0) { - notificationsMessage.writeAll([ - 'Loading: $loadingCalls', - ' | ', - ]); - } - if (successCalls > 0) { - notificationsMessage.writeAll([ - 'Success: $successCalls', - ' | ', - ]); - } - if (redirectCalls > 0) { - notificationsMessage.writeAll([ - 'Redirect: $redirectCalls', - ' | ', - ]); - } - if (errorCalls > 0) { - notificationsMessage.write('Error: $errorCalls'); - } - String notificationMessageString = notificationsMessage.toString(); - if (notificationMessageString.endsWith(' | ')) { - notificationMessageString = notificationMessageString.substring( - 0, - notificationMessageString.length - 3, - ); - } - - return notificationMessageString; - } + String _getNotificationMessage(AliceStats stats) => [ + if (stats.loading > 0) 'Loading: ${stats.loading}', + if (stats.successes > 0) 'Success: ${stats.successes}', + if (stats.redirects > 0) 'Redirect: ${stats.redirects}', + if (stats.errors > 0) 'Error: ${stats.errors}', + ].join(' | '); Future _requestNotificationPermissions() async { if (Platform.isIOS || Platform.isMacOS) { await _flutterLocalNotificationsPlugin - .resolvePlatformSpecificImplementation< + ?.resolvePlatformSpecificImplementation< IOSFlutterLocalNotificationsPlugin>() ?.requestPermissions( alert: true, @@ -218,7 +182,7 @@ class AliceCore { sound: true, ); await _flutterLocalNotificationsPlugin - .resolvePlatformSpecificImplementation< + ?.resolvePlatformSpecificImplementation< MacOSFlutterLocalNotificationsPlugin>() ?.requestPermissions( alert: true, @@ -228,123 +192,70 @@ class AliceCore { } else if (Platform.isAndroid) { final AndroidFlutterLocalNotificationsPlugin? androidImplementation = _flutterLocalNotificationsPlugin - .resolvePlatformSpecificImplementation< + ?.resolvePlatformSpecificImplementation< AndroidFlutterLocalNotificationsPlugin>(); await androidImplementation?.requestNotificationsPermission(); } } - Future _showLocalNotification() async { - _notificationProcessing = true; - const String channelId = 'Alice'; - const String channelName = 'Alice'; - const String channelDescription = 'Alice'; - final AndroidNotificationDetails androidPlatformChannelSpecifics = - AndroidNotificationDetails( - channelId, - channelName, - channelDescription: channelDescription, - enableVibration: false, - playSound: false, - largeIcon: DrawableResourceAndroidBitmap(notificationIcon), - ); - const DarwinNotificationDetails iOSPlatformChannelSpecifics = - DarwinNotificationDetails(presentSound: false); - final NotificationDetails platformChannelSpecifics = NotificationDetails( - android: androidPlatformChannelSpecifics, - iOS: iOSPlatformChannelSpecifics, - ); - final String? message = _notificationMessage; - await _flutterLocalNotificationsPlugin.show( - 0, - 'Alice (total: ${callsSubject.value.length} requests)', - message, - platformChannelSpecifics, - payload: '', - ); + Future _showLocalNotification(AliceStats stats) async { + try { + _notificationProcessing = true; - _notificationMessageShown = message; - _notificationProcessing = false; - } + final String? message = _notificationMessage; - /// Add alice http call to calls subject - void addCall(AliceHttpCall call) { - final int callsCount = callsSubject.value.length; - if (callsCount >= maxCallsCount) { - final List originalCalls = callsSubject.value; - final List calls = [...originalCalls]..sort( - (AliceHttpCall call1, AliceHttpCall call2) => - call1.createdTime.compareTo(call2.createdTime), - ); - final int indexToReplace = originalCalls.indexOf(calls.first); - originalCalls[indexToReplace] = call; + await _flutterLocalNotificationsPlugin?.show( + 0, + 'Alice (total: ${stats.total} requests)', + message, + _notificationDetails, + payload: '', + ); - callsSubject.add(originalCalls); - } else { - callsSubject.add([...callsSubject.value, call]); + _notificationMessageShown = message; + } finally { + _notificationProcessing = false; } } - /// Add error to existing alice http call - void addError(AliceHttpError error, int requestId) { - final AliceHttpCall? selectedCall = _selectCall(requestId); - - if (selectedCall == null) { - return AliceUtils.log('Selected call is null'); - } + /// Add alice http call to calls subject + void addCall(AliceHttpCall call) => _aliceStorage.addCall(call); - selectedCall.error = error; - callsSubject.add([...callsSubject.value]); - } + /// Add error to existing alice http call + void addError(AliceHttpError error, int requestId) => + _aliceStorage.addError(error, requestId); /// Add response to existing alice http call - void addResponse(AliceHttpResponse response, int requestId) { - final AliceHttpCall? selectedCall = _selectCall(requestId); - - if (selectedCall == null) { - return AliceUtils.log('Selected call is null'); - } - - selectedCall - ..loading = false - ..response = response - ..duration = response.time.millisecondsSinceEpoch - - (selectedCall.request?.time.millisecondsSinceEpoch ?? 0); - - callsSubject.add([...callsSubject.value]); - } + void addResponse(AliceHttpResponse response, int requestId) => + _aliceStorage.addResponse(response, requestId); /// Add alice http call to calls subject - void addHttpCall(AliceHttpCall aliceHttpCall) { - assert(aliceHttpCall.request != null, "Http call request can't be null"); - assert(aliceHttpCall.response != null, "Http call response can't be null"); - callsSubject.add([...callsSubject.value, aliceHttpCall]); - } + void addHttpCall(AliceHttpCall aliceHttpCall) => + _aliceStorage.addHttpCall(aliceHttpCall); /// Remove all calls from calls subject - void removeCalls() => callsSubject.add([]); + void removeCalls() => _aliceStorage.removeCalls(); + + @protected + AliceHttpCall? selectCall(int requestId) => + _aliceStorage.selectCall(requestId); + + Stream> get callsStream => _aliceStorage.callsStream; - AliceHttpCall? _selectCall(int requestId) => callsSubject.value - .firstWhereOrNull((AliceHttpCall call) => call.id == requestId); + List getCalls() => _aliceStorage.getCalls(); /// Save all calls to file void saveHttpRequests(BuildContext context) { - AliceSaveHelper.saveCalls(context, callsSubject.value); + AliceSaveHelper.saveCalls(context, _aliceStorage.getCalls()); } /// Adds new log to Alice logger. - void addLog(AliceLog log) { - _aliceLogger.logs.add(log); - } + void addLog(AliceLog log) => _aliceLogger.logs.add(log); /// Adds list of logs to Alice logger - void addLogs(List logs) { - _aliceLogger.logs.addAll(logs); - } + void addLogs(List logs) => _aliceLogger.logs.addAll(logs); /// Returns flag which determines whether inspector is opened - bool isInspectorOpened() { - return _isInspectorOpened; - } + bool get isInspectorOpened => _isInspectorOpened; } diff --git a/packages/alice/lib/core/alice_memory_storage.dart b/packages/alice/lib/core/alice_memory_storage.dart new file mode 100644 index 00000000..eea6ab08 --- /dev/null +++ b/packages/alice/lib/core/alice_memory_storage.dart @@ -0,0 +1,121 @@ +import 'package:alice/core/alice_core.dart'; +import 'package:alice/core/alice_storage.dart'; +import 'package:alice/core/alice_utils.dart'; +import 'package:alice/model/alice_http_call.dart'; +import 'package:alice/model/alice_http_error.dart'; +import 'package:alice/model/alice_http_response.dart'; +import 'package:alice/utils/num_comparison.dart'; +import 'package:collection/collection.dart'; +import 'package:rxdart/subjects.dart'; + +class AliceMemoryStorage implements AliceStorage { + AliceMemoryStorage({ + required this.maxCallsCount, + }) : callsSubject = BehaviorSubject.seeded([]), + assert(maxCallsCount > 0, 'Max calls count should be greater than 0'); + + @override + final int maxCallsCount; + + final BehaviorSubject> callsSubject; + + @override + Stream> get callsStream => callsSubject.stream; + + @override + List getCalls() => callsSubject.value; + + @override + AliceStats getStats() { + final List calls = getCalls(); + + return ( + total: calls.length, + successes: calls + .where( + (AliceHttpCall call) => + (call.response?.status.gte(200) ?? false) && + (call.response?.status.lt(300) ?? false), + ) + .length, + redirects: calls + .where((AliceHttpCall call) => + (call.response?.status.gte(300) ?? false) && + (call.response?.status.lt(400) ?? false)) + .length, + errors: calls + .where( + (AliceHttpCall call) => + (call.response?.status.gte(400) ?? false) && + (call.response?.status.lt(600) ?? false), + ) + .length, + loading: calls.where((AliceHttpCall call) => call.loading).length, + ); + } + + @override + void addCall(AliceHttpCall call) { + final int callsCount = callsSubject.value.length; + if (callsCount >= maxCallsCount) { + final List originalCalls = callsSubject.value; + final List calls = [...originalCalls]..sort( + (AliceHttpCall call1, AliceHttpCall call2) => + call1.createdTime.compareTo(call2.createdTime), + ); + final int indexToReplace = originalCalls.indexOf(calls.first); + originalCalls[indexToReplace] = call; + + callsSubject.add(originalCalls); + } else { + callsSubject.add([...callsSubject.value, call]); + } + } + + @override + void addError(AliceHttpError error, int requestId) { + final AliceHttpCall? selectedCall = selectCall(requestId); + + if (selectedCall == null) { + return AliceUtils.log('Selected call is null'); + } + + selectedCall.error = error; + callsSubject.add([...callsSubject.value]); + } + + @override + void addHttpCall(AliceHttpCall aliceHttpCall) { + assert(aliceHttpCall.request != null, "Http call request can't be null"); + assert(aliceHttpCall.response != null, "Http call response can't be null"); + callsSubject.add([...callsSubject.value, aliceHttpCall]); + } + + @override + void addResponse(AliceHttpResponse response, int requestId) { + final AliceHttpCall? selectedCall = selectCall(requestId); + + if (selectedCall == null) { + return AliceUtils.log('Selected call is null'); + } + + selectedCall + ..loading = false + ..response = response + ..duration = response.time.millisecondsSinceEpoch - + (selectedCall.request?.time.millisecondsSinceEpoch ?? 0); + + callsSubject.add([...callsSubject.value]); + } + + @override + void removeCalls() => callsSubject.add([]); + + @override + AliceHttpCall? selectCall(int requestId) => callsSubject.value + .firstWhereOrNull((AliceHttpCall call) => call.id == requestId); + + @override + void subscribeToCallChanges(AliceOnCallsChanged callback) => + callsSubject.listen(callback); +} diff --git a/packages/alice/lib/core/alice_storage.dart b/packages/alice/lib/core/alice_storage.dart new file mode 100644 index 00000000..ea8bb4d5 --- /dev/null +++ b/packages/alice/lib/core/alice_storage.dart @@ -0,0 +1,36 @@ +import 'package:alice/core/alice_core.dart'; +import 'package:alice/model/alice_http_call.dart'; +import 'package:alice/model/alice_http_error.dart'; +import 'package:alice/model/alice_http_response.dart'; + +typedef AliceStats = ({ + int total, + int successes, + int redirects, + int errors, + int loading, +}); + +abstract interface class AliceStorage { + abstract final Stream> callsStream; + + abstract final int maxCallsCount; + + List getCalls(); + + AliceStats getStats(); + + AliceHttpCall? selectCall(int requestId); + + void addCall(AliceHttpCall call); + + void addError(AliceHttpError error, int requestId); + + void addResponse(AliceHttpResponse response, int requestId); + + void addHttpCall(AliceHttpCall aliceHttpCall); + + void removeCalls(); + + void subscribeToCallChanges(AliceOnCallsChanged callback); +} diff --git a/packages/alice/lib/core/alice_store.dart b/packages/alice/lib/core/alice_store.dart new file mode 100644 index 00000000..1d97f0f9 --- /dev/null +++ b/packages/alice/lib/core/alice_store.dart @@ -0,0 +1,3 @@ +abstract interface class AliceStore { + void clear(); +} diff --git a/packages/alice/lib/helper/alice_save_helper.dart b/packages/alice/lib/helper/alice_save_helper.dart index 23026f72..5239d8f2 100644 --- a/packages/alice/lib/helper/alice_save_helper.dart +++ b/packages/alice/lib/helper/alice_save_helper.dart @@ -7,6 +7,7 @@ import 'package:alice/helper/operating_system.dart'; import 'package:alice/model/alice_http_call.dart'; import 'package:alice/ui/common/alice_dialog.dart'; import 'package:alice/utils/alice_parser.dart'; +import 'package:alice/utils/curl.dart'; import 'package:flutter/material.dart'; import 'package:open_filex/open_filex.dart'; import 'package:package_info_plus/package_info_plus.dart'; @@ -207,7 +208,7 @@ class AliceSaveHelper { '--------------------------------------------\n', 'Curl\n', '--------------------------------------------\n', - call.getCurlCommand(), + getCurlCommand(call), '\n', '==============================================\n', '\n', diff --git a/packages/alice/lib/model/alice_http_call.dart b/packages/alice/lib/model/alice_http_call.dart index 2aa34eea..eac7bb6e 100644 --- a/packages/alice/lib/model/alice_http_call.dart +++ b/packages/alice/lib/model/alice_http_call.dart @@ -1,5 +1,3 @@ -import 'dart:io' show HttpHeaders; - import 'package:alice/model/alice_http_error.dart'; import 'package:alice/model/alice_http_request.dart'; import 'package:alice/model/alice_http_response.dart'; @@ -30,61 +28,4 @@ class AliceHttpCall { this.response = response; loading = false; } - - String getCurlCommand() { - bool compressed = false; - final StringBuffer curlCmd = StringBuffer('curl'); - - curlCmd.write(' -X $method'); - - for (final MapEntry header - in request?.headers.entries ?? []) { - if (header.key.toLowerCase() == HttpHeaders.acceptEncodingHeader && - header.value.toString().toLowerCase() == 'gzip') { - compressed = true; - } - - curlCmd.write(' -H "${header.key}: ${header.value}"'); - } - - final String? requestBody = request?.body.toString(); - if (requestBody?.isNotEmpty ?? false) { - // try to keep to a single line and use a subshell to preserve any line - // breaks - curlCmd.write(" --data \$'${requestBody?.replaceAll("\n", r"\n")}'"); - } - - final Map? queryParamMap = request?.queryParameters; - int paramCount = queryParamMap?.keys.length ?? 0; - final StringBuffer queryParams = StringBuffer(); - - if (paramCount > 0) { - queryParams.write('?'); - for (final MapEntry queryParam - in queryParamMap?.entries ?? []) { - queryParams.write('${queryParam.key}=${queryParam.value}'); - paramCount--; - if (paramCount > 0) { - queryParams.write('&'); - } - } - } - - // If server already has http(s) don't add it again - if (server.contains('http://') || server.contains('https://')) { - // ignore: join_return_with_assignment - curlCmd.write( - "${compressed ? " --compressed " : " "}" - "${"'$server$endpoint$queryParams'"}", - ); - } else { - // ignore: join_return_with_assignment - curlCmd.write( - "${compressed ? " --compressed " : " "}" - "${"'${secure ? 'https://' : 'http://'}$server$endpoint$queryParams'"}", - ); - } - - return curlCmd.toString(); - } } diff --git a/packages/alice/lib/ui/call_details/page/alice_call_details_page.dart b/packages/alice/lib/ui/call_details/page/alice_call_details_page.dart index d664f924..077931cf 100644 --- a/packages/alice/lib/ui/call_details/page/alice_call_details_page.dart +++ b/packages/alice/lib/ui/call_details/page/alice_call_details_page.dart @@ -37,7 +37,7 @@ class _AliceCallDetailsPageState extends State return AlicePage( core: widget.core, child: StreamBuilder>( - stream: widget.core.callsSubject, + stream: widget.core.callsStream, initialData: [widget.call], builder: (context, AsyncSnapshot> callsSnapshot) { if (callsSnapshot.hasData && !callsSnapshot.hasError) { diff --git a/packages/alice/lib/ui/calls_list/page/alice_calls_list_page.dart b/packages/alice/lib/ui/calls_list/page/alice_calls_list_page.dart index 79ae2bb0..c9fd7aa2 100644 --- a/packages/alice/lib/ui/calls_list/page/alice_calls_list_page.dart +++ b/packages/alice/lib/ui/calls_list/page/alice_calls_list_page.dart @@ -4,12 +4,11 @@ import 'package:alice/model/alice_http_call.dart'; import 'package:alice/ui/call_details/model/alice_menu_item.dart'; import 'package:alice/ui/calls_list/model/alice_calls_list_sort_option.dart'; import 'package:alice/ui/calls_list/model/alice_calls_list_tab_item.dart'; +import 'package:alice/ui/calls_list/widget/alice_inspector_screen.dart'; import 'package:alice/ui/calls_list/widget/alice_sort_dialog.dart'; import 'package:alice/ui/common/alice_dialog.dart'; import 'package:alice/ui/common/alice_navigation.dart'; import 'package:alice/ui/common/alice_page.dart'; -import 'package:alice/ui/calls_list/widget/alice_calls_list_screen.dart'; -import 'package:alice/ui/calls_list/widget/alice_empty_logs_widget.dart'; import 'package:alice/ui/calls_list/widget/alice_logs_screen.dart'; import 'package:alice/utils/alice_theme.dart'; import 'package:flutter/material.dart'; @@ -123,27 +122,12 @@ class _AliceCallsListPageState extends State body: TabBarView( controller: _tabController, children: [ - StreamBuilder>( - stream: aliceCore.callsSubject, - builder: (context, AsyncSnapshot> snapshot) { - final List calls = snapshot.data ?? []; - final String query = _queryTextEditingController.text.trim(); - if (query.isNotEmpty) { - calls.removeWhere((AliceHttpCall call) => !call.endpoint - .toLowerCase() - .contains(query.toLowerCase())); - } - if (calls.isNotEmpty) { - return AliceCallsListScreen( - calls: calls, - sortOption: _sortOption, - sortAscending: _sortAscending, - onListItemClicked: _onListItemPressed, - ); - } else { - return const AliceEmptyLogsWidget(); - } - }, + AliceInspectorScreen( + aliceCore: aliceCore, + queryTextEditingController: _queryTextEditingController, + sortOption: _sortOption, + sortAscending: _sortAscending, + onListItemPressed: _onListItemPressed, ), AliceLogsScreen( scrollController: _scrollController, diff --git a/packages/alice/lib/ui/calls_list/widget/alice_inspector_screen.dart b/packages/alice/lib/ui/calls_list/widget/alice_inspector_screen.dart new file mode 100644 index 00000000..a90b8c6a --- /dev/null +++ b/packages/alice/lib/ui/calls_list/widget/alice_inspector_screen.dart @@ -0,0 +1,59 @@ +import 'package:alice/core/alice_core.dart'; +import 'package:alice/model/alice_http_call.dart'; +import 'package:alice/ui/calls_list/model/alice_calls_list_sort_option.dart'; +import 'package:alice/ui/calls_list/widget/alice_calls_list_screen.dart'; +import 'package:alice/ui/calls_list/widget/alice_empty_logs_widget.dart'; +import 'package:flutter/material.dart'; + +class AliceInspectorScreen extends StatefulWidget { + const AliceInspectorScreen({ + super.key, + required this.aliceCore, + required this.queryTextEditingController, + required this.sortOption, + required this.sortAscending, + required this.onListItemPressed, + }); + + final AliceCore aliceCore; + final TextEditingController queryTextEditingController; + final AliceCallsListSortOption sortOption; + final bool sortAscending; + final void Function(AliceHttpCall) onListItemPressed; + + @override + State createState() => _AliceInspectorScreenState(); +} + +class _AliceInspectorScreenState extends State + with AutomaticKeepAliveClientMixin { + @override + bool get wantKeepAlive => true; + + @override + Widget build(BuildContext context) { + super.build(context); + + return StreamBuilder>( + stream: widget.aliceCore.callsStream, + builder: (context, AsyncSnapshot> snapshot) { + final List calls = snapshot.data ?? []; + final String query = widget.queryTextEditingController.text.trim(); + if (query.isNotEmpty) { + calls.removeWhere((AliceHttpCall call) => + !call.endpoint.toLowerCase().contains(query.toLowerCase())); + } + if (calls.isNotEmpty) { + return AliceCallsListScreen( + calls: calls, + sortOption: widget.sortOption, + sortAscending: widget.sortAscending, + onListItemClicked: widget.onListItemPressed, + ); + } else { + return const AliceEmptyLogsWidget(); + } + }, + ); + } +} diff --git a/packages/alice/lib/ui/stats/alice_stats_page.dart b/packages/alice/lib/ui/stats/alice_stats_page.dart index d139046e..d969a5d3 100644 --- a/packages/alice/lib/ui/stats/alice_stats_page.dart +++ b/packages/alice/lib/ui/stats/alice_stats_page.dart @@ -174,5 +174,5 @@ class AliceStatsPage extends StatelessWidget { _calls.where((call) => !call.secure).toList().length; /// Get all calls from Alice. - List get _calls => aliceCore.callsSubject.value; + List get _calls => aliceCore.getCalls(); } diff --git a/packages/alice/lib/utils/curl.dart b/packages/alice/lib/utils/curl.dart new file mode 100644 index 00000000..10877ad7 --- /dev/null +++ b/packages/alice/lib/utils/curl.dart @@ -0,0 +1,60 @@ +import 'dart:io' show HttpHeaders; + +import 'package:alice/model/alice_http_call.dart'; + +String getCurlCommand(AliceHttpCall call) { + bool compressed = false; + final StringBuffer curlCmd = StringBuffer('curl'); + + curlCmd.write(' -X ${call.method}'); + + for (final MapEntry header + in call.request?.headers.entries ?? []) { + if (header.key.toLowerCase() == HttpHeaders.acceptEncodingHeader && + header.value.toString().toLowerCase() == 'gzip') { + compressed = true; + } + + curlCmd.write(' -H "${header.key}: ${header.value}"'); + } + + final String? requestBody = call.request?.body.toString(); + if (requestBody?.isNotEmpty ?? false) { + // try to keep to a single line and use a subshell to preserve any line + // breaks + curlCmd.write(" --data \$'${requestBody?.replaceAll("\n", r"\n")}'"); + } + + final Map? queryParamMap = call.request?.queryParameters; + int paramCount = queryParamMap?.keys.length ?? 0; + final StringBuffer queryParams = StringBuffer(); + + if (paramCount > 0) { + queryParams.write('?'); + for (final MapEntry queryParam + in queryParamMap?.entries ?? []) { + queryParams.write('${queryParam.key}=${queryParam.value}'); + paramCount--; + if (paramCount > 0) { + queryParams.write('&'); + } + } + } + + // If server already has http(s) don't add it again + if (call.server.contains('http://') || call.server.contains('https://')) { + // ignore: join_return_with_assignment + curlCmd.write( + "${compressed ? " --compressed " : " "}" + "${"'${call.server}${call.endpoint}$queryParams'"}", + ); + } else { + // ignore: join_return_with_assignment + curlCmd.write( + "${compressed ? " --compressed " : " "}" + "${"'${call.secure ? 'https://' : 'http://'}${call.server}${call.endpoint}$queryParams'"}", + ); + } + + return curlCmd.toString(); +} diff --git a/packages/alice_objectbox/.gitignore b/packages/alice_objectbox/.gitignore new file mode 100644 index 00000000..ac5aa989 --- /dev/null +++ b/packages/alice_objectbox/.gitignore @@ -0,0 +1,29 @@ +# Miscellaneous +*.class +*.log +*.pyc +*.swp +.DS_Store +.atom/ +.buildlog/ +.history +.svn/ +migrate_working_dir/ + +# 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 +# Libraries should not include pubspec.lock, per https://dart.dev/guides/libraries/private-files#pubspeclock. +/pubspec.lock +**/doc/api/ +.dart_tool/ +build/ diff --git a/packages/alice_objectbox/CHANGELOG.md b/packages/alice_objectbox/CHANGELOG.md new file mode 100644 index 00000000..effe43c8 --- /dev/null +++ b/packages/alice_objectbox/CHANGELOG.md @@ -0,0 +1,3 @@ +## 1.0.0 + +- Initial version. diff --git a/packages/alice_objectbox/LICENSE b/packages/alice_objectbox/LICENSE new file mode 100644 index 00000000..261eeb9e --- /dev/null +++ b/packages/alice_objectbox/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/packages/alice_objectbox/README.md b/packages/alice_objectbox/README.md new file mode 100644 index 00000000..0c650610 --- /dev/null +++ b/packages/alice_objectbox/README.md @@ -0,0 +1,4 @@ +# Alice ObjectBox + +## Documentation +You can find documentation [here.](https://jhomlala.github.io/alice/) diff --git a/packages/alice_objectbox/analysis_options.yaml b/packages/alice_objectbox/analysis_options.yaml new file mode 100644 index 00000000..ae9086b0 --- /dev/null +++ b/packages/alice_objectbox/analysis_options.yaml @@ -0,0 +1,10 @@ +analyzer: + exclude: + - "lib/generated_plugin_registrant.dart" + - "**.g.dart" + +include: package:flutter_lints/flutter.yaml + +linter: + rules: + #library_annotations: false diff --git a/packages/alice_objectbox/lib/alice_objectbox.dart b/packages/alice_objectbox/lib/alice_objectbox.dart new file mode 100644 index 00000000..83a95365 --- /dev/null +++ b/packages/alice_objectbox/lib/alice_objectbox.dart @@ -0,0 +1,137 @@ +import 'dart:math' show max; + +import 'package:alice/core/alice_core.dart'; +import 'package:alice/core/alice_storage.dart'; +import 'package:alice/core/alice_utils.dart'; +import 'package:alice/model/alice_http_call.dart'; +import 'package:alice/model/alice_http_error.dart'; +import 'package:alice/model/alice_http_response.dart'; +import 'package:alice_objectbox/alice_objectbox_store.dart'; +import 'package:alice_objectbox/model/cached_alice_http_call.dart'; +import 'package:alice_objectbox/objectbox.g.dart'; + +class AliceObjectBox implements AliceStorage { + const AliceObjectBox({ + required AliceObjectBoxStore store, + required this.maxCallsCount, + }) : _store = store, + assert(maxCallsCount > 0, 'Max calls count should be greater than 0'); + + final AliceObjectBoxStore _store; + + @override + final int maxCallsCount; + + @override + Stream> get callsStream => _store.httpCalls + .query() + .order(CachedAliceHttpCall_.createdTime, flags: Order.descending) + .watch(triggerImmediately: true) + .map((Query query) => query.find()) + .asBroadcastStream(); + + @override + List getCalls() => _store.httpCalls.getAll(); + + @override + CachedAliceHttpCall? selectCall(int requestId) => _store.httpCalls + .query(CachedAliceHttpCall_.id.equals(requestId)) + .build() + .findFirst(); + + @override + void addCall(AliceHttpCall call) { + if (maxCallsCount > 0 && _store.httpCalls.count() >= maxCallsCount) { + final Query overQuota = _store.httpCalls + .query() + .order(CachedAliceHttpCall_.createdTime, flags: Order.descending) + .build() + ..offset = max(maxCallsCount - 1, 0); + + overQuota.remove(); + } + + _store.httpCalls.put(CachedAliceHttpCall.fromAliceHttpCall(call)); + } + + @override + void addError(AliceHttpError error, int requestId) { + final CachedAliceHttpCall? selectedCall = selectCall(requestId); + + if (selectedCall != null) { + selectedCall.error = error; + + _store.httpCalls.put(selectedCall); + } else { + AliceUtils.log('Selected call is null'); + } + } + + @override + void addResponse(AliceHttpResponse response, int requestId) { + final CachedAliceHttpCall? selectedCall = selectCall(requestId); + + if (selectedCall != null) { + selectedCall + ..loading = false + ..response = response + ..duration = response.time.millisecondsSinceEpoch - + (selectedCall.request?.time.millisecondsSinceEpoch ?? 0); + + _store.httpCalls.put(selectedCall); + } else { + AliceUtils.log('Selected call is null'); + } + } + + @override + void addHttpCall(AliceHttpCall aliceHttpCall) { + assert(aliceHttpCall.request != null, "Http call request can't be null"); + assert(aliceHttpCall.response != null, "Http call response can't be null"); + + _store.httpCalls.put(CachedAliceHttpCall.fromAliceHttpCall(aliceHttpCall)); + } + + @override + void removeCalls() => _store.httpCalls.removeAll(); + + @override + void subscribeToCallChanges(AliceOnCallsChanged callback) => + callsStream.listen(callback); + + @override + AliceStats getStats() => ( + total: _store.httpCalls.count(), + successes: (_store.httpCalls.query() + ..link( + CachedAliceHttpCall_.responseRel, + CachedAliceHttpResponse_.status + .greaterOrEqual(200) + .and(CachedAliceHttpResponse_.status.lessThan(300)), + )) + .build() + .count(), + redirects: (_store.httpCalls.query() + ..link( + CachedAliceHttpCall_.responseRel, + CachedAliceHttpResponse_.status + .greaterOrEqual(300) + .and(CachedAliceHttpResponse_.status.lessThan(400)), + )) + .build() + .count(), + errors: (_store.httpCalls.query() + ..link( + CachedAliceHttpCall_.responseRel, + CachedAliceHttpResponse_.status + .greaterOrEqual(400) + .and(CachedAliceHttpResponse_.status.lessThan(600)), + )) + .build() + .count(), + loading: _store.httpCalls + .query(CachedAliceHttpCall_.loading.equals(true)) + .build() + .count(), + ); +} diff --git a/packages/alice_objectbox/lib/alice_objectbox_store.dart b/packages/alice_objectbox/lib/alice_objectbox_store.dart new file mode 100644 index 00000000..b561ee8c --- /dev/null +++ b/packages/alice_objectbox/lib/alice_objectbox_store.dart @@ -0,0 +1,64 @@ +import 'dart:io' show Directory; + +import 'package:alice/core/alice_store.dart'; +import 'package:alice_objectbox/model/cached_alice_http_call.dart'; +import 'package:alice_objectbox/objectbox.g.dart'; +import 'package:path_provider/path_provider.dart'; +import 'package:path/path.dart' as path; + +/// Provides access to the ObjectBox Store throughout the app. +/// +/// Create this in the apps main function. +class AliceObjectBoxStore implements AliceStore { + AliceObjectBoxStore._create( + this._store, { + bool persistent = true, + }) { + /// Initialize all the boxes here + httpCalls = Box(_store); + + if (!persistent) { + clear(); + } + } + + /// Create an instance of ObjectBox to use throughout the app. + static Future create({ + Store? store, + bool persistent = true, + }) async { + final String storeDirectoryPath = path.join( + (await getApplicationDocumentsDirectory()).path, + Store.defaultDirectoryPath, + "alice", + ); + + final Directory storeDirectory = Directory(storeDirectoryPath); + if (!await storeDirectory.exists()) { + await storeDirectory.create(recursive: true); + } + + store ??= Store.isOpen(storeDirectoryPath) + ? Store.attach(getObjectBoxModel(), storeDirectoryPath) + : await openStore(directory: storeDirectoryPath); + + return AliceObjectBoxStore._create(store, persistent: persistent); + } + + late final Store _store; + + /// Boxes + late final Box httpCalls; + + late final Map _boxes = { + CachedAliceHttpCall: httpCalls, + }; + + /// This will remove all the items from all the boxes + @override + void clear() { + for (final Box box in _boxes.values) { + box.removeAll(); + } + } +} diff --git a/packages/alice_objectbox/lib/json_converter/alice_form_data_field_converter.dart b/packages/alice_objectbox/lib/json_converter/alice_form_data_field_converter.dart new file mode 100644 index 00000000..fd32d48b --- /dev/null +++ b/packages/alice_objectbox/lib/json_converter/alice_form_data_field_converter.dart @@ -0,0 +1,22 @@ +import 'package:alice/model/alice_from_data_field.dart'; +import 'package:json_annotation/json_annotation.dart'; + +class AliceFormDataFieldConverter + implements JsonConverter> { + const AliceFormDataFieldConverter(); + + static const AliceFormDataFieldConverter instance = + AliceFormDataFieldConverter(); + + @override + AliceFormDataField fromJson(Map json) => AliceFormDataField( + json['name'] as String, + json['value'] as String, + ); + + @override + Map toJson(AliceFormDataField object) => { + 'name': object.name, + 'value': object.value, + }; +} diff --git a/packages/alice_objectbox/lib/json_converter/alice_form_data_file_converter.dart b/packages/alice_objectbox/lib/json_converter/alice_form_data_file_converter.dart new file mode 100644 index 00000000..92aa1d22 --- /dev/null +++ b/packages/alice_objectbox/lib/json_converter/alice_form_data_file_converter.dart @@ -0,0 +1,24 @@ +import 'package:alice/model/alice_form_data_file.dart'; +import 'package:json_annotation/json_annotation.dart'; + +class AliceFormDataFileConverter + implements JsonConverter> { + const AliceFormDataFileConverter(); + + static const AliceFormDataFileConverter instance = + AliceFormDataFileConverter(); + + @override + AliceFormDataFile fromJson(Map json) => AliceFormDataFile( + json['fileName'] as String?, + json['contentType'] as String, + (json['length'] as num).toInt(), + ); + + @override + Map toJson(AliceFormDataFile object) => { + 'fileName': object.fileName, + 'contentType': object.contentType, + 'length': object.length, + }; +} diff --git a/packages/alice_objectbox/lib/model/cached_alice_http_call.dart b/packages/alice_objectbox/lib/model/cached_alice_http_call.dart new file mode 100644 index 00000000..7d3292f3 --- /dev/null +++ b/packages/alice_objectbox/lib/model/cached_alice_http_call.dart @@ -0,0 +1,130 @@ +import 'package:alice/model/alice_http_call.dart'; +import 'package:alice/model/alice_http_error.dart'; +import 'package:alice/model/alice_http_request.dart'; +import 'package:alice/model/alice_http_response.dart'; +import 'package:alice_objectbox/model/cached_alice_http_error.dart'; +import 'package:alice_objectbox/model/cached_alice_http_request.dart'; +import 'package:alice_objectbox/model/cached_alice_http_response.dart'; +import 'package:meta/meta.dart'; +import 'package:objectbox/objectbox.dart'; + +@Entity() +class CachedAliceHttpCall implements AliceHttpCall { + CachedAliceHttpCall( + this.id, { + this.objectId = 0, + DateTime? createdTime, + this.client = '', + this.loading = true, + this.secure = false, + this.method = '', + this.endpoint = '', + this.server = '', + this.uri = '', + this.duration = 0, + }) : createdTime = createdTime ?? DateTime.now(); + + @Id() + int objectId; + + @override + @Index(type: IndexType.value) + @Unique(onConflict: ConflictStrategy.replace) + final int id; + + @override + @Property(type: PropertyType.dateNano) + DateTime createdTime; + + @override + String client; + + @override + bool loading; + + @override + bool secure; + + @override + String method; + + @override + String endpoint; + + @override + String server; + + @override + String uri; + + @override + int duration; + + @override + @Transient() + AliceHttpRequest? get request => requestRel.target; + + @override + @Transient() + set request(AliceHttpRequest? value) { + requestRel.target = value != null + ? CachedAliceHttpRequest.fromAliceHttpRequest(value) + : null; + } + + @protected + final ToOne requestRel = + ToOne(); + + @override + @Transient() + AliceHttpResponse? get response => responseRel.target; + + @override + @Transient() + set response(AliceHttpResponse? value) { + responseRel.target = value != null + ? CachedAliceHttpResponse.fromAliceHttpResponse(value) + : null; + } + + @protected + final ToOne responseRel = + ToOne(); + + @override + @Transient() + AliceHttpError? get error => errorRel.target; + + @override + @Transient() + set error(AliceHttpError? value) { + errorRel.target = + value != null ? CachedAliceHttpError.fromAliceHttpError(value) : null; + } + + @protected + final ToOne errorRel = ToOne(); + + @override + void setResponse(AliceHttpResponse response) { + this.response = response; + loading = false; + } + + factory CachedAliceHttpCall.fromAliceHttpCall(AliceHttpCall call) => + CachedAliceHttpCall( + call.id, + client: call.client, + loading: call.loading, + secure: call.secure, + method: call.method, + endpoint: call.endpoint, + server: call.server, + uri: call.uri, + duration: call.duration, + ) + ..error = call.error + ..request = call.request + ..response = call.response; +} diff --git a/packages/alice_objectbox/lib/model/cached_alice_http_error.dart b/packages/alice_objectbox/lib/model/cached_alice_http_error.dart new file mode 100644 index 00000000..bb05d9a1 --- /dev/null +++ b/packages/alice_objectbox/lib/model/cached_alice_http_error.dart @@ -0,0 +1,47 @@ +import 'dart:convert' show jsonDecode, jsonEncode; + +import 'package:alice/model/alice_http_error.dart'; +import 'package:objectbox/objectbox.dart'; + +@Entity() +class CachedAliceHttpError implements AliceHttpError { + CachedAliceHttpError({ + this.objectId = 0, + }); + + @Id() + int objectId; + + @override + @Transient() + dynamic error; + + String? get dbError { + if (error != null) { + try { + return jsonEncode(error); + } catch (_) { + return jsonEncode(error.toString()); + } + } + return null; + } + + set dbError(String? value) => + error = value != null ? jsonDecode(value) : null; + + @override + @Transient() + StackTrace? stackTrace; + + String? get dbStackTrace => stackTrace?.toString(); + + set dbStackTrace(String? value) => + stackTrace = value != null ? StackTrace.fromString(value) : null; + + factory CachedAliceHttpError.fromAliceHttpError(AliceHttpError error) { + return CachedAliceHttpError() + ..error = error.error + ..stackTrace = error.stackTrace; + } +} diff --git a/packages/alice_objectbox/lib/model/cached_alice_http_request.dart b/packages/alice_objectbox/lib/model/cached_alice_http_request.dart new file mode 100644 index 00000000..1797da7e --- /dev/null +++ b/packages/alice_objectbox/lib/model/cached_alice_http_request.dart @@ -0,0 +1,133 @@ +import 'dart:convert' show jsonDecode, jsonEncode; +import 'dart:io' show Cookie; +import 'package:alice/model/alice_form_data_file.dart'; +import 'package:alice/model/alice_from_data_field.dart'; +import 'package:alice/model/alice_http_request.dart'; +import 'package:alice_objectbox/json_converter/alice_form_data_field_converter.dart'; +import 'package:alice_objectbox/json_converter/alice_form_data_file_converter.dart'; +import 'package:objectbox/objectbox.dart'; + +@Entity() +class CachedAliceHttpRequest implements AliceHttpRequest { + CachedAliceHttpRequest({ + this.objectId = 0, + this.size = 0, + DateTime? time, + this.headers = const {}, + this.body = '', + this.contentType = '', + this.cookies = const [], + this.queryParameters = const {}, + this.formDataFiles, + this.formDataFields, + }) : time = time ?? DateTime.now(); + + @Id() + int objectId; + + @override + int size; + + @override + @Property(type: PropertyType.dateNano) + DateTime time; + + @override + @Transient() + Map headers; + + String get dbHeaders => jsonEncode(headers); + + set dbHeaders(String value) => + headers = jsonDecode(value) as Map; + + @override + @Transient() + dynamic body; + + String? get dbBody { + if (body != null) { + try { + return jsonEncode(body); + } catch (_) { + return jsonEncode(body.toString()); + } + } + return null; + } + + set dbBody(String? value) => body = value != null ? jsonDecode(value) : null; + + @override + String? contentType; + + @override + @Transient() + List cookies; + + List get dbCookies => + cookies.map((Cookie cookie) => cookie.toString()).toList(); + + set dbCookies(List value) => cookies = + value.map((String cookie) => Cookie.fromSetCookieValue(cookie)).toList(); + + @override + @Transient() + Map queryParameters; + + String get dbQueryParameters => jsonEncode(queryParameters); + + set dbQueryParameters(String value) => + queryParameters = jsonDecode(value) as Map; + + @override + @Transient() + List? formDataFiles; + + List? get dbFormDataFiles => formDataFiles + ?.map( + (AliceFormDataFile file) => + jsonEncode(AliceFormDataFileConverter.instance.toJson(file)), + ) + .toList(); + + set dbFormDataFiles(List? value) => formDataFiles = value + ?.map( + (String file) => + AliceFormDataFileConverter.instance.fromJson(jsonDecode(file)), + ) + .toList(); + + @override + @Transient() + List? formDataFields; + + List? get dbFormDataFields => formDataFields + ?.map( + (AliceFormDataField field) => + jsonEncode(AliceFormDataFieldConverter.instance.toJson(field)), + ) + .toList(); + + set dbFormDataFields(List? value) => formDataFields = value + ?.map( + (String field) => + AliceFormDataFieldConverter.instance.fromJson(jsonDecode(field)), + ) + .toList(); + + factory CachedAliceHttpRequest.fromAliceHttpRequest( + AliceHttpRequest request, + ) => + CachedAliceHttpRequest( + size: request.size, + time: request.time, + headers: request.headers, + body: request.body, + contentType: request.contentType, + cookies: request.cookies, + queryParameters: request.queryParameters, + formDataFiles: request.formDataFiles, + formDataFields: request.formDataFields, + ); +} diff --git a/packages/alice_objectbox/lib/model/cached_alice_http_response.dart b/packages/alice_objectbox/lib/model/cached_alice_http_response.dart new file mode 100644 index 00000000..50be6f7f --- /dev/null +++ b/packages/alice_objectbox/lib/model/cached_alice_http_response.dart @@ -0,0 +1,69 @@ +import 'dart:convert' show jsonDecode, jsonEncode; + +import 'package:alice/model/alice_http_response.dart'; +import 'package:objectbox/objectbox.dart'; + +@Entity() +class CachedAliceHttpResponse implements AliceHttpResponse { + CachedAliceHttpResponse({ + this.objectId = 0, + this.status = 0, + this.size = 0, + DateTime? time, + this.body, + this.headers, + }) : time = time ?? DateTime.now(); + + @Id() + int objectId; + + @override + int? status; + + @override + int size; + + @override + @Property(type: PropertyType.dateNano) + DateTime time; + + @override + @Transient() + dynamic body; + + String? get dbBody { + if (body != null) { + try { + return jsonEncode(body); + } catch (_) { + return jsonEncode(body.toString()); + } + } + return null; + } + + set dbBody(String? value) => body = value != null ? jsonDecode(value) : null; + + @override + @Transient() + Map? headers; + + String? get dbHeaders => headers != null ? jsonEncode(headers) : null; + + set dbHeaders(String? value) => headers = value != null + ? (jsonDecode(value) as Map?)?.map( + (key, value) => MapEntry(key, value.toString()), + ) + : null; + + factory CachedAliceHttpResponse.fromAliceHttpResponse( + AliceHttpResponse response, + ) => + CachedAliceHttpResponse( + status: response.status, + size: response.size, + time: response.time, + body: response.body, + headers: response.headers, + ); +} diff --git a/packages/alice_objectbox/lib/objectbox-model.json b/packages/alice_objectbox/lib/objectbox-model.json new file mode 100644 index 00000000..cb7083bb --- /dev/null +++ b/packages/alice_objectbox/lib/objectbox-model.json @@ -0,0 +1,230 @@ +{ + "_note1": "KEEP THIS FILE! Check it into a version control system (VCS) like git.", + "_note2": "ObjectBox manages crucial IDs for your object model. See docs for details.", + "_note3": "If you have VCS merge conflicts, you must resolve them according to ObjectBox docs.", + "entities": [ + { + "id": "1:9112051699732156539", + "lastPropertyId": "14:4354662762236974529", + "name": "CachedAliceHttpCall", + "properties": [ + { + "id": "1:3762653877211677624", + "name": "objectId", + "type": 6, + "flags": 1 + }, + { + "id": "2:3842302574345199884", + "name": "id", + "type": 6, + "flags": 32808, + "indexId": "1:785825462386371646" + }, + { + "id": "3:2486716479761573541", + "name": "createdTime", + "type": 12 + }, + { + "id": "4:5652467799946319465", + "name": "client", + "type": 9 + }, + { + "id": "5:4684667944697724811", + "name": "loading", + "type": 1 + }, + { + "id": "6:4083857791022485908", + "name": "secure", + "type": 1 + }, + { + "id": "7:2468700886320707450", + "name": "method", + "type": 9 + }, + { + "id": "8:1481352321860840686", + "name": "endpoint", + "type": 9 + }, + { + "id": "9:168907952189534839", + "name": "server", + "type": 9 + }, + { + "id": "10:2352012304390890819", + "name": "uri", + "type": 9 + }, + { + "id": "11:5411203974018974012", + "name": "duration", + "type": 6 + }, + { + "id": "12:6347069823169535184", + "name": "requestRelId", + "type": 11, + "flags": 520, + "indexId": "2:1736187189570129453", + "relationTarget": "CachedAliceHttpRequest" + }, + { + "id": "13:2665094240670232880", + "name": "responseRelId", + "type": 11, + "flags": 520, + "indexId": "3:736342038365124987", + "relationTarget": "CachedAliceHttpResponse" + }, + { + "id": "14:4354662762236974529", + "name": "errorRelId", + "type": 11, + "flags": 520, + "indexId": "4:6876753875053649510", + "relationTarget": "CachedAliceHttpError" + } + ], + "relations": [] + }, + { + "id": "2:1234822196006357971", + "lastPropertyId": "3:6606174208067502830", + "name": "CachedAliceHttpError", + "properties": [ + { + "id": "1:7595183079379424783", + "name": "objectId", + "type": 6, + "flags": 1 + }, + { + "id": "2:2906952339862377074", + "name": "dbError", + "type": 9 + }, + { + "id": "3:6606174208067502830", + "name": "dbStackTrace", + "type": 9 + } + ], + "relations": [] + }, + { + "id": "3:3952435099002188469", + "lastPropertyId": "10:5482956073710374459", + "name": "CachedAliceHttpRequest", + "properties": [ + { + "id": "1:7860030543131149189", + "name": "objectId", + "type": 6, + "flags": 1 + }, + { + "id": "2:400756712209683011", + "name": "size", + "type": 6 + }, + { + "id": "3:1461695331578797514", + "name": "time", + "type": 12 + }, + { + "id": "4:6367092714501419189", + "name": "contentType", + "type": 9 + }, + { + "id": "5:6942973330930551852", + "name": "dbHeaders", + "type": 9 + }, + { + "id": "6:5177090139150472743", + "name": "dbBody", + "type": 9 + }, + { + "id": "7:7073566945336254377", + "name": "dbCookies", + "type": 30 + }, + { + "id": "8:7499443194345193384", + "name": "dbQueryParameters", + "type": 9 + }, + { + "id": "9:3569666611249608973", + "name": "dbFormDataFiles", + "type": 30 + }, + { + "id": "10:5482956073710374459", + "name": "dbFormDataFields", + "type": 30 + } + ], + "relations": [] + }, + { + "id": "4:8890972191143949164", + "lastPropertyId": "6:8264530521928833819", + "name": "CachedAliceHttpResponse", + "properties": [ + { + "id": "1:3995456757645260575", + "name": "objectId", + "type": 6, + "flags": 1 + }, + { + "id": "2:3245463467272928109", + "name": "status", + "type": 6 + }, + { + "id": "3:8483714278633944857", + "name": "size", + "type": 6 + }, + { + "id": "4:2693845057233959128", + "name": "time", + "type": 12 + }, + { + "id": "5:5298588582283239315", + "name": "dbBody", + "type": 9 + }, + { + "id": "6:8264530521928833819", + "name": "dbHeaders", + "type": 9 + } + ], + "relations": [] + } + ], + "lastEntityId": "4:8890972191143949164", + "lastIndexId": "4:6876753875053649510", + "lastRelationId": "0:0", + "lastSequenceId": "0:0", + "modelVersion": 5, + "modelVersionParserMinimum": 5, + "retiredEntityUids": [], + "retiredIndexUids": [], + "retiredPropertyUids": [], + "retiredRelationUids": [], + "version": 1 +} \ No newline at end of file diff --git a/packages/alice_objectbox/lib/objectbox.g.dart b/packages/alice_objectbox/lib/objectbox.g.dart new file mode 100644 index 00000000..f18ed00a --- /dev/null +++ b/packages/alice_objectbox/lib/objectbox.g.dart @@ -0,0 +1,685 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND +// This code was generated by ObjectBox. To update it run the generator again +// with `dart run build_runner build`. +// See also https://docs.objectbox.io/getting-started#generate-objectbox-code + +// ignore_for_file: camel_case_types, depend_on_referenced_packages +// coverage:ignore-file + +import 'dart:typed_data'; + +import 'package:flat_buffers/flat_buffers.dart' as fb; +import 'package:objectbox/internal.dart' + as obx_int; // generated code can access "internal" functionality +import 'package:objectbox/objectbox.dart' as obx; +import 'package:objectbox_flutter_libs/objectbox_flutter_libs.dart'; + +import 'model/cached_alice_http_call.dart'; +import 'model/cached_alice_http_error.dart'; +import 'model/cached_alice_http_request.dart'; +import 'model/cached_alice_http_response.dart'; + +export 'package:objectbox/objectbox.dart'; // so that callers only have to import this file + +final _entities = [ + obx_int.ModelEntity( + id: const obx_int.IdUid(1, 9112051699732156539), + name: 'CachedAliceHttpCall', + lastPropertyId: const obx_int.IdUid(14, 4354662762236974529), + flags: 0, + properties: [ + obx_int.ModelProperty( + id: const obx_int.IdUid(1, 3762653877211677624), + name: 'objectId', + type: 6, + flags: 1), + obx_int.ModelProperty( + id: const obx_int.IdUid(2, 3842302574345199884), + name: 'id', + type: 6, + flags: 32808, + indexId: const obx_int.IdUid(1, 785825462386371646)), + obx_int.ModelProperty( + id: const obx_int.IdUid(3, 2486716479761573541), + name: 'createdTime', + type: 12, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(4, 5652467799946319465), + name: 'client', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(5, 4684667944697724811), + name: 'loading', + type: 1, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(6, 4083857791022485908), + name: 'secure', + type: 1, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(7, 2468700886320707450), + name: 'method', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(8, 1481352321860840686), + name: 'endpoint', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(9, 168907952189534839), + name: 'server', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(10, 2352012304390890819), + name: 'uri', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(11, 5411203974018974012), + name: 'duration', + type: 6, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(12, 6347069823169535184), + name: 'requestRelId', + type: 11, + flags: 520, + indexId: const obx_int.IdUid(2, 1736187189570129453), + relationTarget: 'CachedAliceHttpRequest'), + obx_int.ModelProperty( + id: const obx_int.IdUid(13, 2665094240670232880), + name: 'responseRelId', + type: 11, + flags: 520, + indexId: const obx_int.IdUid(3, 736342038365124987), + relationTarget: 'CachedAliceHttpResponse'), + obx_int.ModelProperty( + id: const obx_int.IdUid(14, 4354662762236974529), + name: 'errorRelId', + type: 11, + flags: 520, + indexId: const obx_int.IdUid(4, 6876753875053649510), + relationTarget: 'CachedAliceHttpError') + ], + relations: [], + backlinks: []), + obx_int.ModelEntity( + id: const obx_int.IdUid(2, 1234822196006357971), + name: 'CachedAliceHttpError', + lastPropertyId: const obx_int.IdUid(3, 6606174208067502830), + flags: 0, + properties: [ + obx_int.ModelProperty( + id: const obx_int.IdUid(1, 7595183079379424783), + name: 'objectId', + type: 6, + flags: 1), + obx_int.ModelProperty( + id: const obx_int.IdUid(2, 2906952339862377074), + name: 'dbError', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(3, 6606174208067502830), + name: 'dbStackTrace', + type: 9, + flags: 0) + ], + relations: [], + backlinks: []), + obx_int.ModelEntity( + id: const obx_int.IdUid(3, 3952435099002188469), + name: 'CachedAliceHttpRequest', + lastPropertyId: const obx_int.IdUid(10, 5482956073710374459), + flags: 0, + properties: [ + obx_int.ModelProperty( + id: const obx_int.IdUid(1, 7860030543131149189), + name: 'objectId', + type: 6, + flags: 1), + obx_int.ModelProperty( + id: const obx_int.IdUid(2, 400756712209683011), + name: 'size', + type: 6, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(3, 1461695331578797514), + name: 'time', + type: 12, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(4, 6367092714501419189), + name: 'contentType', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(5, 6942973330930551852), + name: 'dbHeaders', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(6, 5177090139150472743), + name: 'dbBody', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(7, 7073566945336254377), + name: 'dbCookies', + type: 30, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(8, 7499443194345193384), + name: 'dbQueryParameters', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(9, 3569666611249608973), + name: 'dbFormDataFiles', + type: 30, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(10, 5482956073710374459), + name: 'dbFormDataFields', + type: 30, + flags: 0) + ], + relations: [], + backlinks: []), + obx_int.ModelEntity( + id: const obx_int.IdUid(4, 8890972191143949164), + name: 'CachedAliceHttpResponse', + lastPropertyId: const obx_int.IdUid(6, 8264530521928833819), + flags: 0, + properties: [ + obx_int.ModelProperty( + id: const obx_int.IdUid(1, 3995456757645260575), + name: 'objectId', + type: 6, + flags: 1), + obx_int.ModelProperty( + id: const obx_int.IdUid(2, 3245463467272928109), + name: 'status', + type: 6, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(3, 8483714278633944857), + name: 'size', + type: 6, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(4, 2693845057233959128), + name: 'time', + type: 12, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(5, 5298588582283239315), + name: 'dbBody', + type: 9, + flags: 0), + obx_int.ModelProperty( + id: const obx_int.IdUid(6, 8264530521928833819), + name: 'dbHeaders', + type: 9, + flags: 0) + ], + relations: [], + backlinks: []) +]; + +/// Shortcut for [obx.Store.new] that passes [getObjectBoxModel] and for Flutter +/// apps by default a [directory] using `defaultStoreDirectory()` from the +/// ObjectBox Flutter library. +/// +/// Note: for desktop apps it is recommended to specify a unique [directory]. +/// +/// See [obx.Store.new] for an explanation of all parameters. +/// +/// For Flutter apps, also calls `loadObjectBoxLibraryAndroidCompat()` from +/// the ObjectBox Flutter library to fix loading the native ObjectBox library +/// on Android 6 and older. +Future openStore( + {String? directory, + int? maxDBSizeInKB, + int? maxDataSizeInKB, + int? fileMode, + int? maxReaders, + bool queriesCaseSensitiveDefault = true, + String? macosApplicationGroup}) async { + await loadObjectBoxLibraryAndroidCompat(); + return obx.Store(getObjectBoxModel(), + directory: directory ?? (await defaultStoreDirectory()).path, + maxDBSizeInKB: maxDBSizeInKB, + maxDataSizeInKB: maxDataSizeInKB, + fileMode: fileMode, + maxReaders: maxReaders, + queriesCaseSensitiveDefault: queriesCaseSensitiveDefault, + macosApplicationGroup: macosApplicationGroup); +} + +/// Returns the ObjectBox model definition for this project for use with +/// [obx.Store.new]. +obx_int.ModelDefinition getObjectBoxModel() { + final model = obx_int.ModelInfo( + entities: _entities, + lastEntityId: const obx_int.IdUid(4, 8890972191143949164), + lastIndexId: const obx_int.IdUid(4, 6876753875053649510), + lastRelationId: const obx_int.IdUid(0, 0), + lastSequenceId: const obx_int.IdUid(0, 0), + retiredEntityUids: const [], + retiredIndexUids: const [], + retiredPropertyUids: const [], + retiredRelationUids: const [], + modelVersion: 5, + modelVersionParserMinimum: 5, + version: 1); + + final bindings = { + CachedAliceHttpCall: obx_int.EntityDefinition( + model: _entities[0], + toOneRelations: (CachedAliceHttpCall object) => + [object.requestRel, object.responseRel, object.errorRel], + toManyRelations: (CachedAliceHttpCall object) => {}, + getId: (CachedAliceHttpCall object) => object.objectId, + setId: (CachedAliceHttpCall object, int id) { + object.objectId = id; + }, + objectToFB: (CachedAliceHttpCall object, fb.Builder fbb) { + final clientOffset = fbb.writeString(object.client); + final methodOffset = fbb.writeString(object.method); + final endpointOffset = fbb.writeString(object.endpoint); + final serverOffset = fbb.writeString(object.server); + final uriOffset = fbb.writeString(object.uri); + fbb.startTable(15); + fbb.addInt64(0, object.objectId); + fbb.addInt64(1, object.id); + fbb.addInt64(2, object.createdTime.microsecondsSinceEpoch * 1000); + fbb.addOffset(3, clientOffset); + fbb.addBool(4, object.loading); + fbb.addBool(5, object.secure); + fbb.addOffset(6, methodOffset); + fbb.addOffset(7, endpointOffset); + fbb.addOffset(8, serverOffset); + fbb.addOffset(9, uriOffset); + fbb.addInt64(10, object.duration); + fbb.addInt64(11, object.requestRel.targetId); + fbb.addInt64(12, object.responseRel.targetId); + fbb.addInt64(13, object.errorRel.targetId); + fbb.finish(fbb.endTable()); + return object.objectId; + }, + objectFromFB: (obx.Store store, ByteData fbData) { + final buffer = fb.BufferContext(fbData); + final rootOffset = buffer.derefObject(0); + final idParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 6, 0); + final objectIdParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 4, 0); + final createdTimeParam = DateTime.fromMicrosecondsSinceEpoch( + (const fb.Int64Reader().vTableGet(buffer, rootOffset, 8, 0) / + 1000) + .round()); + final clientParam = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 10, ''); + final loadingParam = + const fb.BoolReader().vTableGet(buffer, rootOffset, 12, false); + final secureParam = + const fb.BoolReader().vTableGet(buffer, rootOffset, 14, false); + final methodParam = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 16, ''); + final endpointParam = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 18, ''); + final serverParam = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 20, ''); + final uriParam = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 22, ''); + final durationParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 24, 0); + final object = CachedAliceHttpCall(idParam, + objectId: objectIdParam, + createdTime: createdTimeParam, + client: clientParam, + loading: loadingParam, + secure: secureParam, + method: methodParam, + endpoint: endpointParam, + server: serverParam, + uri: uriParam, + duration: durationParam); + object.requestRel.targetId = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 26, 0); + object.requestRel.attach(store); + object.responseRel.targetId = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 28, 0); + object.responseRel.attach(store); + object.errorRel.targetId = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 30, 0); + object.errorRel.attach(store); + return object; + }), + CachedAliceHttpError: obx_int.EntityDefinition( + model: _entities[1], + toOneRelations: (CachedAliceHttpError object) => [], + toManyRelations: (CachedAliceHttpError object) => {}, + getId: (CachedAliceHttpError object) => object.objectId, + setId: (CachedAliceHttpError object, int id) { + object.objectId = id; + }, + objectToFB: (CachedAliceHttpError object, fb.Builder fbb) { + final dbErrorOffset = + object.dbError == null ? null : fbb.writeString(object.dbError!); + final dbStackTraceOffset = object.dbStackTrace == null + ? null + : fbb.writeString(object.dbStackTrace!); + fbb.startTable(4); + fbb.addInt64(0, object.objectId); + fbb.addOffset(1, dbErrorOffset); + fbb.addOffset(2, dbStackTraceOffset); + fbb.finish(fbb.endTable()); + return object.objectId; + }, + objectFromFB: (obx.Store store, ByteData fbData) { + final buffer = fb.BufferContext(fbData); + final rootOffset = buffer.derefObject(0); + final objectIdParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 4, 0); + final object = CachedAliceHttpError(objectId: objectIdParam) + ..dbError = const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 6) + ..dbStackTrace = const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 8); + + return object; + }), + CachedAliceHttpRequest: obx_int.EntityDefinition( + model: _entities[2], + toOneRelations: (CachedAliceHttpRequest object) => [], + toManyRelations: (CachedAliceHttpRequest object) => {}, + getId: (CachedAliceHttpRequest object) => object.objectId, + setId: (CachedAliceHttpRequest object, int id) { + object.objectId = id; + }, + objectToFB: (CachedAliceHttpRequest object, fb.Builder fbb) { + final contentTypeOffset = object.contentType == null + ? null + : fbb.writeString(object.contentType!); + final dbHeadersOffset = fbb.writeString(object.dbHeaders); + final dbBodyOffset = + object.dbBody == null ? null : fbb.writeString(object.dbBody!); + final dbCookiesOffset = fbb.writeList( + object.dbCookies.map(fbb.writeString).toList(growable: false)); + final dbQueryParametersOffset = + fbb.writeString(object.dbQueryParameters); + final dbFormDataFilesOffset = object.dbFormDataFiles == null + ? null + : fbb.writeList(object.dbFormDataFiles! + .map(fbb.writeString) + .toList(growable: false)); + final dbFormDataFieldsOffset = object.dbFormDataFields == null + ? null + : fbb.writeList(object.dbFormDataFields! + .map(fbb.writeString) + .toList(growable: false)); + fbb.startTable(11); + fbb.addInt64(0, object.objectId); + fbb.addInt64(1, object.size); + fbb.addInt64(2, object.time.microsecondsSinceEpoch * 1000); + fbb.addOffset(3, contentTypeOffset); + fbb.addOffset(4, dbHeadersOffset); + fbb.addOffset(5, dbBodyOffset); + fbb.addOffset(6, dbCookiesOffset); + fbb.addOffset(7, dbQueryParametersOffset); + fbb.addOffset(8, dbFormDataFilesOffset); + fbb.addOffset(9, dbFormDataFieldsOffset); + fbb.finish(fbb.endTable()); + return object.objectId; + }, + objectFromFB: (obx.Store store, ByteData fbData) { + final buffer = fb.BufferContext(fbData); + final rootOffset = buffer.derefObject(0); + final objectIdParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 4, 0); + final sizeParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 6, 0); + final timeParam = DateTime.fromMicrosecondsSinceEpoch( + (const fb.Int64Reader().vTableGet(buffer, rootOffset, 8, 0) / + 1000) + .round()); + final contentTypeParam = + const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 10); + final object = CachedAliceHttpRequest( + objectId: objectIdParam, + size: sizeParam, + time: timeParam, + contentType: contentTypeParam) + ..dbHeaders = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 12, '') + ..dbBody = const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 14) + ..dbCookies = const fb.ListReader( + fb.StringReader(asciiOptimization: true), + lazy: false) + .vTableGet(buffer, rootOffset, 16, []) + ..dbQueryParameters = const fb.StringReader(asciiOptimization: true) + .vTableGet(buffer, rootOffset, 18, '') + ..dbFormDataFiles = const fb.ListReader( + fb.StringReader(asciiOptimization: true), + lazy: false) + .vTableGetNullable(buffer, rootOffset, 20) + ..dbFormDataFields = const fb.ListReader( + fb.StringReader(asciiOptimization: true), + lazy: false) + .vTableGetNullable(buffer, rootOffset, 22); + + return object; + }), + CachedAliceHttpResponse: obx_int.EntityDefinition( + model: _entities[3], + toOneRelations: (CachedAliceHttpResponse object) => [], + toManyRelations: (CachedAliceHttpResponse object) => {}, + getId: (CachedAliceHttpResponse object) => object.objectId, + setId: (CachedAliceHttpResponse object, int id) { + object.objectId = id; + }, + objectToFB: (CachedAliceHttpResponse object, fb.Builder fbb) { + final dbBodyOffset = + object.dbBody == null ? null : fbb.writeString(object.dbBody!); + final dbHeadersOffset = object.dbHeaders == null + ? null + : fbb.writeString(object.dbHeaders!); + fbb.startTable(7); + fbb.addInt64(0, object.objectId); + fbb.addInt64(1, object.status); + fbb.addInt64(2, object.size); + fbb.addInt64(3, object.time.microsecondsSinceEpoch * 1000); + fbb.addOffset(4, dbBodyOffset); + fbb.addOffset(5, dbHeadersOffset); + fbb.finish(fbb.endTable()); + return object.objectId; + }, + objectFromFB: (obx.Store store, ByteData fbData) { + final buffer = fb.BufferContext(fbData); + final rootOffset = buffer.derefObject(0); + final objectIdParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 4, 0); + final statusParam = + const fb.Int64Reader().vTableGetNullable(buffer, rootOffset, 6); + final sizeParam = + const fb.Int64Reader().vTableGet(buffer, rootOffset, 8, 0); + final timeParam = DateTime.fromMicrosecondsSinceEpoch( + (const fb.Int64Reader().vTableGet(buffer, rootOffset, 10, 0) / + 1000) + .round()); + final object = CachedAliceHttpResponse( + objectId: objectIdParam, + status: statusParam, + size: sizeParam, + time: timeParam) + ..dbBody = const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 12) + ..dbHeaders = const fb.StringReader(asciiOptimization: true) + .vTableGetNullable(buffer, rootOffset, 14); + + return object; + }) + }; + + return obx_int.ModelDefinition(model, bindings); +} + +/// [CachedAliceHttpCall] entity fields to define ObjectBox queries. +class CachedAliceHttpCall_ { + /// See [CachedAliceHttpCall.objectId]. + static final objectId = + obx.QueryIntegerProperty(_entities[0].properties[0]); + + /// See [CachedAliceHttpCall.id]. + static final id = + obx.QueryIntegerProperty(_entities[0].properties[1]); + + /// See [CachedAliceHttpCall.createdTime]. + static final createdTime = obx.QueryDateNanoProperty( + _entities[0].properties[2]); + + /// See [CachedAliceHttpCall.client]. + static final client = + obx.QueryStringProperty(_entities[0].properties[3]); + + /// See [CachedAliceHttpCall.loading]. + static final loading = + obx.QueryBooleanProperty(_entities[0].properties[4]); + + /// See [CachedAliceHttpCall.secure]. + static final secure = + obx.QueryBooleanProperty(_entities[0].properties[5]); + + /// See [CachedAliceHttpCall.method]. + static final method = + obx.QueryStringProperty(_entities[0].properties[6]); + + /// See [CachedAliceHttpCall.endpoint]. + static final endpoint = + obx.QueryStringProperty(_entities[0].properties[7]); + + /// See [CachedAliceHttpCall.server]. + static final server = + obx.QueryStringProperty(_entities[0].properties[8]); + + /// See [CachedAliceHttpCall.uri]. + static final uri = + obx.QueryStringProperty(_entities[0].properties[9]); + + /// See [CachedAliceHttpCall.duration]. + static final duration = obx.QueryIntegerProperty( + _entities[0].properties[10]); + + /// See [CachedAliceHttpCall.requestRel]. + static final requestRel = + obx.QueryRelationToOne( + _entities[0].properties[11]); + + /// See [CachedAliceHttpCall.responseRel]. + static final responseRel = + obx.QueryRelationToOne( + _entities[0].properties[12]); + + /// See [CachedAliceHttpCall.errorRel]. + static final errorRel = + obx.QueryRelationToOne( + _entities[0].properties[13]); +} + +/// [CachedAliceHttpError] entity fields to define ObjectBox queries. +class CachedAliceHttpError_ { + /// See [CachedAliceHttpError.objectId]. + static final objectId = obx.QueryIntegerProperty( + _entities[1].properties[0]); + + /// See [CachedAliceHttpError.dbError]. + static final dbError = + obx.QueryStringProperty(_entities[1].properties[1]); + + /// See [CachedAliceHttpError.dbStackTrace]. + static final dbStackTrace = + obx.QueryStringProperty(_entities[1].properties[2]); +} + +/// [CachedAliceHttpRequest] entity fields to define ObjectBox queries. +class CachedAliceHttpRequest_ { + /// See [CachedAliceHttpRequest.objectId]. + static final objectId = obx.QueryIntegerProperty( + _entities[2].properties[0]); + + /// See [CachedAliceHttpRequest.size]. + static final size = obx.QueryIntegerProperty( + _entities[2].properties[1]); + + /// See [CachedAliceHttpRequest.time]. + static final time = obx.QueryDateNanoProperty( + _entities[2].properties[2]); + + /// See [CachedAliceHttpRequest.contentType]. + static final contentType = obx.QueryStringProperty( + _entities[2].properties[3]); + + /// See [CachedAliceHttpRequest.dbHeaders]. + static final dbHeaders = obx.QueryStringProperty( + _entities[2].properties[4]); + + /// See [CachedAliceHttpRequest.dbBody]. + static final dbBody = obx.QueryStringProperty( + _entities[2].properties[5]); + + /// See [CachedAliceHttpRequest.dbCookies]. + static final dbCookies = + obx.QueryStringVectorProperty( + _entities[2].properties[6]); + + /// See [CachedAliceHttpRequest.dbQueryParameters]. + static final dbQueryParameters = + obx.QueryStringProperty( + _entities[2].properties[7]); + + /// See [CachedAliceHttpRequest.dbFormDataFiles]. + static final dbFormDataFiles = + obx.QueryStringVectorProperty( + _entities[2].properties[8]); + + /// See [CachedAliceHttpRequest.dbFormDataFields]. + static final dbFormDataFields = + obx.QueryStringVectorProperty( + _entities[2].properties[9]); +} + +/// [CachedAliceHttpResponse] entity fields to define ObjectBox queries. +class CachedAliceHttpResponse_ { + /// See [CachedAliceHttpResponse.objectId]. + static final objectId = obx.QueryIntegerProperty( + _entities[3].properties[0]); + + /// See [CachedAliceHttpResponse.status]. + static final status = obx.QueryIntegerProperty( + _entities[3].properties[1]); + + /// See [CachedAliceHttpResponse.size]. + static final size = obx.QueryIntegerProperty( + _entities[3].properties[2]); + + /// See [CachedAliceHttpResponse.time]. + static final time = obx.QueryDateNanoProperty( + _entities[3].properties[3]); + + /// See [CachedAliceHttpResponse.dbBody]. + static final dbBody = obx.QueryStringProperty( + _entities[3].properties[4]); + + /// See [CachedAliceHttpResponse.dbHeaders]. + static final dbHeaders = obx.QueryStringProperty( + _entities[3].properties[5]); +} diff --git a/packages/alice_objectbox/pubspec.yaml b/packages/alice_objectbox/pubspec.yaml new file mode 100644 index 00000000..97306f57 --- /dev/null +++ b/packages/alice_objectbox/pubspec.yaml @@ -0,0 +1,26 @@ +name: alice_objectbox +description: "Alice + ObjectBox integration. It contains a plugin for Alice which stores http requests and responses in an ObjectBox NoSQL database." +version: 1.0.0 +repository: https://github.com/jhomlala/alice +homepage: https://github.com/jhomlala/alice + +environment: + sdk: ^3.0.0 + flutter: ">=3.10.0" + +dependencies: + flutter: + sdk: flutter + alice: ^1.0.0-dev.6 + json_annotation: ^4.9.0 + meta: ^1.9.1 + objectbox: ^4.0.1 + objectbox_flutter_libs: ^4.0.1 + path: ^1.9.0 + path_provider: ^2.1.3 + +dev_dependencies: + build_runner: ^2.4.11 + flutter_lints: ^4.0.0 + json_serializable: ^6.8.0 + objectbox_generator: ^4.0.1