commit 651779b1cb58803c2dfa3afa6d4bb770c153bbbf Author: Nathan Anderson Date: Wed May 20 10:22:58 2026 -0500 initial diff --git a/.firebaserc b/.firebaserc new file mode 100644 index 0000000..f5b9d9e --- /dev/null +++ b/.firebaserc @@ -0,0 +1,5 @@ +{ + "projects": { + "default": "cyberhybridhub" + } +} diff --git a/.flutter-plugins-dependencies b/.flutter-plugins-dependencies new file mode 100644 index 0000000..161b639 --- /dev/null +++ b/.flutter-plugins-dependencies @@ -0,0 +1 @@ +{"info":"This is a generated file; do not edit or check into version control.","plugins":{"ios":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"firebase_auth","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_auth-6.5.1/","native_build":true,"dependencies":["firebase_core"],"dev_dependency":false},{"name":"firebase_core","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_core-4.9.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_mobile","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_mobile-0.2.1/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_ios","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_ios-6.3.0/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"path_provider_foundation","path":"/home/nathan/.pub-cache/hosted/pub.dev/path_provider_foundation-2.6.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"shared_preferences_foundation","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.6/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"sqlite3_flutter_libs","path":"/home/nathan/.pub-cache/hosted/pub.dev/sqlite3_flutter_libs-0.5.42/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"url_launcher_ios","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_ios-6.4.1/","native_build":true,"dependencies":[],"dev_dependency":false}],"android":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"firebase_auth","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_auth-6.5.1/","native_build":true,"dependencies":["firebase_core"],"dev_dependency":false},{"name":"firebase_core","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_core-4.9.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_mobile","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_mobile-0.2.1/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_android","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_android-7.2.11/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"jni","path":"/home/nathan/.pub-cache/hosted/pub.dev/jni-1.0.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"jni_flutter","path":"/home/nathan/.pub-cache/hosted/pub.dev/jni_flutter-1.0.1/","native_build":true,"dependencies":["jni"],"dev_dependency":false},{"name":"path_provider_android","path":"/home/nathan/.pub-cache/hosted/pub.dev/path_provider_android-2.3.1/","native_build":false,"dependencies":["jni","jni_flutter"],"dev_dependency":false},{"name":"shared_preferences_android","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_android-2.4.23/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"sqlite3_flutter_libs","path":"/home/nathan/.pub-cache/hosted/pub.dev/sqlite3_flutter_libs-0.5.42/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"url_launcher_android","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_android-6.3.29/","native_build":true,"dependencies":[],"dev_dependency":false}],"macos":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"firebase_auth","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_auth-6.5.1/","native_build":true,"dependencies":["firebase_core"],"dev_dependency":false},{"name":"firebase_core","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_core-4.9.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_desktop","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_desktop-0.2.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_ios","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_ios-6.3.0/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"path_provider_foundation","path":"/home/nathan/.pub-cache/hosted/pub.dev/path_provider_foundation-2.6.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"shared_preferences_foundation","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_foundation-2.5.6/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"sqlite3_flutter_libs","path":"/home/nathan/.pub-cache/hosted/pub.dev/sqlite3_flutter_libs-0.5.42/","shared_darwin_source":true,"native_build":true,"dependencies":[],"dev_dependency":false},{"name":"url_launcher_macos","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_macos-3.2.5/","native_build":true,"dependencies":[],"dev_dependency":false}],"linux":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_desktop","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_desktop-0.2.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"jni","path":"/home/nathan/.pub-cache/hosted/pub.dev/jni-1.0.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"path_provider_linux","path":"/home/nathan/.pub-cache/hosted/pub.dev/path_provider_linux-2.2.1/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"shared_preferences_linux","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_linux-2.4.1/","native_build":false,"dependencies":["path_provider_linux"],"dev_dependency":false},{"name":"sqlite3_flutter_libs","path":"/home/nathan/.pub-cache/hosted/pub.dev/sqlite3_flutter_libs-0.5.42/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"url_launcher_linux","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_linux-3.2.2/","native_build":true,"dependencies":[],"dev_dependency":false}],"windows":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"firebase_auth","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_auth-6.5.1/","native_build":true,"dependencies":["firebase_core"],"dev_dependency":false},{"name":"firebase_core","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_core-4.9.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_desktop","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_desktop-0.2.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"jni","path":"/home/nathan/.pub-cache/hosted/pub.dev/jni-1.0.0/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"path_provider_windows","path":"/home/nathan/.pub-cache/hosted/pub.dev/path_provider_windows-2.3.0/","native_build":false,"dependencies":[],"dev_dependency":false},{"name":"shared_preferences_windows","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_windows-2.4.1/","native_build":false,"dependencies":["path_provider_windows"],"dev_dependency":false},{"name":"sqlite3_flutter_libs","path":"/home/nathan/.pub-cache/hosted/pub.dev/sqlite3_flutter_libs-0.5.42/","native_build":true,"dependencies":[],"dev_dependency":false},{"name":"url_launcher_windows","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_windows-3.1.5/","native_build":true,"dependencies":[],"dev_dependency":false}],"web":[{"name":"connectivity_plus","path":"/home/nathan/.pub-cache/hosted/pub.dev/connectivity_plus-6.1.5/","dependencies":[],"dev_dependency":false},{"name":"firebase_auth_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_auth_web-6.2.1/","dependencies":["firebase_core_web"],"dev_dependency":false},{"name":"firebase_core_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/firebase_core_web-3.7.0/","dependencies":[],"dev_dependency":false},{"name":"google_sign_in_all_platforms_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_all_platforms_web-0.1.0/","dependencies":["google_sign_in_web"],"dev_dependency":false},{"name":"google_sign_in_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/google_sign_in_web-1.1.3/","dependencies":[],"dev_dependency":false},{"name":"shared_preferences_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/shared_preferences_web-2.4.3/","dependencies":[],"dev_dependency":false},{"name":"url_launcher_web","path":"/home/nathan/.pub-cache/hosted/pub.dev/url_launcher_web-2.4.3/","dependencies":[],"dev_dependency":false}]},"dependencyGraph":[{"name":"connectivity_plus","dependencies":[]},{"name":"firebase_auth","dependencies":["firebase_auth_web","firebase_core"]},{"name":"firebase_auth_web","dependencies":["firebase_core","firebase_core_web"]},{"name":"firebase_core","dependencies":["firebase_core_web"]},{"name":"firebase_core_web","dependencies":[]},{"name":"google_sign_in","dependencies":["google_sign_in_android","google_sign_in_ios","google_sign_in_web"]},{"name":"google_sign_in_all_platforms","dependencies":["google_sign_in_all_platforms_desktop","google_sign_in_all_platforms_mobile","google_sign_in_all_platforms_web"]},{"name":"google_sign_in_all_platforms_desktop","dependencies":["url_launcher"]},{"name":"google_sign_in_all_platforms_mobile","dependencies":["google_sign_in"]},{"name":"google_sign_in_all_platforms_web","dependencies":["google_sign_in","google_sign_in_web","google_sign_in_all_platforms_mobile"]},{"name":"google_sign_in_android","dependencies":[]},{"name":"google_sign_in_ios","dependencies":[]},{"name":"google_sign_in_web","dependencies":[]},{"name":"jni","dependencies":[]},{"name":"jni_flutter","dependencies":["jni"]},{"name":"path_provider","dependencies":["path_provider_android","path_provider_foundation","path_provider_linux","path_provider_windows"]},{"name":"path_provider_android","dependencies":["jni","jni_flutter"]},{"name":"path_provider_foundation","dependencies":[]},{"name":"path_provider_linux","dependencies":[]},{"name":"path_provider_windows","dependencies":[]},{"name":"shared_preferences","dependencies":["shared_preferences_android","shared_preferences_foundation","shared_preferences_linux","shared_preferences_web","shared_preferences_windows"]},{"name":"shared_preferences_android","dependencies":[]},{"name":"shared_preferences_foundation","dependencies":[]},{"name":"shared_preferences_linux","dependencies":["path_provider_linux"]},{"name":"shared_preferences_web","dependencies":[]},{"name":"shared_preferences_windows","dependencies":["path_provider_windows"]},{"name":"sqlite3_flutter_libs","dependencies":[]},{"name":"url_launcher","dependencies":["url_launcher_android","url_launcher_ios","url_launcher_linux","url_launcher_macos","url_launcher_web","url_launcher_windows"]},{"name":"url_launcher_android","dependencies":[]},{"name":"url_launcher_ios","dependencies":[]},{"name":"url_launcher_linux","dependencies":[]},{"name":"url_launcher_macos","dependencies":[]},{"name":"url_launcher_web","dependencies":[]},{"name":"url_launcher_windows","dependencies":[]}],"date_created":"2026-05-19 23:28:49.730528","version":"3.44.0","swift_package_manager_enabled":{"ios":false,"macos":false}} \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..99a1f0a --- /dev/null +++ b/.gitignore @@ -0,0 +1,80 @@ +# Flutter/Dart +.dart_tool/ +.packages +.pub-cache/ +.pub/ +build/ +flutter_*.png +linked_*.ds +unlinked.ds +unlinked_spec.ds + +# IDE +.idea/ +*.iml + +# VS Code (keep .vscode/ committed for team settings) +# .vscode/ + +# Android +**/android/**/gradle-wrapper.jar +**/android/.gradle +**/android/captures/ +**/android/gradlew +**/android/gradlew.bat +**/android/local.properties +**/android/**/GeneratedPluginRegistrant.java +**/android/key.properties +*.jks + +# iOS/macOS +**/ios/**/*.mode1v3 +**/ios/**/*.mode2v3 +**/ios/**/*.moved-aside +**/ios/**/*.pbxuser +**/ios/**/*.perspectivev3 +**/ios/**/*sync/ +**/ios/**/.sconsign.dblite +**/ios/**/.tags* +**/ios/**/.vagrant/ +**/ios/**/DerivedData/ +**/ios/**/Icon? +**/ios/**/Pods/ +**/ios/**/.symlinks/ +**/ios/**/profile +**/ios/**/xcuserdata +**/ios/.generated/ +**/ios/Flutter/.last_build_id +**/ios/Flutter/App.framework +**/ios/Flutter/Flutter.framework +**/ios/Flutter/Flutter.podspec +**/ios/Flutter/Generated.xcconfig +**/ios/Flutter/ephemeral +**/ios/Flutter/app.flx +**/ios/Flutter/app.zip +**/ios/Flutter/flutter_assets/ +**/ios/Flutter/flutter_export_environment.sh +**/ios/ServiceDefinitions.json +**/ios/Runner/GeneratedPluginRegistrant.* + +# Web +lib/generated_plugin_registrant.dart + +# Coverage +coverage/ + +# Misc +*.class +*.log +*.pyc +*.swp +.DS_Store +.atom/ +.build/ +.buildlog/ +.history +.svn/ +.swiftpm/ +migrate_working_dir/ +*.env +*.env.* diff --git a/.metadata b/.metadata new file mode 100644 index 0000000..768732a --- /dev/null +++ b/.metadata @@ -0,0 +1,45 @@ +# 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: "559ffa3f75e7402d65a8def9c28389a9b2e6fe42" + channel: "stable" + +project_type: app + +# Tracks metadata for the flutter migrate command +migration: + platforms: + - platform: root + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: android + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: ios + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: linux + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: macos + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: web + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + - platform: windows + create_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + base_revision: 559ffa3f75e7402d65a8def9c28389a9b2e6fe42 + + # 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/.vscode/extensions.json b/.vscode/extensions.json new file mode 100644 index 0000000..2457acb --- /dev/null +++ b/.vscode/extensions.json @@ -0,0 +1,6 @@ +{ + "recommendations": [ + "Dart-Code.dart-code", + "Dart-Code.flutter" + ] +} diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..82e2586 --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,28 @@ +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Flutter (web)", + "request": "launch", + "type": "dart", + "deviceId": "web-server" + }, + { + "name": "Flutter (debug)", + "request": "launch", + "type": "dart" + }, + { + "name": "Flutter (profile)", + "request": "launch", + "type": "dart", + "flutterMode": "profile" + }, + { + "name": "Flutter (release)", + "request": "launch", + "type": "dart", + "flutterMode": "release" + } + ] +} diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..fdca0ee --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,20 @@ +{ + "dart.flutterSdkPath": "/home/nathan/github-open/flutter", + "terminal.integrated.env.linux": { + "PATH": "/home/nathan/github-open/flutter/bin:${env:PATH}" + }, + "editor.formatOnSave": true, + "editor.rulers": [80], + "files.exclude": { + "**/.dart_tool": true, + "**/.packages": true, + "build": true + }, + "[dart]": { + "editor.formatOnSave": true, + "editor.selectionHighlight": false, + "editor.suggestSelection": "first", + "editor.tabCompletion": "onlySnippets", + "editor.wordBasedSuggestions": "off" + } +} diff --git a/README.md b/README.md new file mode 100644 index 0000000..7345198 --- /dev/null +++ b/README.md @@ -0,0 +1,17 @@ +# cyberhybridhub + +A new Flutter project. + +## Getting Started + +This project is a starting point for a Flutter application. + +A few resources to get you started if this is your first Flutter project: + +- [Learn Flutter](https://docs.flutter.dev/get-started/learn-flutter) +- [Write your first Flutter app](https://docs.flutter.dev/get-started/codelab) +- [Flutter learning resources](https://docs.flutter.dev/reference/learning-resources) + +For help getting started with Flutter development, view the +[online documentation](https://docs.flutter.dev/), which offers tutorials, +samples, guidance on mobile development, and a full API reference. diff --git a/TODO.md b/TODO.md new file mode 100644 index 0000000..2d76460 --- /dev/null +++ b/TODO.md @@ -0,0 +1,238 @@ +# Firebase Google Sign-In setup + +## Quick setup (Ubuntu CLI) + +Reload your shell after the first run (or open a new terminal): + +```bash +source ~/.bashrc +``` + +One-shot setup (installs tools, logs in, configures FlutterFire, registers debug SHA-1): + +```bash +cd ~/cyberhybridhub.com +chmod +x scripts/setup-firebase-google-auth.sh scripts/get-android-sha.sh + +# Optional: set your Firebase project id to skip the interactive picker +export FIREBASE_PROJECT_ID=your-firebase-project-id + +./scripts/setup-firebase-google-auth.sh +``` + +Or run step by step: + +```bash +# 1. Tools (if not already installed) +npm install -g firebase-tools +dart pub global activate flutterfire_cli + +# 2. Login (FlutterFire uses the Firebase CLI session) +firebase login + +# 3. Register apps + download config (android, ios, web) +cd ~/cyberhybridhub.com +flutterfire configure \ + --project=YOUR_FIREBASE_PROJECT_ID \ + --platforms=android,ios,web \ + --android-package-name=com.cyberhybridhub.cyberhybridhub \ + --ios-bundle-id=com.cyberhybridhub.cyberhybridhub \ + --out=lib/firebase_options.dart + +# 4. Debug SHA-1 (Android Google Sign-In) +./scripts/get-android-sha.sh +firebase apps:list --project=YOUR_FIREBASE_PROJECT_ID +firebase apps:android:sha:create ANDROID_APP_ID YOUR_SHA1 --project=YOUR_FIREBASE_PROJECT_ID +flutterfire configure --project=YOUR_FIREBASE_PROJECT_ID --platforms=android,ios,web --yes + +# 5. Enable Google provider (browser — required) +# https://console.firebase.google.com/ → Authentication → Sign-in method → Google → Enable + +# 6. Run +flutter pub get && flutter run +``` + +`~/.bashrc` includes `~/.pub-cache/bin` (for `flutterfire`) and `CYBERHYBRIDHUB_ROOT`. + +--- + +## Linux desktop development (`flutter run -d linux`) + +Official FlutterFire does not register native plugins on Linux, so this app uses: + +- `google_sign_in_all_platforms` (system browser OAuth) +- Firebase Auth REST (`signInWithIdp`) with web API key from `lib/firebase_options.dart` + +### One-time Linux OAuth setup + +1. [Google Cloud Console](https://console.cloud.google.com/) → project **cyberhybridhub** → **APIs & Services** → **Credentials** +2. Open your **Web application** OAuth 2.0 client (or create one) +3. Add authorized redirect URI: `http://127.0.0.1` +4. Copy **Client ID** and **Client secret** into `lib/config/auth_config.dart`: + +```dart +const String? googleWebOAuthClientId = 'YOUR_CLIENT_ID.apps.googleusercontent.com'; +const String? googleOAuthDesktopClientSecret = 'YOUR_CLIENT_SECRET'; +``` + +### Web browser (`flutter run -d web-server` or Chrome) + +Uses the same `googleWebOAuthClientId`. In Google Cloud, on that Web OAuth client, add **Authorized JavaScript origins**: + +- `http://localhost` +- `http://127.0.0.1` + +(Uncomment the `google-signin-client_id` meta tag in `web/index.html` only if you prefer HTML config over Dart.) + +5. Enable Google sign-in in Firebase Console (if not already) +6. Run: + +```bash +flutter run -d linux +``` + +--- + +## Manual checklist + +Complete these steps to replace placeholder config files and enable Google authentication. + +App identifiers used by this project: + +| Platform | Identifier | +|----------|------------| +| Android package | `com.cyberhybridhub.cyberhybridhub` | +| iOS bundle ID | `com.cyberhybridhub.cyberhybridhub` | + +--- + +## 1. Create a Firebase project + +- [ ] Open [Firebase Console](https://console.firebase.google.com/). +- [ ] Create a project (or select an existing one). +- [ ] Enable **Google Analytics** only if you need it (optional for auth). + +--- + +## 2. Register the Flutter app in Firebase + +- [ ] In Firebase Console → **Project settings** → **Your apps**, add: + - [ ] **Android** app with package name `com.cyberhybridhub.cyberhybridhub` + - [ ] **iOS** app with bundle ID `com.cyberhybridhub.cyberhybridhub` + - [ ] **Web** app (required for the web OAuth client used by Google Sign-In on mobile) +- [ ] Download config files when prompted: + - [ ] `google-services.json` → replace `android/app/google-services.json` + - [ ] `GoogleService-Info.plist` → replace `ios/Runner/GoogleService-Info.plist` + +### Recommended: FlutterFire CLI + +```bash +dart pub global activate flutterfire_cli +flutterfire configure +``` + +This regenerates `lib/firebase_options.dart` and downloads the platform config files. + +--- + +## 3. Enable Google sign-in in Firebase + +- [ ] Firebase Console → **Build** → **Authentication** → **Sign-in method** +- [ ] Enable **Google** +- [ ] Set a support email and save + +--- + +## 4. Google Cloud Console (OAuth & credentials) + +Firebase links to a Google Cloud project. Open it from Firebase → **Project settings** → **General** → **Google Cloud Platform resource location** → link to GCP, or go to [Google Cloud Console](https://console.cloud.google.com/) and select the same project. + +### 4.1 OAuth consent screen + +- [ ] **APIs & Services** → **OAuth consent screen** +- [ ] Choose **External** (or Internal for Workspace-only) +- [ ] Fill app name, user support email, developer contact +- [ ] Add scopes if needed (default `email`, `profile`, `openid` are usually enough) +- [ ] Add test users while the app is in **Testing** mode + +### 4.2 OAuth 2.0 Client IDs + +- [ ] **APIs & Services** → **Credentials** +- [ ] Confirm these clients exist (Firebase often creates them automatically): + +| Client type | Purpose | +|-------------|---------| +| **Web application** | `serverClientId` / ID token for Firebase Auth on Android | +| **Android** | Package name + SHA-1 certificate fingerprints | +| **iOS** | Bundle ID | + +#### Android SHA-1 fingerprints + +Debug keystore (local development): + +```bash +keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android +``` + +Release keystore (production): + +```bash +keytool -list -v -keystore /path/to/your-release.keystore -alias your-alias +``` + +- [ ] Add **SHA-1** (and **SHA-256** if offered) to the Android OAuth client in Google Cloud Console +- [ ] Re-download `google-services.json` after adding fingerprints + +#### iOS URL scheme + +- [ ] Open `ios/Runner/GoogleService-Info.plist` and copy `REVERSED_CLIENT_ID` +- [ ] In Xcode (or `ios/Runner/Info.plist`), add a URL type: + - **URL Schemes**: value of `REVERSED_CLIENT_ID` + - Example: `com.googleusercontent.apps.123456789-abcdef` + +--- + +## 5. App code configuration + +- [ ] Replace placeholders in `lib/firebase_options.dart` (or run `flutterfire configure`) +- [ ] If sign-in fails on Android with a `serverClientId` error, set the Web client ID in `lib/config/auth_config.dart`: + +```dart +const String? googleSignInServerClientId = 'YOUR_WEB_CLIENT_ID.apps.googleusercontent.com'; +``` + +Find the Web client ID in Firebase → Project settings → Your apps → Web app, or in `google-services.json` under `oauth_client` with `"client_type": 3`. + +--- + +## 6. Verify + +```bash +flutter pub get +flutter run +``` + +- [ ] Tap **Continue with Google** on the login screen +- [ ] Complete Google account selection +- [ ] Confirm the home screen shows your name and email +- [ ] Sign out and sign in again + +--- + +## Troubleshooting + +| Symptom | Likely fix | +|---------|------------| +| `clientConfigurationError` | Wrong package/bundle ID, missing SHA-1, or invalid `google-services.json` | +| Sign-in cancels immediately after account pick | Missing web OAuth client in `google-services.json` or wrong SHA-1 | +| `serverClientId must be provided on Android` | Add Web app in Firebase, re-download `google-services.json`, or set `googleSignInServerClientId` | +| iOS sign-in fails after Google UI | Add `REVERSED_CLIENT_ID` URL scheme to `Info.plist` | +| `REPLACE_ME` / invalid API key at runtime | Run `flutterfire configure` or paste real values from Firebase | + +--- + +## Reference + +- [FlutterFire setup](https://firebase.flutter.dev/docs/overview) +- [Firebase Auth – Google](https://firebase.google.com/docs/auth/flutter/federated-auth#google) +- [google_sign_in plugin](https://pub.dev/packages/google_sign_in) diff --git a/analysis_options.yaml b/analysis_options.yaml new file mode 100644 index 0000000..0d29021 --- /dev/null +++ b/analysis_options.yaml @@ -0,0 +1,28 @@ +# This file configures the analyzer, which statically analyzes Dart code to +# check for errors, warnings, and lints. +# +# The issues identified by the analyzer are surfaced in the UI of Dart-enabled +# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be +# invoked from the command line by running `flutter analyze`. + +# The following line activates a set of recommended lints for Flutter apps, +# packages, and plugins designed to encourage good coding practices. +include: package:flutter_lints/flutter.yaml + +linter: + # The lint rules applied to this project can be customized in the + # section below to disable rules from the `package:flutter_lints/flutter.yaml` + # included above or to enable additional rules. A list of all available lints + # and their documentation is published at https://dart.dev/lints. + # + # Instead of disabling a lint rule for the entire project in the + # section below, it can also be suppressed for a single line of code + # or a specific dart file by using the `// ignore: name_of_lint` and + # `// ignore_for_file: name_of_lint` syntax on the line or in the file + # producing the lint. + rules: + # avoid_print: false # Uncomment to disable the `avoid_print` rule + # prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule + +# Additional information about this file can be found at +# https://dart.dev/guides/language/analysis-options diff --git a/android/.gitignore b/android/.gitignore new file mode 100644 index 0000000..be3943c --- /dev/null +++ b/android/.gitignore @@ -0,0 +1,14 @@ +gradle-wrapper.jar +/.gradle +/captures/ +/gradlew +/gradlew.bat +/local.properties +GeneratedPluginRegistrant.java +.cxx/ + +# Remember to never publicly share your keystore. +# See https://flutter.dev/to/reference-keystore +key.properties +**/*.keystore +**/*.jks diff --git a/android/app/build.gradle.kts b/android/app/build.gradle.kts new file mode 100644 index 0000000..62b1734 --- /dev/null +++ b/android/app/build.gradle.kts @@ -0,0 +1,46 @@ +plugins { + id("com.android.application") + // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. + id("dev.flutter.flutter-gradle-plugin") + id("com.google.gms.google-services") +} + +android { + namespace = "com.cyberhybridhub.cyberhybridhub" + compileSdk = flutter.compileSdkVersion + ndkVersion = flutter.ndkVersion + + compileOptions { + sourceCompatibility = JavaVersion.VERSION_17 + targetCompatibility = JavaVersion.VERSION_17 + } + + defaultConfig { + // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). + applicationId = "com.cyberhybridhub.cyberhybridhub" + // You can update the following values to match your application needs. + // For more information, see: https://flutter.dev/to/review-gradle-config. + minSdk = flutter.minSdkVersion + targetSdk = flutter.targetSdkVersion + versionCode = flutter.versionCode + versionName = flutter.versionName + } + + 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.getByName("debug") + } + } +} + +kotlin { + compilerOptions { + jvmTarget = org.jetbrains.kotlin.gradle.dsl.JvmTarget.JVM_17 + } +} + +flutter { + source = "../.." +} diff --git a/android/app/google-services.json b/android/app/google-services.json new file mode 100644 index 0000000..0a5a857 --- /dev/null +++ b/android/app/google-services.json @@ -0,0 +1,29 @@ +{ + "project_info": { + "project_number": "155018389717", + "project_id": "cyberhybridhub", + "storage_bucket": "cyberhybridhub.firebasestorage.app" + }, + "client": [ + { + "client_info": { + "mobilesdk_app_id": "1:155018389717:android:90887875660a26509ada5c", + "android_client_info": { + "package_name": "com.cyberhybridhub.cyberhybridhub" + } + }, + "oauth_client": [], + "api_key": [ + { + "current_key": "AIzaSyA71SkNLrv3tyJQW2DCfe2vFcu1c2opDX8" + } + ], + "services": { + "appinvite_service": { + "other_platform_oauth_client": [] + } + } + } + ], + "configuration_version": "1" +} \ No newline at end of file diff --git a/android/app/src/debug/AndroidManifest.xml b/android/app/src/debug/AndroidManifest.xml new file mode 100644 index 0000000..399f698 --- /dev/null +++ b/android/app/src/debug/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/android/app/src/main/AndroidManifest.xml b/android/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..97b29a6 --- /dev/null +++ b/android/app/src/main/AndroidManifest.xml @@ -0,0 +1,45 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/android/app/src/main/kotlin/com/cyberhybridhub/cyberhybridhub/MainActivity.kt b/android/app/src/main/kotlin/com/cyberhybridhub/cyberhybridhub/MainActivity.kt new file mode 100644 index 0000000..9a61d6e --- /dev/null +++ b/android/app/src/main/kotlin/com/cyberhybridhub/cyberhybridhub/MainActivity.kt @@ -0,0 +1,5 @@ +package com.cyberhybridhub.cyberhybridhub + +import io.flutter.embedding.android.FlutterActivity + +class MainActivity : FlutterActivity() diff --git a/android/app/src/main/res/drawable-v21/launch_background.xml b/android/app/src/main/res/drawable-v21/launch_background.xml new file mode 100644 index 0000000..f74085f --- /dev/null +++ b/android/app/src/main/res/drawable-v21/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/android/app/src/main/res/drawable/launch_background.xml b/android/app/src/main/res/drawable/launch_background.xml new file mode 100644 index 0000000..304732f --- /dev/null +++ b/android/app/src/main/res/drawable/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/android/app/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000..db77bb4 Binary files /dev/null and b/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/android/app/src/main/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 0000000..17987b7 Binary files /dev/null and b/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 0000000..09d4391 Binary files /dev/null and b/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 0000000..d5f1c8d Binary files /dev/null and b/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000..4d6372e Binary files /dev/null and b/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/values-night/styles.xml b/android/app/src/main/res/values-night/styles.xml new file mode 100644 index 0000000..06952be --- /dev/null +++ b/android/app/src/main/res/values-night/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/android/app/src/main/res/values/styles.xml b/android/app/src/main/res/values/styles.xml new file mode 100644 index 0000000..cb1ef88 --- /dev/null +++ b/android/app/src/main/res/values/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/android/app/src/profile/AndroidManifest.xml b/android/app/src/profile/AndroidManifest.xml new file mode 100644 index 0000000..399f698 --- /dev/null +++ b/android/app/src/profile/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/android/build.gradle.kts b/android/build.gradle.kts new file mode 100644 index 0000000..dbee657 --- /dev/null +++ b/android/build.gradle.kts @@ -0,0 +1,24 @@ +allprojects { + repositories { + google() + mavenCentral() + } +} + +val newBuildDir: Directory = + rootProject.layout.buildDirectory + .dir("../../build") + .get() +rootProject.layout.buildDirectory.value(newBuildDir) + +subprojects { + val newSubprojectBuildDir: Directory = newBuildDir.dir(project.name) + project.layout.buildDirectory.value(newSubprojectBuildDir) +} +subprojects { + project.evaluationDependsOn(":app") +} + +tasks.register("clean") { + delete(rootProject.layout.buildDirectory) +} diff --git a/android/gradle.properties b/android/gradle.properties new file mode 100644 index 0000000..e96108c --- /dev/null +++ b/android/gradle.properties @@ -0,0 +1,6 @@ +org.gradle.jvmargs=-Xmx8G -XX:MaxMetaspaceSize=4G -XX:ReservedCodeCacheSize=512m -XX:+HeapDumpOnOutOfMemoryError +android.useAndroidX=true +# This newDsl flag was added by the Flutter template +android.newDsl=false +# This builtInKotlin flag was added by the Flutter template +android.builtInKotlin=false diff --git a/android/gradle/wrapper/gradle-wrapper.properties b/android/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..2d428bf --- /dev/null +++ b/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-9.1.0-all.zip diff --git a/android/settings.gradle.kts b/android/settings.gradle.kts new file mode 100644 index 0000000..6ade8d3 --- /dev/null +++ b/android/settings.gradle.kts @@ -0,0 +1,27 @@ +pluginManagement { + val flutterSdkPath = + run { + val properties = java.util.Properties() + file("local.properties").inputStream().use { properties.load(it) } + val flutterSdkPath = properties.getProperty("flutter.sdk") + require(flutterSdkPath != null) { "flutter.sdk not set in local.properties" } + 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 "9.0.1" apply false + id("org.jetbrains.kotlin.android") version "2.3.20" apply false + id("com.google.gms.google-services") version "4.4.2" apply false +} + +include(":app") diff --git a/firebase.json b/firebase.json new file mode 100644 index 0000000..5cfd924 --- /dev/null +++ b/firebase.json @@ -0,0 +1,34 @@ +{ + "auth": { + "providers": { + "googleSignIn": { + "oAuthBrandDisplayName": "Cyber Hybrid Hub", + "supportEmail": "support@cyberhybridhub.firebaseapp.com", + "authorizedRedirectUris": [ + "http://localhost" + ] + } + } + }, + "flutter": { + "platforms": { + "android": { + "default": { + "projectId": "cyberhybridhub", + "appId": "1:155018389717:android:90887875660a26509ada5c", + "fileOutput": "android/app/google-services.json" + } + }, + "dart": { + "lib/firebase_options.dart": { + "projectId": "cyberhybridhub", + "configurations": { + "android": "1:155018389717:android:90887875660a26509ada5c", + "ios": "1:155018389717:ios:102445b8ec5fde749ada5c", + "web": "1:155018389717:web:afb61503456673019ada5c" + } + } + } + } + } +} diff --git a/ios/.gitignore b/ios/.gitignore new file mode 100644 index 0000000..7a7f987 --- /dev/null +++ b/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/ios/Flutter/AppFrameworkInfo.plist b/ios/Flutter/AppFrameworkInfo.plist new file mode 100644 index 0000000..391a902 --- /dev/null +++ b/ios/Flutter/AppFrameworkInfo.plist @@ -0,0 +1,24 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + App + CFBundleIdentifier + io.flutter.flutter.app + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + App + CFBundlePackageType + FMWK + CFBundleShortVersionString + 1.0 + CFBundleSignature + ???? + CFBundleVersion + 1.0 + + diff --git a/ios/Flutter/Debug.xcconfig b/ios/Flutter/Debug.xcconfig new file mode 100644 index 0000000..592ceee --- /dev/null +++ b/ios/Flutter/Debug.xcconfig @@ -0,0 +1 @@ +#include "Generated.xcconfig" diff --git a/ios/Flutter/Release.xcconfig b/ios/Flutter/Release.xcconfig new file mode 100644 index 0000000..592ceee --- /dev/null +++ b/ios/Flutter/Release.xcconfig @@ -0,0 +1 @@ +#include "Generated.xcconfig" diff --git a/ios/Runner.xcodeproj/project.pbxproj b/ios/Runner.xcodeproj/project.pbxproj new file mode 100644 index 0000000..c897a53 --- /dev/null +++ b/ios/Runner.xcodeproj/project.pbxproj @@ -0,0 +1,644 @@ +// !$*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 */; }; + 7884E8682EC3CC0700C636F2 /* SceneDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7884E8672EC3CC0400C636F2 /* SceneDelegate.swift */; }; + 78A318202AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage in Frameworks */ = {isa = PBXBuildFile; productRef = 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */; }; + 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 = ""; }; + 7884E8672EC3CC0400C636F2 /* SceneDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SceneDelegate.swift; sourceTree = ""; }; + 78E0A7A72DC9AD7400C4905E /* FlutterGeneratedPluginSwiftPackage */ = {isa = PBXFileReference; lastKnownFileType = wrapper; name = FlutterGeneratedPluginSwiftPackage; path = Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage; 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 = ( + 78A318202AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 331C8082294A63A400263BE5 /* RunnerTests */ = { + isa = PBXGroup; + children = ( + 331C807B294A618700263BE5 /* RunnerTests.swift */, + ); + path = RunnerTests; + sourceTree = ""; + }; + 9740EEB11CF90186004384FC /* Flutter */ = { + isa = PBXGroup; + children = ( + 78E0A7A72DC9AD7400C4905E /* FlutterGeneratedPluginSwiftPackage */, + 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 */, + 7884E8672EC3CC0400C636F2 /* SceneDelegate.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; + packageProductDependencies = ( + 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */, + ); + 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; + packageReferences = ( + 781AD8BC2B33823900A9FFBB /* XCLocalSwiftPackageReference "Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage" */, + ); + 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 */, + 7884E8682EC3CC0700C636F2 /* SceneDelegate.swift 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 = 13.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)"; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_VERSION = 5.0; + 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.cyberhybridhub.cyberhybridhub.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.cyberhybridhub.cyberhybridhub.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.cyberhybridhub.cyberhybridhub.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 = 13.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 = 13.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)"; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + 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)"; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; + SWIFT_VERSION = 5.0; + 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 */ + +/* Begin XCLocalSwiftPackageReference section */ + 781AD8BC2B33823900A9FFBB /* XCLocalSwiftPackageReference "Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage" */ = { + isa = XCLocalSwiftPackageReference; + relativePath = Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage; + }; +/* End XCLocalSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */ = { + isa = XCSwiftPackageProductDependency; + productName = FlutterGeneratedPluginSwiftPackage; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = 97C146E61CF9000F007C117D /* Project object */; +} diff --git a/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 0000000..919434a --- /dev/null +++ b/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 0000000..18d9810 --- /dev/null +++ b/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 0000000..f9b0d7c --- /dev/null +++ b/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme new file mode 100644 index 0000000..c3fedb2 --- /dev/null +++ b/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -0,0 +1,119 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ios/Runner.xcworkspace/contents.xcworkspacedata b/ios/Runner.xcworkspace/contents.xcworkspacedata new file mode 100644 index 0000000..1d526a1 --- /dev/null +++ b/ios/Runner.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 0000000..18d9810 --- /dev/null +++ b/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 0000000..f9b0d7c --- /dev/null +++ b/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/ios/Runner/AppDelegate.swift b/ios/Runner/AppDelegate.swift new file mode 100644 index 0000000..c30b367 --- /dev/null +++ b/ios/Runner/AppDelegate.swift @@ -0,0 +1,16 @@ +import Flutter +import UIKit + +@main +@objc class AppDelegate: FlutterAppDelegate, FlutterImplicitEngineDelegate { + override func application( + _ application: UIApplication, + didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? + ) -> Bool { + return super.application(application, didFinishLaunchingWithOptions: launchOptions) + } + + func didInitializeImplicitFlutterEngine(_ engineBridge: FlutterImplicitEngineBridge) { + GeneratedPluginRegistrant.register(with: engineBridge.pluginRegistry) + } +} diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 0000000..d36b1fa --- /dev/null +++ b/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/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png new file mode 100644 index 0000000..dc9ada4 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png new file mode 100644 index 0000000..7353c41 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png new file mode 100644 index 0000000..797d452 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png new file mode 100644 index 0000000..6ed2d93 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png new file mode 100644 index 0000000..4cd7b00 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png new file mode 100644 index 0000000..fe73094 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png new file mode 100644 index 0000000..321773c Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png new file mode 100644 index 0000000..797d452 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png new file mode 100644 index 0000000..502f463 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png new file mode 100644 index 0000000..0ec3034 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png new file mode 100644 index 0000000..0ec3034 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png new file mode 100644 index 0000000..e9f5fea Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png new file mode 100644 index 0000000..84ac32a Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png new file mode 100644 index 0000000..8953cba Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png new file mode 100644 index 0000000..0467bf1 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json b/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json new file mode 100644 index 0000000..0bedcf2 --- /dev/null +++ b/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/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png new file mode 100644 index 0000000..9da19ea Binary files /dev/null and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png new file mode 100644 index 0000000..9da19ea Binary files /dev/null and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png new file mode 100644 index 0000000..9da19ea Binary files /dev/null and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md b/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md new file mode 100644 index 0000000..89c2725 --- /dev/null +++ b/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/ios/Runner/Base.lproj/LaunchScreen.storyboard b/ios/Runner/Base.lproj/LaunchScreen.storyboard new file mode 100644 index 0000000..f2e259c --- /dev/null +++ b/ios/Runner/Base.lproj/LaunchScreen.storyboard @@ -0,0 +1,37 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ios/Runner/Base.lproj/Main.storyboard b/ios/Runner/Base.lproj/Main.storyboard new file mode 100644 index 0000000..f3c2851 --- /dev/null +++ b/ios/Runner/Base.lproj/Main.storyboard @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ios/Runner/GoogleService-Info.plist b/ios/Runner/GoogleService-Info.plist new file mode 100644 index 0000000..180139a --- /dev/null +++ b/ios/Runner/GoogleService-Info.plist @@ -0,0 +1,34 @@ + + + + + CLIENT_ID + REPLACE_ME.apps.googleusercontent.com + REVERSED_CLIENT_ID + com.googleusercontent.apps.REPLACE_ME + API_KEY + REPLACE_ME + GCM_SENDER_ID + 000000000000 + PLIST_VERSION + 1 + BUNDLE_ID + com.cyberhybridhub.cyberhybridhub + PROJECT_ID + REPLACE_ME + STORAGE_BUCKET + REPLACE_ME.appspot.com + IS_ADS_ENABLED + + IS_ANALYTICS_ENABLED + + IS_APPINVITE_ENABLED + + IS_GCM_ENABLED + + IS_SIGNIN_ENABLED + + GOOGLE_APP_ID + 1:000000000000:ios:0000000000000000000000 + + diff --git a/ios/Runner/Info.plist b/ios/Runner/Info.plist new file mode 100644 index 0000000..68a2122 --- /dev/null +++ b/ios/Runner/Info.plist @@ -0,0 +1,70 @@ + + + + + CADisableMinimumFrameDurationOnPhone + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleDisplayName + Cyberhybridhub + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + cyberhybridhub + CFBundlePackageType + APPL + CFBundleShortVersionString + $(FLUTTER_BUILD_NAME) + CFBundleSignature + ???? + CFBundleVersion + $(FLUTTER_BUILD_NUMBER) + LSRequiresIPhoneOS + + UIApplicationSceneManifest + + UIApplicationSupportsMultipleScenes + + UISceneConfigurations + + UIWindowSceneSessionRoleApplication + + + UISceneClassName + UIWindowScene + UISceneConfigurationName + flutter + UISceneDelegateClassName + $(PRODUCT_MODULE_NAME).SceneDelegate + UISceneStoryboardFile + Main + + + + + UIApplicationSupportsIndirectInputEvents + + UILaunchStoryboardName + LaunchScreen + UIMainStoryboardFile + Main + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + + diff --git a/ios/Runner/Runner-Bridging-Header.h b/ios/Runner/Runner-Bridging-Header.h new file mode 100644 index 0000000..308a2a5 --- /dev/null +++ b/ios/Runner/Runner-Bridging-Header.h @@ -0,0 +1 @@ +#import "GeneratedPluginRegistrant.h" diff --git a/ios/Runner/SceneDelegate.swift b/ios/Runner/SceneDelegate.swift new file mode 100644 index 0000000..b9ce8ea --- /dev/null +++ b/ios/Runner/SceneDelegate.swift @@ -0,0 +1,6 @@ +import Flutter +import UIKit + +class SceneDelegate: FlutterSceneDelegate { + +} diff --git a/ios/RunnerTests/RunnerTests.swift b/ios/RunnerTests/RunnerTests.swift new file mode 100644 index 0000000..86a7c3b --- /dev/null +++ b/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/lib/bootstrap.dart b/lib/bootstrap.dart new file mode 100644 index 0000000..1413706 --- /dev/null +++ b/lib/bootstrap.dart @@ -0,0 +1,16 @@ +import 'package:firebase_core/firebase_core.dart'; + +import 'firebase_options.dart'; +import 'repositories/user_profile_repository.dart'; +import 'services/auth_service.dart'; + +/// Initializes Firebase (non-Linux), auth, and profile sync storage. +Future bootstrap() async { + if (!AuthService.isLinuxDesktop) { + await Firebase.initializeApp( + options: DefaultFirebaseOptions.currentPlatform, + ); + } + await AuthService.instance.initialize(); + await UserProfileRepository.instance.initialize(); +} diff --git a/lib/config/api_config.dart b/lib/config/api_config.dart new file mode 100644 index 0000000..d3da3af --- /dev/null +++ b/lib/config/api_config.dart @@ -0,0 +1,8 @@ +/// Base URL for the Cyber Hybrid Hub API (Postgres-backed). +/// +/// Override at build/run time: +/// flutter run --dart-define=API_BASE_URL=http://localhost:3000 +const String apiBaseUrl = String.fromEnvironment( + 'API_BASE_URL', + defaultValue: 'http://localhost:3000', +); diff --git a/lib/config/auth_config.dart b/lib/config/auth_config.dart new file mode 100644 index 0000000..ea41360 --- /dev/null +++ b/lib/config/auth_config.dart @@ -0,0 +1,25 @@ +/// Web OAuth 2.0 client ID (Google Cloud → Credentials → Web application). +/// +/// Used for: +/// - **Android**: `serverClientId` when not in `google-services.json` +/// - **Linux desktop**: `google_sign_in_all_platforms` clientId +/// +/// **Web (required for Firefox / strict privacy):** `clientId` for the +/// Google Identity Services fallback when Firebase popup sign-in fails. +/// Popup is tried first; redirect is not used (sessionStorage breaks under ETP). +/// +/// Find it: Firebase Console → Project settings → Your apps → Web app, +/// or Authentication → Sign-in method → Google → Web SDK configuration. +/// +/// Also add authorized JavaScript origins in Google Cloud for web dev: +/// `http://localhost:8080` (see [web_dev_config.yaml] in the project root). +const String? googleWebOAuthClientId = + '155018389717-cers33qaak01urmf4sa04s85h7kcocpt.apps.googleusercontent.com'; + +/// @deprecated Use [googleWebOAuthClientId]. +const String? googleSignInServerClientId = googleWebOAuthClientId; + +/// Linux desktop OAuth client secret (same Web client as above). +const String? googleOAuthDesktopClientId = googleWebOAuthClientId; + +const String? googleOAuthDesktopClientSecret = null; diff --git a/lib/data/local/app_database.dart b/lib/data/local/app_database.dart new file mode 100644 index 0000000..41a8b48 --- /dev/null +++ b/lib/data/local/app_database.dart @@ -0,0 +1,71 @@ +import 'package:drift/drift.dart'; +import 'package:drift_flutter/drift_flutter.dart'; + +import 'tables.dart'; + +part 'app_database.g.dart'; + +@DriftDatabase(tables: [UserProfileRows, SyncOutboxRows]) +class AppDatabase extends _$AppDatabase { + AppDatabase([QueryExecutor? executor]) : super(executor ?? _openConnection()); + + @override + int get schemaVersion => 1; + + static QueryExecutor _openConnection() { + return driftDatabase(name: 'cyberhybridhub'); + } + + Stream watchProfile(String firebaseUid) { + return (select(userProfileRows) + ..where((UserProfileRows t) => t.firebaseUid.equals(firebaseUid))) + .watchSingleOrNull(); + } + + Future getProfile(String firebaseUid) { + return (select(userProfileRows) + ..where((UserProfileRows t) => t.firebaseUid.equals(firebaseUid))) + .getSingleOrNull(); + } + + Future upsertProfile(UserProfileRowsCompanion row) { + return into(userProfileRows).insertOnConflictUpdate(row); + } + + Future deleteProfile(String firebaseUid) { + return (delete(userProfileRows) + ..where((UserProfileRows t) => t.firebaseUid.equals(firebaseUid))) + .go(); + } + + Future clearAllProfiles() { + return delete(userProfileRows).go(); + } + + Future> pendingOutbox(String firebaseUid) { + return (select(syncOutboxRows) + ..where((SyncOutboxRows t) => t.firebaseUid.equals(firebaseUid)) + ..orderBy(>[ + (SyncOutboxRows t) => OrderingTerm(expression: t.createdAt), + ])) + .get(); + } + + Future enqueueOutbox(SyncOutboxRowsCompanion row) { + return into(syncOutboxRows).insert(row); + } + + Future deleteOutboxEntry(int id) { + return (delete(syncOutboxRows) + ..where((SyncOutboxRows t) => t.id.equals(id))) + .go(); + } + + Future clearOutbox(String firebaseUid) { + return (delete(syncOutboxRows) + ..where((SyncOutboxRows t) => t.firebaseUid.equals(firebaseUid))) + .go(); + } + + Future clearAllOutbox() => delete(syncOutboxRows).go(); +} diff --git a/lib/data/local/app_database.g.dart b/lib/data/local/app_database.g.dart new file mode 100644 index 0000000..bde6a0c --- /dev/null +++ b/lib/data/local/app_database.g.dart @@ -0,0 +1,1535 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'app_database.dart'; + +// ignore_for_file: type=lint +class $UserProfileRowsTable extends UserProfileRows + with TableInfo<$UserProfileRowsTable, UserProfileRow> { + @override + final GeneratedDatabase attachedDatabase; + final String? _alias; + $UserProfileRowsTable(this.attachedDatabase, [this._alias]); + static const VerificationMeta _firebaseUidMeta = const VerificationMeta( + 'firebaseUid', + ); + @override + late final GeneratedColumn firebaseUid = GeneratedColumn( + 'firebase_uid', + aliasedName, + false, + type: DriftSqlType.string, + requiredDuringInsert: true, + ); + static const VerificationMeta _emailMeta = const VerificationMeta('email'); + @override + late final GeneratedColumn email = GeneratedColumn( + 'email', + aliasedName, + true, + type: DriftSqlType.string, + requiredDuringInsert: false, + ); + static const VerificationMeta _displayNameMeta = const VerificationMeta( + 'displayName', + ); + @override + late final GeneratedColumn displayName = GeneratedColumn( + 'display_name', + aliasedName, + true, + type: DriftSqlType.string, + requiredDuringInsert: false, + ); + static const VerificationMeta _photoUrlMeta = const VerificationMeta( + 'photoUrl', + ); + @override + late final GeneratedColumn photoUrl = GeneratedColumn( + 'photo_url', + aliasedName, + true, + type: DriftSqlType.string, + requiredDuringInsert: false, + ); + static const VerificationMeta _localeMeta = const VerificationMeta('locale'); + @override + late final GeneratedColumn locale = GeneratedColumn( + 'locale', + aliasedName, + false, + type: DriftSqlType.string, + requiredDuringInsert: false, + defaultValue: const Constant('en'), + ); + static const VerificationMeta _timezoneMeta = const VerificationMeta( + 'timezone', + ); + @override + late final GeneratedColumn timezone = GeneratedColumn( + 'timezone', + aliasedName, + true, + type: DriftSqlType.string, + requiredDuringInsert: false, + ); + static const VerificationMeta _onboardingCompletedMeta = + const VerificationMeta('onboardingCompleted'); + @override + late final GeneratedColumn onboardingCompleted = GeneratedColumn( + 'onboarding_completed', + aliasedName, + false, + type: DriftSqlType.bool, + requiredDuringInsert: false, + defaultConstraints: GeneratedColumn.constraintIsAlways( + 'CHECK ("onboarding_completed" IN (0, 1))', + ), + defaultValue: const Constant(false), + ); + static const VerificationMeta _revisionMeta = const VerificationMeta( + 'revision', + ); + @override + late final GeneratedColumn revision = GeneratedColumn( + 'revision', + aliasedName, + false, + type: DriftSqlType.int, + requiredDuringInsert: false, + defaultValue: const Constant(1), + ); + static const VerificationMeta _updatedAtMeta = const VerificationMeta( + 'updatedAt', + ); + @override + late final GeneratedColumn updatedAt = GeneratedColumn( + 'updated_at', + aliasedName, + false, + type: DriftSqlType.dateTime, + requiredDuringInsert: true, + ); + static const VerificationMeta _lastSyncedAtMeta = const VerificationMeta( + 'lastSyncedAt', + ); + @override + late final GeneratedColumn lastSyncedAt = GeneratedColumn( + 'last_synced_at', + aliasedName, + true, + type: DriftSqlType.dateTime, + requiredDuringInsert: false, + ); + static const VerificationMeta _dirtyMeta = const VerificationMeta('dirty'); + @override + late final GeneratedColumn dirty = GeneratedColumn( + 'dirty', + aliasedName, + false, + type: DriftSqlType.bool, + requiredDuringInsert: false, + defaultConstraints: GeneratedColumn.constraintIsAlways( + 'CHECK ("dirty" IN (0, 1))', + ), + defaultValue: const Constant(false), + ); + @override + List get $columns => [ + firebaseUid, + email, + displayName, + photoUrl, + locale, + timezone, + onboardingCompleted, + revision, + updatedAt, + lastSyncedAt, + dirty, + ]; + @override + String get aliasedName => _alias ?? actualTableName; + @override + String get actualTableName => $name; + static const String $name = 'user_profile_rows'; + @override + VerificationContext validateIntegrity( + Insertable instance, { + bool isInserting = false, + }) { + final context = VerificationContext(); + final data = instance.toColumns(true); + if (data.containsKey('firebase_uid')) { + context.handle( + _firebaseUidMeta, + firebaseUid.isAcceptableOrUnknown( + data['firebase_uid']!, + _firebaseUidMeta, + ), + ); + } else if (isInserting) { + context.missing(_firebaseUidMeta); + } + if (data.containsKey('email')) { + context.handle( + _emailMeta, + email.isAcceptableOrUnknown(data['email']!, _emailMeta), + ); + } + if (data.containsKey('display_name')) { + context.handle( + _displayNameMeta, + displayName.isAcceptableOrUnknown( + data['display_name']!, + _displayNameMeta, + ), + ); + } + if (data.containsKey('photo_url')) { + context.handle( + _photoUrlMeta, + photoUrl.isAcceptableOrUnknown(data['photo_url']!, _photoUrlMeta), + ); + } + if (data.containsKey('locale')) { + context.handle( + _localeMeta, + locale.isAcceptableOrUnknown(data['locale']!, _localeMeta), + ); + } + if (data.containsKey('timezone')) { + context.handle( + _timezoneMeta, + timezone.isAcceptableOrUnknown(data['timezone']!, _timezoneMeta), + ); + } + if (data.containsKey('onboarding_completed')) { + context.handle( + _onboardingCompletedMeta, + onboardingCompleted.isAcceptableOrUnknown( + data['onboarding_completed']!, + _onboardingCompletedMeta, + ), + ); + } + if (data.containsKey('revision')) { + context.handle( + _revisionMeta, + revision.isAcceptableOrUnknown(data['revision']!, _revisionMeta), + ); + } + if (data.containsKey('updated_at')) { + context.handle( + _updatedAtMeta, + updatedAt.isAcceptableOrUnknown(data['updated_at']!, _updatedAtMeta), + ); + } else if (isInserting) { + context.missing(_updatedAtMeta); + } + if (data.containsKey('last_synced_at')) { + context.handle( + _lastSyncedAtMeta, + lastSyncedAt.isAcceptableOrUnknown( + data['last_synced_at']!, + _lastSyncedAtMeta, + ), + ); + } + if (data.containsKey('dirty')) { + context.handle( + _dirtyMeta, + dirty.isAcceptableOrUnknown(data['dirty']!, _dirtyMeta), + ); + } + return context; + } + + @override + Set get $primaryKey => {firebaseUid}; + @override + UserProfileRow map(Map data, {String? tablePrefix}) { + final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; + return UserProfileRow( + firebaseUid: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}firebase_uid'], + )!, + email: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}email'], + ), + displayName: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}display_name'], + ), + photoUrl: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}photo_url'], + ), + locale: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}locale'], + )!, + timezone: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}timezone'], + ), + onboardingCompleted: attachedDatabase.typeMapping.read( + DriftSqlType.bool, + data['${effectivePrefix}onboarding_completed'], + )!, + revision: attachedDatabase.typeMapping.read( + DriftSqlType.int, + data['${effectivePrefix}revision'], + )!, + updatedAt: attachedDatabase.typeMapping.read( + DriftSqlType.dateTime, + data['${effectivePrefix}updated_at'], + )!, + lastSyncedAt: attachedDatabase.typeMapping.read( + DriftSqlType.dateTime, + data['${effectivePrefix}last_synced_at'], + ), + dirty: attachedDatabase.typeMapping.read( + DriftSqlType.bool, + data['${effectivePrefix}dirty'], + )!, + ); + } + + @override + $UserProfileRowsTable createAlias(String alias) { + return $UserProfileRowsTable(attachedDatabase, alias); + } +} + +class UserProfileRow extends DataClass implements Insertable { + final String firebaseUid; + final String? email; + final String? displayName; + final String? photoUrl; + final String locale; + final String? timezone; + final bool onboardingCompleted; + final int revision; + final DateTime updatedAt; + final DateTime? lastSyncedAt; + final bool dirty; + const UserProfileRow({ + required this.firebaseUid, + this.email, + this.displayName, + this.photoUrl, + required this.locale, + this.timezone, + required this.onboardingCompleted, + required this.revision, + required this.updatedAt, + this.lastSyncedAt, + required this.dirty, + }); + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + map['firebase_uid'] = Variable(firebaseUid); + if (!nullToAbsent || email != null) { + map['email'] = Variable(email); + } + if (!nullToAbsent || displayName != null) { + map['display_name'] = Variable(displayName); + } + if (!nullToAbsent || photoUrl != null) { + map['photo_url'] = Variable(photoUrl); + } + map['locale'] = Variable(locale); + if (!nullToAbsent || timezone != null) { + map['timezone'] = Variable(timezone); + } + map['onboarding_completed'] = Variable(onboardingCompleted); + map['revision'] = Variable(revision); + map['updated_at'] = Variable(updatedAt); + if (!nullToAbsent || lastSyncedAt != null) { + map['last_synced_at'] = Variable(lastSyncedAt); + } + map['dirty'] = Variable(dirty); + return map; + } + + UserProfileRowsCompanion toCompanion(bool nullToAbsent) { + return UserProfileRowsCompanion( + firebaseUid: Value(firebaseUid), + email: email == null && nullToAbsent + ? const Value.absent() + : Value(email), + displayName: displayName == null && nullToAbsent + ? const Value.absent() + : Value(displayName), + photoUrl: photoUrl == null && nullToAbsent + ? const Value.absent() + : Value(photoUrl), + locale: Value(locale), + timezone: timezone == null && nullToAbsent + ? const Value.absent() + : Value(timezone), + onboardingCompleted: Value(onboardingCompleted), + revision: Value(revision), + updatedAt: Value(updatedAt), + lastSyncedAt: lastSyncedAt == null && nullToAbsent + ? const Value.absent() + : Value(lastSyncedAt), + dirty: Value(dirty), + ); + } + + factory UserProfileRow.fromJson( + Map json, { + ValueSerializer? serializer, + }) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return UserProfileRow( + firebaseUid: serializer.fromJson(json['firebaseUid']), + email: serializer.fromJson(json['email']), + displayName: serializer.fromJson(json['displayName']), + photoUrl: serializer.fromJson(json['photoUrl']), + locale: serializer.fromJson(json['locale']), + timezone: serializer.fromJson(json['timezone']), + onboardingCompleted: serializer.fromJson( + json['onboardingCompleted'], + ), + revision: serializer.fromJson(json['revision']), + updatedAt: serializer.fromJson(json['updatedAt']), + lastSyncedAt: serializer.fromJson(json['lastSyncedAt']), + dirty: serializer.fromJson(json['dirty']), + ); + } + @override + Map toJson({ValueSerializer? serializer}) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return { + 'firebaseUid': serializer.toJson(firebaseUid), + 'email': serializer.toJson(email), + 'displayName': serializer.toJson(displayName), + 'photoUrl': serializer.toJson(photoUrl), + 'locale': serializer.toJson(locale), + 'timezone': serializer.toJson(timezone), + 'onboardingCompleted': serializer.toJson(onboardingCompleted), + 'revision': serializer.toJson(revision), + 'updatedAt': serializer.toJson(updatedAt), + 'lastSyncedAt': serializer.toJson(lastSyncedAt), + 'dirty': serializer.toJson(dirty), + }; + } + + UserProfileRow copyWith({ + String? firebaseUid, + Value email = const Value.absent(), + Value displayName = const Value.absent(), + Value photoUrl = const Value.absent(), + String? locale, + Value timezone = const Value.absent(), + bool? onboardingCompleted, + int? revision, + DateTime? updatedAt, + Value lastSyncedAt = const Value.absent(), + bool? dirty, + }) => UserProfileRow( + firebaseUid: firebaseUid ?? this.firebaseUid, + email: email.present ? email.value : this.email, + displayName: displayName.present ? displayName.value : this.displayName, + photoUrl: photoUrl.present ? photoUrl.value : this.photoUrl, + locale: locale ?? this.locale, + timezone: timezone.present ? timezone.value : this.timezone, + onboardingCompleted: onboardingCompleted ?? this.onboardingCompleted, + revision: revision ?? this.revision, + updatedAt: updatedAt ?? this.updatedAt, + lastSyncedAt: lastSyncedAt.present ? lastSyncedAt.value : this.lastSyncedAt, + dirty: dirty ?? this.dirty, + ); + UserProfileRow copyWithCompanion(UserProfileRowsCompanion data) { + return UserProfileRow( + firebaseUid: data.firebaseUid.present + ? data.firebaseUid.value + : this.firebaseUid, + email: data.email.present ? data.email.value : this.email, + displayName: data.displayName.present + ? data.displayName.value + : this.displayName, + photoUrl: data.photoUrl.present ? data.photoUrl.value : this.photoUrl, + locale: data.locale.present ? data.locale.value : this.locale, + timezone: data.timezone.present ? data.timezone.value : this.timezone, + onboardingCompleted: data.onboardingCompleted.present + ? data.onboardingCompleted.value + : this.onboardingCompleted, + revision: data.revision.present ? data.revision.value : this.revision, + updatedAt: data.updatedAt.present ? data.updatedAt.value : this.updatedAt, + lastSyncedAt: data.lastSyncedAt.present + ? data.lastSyncedAt.value + : this.lastSyncedAt, + dirty: data.dirty.present ? data.dirty.value : this.dirty, + ); + } + + @override + String toString() { + return (StringBuffer('UserProfileRow(') + ..write('firebaseUid: $firebaseUid, ') + ..write('email: $email, ') + ..write('displayName: $displayName, ') + ..write('photoUrl: $photoUrl, ') + ..write('locale: $locale, ') + ..write('timezone: $timezone, ') + ..write('onboardingCompleted: $onboardingCompleted, ') + ..write('revision: $revision, ') + ..write('updatedAt: $updatedAt, ') + ..write('lastSyncedAt: $lastSyncedAt, ') + ..write('dirty: $dirty') + ..write(')')) + .toString(); + } + + @override + int get hashCode => Object.hash( + firebaseUid, + email, + displayName, + photoUrl, + locale, + timezone, + onboardingCompleted, + revision, + updatedAt, + lastSyncedAt, + dirty, + ); + @override + bool operator ==(Object other) => + identical(this, other) || + (other is UserProfileRow && + other.firebaseUid == this.firebaseUid && + other.email == this.email && + other.displayName == this.displayName && + other.photoUrl == this.photoUrl && + other.locale == this.locale && + other.timezone == this.timezone && + other.onboardingCompleted == this.onboardingCompleted && + other.revision == this.revision && + other.updatedAt == this.updatedAt && + other.lastSyncedAt == this.lastSyncedAt && + other.dirty == this.dirty); +} + +class UserProfileRowsCompanion extends UpdateCompanion { + final Value firebaseUid; + final Value email; + final Value displayName; + final Value photoUrl; + final Value locale; + final Value timezone; + final Value onboardingCompleted; + final Value revision; + final Value updatedAt; + final Value lastSyncedAt; + final Value dirty; + final Value rowid; + const UserProfileRowsCompanion({ + this.firebaseUid = const Value.absent(), + this.email = const Value.absent(), + this.displayName = const Value.absent(), + this.photoUrl = const Value.absent(), + this.locale = const Value.absent(), + this.timezone = const Value.absent(), + this.onboardingCompleted = const Value.absent(), + this.revision = const Value.absent(), + this.updatedAt = const Value.absent(), + this.lastSyncedAt = const Value.absent(), + this.dirty = const Value.absent(), + this.rowid = const Value.absent(), + }); + UserProfileRowsCompanion.insert({ + required String firebaseUid, + this.email = const Value.absent(), + this.displayName = const Value.absent(), + this.photoUrl = const Value.absent(), + this.locale = const Value.absent(), + this.timezone = const Value.absent(), + this.onboardingCompleted = const Value.absent(), + this.revision = const Value.absent(), + required DateTime updatedAt, + this.lastSyncedAt = const Value.absent(), + this.dirty = const Value.absent(), + this.rowid = const Value.absent(), + }) : firebaseUid = Value(firebaseUid), + updatedAt = Value(updatedAt); + static Insertable custom({ + Expression? firebaseUid, + Expression? email, + Expression? displayName, + Expression? photoUrl, + Expression? locale, + Expression? timezone, + Expression? onboardingCompleted, + Expression? revision, + Expression? updatedAt, + Expression? lastSyncedAt, + Expression? dirty, + Expression? rowid, + }) { + return RawValuesInsertable({ + if (firebaseUid != null) 'firebase_uid': firebaseUid, + if (email != null) 'email': email, + if (displayName != null) 'display_name': displayName, + if (photoUrl != null) 'photo_url': photoUrl, + if (locale != null) 'locale': locale, + if (timezone != null) 'timezone': timezone, + if (onboardingCompleted != null) + 'onboarding_completed': onboardingCompleted, + if (revision != null) 'revision': revision, + if (updatedAt != null) 'updated_at': updatedAt, + if (lastSyncedAt != null) 'last_synced_at': lastSyncedAt, + if (dirty != null) 'dirty': dirty, + if (rowid != null) 'rowid': rowid, + }); + } + + UserProfileRowsCompanion copyWith({ + Value? firebaseUid, + Value? email, + Value? displayName, + Value? photoUrl, + Value? locale, + Value? timezone, + Value? onboardingCompleted, + Value? revision, + Value? updatedAt, + Value? lastSyncedAt, + Value? dirty, + Value? rowid, + }) { + return UserProfileRowsCompanion( + firebaseUid: firebaseUid ?? this.firebaseUid, + email: email ?? this.email, + displayName: displayName ?? this.displayName, + photoUrl: photoUrl ?? this.photoUrl, + locale: locale ?? this.locale, + timezone: timezone ?? this.timezone, + onboardingCompleted: onboardingCompleted ?? this.onboardingCompleted, + revision: revision ?? this.revision, + updatedAt: updatedAt ?? this.updatedAt, + lastSyncedAt: lastSyncedAt ?? this.lastSyncedAt, + dirty: dirty ?? this.dirty, + rowid: rowid ?? this.rowid, + ); + } + + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + if (firebaseUid.present) { + map['firebase_uid'] = Variable(firebaseUid.value); + } + if (email.present) { + map['email'] = Variable(email.value); + } + if (displayName.present) { + map['display_name'] = Variable(displayName.value); + } + if (photoUrl.present) { + map['photo_url'] = Variable(photoUrl.value); + } + if (locale.present) { + map['locale'] = Variable(locale.value); + } + if (timezone.present) { + map['timezone'] = Variable(timezone.value); + } + if (onboardingCompleted.present) { + map['onboarding_completed'] = Variable(onboardingCompleted.value); + } + if (revision.present) { + map['revision'] = Variable(revision.value); + } + if (updatedAt.present) { + map['updated_at'] = Variable(updatedAt.value); + } + if (lastSyncedAt.present) { + map['last_synced_at'] = Variable(lastSyncedAt.value); + } + if (dirty.present) { + map['dirty'] = Variable(dirty.value); + } + if (rowid.present) { + map['rowid'] = Variable(rowid.value); + } + return map; + } + + @override + String toString() { + return (StringBuffer('UserProfileRowsCompanion(') + ..write('firebaseUid: $firebaseUid, ') + ..write('email: $email, ') + ..write('displayName: $displayName, ') + ..write('photoUrl: $photoUrl, ') + ..write('locale: $locale, ') + ..write('timezone: $timezone, ') + ..write('onboardingCompleted: $onboardingCompleted, ') + ..write('revision: $revision, ') + ..write('updatedAt: $updatedAt, ') + ..write('lastSyncedAt: $lastSyncedAt, ') + ..write('dirty: $dirty, ') + ..write('rowid: $rowid') + ..write(')')) + .toString(); + } +} + +class $SyncOutboxRowsTable extends SyncOutboxRows + with TableInfo<$SyncOutboxRowsTable, SyncOutboxRow> { + @override + final GeneratedDatabase attachedDatabase; + final String? _alias; + $SyncOutboxRowsTable(this.attachedDatabase, [this._alias]); + static const VerificationMeta _idMeta = const VerificationMeta('id'); + @override + late final GeneratedColumn id = GeneratedColumn( + 'id', + aliasedName, + false, + hasAutoIncrement: true, + type: DriftSqlType.int, + requiredDuringInsert: false, + defaultConstraints: GeneratedColumn.constraintIsAlways( + 'PRIMARY KEY AUTOINCREMENT', + ), + ); + static const VerificationMeta _firebaseUidMeta = const VerificationMeta( + 'firebaseUid', + ); + @override + late final GeneratedColumn firebaseUid = GeneratedColumn( + 'firebase_uid', + aliasedName, + false, + type: DriftSqlType.string, + requiredDuringInsert: true, + ); + static const VerificationMeta _payloadJsonMeta = const VerificationMeta( + 'payloadJson', + ); + @override + late final GeneratedColumn payloadJson = GeneratedColumn( + 'payload_json', + aliasedName, + false, + type: DriftSqlType.string, + requiredDuringInsert: true, + ); + static const VerificationMeta _createdAtMeta = const VerificationMeta( + 'createdAt', + ); + @override + late final GeneratedColumn createdAt = GeneratedColumn( + 'created_at', + aliasedName, + false, + type: DriftSqlType.dateTime, + requiredDuringInsert: true, + ); + @override + List get $columns => [ + id, + firebaseUid, + payloadJson, + createdAt, + ]; + @override + String get aliasedName => _alias ?? actualTableName; + @override + String get actualTableName => $name; + static const String $name = 'sync_outbox_rows'; + @override + VerificationContext validateIntegrity( + Insertable instance, { + bool isInserting = false, + }) { + final context = VerificationContext(); + final data = instance.toColumns(true); + if (data.containsKey('id')) { + context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); + } + if (data.containsKey('firebase_uid')) { + context.handle( + _firebaseUidMeta, + firebaseUid.isAcceptableOrUnknown( + data['firebase_uid']!, + _firebaseUidMeta, + ), + ); + } else if (isInserting) { + context.missing(_firebaseUidMeta); + } + if (data.containsKey('payload_json')) { + context.handle( + _payloadJsonMeta, + payloadJson.isAcceptableOrUnknown( + data['payload_json']!, + _payloadJsonMeta, + ), + ); + } else if (isInserting) { + context.missing(_payloadJsonMeta); + } + if (data.containsKey('created_at')) { + context.handle( + _createdAtMeta, + createdAt.isAcceptableOrUnknown(data['created_at']!, _createdAtMeta), + ); + } else if (isInserting) { + context.missing(_createdAtMeta); + } + return context; + } + + @override + Set get $primaryKey => {id}; + @override + SyncOutboxRow map(Map data, {String? tablePrefix}) { + final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; + return SyncOutboxRow( + id: attachedDatabase.typeMapping.read( + DriftSqlType.int, + data['${effectivePrefix}id'], + )!, + firebaseUid: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}firebase_uid'], + )!, + payloadJson: attachedDatabase.typeMapping.read( + DriftSqlType.string, + data['${effectivePrefix}payload_json'], + )!, + createdAt: attachedDatabase.typeMapping.read( + DriftSqlType.dateTime, + data['${effectivePrefix}created_at'], + )!, + ); + } + + @override + $SyncOutboxRowsTable createAlias(String alias) { + return $SyncOutboxRowsTable(attachedDatabase, alias); + } +} + +class SyncOutboxRow extends DataClass implements Insertable { + final int id; + final String firebaseUid; + final String payloadJson; + final DateTime createdAt; + const SyncOutboxRow({ + required this.id, + required this.firebaseUid, + required this.payloadJson, + required this.createdAt, + }); + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + map['id'] = Variable(id); + map['firebase_uid'] = Variable(firebaseUid); + map['payload_json'] = Variable(payloadJson); + map['created_at'] = Variable(createdAt); + return map; + } + + SyncOutboxRowsCompanion toCompanion(bool nullToAbsent) { + return SyncOutboxRowsCompanion( + id: Value(id), + firebaseUid: Value(firebaseUid), + payloadJson: Value(payloadJson), + createdAt: Value(createdAt), + ); + } + + factory SyncOutboxRow.fromJson( + Map json, { + ValueSerializer? serializer, + }) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return SyncOutboxRow( + id: serializer.fromJson(json['id']), + firebaseUid: serializer.fromJson(json['firebaseUid']), + payloadJson: serializer.fromJson(json['payloadJson']), + createdAt: serializer.fromJson(json['createdAt']), + ); + } + @override + Map toJson({ValueSerializer? serializer}) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return { + 'id': serializer.toJson(id), + 'firebaseUid': serializer.toJson(firebaseUid), + 'payloadJson': serializer.toJson(payloadJson), + 'createdAt': serializer.toJson(createdAt), + }; + } + + SyncOutboxRow copyWith({ + int? id, + String? firebaseUid, + String? payloadJson, + DateTime? createdAt, + }) => SyncOutboxRow( + id: id ?? this.id, + firebaseUid: firebaseUid ?? this.firebaseUid, + payloadJson: payloadJson ?? this.payloadJson, + createdAt: createdAt ?? this.createdAt, + ); + SyncOutboxRow copyWithCompanion(SyncOutboxRowsCompanion data) { + return SyncOutboxRow( + id: data.id.present ? data.id.value : this.id, + firebaseUid: data.firebaseUid.present + ? data.firebaseUid.value + : this.firebaseUid, + payloadJson: data.payloadJson.present + ? data.payloadJson.value + : this.payloadJson, + createdAt: data.createdAt.present ? data.createdAt.value : this.createdAt, + ); + } + + @override + String toString() { + return (StringBuffer('SyncOutboxRow(') + ..write('id: $id, ') + ..write('firebaseUid: $firebaseUid, ') + ..write('payloadJson: $payloadJson, ') + ..write('createdAt: $createdAt') + ..write(')')) + .toString(); + } + + @override + int get hashCode => Object.hash(id, firebaseUid, payloadJson, createdAt); + @override + bool operator ==(Object other) => + identical(this, other) || + (other is SyncOutboxRow && + other.id == this.id && + other.firebaseUid == this.firebaseUid && + other.payloadJson == this.payloadJson && + other.createdAt == this.createdAt); +} + +class SyncOutboxRowsCompanion extends UpdateCompanion { + final Value id; + final Value firebaseUid; + final Value payloadJson; + final Value createdAt; + const SyncOutboxRowsCompanion({ + this.id = const Value.absent(), + this.firebaseUid = const Value.absent(), + this.payloadJson = const Value.absent(), + this.createdAt = const Value.absent(), + }); + SyncOutboxRowsCompanion.insert({ + this.id = const Value.absent(), + required String firebaseUid, + required String payloadJson, + required DateTime createdAt, + }) : firebaseUid = Value(firebaseUid), + payloadJson = Value(payloadJson), + createdAt = Value(createdAt); + static Insertable custom({ + Expression? id, + Expression? firebaseUid, + Expression? payloadJson, + Expression? createdAt, + }) { + return RawValuesInsertable({ + if (id != null) 'id': id, + if (firebaseUid != null) 'firebase_uid': firebaseUid, + if (payloadJson != null) 'payload_json': payloadJson, + if (createdAt != null) 'created_at': createdAt, + }); + } + + SyncOutboxRowsCompanion copyWith({ + Value? id, + Value? firebaseUid, + Value? payloadJson, + Value? createdAt, + }) { + return SyncOutboxRowsCompanion( + id: id ?? this.id, + firebaseUid: firebaseUid ?? this.firebaseUid, + payloadJson: payloadJson ?? this.payloadJson, + createdAt: createdAt ?? this.createdAt, + ); + } + + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + if (id.present) { + map['id'] = Variable(id.value); + } + if (firebaseUid.present) { + map['firebase_uid'] = Variable(firebaseUid.value); + } + if (payloadJson.present) { + map['payload_json'] = Variable(payloadJson.value); + } + if (createdAt.present) { + map['created_at'] = Variable(createdAt.value); + } + return map; + } + + @override + String toString() { + return (StringBuffer('SyncOutboxRowsCompanion(') + ..write('id: $id, ') + ..write('firebaseUid: $firebaseUid, ') + ..write('payloadJson: $payloadJson, ') + ..write('createdAt: $createdAt') + ..write(')')) + .toString(); + } +} + +abstract class _$AppDatabase extends GeneratedDatabase { + _$AppDatabase(QueryExecutor e) : super(e); + $AppDatabaseManager get managers => $AppDatabaseManager(this); + late final $UserProfileRowsTable userProfileRows = $UserProfileRowsTable( + this, + ); + late final $SyncOutboxRowsTable syncOutboxRows = $SyncOutboxRowsTable(this); + @override + Iterable> get allTables => + allSchemaEntities.whereType>(); + @override + List get allSchemaEntities => [ + userProfileRows, + syncOutboxRows, + ]; +} + +typedef $$UserProfileRowsTableCreateCompanionBuilder = + UserProfileRowsCompanion Function({ + required String firebaseUid, + Value email, + Value displayName, + Value photoUrl, + Value locale, + Value timezone, + Value onboardingCompleted, + Value revision, + required DateTime updatedAt, + Value lastSyncedAt, + Value dirty, + Value rowid, + }); +typedef $$UserProfileRowsTableUpdateCompanionBuilder = + UserProfileRowsCompanion Function({ + Value firebaseUid, + Value email, + Value displayName, + Value photoUrl, + Value locale, + Value timezone, + Value onboardingCompleted, + Value revision, + Value updatedAt, + Value lastSyncedAt, + Value dirty, + Value rowid, + }); + +class $$UserProfileRowsTableFilterComposer + extends Composer<_$AppDatabase, $UserProfileRowsTable> { + $$UserProfileRowsTableFilterComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnFilters get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get email => $composableBuilder( + column: $table.email, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get displayName => $composableBuilder( + column: $table.displayName, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get photoUrl => $composableBuilder( + column: $table.photoUrl, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get locale => $composableBuilder( + column: $table.locale, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get timezone => $composableBuilder( + column: $table.timezone, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get onboardingCompleted => $composableBuilder( + column: $table.onboardingCompleted, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get revision => $composableBuilder( + column: $table.revision, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get updatedAt => $composableBuilder( + column: $table.updatedAt, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get lastSyncedAt => $composableBuilder( + column: $table.lastSyncedAt, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get dirty => $composableBuilder( + column: $table.dirty, + builder: (column) => ColumnFilters(column), + ); +} + +class $$UserProfileRowsTableOrderingComposer + extends Composer<_$AppDatabase, $UserProfileRowsTable> { + $$UserProfileRowsTableOrderingComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnOrderings get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get email => $composableBuilder( + column: $table.email, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get displayName => $composableBuilder( + column: $table.displayName, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get photoUrl => $composableBuilder( + column: $table.photoUrl, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get locale => $composableBuilder( + column: $table.locale, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get timezone => $composableBuilder( + column: $table.timezone, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get onboardingCompleted => $composableBuilder( + column: $table.onboardingCompleted, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get revision => $composableBuilder( + column: $table.revision, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get updatedAt => $composableBuilder( + column: $table.updatedAt, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get lastSyncedAt => $composableBuilder( + column: $table.lastSyncedAt, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get dirty => $composableBuilder( + column: $table.dirty, + builder: (column) => ColumnOrderings(column), + ); +} + +class $$UserProfileRowsTableAnnotationComposer + extends Composer<_$AppDatabase, $UserProfileRowsTable> { + $$UserProfileRowsTableAnnotationComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + GeneratedColumn get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => column, + ); + + GeneratedColumn get email => + $composableBuilder(column: $table.email, builder: (column) => column); + + GeneratedColumn get displayName => $composableBuilder( + column: $table.displayName, + builder: (column) => column, + ); + + GeneratedColumn get photoUrl => + $composableBuilder(column: $table.photoUrl, builder: (column) => column); + + GeneratedColumn get locale => + $composableBuilder(column: $table.locale, builder: (column) => column); + + GeneratedColumn get timezone => + $composableBuilder(column: $table.timezone, builder: (column) => column); + + GeneratedColumn get onboardingCompleted => $composableBuilder( + column: $table.onboardingCompleted, + builder: (column) => column, + ); + + GeneratedColumn get revision => + $composableBuilder(column: $table.revision, builder: (column) => column); + + GeneratedColumn get updatedAt => + $composableBuilder(column: $table.updatedAt, builder: (column) => column); + + GeneratedColumn get lastSyncedAt => $composableBuilder( + column: $table.lastSyncedAt, + builder: (column) => column, + ); + + GeneratedColumn get dirty => + $composableBuilder(column: $table.dirty, builder: (column) => column); +} + +class $$UserProfileRowsTableTableManager + extends + RootTableManager< + _$AppDatabase, + $UserProfileRowsTable, + UserProfileRow, + $$UserProfileRowsTableFilterComposer, + $$UserProfileRowsTableOrderingComposer, + $$UserProfileRowsTableAnnotationComposer, + $$UserProfileRowsTableCreateCompanionBuilder, + $$UserProfileRowsTableUpdateCompanionBuilder, + ( + UserProfileRow, + BaseReferences< + _$AppDatabase, + $UserProfileRowsTable, + UserProfileRow + >, + ), + UserProfileRow, + PrefetchHooks Function() + > { + $$UserProfileRowsTableTableManager( + _$AppDatabase db, + $UserProfileRowsTable table, + ) : super( + TableManagerState( + db: db, + table: table, + createFilteringComposer: () => + $$UserProfileRowsTableFilterComposer($db: db, $table: table), + createOrderingComposer: () => + $$UserProfileRowsTableOrderingComposer($db: db, $table: table), + createComputedFieldComposer: () => + $$UserProfileRowsTableAnnotationComposer($db: db, $table: table), + updateCompanionCallback: + ({ + Value firebaseUid = const Value.absent(), + Value email = const Value.absent(), + Value displayName = const Value.absent(), + Value photoUrl = const Value.absent(), + Value locale = const Value.absent(), + Value timezone = const Value.absent(), + Value onboardingCompleted = const Value.absent(), + Value revision = const Value.absent(), + Value updatedAt = const Value.absent(), + Value lastSyncedAt = const Value.absent(), + Value dirty = const Value.absent(), + Value rowid = const Value.absent(), + }) => UserProfileRowsCompanion( + firebaseUid: firebaseUid, + email: email, + displayName: displayName, + photoUrl: photoUrl, + locale: locale, + timezone: timezone, + onboardingCompleted: onboardingCompleted, + revision: revision, + updatedAt: updatedAt, + lastSyncedAt: lastSyncedAt, + dirty: dirty, + rowid: rowid, + ), + createCompanionCallback: + ({ + required String firebaseUid, + Value email = const Value.absent(), + Value displayName = const Value.absent(), + Value photoUrl = const Value.absent(), + Value locale = const Value.absent(), + Value timezone = const Value.absent(), + Value onboardingCompleted = const Value.absent(), + Value revision = const Value.absent(), + required DateTime updatedAt, + Value lastSyncedAt = const Value.absent(), + Value dirty = const Value.absent(), + Value rowid = const Value.absent(), + }) => UserProfileRowsCompanion.insert( + firebaseUid: firebaseUid, + email: email, + displayName: displayName, + photoUrl: photoUrl, + locale: locale, + timezone: timezone, + onboardingCompleted: onboardingCompleted, + revision: revision, + updatedAt: updatedAt, + lastSyncedAt: lastSyncedAt, + dirty: dirty, + rowid: rowid, + ), + withReferenceMapper: (p0) => p0 + .map((e) => (e.readTable(table), BaseReferences(db, table, e))) + .toList(), + prefetchHooksCallback: null, + ), + ); +} + +typedef $$UserProfileRowsTableProcessedTableManager = + ProcessedTableManager< + _$AppDatabase, + $UserProfileRowsTable, + UserProfileRow, + $$UserProfileRowsTableFilterComposer, + $$UserProfileRowsTableOrderingComposer, + $$UserProfileRowsTableAnnotationComposer, + $$UserProfileRowsTableCreateCompanionBuilder, + $$UserProfileRowsTableUpdateCompanionBuilder, + ( + UserProfileRow, + BaseReferences<_$AppDatabase, $UserProfileRowsTable, UserProfileRow>, + ), + UserProfileRow, + PrefetchHooks Function() + >; +typedef $$SyncOutboxRowsTableCreateCompanionBuilder = + SyncOutboxRowsCompanion Function({ + Value id, + required String firebaseUid, + required String payloadJson, + required DateTime createdAt, + }); +typedef $$SyncOutboxRowsTableUpdateCompanionBuilder = + SyncOutboxRowsCompanion Function({ + Value id, + Value firebaseUid, + Value payloadJson, + Value createdAt, + }); + +class $$SyncOutboxRowsTableFilterComposer + extends Composer<_$AppDatabase, $SyncOutboxRowsTable> { + $$SyncOutboxRowsTableFilterComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnFilters get id => $composableBuilder( + column: $table.id, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get payloadJson => $composableBuilder( + column: $table.payloadJson, + builder: (column) => ColumnFilters(column), + ); + + ColumnFilters get createdAt => $composableBuilder( + column: $table.createdAt, + builder: (column) => ColumnFilters(column), + ); +} + +class $$SyncOutboxRowsTableOrderingComposer + extends Composer<_$AppDatabase, $SyncOutboxRowsTable> { + $$SyncOutboxRowsTableOrderingComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnOrderings get id => $composableBuilder( + column: $table.id, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get payloadJson => $composableBuilder( + column: $table.payloadJson, + builder: (column) => ColumnOrderings(column), + ); + + ColumnOrderings get createdAt => $composableBuilder( + column: $table.createdAt, + builder: (column) => ColumnOrderings(column), + ); +} + +class $$SyncOutboxRowsTableAnnotationComposer + extends Composer<_$AppDatabase, $SyncOutboxRowsTable> { + $$SyncOutboxRowsTableAnnotationComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + GeneratedColumn get id => + $composableBuilder(column: $table.id, builder: (column) => column); + + GeneratedColumn get firebaseUid => $composableBuilder( + column: $table.firebaseUid, + builder: (column) => column, + ); + + GeneratedColumn get payloadJson => $composableBuilder( + column: $table.payloadJson, + builder: (column) => column, + ); + + GeneratedColumn get createdAt => + $composableBuilder(column: $table.createdAt, builder: (column) => column); +} + +class $$SyncOutboxRowsTableTableManager + extends + RootTableManager< + _$AppDatabase, + $SyncOutboxRowsTable, + SyncOutboxRow, + $$SyncOutboxRowsTableFilterComposer, + $$SyncOutboxRowsTableOrderingComposer, + $$SyncOutboxRowsTableAnnotationComposer, + $$SyncOutboxRowsTableCreateCompanionBuilder, + $$SyncOutboxRowsTableUpdateCompanionBuilder, + ( + SyncOutboxRow, + BaseReferences<_$AppDatabase, $SyncOutboxRowsTable, SyncOutboxRow>, + ), + SyncOutboxRow, + PrefetchHooks Function() + > { + $$SyncOutboxRowsTableTableManager( + _$AppDatabase db, + $SyncOutboxRowsTable table, + ) : super( + TableManagerState( + db: db, + table: table, + createFilteringComposer: () => + $$SyncOutboxRowsTableFilterComposer($db: db, $table: table), + createOrderingComposer: () => + $$SyncOutboxRowsTableOrderingComposer($db: db, $table: table), + createComputedFieldComposer: () => + $$SyncOutboxRowsTableAnnotationComposer($db: db, $table: table), + updateCompanionCallback: + ({ + Value id = const Value.absent(), + Value firebaseUid = const Value.absent(), + Value payloadJson = const Value.absent(), + Value createdAt = const Value.absent(), + }) => SyncOutboxRowsCompanion( + id: id, + firebaseUid: firebaseUid, + payloadJson: payloadJson, + createdAt: createdAt, + ), + createCompanionCallback: + ({ + Value id = const Value.absent(), + required String firebaseUid, + required String payloadJson, + required DateTime createdAt, + }) => SyncOutboxRowsCompanion.insert( + id: id, + firebaseUid: firebaseUid, + payloadJson: payloadJson, + createdAt: createdAt, + ), + withReferenceMapper: (p0) => p0 + .map((e) => (e.readTable(table), BaseReferences(db, table, e))) + .toList(), + prefetchHooksCallback: null, + ), + ); +} + +typedef $$SyncOutboxRowsTableProcessedTableManager = + ProcessedTableManager< + _$AppDatabase, + $SyncOutboxRowsTable, + SyncOutboxRow, + $$SyncOutboxRowsTableFilterComposer, + $$SyncOutboxRowsTableOrderingComposer, + $$SyncOutboxRowsTableAnnotationComposer, + $$SyncOutboxRowsTableCreateCompanionBuilder, + $$SyncOutboxRowsTableUpdateCompanionBuilder, + ( + SyncOutboxRow, + BaseReferences<_$AppDatabase, $SyncOutboxRowsTable, SyncOutboxRow>, + ), + SyncOutboxRow, + PrefetchHooks Function() + >; + +class $AppDatabaseManager { + final _$AppDatabase _db; + $AppDatabaseManager(this._db); + $$UserProfileRowsTableTableManager get userProfileRows => + $$UserProfileRowsTableTableManager(_db, _db.userProfileRows); + $$SyncOutboxRowsTableTableManager get syncOutboxRows => + $$SyncOutboxRowsTableTableManager(_db, _db.syncOutboxRows); +} diff --git a/lib/data/local/tables.dart b/lib/data/local/tables.dart new file mode 100644 index 0000000..3554f5e --- /dev/null +++ b/lib/data/local/tables.dart @@ -0,0 +1,26 @@ +import 'package:drift/drift.dart'; + +class UserProfileRows extends Table { + TextColumn get firebaseUid => text()(); + TextColumn get email => text().nullable()(); + TextColumn get displayName => text().nullable()(); + TextColumn get photoUrl => text().nullable()(); + TextColumn get locale => text().withDefault(const Constant('en'))(); + TextColumn get timezone => text().nullable()(); + BoolColumn get onboardingCompleted => + boolean().withDefault(const Constant(false))(); + IntColumn get revision => integer().withDefault(const Constant(1))(); + DateTimeColumn get updatedAt => dateTime()(); + DateTimeColumn get lastSyncedAt => dateTime().nullable()(); + BoolColumn get dirty => boolean().withDefault(const Constant(false))(); + + @override + Set> get primaryKey => >{firebaseUid}; +} + +class SyncOutboxRows extends Table { + IntColumn get id => integer().autoIncrement()(); + TextColumn get firebaseUid => text()(); + TextColumn get payloadJson => text()(); + DateTimeColumn get createdAt => dateTime()(); +} diff --git a/lib/data/local/user_profile_local_store.dart b/lib/data/local/user_profile_local_store.dart new file mode 100644 index 0000000..d83892e --- /dev/null +++ b/lib/data/local/user_profile_local_store.dart @@ -0,0 +1,70 @@ +import '../../models/user_profile.dart'; +import 'app_database.dart'; +import 'user_profile_mapper.dart'; + +/// Drift-backed local profile storage (mobile and desktop native). +class UserProfileLocalStore { + UserProfileLocalStore(this._db); + + final AppDatabase _db; + + Stream watchProfile(String firebaseUid) { + return _db.watchProfile(firebaseUid).map((UserProfileRow? row) { + if (row == null) { + return null; + } + return profileFromRow(row); + }); + } + + Future getProfile(String firebaseUid) async { + final UserProfileRow? row = await _db.getProfile(firebaseUid); + if (row == null) { + return null; + } + return profileFromRow(row); + } + + Future saveProfile(UserProfile profile, {required bool dirty}) async { + await _db.upsertProfile( + companionFromProfile(profile.copyWith(dirty: dirty)), + ); + if (dirty) { + await _db.enqueueOutbox( + SyncOutboxRowsCompanion.insert( + firebaseUid: profile.firebaseUid, + payloadJson: encodeProfilePayload(profile), + createdAt: DateTime.now().toUtc(), + ), + ); + } + } + + Future> pendingOutboxProfiles(String firebaseUid) async { + final List rows = await _db.pendingOutbox(firebaseUid); + return rows + .map((SyncOutboxRow row) => decodeProfilePayload(row.payloadJson)) + .toList(); + } + + Future clearOutbox(String firebaseUid) async { + await _db.clearOutbox(firebaseUid); + } + + Future deleteOutboxEntries(String firebaseUid) async { + final List rows = await _db.pendingOutbox(firebaseUid); + for (final SyncOutboxRow row in rows) { + await _db.deleteOutboxEntry(row.id); + } + } + + Future clearLocal(String firebaseUid) async { + await _db.clearOutbox(firebaseUid); + await _db.deleteProfile(firebaseUid); + } + + Future clearAll() async { + await _db.clearAllProfiles(); + await _db.clearAllOutbox(); + } +} diff --git a/lib/data/local/user_profile_mapper.dart b/lib/data/local/user_profile_mapper.dart new file mode 100644 index 0000000..443ad8f --- /dev/null +++ b/lib/data/local/user_profile_mapper.dart @@ -0,0 +1,48 @@ +import 'dart:convert'; + +import 'package:drift/drift.dart'; + +import '../../models/user_profile.dart'; +import 'app_database.dart'; + +UserProfile profileFromRow(UserProfileRow row) { + return UserProfile( + firebaseUid: row.firebaseUid, + email: row.email, + displayName: row.displayName, + photoUrl: row.photoUrl, + locale: row.locale, + timezone: row.timezone, + onboardingCompleted: row.onboardingCompleted, + revision: row.revision, + updatedAt: row.updatedAt.toUtc(), + lastSyncedAt: row.lastSyncedAt?.toUtc(), + dirty: row.dirty, + ); +} + +UserProfileRowsCompanion companionFromProfile(UserProfile profile) { + return UserProfileRowsCompanion( + firebaseUid: Value(profile.firebaseUid), + email: Value(profile.email), + displayName: Value(profile.displayName), + photoUrl: Value(profile.photoUrl), + locale: Value(profile.locale), + timezone: Value(profile.timezone), + onboardingCompleted: Value(profile.onboardingCompleted), + revision: Value(profile.revision), + updatedAt: Value(profile.updatedAt.toUtc()), + lastSyncedAt: Value(profile.lastSyncedAt?.toUtc()), + dirty: Value(profile.dirty), + ); +} + +String encodeProfilePayload(UserProfile profile) { + return jsonEncode(profile.toJson()); +} + +UserProfile decodeProfilePayload(String json) { + return UserProfile.fromJson( + jsonDecode(json) as Map, + ); +} diff --git a/lib/data/remote/profile_api_exception.dart b/lib/data/remote/profile_api_exception.dart new file mode 100644 index 0000000..2be2671 --- /dev/null +++ b/lib/data/remote/profile_api_exception.dart @@ -0,0 +1,19 @@ +import '../../models/user_profile.dart'; + +class ProfileNotFoundException implements Exception {} + +class ProfileConflictException implements Exception { + ProfileConflictException(this.serverProfile); + + final UserProfile serverProfile; +} + +class ProfileApiException implements Exception { + ProfileApiException(this.statusCode, this.body); + + final int statusCode; + final String body; + + @override + String toString() => 'ProfileApiException($statusCode): $body'; +} diff --git a/lib/data/remote/user_profile_remote_store.dart b/lib/data/remote/user_profile_remote_store.dart new file mode 100644 index 0000000..5ffd1ad --- /dev/null +++ b/lib/data/remote/user_profile_remote_store.dart @@ -0,0 +1,90 @@ +import 'dart:convert'; + +import 'package:http/http.dart' as http; + +import '../../config/api_config.dart'; +import '../../models/user_profile.dart'; +import '../../services/auth_service.dart'; +import 'profile_api_exception.dart'; + +/// HTTP client for Postgres-backed profile API. +class UserProfileRemoteStore { + UserProfileRemoteStore({http.Client? client}) : _client = client ?? http.Client(); + + final http.Client _client; + + Future fetchProfile() async { + final String? token = await AuthService.instance.getIdToken(); + if (token == null) { + return null; + } + + final http.Response response = await _client.get( + Uri.parse('$apiBaseUrl/v1/me/profile'), + headers: _authHeaders(token), + ); + + if (response.statusCode == 404) { + return null; + } + if (response.statusCode != 200) { + throw ProfileApiException(response.statusCode, response.body); + } + + return _profileFromApi( + jsonDecode(response.body) as Map, + ); + } + + Future pushProfile(UserProfile profile) async { + final String? token = await AuthService.instance.getIdToken(); + if (token == null) { + throw StateError('Not signed in'); + } + + final http.Response response = await _client.put( + Uri.parse('$apiBaseUrl/v1/me/profile'), + headers: _authHeaders(token), + body: jsonEncode(profile.toApiJson()), + ); + + if (response.statusCode == 409) { + final Map body = + jsonDecode(response.body) as Map; + throw ProfileConflictException( + _profileFromApi(body['profile'] as Map), + ); + } + if (response.statusCode != 200) { + throw ProfileApiException(response.statusCode, response.body); + } + + return _profileFromApi( + jsonDecode(response.body) as Map, + ); + } + + Map _authHeaders(String token) { + return { + 'Authorization': 'Bearer $token', + 'Content-Type': 'application/json', + 'Accept': 'application/json', + }; + } + + UserProfile _profileFromApi(Map json) { + return UserProfile( + firebaseUid: json['firebaseUid'] as String, + email: json['email'] as String?, + displayName: json['displayName'] as String?, + photoUrl: json['photoUrl'] as String?, + locale: json['locale'] as String? ?? 'en', + timezone: json['timezone'] as String?, + onboardingCompleted: json['onboardingCompleted'] as bool? ?? false, + revision: (json['revision'] as num).toInt(), + updatedAt: DateTime.parse(json['updatedAt'] as String).toUtc(), + lastSyncedAt: DateTime.now().toUtc(), + dirty: false, + ); + } +} diff --git a/lib/data/sync/connectivity_service.dart b/lib/data/sync/connectivity_service.dart new file mode 100644 index 0000000..f4ad9c6 --- /dev/null +++ b/lib/data/sync/connectivity_service.dart @@ -0,0 +1,63 @@ +import 'dart:async'; + +import 'package:connectivity_plus/connectivity_plus.dart'; +import 'package:flutter/foundation.dart' show kIsWeb; + +/// Observes network reachability and exposes online/offline transitions. +class ConnectivityService { + ConnectivityService({Connectivity? connectivity}) + : _connectivity = connectivity ?? Connectivity(); + + final Connectivity _connectivity; + final StreamController _onlineController = + StreamController.broadcast(); + + StreamSubscription>? _subscription; + bool _isOnline = true; + + Stream get onlineChanges => _onlineController.stream; + + bool get isOnline => _isOnline; + + Future initialize() async { + if (kIsWeb) { + _isOnline = true; + _onlineController.add(true); + return; + } + + final List results = await _connectivity + .checkConnectivity(); + _emitOnline(_resultsOnline(results)); + + _subscription = _connectivity.onConnectivityChanged.listen( + (List results) => _emitOnline(_resultsOnline(results)), + ); + } + + void dispose() { + _subscription?.cancel(); + _onlineController.close(); + } + + bool _resultsOnline(List results) { + if (results.isEmpty) { + return false; + } + return results.any( + (ConnectivityResult r) => + r == ConnectivityResult.mobile || + r == ConnectivityResult.wifi || + r == ConnectivityResult.ethernet || + r == ConnectivityResult.vpn, + ); + } + + void _emitOnline(bool online) { + if (_isOnline == online) { + return; + } + _isOnline = online; + _onlineController.add(online); + } +} diff --git a/lib/data/sync/profile_sync_coordinator.dart b/lib/data/sync/profile_sync_coordinator.dart new file mode 100644 index 0000000..be3d071 --- /dev/null +++ b/lib/data/sync/profile_sync_coordinator.dart @@ -0,0 +1,115 @@ +import 'package:flutter/foundation.dart' show kIsWeb; + +import '../../models/sync_result.dart'; +import '../../models/user_profile.dart'; +import '../local/user_profile_local_store.dart'; +import '../remote/profile_api_exception.dart'; +import '../remote/user_profile_remote_store.dart'; +import 'connectivity_service.dart'; + +/// Pushes local changes and pulls server state when online. +class ProfileSyncCoordinator { + ProfileSyncCoordinator({ + required UserProfileLocalStore? local, + required UserProfileRemoteStore remote, + required ConnectivityService connectivity, + }) : _local = local, + _remote = remote, + _connectivity = connectivity; + + final UserProfileLocalStore? _local; + final UserProfileRemoteStore _remote; + final ConnectivityService _connectivity; + + bool get hasLocalStore => _local != null; + + Future sync(String firebaseUid) async { + if (!kIsWeb && !_connectivity.isOnline) { + return const SyncResult.offline(); + } + + try { + if (_local == null) { + return const SyncResult.success(); + } + return _syncWithLocal(firebaseUid); + } on ProfileApiException catch (e) { + return SyncResult.error(e.toString()); + } catch (e) { + return SyncResult.error(e.toString()); + } + } + + Future _syncWithLocal(String firebaseUid) async { + final UserProfileLocalStore local = _local!; + + final List pending = await local.pendingOutboxProfiles( + firebaseUid, + ); + for (final UserProfile profile in pending) { + try { + final UserProfile saved = await _remote.pushProfile(profile); + await local.saveProfile( + saved.copyWith(dirty: false, lastSyncedAt: DateTime.now().toUtc()), + dirty: false, + ); + } on ProfileConflictException catch (e) { + await local.saveProfile( + e.serverProfile.copyWith( + dirty: false, + lastSyncedAt: DateTime.now().toUtc(), + ), + dirty: false, + ); + await local.clearOutbox(firebaseUid); + return const SyncResult.conflictResolved(); + } + } + if (pending.isNotEmpty) { + await local.clearOutbox(firebaseUid); + } + + final UserProfile? server = await _remote.fetchProfile(); + if (server == null) { + final UserProfile? localProfile = await local.getProfile(firebaseUid); + if (localProfile != null && localProfile.dirty) { + final UserProfile saved = await _remote.pushProfile(localProfile); + await local.saveProfile( + saved.copyWith(dirty: false, lastSyncedAt: DateTime.now().toUtc()), + dirty: false, + ); + } + return const SyncResult.success(); + } + + final UserProfile? localProfile = await local.getProfile(firebaseUid); + if (localProfile == null || + server.revision > localProfile.revision || + (server.revision == localProfile.revision && + server.updatedAt.isAfter(localProfile.updatedAt))) { + await local.saveProfile( + server.copyWith(dirty: false, lastSyncedAt: DateTime.now().toUtc()), + dirty: false, + ); + } else if (localProfile.dirty) { + try { + final UserProfile saved = await _remote.pushProfile(localProfile); + await local.saveProfile( + saved.copyWith(dirty: false, lastSyncedAt: DateTime.now().toUtc()), + dirty: false, + ); + } on ProfileConflictException catch (e) { + await local.saveProfile( + e.serverProfile.copyWith( + dirty: false, + lastSyncedAt: DateTime.now().toUtc(), + ), + dirty: false, + ); + return const SyncResult.conflictResolved(); + } + } + + return const SyncResult.success(); + } +} diff --git a/lib/firebase_options.dart b/lib/firebase_options.dart new file mode 100644 index 0000000..ddeef7f --- /dev/null +++ b/lib/firebase_options.dart @@ -0,0 +1,65 @@ +// File generated by FlutterFire CLI. +// ignore_for_file: type=lint +import 'package:firebase_core/firebase_core.dart' show FirebaseOptions; +import 'package:flutter/foundation.dart' + show defaultTargetPlatform, kIsWeb, TargetPlatform; + +/// Placeholder Firebase options. +/// +/// Replace this file by running: +/// dart pub global activate flutterfire_cli +/// flutterfire configure +class DefaultFirebaseOptions { + static FirebaseOptions get currentPlatform { + if (kIsWeb) { + return web; + } + switch (defaultTargetPlatform) { + case TargetPlatform.android: + return android; + case TargetPlatform.iOS: + return ios; + case TargetPlatform.macOS: + return macos; + case TargetPlatform.linux: + return linux; + case TargetPlatform.windows: + return web; + default: + throw UnsupportedError( + 'DefaultFirebaseOptions are not configured for this platform.', + ); + } + } + + /// Used for Linux desktop dev (Firebase REST + browser OAuth). + static const FirebaseOptions linux = FirebaseOptions( + apiKey: 'AIzaSyDbYBh6nO-UasqwM-dkVVJAvw5wMhRTsIs', + appId: '1:155018389717:web:afb61503456673019ada5c', + messagingSenderId: '155018389717', + projectId: 'cyberhybridhub', + authDomain: 'cyberhybridhub.firebaseapp.com', + storageBucket: 'cyberhybridhub.firebasestorage.app', + ); + + static const FirebaseOptions web = linux; + + static const FirebaseOptions android = FirebaseOptions( + apiKey: 'AIzaSyA71SkNLrv3tyJQW2DCfe2vFcu1c2opDX8', + appId: '1:155018389717:android:90887875660a26509ada5c', + messagingSenderId: '155018389717', + projectId: 'cyberhybridhub', + storageBucket: 'cyberhybridhub.firebasestorage.app', + ); + + static const FirebaseOptions ios = FirebaseOptions( + apiKey: 'AIzaSyAHqjmp-vC8n_2EClQM2ch33jL_yIGuQLE', + appId: '1:155018389717:ios:102445b8ec5fde749ada5c', + messagingSenderId: '155018389717', + projectId: 'cyberhybridhub', + storageBucket: 'cyberhybridhub.firebasestorage.app', + iosBundleId: 'com.cyberhybridhub.cyberhybridhub', + ); + + static const FirebaseOptions macos = ios; +} \ No newline at end of file diff --git a/lib/main.dart b/lib/main.dart new file mode 100644 index 0000000..acb6d14 --- /dev/null +++ b/lib/main.dart @@ -0,0 +1,62 @@ +import 'package:flutter/material.dart'; + +import 'bootstrap.dart'; +import 'models/app_user.dart'; +import 'screens/home_screen.dart'; +import 'screens/landing_screen.dart'; +import 'services/auth_service.dart'; +import 'theme/app_theme.dart'; +import 'widgets/profile_session.dart'; + +Future main() async { + WidgetsFlutterBinding.ensureInitialized(); + await bootstrap(); + runApp(const CyberHybridHubApp()); +} + +class CyberHybridHubApp extends StatelessWidget { + const CyberHybridHubApp({super.key}); + + @override + Widget build(BuildContext context) { + return MaterialApp( + title: 'Cyber Hybrid Hub', + theme: buildAppTheme(), + home: const AuthGate(), + ); + } +} + +class AuthGate extends StatelessWidget { + const AuthGate({super.key}); + + @override + Widget build(BuildContext context) { + return StreamBuilder( + stream: AuthService.instance.authStateChanges, + builder: (BuildContext context, AsyncSnapshot snapshot) { + if (snapshot.connectionState == ConnectionState.waiting) { + return const Scaffold( + body: Center( + child: CircularProgressIndicator(color: AppColors.accent), + ), + ); + } + + final AppUser? user = snapshot.data; + if (user == null) { + return const LandingScreen(); + } + + return ProfileSession( + user: user, + builder: (context, profile, syncStatus) => HomeScreen( + user: user, + profile: profile, + syncStatus: syncStatus, + ), + ); + }, + ); + } +} diff --git a/lib/models/app_user.dart b/lib/models/app_user.dart new file mode 100644 index 0000000..7968eb2 --- /dev/null +++ b/lib/models/app_user.dart @@ -0,0 +1,14 @@ +/// App-level user model used across mobile, web, and Linux desktop. +class AppUser { + const AppUser({ + required this.uid, + this.email, + this.displayName, + this.photoUrl, + }); + + final String uid; + final String? email; + final String? displayName; + final String? photoUrl; +} diff --git a/lib/models/sync_result.dart b/lib/models/sync_result.dart new file mode 100644 index 0000000..6ca81e0 --- /dev/null +++ b/lib/models/sync_result.dart @@ -0,0 +1,26 @@ +/// Result of a profile sync attempt. +enum SyncResultKind { success, offline, conflictResolved, error } + +class SyncResult { + const SyncResult._(this.kind, {this.message}); + + const SyncResult.success() : this._(SyncResultKind.success); + + const SyncResult.offline() : this._(SyncResultKind.offline); + + const SyncResult.conflictResolved() + : this._(SyncResultKind.conflictResolved); + + const SyncResult.error(String message) + : this._(SyncResultKind.error, message: message); + + final SyncResultKind kind; + final String? message; + + bool get isSuccess => + kind == SyncResultKind.success || + kind == SyncResultKind.conflictResolved; +} + +/// UI-facing sync state from [UserProfileRepository]. +enum ProfileSyncStatus { idle, syncing, synced, offline, error } diff --git a/lib/models/user_profile.dart b/lib/models/user_profile.dart new file mode 100644 index 0000000..7d2e9cd --- /dev/null +++ b/lib/models/user_profile.dart @@ -0,0 +1,123 @@ +/// App-owned user profile synced between local Drift storage and Postgres. +class UserProfile { + const UserProfile({ + required this.firebaseUid, + this.email, + this.displayName, + this.photoUrl, + this.locale = 'en', + this.timezone, + this.onboardingCompleted = false, + this.revision = 1, + required this.updatedAt, + this.lastSyncedAt, + this.dirty = false, + }); + + final String firebaseUid; + final String? email; + final String? displayName; + final String? photoUrl; + final String locale; + final String? timezone; + final bool onboardingCompleted; + final int revision; + final DateTime updatedAt; + final DateTime? lastSyncedAt; + final bool dirty; + + UserProfile copyWith({ + String? email, + String? displayName, + String? photoUrl, + String? locale, + String? timezone, + bool? onboardingCompleted, + int? revision, + DateTime? updatedAt, + DateTime? lastSyncedAt, + bool? dirty, + bool clearLastSyncedAt = false, + }) { + return UserProfile( + firebaseUid: firebaseUid, + email: email ?? this.email, + displayName: displayName ?? this.displayName, + photoUrl: photoUrl ?? this.photoUrl, + locale: locale ?? this.locale, + timezone: timezone ?? this.timezone, + onboardingCompleted: onboardingCompleted ?? this.onboardingCompleted, + revision: revision ?? this.revision, + updatedAt: updatedAt ?? this.updatedAt, + lastSyncedAt: clearLastSyncedAt + ? null + : (lastSyncedAt ?? this.lastSyncedAt), + dirty: dirty ?? this.dirty, + ); + } + + factory UserProfile.fromAuth({ + required String firebaseUid, + String? email, + String? displayName, + String? photoUrl, + }) { + return UserProfile( + firebaseUid: firebaseUid, + email: email, + displayName: displayName, + photoUrl: photoUrl, + updatedAt: DateTime.now().toUtc(), + dirty: true, + ); + } + + factory UserProfile.fromJson(Map json) { + return UserProfile( + firebaseUid: json['firebaseUid'] as String, + email: json['email'] as String?, + displayName: json['displayName'] as String?, + photoUrl: json['photoUrl'] as String?, + locale: json['locale'] as String? ?? 'en', + timezone: json['timezone'] as String?, + onboardingCompleted: json['onboardingCompleted'] as bool? ?? false, + revision: (json['revision'] as num?)?.toInt() ?? 1, + updatedAt: DateTime.parse(json['updatedAt'] as String).toUtc(), + lastSyncedAt: json['lastSyncedAt'] == null + ? null + : DateTime.parse(json['lastSyncedAt'] as String).toUtc(), + dirty: json['dirty'] as bool? ?? false, + ); + } + + Map toJson() { + return { + 'firebaseUid': firebaseUid, + 'email': email, + 'displayName': displayName, + 'photoUrl': photoUrl, + 'locale': locale, + 'timezone': timezone, + 'onboardingCompleted': onboardingCompleted, + 'revision': revision, + 'updatedAt': updatedAt.toUtc().toIso8601String(), + if (lastSyncedAt != null) + 'lastSyncedAt': lastSyncedAt!.toUtc().toIso8601String(), + 'dirty': dirty, + }; + } + + /// Payload sent to the API (server assigns authoritative revision on conflict). + Map toApiJson() { + return { + 'email': email, + 'displayName': displayName, + 'photoUrl': photoUrl, + 'locale': locale, + 'timezone': timezone, + 'onboardingCompleted': onboardingCompleted, + 'revision': revision, + 'updatedAt': updatedAt.toUtc().toIso8601String(), + }; + } +} diff --git a/lib/repositories/user_profile_repository.dart b/lib/repositories/user_profile_repository.dart new file mode 100644 index 0000000..925234b --- /dev/null +++ b/lib/repositories/user_profile_repository.dart @@ -0,0 +1,248 @@ +import 'dart:async'; + +import 'package:flutter/foundation.dart' show kIsWeb; + +import '../data/local/app_database.dart'; +import '../data/local/user_profile_local_store.dart'; +import '../data/remote/profile_api_exception.dart'; +import '../data/remote/user_profile_remote_store.dart'; +import '../data/sync/connectivity_service.dart'; +import '../data/sync/profile_sync_coordinator.dart'; +import '../models/app_user.dart'; +import '../models/sync_result.dart'; +import '../models/user_profile.dart'; + +/// Facade for profile reads/writes with Drift (native) and API sync. +class UserProfileRepository { + UserProfileRepository._(); + + static final UserProfileRepository instance = UserProfileRepository._(); + + final ConnectivityService _connectivity = ConnectivityService(); + final UserProfileRemoteStore _remote = UserProfileRemoteStore(); + + AppDatabase? _database; + UserProfileLocalStore? _local; + ProfileSyncCoordinator? _sync; + StreamSubscription? _connectivitySubscription; + + final StreamController _profileController = + StreamController.broadcast(); + final StreamController _syncStatusController = + StreamController.broadcast(); + + StreamSubscription? _localWatchSubscription; + String? _activeUid; + UserProfile? _cachedProfile; + + Stream get profileStream => _profileController.stream; + + Stream get syncStatusStream => + _syncStatusController.stream; + + UserProfile? get currentProfile => _cachedProfile; + + bool get usesLocalStore => _local != null; + + Future initialize() async { + await _connectivity.initialize(); + + if (!kIsWeb) { + _database = AppDatabase(); + _local = UserProfileLocalStore(_database!); + _sync = ProfileSyncCoordinator( + local: _local, + remote: _remote, + connectivity: _connectivity, + ); + } else { + _sync = ProfileSyncCoordinator( + local: null, + remote: _remote, + connectivity: _connectivity, + ); + } + + _connectivitySubscription = _connectivity.onlineChanges.listen(( + bool online, + ) { + if (online && _activeUid != null) { + unawaited(sync()); + } else if (!online) { + _emitSyncStatus(ProfileSyncStatus.offline); + } + }); + + _emitSyncStatus(ProfileSyncStatus.idle); + } + + Future startSession(AppUser user) async { + if (_activeUid == user.uid) { + return; + } + await endSession(); + + _activeUid = user.uid; + _emitSyncStatus(ProfileSyncStatus.syncing); + + if (_local != null) { + _localWatchSubscription = _local!.watchProfile(user.uid).listen( + (UserProfile? profile) { + _cachedProfile = profile; + _profileController.add(profile); + }, + ); + + UserProfile? profile = await _local!.getProfile(user.uid); + profile ??= UserProfile.fromAuth( + firebaseUid: user.uid, + email: user.email, + displayName: user.displayName, + photoUrl: user.photoUrl, + ); + await _local!.saveProfile(profile, dirty: true); + _cachedProfile = profile; + _profileController.add(profile); + } else { + await _loadRemoteProfile(user); + } + + await sync(); + } + + Future _loadRemoteProfile(AppUser user) async { + try { + UserProfile? profile = await _remote.fetchProfile(); + profile ??= UserProfile.fromAuth( + firebaseUid: user.uid, + email: user.email, + displayName: user.displayName, + photoUrl: user.photoUrl, + ); + if (profile.revision == 1 && !profile.dirty) { + final UserProfile seeded = profile.copyWith(dirty: true); + profile = await _remote.pushProfile(seeded); + } + _cachedProfile = profile; + _profileController.add(profile); + } catch (_) { + final UserProfile fallback = UserProfile.fromAuth( + firebaseUid: user.uid, + email: user.email, + displayName: user.displayName, + photoUrl: user.photoUrl, + ); + _cachedProfile = fallback; + _profileController.add(fallback); + _emitSyncStatus(ProfileSyncStatus.error); + } + } + + Future endSession() async { + await _localWatchSubscription?.cancel(); + _localWatchSubscription = null; + + if (_activeUid != null && _local != null) { + await _local!.clearLocal(_activeUid!); + } + + _activeUid = null; + _cachedProfile = null; + _profileController.add(null); + _emitSyncStatus(ProfileSyncStatus.idle); + } + + Future updateProfile(UserProfile profile) async { + final UserProfile updated = profile.copyWith( + revision: profile.revision + 1, + updatedAt: DateTime.now().toUtc(), + dirty: true, + ); + + if (_local != null) { + await _local!.saveProfile(updated, dirty: true); + } else { + _cachedProfile = updated; + _profileController.add(updated); + await _remote.pushProfile(updated); + } + + unawaited(sync()); + } + + Future sync() async { + final String? uid = _activeUid; + if (uid == null || _sync == null) { + return const SyncResult.error('No active session'); + } + + if (!kIsWeb && !_connectivity.isOnline) { + _emitSyncStatus(ProfileSyncStatus.offline); + return const SyncResult.offline(); + } + + _emitSyncStatus(ProfileSyncStatus.syncing); + final SyncResult result = _local == null + ? await _syncWebOnly() + : await _sync!.sync(uid); + + switch (result.kind) { + case SyncResultKind.success: + case SyncResultKind.conflictResolved: + _emitSyncStatus(ProfileSyncStatus.synced); + case SyncResultKind.offline: + _emitSyncStatus(ProfileSyncStatus.offline); + case SyncResultKind.error: + _emitSyncStatus(ProfileSyncStatus.error); + } + + return result; + } + + Future _syncWebOnly() async { + final UserProfile? profile = _cachedProfile; + if (profile == null) { + return const SyncResult.error('No profile loaded'); + } + + try { + if (profile.dirty) { + try { + final UserProfile saved = await _remote.pushProfile(profile); + _cachedProfile = saved; + _profileController.add(saved); + } on ProfileConflictException catch (e) { + _cachedProfile = e.serverProfile; + _profileController.add(e.serverProfile); + return const SyncResult.conflictResolved(); + } + } else { + final UserProfile? server = await _remote.fetchProfile(); + if (server != null) { + _cachedProfile = server; + _profileController.add(server); + } + } + return const SyncResult.success(); + } on ProfileApiException catch (e) { + return SyncResult.error(e.toString()); + } catch (e) { + return SyncResult.error(e.toString()); + } + } + + Future dispose() async { + await _connectivitySubscription?.cancel(); + _connectivity.dispose(); + await _localWatchSubscription?.cancel(); + await _profileController.close(); + await _syncStatusController.close(); + await _database?.close(); + } + + void _emitSyncStatus(ProfileSyncStatus status) { + if (!_syncStatusController.isClosed) { + _syncStatusController.add(status); + } + } +} diff --git a/lib/screens/home_screen.dart b/lib/screens/home_screen.dart new file mode 100644 index 0000000..94c1a80 --- /dev/null +++ b/lib/screens/home_screen.dart @@ -0,0 +1,199 @@ +import 'package:flutter/material.dart'; + +import '../models/app_user.dart'; +import '../models/sync_result.dart'; +import '../models/user_profile.dart'; +import '../repositories/user_profile_repository.dart'; +import '../services/auth_service.dart'; +import '../theme/app_theme.dart'; + +class HomeScreen extends StatelessWidget { + const HomeScreen({ + super.key, + required this.user, + this.profile, + this.syncStatus = ProfileSyncStatus.idle, + }); + + final AppUser user; + final UserProfile? profile; + final ProfileSyncStatus syncStatus; + + @override + Widget build(BuildContext context) { + final String? photoUrl = profile?.photoUrl ?? user.photoUrl; + final String displayName = + profile?.displayName ?? user.displayName ?? 'there'; + final String? email = profile?.email ?? user.email; + + return Scaffold( + appBar: AppBar( + backgroundColor: Colors.transparent, + title: const Text('Cyber Hybrid Hub'), + actions: [ + IconButton( + onPressed: () => UserProfileRepository.instance.sync(), + tooltip: 'Sync profile', + icon: const Icon(Icons.sync), + ), + IconButton( + onPressed: () => AuthService.instance.signOut(), + tooltip: 'Sign out', + icon: const Icon(Icons.logout), + ), + ], + ), + body: DecoratedBox( + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [AppColors.background, AppColors.surface], + ), + ), + child: SafeArea( + child: Padding( + padding: const EdgeInsets.all(24), + child: Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + Container( + padding: const EdgeInsets.all(24), + decoration: BoxDecoration( + color: AppColors.surfaceElevated, + borderRadius: BorderRadius.circular(16), + border: Border.all( + color: AppColors.accent.withValues(alpha: 0.2), + ), + ), + child: Column( + children: [ + if (photoUrl != null) + CircleAvatar( + radius: 36, + backgroundImage: NetworkImage(photoUrl), + ) + else + const CircleAvatar( + radius: 36, + child: Icon(Icons.person, size: 36), + ), + const SizedBox(height: 16), + Text( + 'Welcome, $displayName', + style: Theme.of(context).textTheme.headlineMedium, + textAlign: TextAlign.center, + ), + if (email != null) ...[ + const SizedBox(height: 8), + Text( + email, + style: Theme.of(context).textTheme.bodyLarge, + textAlign: TextAlign.center, + ), + ], + const SizedBox(height: 12), + _SyncStatusChip(status: syncStatus), + const SizedBox(height: 20), + const Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Icon( + Icons.check_circle, + color: AppColors.success, + size: 20, + ), + SizedBox(width: 8), + Text( + 'You\'re signed in', + style: TextStyle( + color: AppColors.success, + fontWeight: FontWeight.w600, + ), + ), + ], + ), + ], + ), + ), + const SizedBox(height: 24), + Text( + profile?.dirty == true + ? 'Profile saved locally. Will sync when online.' + : 'Profile synced with your account.', + style: Theme.of(context).textTheme.bodyLarge, + textAlign: TextAlign.center, + ), + if (UserProfileRepository.instance.usesLocalStore) ...[ + const SizedBox(height: 16), + OutlinedButton.icon( + onPressed: profile == null + ? null + : () async { + final UserProfile current = profile!; + await UserProfileRepository.instance.updateProfile( + current.copyWith( + onboardingCompleted: + !current.onboardingCompleted, + ), + ); + }, + icon: const Icon(Icons.toggle_on_outlined), + label: Text( + profile?.onboardingCompleted == true + ? 'Mark onboarding incomplete' + : 'Mark onboarding complete', + ), + ), + ], + ], + ), + ), + ), + ), + ); + } +} + +class _SyncStatusChip extends StatelessWidget { + const _SyncStatusChip({required this.status}); + + final ProfileSyncStatus status; + + @override + Widget build(BuildContext context) { + final ({String label, Color color, IconData icon}) style = switch (status) { + ProfileSyncStatus.syncing => ( + label: 'Syncing…', + color: AppColors.accent, + icon: Icons.sync, + ), + ProfileSyncStatus.synced => ( + label: 'Synced', + color: AppColors.success, + icon: Icons.cloud_done_outlined, + ), + ProfileSyncStatus.offline => ( + label: 'Offline', + color: Colors.orange, + icon: Icons.cloud_off_outlined, + ), + ProfileSyncStatus.error => ( + label: 'Sync error', + color: Colors.redAccent, + icon: Icons.error_outline, + ), + ProfileSyncStatus.idle => ( + label: 'Ready', + color: AppColors.accent, + icon: Icons.cloud_queue_outlined, + ), + }; + + return Chip( + avatar: Icon(style.icon, size: 18, color: style.color), + label: Text(style.label), + side: BorderSide(color: style.color.withValues(alpha: 0.4)), + ); + } +} diff --git a/lib/screens/landing_screen.dart b/lib/screens/landing_screen.dart new file mode 100644 index 0000000..d3593a5 --- /dev/null +++ b/lib/screens/landing_screen.dart @@ -0,0 +1,464 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:flutter/foundation.dart' show kIsWeb; +import 'package:flutter/material.dart'; + +import '../theme/app_theme.dart'; +import '../widgets/benefit_row.dart'; +import '../widgets/google_sign_in_button.dart'; + +class LandingScreen extends StatefulWidget { + const LandingScreen({super.key}); + + @override + State createState() => _LandingScreenState(); +} + +class _LandingScreenState extends State { + String? _errorMessage; + + @override + Widget build(BuildContext context) { + return Scaffold( + body: DecoratedBox( + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topLeft, + end: Alignment.bottomRight, + colors: [ + AppColors.background, + Color(0xFF0C1222), + Color(0xFF0A1628), + ], + ), + ), + child: Stack( + children: [ + const _BackgroundGlow(), + SafeArea( + child: LayoutBuilder( + builder: (BuildContext context, BoxConstraints constraints) { + final bool wide = constraints.maxWidth >= 720; + return Center( + child: ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 960), + child: wide + ? _WideLayout( + errorMessage: _errorMessage, + onError: _handleError, + ) + : _NarrowLayout( + errorMessage: _errorMessage, + onError: _handleError, + ), + ), + ); + }, + ), + ), + ], + ), + ), + ); + } + + void _handleError(Object error) { + setState(() { + _errorMessage = _friendlyError(error); + }); + } + + String _friendlyError(Object error) { + if (error is FirebaseAuthException) { + switch (error.code) { + case 'unauthorized-domain': + final String origin = kIsWeb ? Uri.base.origin : 'this app'; + return 'Add $origin under Firebase Console → Authentication → ' + 'Settings → Authorized domains.'; + case 'operation-not-allowed': + return 'Google sign-in is not enabled. Turn on the Google provider ' + 'in Firebase Console → Authentication → Sign-in method.'; + case 'popup-blocked': + return 'The sign-in popup was blocked. Allow popups for this site ' + 'or try again.'; + case 'popup-closed-by-user': + case 'cancelled-popup-request': + return 'Sign-in was cancelled. Tap below to try again.'; + } + } + + final String raw = error.toString(); + if (raw.contains('missing initial state') || + raw.contains('sessionStorage')) { + return 'This browser blocked sign-in storage (common in Firefox). ' + 'Set googleWebOAuthClientId in lib/config/auth_config.dart, allow ' + 'popups for this site, or try Chrome.'; + } + if (raw.contains('canceled') || raw.contains('cancelled')) { + return 'Sign-in was cancelled. Tap below to try again.'; + } + if (raw.contains('network')) { + return 'Network error. Check your connection and try again.'; + } + if (raw.contains('googleWebOAuthClientId')) { + return raw.replaceFirst('Bad state: ', ''); + } + return 'Could not sign in. Please try again.'; + } +} + +class _BackgroundGlow extends StatelessWidget { + const _BackgroundGlow(); + + @override + Widget build(BuildContext context) { + return IgnorePointer( + child: Stack( + children: [ + Positioned( + top: -120, + right: -80, + child: Container( + width: 320, + height: 320, + decoration: BoxDecoration( + shape: BoxShape.circle, + gradient: RadialGradient( + colors: [ + AppColors.accent.withValues(alpha: 0.18), + Colors.transparent, + ], + ), + ), + ), + ), + Positioned( + bottom: -100, + left: -60, + child: Container( + width: 280, + height: 280, + decoration: BoxDecoration( + shape: BoxShape.circle, + gradient: RadialGradient( + colors: [ + AppColors.accentMuted.withValues(alpha: 0.12), + Colors.transparent, + ], + ), + ), + ), + ), + ], + ), + ); + } +} + +class _NarrowLayout extends StatelessWidget { + const _NarrowLayout({required this.errorMessage, required this.onError}); + + final String? errorMessage; + final void Function(Object error) onError; + + @override + Widget build(BuildContext context) { + return SingleChildScrollView( + padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 16), + child: Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + const _BrandHeader(), + const SizedBox(height: 32), + const _HeroCopy(), + const SizedBox(height: 28), + const _BenefitsList(), + const SizedBox(height: 32), + _CtaSection(errorMessage: errorMessage, onError: onError), + const SizedBox(height: 24), + const _TrustFooter(), + ], + ), + ); + } +} + +class _WideLayout extends StatelessWidget { + const _WideLayout({required this.errorMessage, required this.onError}); + + final String? errorMessage; + final void Function(Object error) onError; + + @override + Widget build(BuildContext context) { + return SingleChildScrollView( + padding: const EdgeInsets.symmetric(horizontal: 40, vertical: 24), + child: Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + const Expanded( + flex: 5, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + _BrandHeader(), + SizedBox(height: 40), + _HeroCopy(), + SizedBox(height: 36), + _BenefitsList(), + ], + ), + ), + const SizedBox(width: 48), + Expanded( + flex: 4, + child: _CtaCard(errorMessage: errorMessage, onError: onError), + ), + ], + ), + ); + } +} + +class _BrandHeader extends StatelessWidget { + const _BrandHeader(); + + @override + Widget build(BuildContext context) { + return Row( + children: [ + Container( + width: 40, + height: 40, + decoration: BoxDecoration( + gradient: const LinearGradient( + colors: [AppColors.accent, AppColors.accentMuted], + ), + borderRadius: BorderRadius.circular(10), + ), + child: const Icon( + Icons.shield_outlined, + color: AppColors.background, + size: 22, + ), + ), + const SizedBox(width: 12), + Text( + 'Cyber Hybrid Hub', + style: Theme.of(context).textTheme.titleLarge?.copyWith( + letterSpacing: -0.2, + ), + ), + ], + ); + } +} + +class _HeroCopy extends StatelessWidget { + const _HeroCopy(); + + @override + Widget build(BuildContext context) { + return Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Container( + padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 6), + decoration: BoxDecoration( + color: AppColors.success.withValues(alpha: 0.12), + borderRadius: BorderRadius.circular(20), + border: Border.all( + color: AppColors.success.withValues(alpha: 0.35), + ), + ), + child: const Row( + mainAxisSize: MainAxisSize.min, + children: [ + Icon(Icons.bolt, size: 16, color: AppColors.success), + SizedBox(width: 6), + Text( + 'Free to get started', + style: TextStyle( + fontSize: 13, + fontWeight: FontWeight.w600, + color: AppColors.success, + ), + ), + ], + ), + ), + const SizedBox(height: 20), + Text( + 'Your hybrid security\ncommand center', + style: Theme.of(context).textTheme.headlineLarge, + ), + const SizedBox(height: 16), + Text( + 'Unify threat monitoring, team collaboration, and compliance ' + 'workflows in one secure workspace — built for modern hybrid teams.', + style: Theme.of(context).textTheme.bodyLarge?.copyWith( + fontSize: 17, + ), + ), + ], + ); + } +} + +class _BenefitsList extends StatelessWidget { + const _BenefitsList(); + + static const List<({IconData icon, String title, String subtitle})> _items = + <({IconData icon, String title, String subtitle})>[ + ( + icon: Icons.radar, + title: 'Real-time threat visibility', + subtitle: 'See alerts and incidents as they happen, not hours later.', + ), + ( + icon: Icons.groups_outlined, + title: 'Built for hybrid teams', + subtitle: 'On-site and remote staff share one source of truth.', + ), + ( + icon: Icons.verified_user_outlined, + title: 'Sign in securely with Google', + subtitle: 'No new passwords — enterprise-ready OAuth in one tap.', + ), + ]; + + @override + Widget build(BuildContext context) { + return Column( + children: [ + for (int i = 0; i < _items.length; i++) ...[ + if (i > 0) const SizedBox(height: 20), + BenefitRow( + icon: _items[i].icon, + title: _items[i].title, + subtitle: _items[i].subtitle, + ), + ], + ], + ); + } +} + +class _CtaSection extends StatelessWidget { + const _CtaSection({required this.errorMessage, required this.onError}); + + final String? errorMessage; + final void Function(Object error) onError; + + @override + Widget build(BuildContext context) { + return _CtaCard(errorMessage: errorMessage, onError: onError); + } +} + +class _CtaCard extends StatelessWidget { + const _CtaCard({required this.errorMessage, required this.onError}); + + final String? errorMessage; + final void Function(Object error) onError; + + @override + Widget build(BuildContext context) { + return Container( + padding: const EdgeInsets.all(24), + decoration: BoxDecoration( + color: AppColors.surfaceElevated, + borderRadius: BorderRadius.circular(16), + border: Border.all(color: Colors.white.withValues(alpha: 0.08)), + boxShadow: [ + BoxShadow( + color: AppColors.accent.withValues(alpha: 0.08), + blurRadius: 40, + offset: const Offset(0, 12), + ), + ], + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + Text( + 'Get started in seconds', + style: Theme.of(context).textTheme.headlineMedium?.copyWith( + fontSize: 22, + ), + textAlign: TextAlign.center, + ), + const SizedBox(height: 8), + Text( + 'Use your Google account — no credit card required.', + style: Theme.of(context).textTheme.bodyLarge, + textAlign: TextAlign.center, + ), + const SizedBox(height: 24), + if (errorMessage != null) ...[ + Container( + padding: const EdgeInsets.all(12), + decoration: BoxDecoration( + color: Theme.of(context).colorScheme.error.withValues( + alpha: 0.12, + ), + borderRadius: BorderRadius.circular(8), + ), + child: Text( + errorMessage!, + style: TextStyle( + color: Theme.of(context).colorScheme.error, + fontSize: 14, + ), + textAlign: TextAlign.center, + ), + ), + const SizedBox(height: 16), + ], + GoogleSignInButton( + label: 'Get started with Google', + onError: onError, + ), + const SizedBox(height: 16), + const _SocialProof(), + ], + ), + ); + } +} + +class _SocialProof extends StatelessWidget { + const _SocialProof(); + + @override + Widget build(BuildContext context) { + return Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + const Icon(Icons.lock_outline, size: 14, color: AppColors.textSecondary), + const SizedBox(width: 6), + Flexible( + child: Text( + 'Secured by Firebase · Google OAuth', + style: Theme.of(context).textTheme.bodyLarge?.copyWith(fontSize: 12), + textAlign: TextAlign.center, + overflow: TextOverflow.ellipsis, + maxLines: 2, + ), + ), + ], + ); + } +} + +class _TrustFooter extends StatelessWidget { + const _TrustFooter(); + + @override + Widget build(BuildContext context) { + return Text( + 'By continuing, you agree to our Terms of Service and Privacy Policy.', + style: Theme.of(context).textTheme.bodyLarge?.copyWith(fontSize: 12), + textAlign: TextAlign.center, + ); + } +} diff --git a/lib/services/auth_service.dart b/lib/services/auth_service.dart new file mode 100644 index 0000000..1363594 --- /dev/null +++ b/lib/services/auth_service.dart @@ -0,0 +1,68 @@ +import 'package:flutter/foundation.dart' + show TargetPlatform, defaultTargetPlatform, kIsWeb; + +import '../models/app_user.dart'; +import 'auth_service_firebase.dart'; +import 'auth_service_linux.dart'; + +/// Unified auth API for mobile, web, and Linux desktop development. +class AuthService { + AuthService._(); + + static final AuthService instance = AuthService._(); + + /// True when running the Linux desktop app (not unit tests on a Linux host). + static bool get isLinuxDesktop { + if (kIsWeb) { + return false; + } + if (const bool.fromEnvironment('FLUTTER_TEST')) { + return false; + } + return defaultTargetPlatform == TargetPlatform.linux; + } + + Stream get authStateChanges => isLinuxDesktop + ? AuthServiceLinux.instance.authStateChanges + : AuthServiceFirebase.instance.authStateChanges; + + AppUser? get currentUser => isLinuxDesktop + ? AuthServiceLinux.instance.currentUser + : AuthServiceFirebase.instance.currentUser; + + Future initialize() async { + if (isLinuxDesktop) { + await AuthServiceLinux.instance.initialize(); + } else { + await AuthServiceFirebase.instance.initialize(); + } + } + + Future signInWithGoogle() async { + if (isLinuxDesktop) { + await AuthServiceLinux.instance.signInWithGoogle(); + } else { + await AuthServiceFirebase.instance.signInWithGoogle(); + } + } + + Future signOut() async { + if (isLinuxDesktop) { + await AuthServiceLinux.instance.signOut(); + } else { + await AuthServiceFirebase.instance.signOut(); + } + } + + /// Firebase ID token for API requests. Returns null when signed out. + Future getIdToken({bool forceRefresh = false}) async { + if (isLinuxDesktop) { + return AuthServiceLinux.instance.getIdToken( + forceRefresh: forceRefresh, + ); + } + return AuthServiceFirebase.instance.getIdToken( + forceRefresh: forceRefresh, + ); + } +} diff --git a/lib/services/auth_service_firebase.dart b/lib/services/auth_service_firebase.dart new file mode 100644 index 0000000..b6cedb7 --- /dev/null +++ b/lib/services/auth_service_firebase.dart @@ -0,0 +1,135 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:flutter/foundation.dart' show kIsWeb; +import 'package:google_sign_in/google_sign_in.dart'; + +import '../config/auth_config.dart'; +import '../models/app_user.dart'; + +class AuthServiceFirebase { + AuthServiceFirebase._(); + + static final AuthServiceFirebase instance = AuthServiceFirebase._(); + + final FirebaseAuth _firebaseAuth = FirebaseAuth.instance; + final GoogleSignIn _googleSignIn = GoogleSignIn.instance; + + bool _googleSignInReady = false; + + Stream get authStateChanges => + _firebaseAuth.authStateChanges().map(_mapUser); + + AppUser? get currentUser => _mapUser(_firebaseAuth.currentUser); + + Future initialize() async { + if (kIsWeb) { + if (googleWebOAuthClientId != null) { + await _googleSignIn.initialize(clientId: googleWebOAuthClientId); + _googleSignInReady = true; + } + return; + } + + if (googleWebOAuthClientId != null) { + await _googleSignIn.initialize( + serverClientId: googleWebOAuthClientId, + ); + } else { + await _googleSignIn.initialize(); + } + _googleSignInReady = true; + } + + Future signInWithGoogle() async { + if (kIsWeb) { + await _signInWithGoogleWeb(); + return; + } + + _ensureGoogleSignInReady(); + + final GoogleSignInAccount account = await _googleSignIn.authenticate(); + final String? idToken = account.authentication.idToken; + if (idToken == null) { + throw StateError('Google Sign-In did not return an ID token.'); + } + + final credential = GoogleAuthProvider.credential(idToken: idToken); + await _firebaseAuth.signInWithCredential(credential); + } + + /// Web: popup avoids Firebase redirect sessionStorage (broken under ETP). + /// Falls back to Google Identity Services + [signInWithCredential] when set up. + Future _signInWithGoogleWeb() async { + try { + await _firebaseAuth.signInWithPopup(GoogleAuthProvider()); + return; + } on FirebaseAuthException catch (e) { + if (_isUserCancelledAuth(e)) { + rethrow; + } + } + + if (!_googleSignInReady) { + throw StateError( + 'Google sign-in failed in this browser (often Firefox with strict ' + 'privacy). Set googleWebOAuthClientId in lib/config/auth_config.dart ' + '(Firebase Console → Authentication → Google → Web SDK configuration) ' + 'and add ${Uri.base.origin} to that client\'s authorized JavaScript ' + 'origins in Google Cloud Console.', + ); + } + + final GoogleSignInAccount account = await _googleSignIn.authenticate(); + final String? idToken = account.authentication.idToken; + if (idToken == null) { + throw StateError('Google Sign-In did not return an ID token.'); + } + + await _firebaseAuth.signInWithCredential( + GoogleAuthProvider.credential(idToken: idToken), + ); + } + + Future signOut() async { + if (_googleSignInReady) { + await _googleSignIn.signOut(); + } + await _firebaseAuth.signOut(); + } + + Future getIdToken({bool forceRefresh = false}) async { + final User? user = _firebaseAuth.currentUser; + if (user == null) { + return null; + } + return user.getIdToken(forceRefresh); + } + + void _ensureGoogleSignInReady() { + if (_googleSignInReady) { + return; + } + throw StateError( + 'Google Sign-In is not configured. Set googleWebOAuthClientId in ' + 'lib/config/auth_config.dart for Android, or run a full restart after ' + 'changing auth settings.', + ); + } + + bool _isUserCancelledAuth(FirebaseAuthException e) { + return e.code == 'popup-closed-by-user' || + e.code == 'cancelled-popup-request'; + } + + AppUser? _mapUser(User? user) { + if (user == null) { + return null; + } + return AppUser( + uid: user.uid, + email: user.email, + displayName: user.displayName, + photoUrl: user.photoURL, + ); + } +} diff --git a/lib/services/auth_service_linux.dart b/lib/services/auth_service_linux.dart new file mode 100644 index 0000000..cb97d80 --- /dev/null +++ b/lib/services/auth_service_linux.dart @@ -0,0 +1,201 @@ +import 'dart:async'; +import 'dart:convert'; + +import 'package:google_sign_in_all_platforms/google_sign_in_all_platforms.dart'; +import 'package:http/http.dart' as http; + +import '../config/auth_config.dart'; +import '../firebase_options.dart'; +import '../models/app_user.dart'; + +class AuthServiceLinux { + AuthServiceLinux._(); + + static final AuthServiceLinux instance = AuthServiceLinux._(); + + final StreamController _authController = + StreamController.broadcast(); + + GoogleSignIn? _googleSignIn; + AppUser? _currentUser; + String? _idToken; + String? _refreshToken; + DateTime? _tokenExpiry; + + Stream get authStateChanges => _authController.stream; + + AppUser? get currentUser => _currentUser; + + Future initialize() async { + if (!_hasOAuthConfig) { + _emitUser(null); + return; + } + + _googleSignIn = GoogleSignIn( + params: GoogleSignInParams( + clientId: googleWebOAuthClientId!, + clientSecret: googleOAuthDesktopClientSecret!, + scopes: const ['openid', 'profile', 'email'], + ), + ); + + _googleSignIn!.authenticationState.listen((GoogleSignInCredentials? creds) { + if (creds == null) { + _emitUser(null); + } + }); + + final GoogleSignInCredentials? restored = + await _googleSignIn!.silentSignIn(); + if (restored?.idToken != null) { + try { + final AppUser user = await _signInToFirebaseWithIdp(restored!.idToken!); + _emitUser(user); + } catch (_) { + // Ignore stale desktop sessions on startup. + } + } + } + + bool get _hasOAuthConfig => + googleWebOAuthClientId != null && + googleOAuthDesktopClientSecret != null; + + Future signInWithGoogle() async { + _validateOAuthConfig(); + final GoogleSignInCredentials? credentials = await _googleSignIn!.signIn(); + if (credentials == null) { + throw StateError('Google Sign-In was cancelled.'); + } + + final String? idToken = credentials.idToken; + if (idToken == null) { + throw StateError('Google Sign-In did not return an ID token.'); + } + + final AppUser user = await _signInToFirebaseWithIdp(idToken); + _emitUser(user); + } + + Future signOut() async { + await _googleSignIn?.signOut(); + _clearTokens(); + _emitUser(null); + } + + Future getIdToken({bool forceRefresh = false}) async { + if (_currentUser == null) { + return null; + } + if (!forceRefresh && + _idToken != null && + _tokenExpiry != null && + DateTime.now().isBefore(_tokenExpiry!)) { + return _idToken; + } + if (_refreshToken != null) { + await _refreshIdToken(); + return _idToken; + } + return _idToken; + } + + void _emitUser(AppUser? user) { + _currentUser = user; + if (user == null) { + _clearTokens(); + } + _authController.add(user); + } + + void _clearTokens() { + _idToken = null; + _refreshToken = null; + _tokenExpiry = null; + } + + void _storeTokens(Map data) { + _idToken = data['idToken'] as String?; + _refreshToken = data['refreshToken'] as String?; + final int expiresIn = (data['expiresIn'] as String?) != null + ? int.parse(data['expiresIn'] as String) + : (data['expiresIn'] as num?)?.toInt() ?? 3600; + _tokenExpiry = DateTime.now().add(Duration(seconds: expiresIn - 60)); + } + + Future _refreshIdToken() async { + if (_refreshToken == null) { + return; + } + final Uri uri = Uri.parse( + 'https://securetoken.googleapis.com/v1/token' + '?key=${DefaultFirebaseOptions.web.apiKey}', + ); + final http.Response response = await http.post( + uri, + headers: const { + 'Content-Type': 'application/x-www-form-urlencoded', + }, + body: 'grant_type=refresh_token&refresh_token=$_refreshToken', + ); + if (response.statusCode != 200) { + _clearTokens(); + return; + } + final Map data = + jsonDecode(response.body) as Map; + _idToken = data['id_token'] as String?; + _refreshToken = data['refresh_token'] as String? ?? _refreshToken; + final int expiresIn = (data['expires_in'] as num?)?.toInt() ?? 3600; + _tokenExpiry = DateTime.now().add(Duration(seconds: expiresIn - 60)); + } + + void _validateOAuthConfig() { + if (!_hasOAuthConfig) { + throw StateError( + 'Linux desktop auth requires googleWebOAuthClientId and ' + 'googleOAuthDesktopClientSecret in lib/config/auth_config.dart. ' + 'Create a Web application OAuth client in Google Cloud Console ' + '(same project as Firebase) and add ' + 'http://127.0.0.1 as an authorized redirect URI.', + ); + } + } + + Future _signInToFirebaseWithIdp(String idToken) async { + final Uri uri = Uri.parse( + 'https://identitytoolkit.googleapis.com/v1/accounts:signInWithIdp' + '?key=${DefaultFirebaseOptions.web.apiKey}', + ); + + final http.Response response = await http.post( + uri, + headers: const {'Content-Type': 'application/json'}, + body: jsonEncode({ + 'postBody': 'id_token=$idToken&providerId=google.com', + 'requestUri': 'http://localhost', + 'returnIdpCredential': true, + 'returnSecureToken': true, + }), + ); + + if (response.statusCode != 200) { + throw StateError( + 'Firebase sign-in failed (${response.statusCode}): ${response.body}', + ); + } + + final Map data = + jsonDecode(response.body) as Map; + + _storeTokens(data); + + return AppUser( + uid: data['localId'] as String, + email: data['email'] as String?, + displayName: data['displayName'] as String?, + photoUrl: data['photoUrl'] as String?, + ); + } +} diff --git a/lib/theme/app_theme.dart b/lib/theme/app_theme.dart new file mode 100644 index 0000000..a94d598 --- /dev/null +++ b/lib/theme/app_theme.dart @@ -0,0 +1,59 @@ +import 'package:flutter/material.dart'; + +abstract final class AppColors { + static const Color background = Color(0xFF070B14); + static const Color surface = Color(0xFF111827); + static const Color surfaceElevated = Color(0xFF1A2332); + static const Color accent = Color(0xFF22D3EE); + static const Color accentMuted = Color(0xFF0891B2); + static const Color textPrimary = Color(0xFFF8FAFC); + static const Color textSecondary = Color(0xFF94A3B8); + static const Color success = Color(0xFF34D399); +} + +ThemeData buildAppTheme() { + const ColorScheme scheme = ColorScheme.dark( + primary: AppColors.accent, + onPrimary: AppColors.background, + secondary: AppColors.accentMuted, + surface: AppColors.surface, + onSurface: AppColors.textPrimary, + error: Color(0xFFF87171), + ); + + return ThemeData( + useMaterial3: true, + brightness: Brightness.dark, + colorScheme: scheme, + scaffoldBackgroundColor: AppColors.background, + textTheme: const TextTheme( + headlineLarge: TextStyle( + fontSize: 36, + fontWeight: FontWeight.w700, + height: 1.15, + letterSpacing: -0.5, + color: AppColors.textPrimary, + ), + headlineMedium: TextStyle( + fontSize: 28, + fontWeight: FontWeight.w600, + color: AppColors.textPrimary, + ), + titleLarge: TextStyle( + fontSize: 18, + fontWeight: FontWeight.w600, + color: AppColors.textPrimary, + ), + bodyLarge: TextStyle( + fontSize: 16, + height: 1.5, + color: AppColors.textSecondary, + ), + labelLarge: TextStyle( + fontSize: 16, + fontWeight: FontWeight.w600, + color: AppColors.background, + ), + ), + ); +} diff --git a/lib/widgets/benefit_row.dart b/lib/widgets/benefit_row.dart new file mode 100644 index 0000000..e145775 --- /dev/null +++ b/lib/widgets/benefit_row.dart @@ -0,0 +1,51 @@ +import 'package:flutter/material.dart'; + +import '../theme/app_theme.dart'; + +class BenefitRow extends StatelessWidget { + const BenefitRow({ + super.key, + required this.icon, + required this.title, + required this.subtitle, + }); + + final IconData icon; + final String title; + final String subtitle; + + @override + Widget build(BuildContext context) { + return Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Container( + width: 44, + height: 44, + decoration: BoxDecoration( + color: AppColors.accent.withValues(alpha: 0.12), + borderRadius: BorderRadius.circular(10), + border: Border.all( + color: AppColors.accent.withValues(alpha: 0.25), + ), + ), + child: Icon(icon, color: AppColors.accent, size: 22), + ), + const SizedBox(width: 14), + Expanded( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + title, + style: Theme.of(context).textTheme.titleLarge, + ), + const SizedBox(height: 4), + Text(subtitle, style: Theme.of(context).textTheme.bodyLarge), + ], + ), + ), + ], + ); + } +} diff --git a/lib/widgets/google_sign_in_button.dart b/lib/widgets/google_sign_in_button.dart new file mode 100644 index 0000000..14af37b --- /dev/null +++ b/lib/widgets/google_sign_in_button.dart @@ -0,0 +1,139 @@ +import 'package:flutter/material.dart'; + +import '../services/auth_service.dart'; +import '../theme/app_theme.dart'; + +class GoogleSignInButton extends StatefulWidget { + const GoogleSignInButton({ + super.key, + this.label = 'Continue with Google', + this.compact = false, + this.onSignedIn, + this.onError, + }); + + final String label; + final bool compact; + final VoidCallback? onSignedIn; + final void Function(Object error)? onError; + + @override + State createState() => _GoogleSignInButtonState(); +} + +class _GoogleSignInButtonState extends State { + bool _isLoading = false; + + Future _signIn() async { + setState(() => _isLoading = true); + try { + await AuthService.instance.signInWithGoogle(); + widget.onSignedIn?.call(); + } catch (error) { + widget.onError?.call(error); + } finally { + if (mounted) { + setState(() => _isLoading = false); + } + } + } + + @override + Widget build(BuildContext context) { + final double verticalPadding = widget.compact ? 14 : 16; + + return SizedBox( + width: double.infinity, + child: FilledButton( + onPressed: _isLoading ? null : _signIn, + style: FilledButton.styleFrom( + backgroundColor: Colors.white, + foregroundColor: const Color(0xFF1F2937), + disabledBackgroundColor: Colors.white.withValues(alpha: 0.7), + padding: EdgeInsets.symmetric(vertical: verticalPadding), + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(12), + ), + elevation: 0, + ), + child: _isLoading + ? const SizedBox( + width: 22, + height: 22, + child: CircularProgressIndicator( + strokeWidth: 2.5, + color: AppColors.background, + ), + ) + : Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + const _GoogleLogo(size: 22), + const SizedBox(width: 12), + Flexible( + child: Text( + widget.label, + style: const TextStyle( + fontSize: 16, + fontWeight: FontWeight.w600, + color: Color(0xFF1F2937), + ), + textAlign: TextAlign.center, + overflow: TextOverflow.ellipsis, + maxLines: 1, + ), + ), + ], + ), + ), + ); + } +} + +class _GoogleLogo extends StatelessWidget { + const _GoogleLogo({required this.size}); + + final double size; + + @override + Widget build(BuildContext context) { + return SizedBox( + width: size, + height: size, + child: CustomPaint(painter: _GoogleLogoPainter()), + ); + } +} + +class _GoogleLogoPainter extends CustomPainter { + @override + void paint(Canvas canvas, Size size) { + final double w = size.width; + final double h = size.height; + final Offset center = Offset(w / 2, h / 2); + final double r = w * 0.42; + + void arc(Color color, double start, double sweep) { + final Paint paint = Paint() + ..color = color + ..style = PaintingStyle.stroke + ..strokeWidth = w * 0.18 + ..strokeCap = StrokeCap.round; + canvas.drawArc( + Rect.fromCircle(center: center, radius: r), + start, + sweep, + false, + paint, + ); + } + + arc(const Color(0xFF4285F4), -0.4, 1.6); + arc(const Color(0xFFEA4335), 1.2, 1.3); + arc(const Color(0xFFFBBC05), 2.5, 1.3); + arc(const Color(0xFF34A853), 3.8, 1.3); + } + + @override + bool shouldRepaint(covariant CustomPainter oldDelegate) => false; +} diff --git a/lib/widgets/profile_session.dart b/lib/widgets/profile_session.dart new file mode 100644 index 0000000..b5012ed --- /dev/null +++ b/lib/widgets/profile_session.dart @@ -0,0 +1,79 @@ +import 'dart:async'; + +import 'package:flutter/material.dart'; + +import '../models/app_user.dart'; +import '../models/sync_result.dart'; +import '../models/user_profile.dart'; +import '../repositories/user_profile_repository.dart'; +import '../theme/app_theme.dart'; + +/// Starts a profile sync session for [user] and exposes profile + sync state. +class ProfileSession extends StatefulWidget { + const ProfileSession({ + super.key, + required this.user, + required this.builder, + }); + + final AppUser user; + final Widget Function( + BuildContext context, + UserProfile? profile, + ProfileSyncStatus syncStatus, + ) builder; + + @override + State createState() => _ProfileSessionState(); +} + +class _ProfileSessionState extends State { + late final Future _sessionReady; + ProfileSyncStatus _syncStatus = ProfileSyncStatus.idle; + StreamSubscription? _syncStatusSubscription; + + @override + void initState() { + super.initState(); + _sessionReady = UserProfileRepository.instance.startSession(widget.user); + _syncStatusSubscription = + UserProfileRepository.instance.syncStatusStream.listen(( + ProfileSyncStatus status, + ) { + if (mounted) { + setState(() => _syncStatus = status); + } + }); + } + + @override + void dispose() { + _syncStatusSubscription?.cancel(); + UserProfileRepository.instance.endSession(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return FutureBuilder( + future: _sessionReady, + builder: (BuildContext context, AsyncSnapshot sessionSnap) { + if (sessionSnap.connectionState != ConnectionState.done) { + return const Scaffold( + body: Center( + child: CircularProgressIndicator(color: AppColors.accent), + ), + ); + } + + return StreamBuilder( + stream: UserProfileRepository.instance.profileStream, + initialData: UserProfileRepository.instance.currentProfile, + builder: (BuildContext context, AsyncSnapshot snap) { + return widget.builder(context, snap.data, _syncStatus); + }, + ); + }, + ); + } +} diff --git a/linux/.gitignore b/linux/.gitignore new file mode 100644 index 0000000..d3896c9 --- /dev/null +++ b/linux/.gitignore @@ -0,0 +1 @@ +flutter/ephemeral diff --git a/linux/CMakeLists.txt b/linux/CMakeLists.txt new file mode 100644 index 0000000..14185a3 --- /dev/null +++ b/linux/CMakeLists.txt @@ -0,0 +1,128 @@ +# Project-level configuration. +cmake_minimum_required(VERSION 3.13) +project(runner LANGUAGES CXX) + +# The name of the executable created for the application. Change this to change +# the on-disk name of your application. +set(BINARY_NAME "cyberhybridhub") +# The unique GTK application identifier for this application. See: +# https://wiki.gnome.org/HowDoI/ChooseApplicationID +set(APPLICATION_ID "com.cyberhybridhub.cyberhybridhub") + +# Explicitly opt in to modern CMake behaviors to avoid warnings with recent +# versions of CMake. +cmake_policy(SET CMP0063 NEW) + +# Load bundled libraries from the lib/ directory relative to the binary. +set(CMAKE_INSTALL_RPATH "$ORIGIN/lib") + +# Root filesystem for cross-building. +if(FLUTTER_TARGET_PLATFORM_SYSROOT) + set(CMAKE_SYSROOT ${FLUTTER_TARGET_PLATFORM_SYSROOT}) + set(CMAKE_FIND_ROOT_PATH ${CMAKE_SYSROOT}) + set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) + set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY) + set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY) + set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY) +endif() + +# Define build configuration options. +if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) + set(CMAKE_BUILD_TYPE "Debug" CACHE + STRING "Flutter build mode" FORCE) + set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS + "Debug" "Profile" "Release") +endif() + +# Compilation settings that should be applied to most targets. +# +# Be cautious about adding new options here, as plugins use this function by +# default. In most cases, you should add new options to specific targets instead +# of modifying this function. +function(APPLY_STANDARD_SETTINGS TARGET) + target_compile_features(${TARGET} PUBLIC cxx_std_14) + target_compile_options(${TARGET} PRIVATE -Wall -Werror) + target_compile_options(${TARGET} PRIVATE "$<$>:-O3>") + target_compile_definitions(${TARGET} PRIVATE "$<$>:NDEBUG>") +endfunction() + +# Flutter library and tool build rules. +set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") +add_subdirectory(${FLUTTER_MANAGED_DIR}) + +# System-level dependencies. +find_package(PkgConfig REQUIRED) +pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) + +# Application build; see runner/CMakeLists.txt. +add_subdirectory("runner") + +# Run the Flutter tool portions of the build. This must not be removed. +add_dependencies(${BINARY_NAME} flutter_assemble) + +# Only the install-generated bundle's copy of the executable will launch +# correctly, since the resources must in the right relative locations. To avoid +# people trying to run the unbundled copy, put it in a subdirectory instead of +# the default top-level location. +set_target_properties(${BINARY_NAME} + PROPERTIES + RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/intermediates_do_not_run" +) + + +# Generated plugin build rules, which manage building the plugins and adding +# them to the application. +include(flutter/generated_plugins.cmake) + + +# === Installation === +# By default, "installing" just makes a relocatable bundle in the build +# directory. +set(BUILD_BUNDLE_DIR "${PROJECT_BINARY_DIR}/bundle") +if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) + set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) +endif() + +# Start with a clean build bundle directory every time. +install(CODE " + file(REMOVE_RECURSE \"${BUILD_BUNDLE_DIR}/\") + " COMPONENT Runtime) + +set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") +set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}/lib") + +install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" + COMPONENT Runtime) + +install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" + COMPONENT Runtime) + +install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) + +foreach(bundled_library ${PLUGIN_BUNDLED_LIBRARIES}) + install(FILES "${bundled_library}" + DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) +endforeach(bundled_library) + +# Copy the native assets provided by the build.dart from all packages. +set(NATIVE_ASSETS_DIR "${PROJECT_BUILD_DIR}native_assets/linux/") +install(DIRECTORY "${NATIVE_ASSETS_DIR}" + DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) + +# Fully re-copy the assets directory on each build to avoid having stale files +# from a previous install. +set(FLUTTER_ASSET_DIR_NAME "flutter_assets") +install(CODE " + file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") + " COMPONENT Runtime) +install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" + DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) + +# Install the AOT library on non-Debug builds only. +if(NOT CMAKE_BUILD_TYPE MATCHES "Debug") + install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) +endif() diff --git a/linux/flutter/CMakeLists.txt b/linux/flutter/CMakeLists.txt new file mode 100644 index 0000000..d5bd016 --- /dev/null +++ b/linux/flutter/CMakeLists.txt @@ -0,0 +1,88 @@ +# This file controls Flutter-level build steps. It should not be edited. +cmake_minimum_required(VERSION 3.10) + +set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") + +# Configuration provided via flutter tool. +include(${EPHEMERAL_DIR}/generated_config.cmake) + +# TODO: Move the rest of this into files in ephemeral. See +# https://github.com/flutter/flutter/issues/57146. + +# Serves the same purpose as list(TRANSFORM ... PREPEND ...), +# which isn't available in 3.10. +function(list_prepend LIST_NAME PREFIX) + set(NEW_LIST "") + foreach(element ${${LIST_NAME}}) + list(APPEND NEW_LIST "${PREFIX}${element}") + endforeach(element) + set(${LIST_NAME} "${NEW_LIST}" PARENT_SCOPE) +endfunction() + +# === Flutter Library === +# System-level dependencies. +find_package(PkgConfig REQUIRED) +pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) +pkg_check_modules(GLIB REQUIRED IMPORTED_TARGET glib-2.0) +pkg_check_modules(GIO REQUIRED IMPORTED_TARGET gio-2.0) + +set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/libflutter_linux_gtk.so") + +# Published to parent scope for install step. +set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) +set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) +set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) +set(AOT_LIBRARY "${PROJECT_DIR}/build/lib/libapp.so" PARENT_SCOPE) + +list(APPEND FLUTTER_LIBRARY_HEADERS + "fl_basic_message_channel.h" + "fl_binary_codec.h" + "fl_binary_messenger.h" + "fl_dart_project.h" + "fl_engine.h" + "fl_json_message_codec.h" + "fl_json_method_codec.h" + "fl_message_codec.h" + "fl_method_call.h" + "fl_method_channel.h" + "fl_method_codec.h" + "fl_method_response.h" + "fl_plugin_registrar.h" + "fl_plugin_registry.h" + "fl_standard_message_codec.h" + "fl_standard_method_codec.h" + "fl_string_codec.h" + "fl_value.h" + "fl_view.h" + "flutter_linux.h" +) +list_prepend(FLUTTER_LIBRARY_HEADERS "${EPHEMERAL_DIR}/flutter_linux/") +add_library(flutter INTERFACE) +target_include_directories(flutter INTERFACE + "${EPHEMERAL_DIR}" +) +target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}") +target_link_libraries(flutter INTERFACE + PkgConfig::GTK + PkgConfig::GLIB + PkgConfig::GIO +) +add_dependencies(flutter flutter_assemble) + +# === Flutter tool backend === +# _phony_ is a non-existent file to force this command to run every time, +# since currently there's no way to get a full input/output list from the +# flutter tool. +add_custom_command( + OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} + ${CMAKE_CURRENT_BINARY_DIR}/_phony_ + COMMAND ${CMAKE_COMMAND} -E env + ${FLUTTER_TOOL_ENVIRONMENT} + "${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.sh" + ${FLUTTER_TARGET_PLATFORM} ${CMAKE_BUILD_TYPE} + VERBATIM +) +add_custom_target(flutter_assemble DEPENDS + "${FLUTTER_LIBRARY}" + ${FLUTTER_LIBRARY_HEADERS} +) diff --git a/linux/flutter/generated_plugin_registrant.cc b/linux/flutter/generated_plugin_registrant.cc new file mode 100644 index 0000000..4c0025f --- /dev/null +++ b/linux/flutter/generated_plugin_registrant.cc @@ -0,0 +1,19 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#include "generated_plugin_registrant.h" + +#include +#include + +void fl_register_plugins(FlPluginRegistry* registry) { + g_autoptr(FlPluginRegistrar) sqlite3_flutter_libs_registrar = + fl_plugin_registry_get_registrar_for_plugin(registry, "Sqlite3FlutterLibsPlugin"); + sqlite3_flutter_libs_plugin_register_with_registrar(sqlite3_flutter_libs_registrar); + g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar = + fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin"); + url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar); +} diff --git a/linux/flutter/generated_plugin_registrant.h b/linux/flutter/generated_plugin_registrant.h new file mode 100644 index 0000000..e0f0a47 --- /dev/null +++ b/linux/flutter/generated_plugin_registrant.h @@ -0,0 +1,15 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#ifndef GENERATED_PLUGIN_REGISTRANT_ +#define GENERATED_PLUGIN_REGISTRANT_ + +#include + +// Registers Flutter plugins. +void fl_register_plugins(FlPluginRegistry* registry); + +#endif // GENERATED_PLUGIN_REGISTRANT_ diff --git a/linux/flutter/generated_plugins.cmake b/linux/flutter/generated_plugins.cmake new file mode 100644 index 0000000..75d875c --- /dev/null +++ b/linux/flutter/generated_plugins.cmake @@ -0,0 +1,26 @@ +# +# Generated file, do not edit. +# + +list(APPEND FLUTTER_PLUGIN_LIST + sqlite3_flutter_libs + url_launcher_linux +) + +list(APPEND FLUTTER_FFI_PLUGIN_LIST + jni +) + +set(PLUGIN_BUNDLED_LIBRARIES) + +foreach(plugin ${FLUTTER_PLUGIN_LIST}) + add_subdirectory(flutter/ephemeral/.plugin_symlinks/${plugin}/linux plugins/${plugin}) + target_link_libraries(${BINARY_NAME} PRIVATE ${plugin}_plugin) + list(APPEND PLUGIN_BUNDLED_LIBRARIES $) + list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${plugin}_bundled_libraries}) +endforeach(plugin) + +foreach(ffi_plugin ${FLUTTER_FFI_PLUGIN_LIST}) + add_subdirectory(flutter/ephemeral/.plugin_symlinks/${ffi_plugin}/linux plugins/${ffi_plugin}) + list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${ffi_plugin}_bundled_libraries}) +endforeach(ffi_plugin) diff --git a/linux/runner/CMakeLists.txt b/linux/runner/CMakeLists.txt new file mode 100644 index 0000000..e97dabc --- /dev/null +++ b/linux/runner/CMakeLists.txt @@ -0,0 +1,26 @@ +cmake_minimum_required(VERSION 3.13) +project(runner LANGUAGES CXX) + +# Define the application target. To change its name, change BINARY_NAME in the +# top-level CMakeLists.txt, not the value here, or `flutter run` will no longer +# work. +# +# Any new source files that you add to the application should be added here. +add_executable(${BINARY_NAME} + "main.cc" + "my_application.cc" + "${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" +) + +# Apply the standard set of build settings. This can be removed for applications +# that need different build settings. +apply_standard_settings(${BINARY_NAME}) + +# Add preprocessor definitions for the application ID. +add_definitions(-DAPPLICATION_ID="${APPLICATION_ID}") + +# Add dependency libraries. Add any application-specific dependencies here. +target_link_libraries(${BINARY_NAME} PRIVATE flutter) +target_link_libraries(${BINARY_NAME} PRIVATE PkgConfig::GTK) + +target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") diff --git a/linux/runner/main.cc b/linux/runner/main.cc new file mode 100644 index 0000000..e7c5c54 --- /dev/null +++ b/linux/runner/main.cc @@ -0,0 +1,6 @@ +#include "my_application.h" + +int main(int argc, char** argv) { + g_autoptr(MyApplication) app = my_application_new(); + return g_application_run(G_APPLICATION(app), argc, argv); +} diff --git a/linux/runner/my_application.cc b/linux/runner/my_application.cc new file mode 100644 index 0000000..209af1f --- /dev/null +++ b/linux/runner/my_application.cc @@ -0,0 +1,148 @@ +#include "my_application.h" + +#include +#ifdef GDK_WINDOWING_X11 +#include +#endif + +#include "flutter/generated_plugin_registrant.h" + +struct _MyApplication { + GtkApplication parent_instance; + char** dart_entrypoint_arguments; +}; + +G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION) + +// Called when first Flutter frame received. +static void first_frame_cb(MyApplication* self, FlView* view) { + gtk_widget_show(gtk_widget_get_toplevel(GTK_WIDGET(view))); +} + +// Implements GApplication::activate. +static void my_application_activate(GApplication* application) { + MyApplication* self = MY_APPLICATION(application); + GtkWindow* window = + GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application))); + + // Use a header bar when running in GNOME as this is the common style used + // by applications and is the setup most users will be using (e.g. Ubuntu + // desktop). + // If running on X and not using GNOME then just use a traditional title bar + // in case the window manager does more exotic layout, e.g. tiling. + // If running on Wayland assume the header bar will work (may need changing + // if future cases occur). + gboolean use_header_bar = TRUE; +#ifdef GDK_WINDOWING_X11 + GdkScreen* screen = gtk_window_get_screen(window); + if (GDK_IS_X11_SCREEN(screen)) { + const gchar* wm_name = gdk_x11_screen_get_window_manager_name(screen); + if (g_strcmp0(wm_name, "GNOME Shell") != 0) { + use_header_bar = FALSE; + } + } +#endif + if (use_header_bar) { + GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); + gtk_widget_show(GTK_WIDGET(header_bar)); + gtk_header_bar_set_title(header_bar, "cyberhybridhub"); + gtk_header_bar_set_show_close_button(header_bar, TRUE); + gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); + } else { + gtk_window_set_title(window, "cyberhybridhub"); + } + + gtk_window_set_default_size(window, 1280, 720); + + g_autoptr(FlDartProject) project = fl_dart_project_new(); + fl_dart_project_set_dart_entrypoint_arguments( + project, self->dart_entrypoint_arguments); + + FlView* view = fl_view_new(project); + GdkRGBA background_color; + // Background defaults to black, override it here if necessary, e.g. #00000000 + // for transparent. + gdk_rgba_parse(&background_color, "#000000"); + fl_view_set_background_color(view, &background_color); + gtk_widget_show(GTK_WIDGET(view)); + gtk_container_add(GTK_CONTAINER(window), GTK_WIDGET(view)); + + // Show the window when Flutter renders. + // Requires the view to be realized so we can start rendering. + g_signal_connect_swapped(view, "first-frame", G_CALLBACK(first_frame_cb), + self); + gtk_widget_realize(GTK_WIDGET(view)); + + fl_register_plugins(FL_PLUGIN_REGISTRY(view)); + + gtk_widget_grab_focus(GTK_WIDGET(view)); +} + +// Implements GApplication::local_command_line. +static gboolean my_application_local_command_line(GApplication* application, + gchar*** arguments, + int* exit_status) { + MyApplication* self = MY_APPLICATION(application); + // Strip out the first argument as it is the binary name. + self->dart_entrypoint_arguments = g_strdupv(*arguments + 1); + + g_autoptr(GError) error = nullptr; + if (!g_application_register(application, nullptr, &error)) { + g_warning("Failed to register: %s", error->message); + *exit_status = 1; + return TRUE; + } + + g_application_activate(application); + *exit_status = 0; + + return TRUE; +} + +// Implements GApplication::startup. +static void my_application_startup(GApplication* application) { + // MyApplication* self = MY_APPLICATION(object); + + // Perform any actions required at application startup. + + G_APPLICATION_CLASS(my_application_parent_class)->startup(application); +} + +// Implements GApplication::shutdown. +static void my_application_shutdown(GApplication* application) { + // MyApplication* self = MY_APPLICATION(object); + + // Perform any actions required at application shutdown. + + G_APPLICATION_CLASS(my_application_parent_class)->shutdown(application); +} + +// Implements GObject::dispose. +static void my_application_dispose(GObject* object) { + MyApplication* self = MY_APPLICATION(object); + g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev); + G_OBJECT_CLASS(my_application_parent_class)->dispose(object); +} + +static void my_application_class_init(MyApplicationClass* klass) { + G_APPLICATION_CLASS(klass)->activate = my_application_activate; + G_APPLICATION_CLASS(klass)->local_command_line = + my_application_local_command_line; + G_APPLICATION_CLASS(klass)->startup = my_application_startup; + G_APPLICATION_CLASS(klass)->shutdown = my_application_shutdown; + G_OBJECT_CLASS(klass)->dispose = my_application_dispose; +} + +static void my_application_init(MyApplication* self) {} + +MyApplication* my_application_new() { + // Set the program name to the application ID, which helps various systems + // like GTK and desktop environments map this running application to its + // corresponding .desktop file. This ensures better integration by allowing + // the application to be recognized beyond its binary name. + g_set_prgname(APPLICATION_ID); + + return MY_APPLICATION(g_object_new(my_application_get_type(), + "application-id", APPLICATION_ID, "flags", + G_APPLICATION_NON_UNIQUE, nullptr)); +} diff --git a/linux/runner/my_application.h b/linux/runner/my_application.h new file mode 100644 index 0000000..db16367 --- /dev/null +++ b/linux/runner/my_application.h @@ -0,0 +1,21 @@ +#ifndef FLUTTER_MY_APPLICATION_H_ +#define FLUTTER_MY_APPLICATION_H_ + +#include + +G_DECLARE_FINAL_TYPE(MyApplication, + my_application, + MY, + APPLICATION, + GtkApplication) + +/** + * my_application_new: + * + * Creates a new Flutter-based application. + * + * Returns: a new #MyApplication. + */ +MyApplication* my_application_new(); + +#endif // FLUTTER_MY_APPLICATION_H_ diff --git a/macos/.gitignore b/macos/.gitignore new file mode 100644 index 0000000..746adbb --- /dev/null +++ b/macos/.gitignore @@ -0,0 +1,7 @@ +# Flutter-related +**/Flutter/ephemeral/ +**/Pods/ + +# Xcode-related +**/dgph +**/xcuserdata/ diff --git a/macos/Flutter/Flutter-Debug.xcconfig b/macos/Flutter/Flutter-Debug.xcconfig new file mode 100644 index 0000000..c2efd0b --- /dev/null +++ b/macos/Flutter/Flutter-Debug.xcconfig @@ -0,0 +1 @@ +#include "ephemeral/Flutter-Generated.xcconfig" diff --git a/macos/Flutter/Flutter-Release.xcconfig b/macos/Flutter/Flutter-Release.xcconfig new file mode 100644 index 0000000..c2efd0b --- /dev/null +++ b/macos/Flutter/Flutter-Release.xcconfig @@ -0,0 +1 @@ +#include "ephemeral/Flutter-Generated.xcconfig" diff --git a/macos/Flutter/GeneratedPluginRegistrant.swift b/macos/Flutter/GeneratedPluginRegistrant.swift new file mode 100644 index 0000000..0a7ffc0 --- /dev/null +++ b/macos/Flutter/GeneratedPluginRegistrant.swift @@ -0,0 +1,24 @@ +// +// Generated file. Do not edit. +// + +import FlutterMacOS +import Foundation + +import connectivity_plus +import firebase_auth +import firebase_core +import google_sign_in_ios +import shared_preferences_foundation +import sqlite3_flutter_libs +import url_launcher_macos + +func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { + ConnectivityPlusPlugin.register(with: registry.registrar(forPlugin: "ConnectivityPlusPlugin")) + FLTFirebaseAuthPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseAuthPlugin")) + FLTFirebaseCorePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseCorePlugin")) + FLTGoogleSignInPlugin.register(with: registry.registrar(forPlugin: "FLTGoogleSignInPlugin")) + SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin")) + Sqlite3FlutterLibsPlugin.register(with: registry.registrar(forPlugin: "Sqlite3FlutterLibsPlugin")) + UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin")) +} diff --git a/macos/Runner.xcodeproj/project.pbxproj b/macos/Runner.xcodeproj/project.pbxproj new file mode 100644 index 0000000..a0c59b3 --- /dev/null +++ b/macos/Runner.xcodeproj/project.pbxproj @@ -0,0 +1,729 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 54; + objects = { + +/* Begin PBXAggregateTarget section */ + 33CC111A2044C6BA0003C045 /* Flutter Assemble */ = { + isa = PBXAggregateTarget; + buildConfigurationList = 33CC111B2044C6BA0003C045 /* Build configuration list for PBXAggregateTarget "Flutter Assemble" */; + buildPhases = ( + 33CC111E2044C6BF0003C045 /* ShellScript */, + ); + dependencies = ( + ); + name = "Flutter Assemble"; + productName = FLX; + }; +/* End PBXAggregateTarget section */ + +/* Begin PBXBuildFile section */ + 331C80D8294CF71000263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C80D7294CF71000263BE5 /* RunnerTests.swift */; }; + 335BBD1B22A9A15E00E9071D /* GeneratedPluginRegistrant.swift in Sources */ = {isa = PBXBuildFile; fileRef = 335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */; }; + 33CC10F12044A3C60003C045 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC10F02044A3C60003C045 /* AppDelegate.swift */; }; + 33CC10F32044A3C60003C045 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F22044A3C60003C045 /* Assets.xcassets */; }; + 33CC10F62044A3C60003C045 /* MainMenu.xib in Resources */ = {isa = PBXBuildFile; fileRef = 33CC10F42044A3C60003C045 /* MainMenu.xib */; }; + 33CC11132044BFA00003C045 /* MainFlutterWindow.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33CC11122044BFA00003C045 /* MainFlutterWindow.swift */; }; + 78A318202AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage in Frameworks */ = {isa = PBXBuildFile; productRef = 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 331C80D9294CF71000263BE5 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 33CC10E52044A3C60003C045 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 33CC10EC2044A3C60003C045; + remoteInfo = Runner; + }; + 33CC111F2044C79F0003C045 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 33CC10E52044A3C60003C045 /* Project object */; + proxyType = 1; + remoteGlobalIDString = 33CC111A2044C6BA0003C045; + remoteInfo = FLX; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 33CC110E2044A8840003C045 /* Bundle Framework */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Bundle Framework"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 331C80D5294CF71000263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 331C80D7294CF71000263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = ""; }; + 333000ED22D3DE5D00554162 /* Warnings.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Warnings.xcconfig; sourceTree = ""; }; + 335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = GeneratedPluginRegistrant.swift; sourceTree = ""; }; + 33CC10ED2044A3C60003C045 /* cyberhybridhub.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "cyberhybridhub.app"; sourceTree = BUILT_PRODUCTS_DIR; }; + 33CC10F02044A3C60003C045 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; + 33CC10F22044A3C60003C045 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Assets.xcassets; path = Runner/Assets.xcassets; sourceTree = ""; }; + 33CC10F52044A3C60003C045 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/MainMenu.xib; sourceTree = ""; }; + 33CC10F72044A3C60003C045 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; name = Info.plist; path = Runner/Info.plist; sourceTree = ""; }; + 33CC11122044BFA00003C045 /* MainFlutterWindow.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MainFlutterWindow.swift; sourceTree = ""; }; + 33CEB47222A05771004F2AC0 /* Flutter-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Flutter-Debug.xcconfig"; sourceTree = ""; }; + 33CEB47422A05771004F2AC0 /* Flutter-Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "Flutter-Release.xcconfig"; sourceTree = ""; }; + 33CEB47722A0578A004F2AC0 /* Flutter-Generated.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = "Flutter-Generated.xcconfig"; path = "ephemeral/Flutter-Generated.xcconfig"; sourceTree = ""; }; + 33E51913231747F40026EE4D /* DebugProfile.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = DebugProfile.entitlements; sourceTree = ""; }; + 33E51914231749380026EE4D /* Release.entitlements */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.entitlements; path = Release.entitlements; sourceTree = ""; }; + 33E5194F232828860026EE4D /* AppInfo.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = AppInfo.xcconfig; sourceTree = ""; }; + 78E0A7A72DC9AD7400C4905E /* FlutterGeneratedPluginSwiftPackage */ = {isa = PBXFileReference; lastKnownFileType = wrapper; name = FlutterGeneratedPluginSwiftPackage; path = ephemeral/Packages/FlutterGeneratedPluginSwiftPackage; sourceTree = ""; }; + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Release.xcconfig; sourceTree = ""; }; + 9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; path = Debug.xcconfig; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 331C80D2294CF70F00263BE5 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 33CC10EA2044A3C60003C045 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 78A318202AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 331C80D6294CF71000263BE5 /* RunnerTests */ = { + isa = PBXGroup; + children = ( + 331C80D7294CF71000263BE5 /* RunnerTests.swift */, + ); + path = RunnerTests; + sourceTree = ""; + }; + 33BA886A226E78AF003329D5 /* Configs */ = { + isa = PBXGroup; + children = ( + 33E5194F232828860026EE4D /* AppInfo.xcconfig */, + 9740EEB21CF90195004384FC /* Debug.xcconfig */, + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */, + 333000ED22D3DE5D00554162 /* Warnings.xcconfig */, + ); + path = Configs; + sourceTree = ""; + }; + 33CC10E42044A3C60003C045 = { + isa = PBXGroup; + children = ( + 33FAB671232836740065AC1E /* Runner */, + 33CEB47122A05771004F2AC0 /* Flutter */, + 331C80D6294CF71000263BE5 /* RunnerTests */, + 33CC10EE2044A3C60003C045 /* Products */, + D73912EC22F37F3D000D13A0 /* Frameworks */, + ); + sourceTree = ""; + }; + 33CC10EE2044A3C60003C045 /* Products */ = { + isa = PBXGroup; + children = ( + 33CC10ED2044A3C60003C045 /* cyberhybridhub.app */, + 331C80D5294CF71000263BE5 /* RunnerTests.xctest */, + ); + name = Products; + sourceTree = ""; + }; + 33CC11242044D66E0003C045 /* Resources */ = { + isa = PBXGroup; + children = ( + 33CC10F22044A3C60003C045 /* Assets.xcassets */, + 33CC10F42044A3C60003C045 /* MainMenu.xib */, + 33CC10F72044A3C60003C045 /* Info.plist */, + ); + name = Resources; + path = ..; + sourceTree = ""; + }; + 33CEB47122A05771004F2AC0 /* Flutter */ = { + isa = PBXGroup; + children = ( + 78E0A7A72DC9AD7400C4905E /* FlutterGeneratedPluginSwiftPackage */, + 335BBD1A22A9A15E00E9071D /* GeneratedPluginRegistrant.swift */, + 33CEB47222A05771004F2AC0 /* Flutter-Debug.xcconfig */, + 33CEB47422A05771004F2AC0 /* Flutter-Release.xcconfig */, + 33CEB47722A0578A004F2AC0 /* Flutter-Generated.xcconfig */, + ); + path = Flutter; + sourceTree = ""; + }; + 33FAB671232836740065AC1E /* Runner */ = { + isa = PBXGroup; + children = ( + 33CC10F02044A3C60003C045 /* AppDelegate.swift */, + 33CC11122044BFA00003C045 /* MainFlutterWindow.swift */, + 33E51913231747F40026EE4D /* DebugProfile.entitlements */, + 33E51914231749380026EE4D /* Release.entitlements */, + 33CC11242044D66E0003C045 /* Resources */, + 33BA886A226E78AF003329D5 /* Configs */, + ); + path = Runner; + sourceTree = ""; + }; + D73912EC22F37F3D000D13A0 /* Frameworks */ = { + isa = PBXGroup; + children = ( + ); + name = Frameworks; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + 331C80D4294CF70F00263BE5 /* RunnerTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 331C80DE294CF71000263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; + buildPhases = ( + 331C80D1294CF70F00263BE5 /* Sources */, + 331C80D2294CF70F00263BE5 /* Frameworks */, + 331C80D3294CF70F00263BE5 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + 331C80DA294CF71000263BE5 /* PBXTargetDependency */, + ); + name = RunnerTests; + productName = RunnerTests; + productReference = 331C80D5294CF71000263BE5 /* RunnerTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + 33CC10EC2044A3C60003C045 /* Runner */ = { + isa = PBXNativeTarget; + buildConfigurationList = 33CC10FB2044A3C60003C045 /* Build configuration list for PBXNativeTarget "Runner" */; + buildPhases = ( + 33CC10E92044A3C60003C045 /* Sources */, + 33CC10EA2044A3C60003C045 /* Frameworks */, + 33CC10EB2044A3C60003C045 /* Resources */, + 33CC110E2044A8840003C045 /* Bundle Framework */, + 3399D490228B24CF009A79C7 /* ShellScript */, + ); + buildRules = ( + ); + dependencies = ( + 33CC11202044C79F0003C045 /* PBXTargetDependency */, + ); + name = Runner; + packageProductDependencies = ( + 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */, + ); + productName = Runner; + productReference = 33CC10ED2044A3C60003C045 /* cyberhybridhub.app */; + productType = "com.apple.product-type.application"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 33CC10E52044A3C60003C045 /* Project object */ = { + isa = PBXProject; + attributes = { + BuildIndependentTargetsInParallel = YES; + LastSwiftUpdateCheck = 0920; + LastUpgradeCheck = 1510; + ORGANIZATIONNAME = ""; + TargetAttributes = { + 331C80D4294CF70F00263BE5 = { + CreatedOnToolsVersion = 14.0; + TestTargetID = 33CC10EC2044A3C60003C045; + }; + 33CC10EC2044A3C60003C045 = { + CreatedOnToolsVersion = 9.2; + LastSwiftMigration = 1100; + ProvisioningStyle = Automatic; + SystemCapabilities = { + com.apple.Sandbox = { + enabled = 1; + }; + }; + }; + 33CC111A2044C6BA0003C045 = { + CreatedOnToolsVersion = 9.2; + ProvisioningStyle = Manual; + }; + }; + }; + buildConfigurationList = 33CC10E82044A3C60003C045 /* Build configuration list for PBXProject "Runner" */; + compatibilityVersion = "Xcode 9.3"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = 33CC10E42044A3C60003C045; + packageReferences = ( + 781AD8BC2B33823900A9FFBB /* XCLocalSwiftPackageReference "Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage" */, + ); + productRefGroup = 33CC10EE2044A3C60003C045 /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 33CC10EC2044A3C60003C045 /* Runner */, + 331C80D4294CF70F00263BE5 /* RunnerTests */, + 33CC111A2044C6BA0003C045 /* Flutter Assemble */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 331C80D3294CF70F00263BE5 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 33CC10EB2044A3C60003C045 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 33CC10F32044A3C60003C045 /* Assets.xcassets in Resources */, + 33CC10F62044A3C60003C045 /* MainMenu.xib in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXShellScriptBuildPhase section */ + 3399D490228B24CF009A79C7 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + ); + outputFileListPaths = ( + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "echo \"$PRODUCT_NAME.app\" > \"$PROJECT_DIR\"/Flutter/ephemeral/.app_filename && \"$FLUTTER_ROOT\"/packages/flutter_tools/bin/macos_assemble.sh embed\n"; + }; + 33CC111E2044C6BF0003C045 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + Flutter/ephemeral/FlutterInputs.xcfilelist, + ); + inputPaths = ( + Flutter/ephemeral/tripwire, + ); + outputFileListPaths = ( + Flutter/ephemeral/FlutterOutputs.xcfilelist, + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "\"$FLUTTER_ROOT\"/packages/flutter_tools/bin/macos_assemble.sh && touch Flutter/ephemeral/tripwire"; + }; +/* End PBXShellScriptBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 331C80D1294CF70F00263BE5 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 331C80D8294CF71000263BE5 /* RunnerTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 33CC10E92044A3C60003C045 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 33CC11132044BFA00003C045 /* MainFlutterWindow.swift in Sources */, + 33CC10F12044A3C60003C045 /* AppDelegate.swift in Sources */, + 335BBD1B22A9A15E00E9071D /* GeneratedPluginRegistrant.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 331C80DA294CF71000263BE5 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 33CC10EC2044A3C60003C045 /* Runner */; + targetProxy = 331C80D9294CF71000263BE5 /* PBXContainerItemProxy */; + }; + 33CC11202044C79F0003C045 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 33CC111A2044C6BA0003C045 /* Flutter Assemble */; + targetProxy = 33CC111F2044C79F0003C045 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + 33CC10F42044A3C60003C045 /* MainMenu.xib */ = { + isa = PBXVariantGroup; + children = ( + 33CC10F52044A3C60003C045 /* Base */, + ); + name = MainMenu.xib; + path = Runner; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 331C80DB294CF71000263BE5 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/cyberhybridhub.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/cyberhybridhub"; + }; + name = Debug; + }; + 331C80DC294CF71000263BE5 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/cyberhybridhub.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/cyberhybridhub"; + }; + name = Release; + }; + 331C80DD294CF71000263BE5 /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + BUNDLE_LOADER = "$(TEST_HOST)"; + CURRENT_PROJECT_VERSION = 1; + GENERATE_INFOPLIST_FILE = YES; + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/cyberhybridhub.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/cyberhybridhub"; + }; + name = Profile; + }; + 338D0CE9231458BD00FA5F75 /* Profile */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; + 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_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + 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_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CODE_SIGN_IDENTITY = "-"; + COPY_PHASE_STRIP = NO; + DEAD_CODE_STRIPPING = YES; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu11; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + MACOSX_DEPLOYMENT_TARGET = 10.15; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = macosx; + SWIFT_COMPILATION_MODE = wholemodule; + SWIFT_OPTIMIZATION_LEVEL = "-O"; + }; + name = Profile; + }; + 338D0CEA231458BD00FA5F75 /* Profile */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 33E5194F232828860026EE4D /* AppInfo.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/DebugProfile.entitlements; + CODE_SIGN_STYLE = Automatic; + COMBINE_HIDPI_IMAGES = YES; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/../Frameworks", + ); + PROVISIONING_PROFILE_SPECIFIER = ""; + SWIFT_VERSION = 5.0; + }; + name = Profile; + }; + 338D0CEB231458BD00FA5F75 /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + CODE_SIGN_STYLE = Manual; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Profile; + }; + 33CC10F92044A3C60003C045 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; + 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_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + 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_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CODE_SIGN_IDENTITY = "-"; + COPY_PHASE_STRIP = NO; + DEAD_CODE_STRIPPING = YES; + DEBUG_INFORMATION_FORMAT = dwarf; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu11; + 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_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + MACOSX_DEPLOYMENT_TARGET = 10.15; + MTL_ENABLE_DEBUG_INFO = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = macosx; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + }; + name = Debug; + }; + 33CC10FA2044A3C60003C045 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; + 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_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + 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_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CODE_SIGN_IDENTITY = "-"; + COPY_PHASE_STRIP = NO; + DEAD_CODE_STRIPPING = YES; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu11; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + MACOSX_DEPLOYMENT_TARGET = 10.15; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = macosx; + SWIFT_COMPILATION_MODE = wholemodule; + SWIFT_OPTIMIZATION_LEVEL = "-O"; + }; + name = Release; + }; + 33CC10FC2044A3C60003C045 /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 33E5194F232828860026EE4D /* AppInfo.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/DebugProfile.entitlements; + CODE_SIGN_STYLE = Automatic; + COMBINE_HIDPI_IMAGES = YES; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/../Frameworks", + ); + PROVISIONING_PROFILE_SPECIFIER = ""; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + }; + name = Debug; + }; + 33CC10FD2044A3C60003C045 /* Release */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 33E5194F232828860026EE4D /* AppInfo.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Release.entitlements; + CODE_SIGN_STYLE = Automatic; + COMBINE_HIDPI_IMAGES = YES; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/../Frameworks", + ); + PROVISIONING_PROFILE_SPECIFIER = ""; + SWIFT_VERSION = 5.0; + }; + name = Release; + }; + 33CC111C2044C6BA0003C045 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + CODE_SIGN_STYLE = Manual; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Debug; + }; + 33CC111D2044C6BA0003C045 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + CODE_SIGN_STYLE = Automatic; + PRODUCT_NAME = "$(TARGET_NAME)"; + }; + name = Release; + }; +/* End XCBuildConfiguration section */ + +/* Begin XCConfigurationList section */ + 331C80DE294CF71000263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 331C80DB294CF71000263BE5 /* Debug */, + 331C80DC294CF71000263BE5 /* Release */, + 331C80DD294CF71000263BE5 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 33CC10E82044A3C60003C045 /* Build configuration list for PBXProject "Runner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 33CC10F92044A3C60003C045 /* Debug */, + 33CC10FA2044A3C60003C045 /* Release */, + 338D0CE9231458BD00FA5F75 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 33CC10FB2044A3C60003C045 /* Build configuration list for PBXNativeTarget "Runner" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 33CC10FC2044A3C60003C045 /* Debug */, + 33CC10FD2044A3C60003C045 /* Release */, + 338D0CEA231458BD00FA5F75 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; + 33CC111B2044C6BA0003C045 /* Build configuration list for PBXAggregateTarget "Flutter Assemble" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + 33CC111C2044C6BA0003C045 /* Debug */, + 33CC111D2044C6BA0003C045 /* Release */, + 338D0CEB231458BD00FA5F75 /* Profile */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; +/* End XCConfigurationList section */ + +/* Begin XCLocalSwiftPackageReference section */ + 781AD8BC2B33823900A9FFBB /* XCLocalSwiftPackageReference "Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage" */ = { + isa = XCLocalSwiftPackageReference; + relativePath = Flutter/ephemeral/Packages/FlutterGeneratedPluginSwiftPackage; + }; +/* End XCLocalSwiftPackageReference section */ + +/* Begin XCSwiftPackageProductDependency section */ + 78A3181F2AECB46A00862997 /* FlutterGeneratedPluginSwiftPackage */ = { + isa = XCSwiftPackageProductDependency; + productName = FlutterGeneratedPluginSwiftPackage; + }; +/* End XCSwiftPackageProductDependency section */ + }; + rootObject = 33CC10E52044A3C60003C045 /* Project object */; +} diff --git a/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 0000000..18d9810 --- /dev/null +++ b/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme new file mode 100644 index 0000000..84b467e --- /dev/null +++ b/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -0,0 +1,117 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/macos/Runner.xcworkspace/contents.xcworkspacedata b/macos/Runner.xcworkspace/contents.xcworkspacedata new file mode 100644 index 0000000..1d526a1 --- /dev/null +++ b/macos/Runner.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 0000000..18d9810 --- /dev/null +++ b/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/macos/Runner/AppDelegate.swift b/macos/Runner/AppDelegate.swift new file mode 100644 index 0000000..b3c1761 --- /dev/null +++ b/macos/Runner/AppDelegate.swift @@ -0,0 +1,13 @@ +import Cocoa +import FlutterMacOS + +@main +class AppDelegate: FlutterAppDelegate { + override func applicationShouldTerminateAfterLastWindowClosed(_ sender: NSApplication) -> Bool { + return true + } + + override func applicationSupportsSecureRestorableState(_ app: NSApplication) -> Bool { + return true + } +} diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 0000000..a2ec33f --- /dev/null +++ b/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,68 @@ +{ + "images" : [ + { + "size" : "16x16", + "idiom" : "mac", + "filename" : "app_icon_16.png", + "scale" : "1x" + }, + { + "size" : "16x16", + "idiom" : "mac", + "filename" : "app_icon_32.png", + "scale" : "2x" + }, + { + "size" : "32x32", + "idiom" : "mac", + "filename" : "app_icon_32.png", + "scale" : "1x" + }, + { + "size" : "32x32", + "idiom" : "mac", + "filename" : "app_icon_64.png", + "scale" : "2x" + }, + { + "size" : "128x128", + "idiom" : "mac", + "filename" : "app_icon_128.png", + "scale" : "1x" + }, + { + "size" : "128x128", + "idiom" : "mac", + "filename" : "app_icon_256.png", + "scale" : "2x" + }, + { + "size" : "256x256", + "idiom" : "mac", + "filename" : "app_icon_256.png", + "scale" : "1x" + }, + { + "size" : "256x256", + "idiom" : "mac", + "filename" : "app_icon_512.png", + "scale" : "2x" + }, + { + "size" : "512x512", + "idiom" : "mac", + "filename" : "app_icon_512.png", + "scale" : "1x" + }, + { + "size" : "512x512", + "idiom" : "mac", + "filename" : "app_icon_1024.png", + "scale" : "2x" + } + ], + "info" : { + "version" : 1, + "author" : "xcode" + } +} diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png new file mode 100644 index 0000000..82b6f9d Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png new file mode 100644 index 0000000..13b35eb Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png new file mode 100644 index 0000000..0a3f5fa Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png new file mode 100644 index 0000000..bdb5722 Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png new file mode 100644 index 0000000..f083318 Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png new file mode 100644 index 0000000..326c0e7 Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png differ diff --git a/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png new file mode 100644 index 0000000..2f1632c Binary files /dev/null and b/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png differ diff --git a/macos/Runner/Base.lproj/MainMenu.xib b/macos/Runner/Base.lproj/MainMenu.xib new file mode 100644 index 0000000..80e867a --- /dev/null +++ b/macos/Runner/Base.lproj/MainMenu.xib @@ -0,0 +1,343 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/macos/Runner/Configs/AppInfo.xcconfig b/macos/Runner/Configs/AppInfo.xcconfig new file mode 100644 index 0000000..df19d5a --- /dev/null +++ b/macos/Runner/Configs/AppInfo.xcconfig @@ -0,0 +1,14 @@ +// Application-level settings for the Runner target. +// +// This may be replaced with something auto-generated from metadata (e.g., pubspec.yaml) in the +// future. If not, the values below would default to using the project name when this becomes a +// 'flutter create' template. + +// The application's name. By default this is also the title of the Flutter window. +PRODUCT_NAME = cyberhybridhub + +// The application's bundle identifier +PRODUCT_BUNDLE_IDENTIFIER = com.cyberhybridhub.cyberhybridhub + +// The copyright displayed in application information +PRODUCT_COPYRIGHT = Copyright © 2026 com.cyberhybridhub. All rights reserved. diff --git a/macos/Runner/Configs/Debug.xcconfig b/macos/Runner/Configs/Debug.xcconfig new file mode 100644 index 0000000..36b0fd9 --- /dev/null +++ b/macos/Runner/Configs/Debug.xcconfig @@ -0,0 +1,2 @@ +#include "../../Flutter/Flutter-Debug.xcconfig" +#include "Warnings.xcconfig" diff --git a/macos/Runner/Configs/Release.xcconfig b/macos/Runner/Configs/Release.xcconfig new file mode 100644 index 0000000..dff4f49 --- /dev/null +++ b/macos/Runner/Configs/Release.xcconfig @@ -0,0 +1,2 @@ +#include "../../Flutter/Flutter-Release.xcconfig" +#include "Warnings.xcconfig" diff --git a/macos/Runner/Configs/Warnings.xcconfig b/macos/Runner/Configs/Warnings.xcconfig new file mode 100644 index 0000000..42bcbf4 --- /dev/null +++ b/macos/Runner/Configs/Warnings.xcconfig @@ -0,0 +1,13 @@ +WARNING_CFLAGS = -Wall -Wconditional-uninitialized -Wnullable-to-nonnull-conversion -Wmissing-method-return-type -Woverlength-strings +GCC_WARN_UNDECLARED_SELECTOR = YES +CLANG_UNDEFINED_BEHAVIOR_SANITIZER_NULLABILITY = YES +CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE +CLANG_WARN__DUPLICATE_METHOD_MATCH = YES +CLANG_WARN_PRAGMA_PACK = YES +CLANG_WARN_STRICT_PROTOTYPES = YES +CLANG_WARN_COMMA = YES +GCC_WARN_STRICT_SELECTOR_MATCH = YES +CLANG_WARN_OBJC_REPEATED_USE_OF_WEAK = YES +CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES +GCC_WARN_SHADOW = YES +CLANG_WARN_UNREACHABLE_CODE = YES diff --git a/macos/Runner/DebugProfile.entitlements b/macos/Runner/DebugProfile.entitlements new file mode 100644 index 0000000..dddb8a3 --- /dev/null +++ b/macos/Runner/DebugProfile.entitlements @@ -0,0 +1,12 @@ + + + + + com.apple.security.app-sandbox + + com.apple.security.cs.allow-jit + + com.apple.security.network.server + + + diff --git a/macos/Runner/Info.plist b/macos/Runner/Info.plist new file mode 100644 index 0000000..4789daa --- /dev/null +++ b/macos/Runner/Info.plist @@ -0,0 +1,32 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIconFile + + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + $(PRODUCT_NAME) + CFBundlePackageType + APPL + CFBundleShortVersionString + $(FLUTTER_BUILD_NAME) + CFBundleVersion + $(FLUTTER_BUILD_NUMBER) + LSMinimumSystemVersion + $(MACOSX_DEPLOYMENT_TARGET) + NSHumanReadableCopyright + $(PRODUCT_COPYRIGHT) + NSMainNibFile + MainMenu + NSPrincipalClass + NSApplication + + diff --git a/macos/Runner/MainFlutterWindow.swift b/macos/Runner/MainFlutterWindow.swift new file mode 100644 index 0000000..3cc05eb --- /dev/null +++ b/macos/Runner/MainFlutterWindow.swift @@ -0,0 +1,15 @@ +import Cocoa +import FlutterMacOS + +class MainFlutterWindow: NSWindow { + override func awakeFromNib() { + let flutterViewController = FlutterViewController() + let windowFrame = self.frame + self.contentViewController = flutterViewController + self.setFrame(windowFrame, display: true) + + RegisterGeneratedPlugins(registry: flutterViewController) + + super.awakeFromNib() + } +} diff --git a/macos/Runner/Release.entitlements b/macos/Runner/Release.entitlements new file mode 100644 index 0000000..852fa1a --- /dev/null +++ b/macos/Runner/Release.entitlements @@ -0,0 +1,8 @@ + + + + + com.apple.security.app-sandbox + + + diff --git a/macos/RunnerTests/RunnerTests.swift b/macos/RunnerTests/RunnerTests.swift new file mode 100644 index 0000000..61f3bd1 --- /dev/null +++ b/macos/RunnerTests/RunnerTests.swift @@ -0,0 +1,12 @@ +import Cocoa +import FlutterMacOS +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/pubspec.lock b/pubspec.lock new file mode 100644 index 0000000..d67b05b --- /dev/null +++ b/pubspec.lock @@ -0,0 +1,1090 @@ +# Generated by pub +# See https://dart.dev/tools/pub/glossary#lockfile +packages: + _discoveryapis_commons: + dependency: transitive + description: + name: _discoveryapis_commons + sha256: "113c4100b90a5b70a983541782431b82168b3cae166ab130649c36eb3559d498" + url: "https://pub.dev" + source: hosted + version: "1.0.7" + _fe_analyzer_shared: + dependency: transitive + description: + name: _fe_analyzer_shared + sha256: "3b19a47f6ea7c2632760777c78174f47f6aec1e05f0cd611380d4593b8af1dbc" + url: "https://pub.dev" + source: hosted + version: "96.0.0" + _flutterfire_internals: + dependency: transitive + description: + name: _flutterfire_internals + sha256: "8f89e371e2883de35cdc78f648e725fa4da5f3b6c927269f00fa68f1ea92b598" + url: "https://pub.dev" + source: hosted + version: "1.3.71" + analyzer: + dependency: transitive + description: + name: analyzer + sha256: "0c516bc4ad36a1a75759e54d5047cb9d15cded4459df01aa35a0b5ec7db2c2a0" + url: "https://pub.dev" + source: hosted + version: "10.2.0" + args: + dependency: transitive + description: + name: args + sha256: d0481093c50b1da8910eb0bb301626d4d8eb7284aa739614d2b394ee09e3ea04 + url: "https://pub.dev" + source: hosted + version: "2.7.0" + async: + dependency: transitive + description: + name: async + sha256: e2eb0491ba5ddb6177742d2da23904574082139b07c1e33b8503b9f46f3e1a37 + url: "https://pub.dev" + source: hosted + version: "2.13.1" + boolean_selector: + dependency: transitive + description: + name: boolean_selector + sha256: "8aab1771e1243a5063b8b0ff68042d67334e3feab9e95b9490f9a6ebf73b42ea" + url: "https://pub.dev" + source: hosted + version: "2.1.2" + build: + dependency: transitive + description: + name: build + sha256: a156715e7cd728130c592f30552575908aae5b100005fbc1f0fb16b3c03a3d10 + url: "https://pub.dev" + source: hosted + version: "4.0.6" + build_config: + dependency: transitive + description: + name: build_config + sha256: "4070d2a59f8eec34c97c86ceb44403834899075f66e8a9d59706f8e7834f6f71" + url: "https://pub.dev" + source: hosted + version: "1.3.0" + build_daemon: + dependency: transitive + description: + name: build_daemon + sha256: bf05f6e12cfea92d3c09308d7bcdab1906cd8a179b023269eed00c071004b957 + url: "https://pub.dev" + source: hosted + version: "4.1.1" + build_runner: + dependency: "direct dev" + description: + name: build_runner + sha256: "1523ce62448ebac2c15a8ba5fbad8acac169788658a7dd2a1c2d9c2a9318b9a6" + url: "https://pub.dev" + source: hosted + version: "2.15.0" + built_collection: + dependency: transitive + description: + name: built_collection + sha256: "376e3dd27b51ea877c28d525560790aee2e6fbb5f20e2f85d5081027d94e2100" + url: "https://pub.dev" + source: hosted + version: "5.1.1" + built_value: + dependency: transitive + description: + name: built_value + sha256: "34e4067d30ce212937df995f03b69992eea683539ceeac7f679a1f1eba055b56" + url: "https://pub.dev" + source: hosted + version: "8.12.6" + characters: + dependency: transitive + description: + name: characters + sha256: faf38497bda5ead2a8c7615f4f7939df04333478bf32e4173fcb06d428b5716b + url: "https://pub.dev" + source: hosted + version: "1.4.1" + charcode: + dependency: transitive + description: + name: charcode + sha256: fb0f1107cac15a5ea6ef0a6ef71a807b9e4267c713bb93e00e92d737cc8dbd8a + url: "https://pub.dev" + source: hosted + version: "1.4.0" + checked_yaml: + dependency: transitive + description: + name: checked_yaml + sha256: "959525d3162f249993882720d52b7e0c833978df229be20702b33d48d91de70f" + url: "https://pub.dev" + source: hosted + version: "2.0.4" + cli_util: + dependency: transitive + description: + name: cli_util + sha256: ff6785f7e9e3c38ac98b2fb035701789de90154024a75b6cb926445e83197d1c + url: "https://pub.dev" + source: hosted + version: "0.4.2" + clock: + dependency: transitive + description: + name: clock + sha256: fddb70d9b5277016c77a80201021d40a2247104d9f4aa7bab7157b7e3f05b84b + url: "https://pub.dev" + source: hosted + version: "1.1.2" + code_assets: + dependency: transitive + description: + name: code_assets + sha256: "83ccdaa064c980b5596c35dd64a8d3ecc68620174ab9b90b6343b753aa721687" + url: "https://pub.dev" + source: hosted + version: "1.0.0" + collection: + dependency: transitive + description: + name: collection + sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" + url: "https://pub.dev" + source: hosted + version: "1.19.1" + connectivity_plus: + dependency: "direct main" + description: + name: connectivity_plus + sha256: b5e72753cf63becce2c61fd04dfe0f1c430cc5278b53a1342dc5ad839eab29ec + url: "https://pub.dev" + source: hosted + version: "6.1.5" + connectivity_plus_platform_interface: + dependency: transitive + description: + name: connectivity_plus_platform_interface + sha256: "3c09627c536d22fd24691a905cdd8b14520de69da52c7a97499c8be5284a32ed" + url: "https://pub.dev" + source: hosted + version: "2.1.0" + convert: + dependency: transitive + description: + name: convert + sha256: b30acd5944035672bc15c6b7a8b47d773e41e2f17de064350988c5d02adb1c68 + url: "https://pub.dev" + source: hosted + version: "3.1.2" + crypto: + dependency: transitive + description: + name: crypto + sha256: c8ea0233063ba03258fbcf2ca4d6dadfefe14f02fab57702265467a19f27fadf + url: "https://pub.dev" + source: hosted + version: "3.0.7" + cupertino_icons: + dependency: "direct main" + description: + name: cupertino_icons + sha256: "41e005c33bd814be4d3096aff55b1908d419fde52ca656c8c47719ec745873cd" + url: "https://pub.dev" + source: hosted + version: "1.0.9" + dart_style: + dependency: transitive + description: + name: dart_style + sha256: "29f7ecc274a86d32920b1d9cfc7502fa87220da41ec60b55f329559d5732e2b2" + url: "https://pub.dev" + source: hosted + version: "3.1.7" + dbus: + dependency: transitive + description: + name: dbus + sha256: d0c98dcd4f5169878b6cf8f6e0a52403a9dff371a3e2f019697accbf6f44a270 + url: "https://pub.dev" + source: hosted + version: "0.7.12" + drift: + dependency: "direct main" + description: + name: drift + sha256: "970cd188fddb111b26ea6a9b07a62bf5c2432d74147b8122c67044ae3b97e99e" + url: "https://pub.dev" + source: hosted + version: "2.31.0" + drift_dev: + dependency: "direct dev" + description: + name: drift_dev + sha256: "917184b2fb867b70a548a83bf0d36268423b38d39968c06cce4905683da49587" + url: "https://pub.dev" + source: hosted + version: "2.31.0" + drift_flutter: + dependency: "direct main" + description: + name: drift_flutter + sha256: c07120854742a0cae2f7501a0da02493addde550db6641d284983c08762e60a7 + url: "https://pub.dev" + source: hosted + version: "0.2.8" + extension_google_sign_in_as_googleapis_auth: + dependency: transitive + description: + name: extension_google_sign_in_as_googleapis_auth + sha256: "8a9c887a377ee5b990b2b29be229de8401a43002a7d0d0b29f7d014c1c6bfeac" + url: "https://pub.dev" + source: hosted + version: "3.0.0" + fake_async: + dependency: transitive + description: + name: fake_async + sha256: "5368f224a74523e8d2e7399ea1638b37aecfca824a3cc4dfdf77bf1fa905ac44" + url: "https://pub.dev" + source: hosted + version: "1.3.3" + ffi: + dependency: transitive + description: + name: ffi + sha256: "6d7fd89431262d8f3125e81b50d3847a091d846eafcd4fdb88dd06f36d705a45" + url: "https://pub.dev" + source: hosted + version: "2.2.0" + file: + dependency: transitive + description: + name: file + sha256: a3b4f84adafef897088c160faf7dfffb7696046cb13ae90b508c2cbc95d3b8d4 + url: "https://pub.dev" + source: hosted + version: "7.0.1" + firebase_auth: + dependency: "direct main" + description: + name: firebase_auth + sha256: "2df96699cf10c6125a33059f33100daff539906195bccaced7cb0002bdeb860e" + url: "https://pub.dev" + source: hosted + version: "6.5.1" + firebase_auth_platform_interface: + dependency: transitive + description: + name: firebase_auth_platform_interface + sha256: f37e163b063edbb62b911f100edb4546b10ca06857f879f5c3eb8d9cb19c275b + url: "https://pub.dev" + source: hosted + version: "9.0.1" + firebase_auth_web: + dependency: transitive + description: + name: firebase_auth_web + sha256: "82fdbe2126b2dd6b8635fb091e7c86bebc58ef5d7da350d623b47a8685dd0947" + url: "https://pub.dev" + source: hosted + version: "6.2.1" + firebase_core: + dependency: "direct main" + description: + name: firebase_core + sha256: "93a5bde9775fd5adcc937f39dfa04ae0bc89c4d79bea6abc49de3f7b049d9ff6" + url: "https://pub.dev" + source: hosted + version: "4.9.0" + firebase_core_platform_interface: + dependency: transitive + description: + name: firebase_core_platform_interface + sha256: "4a120366dbf7d5a8ee9438978530b664b855728fb8dcc3a201017660817e555b" + url: "https://pub.dev" + source: hosted + version: "7.0.1" + firebase_core_web: + dependency: transitive + description: + name: firebase_core_web + sha256: "7c98f10b8c8e5adedc0b810b66a877120696675e2c22d9ca9caca092da0d9e57" + url: "https://pub.dev" + source: hosted + version: "3.7.0" + fixnum: + dependency: transitive + description: + name: fixnum + sha256: b6dc7065e46c974bc7c5f143080a6764ec7a4be6da1285ececdc37be96de53be + url: "https://pub.dev" + source: hosted + version: "1.1.1" + flutter: + dependency: "direct main" + description: flutter + source: sdk + version: "0.0.0" + flutter_lints: + dependency: "direct dev" + description: + name: flutter_lints + sha256: "3105dc8492f6183fb076ccf1f351ac3d60564bff92e20bfc4af9cc1651f4e7e1" + url: "https://pub.dev" + source: hosted + version: "6.0.0" + flutter_test: + dependency: "direct dev" + description: flutter + source: sdk + version: "0.0.0" + flutter_web_plugins: + dependency: transitive + description: flutter + source: sdk + version: "0.0.0" + glob: + dependency: transitive + description: + name: glob + sha256: c3f1ee72c96f8f78935e18aa8cecced9ab132419e8625dc187e1c2408efc20de + url: "https://pub.dev" + source: hosted + version: "2.1.3" + google_cloud: + dependency: transitive + description: + name: google_cloud + sha256: b385e20726ef5315d302c5933bfb728103116c5be2d3d17094b01a82da538c1f + url: "https://pub.dev" + source: hosted + version: "0.5.0" + google_identity_services_web: + dependency: transitive + description: + name: google_identity_services_web + sha256: "5d187c46dc59e02646e10fe82665fc3884a9b71bc1c90c2b8b749316d33ee454" + url: "https://pub.dev" + source: hosted + version: "0.3.3+1" + google_sign_in: + dependency: "direct main" + description: + name: google_sign_in + sha256: "521031b65853b4409b8213c0387d57edaad7e2a949ce6dea0d8b2afc9cb29763" + url: "https://pub.dev" + source: hosted + version: "7.2.0" + google_sign_in_all_platforms: + dependency: "direct main" + description: + name: google_sign_in_all_platforms + sha256: "3faab55abf133ee5650de9afec4d63663fbe1b99a62bb924f71c73b907a97777" + url: "https://pub.dev" + source: hosted + version: "2.0.2" + google_sign_in_all_platforms_desktop: + dependency: transitive + description: + name: google_sign_in_all_platforms_desktop + sha256: d0ad9f7ed30eb7d8aa5a781ef58d4f1c9c7ad9d7634e27652411436fb6b2c670 + url: "https://pub.dev" + source: hosted + version: "0.2.0" + google_sign_in_all_platforms_interface: + dependency: transitive + description: + name: google_sign_in_all_platforms_interface + sha256: bf9b2a17067d2e023862cd565a076555c8de088c64e4d35ee68826cad0ad972d + url: "https://pub.dev" + source: hosted + version: "0.2.1" + google_sign_in_all_platforms_mobile: + dependency: transitive + description: + name: google_sign_in_all_platforms_mobile + sha256: "4d09eaf2676bf44925550fc9346ea8e6ff3a32c9885c566025bf0e90feab2aee" + url: "https://pub.dev" + source: hosted + version: "0.2.1" + google_sign_in_all_platforms_web: + dependency: transitive + description: + name: google_sign_in_all_platforms_web + sha256: "1af470d499396a522740044f70906d3e220a7da40c2976d5a46ec0ede5d2cfb1" + url: "https://pub.dev" + source: hosted + version: "0.1.0" + google_sign_in_android: + dependency: transitive + description: + name: google_sign_in_android + sha256: df5c15533814ed20b7d9e1c5a40a73f174e5d5017bd2669b1c72fb6596fde812 + url: "https://pub.dev" + source: hosted + version: "7.2.11" + google_sign_in_ios: + dependency: transitive + description: + name: google_sign_in_ios + sha256: ac1e4c1205267cb7999d1d81333fccffdfda29e853f434bbaf71525498bb6950 + url: "https://pub.dev" + source: hosted + version: "6.3.0" + google_sign_in_platform_interface: + dependency: transitive + description: + name: google_sign_in_platform_interface + sha256: "7f59208c42b415a3cca203571128d6f84f885fead2d5b53eb65a9e27f2965bb5" + url: "https://pub.dev" + source: hosted + version: "3.1.0" + google_sign_in_web: + dependency: transitive + description: + name: google_sign_in_web + sha256: d473003eeca892f96a01a64fc803378be765071cb0c265ee872c7f8683245d14 + url: "https://pub.dev" + source: hosted + version: "1.1.3" + googleapis: + dependency: transitive + description: + name: googleapis + sha256: "5c9e0f25be1dec13d8d2158263141104c51b5ba83487537c17a2330581e505ee" + url: "https://pub.dev" + source: hosted + version: "14.0.0" + googleapis_auth: + dependency: transitive + description: + name: googleapis_auth + sha256: "2a8895c3885197f96bb2fd91ee0ae77b53ff3874c7b1f1eadb6566248e880958" + url: "https://pub.dev" + source: hosted + version: "2.3.1" + graphs: + dependency: transitive + description: + name: graphs + sha256: "741bbf84165310a68ff28fe9e727332eef1407342fca52759cb21ad8177bb8d0" + url: "https://pub.dev" + source: hosted + version: "2.3.2" + hooks: + dependency: transitive + description: + name: hooks + sha256: "025f060e86d2d4c3c47b56e33caf7f93bf9283340f26d23424ebcfccf34f621e" + url: "https://pub.dev" + source: hosted + version: "1.0.3" + http: + dependency: "direct main" + description: + name: http + sha256: "87721a4a50b19c7f1d49001e51409bddc46303966ce89a65af4f4e6004896412" + url: "https://pub.dev" + source: hosted + version: "1.6.0" + http_methods: + dependency: transitive + description: + name: http_methods + sha256: "6bccce8f1ec7b5d701e7921dca35e202d425b57e317ba1a37f2638590e29e566" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + http_multi_server: + dependency: transitive + description: + name: http_multi_server + sha256: aa6199f908078bb1c5efb8d8638d4ae191aac11b311132c3ef48ce352fb52ef8 + url: "https://pub.dev" + source: hosted + version: "3.2.2" + http_parser: + dependency: transitive + description: + name: http_parser + sha256: "178d74305e7866013777bab2c3d8726205dc5a4dd935297175b19a23a2e66571" + url: "https://pub.dev" + source: hosted + version: "4.1.2" + io: + dependency: transitive + description: + name: io + sha256: dfd5a80599cf0165756e3181807ed3e77daf6dd4137caaad72d0b7931597650b + url: "https://pub.dev" + source: hosted + version: "1.0.5" + jni: + dependency: transitive + description: + name: jni + sha256: c2230682d5bc2362c1c9e8d3c7f406d9cbba23ab3f2e203a025dd47e0fb2e68f + url: "https://pub.dev" + source: hosted + version: "1.0.0" + jni_flutter: + dependency: transitive + description: + name: jni_flutter + sha256: "8b59e590786050b1cd866677dddaf76b1ade5e7bc751abe04b86e84d379d3ba6" + url: "https://pub.dev" + source: hosted + version: "1.0.1" + json_annotation: + dependency: transitive + description: + name: json_annotation + sha256: "2a743920d81b7910627f68ee2c9ac1fc0bfee32b9fc3403587d7c6791ca12f80" + url: "https://pub.dev" + source: hosted + version: "4.12.0" + leak_tracker: + dependency: transitive + description: + name: leak_tracker + sha256: "33e2e26bdd85a0112ec15400c8cbffea70d0f9c3407491f672a2fad47915e2de" + url: "https://pub.dev" + source: hosted + version: "11.0.2" + leak_tracker_flutter_testing: + dependency: transitive + description: + name: leak_tracker_flutter_testing + sha256: "1dbc140bb5a23c75ea9c4811222756104fbcd1a27173f0c34ca01e16bea473c1" + url: "https://pub.dev" + source: hosted + version: "3.0.10" + leak_tracker_testing: + dependency: transitive + description: + name: leak_tracker_testing + sha256: "8d5a2d49f4a66b49744b23b018848400d23e54caf9463f4eb20df3eb8acb2eb1" + url: "https://pub.dev" + source: hosted + version: "3.0.2" + lints: + dependency: transitive + description: + name: lints + sha256: "12f842a479589fea194fe5c5a3095abc7be0c1f2ddfa9a0e76aed1dbd26a87df" + url: "https://pub.dev" + source: hosted + version: "6.1.0" + logging: + dependency: transitive + description: + name: logging + sha256: c8245ada5f1717ed44271ed1c26b8ce85ca3228fd2ffdb75468ab01979309d61 + url: "https://pub.dev" + source: hosted + version: "1.3.0" + matcher: + dependency: transitive + description: + name: matcher + sha256: dc0b7dc7651697ea4ff3e69ef44b0407ea32c487a39fff6a4004fa585e901861 + url: "https://pub.dev" + source: hosted + version: "0.12.19" + material_color_utilities: + dependency: transitive + description: + name: material_color_utilities + sha256: "9c337007e82b1889149c82ed242ed1cb24a66044e30979c44912381e9be4c48b" + url: "https://pub.dev" + source: hosted + version: "0.13.0" + meta: + dependency: transitive + description: + name: meta + sha256: "1741988757a65eb6b36abe716829688cf01910bbf91c34354ff7ec1c3de2b349" + url: "https://pub.dev" + source: hosted + version: "1.18.0" + mime: + dependency: transitive + description: + name: mime + sha256: "41a20518f0cb1256669420fdba0cd90d21561e560ac240f26ef8322e45bb7ed6" + url: "https://pub.dev" + source: hosted + version: "2.0.0" + nm: + dependency: transitive + description: + name: nm + sha256: "2c9aae4127bdc8993206464fcc063611e0e36e72018696cd9631023a31b24254" + url: "https://pub.dev" + source: hosted + version: "0.5.0" + objective_c: + dependency: transitive + description: + name: objective_c + sha256: "02f4ff97ac95fa4abe561242eb033239e08e8903a46eaecf61b047761b89fe1f" + url: "https://pub.dev" + source: hosted + version: "9.4.0" + package_config: + dependency: transitive + description: + name: package_config + sha256: f096c55ebb7deb7e384101542bfba8c52696c1b56fca2eb62827989ef2353bbc + url: "https://pub.dev" + source: hosted + version: "2.2.0" + path: + dependency: "direct main" + description: + name: path + sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" + url: "https://pub.dev" + source: hosted + version: "1.9.1" + path_provider: + dependency: "direct main" + description: + name: path_provider + sha256: "50c5dd5b6e1aaf6fb3a78b33f6aa3afca52bf903a8a5298f53101fdaee55bbcd" + url: "https://pub.dev" + source: hosted + version: "2.1.5" + path_provider_android: + dependency: transitive + description: + name: path_provider_android + sha256: "69cbd515a62b94d32a7944f086b2f82b4ac40a1d45bebfc00813a430ab2dabcd" + url: "https://pub.dev" + source: hosted + version: "2.3.1" + path_provider_foundation: + dependency: transitive + description: + name: path_provider_foundation + sha256: "2a376b7d6392d80cd3705782d2caa734ca4727776db0b6ec36ef3f1855197699" + url: "https://pub.dev" + source: hosted + version: "2.6.0" + path_provider_linux: + dependency: transitive + description: + name: path_provider_linux + sha256: f7a1fe3a634fe7734c8d3f2766ad746ae2a2884abe22e241a8b301bf5cac3279 + url: "https://pub.dev" + source: hosted + version: "2.2.1" + path_provider_platform_interface: + dependency: transitive + description: + name: path_provider_platform_interface + sha256: "88f5779f72ba699763fa3a3b06aa4bf6de76c8e5de842cf6f29e2e06476c2334" + url: "https://pub.dev" + source: hosted + version: "2.1.2" + path_provider_windows: + dependency: transitive + description: + name: path_provider_windows + sha256: bd6f00dbd873bfb70d0761682da2b3a2c2fccc2b9e84c495821639601d81afe7 + url: "https://pub.dev" + source: hosted + version: "2.3.0" + petitparser: + dependency: transitive + description: + name: petitparser + sha256: "91bd59303e9f769f108f8df05e371341b15d59e995e6806aefab827b58336675" + url: "https://pub.dev" + source: hosted + version: "7.0.2" + platform: + dependency: transitive + description: + name: platform + sha256: "5d6b1b0036a5f331ebc77c850ebc8506cbc1e9416c27e59b439f917a902a4984" + url: "https://pub.dev" + source: hosted + version: "3.1.6" + plugin_platform_interface: + dependency: transitive + description: + name: plugin_platform_interface + sha256: "4820fbfdb9478b1ebae27888254d445073732dae3d6ea81f0b7e06d5dedc3f02" + url: "https://pub.dev" + source: hosted + version: "2.1.8" + pool: + dependency: transitive + description: + name: pool + sha256: "978783255c543aa3586a1b3c21f6e9d720eb315376a915872c61ef8b5c20177d" + url: "https://pub.dev" + source: hosted + version: "1.5.2" + pub_semver: + dependency: transitive + description: + name: pub_semver + sha256: "5bfcf68ca79ef689f8990d1160781b4bad40a3bd5e5218ad4076ddb7f4081585" + url: "https://pub.dev" + source: hosted + version: "2.2.0" + pubspec_parse: + dependency: transitive + description: + name: pubspec_parse + sha256: "0560ba233314abbed0a48a2956f7f022cce7c3e1e73df540277da7544cad4082" + url: "https://pub.dev" + source: hosted + version: "1.5.0" + recase: + dependency: transitive + description: + name: recase + sha256: e4eb4ec2dcdee52dcf99cb4ceabaffc631d7424ee55e56f280bc039737f89213 + url: "https://pub.dev" + source: hosted + version: "4.1.0" + record_use: + dependency: transitive + description: + name: record_use + sha256: "2551bd8eecfe95d14ae75f6021ad0248be5c27f138c2ec12fcb52b500b3ba1ed" + url: "https://pub.dev" + source: hosted + version: "0.6.0" + shared_preferences: + dependency: transitive + description: + name: shared_preferences + sha256: c3025c5534b01739267eb7d76959bbc25a6d10f6988e1c2a3036940133dd10bf + url: "https://pub.dev" + source: hosted + version: "2.5.5" + shared_preferences_android: + dependency: transitive + description: + name: shared_preferences_android + sha256: e8d4762b1e2e8578fc4d0fd548cebf24afd24f49719c08974df92834565e2c53 + url: "https://pub.dev" + source: hosted + version: "2.4.23" + shared_preferences_foundation: + dependency: transitive + description: + name: shared_preferences_foundation + sha256: "4e7eaffc2b17ba398759f1151415869a34771ba11ebbccd1b0145472a619a64f" + url: "https://pub.dev" + source: hosted + version: "2.5.6" + shared_preferences_linux: + dependency: transitive + description: + name: shared_preferences_linux + sha256: "580abfd40f415611503cae30adf626e6656dfb2f0cee8f465ece7b6defb40f2f" + url: "https://pub.dev" + source: hosted + version: "2.4.1" + shared_preferences_platform_interface: + dependency: transitive + description: + name: shared_preferences_platform_interface + sha256: "649dc798a33931919ea356c4305c2d1f81619ea6e92244070b520187b5140ef9" + url: "https://pub.dev" + source: hosted + version: "2.4.2" + shared_preferences_web: + dependency: transitive + description: + name: shared_preferences_web + sha256: c49bd060261c9a3f0ff445892695d6212ff603ef3115edbb448509d407600019 + url: "https://pub.dev" + source: hosted + version: "2.4.3" + shared_preferences_windows: + dependency: transitive + description: + name: shared_preferences_windows + sha256: "94ef0f72b2d71bc3e700e025db3710911bd51a71cefb65cc609dd0d9a982e3c1" + url: "https://pub.dev" + source: hosted + version: "2.4.1" + shelf: + dependency: transitive + description: + name: shelf + sha256: e7dd780a7ffb623c57850b33f43309312fc863fb6aa3d276a754bb299839ef12 + url: "https://pub.dev" + source: hosted + version: "1.4.2" + shelf_router: + dependency: transitive + description: + name: shelf_router + sha256: f5e5d492440a7fb165fe1e2e1a623f31f734d3370900070b2b1e0d0428d59864 + url: "https://pub.dev" + source: hosted + version: "1.1.4" + shelf_web_socket: + dependency: transitive + description: + name: shelf_web_socket + sha256: "3632775c8e90d6c9712f883e633716432a27758216dfb61bd86a8321c0580925" + url: "https://pub.dev" + source: hosted + version: "3.0.0" + sky_engine: + dependency: transitive + description: flutter + source: sdk + version: "0.0.0" + source_gen: + dependency: transitive + description: + name: source_gen + sha256: ec37cc0e6694374cbef59ed79685572c870a54ede6fa30a3e420feb3adffea02 + url: "https://pub.dev" + source: hosted + version: "4.2.3" + source_span: + dependency: transitive + description: + name: source_span + sha256: "56a02f1f4cd1a2d96303c0144c93bd6d909eea6bee6bf5a0e0b685edbd4c47ab" + url: "https://pub.dev" + source: hosted + version: "1.10.2" + sqlite3: + dependency: transitive + description: + name: sqlite3 + sha256: "3145bd74dcdb4fd6f5c6dda4d4e4490a8087d7f286a14dee5d37087290f0f8a2" + url: "https://pub.dev" + source: hosted + version: "2.9.4" + sqlite3_flutter_libs: + dependency: "direct main" + description: + name: sqlite3_flutter_libs + sha256: eeb9e3a45207649076b808f8a5a74d68770d0b7f26ccef6d5f43106eee5375ad + url: "https://pub.dev" + source: hosted + version: "0.5.42" + sqlparser: + dependency: transitive + description: + name: sqlparser + sha256: "337e9997f7141ffdd054259128553c348635fa318f7ca492f07a4ab76f850d19" + url: "https://pub.dev" + source: hosted + version: "0.43.1" + stack_trace: + dependency: transitive + description: + name: stack_trace + sha256: "8b27215b45d22309b5cddda1aa2b19bdfec9df0e765f2de506401c071d38d1b1" + url: "https://pub.dev" + source: hosted + version: "1.12.1" + stream_channel: + dependency: transitive + description: + name: stream_channel + sha256: "969e04c80b8bcdf826f8f16579c7b14d780458bd97f56d107d3950fdbeef059d" + url: "https://pub.dev" + source: hosted + version: "2.1.4" + stream_transform: + dependency: transitive + description: + name: stream_transform + sha256: ad47125e588cfd37a9a7f86c7d6356dde8dfe89d071d293f80ca9e9273a33871 + url: "https://pub.dev" + source: hosted + version: "2.1.1" + string_scanner: + dependency: transitive + description: + name: string_scanner + sha256: "921cd31725b72fe181906c6a94d987c78e3b98c2e205b397ea399d4054872b43" + url: "https://pub.dev" + source: hosted + version: "1.4.1" + term_glyph: + dependency: transitive + description: + name: term_glyph + sha256: "7f554798625ea768a7518313e58f83891c7f5024f88e46e7182a4558850a4b8e" + url: "https://pub.dev" + source: hosted + version: "1.2.2" + test_api: + dependency: transitive + description: + name: test_api + sha256: "949a932224383300f01be9221c39180316445ecb8e7547f70a41a35bf421fb9e" + url: "https://pub.dev" + source: hosted + version: "0.7.11" + typed_data: + dependency: transitive + description: + name: typed_data + sha256: f9049c039ebfeb4cf7a7104a675823cd72dba8297f264b6637062516699fa006 + url: "https://pub.dev" + source: hosted + version: "1.4.0" + url_launcher: + dependency: transitive + description: + name: url_launcher + sha256: f6a7e5c4835bb4e3026a04793a4199ca2d14c739ec378fdfe23fc8075d0439f8 + url: "https://pub.dev" + source: hosted + version: "6.3.2" + url_launcher_android: + dependency: transitive + description: + name: url_launcher_android + sha256: "3bb000251e55d4a209aa0e2e563309dc9bb2befea2295fd0cec1f51760aac572" + url: "https://pub.dev" + source: hosted + version: "6.3.29" + url_launcher_ios: + dependency: transitive + description: + name: url_launcher_ios + sha256: "580fe5dfb51671ae38191d316e027f6b76272b026370708c2d898799750a02b0" + url: "https://pub.dev" + source: hosted + version: "6.4.1" + url_launcher_linux: + dependency: transitive + description: + name: url_launcher_linux + sha256: d5e14138b3bc193a0f63c10a53c94b91d399df0512b1f29b94a043db7482384a + url: "https://pub.dev" + source: hosted + version: "3.2.2" + url_launcher_macos: + dependency: transitive + description: + name: url_launcher_macos + sha256: "368adf46f71ad3c21b8f06614adb38346f193f3a59ba8fe9a2fd74133070ba18" + url: "https://pub.dev" + source: hosted + version: "3.2.5" + url_launcher_platform_interface: + dependency: transitive + description: + name: url_launcher_platform_interface + sha256: "552f8a1e663569be95a8190206a38187b531910283c3e982193e4f2733f01029" + url: "https://pub.dev" + source: hosted + version: "2.3.2" + url_launcher_web: + dependency: transitive + description: + name: url_launcher_web + sha256: "85c81589622fbc87c1c683aaea164d3604a7777495a79d91e39ffcdec39ddb34" + url: "https://pub.dev" + source: hosted + version: "2.4.3" + url_launcher_windows: + dependency: transitive + description: + name: url_launcher_windows + sha256: "712c70ab1b99744ff066053cbe3e80c73332b38d46e5e945c98689b2e66fc15f" + url: "https://pub.dev" + source: hosted + version: "3.1.5" + vector_math: + dependency: transitive + description: + name: vector_math + sha256: d530bd74fea330e6e364cda7a85019c434070188383e1cd8d9777ee586914c5b + url: "https://pub.dev" + source: hosted + version: "2.2.0" + vm_service: + dependency: transitive + description: + name: vm_service + sha256: "0016aef94fc66495ac78af5859181e3f3bf2026bd8eecc72b9565601e19ab360" + url: "https://pub.dev" + source: hosted + version: "15.2.0" + watcher: + dependency: transitive + description: + name: watcher + sha256: "1398c9f081a753f9226febe8900fce8f7d0a67163334e1c94a2438339d79d635" + url: "https://pub.dev" + source: hosted + version: "1.2.1" + web: + dependency: transitive + description: + name: web + sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + web_socket: + dependency: transitive + description: + name: web_socket + sha256: "34d64019aa8e36bf9842ac014bb5d2f5586ca73df5e4d9bf5c936975cae6982c" + url: "https://pub.dev" + source: hosted + version: "1.0.1" + web_socket_channel: + dependency: transitive + description: + name: web_socket_channel + sha256: d645757fb0f4773d602444000a8131ff5d48c9e47adfe9772652dd1a4f2d45c8 + url: "https://pub.dev" + source: hosted + version: "3.0.3" + xdg_directories: + dependency: transitive + description: + name: xdg_directories + sha256: "7a3f37b05d989967cdddcbb571f1ea834867ae2faa29725fd085180e0883aa15" + url: "https://pub.dev" + source: hosted + version: "1.1.0" + xml: + dependency: transitive + description: + name: xml + sha256: "971043b3a0d3da28727e40ed3e0b5d18b742fa5a68665cca88e74b7876d5e025" + url: "https://pub.dev" + source: hosted + version: "6.6.1" + yaml: + dependency: transitive + description: + name: yaml + sha256: b9da305ac7c39faa3f030eccd175340f968459dae4af175130b3fc47e40d76ce + url: "https://pub.dev" + source: hosted + version: "3.1.3" +sdks: + dart: ">=3.12.0 <4.0.0" + flutter: ">=3.38.4" diff --git a/pubspec.yaml b/pubspec.yaml new file mode 100644 index 0000000..651eaf3 --- /dev/null +++ b/pubspec.yaml @@ -0,0 +1,102 @@ +name: cyberhybridhub +description: "A new Flutter project." +# The following line prevents the package from being accidentally published to +# pub.dev using `flutter pub publish`. This is preferred for private packages. +publish_to: 'none' # Remove this line if you wish to publish to pub.dev + +# The following defines the version and build number for your application. +# A version number is three numbers separated by dots, like 1.2.43 +# followed by an optional build number separated by a +. +# Both the version and the builder number may be overridden in flutter +# build by specifying --build-name and --build-number, respectively. +# In Android, build-name is used as versionName while build-number used as versionCode. +# Read more about Android versioning at https://developer.android.com/studio/publish/versioning +# In iOS, build-name is used as CFBundleShortVersionString while build-number is used as CFBundleVersion. +# Read more about iOS versioning at +# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html +# In Windows, build-name is used as the major, minor, and patch parts +# of the product and file versions while build-number is used as the build suffix. +version: 1.0.0+1 + +environment: + sdk: ^3.12.0 + +# Dependencies specify other packages that your package needs in order to work. +# To automatically upgrade your package dependencies to the latest versions +# consider running `flutter pub upgrade --major-versions`. Alternatively, +# dependencies can be manually updated by changing the version numbers below to +# the latest version available on pub.dev. To see which dependencies have newer +# versions available, run `flutter pub outdated`. +dependencies: + flutter: + sdk: flutter + + # The following adds the Cupertino Icons font to your application. + # Use with the CupertinoIcons class for iOS style icons. + cupertino_icons: ^1.0.8 + firebase_core: ^4.9.0 + firebase_auth: ^6.5.1 + google_sign_in: ^7.2.0 + google_sign_in_all_platforms: ^2.0.2 + http: ^1.6.0 + drift: ^2.28.0 + drift_flutter: ^0.2.5 + sqlite3_flutter_libs: ^0.5.34 + path: ^1.9.1 + path_provider: ^2.1.5 + connectivity_plus: ^6.1.4 + +dev_dependencies: + flutter_test: + sdk: flutter + + # The "flutter_lints" package below contains a set of recommended lints to + # encourage good coding practices. The lint set provided by the package is + # activated in the `analysis_options.yaml` file located at the root of your + # package. See that file for information about deactivating specific lint + # rules and activating additional ones. + flutter_lints: ^6.0.0 + drift_dev: ^2.28.0 + build_runner: ^2.7.0 + +# For information on the generic Dart part of this file, see the +# following page: https://dart.dev/tools/pub/pubspec + +# The following section is specific to Flutter packages. +flutter: + + # The following line ensures that the Material Icons font is + # included with your application, so that you can use the icons in + # the material Icons class. + uses-material-design: true + + # To add assets to your application, add an assets section, like this: + # assets: + # - images/a_dot_burr.jpeg + # - images/a_dot_ham.jpeg + + # An image asset can refer to one or more resolution-specific "variants", see + # https://flutter.dev/to/resolution-aware-images + + # For details regarding adding assets from package dependencies, see + # https://flutter.dev/to/asset-from-package + + # To add custom fonts to your application, add a fonts section here, + # in this "flutter" section. Each entry in this list should have a + # "family" key with the font family name, and a "fonts" key with a + # list giving the asset and other descriptors for the font. For + # example: + # fonts: + # - family: Schyler + # fonts: + # - asset: fonts/Schyler-Regular.ttf + # - asset: fonts/Schyler-Italic.ttf + # style: italic + # - family: Trajan Pro + # fonts: + # - asset: fonts/TrajanPro.ttf + # - asset: fonts/TrajanPro_Bold.ttf + # weight: 700 + # + # For details regarding fonts from package dependencies, + # see https://flutter.dev/to/font-from-package diff --git a/scripts/flutter-env.sh b/scripts/flutter-env.sh new file mode 100644 index 0000000..f35c8e1 --- /dev/null +++ b/scripts/flutter-env.sh @@ -0,0 +1,3 @@ +# Source this file before flutter/dart commands in any shell: +# source /home/nathan/cyberhybridhub.com/scripts/flutter-env.sh +export PATH="$HOME/github-open/flutter/bin:$PATH" diff --git a/scripts/get-android-sha.sh b/scripts/get-android-sha.sh new file mode 100755 index 0000000..6171914 --- /dev/null +++ b/scripts/get-android-sha.sh @@ -0,0 +1,17 @@ +#!/usr/bin/env bash +# Print Android debug SHA-1 / SHA-256 for Google Sign-In / Firebase. +set -euo pipefail + +KEYSTORE="${1:-$HOME/.android/debug.keystore}" +ALIAS="${2:-androiddebugkey}" + +if [ ! -f "$KEYSTORE" ]; then + echo "Keystore not found: $KEYSTORE" >&2 + exit 1 +fi + +keytool -list -v \ + -keystore "$KEYSTORE" \ + -alias "$ALIAS" \ + -storepass android -keypass android \ + | grep -E 'SHA1:|SHA256:' diff --git a/scripts/setup-firebase-google-auth.sh b/scripts/setup-firebase-google-auth.sh new file mode 100755 index 0000000..8a77d3d --- /dev/null +++ b/scripts/setup-firebase-google-auth.sh @@ -0,0 +1,242 @@ +#!/usr/bin/env bash +# Setup Firebase + Google Sign-In for cyberhybridhub (Ubuntu). +# Run from anywhere: ./scripts/setup-firebase-google-auth.sh +set -euo pipefail + +ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)" +ANDROID_PACKAGE="com.cyberhybridhub.cyberhybridhub" +IOS_BUNDLE_ID="com.cyberhybridhub.cyberhybridhub" +BASHRC_MARKER="# cyberhybridhub: dart pub global + project paths" + +log() { printf '\n==> %s\n' "$*"; } +warn() { printf 'WARN: %s\n' "$*" >&2; } + +ensure_bashrc() { + if grep -qF "$BASHRC_MARKER" "$HOME/.bashrc" 2>/dev/null; then + log "bashrc already configured ($BASHRC_MARKER)" + return + fi + + log "Appending PATH entries to ~/.bashrc" + cat >>"$HOME/.bashrc" <<'EOF' + +# cyberhybridhub: dart pub global + project paths +if [ -d "$HOME/.pub-cache/bin" ]; then + case ":$PATH:" in + *":$HOME/.pub-cache/bin:"*) ;; + *) export PATH="$HOME/.pub-cache/bin:$PATH" ;; + esac +fi +export CYBERHYBRIDHUB_ROOT="$HOME/cyberhybridhub.com" +EOF + # shellcheck source=/dev/null + source "$HOME/.bashrc" 2>/dev/null || true +} + +ensure_nvm() { + export NVM_DIR="${NVM_DIR:-$HOME/.nvm}" + if [ -s "$NVM_DIR/nvm.sh" ]; then + # shellcheck source=/dev/null + . "$NVM_DIR/nvm.sh" + fi +} + +ensure_node_tools() { + ensure_nvm + if ! command -v node >/dev/null 2>&1; then + warn "Node.js not found. Install nvm, then: nvm install --lts" + exit 1 + fi + if ! command -v firebase >/dev/null 2>&1; then + log "Installing Firebase CLI (firebase-tools)" + npm install -g firebase-tools + else + log "Firebase CLI: $(firebase --version)" + fi +} + +ensure_flutter_tools() { + if ! command -v flutter >/dev/null 2>&1; then + warn "Flutter not on PATH. Ensure ~/github-open/flutter/bin is in ~/.bashrc" + exit 1 + fi + log "Flutter: $(flutter --version | head -1)" + if ! command -v flutterfire >/dev/null 2>&1; then + log "Installing FlutterFire CLI" + dart pub global activate flutterfire_cli + else + log "FlutterFire CLI already installed" + fi +} + +login_if_needed() { + log "Firebase login (browser may open; FlutterFire uses this session)" + firebase login --no-localhost 2>/dev/null || firebase login +} + +ensure_firebaserc() { + local project="${FIREBASE_PROJECT_ID:-cyberhybridhub}" + if [ ! -f "$ROOT/.firebaserc" ]; then + log "Creating .firebaserc (default project: $project)" + printf '%s\n' "{\"projects\":{\"default\":\"$project\"}}" >"$ROOT/.firebaserc" + fi +} + +enable_google_auth() { + local project="${FIREBASE_PROJECT_ID:-cyberhybridhub}" + if ! grep -q '"googleSignIn"' "$ROOT/firebase.json" 2>/dev/null; then + warn "firebase.json has no auth.providers.googleSignIn — run: firebase init auth" + return + fi + + log "Enabling Google sign-in (firebase deploy --only auth)" + ( + cd "$ROOT" + firebase deploy --only auth --project="$project" + ) +} + +configure_flutterfire() { + cd "$ROOT" + if [ -n "${FIREBASE_PROJECT_ID:-}" ]; then + log "Running flutterfire configure for project: $FIREBASE_PROJECT_ID" + flutterfire configure \ + --project="$FIREBASE_PROJECT_ID" \ + --platforms=android,ios,web \ + --android-package-name="$ANDROID_PACKAGE" \ + --ios-bundle-id="$IOS_BUNDLE_ID" \ + --out=lib/firebase_options.dart \ + --yes + else + log "Running flutterfire configure (interactive — pick your Firebase project)" + log "Tip: export FIREBASE_PROJECT_ID=your-project-id to skip the project picker" + flutterfire configure \ + --platforms=android,ios,web \ + --android-package-name="$ANDROID_PACKAGE" \ + --ios-bundle-id="$IOS_BUNDLE_ID" \ + --out=lib/firebase_options.dart + fi +} + +add_android_debug_sha() { + local keystore="$HOME/.android/debug.keystore" + if [ ! -f "$keystore" ]; then + warn "Debug keystore not found at $keystore — run the app once or create the keystore" + return + fi + + local sha1 + sha1="$( + keytool -list -v \ + -keystore "$keystore" \ + -alias androiddebugkey \ + -storepass android -keypass android 2>/dev/null \ + | awk -F': ' '/SHA1:/ {print $2; exit}' \ + | tr -d '[:space:]' + )" + + if [ -z "$sha1" ]; then + warn "Could not read SHA-1 from debug keystore" + return + fi + + log "Android debug SHA-1: $sha1" + + if [ -z "${FIREBASE_PROJECT_ID:-}" ]; then + warn "Set FIREBASE_PROJECT_ID to register SHA-1 via CLI, or add it in Google Cloud Console" + return + fi + + local android_app_id + android_app_id="$( + firebase apps:list --project="$FIREBASE_PROJECT_ID" --json 2>/dev/null \ + | python3 -c " +import json, sys +apps = json.load(sys.stdin).get('result', []) +for a in apps: + if a.get('platform') == 'ANDROID': + print(a['appId']) + break +" 2>/dev/null || true + )" + + if [ -z "$android_app_id" ]; then + warn "Could not find Android app ID. Add SHA-1 manually in Firebase/Google Cloud Console" + return + fi + + log "Registering SHA-1 with Firebase app: $android_app_id" + firebase apps:android:sha:create "$android_app_id" "$sha1" \ + --project="$FIREBASE_PROJECT_ID" 2>/dev/null \ + || warn "SHA-1 may already be registered (or run failed — add manually if needed)" + + log "Re-running flutterfire configure to refresh google-services.json" + flutterfire configure \ + --project="$FIREBASE_PROJECT_ID" \ + --platforms=android,ios,web \ + --android-package-name="$ANDROID_PACKAGE" \ + --ios-bundle-id="$IOS_BUNDLE_ID" \ + --out=lib/firebase_options.dart \ + --yes +} + +print_ios_url_scheme_hint() { + local plist="$ROOT/ios/Runner/GoogleService-Info.plist" + if [ ! -f "$plist" ]; then + return + fi + if command -v /usr/libexec/PlistBuddy >/dev/null 2>&1; then + local reversed + reversed="$(/usr/libexec/PlistBuddy -c 'Print :REVERSED_CLIENT_ID' "$plist" 2>/dev/null || true)" + if [ -n "$reversed" ] && [ "$reversed" != "com.googleusercontent.apps.REPLACE_ME" ]; then + log "iOS: add URL scheme to ios/Runner/Info.plist: $reversed" + fi + fi +} + +print_manual_steps() { + cat <<'EOF' + +---------------------------------------------------------------------- +CLI already handled: + • flutterfire configure → lib/firebase_options.dart, google-services.json, GoogleService-Info.plist + • firebase deploy --only auth → Google sign-in provider enabled + +Optional — raw web SDK JSON (not used by Flutter directly): + firebase apps:sdkconfig WEB --project=cyberhybridhub + +Manual if needed: + • Google Cloud → Credentials → Web client → Authorized JavaScript origins: + http://localhost:8080 (see web_dev_config.yaml) + • OAuth consent screen if Google prompts during first sign-in + • iOS: add REVERSED_CLIENT_ID as URL scheme in Info.plist (see above) +---------------------------------------------------------------------- + +Then verify: + cd ~/cyberhybridhub.com + flutter pub get + flutter run -d web-server + +EOF +} + +main() { + log "Cyber Hybrid Hub — Firebase Google auth setup" + ensure_bashrc + ensure_node_tools + ensure_flutter_tools + login_if_needed + ensure_firebaserc + configure_flutterfire + enable_google_auth + add_android_debug_sha + print_ios_url_scheme_hint + + cd "$ROOT" + flutter pub get + + print_manual_steps + log "Done." +} + +main "$@" diff --git a/server/README.md b/server/README.md new file mode 100644 index 0000000..06489ba --- /dev/null +++ b/server/README.md @@ -0,0 +1,41 @@ +# Cyber Hybrid Hub API + +Postgres-backed profile API for the Flutter app. + +## Setup + +1. Create the database (once): + + ```bash + createdb cyberhybridhub + ``` + +2. Copy and edit environment variables: + + ```bash + cp .env.example .env + ``` + +3. Install dependencies and run (from this `server/` directory): + + ```bash + dart pub get + dart run bin/server.dart + ``` + +The API listens on `http://localhost:3000` by default (`PORT` in `.env`). + +## Endpoints + +| Method | Path | Auth | +|--------|------|------| +| `GET` | `/v1/me/profile` | `Authorization: Bearer ` | +| `PUT` | `/v1/me/profile` | same | + +## Flutter client + +Run the app with the API URL (defaults to `http://localhost:3000`): + +```bash +flutter run --dart-define=API_BASE_URL=http://localhost:3000 +``` diff --git a/server/bin/server.dart b/server/bin/server.dart new file mode 100644 index 0000000..5c95ca0 --- /dev/null +++ b/server/bin/server.dart @@ -0,0 +1,40 @@ +import 'dart:io'; + +import 'package:shelf/shelf.dart'; +import 'package:shelf/shelf_io.dart' as shelf_io; + +import '../lib/db.dart'; +import '../lib/env.dart'; +import '../lib/firebase_auth.dart'; +import '../lib/handlers/profile_handler.dart'; + +Future main() async { + final Directory serverRoot = Directory.current; + if (!File('migrations/001_users.sql').existsSync()) { + final String cwd = serverRoot.path; + stderr.writeln( + 'Run the API from the server/ directory (cd server && dart run bin/server.dart). ' + 'Current directory: $cwd', + ); + exit(1); + } + + final ServerEnv env = ServerEnv.load(); + final ProfileDb db = await ProfileDb.connect(env.databaseUrl); + await db.migrate(); + + final FirebaseAuthVerifier auth = FirebaseAuthVerifier(env.firebaseWebApiKey); + final Handler handler = Pipeline() + .addMiddleware(logRequests()) + .addHandler(profileHandler(db: db, auth: auth)); + + final HttpServer server = await shelf_io.serve( + handler, + InternetAddress.anyIPv4, + env.port, + ); + + stdout.writeln( + 'Cyber Hybrid Hub API listening on http://localhost:${server.port}', + ); +} diff --git a/server/lib/db.dart b/server/lib/db.dart new file mode 100644 index 0000000..53f238a --- /dev/null +++ b/server/lib/db.dart @@ -0,0 +1,133 @@ +import 'dart:io'; + +import 'package:postgres/postgres.dart'; + +class ProfileDb { + ProfileDb(this._connection); + + final Connection _connection; + + static Future connect(String databaseUrl) async { + final Uri uri = Uri.parse(databaseUrl); + final Connection connection = await Connection.open( + Endpoint( + host: uri.host.isEmpty ? 'localhost' : uri.host, + port: uri.hasPort ? uri.port : 5432, + database: uri.pathSegments.isNotEmpty ? uri.pathSegments.last : 'postgres', + username: uri.userInfo.isNotEmpty ? uri.userInfo.split(':').first : null, + password: uri.userInfo.contains(':') + ? uri.userInfo.split(':').skip(1).join(':') + : null, + ), + settings: const ConnectionSettings(sslMode: SslMode.disable), + ); + return ProfileDb(connection); + } + + Future migrate() async { + final String sql = await File('migrations/001_users.sql').readAsString(); + final List statements = sql + .split(';') + .map((String s) => s.trim()) + .where((String s) => s.isNotEmpty) + .toList(); + + for (final String statement in statements) { + await _connection.execute(statement); + } + } + + Future?> getProfile(String firebaseUid) async { + final Result result = await _connection.execute( + Sql.named( + 'SELECT firebase_uid, email, display_name, photo_url, locale, timezone, ' + 'onboarding_done, revision, updated_at ' + 'FROM users WHERE firebase_uid = @uid', + ), + parameters: {'uid': firebaseUid}, + ); + if (result.isEmpty) { + return null; + } + final ResultRow row = result.first; + return _rowToJson(row); + } + + Future> upsertProfile({ + required String firebaseUid, + required Map body, + required int clientRevision, + }) async { + final Result existing = await _connection.execute( + Sql.named('SELECT revision FROM users WHERE firebase_uid = @uid'), + parameters: {'uid': firebaseUid}, + ); + + if (existing.isNotEmpty) { + final int serverRevision = (existing.first[0]! as num).toInt(); + if (clientRevision < serverRevision) { + final Map? current = await getProfile(firebaseUid); + throw StaleRevisionException(current!); + } + } + + await _connection.execute( + Sql.named( + ''' + INSERT INTO users ( + firebase_uid, email, display_name, photo_url, locale, timezone, + onboarding_done, revision, updated_at + ) VALUES ( + @uid, @email, @display_name, @photo_url, @locale, @timezone, + @onboarding_done, @revision, @updated_at + ) + ON CONFLICT (firebase_uid) DO UPDATE SET + email = EXCLUDED.email, + display_name = EXCLUDED.display_name, + photo_url = EXCLUDED.photo_url, + locale = EXCLUDED.locale, + timezone = EXCLUDED.timezone, + onboarding_done = EXCLUDED.onboarding_done, + revision = EXCLUDED.revision, + updated_at = EXCLUDED.updated_at + ''', + ), + parameters: { + 'uid': firebaseUid, + 'email': body['email'], + 'display_name': body['displayName'], + 'photo_url': body['photoUrl'], + 'locale': body['locale'] ?? 'en', + 'timezone': body['timezone'], + 'onboarding_done': body['onboardingCompleted'] ?? false, + 'revision': clientRevision, + 'updated_at': DateTime.parse(body['updatedAt'] as String).toUtc(), + }, + ); + + return (await getProfile(firebaseUid))!; + } + + Future close() => _connection.close(); + + Map _rowToJson(ResultRow row) { + final DateTime updatedAt = row[8]! as DateTime; + return { + 'firebaseUid': row[0]! as String, + 'email': row[1] as String?, + 'displayName': row[2] as String?, + 'photoUrl': row[3] as String?, + 'locale': row[4]! as String, + 'timezone': row[5] as String?, + 'onboardingCompleted': row[6]! as bool, + 'revision': (row[7]! as num).toInt(), + 'updatedAt': updatedAt.toUtc().toIso8601String(), + }; + } +} + +class StaleRevisionException implements Exception { + StaleRevisionException(this.serverProfile); + + final Map serverProfile; +} diff --git a/server/lib/env.dart b/server/lib/env.dart new file mode 100644 index 0000000..4a93d4c --- /dev/null +++ b/server/lib/env.dart @@ -0,0 +1,32 @@ +import 'dart:io'; + +import 'package:dotenv/dotenv.dart'; + +class ServerEnv { + ServerEnv._(this.databaseUrl, this.port, this.firebaseWebApiKey); + + final String databaseUrl; + final int port; + final String firebaseWebApiKey; + + static ServerEnv load() { + final DotEnv env = DotEnv(includePlatformEnvironment: true) + ..load(['.env']); + + final String? databaseUrl = env['DATABASE_URL']; + if (databaseUrl == null || databaseUrl.isEmpty) { + stderr.writeln('DATABASE_URL is required in server/.env'); + exit(1); + } + + final String? apiKey = env['FIREBASE_WEB_API_KEY']; + if (apiKey == null || apiKey.isEmpty) { + stderr.writeln('FIREBASE_WEB_API_KEY is required in server/.env'); + exit(1); + } + + final int port = int.tryParse(env['PORT'] ?? '3000') ?? 3000; + + return ServerEnv._(databaseUrl, port, apiKey); + } +} diff --git a/server/lib/firebase_auth.dart b/server/lib/firebase_auth.dart new file mode 100644 index 0000000..8f89ee9 --- /dev/null +++ b/server/lib/firebase_auth.dart @@ -0,0 +1,40 @@ +import 'dart:convert'; + +import 'package:http/http.dart' as http; + +class FirebaseAuthVerifier { + FirebaseAuthVerifier(this._webApiKey); + + final String _webApiKey; + + Future verifyBearerToken(String? authorization) async { + if (authorization == null || !authorization.startsWith('Bearer ')) { + return null; + } + final String idToken = authorization.substring('Bearer '.length).trim(); + if (idToken.isEmpty) { + return null; + } + + final Uri uri = Uri.parse( + 'https://identitytoolkit.googleapis.com/v1/accounts:lookup?key=$_webApiKey', + ); + final http.Response response = await http.post( + uri, + headers: const {'Content-Type': 'application/json'}, + body: jsonEncode({'idToken': idToken}), + ); + + if (response.statusCode != 200) { + return null; + } + + final Map data = + jsonDecode(response.body) as Map; + final List users = data['users'] as List? ?? []; + if (users.isEmpty) { + return null; + } + return users.first['localId'] as String?; + } +} diff --git a/server/lib/handlers/profile_handler.dart b/server/lib/handlers/profile_handler.dart new file mode 100644 index 0000000..95b915f --- /dev/null +++ b/server/lib/handlers/profile_handler.dart @@ -0,0 +1,82 @@ +import 'dart:convert'; +import 'dart:io'; + +import 'package:shelf/shelf.dart'; + +import '../db.dart'; +import '../firebase_auth.dart'; + +Handler profileHandler({ + required ProfileDb db, + required FirebaseAuthVerifier auth, +}) { + return (Request request) async { + if (request.method == 'OPTIONS') { + return Response.ok('', headers: _corsHeaders); + } + + final String? firebaseUid = await auth.verifyBearerToken( + request.headers['Authorization'] ?? request.headers['authorization'], + ); + if (firebaseUid == null) { + return _jsonResponse(401, {'error': 'Unauthorized'}); + } + + if (request.requestedUri.path != '/v1/me/profile') { + return _jsonResponse(404, {'error': 'Not found'}); + } + + try { + if (request.method == 'GET') { + final Map? profile = await db.getProfile(firebaseUid); + if (profile == null) { + return _jsonResponse(404, {'error': 'Not found'}); + } + return _jsonResponse(200, profile); + } + + if (request.method == 'PUT') { + final String body = await request.readAsString(); + final Map json = + jsonDecode(body) as Map; + final int revision = (json['revision'] as num?)?.toInt() ?? 1; + + try { + final Map saved = await db.upsertProfile( + firebaseUid: firebaseUid, + body: json, + clientRevision: revision, + ); + return _jsonResponse(200, saved); + } on StaleRevisionException catch (e) { + return _jsonResponse(409, { + 'error': 'Conflict', + 'profile': e.serverProfile, + }); + } + } + + return _jsonResponse(405, {'error': 'Method not allowed'}); + } catch (e, st) { + stderr.writeln('Profile handler error: $e\n$st'); + return _jsonResponse(500, {'error': 'Internal error'}); + } + }; +} + +Map get _corsHeaders => { + 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Methods': 'GET, PUT, OPTIONS', + 'Access-Control-Allow-Headers': 'Authorization, Content-Type', +}; + +Response _jsonResponse(int status, Map body) { + return Response( + status, + body: jsonEncode(body), + headers: { + ..._corsHeaders, + 'Content-Type': 'application/json', + }, + ); +} diff --git a/server/migrations/001_users.sql b/server/migrations/001_users.sql new file mode 100644 index 0000000..dca35d3 --- /dev/null +++ b/server/migrations/001_users.sql @@ -0,0 +1,14 @@ +CREATE TABLE IF NOT EXISTS users ( + firebase_uid TEXT PRIMARY KEY, + email TEXT, + display_name TEXT, + photo_url TEXT, + locale TEXT NOT NULL DEFAULT 'en', + timezone TEXT, + onboarding_done BOOLEAN NOT NULL DEFAULT FALSE, + revision BIGINT NOT NULL DEFAULT 1, + updated_at TIMESTAMPTZ NOT NULL DEFAULT now(), + created_at TIMESTAMPTZ NOT NULL DEFAULT now() +); + +CREATE INDEX IF NOT EXISTS users_updated_at_idx ON users (updated_at); diff --git a/server/pubspec.lock b/server/pubspec.lock new file mode 100644 index 0000000..d03f3ab --- /dev/null +++ b/server/pubspec.lock @@ -0,0 +1,197 @@ +# Generated by pub +# See https://dart.dev/tools/pub/glossary#lockfile +packages: + args: + dependency: transitive + description: + name: args + sha256: d0481093c50b1da8910eb0bb301626d4d8eb7284aa739614d2b394ee09e3ea04 + url: "https://pub.dev" + source: hosted + version: "2.7.0" + async: + dependency: transitive + description: + name: async + sha256: e2eb0491ba5ddb6177742d2da23904574082139b07c1e33b8503b9f46f3e1a37 + url: "https://pub.dev" + source: hosted + version: "2.13.1" + buffer: + dependency: transitive + description: + name: buffer + sha256: "389da2ec2c16283c8787e0adaede82b1842102f8c8aae2f49003a766c5c6b3d1" + url: "https://pub.dev" + source: hosted + version: "1.2.3" + charcode: + dependency: transitive + description: + name: charcode + sha256: fb0f1107cac15a5ea6ef0a6ef71a807b9e4267c713bb93e00e92d737cc8dbd8a + url: "https://pub.dev" + source: hosted + version: "1.4.0" + collection: + dependency: transitive + description: + name: collection + sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" + url: "https://pub.dev" + source: hosted + version: "1.19.1" + crypto: + dependency: transitive + description: + name: crypto + sha256: c8ea0233063ba03258fbcf2ca4d6dadfefe14f02fab57702265467a19f27fadf + url: "https://pub.dev" + source: hosted + version: "3.0.7" + dotenv: + dependency: "direct main" + description: + name: dotenv + sha256: "379e64b6fc82d3df29461d349a1796ecd2c436c480d4653f3af6872eccbc90e1" + url: "https://pub.dev" + source: hosted + version: "4.2.0" + http: + dependency: "direct main" + description: + name: http + sha256: "87721a4a50b19c7f1d49001e51409bddc46303966ce89a65af4f4e6004896412" + url: "https://pub.dev" + source: hosted + version: "1.6.0" + http_methods: + dependency: transitive + description: + name: http_methods + sha256: "6bccce8f1ec7b5d701e7921dca35e202d425b57e317ba1a37f2638590e29e566" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + http_parser: + dependency: transitive + description: + name: http_parser + sha256: "178d74305e7866013777bab2c3d8726205dc5a4dd935297175b19a23a2e66571" + url: "https://pub.dev" + source: hosted + version: "4.1.2" + meta: + dependency: transitive + description: + name: meta + sha256: df0c643f44ad098eb37988027a8e2b2b5a031fd3977f06bbfd3a76637e8df739 + url: "https://pub.dev" + source: hosted + version: "1.18.2" + path: + dependency: transitive + description: + name: path + sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" + url: "https://pub.dev" + source: hosted + version: "1.9.1" + pool: + dependency: transitive + description: + name: pool + sha256: "978783255c543aa3586a1b3c21f6e9d720eb315376a915872c61ef8b5c20177d" + url: "https://pub.dev" + source: hosted + version: "1.5.2" + postgres: + dependency: "direct main" + description: + name: postgres + sha256: fe4a19a69fad564efa8ced72d58b14e86d12a548f5f0a511b377089bf3b592ed + url: "https://pub.dev" + source: hosted + version: "3.5.11" + shelf: + dependency: "direct main" + description: + name: shelf + sha256: e7dd780a7ffb623c57850b33f43309312fc863fb6aa3d276a754bb299839ef12 + url: "https://pub.dev" + source: hosted + version: "1.4.2" + shelf_cors_headers: + dependency: "direct main" + description: + name: shelf_cors_headers + sha256: a127c80f99bbef3474293db67a7608e3a0f1f0fcdb171dad77fa9bd2cd123ae4 + url: "https://pub.dev" + source: hosted + version: "0.1.5" + shelf_router: + dependency: "direct main" + description: + name: shelf_router + sha256: f5e5d492440a7fb165fe1e2e1a623f31f734d3370900070b2b1e0d0428d59864 + url: "https://pub.dev" + source: hosted + version: "1.1.4" + source_span: + dependency: transitive + description: + name: source_span + sha256: "56a02f1f4cd1a2d96303c0144c93bd6d909eea6bee6bf5a0e0b685edbd4c47ab" + url: "https://pub.dev" + source: hosted + version: "1.10.2" + stack_trace: + dependency: transitive + description: + name: stack_trace + sha256: "8b27215b45d22309b5cddda1aa2b19bdfec9df0e765f2de506401c071d38d1b1" + url: "https://pub.dev" + source: hosted + version: "1.12.1" + stream_channel: + dependency: transitive + description: + name: stream_channel + sha256: "969e04c80b8bcdf826f8f16579c7b14d780458bd97f56d107d3950fdbeef059d" + url: "https://pub.dev" + source: hosted + version: "2.1.4" + string_scanner: + dependency: transitive + description: + name: string_scanner + sha256: "921cd31725b72fe181906c6a94d987c78e3b98c2e205b397ea399d4054872b43" + url: "https://pub.dev" + source: hosted + version: "1.4.1" + term_glyph: + dependency: transitive + description: + name: term_glyph + sha256: "7f554798625ea768a7518313e58f83891c7f5024f88e46e7182a4558850a4b8e" + url: "https://pub.dev" + source: hosted + version: "1.2.2" + typed_data: + dependency: transitive + description: + name: typed_data + sha256: f9049c039ebfeb4cf7a7104a675823cd72dba8297f264b6637062516699fa006 + url: "https://pub.dev" + source: hosted + version: "1.4.0" + web: + dependency: transitive + description: + name: web + sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" + url: "https://pub.dev" + source: hosted + version: "1.1.1" +sdks: + dart: ">=3.12.0 <4.0.0" diff --git a/server/pubspec.yaml b/server/pubspec.yaml new file mode 100644 index 0000000..3566f3f --- /dev/null +++ b/server/pubspec.yaml @@ -0,0 +1,14 @@ +name: cyberhybridhub_server +description: API server for Cyber Hybrid Hub user profiles (Postgres). +publish_to: 'none' + +environment: + sdk: ^3.12.0 + +dependencies: + shelf: ^1.4.2 + shelf_router: ^1.1.4 + shelf_cors_headers: ^0.1.5 + postgres: ^3.5.6 + dotenv: ^4.2.0 + http: ^1.6.0 diff --git a/test/widget_test.dart b/test/widget_test.dart new file mode 100644 index 0000000..94cbbdb --- /dev/null +++ b/test/widget_test.dart @@ -0,0 +1,28 @@ +import 'package:cyberhybridhub/screens/landing_screen.dart'; +import 'package:cyberhybridhub/theme/app_theme.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_test/flutter_test.dart'; + +void main() { + testWidgets('Landing page shows primary CTA and value proposition', ( + WidgetTester tester, + ) async { + tester.view.physicalSize = const Size(800, 1200); + tester.view.devicePixelRatio = 1.0; + addTearDown(tester.view.reset); + + await tester.pumpWidget( + MaterialApp( + theme: buildAppTheme(), + home: const LandingScreen(), + ), + ); + + expect(find.text('Get started with Google'), findsOneWidget); + expect( + find.text('Your hybrid security\ncommand center'), + findsOneWidget, + ); + expect(find.text('Free to get started'), findsOneWidget); + }); +} diff --git a/web/favicon.png b/web/favicon.png new file mode 100644 index 0000000..8aaa46a Binary files /dev/null and b/web/favicon.png differ diff --git a/web/icons/Icon-192.png b/web/icons/Icon-192.png new file mode 100644 index 0000000..b749bfe Binary files /dev/null and b/web/icons/Icon-192.png differ diff --git a/web/icons/Icon-512.png b/web/icons/Icon-512.png new file mode 100644 index 0000000..88cfd48 Binary files /dev/null and b/web/icons/Icon-512.png differ diff --git a/web/icons/Icon-maskable-192.png b/web/icons/Icon-maskable-192.png new file mode 100644 index 0000000..eb9b4d7 Binary files /dev/null and b/web/icons/Icon-maskable-192.png differ diff --git a/web/icons/Icon-maskable-512.png b/web/icons/Icon-maskable-512.png new file mode 100644 index 0000000..d69c566 Binary files /dev/null and b/web/icons/Icon-maskable-512.png differ diff --git a/web/index.html b/web/index.html new file mode 100644 index 0000000..298feed --- /dev/null +++ b/web/index.html @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + + + + + + + + + cyberhybridhub + + + + + + + diff --git a/web/manifest.json b/web/manifest.json new file mode 100644 index 0000000..4548d4b --- /dev/null +++ b/web/manifest.json @@ -0,0 +1,35 @@ +{ + "name": "cyberhybridhub", + "short_name": "cyberhybridhub", + "start_url": ".", + "display": "standalone", + "background_color": "#0175C2", + "theme_color": "#0175C2", + "description": "A new Flutter project.", + "orientation": "portrait-primary", + "prefer_related_applications": false, + "icons": [ + { + "src": "icons/Icon-192.png", + "sizes": "192x192", + "type": "image/png" + }, + { + "src": "icons/Icon-512.png", + "sizes": "512x512", + "type": "image/png" + }, + { + "src": "icons/Icon-maskable-192.png", + "sizes": "192x192", + "type": "image/png", + "purpose": "maskable" + }, + { + "src": "icons/Icon-maskable-512.png", + "sizes": "512x512", + "type": "image/png", + "purpose": "maskable" + } + ] +} diff --git a/web_dev_config.yaml b/web_dev_config.yaml new file mode 100644 index 0000000..393e26b --- /dev/null +++ b/web_dev_config.yaml @@ -0,0 +1,5 @@ +# Static port for `flutter run` on web (OAuth redirect URIs, bookmarks). +# See: https://docs.flutter.dev/platform-integration/web/building +server: + host: localhost + port: 8080 diff --git a/windows/.gitignore b/windows/.gitignore new file mode 100644 index 0000000..d492d0d --- /dev/null +++ b/windows/.gitignore @@ -0,0 +1,17 @@ +flutter/ephemeral/ + +# Visual Studio user-specific files. +*.suo +*.user +*.userosscache +*.sln.docstates + +# Visual Studio build-related files. +x64/ +x86/ + +# Visual Studio cache files +# files ending in .cache can be ignored +*.[Cc]ache +# but keep track of directories ending in .cache +!*.[Cc]ache/ diff --git a/windows/CMakeLists.txt b/windows/CMakeLists.txt new file mode 100644 index 0000000..b34f56c --- /dev/null +++ b/windows/CMakeLists.txt @@ -0,0 +1,108 @@ +# Project-level configuration. +cmake_minimum_required(VERSION 3.14) +project(cyberhybridhub LANGUAGES CXX) + +# The name of the executable created for the application. Change this to change +# the on-disk name of your application. +set(BINARY_NAME "cyberhybridhub") + +# Explicitly opt in to modern CMake behaviors to avoid warnings with recent +# versions of CMake. +cmake_policy(VERSION 3.14...3.25) + +# Define build configuration option. +get_property(IS_MULTICONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG) +if(IS_MULTICONFIG) + set(CMAKE_CONFIGURATION_TYPES "Debug;Profile;Release" + CACHE STRING "" FORCE) +else() + if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) + set(CMAKE_BUILD_TYPE "Debug" CACHE + STRING "Flutter build mode" FORCE) + set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS + "Debug" "Profile" "Release") + endif() +endif() +# Define settings for the Profile build mode. +set(CMAKE_EXE_LINKER_FLAGS_PROFILE "${CMAKE_EXE_LINKER_FLAGS_RELEASE}") +set(CMAKE_SHARED_LINKER_FLAGS_PROFILE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE}") +set(CMAKE_C_FLAGS_PROFILE "${CMAKE_C_FLAGS_RELEASE}") +set(CMAKE_CXX_FLAGS_PROFILE "${CMAKE_CXX_FLAGS_RELEASE}") + +# Use Unicode for all projects. +add_definitions(-DUNICODE -D_UNICODE) + +# Compilation settings that should be applied to most targets. +# +# Be cautious about adding new options here, as plugins use this function by +# default. In most cases, you should add new options to specific targets instead +# of modifying this function. +function(APPLY_STANDARD_SETTINGS TARGET) + target_compile_features(${TARGET} PUBLIC cxx_std_17) + target_compile_options(${TARGET} PRIVATE /W4 /WX /wd"4100") + target_compile_options(${TARGET} PRIVATE /EHsc) + target_compile_definitions(${TARGET} PRIVATE "_HAS_EXCEPTIONS=0") + target_compile_definitions(${TARGET} PRIVATE "$<$:_DEBUG>") +endfunction() + +# Flutter library and tool build rules. +set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") +add_subdirectory(${FLUTTER_MANAGED_DIR}) + +# Application build; see runner/CMakeLists.txt. +add_subdirectory("runner") + + +# Generated plugin build rules, which manage building the plugins and adding +# them to the application. +include(flutter/generated_plugins.cmake) + + +# === Installation === +# Support files are copied into place next to the executable, so that it can +# run in place. This is done instead of making a separate bundle (as on Linux) +# so that building and running from within Visual Studio will work. +set(BUILD_BUNDLE_DIR "$") +# Make the "install" step default, as it's required to run. +set(CMAKE_VS_INCLUDE_INSTALL_TO_DEFAULT_BUILD 1) +if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) + set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) +endif() + +set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") +set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}") + +install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" + COMPONENT Runtime) + +install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" + COMPONENT Runtime) + +install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) + +if(PLUGIN_BUNDLED_LIBRARIES) + install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" + DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) +endif() + +# Copy the native assets provided by the build.dart from all packages. +set(NATIVE_ASSETS_DIR "${PROJECT_BUILD_DIR}native_assets/windows/") +install(DIRECTORY "${NATIVE_ASSETS_DIR}" + DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" + COMPONENT Runtime) + +# Fully re-copy the assets directory on each build to avoid having stale files +# from a previous install. +set(FLUTTER_ASSET_DIR_NAME "flutter_assets") +install(CODE " + file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") + " COMPONENT Runtime) +install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" + DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) + +# Install the AOT library on non-Debug builds only. +install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" + CONFIGURATIONS Profile;Release + COMPONENT Runtime) diff --git a/windows/flutter/CMakeLists.txt b/windows/flutter/CMakeLists.txt new file mode 100644 index 0000000..903f489 --- /dev/null +++ b/windows/flutter/CMakeLists.txt @@ -0,0 +1,109 @@ +# This file controls Flutter-level build steps. It should not be edited. +cmake_minimum_required(VERSION 3.14) + +set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") + +# Configuration provided via flutter tool. +include(${EPHEMERAL_DIR}/generated_config.cmake) + +# TODO: Move the rest of this into files in ephemeral. See +# https://github.com/flutter/flutter/issues/57146. +set(WRAPPER_ROOT "${EPHEMERAL_DIR}/cpp_client_wrapper") + +# Set fallback configurations for older versions of the flutter tool. +if (NOT DEFINED FLUTTER_TARGET_PLATFORM) + set(FLUTTER_TARGET_PLATFORM "windows-x64") +endif() + +# === Flutter Library === +set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/flutter_windows.dll") + +# Published to parent scope for install step. +set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) +set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) +set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) +set(AOT_LIBRARY "${PROJECT_DIR}/build/windows/app.so" PARENT_SCOPE) + +list(APPEND FLUTTER_LIBRARY_HEADERS + "flutter_export.h" + "flutter_windows.h" + "flutter_messenger.h" + "flutter_plugin_registrar.h" + "flutter_texture_registrar.h" +) +list(TRANSFORM FLUTTER_LIBRARY_HEADERS PREPEND "${EPHEMERAL_DIR}/") +add_library(flutter INTERFACE) +target_include_directories(flutter INTERFACE + "${EPHEMERAL_DIR}" +) +target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}.lib") +add_dependencies(flutter flutter_assemble) + +# === Wrapper === +list(APPEND CPP_WRAPPER_SOURCES_CORE + "core_implementations.cc" + "standard_codec.cc" +) +list(TRANSFORM CPP_WRAPPER_SOURCES_CORE PREPEND "${WRAPPER_ROOT}/") +list(APPEND CPP_WRAPPER_SOURCES_PLUGIN + "plugin_registrar.cc" +) +list(TRANSFORM CPP_WRAPPER_SOURCES_PLUGIN PREPEND "${WRAPPER_ROOT}/") +list(APPEND CPP_WRAPPER_SOURCES_APP + "flutter_engine.cc" + "flutter_view_controller.cc" +) +list(TRANSFORM CPP_WRAPPER_SOURCES_APP PREPEND "${WRAPPER_ROOT}/") + +# Wrapper sources needed for a plugin. +add_library(flutter_wrapper_plugin STATIC + ${CPP_WRAPPER_SOURCES_CORE} + ${CPP_WRAPPER_SOURCES_PLUGIN} +) +apply_standard_settings(flutter_wrapper_plugin) +set_target_properties(flutter_wrapper_plugin PROPERTIES + POSITION_INDEPENDENT_CODE ON) +set_target_properties(flutter_wrapper_plugin PROPERTIES + CXX_VISIBILITY_PRESET hidden) +target_link_libraries(flutter_wrapper_plugin PUBLIC flutter) +target_include_directories(flutter_wrapper_plugin PUBLIC + "${WRAPPER_ROOT}/include" +) +add_dependencies(flutter_wrapper_plugin flutter_assemble) + +# Wrapper sources needed for the runner. +add_library(flutter_wrapper_app STATIC + ${CPP_WRAPPER_SOURCES_CORE} + ${CPP_WRAPPER_SOURCES_APP} +) +apply_standard_settings(flutter_wrapper_app) +target_link_libraries(flutter_wrapper_app PUBLIC flutter) +target_include_directories(flutter_wrapper_app PUBLIC + "${WRAPPER_ROOT}/include" +) +add_dependencies(flutter_wrapper_app flutter_assemble) + +# === Flutter tool backend === +# _phony_ is a non-existent file to force this command to run every time, +# since currently there's no way to get a full input/output list from the +# flutter tool. +set(PHONY_OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/_phony_") +set_source_files_properties("${PHONY_OUTPUT}" PROPERTIES SYMBOLIC TRUE) +add_custom_command( + OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} + ${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} + ${CPP_WRAPPER_SOURCES_APP} + ${PHONY_OUTPUT} + COMMAND ${CMAKE_COMMAND} -E env + ${FLUTTER_TOOL_ENVIRONMENT} + "${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.bat" + ${FLUTTER_TARGET_PLATFORM} $ + VERBATIM +) +add_custom_target(flutter_assemble DEPENDS + "${FLUTTER_LIBRARY}" + ${FLUTTER_LIBRARY_HEADERS} + ${CPP_WRAPPER_SOURCES_CORE} + ${CPP_WRAPPER_SOURCES_PLUGIN} + ${CPP_WRAPPER_SOURCES_APP} +) diff --git a/windows/flutter/generated_plugin_registrant.cc b/windows/flutter/generated_plugin_registrant.cc new file mode 100644 index 0000000..4d662be --- /dev/null +++ b/windows/flutter/generated_plugin_registrant.cc @@ -0,0 +1,26 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#include "generated_plugin_registrant.h" + +#include +#include +#include +#include +#include + +void RegisterPlugins(flutter::PluginRegistry* registry) { + ConnectivityPlusWindowsPluginRegisterWithRegistrar( + registry->GetRegistrarForPlugin("ConnectivityPlusWindowsPlugin")); + FirebaseAuthPluginCApiRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FirebaseAuthPluginCApi")); + FirebaseCorePluginCApiRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FirebaseCorePluginCApi")); + Sqlite3FlutterLibsPluginRegisterWithRegistrar( + registry->GetRegistrarForPlugin("Sqlite3FlutterLibsPlugin")); + UrlLauncherWindowsRegisterWithRegistrar( + registry->GetRegistrarForPlugin("UrlLauncherWindows")); +} diff --git a/windows/flutter/generated_plugin_registrant.h b/windows/flutter/generated_plugin_registrant.h new file mode 100644 index 0000000..dc139d8 --- /dev/null +++ b/windows/flutter/generated_plugin_registrant.h @@ -0,0 +1,15 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#ifndef GENERATED_PLUGIN_REGISTRANT_ +#define GENERATED_PLUGIN_REGISTRANT_ + +#include + +// Registers Flutter plugins. +void RegisterPlugins(flutter::PluginRegistry* registry); + +#endif // GENERATED_PLUGIN_REGISTRANT_ diff --git a/windows/flutter/generated_plugins.cmake b/windows/flutter/generated_plugins.cmake new file mode 100644 index 0000000..ea130c3 --- /dev/null +++ b/windows/flutter/generated_plugins.cmake @@ -0,0 +1,29 @@ +# +# Generated file, do not edit. +# + +list(APPEND FLUTTER_PLUGIN_LIST + connectivity_plus + firebase_auth + firebase_core + sqlite3_flutter_libs + url_launcher_windows +) + +list(APPEND FLUTTER_FFI_PLUGIN_LIST + jni +) + +set(PLUGIN_BUNDLED_LIBRARIES) + +foreach(plugin ${FLUTTER_PLUGIN_LIST}) + add_subdirectory(flutter/ephemeral/.plugin_symlinks/${plugin}/windows plugins/${plugin}) + target_link_libraries(${BINARY_NAME} PRIVATE ${plugin}_plugin) + list(APPEND PLUGIN_BUNDLED_LIBRARIES $) + list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${plugin}_bundled_libraries}) +endforeach(plugin) + +foreach(ffi_plugin ${FLUTTER_FFI_PLUGIN_LIST}) + add_subdirectory(flutter/ephemeral/.plugin_symlinks/${ffi_plugin}/windows plugins/${ffi_plugin}) + list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${ffi_plugin}_bundled_libraries}) +endforeach(ffi_plugin) diff --git a/windows/runner/CMakeLists.txt b/windows/runner/CMakeLists.txt new file mode 100644 index 0000000..394917c --- /dev/null +++ b/windows/runner/CMakeLists.txt @@ -0,0 +1,40 @@ +cmake_minimum_required(VERSION 3.14) +project(runner LANGUAGES CXX) + +# Define the application target. To change its name, change BINARY_NAME in the +# top-level CMakeLists.txt, not the value here, or `flutter run` will no longer +# work. +# +# Any new source files that you add to the application should be added here. +add_executable(${BINARY_NAME} WIN32 + "flutter_window.cpp" + "main.cpp" + "utils.cpp" + "win32_window.cpp" + "${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" + "Runner.rc" + "runner.exe.manifest" +) + +# Apply the standard set of build settings. This can be removed for applications +# that need different build settings. +apply_standard_settings(${BINARY_NAME}) + +# Add preprocessor definitions for the build version. +target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION=\"${FLUTTER_VERSION}\"") +target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MAJOR=${FLUTTER_VERSION_MAJOR}") +target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MINOR=${FLUTTER_VERSION_MINOR}") +target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_PATCH=${FLUTTER_VERSION_PATCH}") +target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_BUILD=${FLUTTER_VERSION_BUILD}") + +# Disable Windows macros that collide with C++ standard library functions. +target_compile_definitions(${BINARY_NAME} PRIVATE "NOMINMAX") + +# Add dependency libraries and include directories. Add any application-specific +# dependencies here. +target_link_libraries(${BINARY_NAME} PRIVATE flutter flutter_wrapper_app) +target_link_libraries(${BINARY_NAME} PRIVATE "dwmapi.lib") +target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") + +# Run the Flutter tool portions of the build. This must not be removed. +add_dependencies(${BINARY_NAME} flutter_assemble) diff --git a/windows/runner/Runner.rc b/windows/runner/Runner.rc new file mode 100644 index 0000000..31017b5 --- /dev/null +++ b/windows/runner/Runner.rc @@ -0,0 +1,121 @@ +// Microsoft Visual C++ generated resource script. +// +#pragma code_page(65001) +#include "resource.h" + +#define APSTUDIO_READONLY_SYMBOLS +///////////////////////////////////////////////////////////////////////////// +// +// Generated from the TEXTINCLUDE 2 resource. +// +#include "winres.h" + +///////////////////////////////////////////////////////////////////////////// +#undef APSTUDIO_READONLY_SYMBOLS + +///////////////////////////////////////////////////////////////////////////// +// English (United States) resources + +#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU) +LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US + +#ifdef APSTUDIO_INVOKED +///////////////////////////////////////////////////////////////////////////// +// +// TEXTINCLUDE +// + +1 TEXTINCLUDE +BEGIN + "resource.h\0" +END + +2 TEXTINCLUDE +BEGIN + "#include ""winres.h""\r\n" + "\0" +END + +3 TEXTINCLUDE +BEGIN + "\r\n" + "\0" +END + +#endif // APSTUDIO_INVOKED + + +///////////////////////////////////////////////////////////////////////////// +// +// Icon +// + +// Icon with lowest ID value placed first to ensure application icon +// remains consistent on all systems. +IDI_APP_ICON ICON "resources\\app_icon.ico" + + +///////////////////////////////////////////////////////////////////////////// +// +// Version +// + +#if defined(FLUTTER_VERSION_MAJOR) && defined(FLUTTER_VERSION_MINOR) && defined(FLUTTER_VERSION_PATCH) && defined(FLUTTER_VERSION_BUILD) +#define VERSION_AS_NUMBER FLUTTER_VERSION_MAJOR,FLUTTER_VERSION_MINOR,FLUTTER_VERSION_PATCH,FLUTTER_VERSION_BUILD +#else +#define VERSION_AS_NUMBER 1,0,0,0 +#endif + +#if defined(FLUTTER_VERSION) +#define VERSION_AS_STRING FLUTTER_VERSION +#else +#define VERSION_AS_STRING "1.0.0" +#endif + +VS_VERSION_INFO VERSIONINFO + FILEVERSION VERSION_AS_NUMBER + PRODUCTVERSION VERSION_AS_NUMBER + FILEFLAGSMASK VS_FFI_FILEFLAGSMASK +#ifdef _DEBUG + FILEFLAGS VS_FF_DEBUG +#else + FILEFLAGS 0x0L +#endif + FILEOS VOS__WINDOWS32 + FILETYPE VFT_APP + FILESUBTYPE 0x0L +BEGIN + BLOCK "StringFileInfo" + BEGIN + BLOCK "040904e4" + BEGIN + VALUE "CompanyName", "com.cyberhybridhub" "\0" + VALUE "FileDescription", "cyberhybridhub" "\0" + VALUE "FileVersion", VERSION_AS_STRING "\0" + VALUE "InternalName", "cyberhybridhub" "\0" + VALUE "LegalCopyright", "Copyright (C) 2026 com.cyberhybridhub. All rights reserved." "\0" + VALUE "OriginalFilename", "cyberhybridhub.exe" "\0" + VALUE "ProductName", "cyberhybridhub" "\0" + VALUE "ProductVersion", VERSION_AS_STRING "\0" + END + END + BLOCK "VarFileInfo" + BEGIN + VALUE "Translation", 0x409, 1252 + END +END + +#endif // English (United States) resources +///////////////////////////////////////////////////////////////////////////// + + + +#ifndef APSTUDIO_INVOKED +///////////////////////////////////////////////////////////////////////////// +// +// Generated from the TEXTINCLUDE 3 resource. +// + + +///////////////////////////////////////////////////////////////////////////// +#endif // not APSTUDIO_INVOKED diff --git a/windows/runner/flutter_window.cpp b/windows/runner/flutter_window.cpp new file mode 100644 index 0000000..955ee30 --- /dev/null +++ b/windows/runner/flutter_window.cpp @@ -0,0 +1,71 @@ +#include "flutter_window.h" + +#include + +#include "flutter/generated_plugin_registrant.h" + +FlutterWindow::FlutterWindow(const flutter::DartProject& project) + : project_(project) {} + +FlutterWindow::~FlutterWindow() {} + +bool FlutterWindow::OnCreate() { + if (!Win32Window::OnCreate()) { + return false; + } + + RECT frame = GetClientArea(); + + // The size here must match the window dimensions to avoid unnecessary surface + // creation / destruction in the startup path. + flutter_controller_ = std::make_unique( + frame.right - frame.left, frame.bottom - frame.top, project_); + // Ensure that basic setup of the controller was successful. + if (!flutter_controller_->engine() || !flutter_controller_->view()) { + return false; + } + RegisterPlugins(flutter_controller_->engine()); + SetChildContent(flutter_controller_->view()->GetNativeWindow()); + + flutter_controller_->engine()->SetNextFrameCallback([&]() { + this->Show(); + }); + + // Flutter can complete the first frame before the "show window" callback is + // registered. The following call ensures a frame is pending to ensure the + // window is shown. It is a no-op if the first frame hasn't completed yet. + flutter_controller_->ForceRedraw(); + + return true; +} + +void FlutterWindow::OnDestroy() { + if (flutter_controller_) { + flutter_controller_ = nullptr; + } + + Win32Window::OnDestroy(); +} + +LRESULT +FlutterWindow::MessageHandler(HWND hwnd, UINT const message, + WPARAM const wparam, + LPARAM const lparam) noexcept { + // Give Flutter, including plugins, an opportunity to handle window messages. + if (flutter_controller_) { + std::optional result = + flutter_controller_->HandleTopLevelWindowProc(hwnd, message, wparam, + lparam); + if (result) { + return *result; + } + } + + switch (message) { + case WM_FONTCHANGE: + flutter_controller_->engine()->ReloadSystemFonts(); + break; + } + + return Win32Window::MessageHandler(hwnd, message, wparam, lparam); +} diff --git a/windows/runner/flutter_window.h b/windows/runner/flutter_window.h new file mode 100644 index 0000000..6da0652 --- /dev/null +++ b/windows/runner/flutter_window.h @@ -0,0 +1,33 @@ +#ifndef RUNNER_FLUTTER_WINDOW_H_ +#define RUNNER_FLUTTER_WINDOW_H_ + +#include +#include + +#include + +#include "win32_window.h" + +// A window that does nothing but host a Flutter view. +class FlutterWindow : public Win32Window { + public: + // Creates a new FlutterWindow hosting a Flutter view running |project|. + explicit FlutterWindow(const flutter::DartProject& project); + virtual ~FlutterWindow(); + + protected: + // Win32Window: + bool OnCreate() override; + void OnDestroy() override; + LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, + LPARAM const lparam) noexcept override; + + private: + // The project to run. + flutter::DartProject project_; + + // The Flutter instance hosted by this window. + std::unique_ptr flutter_controller_; +}; + +#endif // RUNNER_FLUTTER_WINDOW_H_ diff --git a/windows/runner/main.cpp b/windows/runner/main.cpp new file mode 100644 index 0000000..98c6138 --- /dev/null +++ b/windows/runner/main.cpp @@ -0,0 +1,43 @@ +#include +#include +#include + +#include "flutter_window.h" +#include "utils.h" + +int APIENTRY wWinMain(_In_ HINSTANCE instance, _In_opt_ HINSTANCE prev, + _In_ wchar_t *command_line, _In_ int show_command) { + // Attach to console when present (e.g., 'flutter run') or create a + // new console when running with a debugger. + if (!::AttachConsole(ATTACH_PARENT_PROCESS) && ::IsDebuggerPresent()) { + CreateAndAttachConsole(); + } + + // Initialize COM, so that it is available for use in the library and/or + // plugins. + ::CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED); + + flutter::DartProject project(L"data"); + + std::vector command_line_arguments = + GetCommandLineArguments(); + + project.set_dart_entrypoint_arguments(std::move(command_line_arguments)); + + FlutterWindow window(project); + Win32Window::Point origin(10, 10); + Win32Window::Size size(1280, 720); + if (!window.Create(L"cyberhybridhub", origin, size)) { + return EXIT_FAILURE; + } + window.SetQuitOnClose(true); + + ::MSG msg; + while (::GetMessage(&msg, nullptr, 0, 0)) { + ::TranslateMessage(&msg); + ::DispatchMessage(&msg); + } + + ::CoUninitialize(); + return EXIT_SUCCESS; +} diff --git a/windows/runner/resource.h b/windows/runner/resource.h new file mode 100644 index 0000000..66a65d1 --- /dev/null +++ b/windows/runner/resource.h @@ -0,0 +1,16 @@ +//{{NO_DEPENDENCIES}} +// Microsoft Visual C++ generated include file. +// Used by Runner.rc +// +#define IDI_APP_ICON 101 + +// Next default values for new objects +// +#ifdef APSTUDIO_INVOKED +#ifndef APSTUDIO_READONLY_SYMBOLS +#define _APS_NEXT_RESOURCE_VALUE 102 +#define _APS_NEXT_COMMAND_VALUE 40001 +#define _APS_NEXT_CONTROL_VALUE 1001 +#define _APS_NEXT_SYMED_VALUE 101 +#endif +#endif diff --git a/windows/runner/resources/app_icon.ico b/windows/runner/resources/app_icon.ico new file mode 100644 index 0000000..c04e20c Binary files /dev/null and b/windows/runner/resources/app_icon.ico differ diff --git a/windows/runner/runner.exe.manifest b/windows/runner/runner.exe.manifest new file mode 100644 index 0000000..153653e --- /dev/null +++ b/windows/runner/runner.exe.manifest @@ -0,0 +1,14 @@ + + + + + PerMonitorV2 + + + + + + + + + diff --git a/windows/runner/utils.cpp b/windows/runner/utils.cpp new file mode 100644 index 0000000..3cb7146 --- /dev/null +++ b/windows/runner/utils.cpp @@ -0,0 +1,69 @@ +#include "utils.h" + +#include +#include +#include +#include + +#include + +void CreateAndAttachConsole() { + if (::AllocConsole()) { + FILE *unused; + if (freopen_s(&unused, "CONOUT$", "w", stdout)) { + _dup2(_fileno(stdout), 1); + } + if (freopen_s(&unused, "CONOUT$", "w", stderr)) { + _dup2(_fileno(stdout), 2); + } + std::ios::sync_with_stdio(); + FlutterDesktopResyncOutputStreams(); + } +} + +std::vector GetCommandLineArguments() { + // Convert the UTF-16 command line arguments to UTF-8 for the Engine to use. + int argc; + wchar_t** argv = ::CommandLineToArgvW(::GetCommandLineW(), &argc); + if (argv == nullptr) { + return std::vector(); + } + + std::vector command_line_arguments; + + // Skip the first argument as it's the binary name. + for (int i = 1; i < argc; i++) { + command_line_arguments.push_back(Utf8FromUtf16(argv[i])); + } + + ::LocalFree(argv); + + return command_line_arguments; +} + +std::string Utf8FromUtf16(const wchar_t* utf16_string) { + if (utf16_string == nullptr) { + return std::string(); + } + // First, find the length of the string with a safe upper bound (CWE-126). + // UNICODE_STRING_MAX_CHARS (32767) is the maximum length of a UNICODE_STRING. + int input_length = static_cast(wcsnlen(utf16_string, UNICODE_STRING_MAX_CHARS)); + // Now use that bounded length to determine the required buffer size. + // When an explicit length is passed, WideCharToMultiByte does not include + // the null terminator in its returned size. + int target_length = ::WideCharToMultiByte( + CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, + input_length, nullptr, 0, nullptr, nullptr); + std::string utf8_string; + if (target_length == 0 || static_cast(target_length) > utf8_string.max_size()) { + return utf8_string; + } + utf8_string.resize(target_length); + int converted_length = ::WideCharToMultiByte( + CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, + input_length, utf8_string.data(), target_length, nullptr, nullptr); + if (converted_length == 0) { + return std::string(); + } + return utf8_string; +} diff --git a/windows/runner/utils.h b/windows/runner/utils.h new file mode 100644 index 0000000..3879d54 --- /dev/null +++ b/windows/runner/utils.h @@ -0,0 +1,19 @@ +#ifndef RUNNER_UTILS_H_ +#define RUNNER_UTILS_H_ + +#include +#include + +// Creates a console for the process, and redirects stdout and stderr to +// it for both the runner and the Flutter library. +void CreateAndAttachConsole(); + +// Takes a null-terminated wchar_t* encoded in UTF-16 and returns a std::string +// encoded in UTF-8. Returns an empty std::string on failure. +std::string Utf8FromUtf16(const wchar_t* utf16_string); + +// Gets the command line arguments passed in as a std::vector, +// encoded in UTF-8. Returns an empty std::vector on failure. +std::vector GetCommandLineArguments(); + +#endif // RUNNER_UTILS_H_ diff --git a/windows/runner/win32_window.cpp b/windows/runner/win32_window.cpp new file mode 100644 index 0000000..60608d0 --- /dev/null +++ b/windows/runner/win32_window.cpp @@ -0,0 +1,288 @@ +#include "win32_window.h" + +#include +#include + +#include "resource.h" + +namespace { + +/// Window attribute that enables dark mode window decorations. +/// +/// Redefined in case the developer's machine has a Windows SDK older than +/// version 10.0.22000.0. +/// See: https://docs.microsoft.com/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute +#ifndef DWMWA_USE_IMMERSIVE_DARK_MODE +#define DWMWA_USE_IMMERSIVE_DARK_MODE 20 +#endif + +constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW"; + +/// Registry key for app theme preference. +/// +/// A value of 0 indicates apps should use dark mode. A non-zero or missing +/// value indicates apps should use light mode. +constexpr const wchar_t kGetPreferredBrightnessRegKey[] = + L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize"; +constexpr const wchar_t kGetPreferredBrightnessRegValue[] = L"AppsUseLightTheme"; + +// The number of Win32Window objects that currently exist. +static int g_active_window_count = 0; + +using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd); + +// Scale helper to convert logical scaler values to physical using passed in +// scale factor +int Scale(int source, double scale_factor) { + return static_cast(source * scale_factor); +} + +// Dynamically loads the |EnableNonClientDpiScaling| from the User32 module. +// This API is only needed for PerMonitor V1 awareness mode. +void EnableFullDpiSupportIfAvailable(HWND hwnd) { + HMODULE user32_module = LoadLibraryA("User32.dll"); + if (!user32_module) { + return; + } + auto enable_non_client_dpi_scaling = + reinterpret_cast( + GetProcAddress(user32_module, "EnableNonClientDpiScaling")); + if (enable_non_client_dpi_scaling != nullptr) { + enable_non_client_dpi_scaling(hwnd); + } + FreeLibrary(user32_module); +} + +} // namespace + +// Manages the Win32Window's window class registration. +class WindowClassRegistrar { + public: + ~WindowClassRegistrar() = default; + + // Returns the singleton registrar instance. + static WindowClassRegistrar* GetInstance() { + if (!instance_) { + instance_ = new WindowClassRegistrar(); + } + return instance_; + } + + // Returns the name of the window class, registering the class if it hasn't + // previously been registered. + const wchar_t* GetWindowClass(); + + // Unregisters the window class. Should only be called if there are no + // instances of the window. + void UnregisterWindowClass(); + + private: + WindowClassRegistrar() = default; + + static WindowClassRegistrar* instance_; + + bool class_registered_ = false; +}; + +WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr; + +const wchar_t* WindowClassRegistrar::GetWindowClass() { + if (!class_registered_) { + WNDCLASS window_class{}; + window_class.hCursor = LoadCursor(nullptr, IDC_ARROW); + window_class.lpszClassName = kWindowClassName; + window_class.style = CS_HREDRAW | CS_VREDRAW; + window_class.cbClsExtra = 0; + window_class.cbWndExtra = 0; + window_class.hInstance = GetModuleHandle(nullptr); + window_class.hIcon = + LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON)); + window_class.hbrBackground = 0; + window_class.lpszMenuName = nullptr; + window_class.lpfnWndProc = Win32Window::WndProc; + RegisterClass(&window_class); + class_registered_ = true; + } + return kWindowClassName; +} + +void WindowClassRegistrar::UnregisterWindowClass() { + UnregisterClass(kWindowClassName, nullptr); + class_registered_ = false; +} + +Win32Window::Win32Window() { + ++g_active_window_count; +} + +Win32Window::~Win32Window() { + --g_active_window_count; + Destroy(); +} + +bool Win32Window::Create(const std::wstring& title, + const Point& origin, + const Size& size) { + Destroy(); + + const wchar_t* window_class = + WindowClassRegistrar::GetInstance()->GetWindowClass(); + + const POINT target_point = {static_cast(origin.x), + static_cast(origin.y)}; + HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST); + UINT dpi = FlutterDesktopGetDpiForMonitor(monitor); + double scale_factor = dpi / 96.0; + + HWND window = CreateWindow( + window_class, title.c_str(), WS_OVERLAPPEDWINDOW, + Scale(origin.x, scale_factor), Scale(origin.y, scale_factor), + Scale(size.width, scale_factor), Scale(size.height, scale_factor), + nullptr, nullptr, GetModuleHandle(nullptr), this); + + if (!window) { + return false; + } + + UpdateTheme(window); + + return OnCreate(); +} + +bool Win32Window::Show() { + return ShowWindow(window_handle_, SW_SHOWNORMAL); +} + +// static +LRESULT CALLBACK Win32Window::WndProc(HWND const window, + UINT const message, + WPARAM const wparam, + LPARAM const lparam) noexcept { + if (message == WM_NCCREATE) { + auto window_struct = reinterpret_cast(lparam); + SetWindowLongPtr(window, GWLP_USERDATA, + reinterpret_cast(window_struct->lpCreateParams)); + + auto that = static_cast(window_struct->lpCreateParams); + EnableFullDpiSupportIfAvailable(window); + that->window_handle_ = window; + } else if (Win32Window* that = GetThisFromHandle(window)) { + return that->MessageHandler(window, message, wparam, lparam); + } + + return DefWindowProc(window, message, wparam, lparam); +} + +LRESULT +Win32Window::MessageHandler(HWND hwnd, + UINT const message, + WPARAM const wparam, + LPARAM const lparam) noexcept { + switch (message) { + case WM_DESTROY: + window_handle_ = nullptr; + Destroy(); + if (quit_on_close_) { + PostQuitMessage(0); + } + return 0; + + case WM_DPICHANGED: { + auto newRectSize = reinterpret_cast(lparam); + LONG newWidth = newRectSize->right - newRectSize->left; + LONG newHeight = newRectSize->bottom - newRectSize->top; + + SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth, + newHeight, SWP_NOZORDER | SWP_NOACTIVATE); + + return 0; + } + case WM_SIZE: { + RECT rect = GetClientArea(); + if (child_content_ != nullptr) { + // Size and position the child window. + MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left, + rect.bottom - rect.top, TRUE); + } + return 0; + } + + case WM_ACTIVATE: + if (child_content_ != nullptr) { + SetFocus(child_content_); + } + return 0; + + case WM_DWMCOLORIZATIONCOLORCHANGED: + UpdateTheme(hwnd); + return 0; + } + + return DefWindowProc(window_handle_, message, wparam, lparam); +} + +void Win32Window::Destroy() { + OnDestroy(); + + if (window_handle_) { + DestroyWindow(window_handle_); + window_handle_ = nullptr; + } + if (g_active_window_count == 0) { + WindowClassRegistrar::GetInstance()->UnregisterWindowClass(); + } +} + +Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept { + return reinterpret_cast( + GetWindowLongPtr(window, GWLP_USERDATA)); +} + +void Win32Window::SetChildContent(HWND content) { + child_content_ = content; + SetParent(content, window_handle_); + RECT frame = GetClientArea(); + + MoveWindow(content, frame.left, frame.top, frame.right - frame.left, + frame.bottom - frame.top, true); + + SetFocus(child_content_); +} + +RECT Win32Window::GetClientArea() { + RECT frame; + GetClientRect(window_handle_, &frame); + return frame; +} + +HWND Win32Window::GetHandle() { + return window_handle_; +} + +void Win32Window::SetQuitOnClose(bool quit_on_close) { + quit_on_close_ = quit_on_close; +} + +bool Win32Window::OnCreate() { + // No-op; provided for subclasses. + return true; +} + +void Win32Window::OnDestroy() { + // No-op; provided for subclasses. +} + +void Win32Window::UpdateTheme(HWND const window) { + DWORD light_mode; + DWORD light_mode_size = sizeof(light_mode); + LSTATUS result = RegGetValue(HKEY_CURRENT_USER, kGetPreferredBrightnessRegKey, + kGetPreferredBrightnessRegValue, + RRF_RT_REG_DWORD, nullptr, &light_mode, + &light_mode_size); + + if (result == ERROR_SUCCESS) { + BOOL enable_dark_mode = light_mode == 0; + DwmSetWindowAttribute(window, DWMWA_USE_IMMERSIVE_DARK_MODE, + &enable_dark_mode, sizeof(enable_dark_mode)); + } +} diff --git a/windows/runner/win32_window.h b/windows/runner/win32_window.h new file mode 100644 index 0000000..e901dde --- /dev/null +++ b/windows/runner/win32_window.h @@ -0,0 +1,102 @@ +#ifndef RUNNER_WIN32_WINDOW_H_ +#define RUNNER_WIN32_WINDOW_H_ + +#include + +#include +#include +#include + +// A class abstraction for a high DPI-aware Win32 Window. Intended to be +// inherited from by classes that wish to specialize with custom +// rendering and input handling +class Win32Window { + public: + struct Point { + unsigned int x; + unsigned int y; + Point(unsigned int x, unsigned int y) : x(x), y(y) {} + }; + + struct Size { + unsigned int width; + unsigned int height; + Size(unsigned int width, unsigned int height) + : width(width), height(height) {} + }; + + Win32Window(); + virtual ~Win32Window(); + + // Creates a win32 window with |title| that is positioned and sized using + // |origin| and |size|. New windows are created on the default monitor. Window + // sizes are specified to the OS in physical pixels, hence to ensure a + // consistent size this function will scale the inputted width and height as + // as appropriate for the default monitor. The window is invisible until + // |Show| is called. Returns true if the window was created successfully. + bool Create(const std::wstring& title, const Point& origin, const Size& size); + + // Show the current window. Returns true if the window was successfully shown. + bool Show(); + + // Release OS resources associated with window. + void Destroy(); + + // Inserts |content| into the window tree. + void SetChildContent(HWND content); + + // Returns the backing Window handle to enable clients to set icon and other + // window properties. Returns nullptr if the window has been destroyed. + HWND GetHandle(); + + // If true, closing this window will quit the application. + void SetQuitOnClose(bool quit_on_close); + + // Return a RECT representing the bounds of the current client area. + RECT GetClientArea(); + + protected: + // Processes and route salient window messages for mouse handling, + // size change and DPI. Delegates handling of these to member overloads that + // inheriting classes can handle. + virtual LRESULT MessageHandler(HWND window, + UINT const message, + WPARAM const wparam, + LPARAM const lparam) noexcept; + + // Called when CreateAndShow is called, allowing subclass window-related + // setup. Subclasses should return false if setup fails. + virtual bool OnCreate(); + + // Called when Destroy is called. + virtual void OnDestroy(); + + private: + friend class WindowClassRegistrar; + + // OS callback called by message pump. Handles the WM_NCCREATE message which + // is passed when the non-client area is being created and enables automatic + // non-client DPI scaling so that the non-client area automatically + // responds to changes in DPI. All other messages are handled by + // MessageHandler. + static LRESULT CALLBACK WndProc(HWND const window, + UINT const message, + WPARAM const wparam, + LPARAM const lparam) noexcept; + + // Retrieves a class instance pointer for |window| + static Win32Window* GetThisFromHandle(HWND const window) noexcept; + + // Update the window frame's theme to match the system theme. + static void UpdateTheme(HWND const window); + + bool quit_on_close_ = false; + + // window handle for top level window. + HWND window_handle_ = nullptr; + + // window handle for hosted content. + HWND child_content_ = nullptr; +}; + +#endif // RUNNER_WIN32_WINDOW_H_