diff --git a/README.md b/README.md index 48a05e012..d08f74474 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,3 @@ - # Define 3.0 The official template repository for Define 3.0 @@ -6,35 +5,51 @@ The official template repository for Define 3.0 -# < Project Name > - Cover Image If applicable +# FINFLOW ### Team Information -- **Team Name**: -- **Track**: < Track > +- *Team Name*: DEBUG DRAGONS +- *Track*: FINTECH ### Team Members | Name | Role | GitHub | LinkedIn | |------|------|--------|----------| -| [Full Name] | [Role] | [@username](https://github.com/username) | [Profile](https://linkedin.com/in/username) | -| [Full Name] | [Role] | [@username](https://github.com/username) | [Profile](https://linkedin.com/in/username) | -| [Full Name] | [Role] | [@username](https://github.com/username) | [Profile](https://linkedin.com/in/username) | -| [Full Name] | [Role] | [@username](https://github.com/username) | [Profile](https://linkedin.com/in/username) | +| MUTHUVEL A | BACKEND DEVELOPMENT | [MuthuvelA](https://github.com/muthuvela) | [Profile](https://linkedin.com/in/muthuvela) | +| KAMALESH R | BACKEND DEVELOPMENT | [Kamaleshbala01](https://github.com/kamaleshbala01) | [Profile](https://linkedin.com/in/kamaleshbala) | +| PRAKASH KUMAR M | FRONTEND DEVELOPMENT | [Pra-Kas](https://github.com/Pra-kas) | [Profile](https://linkedin.com/in/prakashkumar-m) | +| GOPINATH S | FRONTEND DEVELOPMENT | [igopi77](https://github.com/igopi77) | [Profile](https://linkedin.com/in/gopinath-siva-191a27259) | ## Project Details ### Overview -_A concise summary of your project (2-3 sentences)_ +1. Mobile Application to assist user on financial FAQs +2. Dashboard to visualise the income and expense ### Problem Statement -_Describe the problem your project aims to solve_ +Develop a basic prototype of a fintech website with an AI-powered chatbot that provides users with financial insights and customer support. The website should have a clean, user-friendly interface with a simple dashboard displaying sample financial data and a chatbot capable of handling basic FAQs related to fintech services ### Solution -_Explain your approach to solving the problem_ +Start + Users register and integrate their past financial transactions with AI for personalized insights. + Only finance-related questions are allowed. +User Interaction & Data Input + Users can provide financial data such as expenses and income through: + Bills (image upload for AI processing) + Bank statements + Manual entry +AI-Powered Insights & Responses + AI processes financial data to answer finance-related queries. + Users receive personalized insights based on their financial history. + Follw-up questions can be asked to refine responses. +Financial Management Features + AI-driven financial improvement suggestions are sent via push notifications. + An analytics dashboard visually summarizes the user’s financial history. +Finish + The chatbot ensures a seamless and interactive financial advisory experience, helping users make better financial decisions._ ### Demo [![Project Demo](https://img.youtube.com/vi/VIDEO_ID/0.jpg)](https://www.youtube.com/watch?v=VIDEO_ID) -_Replace VIDEO_ID with your YouTube video ID or provide an alternative demo link_ +Replace VIDEO_ID with your YouTube video ID or provide an alternative demo link ### Live Project [Project Name](https://your-project-url.com) @@ -42,17 +57,15 @@ _Replace VIDEO_ID with your YouTube video ID or provide an alternative demo link ## Technical Implementation ### Technologies Used -- **Frontend**: [Technologies] -- **Backend**: [Technologies] -- **Database**: [Technologies] -- **APIs**: [Technologies] -- **DevOps**: [Technologies] -- **Other Tools**: [Technologies] +- *Frontend*: [Flutter] +- *Backend*: [Nodejs,Express] +- *Database*: [MongoDB] +- *APIs*: [GeminiAPI] ### Key Features -- Feature 1 -- Feature 2 -- Feature 3 +- Recurring items automatic suggestion +- Personalised insights +- Data extraction from document ## Setup Instructions @@ -62,25 +75,25 @@ _Replace VIDEO_ID with your YouTube video ID or provide an alternative demo link - Requirement 3 ### Installation -```bash +bash + -``` ### Running the Project -```bash +bash + -``` ## Additional Resources ### Project Timeline -_Brief overview of your development process and milestones_ +Brief overview of your development process and milestones ### Challenges Faced -_Discuss technical challenges and how you overcame them_ +Discuss technical challenges and how you overcame them ### Future Enhancements -_Share your vision for future development_ +Share your vision for future development ### References (if any) - [Reference 1](link) diff --git a/flutter_application_1/.gitignore b/flutter_application_1/.gitignore new file mode 100644 index 000000000..79c113f9b --- /dev/null +++ b/flutter_application_1/.gitignore @@ -0,0 +1,45 @@ +# Miscellaneous +*.class +*.log +*.pyc +*.swp +.DS_Store +.atom/ +.build/ +.buildlog/ +.history +.svn/ +.swiftpm/ +migrate_working_dir/ + +# IntelliJ related +*.iml +*.ipr +*.iws +.idea/ + +# The .vscode folder contains launch configuration and tasks you configure in +# VS Code which you may wish to be included in version control, so this line +# is commented out by default. +#.vscode/ + +# Flutter/Dart/Pub related +**/doc/api/ +**/ios/Flutter/.last_build_id +.dart_tool/ +.flutter-plugins +.flutter-plugins-dependencies +.pub-cache/ +.pub/ +/build/ + +# Symbolication related +app.*.symbols + +# Obfuscation related +app.*.map.json + +# Android Studio will place build artifacts here +/android/app/debug +/android/app/profile +/android/app/release diff --git a/flutter_application_1/.metadata b/flutter_application_1/.metadata new file mode 100644 index 000000000..1f19d868c --- /dev/null +++ b/flutter_application_1/.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: "c519ee916eaeb88923e67befb89c0f1dabfa83e6" + channel: "stable" + +project_type: app + +# Tracks metadata for the flutter migrate command +migration: + platforms: + - platform: root + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: android + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: ios + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: linux + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: macos + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: web + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + - platform: windows + create_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + base_revision: c519ee916eaeb88923e67befb89c0f1dabfa83e6 + + # 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/flutter_application_1/README.md b/flutter_application_1/README.md new file mode 100644 index 000000000..2c8a8aef4 --- /dev/null +++ b/flutter_application_1/README.md @@ -0,0 +1,16 @@ +# flutter_application_1 + +A new Flutter project. + +## Getting Started + +This project is a starting point for a Flutter application. + +A few resources to get you started if this is your first Flutter project: + +- [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab) +- [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook) + +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/flutter_application_1/analysis_options.yaml b/flutter_application_1/analysis_options.yaml new file mode 100644 index 000000000..0d2902135 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/android/.gitignore b/flutter_application_1/android/.gitignore new file mode 100644 index 000000000..55afd919c --- /dev/null +++ b/flutter_application_1/android/.gitignore @@ -0,0 +1,13 @@ +gradle-wrapper.jar +/.gradle +/captures/ +/gradlew +/gradlew.bat +/local.properties +GeneratedPluginRegistrant.java + +# Remember to never publicly share your keystore. +# See https://flutter.dev/to/reference-keystore +key.properties +**/*.keystore +**/*.jks diff --git a/flutter_application_1/android/app/build.gradle b/flutter_application_1/android/app/build.gradle new file mode 100644 index 000000000..cd44dac59 --- /dev/null +++ b/flutter_application_1/android/app/build.gradle @@ -0,0 +1,66 @@ +plugins { + id "com.android.application" + id "kotlin-android" + // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. + id "dev.flutter.flutter-gradle-plugin" + id "com.google.gms.google-services" // Google services for Firebase +} + +android { + namespace = "com.example.flutter_application_1" + compileSdk = flutter.compileSdkVersion + ndkVersion = flutter.ndkVersion + + signingConfigs { + release { + storeFile file("upload-keystore.jks") + storePassword "Gopinath S" + keyAlias "upload" + keyPassword "Gopinath S" + } + } + buildTypes { + release { + signingConfig signingConfigs.release + minifyEnabled false + shrinkResources false + } + } + + compileOptions { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 + } + + kotlinOptions { + jvmTarget = JavaVersion.VERSION_1_8 + } + + defaultConfig { + // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). + applicationId = "com.example.flutter_application_1" + // You can update the following values to match your application needs. + // For more information, see: https://flutter.dev/to/review-gradle-config. + minSdk = 23 + 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.debug + } + } +} + +flutter { + source = "../.." +} + +dependencies { + implementation platform('com.google.firebase:firebase-bom:32.7.0') // Firebase BOM (use latest version) + implementation 'com.google.firebase:firebase-messaging' // Firebase Cloud Messaging +} diff --git a/flutter_application_1/android/app/google-services.json b/flutter_application_1/android/app/google-services.json new file mode 100644 index 000000000..6f2a45869 --- /dev/null +++ b/flutter_application_1/android/app/google-services.json @@ -0,0 +1,48 @@ +{ + "project_info": { + "project_number": "101217440706", + "project_id": "finnfloww", + "storage_bucket": "finnfloww.firebasestorage.app" + }, + "client": [ + { + "client_info": { + "mobilesdk_app_id": "1:101217440706:android:a9c09bb6b54d47d08b7c5b", + "android_client_info": { + "package_name": "com.example.flutter_application_1" + } + }, + "oauth_client": [], + "api_key": [ + { + "current_key": "AIzaSyDFcAakxrfNnQsARzoFKAKhohW_-nMn-CE" + } + ], + "services": { + "appinvite_service": { + "other_platform_oauth_client": [] + } + } + }, + { + "client_info": { + "mobilesdk_app_id": "1:101217440706:android:c3004202794325658b7c5b", + "android_client_info": { + "package_name": "com.finnfloww.app" + } + }, + "oauth_client": [], + "api_key": [ + { + "current_key": "AIzaSyDFcAakxrfNnQsARzoFKAKhohW_-nMn-CE" + } + ], + "services": { + "appinvite_service": { + "other_platform_oauth_client": [] + } + } + } + ], + "configuration_version": "1" +} \ No newline at end of file diff --git a/flutter_application_1/android/app/src/debug/AndroidManifest.xml b/flutter_application_1/android/app/src/debug/AndroidManifest.xml new file mode 100644 index 000000000..399f6981d --- /dev/null +++ b/flutter_application_1/android/app/src/debug/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/flutter_application_1/android/app/src/main/AndroidManifest.xml b/flutter_application_1/android/app/src/main/AndroidManifest.xml new file mode 100644 index 000000000..c814c9954 --- /dev/null +++ b/flutter_application_1/android/app/src/main/AndroidManifest.xml @@ -0,0 +1,55 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/android/app/src/main/kotlin/com/example/flutter_application_1/MainActivity.kt b/flutter_application_1/android/app/src/main/kotlin/com/example/flutter_application_1/MainActivity.kt new file mode 100644 index 000000000..775170412 --- /dev/null +++ b/flutter_application_1/android/app/src/main/kotlin/com/example/flutter_application_1/MainActivity.kt @@ -0,0 +1,5 @@ +package com.example.flutter_application_1 + +import io.flutter.embedding.android.FlutterActivity + +class MainActivity: FlutterActivity() diff --git a/flutter_application_1/android/app/src/main/res/drawable-v21/launch_background.xml b/flutter_application_1/android/app/src/main/res/drawable-v21/launch_background.xml new file mode 100644 index 000000000..f74085f3f --- /dev/null +++ b/flutter_application_1/android/app/src/main/res/drawable-v21/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/flutter_application_1/android/app/src/main/res/drawable/launch_background.xml b/flutter_application_1/android/app/src/main/res/drawable/launch_background.xml new file mode 100644 index 000000000..304732f88 --- /dev/null +++ b/flutter_application_1/android/app/src/main/res/drawable/launch_background.xml @@ -0,0 +1,12 @@ + + + + + + + + diff --git a/flutter_application_1/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/flutter_application_1/android/app/src/main/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 000000000..ecad941a7 Binary files /dev/null and b/flutter_application_1/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ diff --git a/flutter_application_1/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/flutter_application_1/android/app/src/main/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 000000000..9ae7d4076 Binary files /dev/null and b/flutter_application_1/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ diff --git a/flutter_application_1/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/flutter_application_1/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 000000000..3d1d3e8dd Binary files /dev/null and b/flutter_application_1/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/flutter_application_1/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/flutter_application_1/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 000000000..d3d67e988 Binary files /dev/null and b/flutter_application_1/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/flutter_application_1/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/flutter_application_1/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 000000000..ee312861b Binary files /dev/null and b/flutter_application_1/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/flutter_application_1/android/app/src/main/res/values-night/styles.xml b/flutter_application_1/android/app/src/main/res/values-night/styles.xml new file mode 100644 index 000000000..06952be74 --- /dev/null +++ b/flutter_application_1/android/app/src/main/res/values-night/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/flutter_application_1/android/app/src/main/res/values/styles.xml b/flutter_application_1/android/app/src/main/res/values/styles.xml new file mode 100644 index 000000000..cb1ef8805 --- /dev/null +++ b/flutter_application_1/android/app/src/main/res/values/styles.xml @@ -0,0 +1,18 @@ + + + + + + + diff --git a/flutter_application_1/android/app/src/profile/AndroidManifest.xml b/flutter_application_1/android/app/src/profile/AndroidManifest.xml new file mode 100644 index 000000000..399f6981d --- /dev/null +++ b/flutter_application_1/android/app/src/profile/AndroidManifest.xml @@ -0,0 +1,7 @@ + + + + diff --git a/flutter_application_1/android/build.gradle b/flutter_application_1/android/build.gradle new file mode 100644 index 000000000..5f48c79db --- /dev/null +++ b/flutter_application_1/android/build.gradle @@ -0,0 +1,28 @@ +buildscript { + repositories { + google() + mavenCentral() + } + dependencies { + classpath 'com.google.gms:google-services:4.4.2' // Firebase Google services + } +} + +allprojects { + repositories { + google() + mavenCentral() + } +} + +rootProject.buildDir = "../build" +subprojects { + project.buildDir = "${rootProject.buildDir}/${project.name}" +} +subprojects { + project.evaluationDependsOn(":app") +} + +tasks.register("clean", Delete) { + delete rootProject.buildDir +} diff --git a/flutter_application_1/android/gradle.properties b/flutter_application_1/android/gradle.properties new file mode 100644 index 000000000..4f5d1e015 --- /dev/null +++ b/flutter_application_1/android/gradle.properties @@ -0,0 +1,4 @@ +org.gradle.jvmargs=-Xmx4G -XX:MaxMetaspaceSize=2G -XX:+HeapDumpOnOutOfMemoryError +android.useAndroidX=true +android.enableJetifier=true +org.gradle.java.home= C:/Program Files/Java/jdk-17 \ No newline at end of file diff --git a/flutter_application_1/android/gradle/wrapper/gradle-wrapper.properties b/flutter_application_1/android/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 000000000..7bb2df6ba --- /dev/null +++ b/flutter_application_1/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-8.3-all.zip diff --git a/flutter_application_1/android/settings.gradle b/flutter_application_1/android/settings.gradle new file mode 100644 index 000000000..9759a220b --- /dev/null +++ b/flutter_application_1/android/settings.gradle @@ -0,0 +1,28 @@ +pluginManagement { + def flutterSdkPath = { + def properties = new Properties() + file("local.properties").withInputStream { properties.load(it) } + def flutterSdkPath = properties.getProperty("flutter.sdk") + assert flutterSdkPath != null, "flutter.sdk not set in local.properties" + return flutterSdkPath + }() + + includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") + + repositories { + google() + mavenCentral() + gradlePluginPortal() + } +} + +plugins { + id "dev.flutter.flutter-plugin-loader" version "1.0.0" + id "com.android.application" version "8.1.0" apply false + // START: FlutterFire Configuration + id "com.google.gms.google-services" version "4.3.15" apply false + // END: FlutterFire Configuration + id "org.jetbrains.kotlin.android" version "1.8.22" apply false +} + +include ":app" diff --git a/flutter_application_1/assets/fonts/Satoshi-Black.ttf b/flutter_application_1/assets/fonts/Satoshi-Black.ttf new file mode 100644 index 000000000..7777885fc Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Black.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-BlackItalic.ttf b/flutter_application_1/assets/fonts/Satoshi-BlackItalic.ttf new file mode 100644 index 000000000..bbd30776a Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-BlackItalic.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-Bold.ttf b/flutter_application_1/assets/fonts/Satoshi-Bold.ttf new file mode 100644 index 000000000..bc8a9a0ba Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Bold.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-BoldItalic.ttf b/flutter_application_1/assets/fonts/Satoshi-BoldItalic.ttf new file mode 100644 index 000000000..08b262dde Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-BoldItalic.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-Italic.ttf b/flutter_application_1/assets/fonts/Satoshi-Italic.ttf new file mode 100644 index 000000000..2c4fc3f96 Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Italic.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-Light.ttf b/flutter_application_1/assets/fonts/Satoshi-Light.ttf new file mode 100644 index 000000000..b116dd8d3 Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Light.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-LightItalic.ttf b/flutter_application_1/assets/fonts/Satoshi-LightItalic.ttf new file mode 100644 index 000000000..a77c5c26e Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-LightItalic.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-Medium.ttf b/flutter_application_1/assets/fonts/Satoshi-Medium.ttf new file mode 100644 index 000000000..f57b57aa4 Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Medium.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-MediumItalic.ttf b/flutter_application_1/assets/fonts/Satoshi-MediumItalic.ttf new file mode 100644 index 000000000..b596520aa Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-MediumItalic.ttf differ diff --git a/flutter_application_1/assets/fonts/Satoshi-Regular.ttf b/flutter_application_1/assets/fonts/Satoshi-Regular.ttf new file mode 100644 index 000000000..2d6788f60 Binary files /dev/null and b/flutter_application_1/assets/fonts/Satoshi-Regular.ttf differ diff --git a/flutter_application_1/assets/images/applogo.png b/flutter_application_1/assets/images/applogo.png new file mode 100644 index 000000000..682520353 Binary files /dev/null and b/flutter_application_1/assets/images/applogo.png differ diff --git a/flutter_application_1/devtools_options.yaml b/flutter_application_1/devtools_options.yaml new file mode 100644 index 000000000..fa0b357c4 --- /dev/null +++ b/flutter_application_1/devtools_options.yaml @@ -0,0 +1,3 @@ +description: This file stores settings for Dart & Flutter DevTools. +documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states +extensions: diff --git a/flutter_application_1/ios/.gitignore b/flutter_application_1/ios/.gitignore new file mode 100644 index 000000000..7a7f9873a --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/ios/Flutter/AppFrameworkInfo.plist b/flutter_application_1/ios/Flutter/AppFrameworkInfo.plist new file mode 100644 index 000000000..7c5696400 --- /dev/null +++ b/flutter_application_1/ios/Flutter/AppFrameworkInfo.plist @@ -0,0 +1,26 @@ + + + + + CFBundleDevelopmentRegion + en + CFBundleExecutable + App + CFBundleIdentifier + io.flutter.flutter.app + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + App + CFBundlePackageType + FMWK + CFBundleShortVersionString + 1.0 + CFBundleSignature + ???? + CFBundleVersion + 1.0 + MinimumOSVersion + 12.0 + + diff --git a/flutter_application_1/ios/Flutter/Debug.xcconfig b/flutter_application_1/ios/Flutter/Debug.xcconfig new file mode 100644 index 000000000..592ceee85 --- /dev/null +++ b/flutter_application_1/ios/Flutter/Debug.xcconfig @@ -0,0 +1 @@ +#include "Generated.xcconfig" diff --git a/flutter_application_1/ios/Flutter/Release.xcconfig b/flutter_application_1/ios/Flutter/Release.xcconfig new file mode 100644 index 000000000..592ceee85 --- /dev/null +++ b/flutter_application_1/ios/Flutter/Release.xcconfig @@ -0,0 +1 @@ +#include "Generated.xcconfig" diff --git a/flutter_application_1/ios/Runner.xcodeproj/project.pbxproj b/flutter_application_1/ios/Runner.xcodeproj/project.pbxproj new file mode 100644 index 000000000..66696842c --- /dev/null +++ b/flutter_application_1/ios/Runner.xcodeproj/project.pbxproj @@ -0,0 +1,616 @@ +// !$*UTF8*$! +{ + archiveVersion = 1; + classes = { + }; + objectVersion = 54; + objects = { + +/* Begin PBXBuildFile section */ + 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; }; + 331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; + 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; + 74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; + 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; + 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; + 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; }; +/* End PBXBuildFile section */ + +/* Begin PBXContainerItemProxy section */ + 331C8085294A63A400263BE5 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 97C146E61CF9000F007C117D /* Project object */; + proxyType = 1; + remoteGlobalIDString = 97C146ED1CF9000F007C117D; + remoteInfo = Runner; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + 9705A1C41CF9048500538489 /* Embed Frameworks */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 10; + files = ( + ); + name = "Embed Frameworks"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + +/* Begin PBXFileReference section */ + 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = ""; }; + 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = ""; }; + 331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = ""; }; + 331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = ""; }; + 74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = ""; }; + 74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = ""; }; + 9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = ""; }; + 9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = ""; }; + 97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; + 97C146FB1CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = ""; }; + 97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; + 97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; + 97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; +/* End PBXFileReference section */ + +/* Begin PBXFrameworksBuildPhase section */ + 97C146EB1CF9000F007C117D /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXFrameworksBuildPhase section */ + +/* Begin PBXGroup section */ + 331C8082294A63A400263BE5 /* RunnerTests */ = { + isa = PBXGroup; + children = ( + 331C807B294A618700263BE5 /* RunnerTests.swift */, + ); + path = RunnerTests; + sourceTree = ""; + }; + 9740EEB11CF90186004384FC /* Flutter */ = { + isa = PBXGroup; + children = ( + 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */, + 9740EEB21CF90195004384FC /* Debug.xcconfig */, + 7AFA3C8E1D35360C0083082E /* Release.xcconfig */, + 9740EEB31CF90195004384FC /* Generated.xcconfig */, + ); + name = Flutter; + sourceTree = ""; + }; + 97C146E51CF9000F007C117D = { + isa = PBXGroup; + children = ( + 9740EEB11CF90186004384FC /* Flutter */, + 97C146F01CF9000F007C117D /* Runner */, + 97C146EF1CF9000F007C117D /* Products */, + 331C8082294A63A400263BE5 /* RunnerTests */, + ); + sourceTree = ""; + }; + 97C146EF1CF9000F007C117D /* Products */ = { + isa = PBXGroup; + children = ( + 97C146EE1CF9000F007C117D /* Runner.app */, + 331C8081294A63A400263BE5 /* RunnerTests.xctest */, + ); + name = Products; + sourceTree = ""; + }; + 97C146F01CF9000F007C117D /* Runner */ = { + isa = PBXGroup; + children = ( + 97C146FA1CF9000F007C117D /* Main.storyboard */, + 97C146FD1CF9000F007C117D /* Assets.xcassets */, + 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */, + 97C147021CF9000F007C117D /* Info.plist */, + 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */, + 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */, + 74858FAE1ED2DC5600515810 /* AppDelegate.swift */, + 74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */, + ); + path = Runner; + sourceTree = ""; + }; +/* End PBXGroup section */ + +/* Begin PBXNativeTarget section */ + 331C8080294A63A400263BE5 /* RunnerTests */ = { + isa = PBXNativeTarget; + buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; + buildPhases = ( + 331C807D294A63A400263BE5 /* Sources */, + 331C807F294A63A400263BE5 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + 331C8086294A63A400263BE5 /* PBXTargetDependency */, + ); + name = RunnerTests; + productName = RunnerTests; + productReference = 331C8081294A63A400263BE5 /* RunnerTests.xctest */; + productType = "com.apple.product-type.bundle.unit-test"; + }; + 97C146ED1CF9000F007C117D /* Runner */ = { + isa = PBXNativeTarget; + buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; + buildPhases = ( + 9740EEB61CF901F6004384FC /* Run Script */, + 97C146EA1CF9000F007C117D /* Sources */, + 97C146EB1CF9000F007C117D /* Frameworks */, + 97C146EC1CF9000F007C117D /* Resources */, + 9705A1C41CF9048500538489 /* Embed Frameworks */, + 3B06AD1E1E4923F5004D2608 /* Thin Binary */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = Runner; + productName = Runner; + productReference = 97C146EE1CF9000F007C117D /* Runner.app */; + productType = "com.apple.product-type.application"; + }; +/* End PBXNativeTarget section */ + +/* Begin PBXProject section */ + 97C146E61CF9000F007C117D /* Project object */ = { + isa = PBXProject; + attributes = { + BuildIndependentTargetsInParallel = YES; + LastUpgradeCheck = 1510; + ORGANIZATIONNAME = ""; + TargetAttributes = { + 331C8080294A63A400263BE5 = { + CreatedOnToolsVersion = 14.0; + TestTargetID = 97C146ED1CF9000F007C117D; + }; + 97C146ED1CF9000F007C117D = { + CreatedOnToolsVersion = 7.3.1; + LastSwiftMigration = 1100; + }; + }; + }; + buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */; + compatibilityVersion = "Xcode 9.3"; + developmentRegion = en; + hasScannedForEncodings = 0; + knownRegions = ( + en, + Base, + ); + mainGroup = 97C146E51CF9000F007C117D; + productRefGroup = 97C146EF1CF9000F007C117D /* Products */; + projectDirPath = ""; + projectRoot = ""; + targets = ( + 97C146ED1CF9000F007C117D /* Runner */, + 331C8080294A63A400263BE5 /* RunnerTests */, + ); + }; +/* End PBXProject section */ + +/* Begin PBXResourcesBuildPhase section */ + 331C807F294A63A400263BE5 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 97C146EC1CF9000F007C117D /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */, + 3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */, + 97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */, + 97C146FC1CF9000F007C117D /* Main.storyboard in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXResourcesBuildPhase section */ + +/* Begin PBXShellScriptBuildPhase section */ + 3B06AD1E1E4923F5004D2608 /* Thin Binary */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputPaths = ( + "${TARGET_BUILD_DIR}/${INFOPLIST_PATH}", + ); + name = "Thin Binary"; + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" embed_and_thin"; + }; + 9740EEB61CF901F6004384FC /* Run Script */ = { + isa = PBXShellScriptBuildPhase; + alwaysOutOfDate = 1; + buildActionMask = 2147483647; + files = ( + ); + inputPaths = ( + ); + name = "Run Script"; + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; + }; +/* End PBXShellScriptBuildPhase section */ + +/* Begin PBXSourcesBuildPhase section */ + 331C807D294A63A400263BE5 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; + 97C146EA1CF9000F007C117D /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + 74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */, + 1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXSourcesBuildPhase section */ + +/* Begin PBXTargetDependency section */ + 331C8086294A63A400263BE5 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = 97C146ED1CF9000F007C117D /* Runner */; + targetProxy = 331C8085294A63A400263BE5 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + +/* Begin PBXVariantGroup section */ + 97C146FA1CF9000F007C117D /* Main.storyboard */ = { + isa = PBXVariantGroup; + children = ( + 97C146FB1CF9000F007C117D /* Base */, + ); + name = Main.storyboard; + sourceTree = ""; + }; + 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = { + isa = PBXVariantGroup; + children = ( + 97C147001CF9000F007C117D /* Base */, + ); + name = LaunchScreen.storyboard; + sourceTree = ""; + }; +/* End PBXVariantGroup section */ + +/* Begin XCBuildConfiguration section */ + 249021D3217E4FDB00AE95B9 /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = iphoneos; + SUPPORTED_PLATFORMS = iphoneos; + TARGETED_DEVICE_FAMILY = "1,2"; + VALIDATE_PRODUCT = YES; + }; + name = Profile; + }; + 249021D4217E4FDB00AE95B9 /* Profile */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.example.flutterApplication1; + 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.example.flutterApplication1.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.example.flutterApplication1.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.example.flutterApplication1.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/Runner.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/Runner"; + }; + name = Profile; + }; + 97C147031CF9000F007C117D /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = dwarf; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_TESTABILITY = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_DYNAMIC_NO_PIC = NO; + GCC_NO_COMMON_BLOCKS = YES; + GCC_OPTIMIZATION_LEVEL = 0; + GCC_PREPROCESSOR_DEFINITIONS = ( + "DEBUG=1", + "$(inherited)", + ); + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = YES; + ONLY_ACTIVE_ARCH = YES; + SDKROOT = iphoneos; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Debug; + }; + 97C147041CF9000F007C117D /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ALWAYS_SEARCH_USER_PATHS = NO; + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_ANALYZER_NONNULL = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x"; + CLANG_CXX_LIBRARY = "libc++"; + CLANG_ENABLE_MODULES = YES; + CLANG_ENABLE_OBJC_ARC = YES; + CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; + CLANG_WARN_BOOL_CONVERSION = YES; + CLANG_WARN_COMMA = YES; + CLANG_WARN_CONSTANT_CONVERSION = YES; + CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; + CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; + CLANG_WARN_EMPTY_BODY = YES; + CLANG_WARN_ENUM_CONVERSION = YES; + CLANG_WARN_INFINITE_RECURSION = YES; + CLANG_WARN_INT_CONVERSION = YES; + CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; + CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; + CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; + CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; + CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; + CLANG_WARN_STRICT_PROTOTYPES = YES; + CLANG_WARN_SUSPICIOUS_MOVE = YES; + CLANG_WARN_UNREACHABLE_CODE = YES; + CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; + "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; + COPY_PHASE_STRIP = NO; + DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; + ENABLE_NS_ASSERTIONS = NO; + ENABLE_STRICT_OBJC_MSGSEND = YES; + ENABLE_USER_SCRIPT_SANDBOXING = NO; + GCC_C_LANGUAGE_STANDARD = gnu99; + GCC_NO_COMMON_BLOCKS = YES; + GCC_WARN_64_TO_32_BIT_CONVERSION = YES; + GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; + GCC_WARN_UNDECLARED_SELECTOR = YES; + GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; + GCC_WARN_UNUSED_FUNCTION = YES; + GCC_WARN_UNUSED_VARIABLE = YES; + IPHONEOS_DEPLOYMENT_TARGET = 12.0; + MTL_ENABLE_DEBUG_INFO = NO; + SDKROOT = iphoneos; + SUPPORTED_PLATFORMS = iphoneos; + SWIFT_COMPILATION_MODE = wholemodule; + SWIFT_OPTIMIZATION_LEVEL = "-O"; + TARGETED_DEVICE_FAMILY = "1,2"; + VALIDATE_PRODUCT = YES; + }; + name = Release; + }; + 97C147061CF9000F007C117D /* Debug */ = { + isa = XCBuildConfiguration; + baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; + buildSettings = { + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CLANG_ENABLE_MODULES = YES; + CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; + ENABLE_BITCODE = NO; + INFOPLIST_FILE = Runner/Info.plist; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + ); + PRODUCT_BUNDLE_IDENTIFIER = com.example.flutterApplication1; + 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.example.flutterApplication1; + 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 */ + }; + rootObject = 97C146E61CF9000F007C117D /* Project object */; +} diff --git a/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata new file mode 100644 index 000000000..919434a62 --- /dev/null +++ b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 000000000..18d981003 --- /dev/null +++ b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 000000000..f9b0d7c5e --- /dev/null +++ b/flutter_application_1/ios/Runner.xcodeproj/project.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/flutter_application_1/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/flutter_application_1/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme new file mode 100644 index 000000000..8e3ca5dfe --- /dev/null +++ b/flutter_application_1/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -0,0 +1,98 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/ios/Runner.xcworkspace/contents.xcworkspacedata b/flutter_application_1/ios/Runner.xcworkspace/contents.xcworkspacedata new file mode 100644 index 000000000..1d526a16e --- /dev/null +++ b/flutter_application_1/ios/Runner.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 000000000..18d981003 --- /dev/null +++ b/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings b/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings new file mode 100644 index 000000000..f9b0d7c5e --- /dev/null +++ b/flutter_application_1/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings @@ -0,0 +1,8 @@ + + + + + PreviewsEnabled + + + diff --git a/flutter_application_1/ios/Runner/AppDelegate.swift b/flutter_application_1/ios/Runner/AppDelegate.swift new file mode 100644 index 000000000..626664468 --- /dev/null +++ b/flutter_application_1/ios/Runner/AppDelegate.swift @@ -0,0 +1,13 @@ +import Flutter +import UIKit + +@main +@objc class AppDelegate: FlutterAppDelegate { + override func application( + _ application: UIApplication, + didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? + ) -> Bool { + GeneratedPluginRegistrant.register(with: self) + return super.application(application, didFinishLaunchingWithOptions: launchOptions) + } +} diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 000000000..d36b1fab2 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png new file mode 100644 index 000000000..dc9ada472 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png new file mode 100644 index 000000000..7353c41ec Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png new file mode 100644 index 000000000..797d452e4 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png new file mode 100644 index 000000000..6ed2d933e Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png new file mode 100644 index 000000000..4cd7b0099 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png new file mode 100644 index 000000000..fe730945a Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png new file mode 100644 index 000000000..321773cd8 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png new file mode 100644 index 000000000..797d452e4 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png new file mode 100644 index 000000000..502f463a9 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png new file mode 100644 index 000000000..0ec303439 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png new file mode 100644 index 000000000..0ec303439 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png new file mode 100644 index 000000000..e9f5fea27 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png new file mode 100644 index 000000000..84ac32ae7 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png new file mode 100644 index 000000000..8953cba09 Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png new file mode 100644 index 000000000..0467bf12a Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json new file mode 100644 index 000000000..0bedcf2fd --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png new file mode 100644 index 000000000..9da19eaca Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png new file mode 100644 index 000000000..9da19eaca Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png new file mode 100644 index 000000000..9da19eaca Binary files /dev/null and b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png differ diff --git a/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md b/flutter_application_1/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md new file mode 100644 index 000000000..89c2725b7 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/ios/Runner/Base.lproj/LaunchScreen.storyboard b/flutter_application_1/ios/Runner/Base.lproj/LaunchScreen.storyboard new file mode 100644 index 000000000..f2e259c7c --- /dev/null +++ b/flutter_application_1/ios/Runner/Base.lproj/LaunchScreen.storyboard @@ -0,0 +1,37 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/ios/Runner/Base.lproj/Main.storyboard b/flutter_application_1/ios/Runner/Base.lproj/Main.storyboard new file mode 100644 index 000000000..f3c28516f --- /dev/null +++ b/flutter_application_1/ios/Runner/Base.lproj/Main.storyboard @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/ios/Runner/Info.plist b/flutter_application_1/ios/Runner/Info.plist new file mode 100644 index 000000000..6dc955a78 --- /dev/null +++ b/flutter_application_1/ios/Runner/Info.plist @@ -0,0 +1,49 @@ + + + + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleDisplayName + Flutter Application 1 + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + flutter_application_1 + CFBundlePackageType + APPL + CFBundleShortVersionString + $(FLUTTER_BUILD_NAME) + CFBundleSignature + ???? + CFBundleVersion + $(FLUTTER_BUILD_NUMBER) + LSRequiresIPhoneOS + + UILaunchStoryboardName + LaunchScreen + UIMainStoryboardFile + Main + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + CADisableMinimumFrameDurationOnPhone + + UIApplicationSupportsIndirectInputEvents + + + diff --git a/flutter_application_1/ios/Runner/Runner-Bridging-Header.h b/flutter_application_1/ios/Runner/Runner-Bridging-Header.h new file mode 100644 index 000000000..308a2a560 --- /dev/null +++ b/flutter_application_1/ios/Runner/Runner-Bridging-Header.h @@ -0,0 +1 @@ +#import "GeneratedPluginRegistrant.h" diff --git a/flutter_application_1/ios/RunnerTests/RunnerTests.swift b/flutter_application_1/ios/RunnerTests/RunnerTests.swift new file mode 100644 index 000000000..86a7c3b1b --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_bloc.dart b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_bloc.dart new file mode 100644 index 000000000..b3d729930 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_bloc.dart @@ -0,0 +1,125 @@ +import 'dart:math'; +import 'package:bloc/bloc.dart'; +import 'package:flutter_application_1/service/helper/global_service.dart'; +import 'package:meta/meta.dart'; + +part 'analytics_bloc_event.dart'; +part 'analytics_bloc_state.dart'; + +class AnalyticsBlocBloc extends Bloc { + AnalyticsBlocBloc() : super(AnalyticsBlocInitial()) { + on(analyticsLoadEvent); + on(analyticsMonthSelectedEvent); + on(_handleQueryEvent); + on(_handleChipEvent); + } + + Future analyticsLoadEvent( + AnalyticsLoadEvent event, Emitter emit) async { + emit(AnalyticsLoadingState()); + Map response = await GlobalService.globalService(endpoint: "getPrevYear", method: "GET"); + if (response["status"] == false) { + return ; + } + Map data = Map.from(response["message"]); + // Generate random expense data for months + List xList = List.from(data['xList']); + List yList = List.from(data['yList']); + String xTitle = data['xTitle']; + String yTitle = data['yTitle']; + + // Parse weekly data + Map> weeklyExpense = {}; + Map> weeklyIncome = {}; + Map> weeklyData = {}; // For chart + + Map weeklyDataMap = data['weeklyData']; + weeklyDataMap.forEach((month, monthData) { + int monthIndex = int.parse(month); + weeklyExpense[monthIndex] = List.from(monthData['expense']); + weeklyIncome[monthIndex] = List.from(monthData['income']); + weeklyData[monthIndex] = List.from(monthData['expense']); + }); + + emit(AnalyticsLoadedState( + xList: xList, + yList: yList, + xTitle: xTitle, + yTitle: yTitle, + weeklyData: weeklyData, + weeklyExpense: weeklyExpense, + weeklyIncome: weeklyIncome, + selectedMonth: data['selectedMonth'] ?? -1 + )); + } + + Future analyticsMonthSelectedEvent( + AnalyticsMonthSelectedEvent event, Emitter emit) async { + if (state is AnalyticsLoadedState) { + final currentState = state as AnalyticsLoadedState; + + emit(AnalyticsLoadedState( + xList: currentState.xList, + yList: currentState.yList, + xTitle: currentState.xTitle, + yTitle: currentState.yTitle, + weeklyData: currentState.weeklyData, + weeklyExpense: currentState.weeklyExpense, + weeklyIncome: currentState.weeklyIncome, + selectedMonth: event.monthIndex + )); + } + } + + Future _handleQueryEvent( + SendAnalyticsQueryEvent event, + Emitter emit + ) async { + emit(AnalyticsChatBotLoadingState()); + + try { + // Simulate API call + + // This would be replaced with actual analytics data fetching + Map requestBody = { + "botQuery": event.query + }; + Map response = await GlobalService.globalService(endpoint: "botQuery", method: "POST", requestBody: requestBody); + print("response form gemini : ${response["answer"]}"); + String message = (response["answer"] ?? _generateResponse(event.query)); + if (response["isGraph"] == true && false) { + List xList = List.from(response["xList"]); + List yList = List.from(response["yList"]); + String xTitle = response["xTitle"]; + String yTitle = response["yTitle"]; + emit(AnalyticsGraphShowingState(xList: xList, yList: yList, xTitle: xTitle, yTitle: yTitle)); + } + else { + emit(AnalyticsChatBotLoadedState(message: message)); + } + } catch (e) { + emit(AnalyticsErrorState(error: e.toString())); + } + } + + Future _handleChipEvent( + SelectAnalyticsChipEvent event, + Emitter emit + ) async { + emit(AnalyticsChipSelectedState(selectedChip: event.chipName)); + } + + String _generateResponse(String query) { + if (query.toLowerCase().contains('revenue')) { + return 'Based on current analytics, your revenue has increased by 15% compared to last month. The top performing channel is organic search with 43% contribution.'; + } else if (query.toLowerCase().contains('users') || query.toLowerCase().contains('visitors')) { + return 'You had 24,582 unique visitors this month, which is up 8.3% from last month. Average session duration is 3:24 minutes.'; + } else if (query.toLowerCase().contains('conversion')) { + return 'Your overall conversion rate is 3.2%, which is about industry average. Your best converting page is the Summer Sale landing page at 7.8%.'; + } else if (query.toLowerCase().contains('trend') || query.toLowerCase().contains('growth')) { + return 'I\'ve analyzed your growth trend. You\'re seeing consistent week-over-week growth of 2.1% in traffic and 1.7% in conversions over the past 3 months.'; + } else { + return 'I\'ve analyzed your analytics data. Overall performance looks positive with a 12% improvement in key metrics compared to the previous period. Would you like details on a specific metric?'; + } + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_event.dart b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_event.dart new file mode 100644 index 000000000..82548a0d4 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_event.dart @@ -0,0 +1,24 @@ +part of 'analytics_bloc_bloc.dart'; + +@immutable +abstract class AnalyticsBlocEvent {} + +class AnalyticsLoadEvent extends AnalyticsBlocEvent {} + +class AnalyticsMonthSelectedEvent extends AnalyticsBlocEvent { + final int monthIndex; + + AnalyticsMonthSelectedEvent(this.monthIndex); +} + +class SendAnalyticsQueryEvent extends AnalyticsBlocEvent { + final String query; + + SendAnalyticsQueryEvent(this.query); +} + +class SelectAnalyticsChipEvent extends AnalyticsBlocEvent { + final String chipName; + + SelectAnalyticsChipEvent(this.chipName); +} diff --git a/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_state.dart b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_state.dart new file mode 100644 index 000000000..c62c43684 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/AnalyticsBloc/analytics_bloc_state.dart @@ -0,0 +1,65 @@ +part of 'analytics_bloc_bloc.dart'; + +@immutable +abstract class AnalyticsBlocState {} + +class AnalyticsBlocInitial extends AnalyticsBlocState {} + +class AnalyticsLoadingState extends AnalyticsBlocState {} + +class AnalyticsLoadedState extends AnalyticsBlocState { + final List xList; + final List yList; + final String xTitle; + final String yTitle; + final Map> weeklyData; + final Map> weeklyExpense; + final Map> weeklyIncome; + final int selectedMonth; + + AnalyticsLoadedState({ + required this.xList, + required this.yList, + required this.xTitle, + required this.yTitle, + required this.weeklyData, + required this.weeklyExpense, + required this.weeklyIncome, + required this.selectedMonth + }); +} + +class AnalyticsInitialState extends AnalyticsBlocState {} + +class AnalyticsChatBotLoadingState extends AnalyticsBlocState {} + +class AnalyticsChatBotLoadedState extends AnalyticsBlocState { + final String message; + final dynamic data; + + AnalyticsChatBotLoadedState({required this.message, this.data}); +} + +class AnalyticsErrorState extends AnalyticsBlocState { + final String error; + + AnalyticsErrorState({required this.error}); +} + +class AnalyticsChipSelectedState extends AnalyticsBlocState { + final String selectedChip; + + AnalyticsChipSelectedState({required this.selectedChip}); +} + +class AnalyticsGraphShowingState extends AnalyticsBlocState { + final List xList; + final List yList; + final String xTitle; + final String yTitle; + AnalyticsGraphShowingState( + {required this.xList, + required this.yList, + required this.xTitle, + required this.yTitle}); +} diff --git a/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_bloc.dart b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_bloc.dart new file mode 100644 index 000000000..f60fe117f --- /dev/null +++ b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_bloc.dart @@ -0,0 +1,50 @@ +import 'dart:async'; + +import 'package:bloc/bloc.dart'; +import 'package:flutter_application_1/service/helper/global_service.dart'; +import 'package:flutter_application_1/service/helper/update_expense.dart'; +import 'package:meta/meta.dart'; + +part 'edit_expense_event.dart'; +part 'edit_expense_state.dart'; + +class EditExpenseBloc extends Bloc { + EditExpenseBloc() : super(EditExpenseInitial()) { + on (editExpenseSaveEvent); + on (addIncomeSaveEvent); + } + + Future editExpenseSaveEvent(EditExpenseSaveEvent event, Emitter emit) async { + emit(EditExpenseSavingState()); + Map expense = event.expense; + print("expense ${expense} bool : ${event.isEdit}"); + // do api call here + if (event.isEdit) { + bool isUpdated = await updateExpense(expense); + print("bool : $isUpdated"); + if (isUpdated) { + emit(EditExpenseSavedState()); + } else { + emit(EditExpenseErrorState()); + } + } else { + bool isCreated = await createExpense(expense); + if (isCreated) { + emit(EditExpenseSavedState()); + } else { + emit(EditExpenseErrorState()); + } + } + } + + Future addIncomeSaveEvent(AddIncomeSaveEvent event, Emitter emit) async { + emit(EditExpenseSavingState()); + Map incomeData = event.income; + bool isIncomeCreated = await addIncome(incomeData); + if (isIncomeCreated) { + emit(EditExpenseSavedState()); + } else { + emit(EditExpenseErrorState()); + } + } +} diff --git a/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_event.dart b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_event.dart new file mode 100644 index 000000000..42a8ce014 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_event.dart @@ -0,0 +1,14 @@ +part of 'edit_expense_bloc.dart'; + +abstract class EditExpenseEvent {} + +class EditExpenseSaveEvent extends EditExpenseEvent { + final Map expense; + bool isEdit; + EditExpenseSaveEvent({required this.expense, required this.isEdit}); +} + +class AddIncomeSaveEvent extends EditExpenseEvent { + final Map income; + AddIncomeSaveEvent({required this.income}); +} diff --git a/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_state.dart b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_state.dart new file mode 100644 index 000000000..27368b2ab --- /dev/null +++ b/flutter_application_1/lib/components/bloc/edit_expense_bloc/edit_expense_state.dart @@ -0,0 +1,13 @@ +part of 'edit_expense_bloc.dart'; + +abstract class EditExpenseState {} + +abstract class EditExpenseActionState extends EditExpenseState {} + +class EditExpenseInitial extends EditExpenseState {} + +class EditExpenseSavedState extends EditExpenseActionState {} + +class EditExpenseSavingState extends EditExpenseActionState {} + +class EditExpenseErrorState extends EditExpenseActionState {} \ No newline at end of file diff --git a/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_bloc.dart b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_bloc.dart new file mode 100644 index 000000000..82c2ac41b --- /dev/null +++ b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_bloc.dart @@ -0,0 +1,60 @@ +import 'dart:async'; +import 'dart:convert'; +import 'dart:math'; + +import 'package:bloc/bloc.dart'; +import 'package:file_picker/file_picker.dart'; +import 'package:flutter_application_1/components/models/expense_model.dart'; +import 'package:flutter_application_1/service/expense_service.dart'; +import 'package:image_picker/image_picker.dart'; + +part 'expenses_event.dart'; +part 'expenses_state.dart'; + +class ExpensesBloc extends Bloc { + ExpensesBloc() : super(ExpensesInitial()) { + on (expensesInitialEvent); + on(expenseImagePickedEvent); + on(expensePdfPickedEvent); + on(expenseChoiceShipSelectedEvent); + on(expensechatBotEvent); + } + + Future expensechatBotEvent(ExpensechatBotEvent event, Emitter emit) async { + emit(ExpenseChatBotLoadingState()); + String message = await getChatBotResponse(event.message); + emit(ExpenseChatBotLoadedState(message)); + } + + + Future expenseChoiceShipSelectedEvent(ExpenseChoiceShipSelectedEvent event, Emitter emit) async { + emit(ExpenseChoiceShipSelected(event.choiceShip)); + } + + + Future expensePdfPickedEvent(ExpensePdfPickedEvent event, Emitter emit)async{ + emit(ExpensePdfProcessingState()); + await uploadPdf(event.pdf); + dynamic val = await getExpenses(); + List> expenses = val["expenses"]; + List> recurringExpenses = val["recurringExpenses"]; + emit(ExpensePdfProcessedState(expenses)); + } + + Future expenseImagePickedEvent(ExpenseImagePickedEvent event, Emitter emit)async{ + emit(ExpensesImageProcessingState()); + await uploadBill(event.image); + dynamic val = await getExpenses(); + List> expenses = val["expenses"]; + List> recurringExpenses = val["recurringExpenses"]; + emit(ExpenseImageProcessedState(expenses)); + } + + Future expensesInitialEvent(ExpensesInitialEvent event, Emitter emit) async { + emit(ExpensesLoadingState()); + dynamic val = await getExpenses(); + List> expenses = val["expenses"]; + List> recurringExpenses = val["recurringExpenses"]; + emit(ExpensesLoadedState(expenses,recurringExpenses)); + } +} diff --git a/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_event.dart b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_event.dart new file mode 100644 index 000000000..fa701dcaf --- /dev/null +++ b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_event.dart @@ -0,0 +1,25 @@ +part of 'expenses_bloc.dart'; + +abstract class ExpensesEvent {} + +class ExpensesInitialEvent extends ExpensesEvent {} + +class ExpenseImagePickedEvent extends ExpensesEvent { + final XFile image; + ExpenseImagePickedEvent(this.image); +} + +class ExpensePdfPickedEvent extends ExpensesEvent { + final FilePickerResult pdf; + ExpensePdfPickedEvent(this.pdf); +} + +class ExpenseChoiceShipSelectedEvent extends ExpensesEvent { + final String choiceShip; + ExpenseChoiceShipSelectedEvent(this.choiceShip); +} + +class ExpensechatBotEvent extends ExpensesEvent { + final String message; + ExpensechatBotEvent(this.message); +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_state.dart b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_state.dart new file mode 100644 index 000000000..7e294454f --- /dev/null +++ b/flutter_application_1/lib/components/bloc/expenses_bloc/expenses_state.dart @@ -0,0 +1,44 @@ +part of 'expenses_bloc.dart'; + +abstract class ExpensesState {} + +abstract class ExpensesActionState extends ExpensesState {} + +class ExpensesInitial extends ExpensesState {} + +class ExpensesLoadingState extends ExpensesActionState {} + +class ExpensesLoadedState extends ExpensesActionState { + final List> expenses; + final List> recurringExpenses; + + ExpensesLoadedState(this.expenses,this.recurringExpenses); +} + +class ExpensesErrorState extends ExpensesActionState {} + +class ExpensesImageProcessingState extends ExpensesActionState {} + +class ExpenseImageProcessedState extends ExpensesActionState { + List> expenses; + ExpenseImageProcessedState(this.expenses); +} + +class ExpensePdfProcessingState extends ExpensesActionState {} + +class ExpensePdfProcessedState extends ExpensesActionState { + List> expenses; + ExpensePdfProcessedState(this.expenses); +} + +class ExpenseChoiceShipSelected extends ExpensesActionState { + final String choiceShip; + ExpenseChoiceShipSelected(this.choiceShip); +} + +class ExpenseChatBotLoadingState extends ExpensesActionState {} + +class ExpenseChatBotLoadedState extends ExpensesActionState { + final String message; + ExpenseChatBotLoadedState(this.message); +} diff --git a/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_bloc.dart b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_bloc.dart new file mode 100644 index 000000000..64e4d9574 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_bloc.dart @@ -0,0 +1,16 @@ +import 'dart:async'; + +import 'package:bloc/bloc.dart'; + +part 'main_screen_event.dart'; +part 'main_screen_state.dart'; + +class MainScreenBloc extends Bloc { + MainScreenBloc() : super(MainScreenInitial()) { + on (mainScreenBottomNavigationBarSwitching); + } + + FutureOr mainScreenBottomNavigationBarSwitching(MainScreenBottomNavigationBarSwitchingEvent event, Emitter emit) { + emit(MainScreenBottomNavigationBarSwitchingState(event.index)); + } +} diff --git a/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_event.dart b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_event.dart new file mode 100644 index 000000000..23d773c13 --- /dev/null +++ b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_event.dart @@ -0,0 +1,9 @@ +part of 'main_screen_bloc.dart'; + +abstract class MainScreenEvent {} + +class MainScreenBottomNavigationBarSwitchingEvent extends MainScreenEvent { + final int index; + + MainScreenBottomNavigationBarSwitchingEvent(this.index); +} diff --git a/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_state.dart b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_state.dart new file mode 100644 index 000000000..aefdb75ec --- /dev/null +++ b/flutter_application_1/lib/components/bloc/main_screen_bloc/main_screen_state.dart @@ -0,0 +1,13 @@ +part of 'main_screen_bloc.dart'; + +abstract class MainScreenState {} + +abstract class MainScreenActionState extends MainScreenState {} + +final class MainScreenInitial extends MainScreenState {} + +class MainScreenBottomNavigationBarSwitchingState extends MainScreenActionState { + final int index; + + MainScreenBottomNavigationBarSwitchingState(this.index); +} diff --git a/flutter_application_1/lib/components/models/expense_model.dart b/flutter_application_1/lib/components/models/expense_model.dart new file mode 100644 index 000000000..5d1dc504c --- /dev/null +++ b/flutter_application_1/lib/components/models/expense_model.dart @@ -0,0 +1,55 @@ +import 'dart:convert'; + +class ExpenseModel { + String expenseTitle; + int amountSpent; + String category; + DateTime dateTime; + String paymentMethod; + String merchantName; + + ExpenseModel({ + required this.expenseTitle, + required this.amountSpent, + required this.category, + required this.dateTime, + required this.paymentMethod, + required this.merchantName, + }); + + // Convert JSON to Expense object + factory ExpenseModel.fromJson(Map json) { + return ExpenseModel( + expenseTitle: json['expenseTitle'], + amountSpent: json['amountSpent'], + category: json['category'], + dateTime: DateTime.parse(json['dateTime']), + paymentMethod: json['paymentMethod'], + merchantName: json['merchantName'], + ); + } + + // Convert Expense object to JSON + Map toJson() { + return { + 'expenseTitle': expenseTitle, + 'amountSpent': amountSpent, + 'category': category, + 'dateTime': dateTime.toIso8601String(), + 'paymentMethod': paymentMethod, + 'merchantName': merchantName, + }; + } + + // Convert list of JSON to List + static List fromJsonList(String jsonString) { + final List jsonList = json.decode(jsonString); + return jsonList.map((json) => ExpenseModel.fromJson(json)).toList(); + } + + // Convert List to JSON string + static String toJsonList(List expenses) { + final List> jsonList = expenses.map((e) => e.toJson()).toList(); + return json.encode(jsonList); + } +} diff --git a/flutter_application_1/lib/components/models/models b/flutter_application_1/lib/components/models/models new file mode 100644 index 000000000..e69de29bb diff --git a/flutter_application_1/lib/components/models/user_model.dart b/flutter_application_1/lib/components/models/user_model.dart new file mode 100644 index 000000000..ef2169ebe --- /dev/null +++ b/flutter_application_1/lib/components/models/user_model.dart @@ -0,0 +1,6 @@ +class UserModel { + static String name = ""; + static String email = ""; + static String phone = ""; + static String photoUrl = ""; +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/utils/helper/styles.dart b/flutter_application_1/lib/components/utils/helper/styles.dart new file mode 100644 index 000000000..1d825c5f6 --- /dev/null +++ b/flutter_application_1/lib/components/utils/helper/styles.dart @@ -0,0 +1,13 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/theme/colors.dart'; + +class AppStyles { + static TextStyle setAppStyle(Color? color, double? fontSize, FontWeight? fontWeight, String? fontFamily) { + return TextStyle( + color: color ?? black, + fontSize: fontSize ?? 16, + fontWeight: fontWeight ?? FontWeight.normal, + fontFamily: fontFamily ?? "medium" + ); + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/utils/utilities b/flutter_application_1/lib/components/utils/utilities new file mode 100644 index 000000000..e69de29bb diff --git a/flutter_application_1/lib/components/view/Auth/login_view.dart b/flutter_application_1/lib/components/view/Auth/login_view.dart new file mode 100644 index 000000000..358904d2b --- /dev/null +++ b/flutter_application_1/lib/components/view/Auth/login_view.dart @@ -0,0 +1,251 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/view/main_screeen.dart'; +import 'package:flutter_application_1/data/appvalues.dart'; +import 'package:flutter_application_1/service/auth/auth.dart'; +import 'package:google_sign_in/google_sign_in.dart'; + +class LoginPage extends StatelessWidget { + const LoginPage({super.key}); + + @override + Widget build(BuildContext context) { + return Scaffold( + backgroundColor: Colors.white, + body: SafeArea( + child: SingleChildScrollView( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 24.0), + child: Column( + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + const SizedBox(height: 60), + + // App logo + Container( + height: 100, + width: 100, + decoration: BoxDecoration( + color: Colors.blue, + borderRadius: BorderRadius.circular(20), + ), + child: const Icon( + Icons.account_balance_wallet, + color: Colors.white, + size: 60, + ), + ), + + const SizedBox(height: 30), + + // App title + const Text( + "FinTrack", + style: TextStyle( + fontSize: 32, + fontWeight: FontWeight.bold, + color: Colors.blue, + ), + ), + + const SizedBox(height: 10), + + // App tagline + const Text( + "Your personal finance companion", + style: TextStyle( + fontSize: 16, + color: Colors.grey, + ), + textAlign: TextAlign.center, + ), + + const SizedBox(height: 50), + + // Welcome message + Container( + padding: const EdgeInsets.all(16), + decoration: BoxDecoration( + color: Colors.blue.withOpacity(0.1), + borderRadius: BorderRadius.circular(16), + ), + child: Column( + children: [ + const Text( + "Welcome!", + style: TextStyle( + fontSize: 24, + fontWeight: FontWeight.bold, + color: Colors.blue, + ), + ), + const SizedBox(height: 16), + const Text( + "Sign in with your Google account to access intelligent financial insights, track your expenses, and manage your money effortlessly.", + style: TextStyle(fontSize: 16, color: Colors.black87), + textAlign: TextAlign.center, + ), + ], + ), + ), + + const SizedBox(height: 40), + + // Features section + Row( + children: [ + Expanded( + child: _buildFeatureItem( + Icons.insights, + "Personalized financial insights" + ), + ), + ], + ), + const SizedBox(height: 16), + Row( + children: [ + Expanded( + child: _buildFeatureItem( + Icons.track_changes, + "Smart budgeting & expense tracking" + ), + ), + ], + ), + const SizedBox(height: 16), + Row( + children: [ + Expanded( + child: _buildFeatureItem( + Icons.auto_awesome, + "AI-powered financial recommendations" + ), + ), + ], + ), + + const SizedBox(height: 60), + + // Google Sign in button + ElevatedButton( + onPressed: () { + // Implement Google Sign-in functionality + _handleGoogleSignIn(context); + }, + style: ElevatedButton.styleFrom( + backgroundColor: Colors.white, + foregroundColor: Colors.black87, + elevation: 2, + padding: const EdgeInsets.symmetric(vertical: 16), + minimumSize: const Size(double.infinity, 50), + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(8), + side: BorderSide(color: Colors.grey.shade300), + ), + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Image.network( + 'https://th.bing.com/th/id/OIP.JflGW8e1fT4_ttSuFTQXJwHaHj?rs=1&pid=ImgDetMain', + height: 24, + width: 24, + ), + const SizedBox(width: 12), + const Text( + "Sign in with Google", + style: TextStyle( + fontSize: 16, + fontWeight: FontWeight.w500, + ), + ), + ], + ), + ), + + const SizedBox(height: 16), + ], + ), + ), + ), + ), + ); + } + + Widget _buildFeatureItem(IconData icon, String text) { + return Row( + children: [ + Container( + padding: const EdgeInsets.all(8), + decoration: BoxDecoration( + color: Colors.blue.withOpacity(0.1), + borderRadius: BorderRadius.circular(8), + ), + child: Icon(icon, color: Colors.blue, size: 20), + ), + const SizedBox(width: 16), + Expanded( + child: Text( + text, + style: const TextStyle( + fontSize: 14, + fontWeight: FontWeight.w500, + ), + ), + ), + ], + ); + } + + Future _handleGoogleSignIn(BuildContext context) async { + GoogleSignIn _googleSignIn = GoogleSignIn(); + try { + if (await _googleSignIn.isSignedIn()) { + await _googleSignIn.signOut(); + await FirebaseAuth.instance.signOut(); + + try { + await _googleSignIn.disconnect(); + } catch (e, stackTrace) { + print("Error during disconnect: $e"); + } + } + final GoogleSignInAccount? googleUser = await GoogleSignIn().signIn(); + final GoogleSignInAuthentication? googleAuth = await googleUser?.authentication; + final credential = GoogleAuthProvider.credential( + accessToken: googleAuth?.accessToken, + idToken: googleAuth?.idToken, + ); + await FirebaseAuth.instance.signInWithCredential(credential); + String? jwtToken = await FirebaseAuth.instance.currentUser?.getIdToken(); + print("JWT token : $jwtToken"); + if (jwtToken == null) { + throw Exception('Failed to get JWT token'); + } + AppValues.jwtToken = jwtToken; + bool isAuthenticated = await authService(); + if (isAuthenticated) { + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => const MainScreen()), + (Route route) => false, + ); + } + else { + throw Exception("Unauthorized"); + } + } catch (e) { + print("Error in google sign in $e"); + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('Sign-in failed. Please try again.')), + ); + } + + if (AppValues.jwtToken == "") { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('Sign-in failed. Please try again.')), + ); + } + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/view/analytics/analytics_view.dart b/flutter_application_1/lib/components/view/analytics/analytics_view.dart new file mode 100644 index 000000000..a0e2e2d41 --- /dev/null +++ b/flutter_application_1/lib/components/view/analytics/analytics_view.dart @@ -0,0 +1,773 @@ +import 'package:flutter/material.dart'; +import 'package:fl_chart/fl_chart.dart'; +import 'package:flutter_application_1/components/bloc/AnalyticsBloc/analytics_bloc_bloc.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; +import 'package:flutter_spinkit/flutter_spinkit.dart'; + +import '../../../theme/colors.dart'; +import '../../utils/helper/styles.dart'; + +class AnalyticsView extends StatefulWidget { + const AnalyticsView({super.key}); + + @override + State createState() => _AnalyticsViewState(); +} + +class _AnalyticsViewState extends State + with TickerProviderStateMixin { + List xAxis = []; + List month = [ + "January", + "February", + "March", + "April", + "May", + "June", + "July", + "August", + "September", + "October", + "November", + "December" + ]; + List barTouchData = [ + "January", + "February", + "March", + "April", + "May", + "June", + "July", + "August", + "September", + "October", + "November", + "December" + ]; + List yAxis = []; + String xTitle = ""; + String yTitle = ""; + late TabController tabController; + final analyticsBlocBloc = AnalyticsBlocBloc(); + int selectedMonth = -1; + + @override + void initState() { + tabController = TabController(length: 2, vsync: this); + analyticsBlocBloc.add(AnalyticsLoadEvent()); + barTouchData = month; + super.initState(); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar( + title: Text( + 'Analytics', + style: AppStyles.setAppStyle(black, 20, FontWeight.bold, 'black'), + ), + forceMaterialTransparency: true, + leading: Icon(Icons.analytics_outlined), + ), + floatingActionButton: ElevatedButton( + onPressed: () { + showModalBottomSheet( + context: context, + useSafeArea: true, + isScrollControlled: true, + builder: (BuildContext context) { + return Padding( + padding: EdgeInsets.only( + bottom: MediaQuery.of(context).viewInsets.bottom, // Moves content up + ), + child: AnalyticsChatBottomSheet(analyticsChatBloc: analyticsBlocBloc), + ); + }); + }, + style: ElevatedButton.styleFrom( + backgroundColor: white, + shape: + CircleBorder(side: BorderSide(color: border, width: 0.5)), + minimumSize: const Size(50, 50), + ), + child: Icon( + Icons.reddit, + size: 35, + color: Colors.orange, + ), + ), + body: Padding( + padding: const EdgeInsets.all(8.0), + child: BlocBuilder( + bloc: analyticsBlocBloc, + builder: (context, state) { + if (state is AnalyticsLoadingState) { + return Center(child: SpinKitCircle(color: primary,)); + } + if (state is AnalyticsLoadedState) { + xAxis = state.xList; + yAxis = state.yList; + xTitle = state.xTitle; + yTitle = state.yTitle; + selectedMonth = state.selectedMonth; + + return SingleChildScrollView( + scrollDirection: Axis.vertical, + child: Column( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + SingleChildScrollView( + scrollDirection: Axis.horizontal, + child: Row( + children: [ + for (int i = 0; i < 12; i++) + monthlyExpenseContainer( + value: state.yList[i], + month: month[i], + expense: state.yList[i], + income: state.yList[i] + 500, + index: i, + isSelected: selectedMonth == i, + onTap: () { + analyticsBlocBloc.add(AnalyticsMonthSelectedEvent(i)); + } + ) + ], + ), + ), + const SizedBox(height: 20), + + // Show weekly data if a month is selected, otherwise show monthly data + if (selectedMonth >= 0) ...[ + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text('${month[selectedMonth]} Weekly Analytics', + style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold)), + Expanded( + child: TextButton( + onPressed: () { + analyticsBlocBloc.add(AnalyticsMonthSelectedEvent(-1)); + }, + child: const Text('Back to Monthly View',style: TextStyle(overflow: TextOverflow.ellipsis),), + ), + ), + ], + ), + const SizedBox(height: 20), + weeklyExpenseGraph( + context, + List.generate(4, (index) => index), + state.weeklyData[selectedMonth] ?? [], + "Weeks", + "Weekly Amount", + selectedMonth + ), + const SizedBox(height: 20), + SingleChildScrollView( + scrollDirection: Axis.horizontal, + child: Row( + children: [ + for (int i = 0; i < 4; i++) + weeklyDetailContainer( + week: "Week ${i + 1}", + expense: state.weeklyExpense[selectedMonth]?[i] ?? 0, + income: state.weeklyIncome[selectedMonth]?[i] ?? 0, + ) + ], + ), + ), + ] else ...[ + const Text('Expense Graphs', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)), + const SizedBox(height: 20), + expenseGraph(context, xAxis, yAxis, xTitle, yTitle, month), + ], + + // const SizedBox(height: 20), + // TabBar( + // controller: tabController, + // dividerColor: Colors.transparent, + // tabs: const [ + // Tab(child: Text("Expenses")), + // Tab(child: Text("Stocks")) + // ] + // ), + ], + ), + ); + } + return const Center(child: Text("No data available")); + }, + ), + ), + ); + } +} + +Widget monthlyExpenseContainer({ + required int value, + required String month, + required int expense, + required int income, + required int index, + required bool isSelected, + required VoidCallback onTap +}) { + return GestureDetector( + onTap: onTap, + child: Card( + color: isSelected ? Colors.blue[100] : Colors.grey[200], + elevation: isSelected ? 4 : 1, + child: Padding( + padding: const EdgeInsets.all(18.0), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(month, style: const TextStyle(fontWeight: FontWeight.bold)), + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text('Expense'), + Text('₹$expense'), + ], + ), + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text('Income'), + Text('₹$income'), + ], + ), + if (isSelected) + const Padding( + padding: EdgeInsets.only(top: 8.0), + child: Text('Tap to view weekly', style: TextStyle(fontSize: 12, color: Colors.blue)), + ), + ], + ), + ), + ), + ); +} + +Widget weeklyDetailContainer({ + required String week, + required int expense, + required int income, +}) { + return Card( + color: Colors.grey[200], + child: Padding( + padding: const EdgeInsets.all(18.0), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(week, style: const TextStyle(fontWeight: FontWeight.bold)), + const SizedBox(height: 8), + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text('Expense'), + Text('₹$expense'), + ], + ), + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text('Income'), + Text('₹$income'), + ], + ), + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text('Balance'), + Text('₹${income - expense}', + style: TextStyle( + fontWeight: FontWeight.bold, + color: income - expense >= 0 ? Colors.green : Colors.red + )), + ], + ), + ], + ), + ), + ); +} + +Widget expenseGraph(BuildContext context, List xList, List yList, + String xTitle, String yTitle, List month) { + return SizedBox( + height: MediaQuery.of(context).size.width * 0.9, + child: BarChart(BarChartData( + alignment: BarChartAlignment.spaceEvenly, + borderData: FlBorderData(show: false), + gridData: FlGridData(show: false), + groupsSpace: 12, + barTouchData: BarTouchData( + touchTooltipData: BarTouchTooltipData( + getTooltipItem: (group, groupIndex, rod, rodIndex) { + return BarTooltipItem( + month[group.x.toInt()], + const TextStyle(color: Colors.white)); + }, + ), + ), + barGroups: [ + for (int i = 0; i < xList.length; i++) + BarChartGroupData( + x: i, + barRods: [ + BarChartRodData( + toY: yList[i].toDouble(), + fromY: 0, + width: 25, + color: Colors.blue, + ), + ], + barsSpace: 4, + ), + ], + titlesData: FlTitlesData( + leftTitles: AxisTitles( + axisNameWidget: Text(yTitle), + sideTitles: const SideTitles(showTitles: false, reservedSize: 30), + ), + bottomTitles: AxisTitles( + axisNameWidget: Text(xTitle), + sideTitles: SideTitles( + showTitles: true, + getTitlesWidget: (value, meta) { + if (value.toInt() >= 0 && value.toInt() < month.length) { + return Padding( + padding: const EdgeInsets.only(top: 8.0), + child: Text(month[value.toInt()].substring(0, 3)), + ); + } + return const Text(''); + }, + reservedSize: 30 + ), + ), + rightTitles: const AxisTitles( + sideTitles: SideTitles(showTitles: false), + ), + topTitles: const AxisTitles( + sideTitles: SideTitles(showTitles: false), + )))), + ); +} + +Widget weeklyExpenseGraph(BuildContext context, List xList, List yList, + String xTitle, String yTitle, int monthIndex) { + print("ylist data ${yList[1].toDouble()}"); + return SizedBox( + height: MediaQuery.of(context).size.width * 0.7, + child: BarChart(BarChartData( + alignment: BarChartAlignment.spaceEvenly, + borderData: FlBorderData(show: false), + gridData: const FlGridData(show: false), + groupsSpace: 12, + barTouchData: BarTouchData( + touchTooltipData: BarTouchTooltipData( + getTooltipItem: (group, groupIndex, rod, rodIndex) { + return BarTooltipItem( + "Week ${group.x.toInt() + 1}", + const TextStyle(color: Colors.white)); + }, + ), + ), + barGroups: [ + for (int i = 0; i < xList.length; i++) + BarChartGroupData( + x: i, + barRods: [ + BarChartRodData( + toY: yList[i].toDouble(), + fromY: 0, + width: 30, + color: Colors.green, + ), + ], + barsSpace: 4, + ), + ], + titlesData: FlTitlesData( + leftTitles: AxisTitles( + axisNameWidget: Text(yTitle), + sideTitles: const SideTitles(showTitles: false, reservedSize: 30), + ), + bottomTitles: AxisTitles( + axisNameWidget: Text(xTitle), + sideTitles: SideTitles( + showTitles: true, + getTitlesWidget: (value, meta) { + return Padding( + padding: const EdgeInsets.only(top: 8.0), + child: Text("Week ${value.toInt() + 1}"), + ); + }, + reservedSize: 30 + ), + ), + rightTitles: const AxisTitles( + sideTitles: SideTitles(showTitles: false), + ), + topTitles: const AxisTitles( + sideTitles: SideTitles(showTitles: false), + )))), + ); +} +// The Bottom Sheet Widget +class AnalyticsChatBottomSheet extends StatefulWidget { + final AnalyticsBlocBloc analyticsChatBloc; + + const AnalyticsChatBottomSheet({ + required this.analyticsChatBloc, + Key? key, + }) : super(key: key); + + @override + State createState() => _AnalyticsChatBottomSheetState(); +} + +class _AnalyticsChatBottomSheetState extends State { + final TextEditingController _controller = TextEditingController(); + final ScrollController _scrollController = ScrollController(); + String? selectedChip; + List messages = []; + + @override + void dispose() { + _controller.dispose(); + _scrollController.dispose(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + // Common analytics questions + List analyticsChips = [ + "Revenue Analysis", + "Conversion Rates", + "Growth Trends", + "Custom Query" + ]; + + return Container( + decoration: BoxDecoration( + color: Theme.of(context).scaffoldBackgroundColor, + borderRadius: const BorderRadius.vertical(top: Radius.circular(20)), + boxShadow: [ + BoxShadow( + color: Colors.black.withOpacity(0.1), + blurRadius: 10, + spreadRadius: 1, + ) + ], + ), + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), + child: BlocConsumer( + bloc: widget.analyticsChatBloc, + listener: (context, state) { + if (state is AnalyticsChatBotLoadedState) { + setState(() { + messages.add(ChatMessage( + content: state.message, + isFromBot: true, + timestamp: DateTime.now(), + )); + }); + + // Auto-scroll to bottom + WidgetsBinding.instance.addPostFrameCallback((_) { + if (_scrollController.hasClients) { + _scrollController.animateTo( + _scrollController.position.maxScrollExtent, + duration: const Duration(milliseconds: 300), + curve: Curves.easeOut, + ); + } + }); + + // Clear text field after response + _controller.clear(); + } else if (state is AnalyticsChipSelectedState) { + setState(() { + selectedChip = state.selectedChip; + + if (selectedChip != "Custom Query") { + _controller.text = selectedChip!; + } else { + _controller.clear(); + } + }); + } + }, + builder: (context, state) { + return Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + // Handle bar at the top + Center( + child: Container( + margin: const EdgeInsets.only(top: 8, bottom: 16), + width: 40, + height: 4, + decoration: BoxDecoration( + color: Colors.grey.shade300, + borderRadius: BorderRadius.circular(2), + ), + ), + ), + + // Header with analytics icon + Row( + children: [ + Icon( + Icons.analytics, + color: Theme.of(context).primaryColor, + size: 28, + ), + const SizedBox(width: 10), + Text( + "Analytics Assistant", + style: TextStyle( + fontFamily: "medium", + fontSize: 22, + fontWeight: FontWeight.bold, + ), + ), + ], + ), + + const SizedBox(height: 16), + + // Analytics topic chips + SingleChildScrollView( + scrollDirection: Axis.horizontal, + child: Row( + children: analyticsChips.map((label) { + return Padding( + padding: const EdgeInsets.only(right: 8.0), + child: ChoiceChip( + selected: selectedChip == label, + label: Text(label), + labelStyle: TextStyle( + fontWeight: selectedChip == label ? FontWeight.bold : FontWeight.normal, + ), + backgroundColor: Colors.grey.shade200, + selectedColor: Theme.of(context).primaryColor.withOpacity(0.2), + onSelected: (value) { + widget.analyticsChatBloc.add(SelectAnalyticsChipEvent(label)); + }, + ), + ); + }).toList(), + ), + ), + + const SizedBox(height: 16), + + // Chat messages area + Flexible( + child: messages.isEmpty + ? Center( + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Icon( + Icons.chat_bubble_outline, + size: 48, + color: Colors.grey.shade400, + ), + const SizedBox(height: 16), + Text( + "Ask me about your analytics data", + style: TextStyle( + color: Colors.grey.shade600, + fontSize: 16, + ), + ), + ], + ), + ) + : ListView.builder( + controller: _scrollController, + itemCount: messages.length, + itemBuilder: (context, index) { + final message = messages[index]; + return ChatBubble( + message: message.content, + isFromBot: message.isFromBot, + ); + }, + ), + ), + + // Loading indicator + if (state is AnalyticsChatBotLoadingState) + Center( + child: Padding( + padding: const EdgeInsets.symmetric(vertical: 16.0), + child: SpinKitPulse( + color: Theme.of(context).primaryColor, + size: 40.0, + ), + ), + ), + + const SizedBox(height: 16), + + // Input field with send button + Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(24), + border: Border.all( + color: Theme.of(context).primaryColor.withOpacity(0.5), + width: 1.5, + ), + color: Colors.grey.shade50, + ), + child: Row( + children: [ + Expanded( + child: TextField( + controller: _controller, + decoration: InputDecoration( + hintText: "Ask about analytics data...", + hintStyle: TextStyle( + fontFamily: "medium", + color: Colors.grey.shade500, + ), + contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + border: InputBorder.none, + ), + maxLines: null, + textCapitalization: TextCapitalization.sentences, + ), + ), + Material( + color: Colors.transparent, + child: InkWell( + borderRadius: BorderRadius.circular(24), + onTap: () { + if (_controller.text.isNotEmpty) { + final query = _controller.text; + + // Add user message to chat + setState(() { + messages.add(ChatMessage( + content: query, + isFromBot: false, + timestamp: DateTime.now(), + )); + }); + + // Send query to bloc + widget.analyticsChatBloc.add(SendAnalyticsQueryEvent(query)); + } + }, + child: Padding( + padding: const EdgeInsets.all(12.0), + child: Icon( + Icons.send_rounded, + color: Theme.of(context).primaryColor, + ), + ), + ), + ), + ], + ), + ), + + const SizedBox(height: 16), + ], + ); + }, + ), + ), + ); + } +} + +// Chat Message Model +class ChatMessage { + final String content; + final bool isFromBot; + final DateTime timestamp; + + ChatMessage({ + required this.content, + required this.isFromBot, + required this.timestamp, + }); +} + +// Custom chat bubble widget +class ChatBubble extends StatelessWidget { + final String message; + final bool isFromBot; + + + const ChatBubble({ + required this.message, + required this.isFromBot, + Key? key, + }) : super(key: key); + + @override + Widget build(BuildContext context) { + return Container( + margin: const EdgeInsets.symmetric(vertical: 8), + child: Row( + mainAxisAlignment: isFromBot ? MainAxisAlignment.start : MainAxisAlignment.end, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + if (isFromBot) + Container( + margin: const EdgeInsets.only(right: 8), + child: CircleAvatar( + backgroundColor: Theme.of(context).primaryColor.withOpacity(0.2), + radius: 16, + child: Icon( + Icons.analytics, + size: 20, + color: Theme.of(context).primaryColor, + ), + ), + ), + Flexible( + child: Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + decoration: BoxDecoration( + color: isFromBot + ? Colors.grey.shade100 + : Theme.of(context).primaryColor.withOpacity(0.1), + borderRadius: BorderRadius.circular(16), + border: Border.all( + color: isFromBot + ? Colors.grey.shade300 + : Theme.of(context).primaryColor.withOpacity(0.3), + width: 1, + ), + ), + child: Text( + message, + style: TextStyle( + fontSize: 15, + color: Colors.black87, + ), + ), + ), + ), + ], + ), + ); + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/view/expenses/add_income.dart b/flutter_application_1/lib/components/view/expenses/add_income.dart new file mode 100644 index 000000000..8e3dfa8a6 --- /dev/null +++ b/flutter_application_1/lib/components/view/expenses/add_income.dart @@ -0,0 +1,210 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/bloc/edit_expense_bloc/edit_expense_bloc.dart'; +import 'package:flutter_application_1/components/utils/helper/styles.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; +import 'package:flutter_spinkit/flutter_spinkit.dart'; +import 'package:intl/intl.dart'; + +import '../../../theme/colors.dart'; +import '../../bloc/expenses_bloc/expenses_bloc.dart'; + +class AddIncome extends StatefulWidget { + const AddIncome({super.key}); + + @override + State createState() => _AddIncomeState(); +} + +class _AddIncomeState extends State { + + TextEditingController amountController = TextEditingController(); + TextEditingController incomeTitleController = TextEditingController(); + TextEditingController dateController = TextEditingController(); + EditExpenseBloc expensesBloc = EditExpenseBloc(); + bool isSaving = false; + + @override + Widget build(BuildContext context) { + return BlocConsumer( + bloc: expensesBloc, + listener: (context, state) { + if (state is EditExpenseSavedState) { + isSaving = false; + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text("Income added successfully"), + backgroundColor: primary, + ), + ); + Navigator.pop(context); + } + if (state is EditExpenseErrorState) { + isSaving = false; + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text("Error in adding income"), + backgroundColor: primary, + ), + ); + } + if (state is EditExpenseSavingState) { + isSaving = true; + } + }, + builder: (context, state) { + return Scaffold( + backgroundColor: bgColor, + appBar: AppBar( + backgroundColor: bgColor, + title: Text( + "Add Income", + style: AppStyles.setAppStyle(black, 20, FontWeight.bold, "black"), + ), + ), + body: bodyPartOfIncome(), + ); + }, + ); + } + + Widget bodyPartOfIncome() { + return Padding( + padding: const EdgeInsets.all(20), + child: Column( + children: [ + textFieldForCreateScreen( + "Income Title (source)", incomeTitleController), + textFieldForCreateScreen("Salary amount", amountController), + dateWidget(), + isSaving ? SpinKitCircle(color: primary,) : ElevatedButton( + onPressed: () { + Map incomeData = { + "expense_title": incomeTitleController.text, + "merchant_name" : "", + "amount_spent": amountController.text, + "date": dateController.text, + "category": "income", + "payment_method": "salary", + }; + expensesBloc.add(AddIncomeSaveEvent(income: incomeData)); + }, + style: ElevatedButton.styleFrom( + backgroundColor: primary, + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(10)), + ), + child: Text("Add income", style: AppStyles.setAppStyle( + white, 16, FontWeight.normal, "medium"),), + ), + ], + ), + ); + } + + + Widget dateWidget() { + return Padding( + padding: const EdgeInsets.only(bottom: 16.0), + child: Container( + padding: const EdgeInsets.symmetric( + horizontal: 12.0, vertical: 16.0), + decoration: BoxDecoration( + color: filterSortBackground, + borderRadius: BorderRadius.circular(8), + border: Border.all(color: border)), + child: InkWell( + enableFeedback: true, + onTap: () async { + DateTime? pickedDate = await showDatePicker( + context: context, + initialDate: + (dateController.text.isNotEmpty) + ? DateTime + .parse(dateController.text) + : DateTime.now(), + firstDate: DateTime(2000), lastDate: DateTime(2050), + ); + if (pickedDate != null) { + setState(() { + dateController.text = + DateFormat('dd-MM-yyyy') + .format(pickedDate); + }); + } + }, + child: Row( + mainAxisAlignment: + MainAxisAlignment.spaceBetween, + children: [ + Text( + dateController.text.isNotEmpty + ? dateController.text + : 'dd-mm-yyyy', + style: const TextStyle( + fontFamily: 'medium', + color: Colors.grey, + ), + ), + Icon(Icons.calendar_month) + ], + ), + ), + ), + ); + } + + Widget textFieldForCreateScreen(String labelText, + TextEditingController textEditingController) { + return Padding( + padding: const EdgeInsets.only(bottom: 10), + child: TextFormField( + keyboardType: (labelText == "Salary amount") + ? TextInputType.number + : TextInputType.text, + controller: textEditingController, + style: const TextStyle(fontFamily: "regular"), + decoration: InputDecoration( + border: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + enabledBorder: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + focusedBorder: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + labelText: labelText, + labelStyle: TextStyle( + fontFamily: "regular", + color: grey, + fontSize: 16, + fontWeight: FontWeight.w400, + fontStyle: FontStyle.normal, + ), + floatingLabelStyle: TextStyle( + fontFamily: "regular", + color: primary, + fontSize: 16, + fontWeight: FontWeight.w400, + fontStyle: FontStyle.normal, + ), + fillColor: white, + filled: true), + maxLines: 10, + minLines: 1, + ), + ); + } +} diff --git a/flutter_application_1/lib/components/view/expenses/edit_expenses.dart b/flutter_application_1/lib/components/view/expenses/edit_expenses.dart new file mode 100644 index 000000000..9742f9608 --- /dev/null +++ b/flutter_application_1/lib/components/view/expenses/edit_expenses.dart @@ -0,0 +1,334 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/utils/helper/styles.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; +import 'package:flutter_spinkit/flutter_spinkit.dart'; +import 'package:intl/intl.dart'; + +import '../../../theme/colors.dart'; +import '../../bloc/edit_expense_bloc/edit_expense_bloc.dart'; + +class EditExpenseScreen extends StatefulWidget { + final Map expenseData; + final bool isEdit; + + const EditExpenseScreen({super.key, required this.expenseData, required this.isEdit}); + + @override + State createState() => _EditExpenseScreenState(); +} + +class _EditExpenseScreenState extends State { + final _formKey = GlobalKey(); + + TextEditingController expenseTitleController = TextEditingController(); + TextEditingController merchantNameController = TextEditingController(); + TextEditingController totalAmountController = TextEditingController(); + TextEditingController paymentMethodController = TextEditingController(); + TextEditingController dateController = TextEditingController(); + TextEditingController categoryController = TextEditingController(); + + List> items = []; + String selectedCategory = "Shopping"; + String selectedPaymentMethod = "Cash"; + DateTime? selectedDate; + EditExpenseBloc editExpenseBloc = EditExpenseBloc(); + bool isSaving = false; + + @override + void initState() { + super.initState(); + print("AMount spent : ${widget.expenseData["date_time"]}"); + if (widget.isEdit) { + expenseTitleController = + TextEditingController(text: widget.expenseData["expense_title"] ?? ""); + categoryController = TextEditingController(text: widget.expenseData["category"] ?? ""); + paymentMethodController = TextEditingController(text: widget.expenseData["payment_method"] ?? ""); + merchantNameController = TextEditingController(text: widget.expenseData["merchant_name"] ?? ""); + totalAmountController = TextEditingController(text: widget.expenseData["amount_spent"].toString() ?? ""); + dateController = TextEditingController(text: widget.expenseData["date_time"] ?? ""); + selectedDate = widget.expenseData["date_time"] != null + ? DateTime.tryParse(widget.expenseData["date_time"]) + : null; + print("tags from josn : ${widget.expenseData["tags"]}"); + items = List>.from(widget.expenseData["items"] ?? []); + } + } + + @override + void dispose() { + expenseTitleController.dispose(); + merchantNameController.dispose(); + totalAmountController.dispose(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar(title: Text(widget.isEdit ? "Edit Expense" : "Create Expense",style: AppStyles.setAppStyle(black, 20, FontWeight.bold, "black"),)), + body: bodyPartOfEditExpense(), + ); + } + + Widget bodyPartOfEditExpense() { + return BlocConsumer( + bloc: editExpenseBloc, + listener: (context, state) { + if (state is EditExpenseSavingState) { + isSaving = true; + } + if (state is EditExpenseSavedState) { + isSaving = false; + Navigator.pop(context); + } + if (state is EditExpenseErrorState) { + isSaving = false; + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text("Error while saving expense"), + ), + ); + } + }, + builder: (context, state) { + return Padding( + padding: const EdgeInsets.all(20), + child: ListView( + children: [ + textFieldForCreateScreen("Expense Title", expenseTitleController), + textFieldForCreateScreen("Merchant Name", merchantNameController), + textFieldForCreateScreen("Payment Method", paymentMethodController), + dateWidget(), + textFieldForCreateScreen("Category", categoryController), + const SizedBox(height: 20), + const Text("Items", style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)), + Expanded( + child: Column( + children: [ + ListView.builder( + shrinkWrap: true, + itemCount: items.length, + itemBuilder: (context, index) { + return itemRow(index); + }, + ), + ElevatedButton( + onPressed: () { + setState(() { + items.add({ + "title": "", + "amount_spent": 0.0, + }); + }); + }, + style: ElevatedButton.styleFrom( + backgroundColor: primary, + shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10)), + ), + child: const Text("Add Item", style: TextStyle(color: Colors.white)), + ), + ], + ), + ), + isSaving + ? Center( + child : SpinKitCircle( + color: primary, + ), + ) + : + ElevatedButton( + onPressed: () { + if (true) { + for (int i=0;i expenseData = widget.isEdit ? { + "expense_title": expenseTitleController.text, + "merchant_name": merchantNameController.text, + "amount_spent": -double.parse(totalAmountController.text), + "payment_method": paymentMethodController.text, + "date_time": dateController.text.trim(), + "category": selectedCategory, + "items": items, + "uuid" : widget.expenseData["uuid"], + } : { + "expense_title": expenseTitleController.text, + "merchant_name": merchantNameController.text, + "amount_spent": -double.parse(totalAmountController.text), + "payment_method": paymentMethodController.text, + "date_time": dateController.text.trim(), + "category": selectedCategory, + "items": items, + }; + + if (widget.isEdit) { + editExpenseBloc.add(EditExpenseSaveEvent(expense : expenseData, isEdit: true)); + } else { + // Add new expense + editExpenseBloc.add(EditExpenseSaveEvent(expense : expenseData, isEdit: false)); + } + } + }, + style: ElevatedButton.styleFrom( + backgroundColor: primary, + shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10)), + ), + child: Text(widget.isEdit ? "Update Expense" : "Add Expense",style: AppStyles.setAppStyle(white, 16, FontWeight.normal, "medium"),), + ), + ], + ), + ); + }, +); + } + + Widget itemRow(int index) { + TextEditingController titleController = TextEditingController(text: items[index]['title']); + TextEditingController amountController = TextEditingController(text: items[index]['amount_spent'].toString()); + + return Padding( + padding: const EdgeInsets.symmetric(vertical: 8.0), + child: Row( + children: [ + Expanded( + child: TextFormField( + controller: titleController, + onChanged: (value) => items[index]['title'] = value, + decoration: InputDecoration(labelText: 'Item Title'), + ), + ), + const SizedBox(width: 10), + SizedBox( + width: 100, + child: TextFormField( + controller: amountController, + keyboardType: TextInputType.number, + onChanged: (value) => items[index]['amount_spent'] = double.tryParse(value) ?? 0.0, + decoration: InputDecoration(labelText: 'Amount'), + ), + ), + IconButton( + icon: const Icon(Icons.delete, color: Colors.red), + onPressed: () { + setState(() { + items.removeAt(index); + }); + }, + ) + ], + ), + ); + } + + + Widget dateWidget() { + return Padding( + padding: const EdgeInsets.only(bottom: 16.0), + child: Container( + padding: const EdgeInsets.symmetric( + horizontal: 12.0, vertical: 16.0), + decoration: BoxDecoration( + color: filterSortBackground, + borderRadius: BorderRadius.circular(8), + border: Border.all(color: border)), + child: InkWell( + enableFeedback: true, + onTap: () async { + DateTime? pickedDate = await showDatePicker( + context: context, + initialDate: + (dateController.text.isNotEmpty) + ? DateTime + .parse(dateController.text) + : DateTime.now(), + firstDate: DateTime(2000), lastDate: DateTime(2050), + ); + if (pickedDate != null) { + setState(() { + dateController.text = + DateFormat('dd-MM-yyyy') + .format(pickedDate); + }); + } + }, + child: Row( + mainAxisAlignment: + MainAxisAlignment.spaceBetween, + children: [ + Text( + dateController.text.isNotEmpty + ? dateController.text + : 'dd-mm-yyyy', + style: const TextStyle( + fontFamily: 'medium', + color: Colors.grey, + ), + ), + Icon(Icons.calendar_month) + ], + ), + ), + ), + ); + } + + Widget textFieldForCreateScreen( + String labelText, + TextEditingController textEditingController) { + return Padding( + padding: const EdgeInsets.only(bottom: 10), + child: TextFormField( + keyboardType: (labelText == "Amount Spent") + ? TextInputType.number + : TextInputType.text, + controller: textEditingController, + style: const TextStyle(fontFamily: "regular"), + decoration: InputDecoration( + border: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + enabledBorder: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + focusedBorder: OutlineInputBorder( + borderRadius: BorderRadius.circular(10), + borderSide: BorderSide( + color: primary, + width: 0.5 + ), + ), + labelText: labelText, + labelStyle: TextStyle( + fontFamily: "regular", + color: grey, + fontSize: 16, + fontWeight: FontWeight.w400, + fontStyle: FontStyle.normal, + ), + floatingLabelStyle: TextStyle( + fontFamily: "regular", + color: primary, + fontSize: 16, + fontWeight: FontWeight.w400, + fontStyle: FontStyle.normal, + ), + fillColor: white, + filled: true), + maxLines: 10, + minLines: 1, + ), + ); + } +} diff --git a/flutter_application_1/lib/components/view/expenses/expenses_view.dart b/flutter_application_1/lib/components/view/expenses/expenses_view.dart new file mode 100644 index 000000000..503f74d03 --- /dev/null +++ b/flutter_application_1/lib/components/view/expenses/expenses_view.dart @@ -0,0 +1,632 @@ +import 'dart:math'; + +import 'package:file_picker/file_picker.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/models/expense_model.dart'; +import 'package:flutter_application_1/components/view/expenses/add_income.dart'; +import 'package:flutter_application_1/components/view/expenses/edit_expenses.dart'; +import 'package:flutter_application_1/components/widgets/expense_card.dart'; +import 'package:flutter_application_1/service/helper/update_expense.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; +import 'package:flutter_spinkit/flutter_spinkit.dart'; +import '../../../theme/colors.dart'; +import '../../bloc/expenses_bloc/expenses_bloc.dart'; +import 'package:image_picker/image_picker.dart'; +import '../../utils/helper/styles.dart'; + +class ExpensesView extends StatefulWidget { + const ExpensesView({super.key}); + + @override + State createState() => _ExpensesViewState(); +} + +class _ExpensesViewState extends State { + bool initialLoading = true; + bool isRecurring = false; + List> expenses = []; + List> recurringExpenses = []; + ExpensesBloc expensesBloc = ExpensesBloc(); + + @override + void initState() { + expensesBloc.add(ExpensesInitialEvent()); + super.initState(); + } + + @override + Widget build(BuildContext context) { + return BlocConsumer( + bloc: expensesBloc, + listener: (context, state) { + if (state is ExpensesLoadingState) { + initialLoading = true; + } + if (state is ExpensesLoadedState) { + initialLoading = false; + expenses = state.expenses; + recurringExpenses = state.recurringExpenses; + } + if (state is ExpensesErrorState) { + initialLoading = false; + ScaffoldMessenger.of(context) + .showSnackBar(SnackBar(content: Text("something went wrong"))); + } + }, + builder: (context, state) { + if (state is ExpensesImageProcessingState || + state is ExpensePdfProcessingState) { + return SpinKitCircle( + color: primary, + ); + } + if (state is ExpenseImageProcessedState) { + expenses = state.expenses; + } + return Scaffold( + backgroundColor: bgColor, + appBar: AppBar( + backgroundColor: bgColor, + forceMaterialTransparency: true, + title: Text( + "Expenses", + style: AppStyles.setAppStyle(black, 20, FontWeight.bold, "black"), + ), + actions: [ + IconButton( + onPressed: () { + showModalBottomSheet( + context: context, + builder: (BuildContext context) { + return Column( + children: [ + Padding( + padding: const EdgeInsets.all(18.0), + child: Text( + "The below displayed expenses are considered as recurring expenses. Please verify them. You can simple add those by clicking on the check icon.",style: TextStyle(fontFamily: "medium"),), + ), + RecurringExpensesWidget(initialExpenses: recurringExpenses,), + ElevatedButton( + onPressed: () { + Navigator.pop(context); + }, + child: Text("Continue"), + ) + ], + ); + }); + }, + icon: Icon(Icons.info,color: primary,), + ) + ], + leading: Icon(Icons.money, color: black), + ), + floatingActionButton: + Column(mainAxisAlignment: MainAxisAlignment.end, children: [ + ElevatedButton( + onPressed: () { + showModalBottomSheet( + isScrollControlled: true, + context: context, + builder: (BuildContext context) { + return Wrap( + children: [ + ListTile( + title: Center( + child: Text( + "Add Income", + style: TextStyle( + color: Colors.grey, fontFamily: "medium"), + )), + onTap: () { + Navigator.pop(context); + Navigator.push(context, MaterialPageRoute(builder: (context) => AddIncome())); + }, + ), + ListTile( + title: Center( + child: Text( + "Add Expense", + style: TextStyle( + color: Colors.grey, fontFamily: "medium"), + )), + onTap: () { + Navigator.pop(context); + Navigator.push( + context, + MaterialPageRoute( + builder: (context) => EditExpenseScreen( + expenseData: {}, + isEdit: false, + ))); + + }, + ), + ListTile( + title: Center( + child: Text( + "Upload Image", + style: TextStyle( + color: Colors.grey, fontFamily: "medium"), + )), + onTap: () async { + final value = await ImagePicker() + .pickImage(source: ImageSource.gallery); + if (value != null) { + expensesBloc + .add(ExpenseImagePickedEvent(value)); + Navigator.pop(context); + print(value.path); + } + }, + ), + ListTile( + title: Center( + child: Text( + "Upload pdf", + style: TextStyle( + color: Colors.grey, fontFamily: "medium"), + )), + onTap: () async { + final value = await FilePicker.platform.pickFiles( + type: FileType.custom, + allowedExtensions: ['pdf']); + if (value != null) { + expensesBloc.add(ExpensePdfPickedEvent(value)); + print(value); + Navigator.pop(context); + } + }, + ), + ], + ); + }); + }, + style: ElevatedButton.styleFrom( + backgroundColor: white, + shape: + CircleBorder(side: BorderSide(color: border, width: 0.5)), + minimumSize: const Size(50, 50), + ), + child: Icon( + Icons.add, + size: 35, + color: Colors.orange, + ), + ), + SizedBox(height: 20), + ElevatedButton( + onPressed: () { + showModalBottomSheet( + context: context, + useSafeArea: true, + isScrollControlled: true, + builder: (BuildContext context) { + return Padding( + padding: EdgeInsets.only( + bottom: MediaQuery.of(context) + .viewInsets + .bottom, // Moves content up + ), + child: ChatBotBottomSheet(expensesBloc: expensesBloc), + ); + }); + }, + style: ElevatedButton.styleFrom( + backgroundColor: white, + shape: + CircleBorder(side: BorderSide(color: border, width: 0.5)), + minimumSize: const Size(50, 50), + ), + child: Icon( + Icons.reddit, + size: 35, + color: Colors.orange, + ), + ), + ]), + body: bodyPartOfExpensesView(expenses), + ); + }, + ); + } + + Widget bodyPartOfExpensesView(List> expenses) { + return initialLoading + ? Center( + child: SpinKitCircle( + color: primary, + )) + : ListView.builder( + physics: const AlwaysScrollableScrollPhysics(), + shrinkWrap: true, + itemCount: expenses.length, + itemBuilder: (context, index) { + return ExpenseCard( + expenseTitle: expenses[index]["expense_title"], + amountSpent: expenses[index]["amount_spent"], + category: expenses[index]["category"], + dateTime: expenses[index]["date_time"], + paymentMethod: expenses[index]["payment_method"], + merchantName: expenses[index]["merchant_name"], + expenseData: expenses[index], + ); + }, + ); + } +} + +class ChatBotBottomSheet extends StatefulWidget { + final ExpensesBloc expensesBloc; + + const ChatBotBottomSheet({ + required this.expensesBloc, + super.key, + }); + + @override + State createState() => _ChatBotBottomSheetState(); +} + +class _ChatBotBottomSheetState extends State { + String? selectedChip; + final _controller = TextEditingController(); + final _scrollController = ScrollController(); + final messages = []; + + @override + void dispose() { + _controller.dispose(); + _scrollController.dispose(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + List chipLabels = [ + "Financial Insights", + "Budgeting Assistance", + "Expense Tracking", + "Others" + ]; + + return Container( + decoration: BoxDecoration( + color: Theme.of(context).scaffoldBackgroundColor, + borderRadius: const BorderRadius.vertical(top: Radius.circular(20)), + boxShadow: [ + BoxShadow( + color: Colors.black.withOpacity(0.1), + blurRadius: 10, + spreadRadius: 1, + ) + ], + ), + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), + child: BlocBuilder( + bloc: widget.expensesBloc, + builder: (context, state) { + if (state is ExpenseChoiceShipSelected) { + selectedChip = state.choiceShip; + } + + // Auto-scroll to bottom when new messages arrive + WidgetsBinding.instance.addPostFrameCallback((_) { + if (_scrollController.hasClients) { + _scrollController.animateTo( + _scrollController.position.maxScrollExtent, + duration: const Duration(milliseconds: 300), + curve: Curves.easeOut, + ); + } + }); + + // Clear text controller after response is received + if (state is ExpenseChatBotLoadedState) { + _controller.clear(); + } + + return Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + // Handle bar at the top + Center( + child: Container( + margin: const EdgeInsets.only(top: 8, bottom: 16), + width: 40, + height: 4, + decoration: BoxDecoration( + color: Colors.grey.shade300, + borderRadius: BorderRadius.circular(2), + ), + ), + ), + + // Title with an icon + Row( + children: [ + Icon( + Icons.support_agent, + color: Theme.of(context).primaryColor, + size: 28, + ), + const SizedBox(width: 10), + Text( + "Finance Assistant", + style: TextStyle( + fontFamily: "medium", + fontSize: 22, + fontWeight: FontWeight.bold, + ), + ), + ], + ), + + const SizedBox(height: 16), + + // Chat topic chips in a horizontally scrollable container + SingleChildScrollView( + scrollDirection: Axis.horizontal, + child: Row( + children: chipLabels.map((label) { + return Padding( + padding: const EdgeInsets.only(right: 8.0), + child: ChoiceChip( + selected: selectedChip == label, + label: Text(label), + labelStyle: TextStyle( + fontWeight: selectedChip == label + ? FontWeight.bold + : FontWeight.normal, + ), + backgroundColor: Colors.grey.shade200, + selectedColor: + Theme.of(context).primaryColor.withOpacity(0.2), + onSelected: (value) { + setState(() { + if (selectedChip == label) { + selectedChip = null; + _controller.clear(); + } else { + selectedChip = label; + // Set the text controller to the chip label + if (label != "Others") { + _controller.text = label; + } else { + _controller.clear(); + } + } + }); + + widget.expensesBloc.add( + ExpenseChoiceShipSelectedEvent( + selectedChip ?? "")); + }, + ), + ); + }).toList(), + ), + ), + + const SizedBox(height: 16), + + // Chat messages area + Flexible( + child: SingleChildScrollView( + controller: _scrollController, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + if (state is ExpenseChatBotLoadedState) + ChatBubble( + message: state.message, + isFromBot: true, + ), + if (state is ExpenseChatBotLoadingState) + Padding( + padding: const EdgeInsets.all(16.0), + child: Center( + child: SpinKitPulse( + color: Theme.of(context).primaryColor, + size: 40.0, + ), + ), + ), + ], + ), + ), + ), + + const SizedBox(height: 16), + + // Input field with send button - always visible + Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(24), + border: Border.all( + color: Theme.of(context).primaryColor.withOpacity(0.5), + width: 1.5, + ), + color: Colors.grey.shade50, + ), + child: Row( + children: [ + Expanded( + child: TextField( + controller: _controller, + decoration: InputDecoration( + hintText: "Type your question here...", + hintStyle: TextStyle( + fontFamily: "medium", + color: Colors.grey.shade500, + ), + contentPadding: const EdgeInsets.symmetric( + horizontal: 16, vertical: 12), + border: InputBorder.none, + ), + maxLines: null, + textCapitalization: TextCapitalization.sentences, + ), + ), + Material( + color: Colors.transparent, + child: InkWell( + borderRadius: BorderRadius.circular(24), + onTap: () { + if (_controller.text.isNotEmpty) { + // Send message to bloc and wait for response + widget.expensesBloc + .add(ExpensechatBotEvent(_controller.text)); + // Controller will be cleared after response is received + } + }, + child: Padding( + padding: const EdgeInsets.all(12.0), + child: Icon( + Icons.send_rounded, + color: Theme.of(context).primaryColor, + ), + ), + ), + ), + ], + ), + ), + + const SizedBox(height: 16), + ], + ); + }, + ), + ), + ); + } +} + +// Custom chat bubble widget +class ChatBubble extends StatelessWidget { + final String message; + final bool isFromBot; + + const ChatBubble({ + required this.message, + required this.isFromBot, + Key? key, + }) : super(key: key); + + @override + Widget build(BuildContext context) { + return Container( + margin: const EdgeInsets.symmetric(vertical: 8), + child: Row( + mainAxisAlignment: + isFromBot ? MainAxisAlignment.start : MainAxisAlignment.end, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + if (isFromBot) + Container( + margin: const EdgeInsets.only(right: 8), + child: CircleAvatar( + backgroundColor: + Theme.of(context).primaryColor.withOpacity(0.2), + radius: 16, + child: Icon( + Icons.android, + size: 20, + color: Theme.of(context).primaryColor, + ), + ), + ), + Flexible( + child: Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + decoration: BoxDecoration( + color: isFromBot + ? Colors.grey.shade100 + : Theme.of(context).primaryColor.withOpacity(0.1), + borderRadius: BorderRadius.circular(16), + border: Border.all( + color: isFromBot + ? Colors.grey.shade300 + : Theme.of(context).primaryColor.withOpacity(0.3), + width: 1, + ), + ), + child: Text( + message, + style: TextStyle( + fontSize: 15, + color: Colors.black87, + ), + ), + ), + ), + ], + ), + ); + } +} + +class RecurringExpensesWidget extends StatefulWidget { + final List> initialExpenses; + + const RecurringExpensesWidget({Key? key, required this.initialExpenses}) : super(key: key); + + @override + _RecurringExpensesWidgetState createState() => _RecurringExpensesWidgetState(); +} + +class _RecurringExpensesWidgetState extends State { + late List> recurringExpenses; + + @override + void initState() { + super.initState(); + recurringExpenses = List.from(widget.initialExpenses); + } + + void removeItem(int index, bool canSend) { + if (index >= 0 && index < recurringExpenses.length) { + createExpense(recurringExpenses[index]); + setState(() { + recurringExpenses.removeAt(index); + }); + } + } + + @override + Widget build(BuildContext context) { + return Column( + children: recurringExpenses.asMap().entries.map((entry) { + int index = entry.key; + Map expense = entry.value; + return Card( + margin: EdgeInsets.symmetric(vertical: 5, horizontal: 10), + child: ListTile( + title: Text( + expense['expense_title'], + style: TextStyle(fontSize: 16, fontWeight: FontWeight.w500), + ), + subtitle: Text("Amount: ${expense['amount_spent']}"), + trailing: Row( + mainAxisSize: MainAxisSize.min, + children: [ + IconButton( + icon: Icon(Icons.check_circle, color: Colors.green), + onPressed: () => removeItem(index,true), + ), + IconButton( + icon: Icon(Icons.cancel, color: Colors.red), + onPressed: () => removeItem(index,false), + ), + ], + ), + ), + ); + }).toList(), + ); + } +} + diff --git a/flutter_application_1/lib/components/view/expenses/view_expense.dart b/flutter_application_1/lib/components/view/expenses/view_expense.dart new file mode 100644 index 000000000..299b721fc --- /dev/null +++ b/flutter_application_1/lib/components/view/expenses/view_expense.dart @@ -0,0 +1,33 @@ + +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/utils/helper/styles.dart'; +import 'package:flutter_application_1/theme/colors.dart'; + +class ViewExpense extends StatefulWidget { + Map expenseModel; + ViewExpense({super.key, required this.expenseModel}); + + @override + State createState() => _ViewExpenseState(); +} + +class _ViewExpenseState extends State { + @override + Widget build(BuildContext context) { + return Scaffold( + backgroundColor: bgColor, + appBar: AppBar( + backgroundColor: bgColor, + title: Text( + "View Expenses", + style: AppStyles.setAppStyle(black, 29, FontWeight.bold, "bold"), + ), + ), + body: bodyPartOfViewExpense(), + ); + } + + Widget bodyPartOfViewExpense() { + return SizedBox(); + } +} diff --git a/flutter_application_1/lib/components/view/main_screeen.dart b/flutter_application_1/lib/components/view/main_screeen.dart new file mode 100644 index 000000000..f87023a7f --- /dev/null +++ b/flutter_application_1/lib/components/view/main_screeen.dart @@ -0,0 +1,121 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:firebase_messaging/firebase_messaging.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/models/user_model.dart'; +import 'package:flutter_application_1/components/view/analytics/analytics_view.dart'; +import 'package:flutter_application_1/components/view/expenses/expenses_view.dart'; +import 'package:flutter_application_1/components/view/settings/settings.dart'; +import 'package:flutter_application_1/service/auth/auth.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; + +import '../../service/notification_service.dart'; +import '../../theme/colors.dart'; +import '../bloc/main_screen_bloc/main_screen_bloc.dart'; + +class MainScreen extends StatefulWidget { + const MainScreen({super.key}); + + @override + State createState() => _MainScreenState(); +} + +class _MainScreenState extends State { + + List mainScreens = [ + ExpensesView(), + AnalyticsView(), + ProfilePage(), + ]; + MainScreenBloc mainScreenBloc = MainScreenBloc(); + int selectedIndex = 0; + + void _onTap(int index) { + mainScreenBloc.add(MainScreenBottomNavigationBarSwitchingEvent(index)); + } + + @override + void initState() { + // TODO: implement initState + getUserDetails(); + getFirebaseNotification(); + super.initState(); + } + + Future getUserDetails() async { + print("name from firebase : ${FirebaseAuth.instance.currentUser!.displayName!}"); + if (UserModel.name.isEmpty) { + UserModel.name = FirebaseAuth.instance.currentUser!.displayName!; + UserModel.email = FirebaseAuth.instance.currentUser!.email!; + UserModel.photoUrl = FirebaseAuth.instance.currentUser!.photoURL!; + } + } + + Future getFirebaseNotification() async { + await NotificationService.instance.initialize(context); + NotificationSettings settings = + await FirebaseMessaging.instance.getNotificationSettings(); + if (settings.authorizationStatus == AuthorizationStatus.notDetermined) { + settings = await FirebaseMessaging.instance.requestPermission(); + } + + if (settings.authorizationStatus == AuthorizationStatus.authorized) { + print("Has permission"); + // do api call to backend to create FCM + await sendFCM(); // enabled after the backend implemetation. + } else { + + } + } + + @override + Widget build(BuildContext context) { + return BlocConsumer( + bloc: mainScreenBloc, + listener: (context, state) { + if (state is MainScreenBottomNavigationBarSwitchingState) { + selectedIndex = state.index; + } + }, + builder: (context, state) { + return Scaffold( + backgroundColor: bgColor, + body: mainScreens[selectedIndex], + bottomNavigationBar: BottomNavigationBar( + backgroundColor: Colors.white, // Light theme background + currentIndex: selectedIndex, + onTap: _onTap, + type: BottomNavigationBarType.fixed, + selectedItemColor: Colors.blueAccent, // Active item color + unselectedItemColor: Colors.grey, // Inactive item color + showSelectedLabels: true, + showUnselectedLabels: false, + elevation: 8, // Adds a floating effect + items: [ + BottomNavigationBarItem( + icon: Icon( + selectedIndex == 0 ? Icons.money : Icons.money_off, + size: selectedIndex == 0 ? 30 : 24, // Scale effect + ), + label: "Expenses", + ), + BottomNavigationBarItem( + icon: Icon( + selectedIndex == 1 ? Icons.analytics : Icons.analytics_outlined, + size: selectedIndex == 1 ? 30 : 24, + ), + label: "Analytics", + ), + BottomNavigationBarItem( + icon: Icon( + selectedIndex == 2 ? Icons.settings : Icons.settings_outlined, + size: selectedIndex == 2 ? 30 : 24, + ), + label: "Settings", + ), + ], + ), + ); + }, + ); + } +} diff --git a/flutter_application_1/lib/components/view/settings/settings.dart b/flutter_application_1/lib/components/view/settings/settings.dart new file mode 100644 index 000000000..599ec7c23 --- /dev/null +++ b/flutter_application_1/lib/components/view/settings/settings.dart @@ -0,0 +1,245 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:firebase_core/firebase_core.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/models/user_model.dart'; +import 'package:flutter_application_1/data/appvalues.dart'; + +import '../Auth/login_view.dart'; + +class ProfilePage extends StatefulWidget { + const ProfilePage({super.key}); + + @override + State createState() => _ProfilePageState(); +} + +class _ProfilePageState extends State { + // Mock user data - replace with actual user data from your app + final Map userData = { + 'name': UserModel.name.isNotEmpty ? UserModel.name : "Not provided", + 'email': UserModel.email.isNotEmpty ? UserModel.email : "Not provided", + 'phone': UserModel.phone.isNotEmpty ? UserModel.phone : "Not provided", + 'accountType': 'Premium', + 'profilePicture': UserModel.photoUrl, + }; + + @override + Widget build(BuildContext context) { + return Scaffold( + backgroundColor: Colors.white, + appBar: AppBar( + title: const Text( + 'My Profile', + style: TextStyle(color: Colors.white, fontFamily: "medium"), + ), + backgroundColor: Colors.blue, + elevation: 0, + leading: Icon(Icons.settings_outlined), + ), + body: SingleChildScrollView( + child: Column( + children: [ + // Blue curved header with profile image + Container( + height: 150, + decoration: const BoxDecoration( + color: Colors.blue, + borderRadius: BorderRadius.only( + bottomLeft: Radius.circular(30), + bottomRight: Radius.circular(30), + ), + ), + child: Center( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + const SizedBox(height: 10), + // Profile picture + CircleAvatar( + radius: 50, + backgroundColor: Colors.white, + child: CircleAvatar( + radius: 47, + backgroundImage: NetworkImage(userData['profilePicture']), + // If image fails to load, show a fallback + onBackgroundImageError: (_, __) {}, + child: userData['profilePicture'] == null + ? const Icon(Icons.person, size: 60, color: Colors.blue) + : null, + ), + ), + ], + ), + ), + ), + + const SizedBox(height: 20), + + // User name with larger font + Padding( + padding: const EdgeInsets.symmetric(horizontal: 20), + child: Text( + userData['name'], + style: const TextStyle( + fontSize: 24, + fontWeight: FontWeight.bold, + color: Colors.blue, + fontFamily: "medium", + ), + textAlign: TextAlign.center, + ), + ), + + // Account type chip + Container( + margin: const EdgeInsets.symmetric(vertical: 10), + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + decoration: BoxDecoration( + color: Colors.blue.shade100, + borderRadius: BorderRadius.circular(20), + ), + child: Text( + '${userData['accountType']} Account', + style: TextStyle( + color: Colors.blue.shade800, + fontWeight: FontWeight.bold, + fontFamily: "medium" + ), + ), + ), + + // Text( + // "Will implement subscription things in future" + // ), + + const SizedBox(height: 30), + + // Profile information cards + _buildInfoCard('Email', userData['email'], Icons.email), + _buildInfoCard('Phone', userData['phone'], Icons.phone), + + const SizedBox(height: 40), + + // Logout button + Padding( + padding: const EdgeInsets.symmetric(horizontal: 20), + child: ElevatedButton( + onPressed: () { + // Add logout functionality here + _showLogoutDialog(context); + }, + style: ElevatedButton.styleFrom( + backgroundColor: Colors.red.shade400, + foregroundColor: Colors.white, + minimumSize: const Size(double.infinity, 50), + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(10), + ), + ), + child: const Text( + 'Logout', + style: TextStyle(fontSize: 16,fontFamily: "medium"), + ), + ), + ), + + const SizedBox(height: 30), + ], + ), + ), + ); + } + + Widget _buildInfoCard(String title, String value, IconData icon) { + return Container( + margin: const EdgeInsets.symmetric(horizontal: 20, vertical: 10), + padding: const EdgeInsets.all(15), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(15), + boxShadow: [ + BoxShadow( + color: Colors.blue.withOpacity(0.1), + spreadRadius: 2, + blurRadius: 7, + offset: const Offset(0, 3), + ), + ], + ), + child: Row( + children: [ + Icon(icon, color: Colors.blue, size: 28), + const SizedBox(width: 15), + Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + title, + style: TextStyle( + fontSize: 14, + color: Colors.blue.shade300, + fontFamily: "medium" + ), + ), + const SizedBox(height: 5), + Text( + value, + style: const TextStyle( + fontSize: 16, + color: Colors.black87, + fontWeight: FontWeight.w500, + fontFamily: "medium" + ), + ), + ], + ), + ], + ), + ); + } + + // Logout confirmation dialog + void _showLogoutDialog(BuildContext context) { + showDialog( + context: context, + builder: (BuildContext context) { + return AlertDialog( + title: const Text("Logout"), + content: const Text("Are you sure you want to logout?"), + actions: [ + TextButton( + onPressed: () => Navigator.of(context).pop(), + child: Text("Cancel", style: TextStyle( + color: Colors.blue.shade700, + fontFamily: "medium" + )), + ), + ElevatedButton( + onPressed: () async { + // Implement your logout logic here + await FirebaseAuth.instance.signOut(); + AppValues.jwtToken = ""; + UserModel.phone = ""; + UserModel.email = ""; + UserModel.name = ""; + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => const LoginPage()), + (Route route) => false, + ); + // You might want to navigate to login page or clear user session + }, + style: ElevatedButton.styleFrom( + backgroundColor: Colors.red.shade400, + ), + child: const Text("Logout", style: TextStyle( + color: Colors.white, + fontFamily: "medium" + )), + ), + ], + ); + }, + ); + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/components/view/splash_screen.dart b/flutter_application_1/lib/components/view/splash_screen.dart new file mode 100644 index 000000000..7f58770f7 --- /dev/null +++ b/flutter_application_1/lib/components/view/splash_screen.dart @@ -0,0 +1,80 @@ +import 'package:firebase_auth/firebase_auth.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/view/Auth/login_view.dart'; +import 'package:flutter_application_1/components/view/main_screeen.dart'; +import 'package:flutter_application_1/data/appvalues.dart'; + +import '../../theme/colors.dart'; + +class SplashScreen extends StatefulWidget { + const SplashScreen({super.key}); + + @override + State createState() => _SplashScreenState(); +} + +class _SplashScreenState extends State { + + @override + void initState() { + navigateToMainScreen(); + super.initState(); + } + + Future navigateToMainScreen() async { + await Future.delayed(const Duration(seconds: 2)); + + try { + var user = FirebaseAuth.instance.currentUser; + + if (user == null) { + // User is not logged in, navigate to LoginPage + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => const LoginPage()), + (Route route) => false, + ); + } else { + String? jwtToken = await user.getIdToken(); + if (jwtToken == null) { + throw Exception("Unauthorized"); + } + AppValues.jwtToken = jwtToken; + // User is logged in, navigate to MainScreen + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => const MainScreen()), + (Route route) => false, + ); + } + } catch (e) { + print("Error in splash screen: $e"); + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => const LoginPage()), + (Route route) => false, + ); + } + } + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Center( + child: Container( + height: 100, + width: 100, + decoration: BoxDecoration( + color: Colors.blue, + borderRadius: BorderRadius.circular(20), + ), + child: const Icon( + Icons.account_balance_wallet, + color: Colors.white, + size: 60, + ), + ), + ), + ); + } +} diff --git a/flutter_application_1/lib/components/widgets/analytics_graph.dart b/flutter_application_1/lib/components/widgets/analytics_graph.dart new file mode 100644 index 000000000..fe1632ac4 --- /dev/null +++ b/flutter_application_1/lib/components/widgets/analytics_graph.dart @@ -0,0 +1,68 @@ +import 'package:fl_chart/fl_chart.dart'; +import 'package:flutter/material.dart'; + +class CustomGraph extends StatelessWidget { + final List xList; + final List yList; + final String xTitle; + final String yTitle; + + const CustomGraph({ + Key? key, + required this.xList, + required this.yList, + required this.xTitle, + required this.yTitle, + }) : super(key: key); + + @override + Widget build(BuildContext context) { + return AspectRatio( + aspectRatio: 1.5, // Adjust aspect ratio + child: Padding( + padding: const EdgeInsets.all(16.0), + child: LineChart( + LineChartData( + gridData: FlGridData(show: false), // Hide background grid + titlesData: FlTitlesData( + leftTitles: AxisTitles( + sideTitles: SideTitles( + showTitles: true, + reservedSize: 40, + getTitlesWidget: (value, meta) { + return Text(value.toString(), style: const TextStyle(fontSize: 12)); + }, + ), + ), + bottomTitles: AxisTitles( + sideTitles: SideTitles( + showTitles: true, + reservedSize: 40, + getTitlesWidget: (value, meta) { + return Text(value.toString(), style: const TextStyle(fontSize: 12)); + }, + ), + ), + topTitles: const AxisTitles(sideTitles: SideTitles(showTitles: false)), + rightTitles: const AxisTitles(sideTitles: SideTitles(showTitles: false)), + ), + borderData: FlBorderData(show: true), // Show chart border + lineBarsData: [ + LineChartBarData( + spots: List.generate( + xList.length, + (index) => FlSpot(xList[index], yList[index]), + ), + isCurved: true, // Smooth curve + color: Colors.blue, + barWidth: 3, + belowBarData: BarAreaData(show: false), // No fill color + dotData: FlDotData(show: true), // Show points + ), + ], + ), + ), + ), + ); + } +} diff --git a/flutter_application_1/lib/components/widgets/expense_card.dart b/flutter_application_1/lib/components/widgets/expense_card.dart new file mode 100644 index 000000000..4931ac08f --- /dev/null +++ b/flutter_application_1/lib/components/widgets/expense_card.dart @@ -0,0 +1,97 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/view/expenses/edit_expenses.dart'; +import 'package:intl/intl.dart'; + +import '../../theme/colors.dart'; + +class ExpenseCard extends StatelessWidget { + final String expenseTitle; + final int amountSpent; + final String category; + final String dateTime; // Accept date as String from JSON + final String paymentMethod; + final String merchantName; + final Map expenseData; + + const ExpenseCard({ + super.key, + required this.expenseTitle, + required this.amountSpent, + required this.category, + required this.dateTime, + required this.paymentMethod, + required this.merchantName, + required this.expenseData, + }); + + @override + Widget build(BuildContext context) { + // Safely parse the date string to DateTime + DateTime parsedDate = DateTime.tryParse(dateTime) ?? DateTime.now(); + + return GestureDetector( + onTap: () { + Navigator.push(context, MaterialPageRoute(builder: (context) => EditExpenseScreen(expenseData: expenseData, isEdit: true,))); + }, + child: Card( + color: white, + shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12), side: BorderSide(color: border, width: 0.5)), + elevation: 3, + margin: const EdgeInsets.symmetric(vertical: 8, horizontal: 16), + child: Padding( + padding: const EdgeInsets.all(12), + child: Row( + children: [ + // Icon for category (currently hardcoded for movies) + CircleAvatar( + radius: 24, + backgroundColor: Colors.blueAccent, + child: Icon(Icons.movie, color: Colors.white, size: 28), + ), + const SizedBox(width: 12), + + // Expense details + Expanded( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + expenseTitle, + style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold), + ), + const SizedBox(height: 4), + Text( + merchantName, + style: TextStyle(fontSize: 14, color: Colors.grey[700]), + ), + const SizedBox(height: 4), + Text( + DateFormat.yMMMd().format(parsedDate), + style: TextStyle(fontSize: 12, color: Colors.grey[600]), + ), + ], + ), + ), + + // Amount spent + Column( + crossAxisAlignment: CrossAxisAlignment.end, + children: [ + Text( + '₹$amountSpent', + style: const TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.redAccent), + ), + const SizedBox(height: 4), + Text( + paymentMethod, + style: TextStyle(fontSize: 12, color: Colors.grey[700]), + ), + ], + ), + ], + ), + ), + ), + ); + } +} diff --git a/flutter_application_1/lib/data/appvalues.dart b/flutter_application_1/lib/data/appvalues.dart new file mode 100644 index 000000000..77dcef485 --- /dev/null +++ b/flutter_application_1/lib/data/appvalues.dart @@ -0,0 +1,6 @@ + +class AppValues { + static String ip = "http://10.18.0.146:3000/"; + static String fcm = ""; + static String jwtToken = ""; +} \ No newline at end of file diff --git a/flutter_application_1/lib/firebase_options.dart b/flutter_application_1/lib/firebase_options.dart new file mode 100644 index 000000000..9c03fd8cc --- /dev/null +++ b/flutter_application_1/lib/firebase_options.dart @@ -0,0 +1,62 @@ +// 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; + +/// Default [FirebaseOptions] for use with your Firebase apps. +/// +/// Example: +/// ```dart +/// import 'firebase_options.dart'; +/// // ... +/// await Firebase.initializeApp( +/// options: DefaultFirebaseOptions.currentPlatform, +/// ); +/// ``` +class DefaultFirebaseOptions { + static FirebaseOptions get currentPlatform { + if (kIsWeb) { + throw UnsupportedError( + 'DefaultFirebaseOptions have not been configured for web - ' + 'you can reconfigure this by running the FlutterFire CLI again.', + ); + } + switch (defaultTargetPlatform) { + case TargetPlatform.android: + return android; + case TargetPlatform.iOS: + throw UnsupportedError( + 'DefaultFirebaseOptions have not been configured for ios - ' + 'you can reconfigure this by running the FlutterFire CLI again.', + ); + case TargetPlatform.macOS: + throw UnsupportedError( + 'DefaultFirebaseOptions have not been configured for macos - ' + 'you can reconfigure this by running the FlutterFire CLI again.', + ); + case TargetPlatform.windows: + throw UnsupportedError( + 'DefaultFirebaseOptions have not been configured for windows - ' + 'you can reconfigure this by running the FlutterFire CLI again.', + ); + case TargetPlatform.linux: + throw UnsupportedError( + 'DefaultFirebaseOptions have not been configured for linux - ' + 'you can reconfigure this by running the FlutterFire CLI again.', + ); + default: + throw UnsupportedError( + 'DefaultFirebaseOptions are not supported for this platform.', + ); + } + } + + static const FirebaseOptions android = FirebaseOptions( + apiKey: 'AIzaSyDFcAakxrfNnQsARzoFKAKhohW_-nMn-CE', + appId: '1:101217440706:android:a9c09bb6b54d47d08b7c5b', + messagingSenderId: '101217440706', + projectId: 'finnfloww', + storageBucket: 'finnfloww.firebasestorage.app', + ); +} diff --git a/flutter_application_1/lib/main.dart b/flutter_application_1/lib/main.dart new file mode 100644 index 000000000..6742b17c5 --- /dev/null +++ b/flutter_application_1/lib/main.dart @@ -0,0 +1,25 @@ + +import 'package:firebase_core/firebase_core.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_application_1/components/view/analytics/analytics_view.dart'; +import 'package:flutter_application_1/components/view/expenses/expenses_view.dart'; +import 'package:flutter_application_1/components/view/main_screeen.dart'; +import 'package:flutter_application_1/components/view/settings/settings.dart'; +import 'package:flutter_application_1/components/view/splash_screen.dart'; + +Future main() async { + WidgetsFlutterBinding.ensureInitialized(); + await Firebase.initializeApp(); + Widget myApp = MaterialApp( + debugShowCheckedModeBanner: false, + initialRoute: "/", + routes: { + "/" : (BuildContext context) => SplashScreen(), + "/main" : (BuildContext context) => MainScreen(), + "/analytics" : (BuildContext context) => AnalyticsView(), + "/expenses" : (BuildContext context) => ExpensesView(), + "/settings" : (BuildContext context) => ProfilePage() + }, + ); + runApp(myApp); +} \ No newline at end of file diff --git a/flutter_application_1/lib/service/auth/auth.dart b/flutter_application_1/lib/service/auth/auth.dart new file mode 100644 index 000000000..17a2154c1 --- /dev/null +++ b/flutter_application_1/lib/service/auth/auth.dart @@ -0,0 +1,46 @@ + +import 'dart:convert'; +import 'package:http/http.dart' as http; + +import '../../data/appvalues.dart'; +Future authService() async { + try { + var request = await http.get( + Uri.parse("${AppValues.ip}login"), + headers: { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + } + ); + if (request.statusCode == 200) { + return true; + } + return false; + } catch (e) { + print("Error on add income : $e"); + return false; + } +} + +Future sendFCM() async { + try { + Map body = { + "fcm" : AppValues.fcm + }; + var request = await http.post( + Uri.parse("${AppValues.ip}fcm"), + headers: { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + }, + body: jsonEncode(body) + ); + if (request.statusCode == 200) { + return true; + } + return false; + } catch (e) { + print("Error on add income : $e"); + return false; + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/service/expense_service.dart b/flutter_application_1/lib/service/expense_service.dart new file mode 100644 index 000000000..26bae708c --- /dev/null +++ b/flutter_application_1/lib/service/expense_service.dart @@ -0,0 +1,120 @@ +import 'dart:convert'; +import 'dart:io'; +import 'package:file_picker/file_picker.dart'; +import 'package:flutter_application_1/components/models/expense_model.dart'; +import 'package:flutter_application_1/data/appvalues.dart'; +import 'package:image_picker/image_picker.dart'; +import 'package:http/http.dart' as http; + +Future uploadBill(XFile image) async { + try { + List imageBytes = await image.readAsBytes(); + String base64String = base64Encode(imageBytes); + print("The string is $base64String"); + var response = await http.post( + Uri.parse("${AppValues.ip}addExpenseByBill"), + headers: { + "Authorization" : "Bearer ${AppValues.jwtToken}" + }, + body: {"image64": base64String}, + ); + if (response.statusCode == 200) { + return "Upload successful: ${response.body}"; + } else { + return "Upload failed: ${response.statusCode} - ${response.body}"; + } + } catch (e) { + return "Error: $e"; + } +} + +Future uploadPdf(FilePickerResult pdf) async { + try { + if (pdf.files.isEmpty) { + return "No file selected"; + } + + var request = http.MultipartRequest( + "POST", + Uri.parse("${AppValues.ip}addBankStatement"), + ); + request.headers.addAll({ + "Content-Type": "multipart/form-data", + "Authorization" : "Bearer ${AppValues.jwtToken}" + }); + File file = File(pdf.files.single.path!); + request.files.add(await http.MultipartFile.fromPath( + 'file', + file.path, + filename: pdf.files.single.name, + )); + + var response = await request.send(); + + if (response.statusCode == 200) { + return "Upload successful"; + } else { + return "Upload failed: ${response.statusCode}"; + } + } catch (e) { + return "Error: $e"; + } +} + + +Future getExpenses() async { + try{ + List> expenses = []; + // print("JWT token : ${AppValues.jwtToken}"); + var response = await http.get( + Uri.parse("${AppValues.ip}getExpense"), + headers: { + "Content-Type": "multipart/form-data", + "Authorization" : "Bearer ${AppValues.jwtToken}" + } + ); + List> recurringExpenses = []; + if(response.statusCode == 200){ + var data = jsonDecode(response.body); + for(var expense in data["data"]){ + recurringExpenses.add(expense); + } + for(var expense in data["message"]){ + expenses.add(expense); + } + print("Success: $expenses"); + return { + "expenses": expenses, + "recurringExpenses": recurringExpenses + }; + } + } + catch(e){ + print("Error: $e"); + } + return []; +} + +Future getChatBotResponse(String message) async { + try { + var response = await http.post( + Uri.parse("${AppValues.ip}botQuery"), + headers: { + "Authorization" : "Bearer ${AppValues.jwtToken}" + }, + body: {"botQuery": message}, + ); + var decodedResponse = jsonDecode(response.body); + if (decodedResponse["status"] == 200) { + return jsonDecode(response.body)["answer"]; + } + else if(decodedResponse["status"] == 404){ + return decodedResponse["message"]; + } + else { + return "Error: ${response.statusCode} - ${response.body}"; + } + } catch (e) { + return "Error: $e"; + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/service/helper/global_service.dart b/flutter_application_1/lib/service/helper/global_service.dart new file mode 100644 index 000000000..94c80ae8e --- /dev/null +++ b/flutter_application_1/lib/service/helper/global_service.dart @@ -0,0 +1,74 @@ +import 'dart:convert'; +import 'package:http/http.dart' as http; + +import '../../data/appvalues.dart'; + +class GlobalService { + static Future globalService( + {required String endpoint, + Map? requestBody, + required String method}) async { + final Uri url = + Uri.parse("${AppValues.ip}$endpoint"); // ✅ Uses passed endpoint + final Map headers = { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + }; + + try { + http.Response response = + await _makeRequest(url, headers, method, requestBody); + + if (response.statusCode == 401) { + // 🔄 Refresh token and retry + response = await _makeRequest(url, headers, method, requestBody); + } + + // 🛠 Handle all response codes properly + return _handleResponse(response); + } catch (e) { + return {"status": false, "message": "Request failed: $e"}; + } + } + + /// 🔹 Helper function to make HTTP requests + static Future _makeRequest( + Uri url, + Map headers, + String method, + Map? requestBody) async { + switch (method.toUpperCase()) { + case "POST": + return await http.post(url, + headers: headers, body: jsonEncode(requestBody)); + case "GET": + return await http.get(url, headers: headers); + case "PUT": + return await http.put(url, + headers: headers, body: jsonEncode(requestBody)); + case "DELETE": + return await http.delete(url, headers: headers); + default: + throw Exception("Invalid HTTP method: $method"); + } + } + + /// 🔹 Helper function to process API response + static dynamic _handleResponse(http.Response response) { + try { + final Map decodedBody = jsonDecode(response.body); + + if (response.statusCode >= 200 && response.statusCode < 300) { + decodedBody["status"] = true; + return decodedBody; + } else { + return { + "status": false, + "message": decodedBody["message"] ?? "Something went wrong" + }; + } + } catch (e) { + return {"status": false, "message": "Invalid JSON response: $e"}; + } + } +} diff --git a/flutter_application_1/lib/service/helper/update_expense.dart b/flutter_application_1/lib/service/helper/update_expense.dart new file mode 100644 index 000000000..ea88c930a --- /dev/null +++ b/flutter_application_1/lib/service/helper/update_expense.dart @@ -0,0 +1,68 @@ +import "dart:convert"; + +import "package:flutter_application_1/data/appvalues.dart"; +import "package:http/http.dart" as http; + + +Future updateExpense(Map expense) async { + try { + String uuid = expense["uuid"]; + expense.remove("uuid"); + var request = await http.put( + Uri.parse("${AppValues.ip}updateExpense/$uuid"), + body: jsonEncode(expense), + headers: { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + } + ); + if (request.statusCode == 200) { + return true; + } + return false; + } catch (e) { + print("Error on update expense : $e"); + return false; + } +} + +Future createExpense(Map expense) async { + try { + var request = await http.post( + Uri.parse("${AppValues.ip}addExpense"), + body: jsonEncode({"data":expense}), + headers: { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + } + ); + print("Completed request ${request.statusCode}"); + if (request.statusCode == 200) { + return true; + } + return false; + } catch (e) { + print("Error on create expense : $e"); + return false; + } +} + +Future addIncome(Map income) async { + try { + var request = await http.post( + Uri.parse("${AppValues.ip}/addExpense"), + body: jsonEncode(income), + headers: { + "Content-Type": "application/json", + "Authorization" : "Bearer ${AppValues.jwtToken}" + } + ); + if (request.statusCode == 200) { + return true; + } + return false; + } catch (e) { + print("Error on add income : $e"); + return false; + } +} \ No newline at end of file diff --git a/flutter_application_1/lib/service/notification_service.dart b/flutter_application_1/lib/service/notification_service.dart new file mode 100644 index 000000000..6f07ab491 --- /dev/null +++ b/flutter_application_1/lib/service/notification_service.dart @@ -0,0 +1,143 @@ + +import 'dart:convert'; + +import 'package:firebase_messaging/firebase_messaging.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_local_notifications/flutter_local_notifications.dart'; + +import '../data/appvalues.dart'; + +class NotificationService { + NotificationService._(); + static final NotificationService instance = NotificationService._(); + final FirebaseMessaging _fcm = FirebaseMessaging.instance; + final FlutterLocalNotificationsPlugin _localNotifications = + FlutterLocalNotificationsPlugin(); + + bool isFlutterNotificationInitialized = false; + + Future initialize(BuildContext context) async { + FirebaseMessaging.onBackgroundMessage(_firebaseMessagingBackgroundHandler); + + try { + String? token = await _fcm.getToken(); + if (token != null) { + AppValues.fcm = token; + print("FCM Token: $token"); + } else { + print("FCM Token is null"); + } + } catch (e, stackTrace) { + print("Error obtaining FCM token: $e"); + } + await requestPermission(); + await _setupFlutterNotifications(context); + await _setupMessageHandlers(context); + } + + Future requestPermission() async { + final settings = await _fcm.requestPermission( + alert: true, + badge: true, + sound: true, + ); + print( + "Notification permission status: ${settings.authorizationStatus}"); + } + + Future _setupFlutterNotifications(BuildContext context) async { + if (isFlutterNotificationInitialized) { + print("Already it's initialized"); + return; + } + + const channel = AndroidNotificationChannel( + 'finance_updates_channel', + 'finance Updates Channel', + description: 'This channel contains updates of finance in the application.', + importance: Importance.max, + ); + + await _localNotifications + .resolvePlatformSpecificImplementation< + AndroidFlutterLocalNotificationsPlugin>() + ?.createNotificationChannel(channel); + + const androidSettings = + AndroidInitializationSettings('@mipmap/ic_launcher'); + const initializationSettings = InitializationSettings( + android: androidSettings, + ); + + await _localNotifications.initialize( + initializationSettings, + onDidReceiveNotificationResponse: (response) async { + if (response.payload != null) { + print("From initialization"); + } + }, + ); + // await _checkInitialNotification(context); + + isFlutterNotificationInitialized = true; + } + + Future _setupMessageHandlers(BuildContext context) async { + FirebaseMessaging.onMessage.listen((message) { + print("Notification received"); + if (message.notification != null) { + showNotification(message); + } + }); + + FirebaseMessaging.onMessageOpenedApp.listen((message) { + if (message.data.isNotEmpty) { + + } + }); + + try { + // Check for initial message when app starts + RemoteMessage? initialMessage = await _fcm.getInitialMessage(); + + if (initialMessage != null) { + print("Initial message received when app was terminated"); + } + } catch (e, stackTrace) { + print("Error handling initial message: $e"); + } + } + + Future showNotification(RemoteMessage message) async { + final notification = message.notification; + if (notification == null) return; + + AndroidNotificationDetails androidDetails = AndroidNotificationDetails( + 'finance_updates_channel', + 'finance Updates Channel', + importance: Importance.max, + priority: Priority.high, + ); + + NotificationDetails details = NotificationDetails(android: androidDetails); + + await _localNotifications.show( + notification.hashCode, + notification.title, + notification.body, + details, + payload: jsonEncode(message.data), + ); + } + + +} + +@pragma('vm:entry-point') +Future _firebaseMessagingBackgroundHandler(RemoteMessage message) async { + print( + "Notification from the background handler function : ${message.data}"); + // await Firebase.initializeApp(); + // await NotificationService.instance._setupFlutterNotifications(); + // await NotificationService.instance.showMessage(message,"background"); +} \ No newline at end of file diff --git a/flutter_application_1/lib/theme/colors.dart b/flutter_application_1/lib/theme/colors.dart new file mode 100644 index 000000000..32782c837 --- /dev/null +++ b/flutter_application_1/lib/theme/colors.dart @@ -0,0 +1,109 @@ +import 'package:flutter/material.dart'; + +Color primary = const Color.fromRGBO(0, 123, 255, 1); +Color success = const Color.fromRGBO(40, 167, 69, 1); +Color bgColor = const Color.fromRGBO(255, 255, 255, 1); +Color warning = const Color.fromRGBO(220, 53, 69, 1); +Color textColor = const Color.fromRGBO(52, 58, 64, 1); +Color black = const Color.fromRGBO(0, 0, 0, 1); +Color white = const Color.fromRGBO(255, 255, 255, 1); +Color border = const Color.fromRGBO(218, 218, 218, 1); +Color grey = const Color.fromRGBO(128, 128, 128, 1); + +Color scanned = const Color.fromRGBO(0, 150, 136, 1); +Color contact = const Color.fromRGBO(243, 149, 35, 1); +Color sales = const Color.fromRGBO(211, 91, 115, 1); +Color transport = const Color.fromRGBO(79, 195, 247, 1); +Color construction = const Color.fromRGBO(186, 104, 200, 1); +Color housing = const Color.fromRGBO(255, 183, 77, 1); +Color it = const Color.fromRGBO(129, 199, 132, 1); +Color marketing = const Color.fromRGBO(255, 213, 79, 1); +Color searchHintText = const Color(0xff808080); +Color filterSortBackground = const Color(0xFFF8F9FA); +Color activeSwitchTrackColor = const Color(0xFFDCEAFF); +Color inactiveSwitchTrackColor = const Color(0xFFE0E0E0); +Color unfilledGradientColor = const Color(0xFFC4F3CF); +Color sharedUserContainerColor = const Color(0xffFBFBFB); +Color shareCardRejectButtonColor = const Color(0xffDC3545); +Color categoryChipViewcardColor = const Color(0xFFF1F1F1); +Color onBoardingHeadLineColor = const Color(0xff292929); +Color inactiveTrackColor = const Color(0xFFD9D9D9); +Color inactiveVoteColor = const Color(0xff80806B); +Color disabledButtonColor = const Color(0xff007BFF); +Color totalScansProfileViewColor = const Color(0xffF8F8F8); +Color subscribeTableColor = const Color(0xFFEDEDED); +Color crownBackgroundColor = const Color(0xffEAEAEA); +Color photoClickedCount = const Color(0xFF007BFF); +Color cameraScreenIconBackgroundColor = const Color(0xFF252525); +Color onboardCardBackGroundColor = const Color(0xFFEEEEEE); + +List filled_0_25 = const [Color(0xff39904D), Color(0xff28A745)]; +List unfilled_0_25 = const [Color(0xFFC4F3CF), Color(0xFFC4F3CF)]; +List filled_26_50 = const [Color(0xffF8CE51), Color(0xffF8CE51)]; +List unfilled_26_50 = const [Color(0xffDACCA2), Color(0xffDACCA2)]; +List filled_51_75 = const [Color(0xffF8CE51), Color(0xffFFA500)]; +List unfilled_51_75 = const [Color(0xffF5F5F5), Color(0xffF5F5F5)]; +List filled_76_100 = const [ + Color(0xffF8CE51), + Color(0xffF8CE51), + Color(0xffFFA501), + Color(0xffFF0000), +]; +List unfilled_76_100 = const [Color(0xffF5F5F5), Color(0xffF5F5F5)]; + +Gradient draftGradient = const LinearGradient( + begin: Alignment(1.00, 0.00), + end: Alignment(1, 0), + colors: [Color(0xFFF44336), Color(0xFFF1754E)], +); + +Gradient appTutorialGradient = const LinearGradient( + colors: [Color(0xff4E9BEE), Color(0xff2D5988)], + begin: Alignment.topCenter, + end: Alignment.bottomCenter, +); + +Gradient profileGradient = const LinearGradient( + begin: Alignment(-1, 0), + end: Alignment(1.00, 0.00), + colors : [Color(0xffD7E7F9),Color(0xff9BCBFF)] +); + +Gradient gradient0_25 = const LinearGradient( + begin: Alignment(1.00, 0.00), + end: Alignment(-1, 0), + colors: [Color(0xFF50CC6C), Color(0xFF28A745)], +); + +Gradient gradient26_50 = const LinearGradient( + begin: Alignment(1.00, 0.00), + end: Alignment(-2, -1), + colors: [Color(0xFFFFEB3B), Color(0xFFFF9800)], +); + +Gradient gradient51_75 = const LinearGradient( + begin: Alignment(-2, 1.5), + end: Alignment(0.75, 2), + colors: [Color(0xFFFF5722), Color(0xFFD2913B)], +); + +Gradient gradient76_100 = const LinearGradient( + begin: Alignment(-3, 1.5), + end: Alignment(0.75, 2), + colors: [Color(0xFFF44336), Color(0xFFF1754E)], +); + +Gradient subscriptionGradient = const LinearGradient( + begin: Alignment.topLeft, + end: Alignment.bottomRight, + colors: [ + Color(0xFF007BFF), + Color(0xFF004A99), + ], +); + +Gradient premiumBannerGradient = const LinearGradient( + colors: [Color(0xFFFFD700), Color(0xFF998100)], + begin: Alignment.topLeft, + end: Alignment.bottomRight, +); diff --git a/flutter_application_1/linux/.gitignore b/flutter_application_1/linux/.gitignore new file mode 100644 index 000000000..d3896c984 --- /dev/null +++ b/flutter_application_1/linux/.gitignore @@ -0,0 +1 @@ +flutter/ephemeral diff --git a/flutter_application_1/linux/CMakeLists.txt b/flutter_application_1/linux/CMakeLists.txt new file mode 100644 index 000000000..900e42173 --- /dev/null +++ b/flutter_application_1/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 "flutter_application_1") +# The unique GTK application identifier for this application. See: +# https://wiki.gnome.org/HowDoI/ChooseApplicationID +set(APPLICATION_ID "com.example.flutter_application_1") + +# 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/flutter_application_1/linux/flutter/CMakeLists.txt b/flutter_application_1/linux/flutter/CMakeLists.txt new file mode 100644 index 000000000..d5bd01648 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/linux/flutter/generated_plugin_registrant.cc b/flutter_application_1/linux/flutter/generated_plugin_registrant.cc new file mode 100644 index 000000000..64a0ecea4 --- /dev/null +++ b/flutter_application_1/linux/flutter/generated_plugin_registrant.cc @@ -0,0 +1,15 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#include "generated_plugin_registrant.h" + +#include + +void fl_register_plugins(FlPluginRegistry* registry) { + g_autoptr(FlPluginRegistrar) file_selector_linux_registrar = + fl_plugin_registry_get_registrar_for_plugin(registry, "FileSelectorPlugin"); + file_selector_plugin_register_with_registrar(file_selector_linux_registrar); +} diff --git a/flutter_application_1/linux/flutter/generated_plugin_registrant.h b/flutter_application_1/linux/flutter/generated_plugin_registrant.h new file mode 100644 index 000000000..e0f0a47bc --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/linux/flutter/generated_plugins.cmake b/flutter_application_1/linux/flutter/generated_plugins.cmake new file mode 100644 index 000000000..2db3c22ae --- /dev/null +++ b/flutter_application_1/linux/flutter/generated_plugins.cmake @@ -0,0 +1,24 @@ +# +# Generated file, do not edit. +# + +list(APPEND FLUTTER_PLUGIN_LIST + file_selector_linux +) + +list(APPEND FLUTTER_FFI_PLUGIN_LIST +) + +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/flutter_application_1/linux/runner/CMakeLists.txt b/flutter_application_1/linux/runner/CMakeLists.txt new file mode 100644 index 000000000..e97dabc70 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/linux/runner/main.cc b/flutter_application_1/linux/runner/main.cc new file mode 100644 index 000000000..e7c5c5437 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/linux/runner/my_application.cc b/flutter_application_1/linux/runner/my_application.cc new file mode 100644 index 000000000..98b22734c --- /dev/null +++ b/flutter_application_1/linux/runner/my_application.cc @@ -0,0 +1,130 @@ +#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) + +// 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, "flutter_application_1"); + 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, "flutter_application_1"); + } + + gtk_window_set_default_size(window, 1280, 720); + gtk_widget_show(GTK_WIDGET(window)); + + 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); + gtk_widget_show(GTK_WIDGET(view)); + gtk_container_add(GTK_CONTAINER(window), 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/flutter_application_1/linux/runner/my_application.h b/flutter_application_1/linux/runner/my_application.h new file mode 100644 index 000000000..72271d5e4 --- /dev/null +++ b/flutter_application_1/linux/runner/my_application.h @@ -0,0 +1,18 @@ +#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/flutter_application_1/macos/.gitignore b/flutter_application_1/macos/.gitignore new file mode 100644 index 000000000..746adbb6b --- /dev/null +++ b/flutter_application_1/macos/.gitignore @@ -0,0 +1,7 @@ +# Flutter-related +**/Flutter/ephemeral/ +**/Pods/ + +# Xcode-related +**/dgph +**/xcuserdata/ diff --git a/flutter_application_1/macos/Flutter/Flutter-Debug.xcconfig b/flutter_application_1/macos/Flutter/Flutter-Debug.xcconfig new file mode 100644 index 000000000..c2efd0b60 --- /dev/null +++ b/flutter_application_1/macos/Flutter/Flutter-Debug.xcconfig @@ -0,0 +1 @@ +#include "ephemeral/Flutter-Generated.xcconfig" diff --git a/flutter_application_1/macos/Flutter/Flutter-Release.xcconfig b/flutter_application_1/macos/Flutter/Flutter-Release.xcconfig new file mode 100644 index 000000000..c2efd0b60 --- /dev/null +++ b/flutter_application_1/macos/Flutter/Flutter-Release.xcconfig @@ -0,0 +1 @@ +#include "ephemeral/Flutter-Generated.xcconfig" diff --git a/flutter_application_1/macos/Flutter/GeneratedPluginRegistrant.swift b/flutter_application_1/macos/Flutter/GeneratedPluginRegistrant.swift new file mode 100644 index 000000000..90c0b335e --- /dev/null +++ b/flutter_application_1/macos/Flutter/GeneratedPluginRegistrant.swift @@ -0,0 +1,24 @@ +// +// Generated file. Do not edit. +// + +import FlutterMacOS +import Foundation + +import file_picker +import file_selector_macos +import firebase_auth +import firebase_core +import firebase_messaging +import flutter_local_notifications +import google_sign_in_ios + +func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { + FilePickerPlugin.register(with: registry.registrar(forPlugin: "FilePickerPlugin")) + FileSelectorPlugin.register(with: registry.registrar(forPlugin: "FileSelectorPlugin")) + FLTFirebaseAuthPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseAuthPlugin")) + FLTFirebaseCorePlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseCorePlugin")) + FLTFirebaseMessagingPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseMessagingPlugin")) + FlutterLocalNotificationsPlugin.register(with: registry.registrar(forPlugin: "FlutterLocalNotificationsPlugin")) + FLTGoogleSignInPlugin.register(with: registry.registrar(forPlugin: "FLTGoogleSignInPlugin")) +} diff --git a/flutter_application_1/macos/Runner.xcodeproj/project.pbxproj b/flutter_application_1/macos/Runner.xcodeproj/project.pbxproj new file mode 100644 index 000000000..2e7fb957e --- /dev/null +++ b/flutter_application_1/macos/Runner.xcodeproj/project.pbxproj @@ -0,0 +1,705 @@ +// !$*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 */; }; +/* 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 /* flutter_application_1.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = "flutter_application_1.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 = ""; }; + 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 = ( + ); + 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 /* flutter_application_1.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 = ( + 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; + productName = Runner; + productReference = 33CC10ED2044A3C60003C045 /* flutter_application_1.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; + 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.example.flutterApplication1.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/flutter_application_1.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/flutter_application_1"; + }; + 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.example.flutterApplication1.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/flutter_application_1.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/flutter_application_1"; + }; + 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.example.flutterApplication1.RunnerTests; + PRODUCT_NAME = "$(TARGET_NAME)"; + SWIFT_VERSION = 5.0; + TEST_HOST = "$(BUILT_PRODUCTS_DIR)/flutter_application_1.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/flutter_application_1"; + }; + 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.14; + 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.14; + 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.14; + 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 */ + }; + rootObject = 33CC10E52044A3C60003C045 /* Project object */; +} diff --git a/flutter_application_1/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/flutter_application_1/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 000000000..18d981003 --- /dev/null +++ b/flutter_application_1/macos/Runner.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/flutter_application_1/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/flutter_application_1/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme new file mode 100644 index 000000000..369e63264 --- /dev/null +++ b/flutter_application_1/macos/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -0,0 +1,98 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/macos/Runner.xcworkspace/contents.xcworkspacedata b/flutter_application_1/macos/Runner.xcworkspace/contents.xcworkspacedata new file mode 100644 index 000000000..1d526a16e --- /dev/null +++ b/flutter_application_1/macos/Runner.xcworkspace/contents.xcworkspacedata @@ -0,0 +1,7 @@ + + + + + diff --git a/flutter_application_1/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/flutter_application_1/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist new file mode 100644 index 000000000..18d981003 --- /dev/null +++ b/flutter_application_1/macos/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -0,0 +1,8 @@ + + + + + IDEDidComputeMac32BitWarning + + + diff --git a/flutter_application_1/macos/Runner/AppDelegate.swift b/flutter_application_1/macos/Runner/AppDelegate.swift new file mode 100644 index 000000000..b3c176141 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 000000000..a2ec33f19 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png new file mode 100644 index 000000000..82b6f9d9a Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_1024.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png new file mode 100644 index 000000000..13b35eba5 Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_128.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png new file mode 100644 index 000000000..0a3f5fa40 Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_16.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png new file mode 100644 index 000000000..bdb57226d Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_256.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png new file mode 100644 index 000000000..f083318e0 Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_32.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png new file mode 100644 index 000000000..326c0e72c Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_512.png differ diff --git a/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png new file mode 100644 index 000000000..2f1632cfd Binary files /dev/null and b/flutter_application_1/macos/Runner/Assets.xcassets/AppIcon.appiconset/app_icon_64.png differ diff --git a/flutter_application_1/macos/Runner/Base.lproj/MainMenu.xib b/flutter_application_1/macos/Runner/Base.lproj/MainMenu.xib new file mode 100644 index 000000000..80e867a4e --- /dev/null +++ b/flutter_application_1/macos/Runner/Base.lproj/MainMenu.xib @@ -0,0 +1,343 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/flutter_application_1/macos/Runner/Configs/AppInfo.xcconfig b/flutter_application_1/macos/Runner/Configs/AppInfo.xcconfig new file mode 100644 index 000000000..e5b4e71e8 --- /dev/null +++ b/flutter_application_1/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 = flutter_application_1 + +// The application's bundle identifier +PRODUCT_BUNDLE_IDENTIFIER = com.example.flutterApplication1 + +// The copyright displayed in application information +PRODUCT_COPYRIGHT = Copyright © 2025 com.example. All rights reserved. diff --git a/flutter_application_1/macos/Runner/Configs/Debug.xcconfig b/flutter_application_1/macos/Runner/Configs/Debug.xcconfig new file mode 100644 index 000000000..36b0fd946 --- /dev/null +++ b/flutter_application_1/macos/Runner/Configs/Debug.xcconfig @@ -0,0 +1,2 @@ +#include "../../Flutter/Flutter-Debug.xcconfig" +#include "Warnings.xcconfig" diff --git a/flutter_application_1/macos/Runner/Configs/Release.xcconfig b/flutter_application_1/macos/Runner/Configs/Release.xcconfig new file mode 100644 index 000000000..dff4f4956 --- /dev/null +++ b/flutter_application_1/macos/Runner/Configs/Release.xcconfig @@ -0,0 +1,2 @@ +#include "../../Flutter/Flutter-Release.xcconfig" +#include "Warnings.xcconfig" diff --git a/flutter_application_1/macos/Runner/Configs/Warnings.xcconfig b/flutter_application_1/macos/Runner/Configs/Warnings.xcconfig new file mode 100644 index 000000000..42bcbf478 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/DebugProfile.entitlements b/flutter_application_1/macos/Runner/DebugProfile.entitlements new file mode 100644 index 000000000..dddb8a30c --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/Info.plist b/flutter_application_1/macos/Runner/Info.plist new file mode 100644 index 000000000..4789daa6a --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/MainFlutterWindow.swift b/flutter_application_1/macos/Runner/MainFlutterWindow.swift new file mode 100644 index 000000000..3cc05eb23 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/macos/Runner/Release.entitlements b/flutter_application_1/macos/Runner/Release.entitlements new file mode 100644 index 000000000..852fa1a47 --- /dev/null +++ b/flutter_application_1/macos/Runner/Release.entitlements @@ -0,0 +1,8 @@ + + + + + com.apple.security.app-sandbox + + + diff --git a/flutter_application_1/macos/RunnerTests/RunnerTests.swift b/flutter_application_1/macos/RunnerTests/RunnerTests.swift new file mode 100644 index 000000000..61f3bd1fc --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/pubspec.lock b/flutter_application_1/pubspec.lock new file mode 100644 index 000000000..5a87c34eb --- /dev/null +++ b/flutter_application_1/pubspec.lock @@ -0,0 +1,666 @@ +# Generated by pub +# See https://dart.dev/tools/pub/glossary#lockfile +packages: + _flutterfire_internals: + dependency: transitive + description: + name: _flutterfire_internals + sha256: "7fd72d77a7487c26faab1d274af23fb008763ddc10800261abbfb2c067f183d5" + url: "https://pub.dev" + source: hosted + version: "1.3.53" + args: + dependency: transitive + description: + name: args + sha256: bf9f5caeea8d8fe6721a9c358dd8a5c1947b27f1cfaa18b39c301273594919e6 + url: "https://pub.dev" + source: hosted + version: "2.6.0" + async: + dependency: transitive + description: + name: async + sha256: d2872f9c19731c2e5f10444b14686eb7cc85c76274bd6c16e1816bff9a3bab63 + url: "https://pub.dev" + source: hosted + version: "2.12.0" + bloc: + dependency: "direct main" + description: + name: bloc + sha256: "52c10575f4445c61dd9e0cafcc6356fdd827c4c64dd7945ef3c4105f6b6ac189" + url: "https://pub.dev" + source: hosted + version: "9.0.0" + boolean_selector: + dependency: transitive + description: + name: boolean_selector + sha256: "8aab1771e1243a5063b8b0ff68042d67334e3feab9e95b9490f9a6ebf73b42ea" + url: "https://pub.dev" + source: hosted + version: "2.1.2" + characters: + dependency: transitive + description: + name: characters + sha256: f71061c654a3380576a52b451dd5532377954cf9dbd272a78fc8479606670803 + url: "https://pub.dev" + source: hosted + version: "1.4.0" + clock: + dependency: transitive + description: + name: clock + sha256: fddb70d9b5277016c77a80201021d40a2247104d9f4aa7bab7157b7e3f05b84b + url: "https://pub.dev" + source: hosted + version: "1.1.2" + collection: + dependency: transitive + description: + name: collection + sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" + url: "https://pub.dev" + source: hosted + version: "1.19.1" + cross_file: + dependency: transitive + description: + name: cross_file + sha256: "7caf6a750a0c04effbb52a676dce9a4a592e10ad35c34d6d2d0e4811160d5670" + url: "https://pub.dev" + source: hosted + version: "0.3.4+2" + cupertino_icons: + dependency: "direct main" + description: + name: cupertino_icons + sha256: ba631d1c7f7bef6b729a622b7b752645a2d076dba9976925b8f25725a30e1ee6 + url: "https://pub.dev" + source: hosted + version: "1.0.8" + dbus: + dependency: transitive + description: + name: dbus + sha256: "79e0c23480ff85dc68de79e2cd6334add97e48f7f4865d17686dd6ea81a47e8c" + url: "https://pub.dev" + source: hosted + version: "0.7.11" + equatable: + dependency: transitive + description: + name: equatable + sha256: "567c64b3cb4cf82397aac55f4f0cbd3ca20d77c6c03bedbc4ceaddc08904aef7" + url: "https://pub.dev" + source: hosted + version: "2.0.7" + fake_async: + dependency: transitive + description: + name: fake_async + sha256: "6a95e56b2449df2273fd8c45a662d6947ce1ebb7aafe80e550a3f68297f3cacc" + url: "https://pub.dev" + source: hosted + version: "1.3.2" + ffi: + dependency: transitive + description: + name: ffi + sha256: "289279317b4b16eb2bb7e271abccd4bf84ec9bdcbe999e278a94b804f5630418" + url: "https://pub.dev" + source: hosted + version: "2.1.4" + file_picker: + dependency: "direct main" + description: + name: file_picker + sha256: "7423298f08f6fc8cce05792bae329f9a93653fc9c08712831b1a55540127995d" + url: "https://pub.dev" + source: hosted + version: "9.0.2" + file_selector_linux: + dependency: transitive + description: + name: file_selector_linux + sha256: "54cbbd957e1156d29548c7d9b9ec0c0ebb6de0a90452198683a7d23aed617a33" + url: "https://pub.dev" + source: hosted + version: "0.9.3+2" + file_selector_macos: + dependency: transitive + description: + name: file_selector_macos + sha256: "271ab9986df0c135d45c3cdb6bd0faa5db6f4976d3e4b437cf7d0f258d941bfc" + url: "https://pub.dev" + source: hosted + version: "0.9.4+2" + file_selector_platform_interface: + dependency: transitive + description: + name: file_selector_platform_interface + sha256: a3994c26f10378a039faa11de174d7b78eb8f79e4dd0af2a451410c1a5c3f66b + url: "https://pub.dev" + source: hosted + version: "2.6.2" + file_selector_windows: + dependency: transitive + description: + name: file_selector_windows + sha256: "320fcfb6f33caa90f0b58380489fc5ac05d99ee94b61aa96ec2bff0ba81d3c2b" + url: "https://pub.dev" + source: hosted + version: "0.9.3+4" + firebase_auth: + dependency: "direct main" + description: + name: firebase_auth + sha256: "91587615d7d9165c65a030426e3cf40bbec37c486f52ff654af17aba5be3d208" + url: "https://pub.dev" + source: hosted + version: "5.5.1" + firebase_auth_platform_interface: + dependency: transitive + description: + name: firebase_auth_platform_interface + sha256: "1dcf1dbdd90fe97fa37ab3631b561bf584adb88f6be0b0dd915fff799ad53192" + url: "https://pub.dev" + source: hosted + version: "7.6.1" + firebase_auth_web: + dependency: transitive + description: + name: firebase_auth_web + sha256: "3774cb13547e28b180fed2a5e696b4b36f97f4b1fadc7b04a0200e5009344d98" + url: "https://pub.dev" + source: hosted + version: "5.14.1" + firebase_core: + dependency: "direct main" + description: + name: firebase_core + sha256: f4d8f49574a4e396f34567f3eec4d38ab9c3910818dec22ca42b2a467c685d8b + url: "https://pub.dev" + source: hosted + version: "3.12.1" + firebase_core_platform_interface: + dependency: transitive + description: + name: firebase_core_platform_interface + sha256: d7253d255ff10f85cfd2adaba9ac17bae878fa3ba577462451163bd9f1d1f0bf + url: "https://pub.dev" + source: hosted + version: "5.4.0" + firebase_core_web: + dependency: transitive + description: + name: firebase_core_web + sha256: faa5a76f6380a9b90b53bc3bdcb85bc7926a382e0709b9b5edac9f7746651493 + url: "https://pub.dev" + source: hosted + version: "2.21.1" + firebase_messaging: + dependency: "direct main" + description: + name: firebase_messaging + sha256: "5fc345c6341f9dc69fd0ffcbf508c784fd6d1b9e9f249587f30434dd8b6aa281" + url: "https://pub.dev" + source: hosted + version: "15.2.4" + firebase_messaging_platform_interface: + dependency: transitive + description: + name: firebase_messaging_platform_interface + sha256: a935924cf40925985c8049df4968b1dde5c704f570f3ce380b31d3de6990dd94 + url: "https://pub.dev" + source: hosted + version: "4.6.4" + firebase_messaging_web: + dependency: transitive + description: + name: firebase_messaging_web + sha256: fafebf6a1921931334f3f10edb5037a5712288efdd022881e2d093e5654a2fd4 + url: "https://pub.dev" + source: hosted + version: "3.10.4" + fl_chart: + dependency: "direct main" + description: + name: fl_chart + sha256: "5276944c6ffc975ae796569a826c38a62d2abcf264e26b88fa6f482e107f4237" + url: "https://pub.dev" + source: hosted + version: "0.70.2" + flutter: + dependency: "direct main" + description: flutter + source: sdk + version: "0.0.0" + flutter_bloc: + dependency: "direct main" + description: + name: flutter_bloc + sha256: "1046d719fbdf230330d3443187cc33cc11963d15c9089f6cc56faa42a4c5f0cc" + url: "https://pub.dev" + source: hosted + version: "9.1.0" + flutter_lints: + dependency: "direct dev" + description: + name: flutter_lints + sha256: "5398f14efa795ffb7a33e9b6a08798b26a180edac4ad7db3f231e40f82ce11e1" + url: "https://pub.dev" + source: hosted + version: "5.0.0" + flutter_local_notifications: + dependency: "direct main" + description: + name: flutter_local_notifications + sha256: ef41ae901e7529e52934feba19ed82827b11baa67336829564aeab3129460610 + url: "https://pub.dev" + source: hosted + version: "18.0.1" + flutter_local_notifications_linux: + dependency: transitive + description: + name: flutter_local_notifications_linux + sha256: "8f685642876742c941b29c32030f6f4f6dacd0e4eaecb3efbb187d6a3812ca01" + url: "https://pub.dev" + source: hosted + version: "5.0.0" + flutter_local_notifications_platform_interface: + dependency: transitive + description: + name: flutter_local_notifications_platform_interface + sha256: "6c5b83c86bf819cdb177a9247a3722067dd8cc6313827ce7c77a4b238a26fd52" + url: "https://pub.dev" + source: hosted + version: "8.0.0" + flutter_plugin_android_lifecycle: + dependency: transitive + description: + name: flutter_plugin_android_lifecycle + sha256: "5a1e6fb2c0561958d7e4c33574674bda7b77caaca7a33b758876956f2902eea3" + url: "https://pub.dev" + source: hosted + version: "2.0.27" + flutter_spinkit: + dependency: "direct main" + description: + name: flutter_spinkit + sha256: d2696eed13732831414595b98863260e33e8882fc069ee80ec35d4ac9ddb0472 + url: "https://pub.dev" + source: hosted + version: "5.2.1" + 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" + google_identity_services_web: + dependency: transitive + description: + name: google_identity_services_web + sha256: "55580f436822d64c8ff9a77e37d61f5fb1e6c7ec9d632a43ee324e2a05c3c6c9" + url: "https://pub.dev" + source: hosted + version: "0.3.3" + google_sign_in: + dependency: "direct main" + description: + name: google_sign_in + sha256: d0a2c3bcb06e607bb11e4daca48bd4b6120f0bbc4015ccebbe757d24ea60ed2a + url: "https://pub.dev" + source: hosted + version: "6.3.0" + google_sign_in_android: + dependency: transitive + description: + name: google_sign_in_android + sha256: "4e52c64366bdb3fe758f683b088ee514cc7a95e69c52b5ee9fc5919e1683d21b" + url: "https://pub.dev" + source: hosted + version: "6.2.0" + google_sign_in_ios: + dependency: transitive + description: + name: google_sign_in_ios + sha256: "29cd125f58f50ceb40e8253d3c0209e321eee3e5df16cd6d262495f7cad6a2bd" + url: "https://pub.dev" + source: hosted + version: "5.8.1" + google_sign_in_platform_interface: + dependency: transitive + description: + name: google_sign_in_platform_interface + sha256: "5f6f79cf139c197261adb6ac024577518ae48fdff8e53205c5373b5f6430a8aa" + url: "https://pub.dev" + source: hosted + version: "2.5.0" + google_sign_in_web: + dependency: transitive + description: + name: google_sign_in_web + sha256: "460547beb4962b7623ac0fb8122d6b8268c951cf0b646dd150d60498430e4ded" + url: "https://pub.dev" + source: hosted + version: "0.12.4+4" + http: + dependency: "direct main" + description: + name: http + sha256: fe7ab022b76f3034adc518fb6ea04a82387620e19977665ea18d30a1cf43442f + url: "https://pub.dev" + source: hosted + version: "1.3.0" + http_parser: + dependency: transitive + description: + name: http_parser + sha256: "178d74305e7866013777bab2c3d8726205dc5a4dd935297175b19a23a2e66571" + url: "https://pub.dev" + source: hosted + version: "4.1.2" + image_picker: + dependency: "direct main" + description: + name: image_picker + sha256: "021834d9c0c3de46bf0fe40341fa07168407f694d9b2bb18d532dc1261867f7a" + url: "https://pub.dev" + source: hosted + version: "1.1.2" + image_picker_android: + dependency: transitive + description: + name: image_picker_android + sha256: "8bd392ba8b0c8957a157ae0dc9fcf48c58e6c20908d5880aea1d79734df090e9" + url: "https://pub.dev" + source: hosted + version: "0.8.12+22" + image_picker_for_web: + dependency: transitive + description: + name: image_picker_for_web + sha256: "717eb042ab08c40767684327be06a5d8dbb341fe791d514e4b92c7bbe1b7bb83" + url: "https://pub.dev" + source: hosted + version: "3.0.6" + image_picker_ios: + dependency: transitive + description: + name: image_picker_ios + sha256: "05da758e67bc7839e886b3959848aa6b44ff123ab4b28f67891008afe8ef9100" + url: "https://pub.dev" + source: hosted + version: "0.8.12+2" + image_picker_linux: + dependency: transitive + description: + name: image_picker_linux + sha256: "4ed1d9bb36f7cd60aa6e6cd479779cc56a4cb4e4de8f49d487b1aaad831300fa" + url: "https://pub.dev" + source: hosted + version: "0.2.1+1" + image_picker_macos: + dependency: transitive + description: + name: image_picker_macos + sha256: "1b90ebbd9dcf98fb6c1d01427e49a55bd96b5d67b8c67cf955d60a5de74207c1" + url: "https://pub.dev" + source: hosted + version: "0.2.1+2" + image_picker_platform_interface: + dependency: transitive + description: + name: image_picker_platform_interface + sha256: "886d57f0be73c4b140004e78b9f28a8914a09e50c2d816bdd0520051a71236a0" + url: "https://pub.dev" + source: hosted + version: "2.10.1" + image_picker_windows: + dependency: transitive + description: + name: image_picker_windows + sha256: "6ad07afc4eb1bc25f3a01084d28520496c4a3bb0cb13685435838167c9dcedeb" + url: "https://pub.dev" + source: hosted + version: "0.2.1+1" + intl: + dependency: "direct main" + description: + name: intl + sha256: "3df61194eb431efc39c4ceba583b95633a403f46c9fd341e550ce0bfa50e9aa5" + url: "https://pub.dev" + source: hosted + version: "0.20.2" + leak_tracker: + dependency: transitive + description: + name: leak_tracker + sha256: c35baad643ba394b40aac41080300150a4f08fd0fd6a10378f8f7c6bc161acec + url: "https://pub.dev" + source: hosted + version: "10.0.8" + leak_tracker_flutter_testing: + dependency: transitive + description: + name: leak_tracker_flutter_testing + sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 + url: "https://pub.dev" + source: hosted + version: "3.0.9" + leak_tracker_testing: + dependency: transitive + description: + name: leak_tracker_testing + sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" + url: "https://pub.dev" + source: hosted + version: "3.0.1" + lints: + dependency: transitive + description: + name: lints + sha256: c35bb79562d980e9a453fc715854e1ed39e24e7d0297a880ef54e17f9874a9d7 + url: "https://pub.dev" + source: hosted + version: "5.1.1" + matcher: + dependency: transitive + description: + name: matcher + sha256: dc58c723c3c24bf8d3e2d3ad3f2f9d7bd9cf43ec6feaa64181775e60190153f2 + url: "https://pub.dev" + source: hosted + version: "0.12.17" + material_color_utilities: + dependency: transitive + description: + name: material_color_utilities + sha256: f7142bb1154231d7ea5f96bc7bde4bda2a0945d2806bb11670e30b850d56bdec + url: "https://pub.dev" + source: hosted + version: "0.11.1" + meta: + dependency: transitive + description: + name: meta + sha256: e3641ec5d63ebf0d9b41bd43201a66e3fc79a65db5f61fc181f04cd27aab950c + url: "https://pub.dev" + source: hosted + version: "1.16.0" + mime: + dependency: transitive + description: + name: mime + sha256: "41a20518f0cb1256669420fdba0cd90d21561e560ac240f26ef8322e45bb7ed6" + url: "https://pub.dev" + source: hosted + version: "2.0.0" + nested: + dependency: transitive + description: + name: nested + sha256: "03bac4c528c64c95c722ec99280375a6f2fc708eec17c7b3f07253b626cd2a20" + url: "https://pub.dev" + source: hosted + version: "1.0.0" + path: + dependency: transitive + description: + name: path + sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" + url: "https://pub.dev" + source: hosted + version: "1.9.1" + petitparser: + dependency: transitive + description: + name: petitparser + sha256: "07c8f0b1913bcde1ff0d26e57ace2f3012ccbf2b204e070290dad3bb22797646" + url: "https://pub.dev" + source: hosted + version: "6.1.0" + plugin_platform_interface: + dependency: transitive + description: + name: plugin_platform_interface + sha256: "4820fbfdb9478b1ebae27888254d445073732dae3d6ea81f0b7e06d5dedc3f02" + url: "https://pub.dev" + source: hosted + version: "2.1.8" + provider: + dependency: transitive + description: + name: provider + sha256: c8a055ee5ce3fd98d6fc872478b03823ffdb448699c6ebdbbc71d59b596fd48c + url: "https://pub.dev" + source: hosted + version: "6.1.2" + sky_engine: + dependency: transitive + description: flutter + source: sdk + version: "0.0.0" + source_span: + dependency: transitive + description: + name: source_span + sha256: "254ee5351d6cb365c859e20ee823c3bb479bf4a293c22d17a9f1bf144ce86f7c" + url: "https://pub.dev" + source: hosted + version: "1.10.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" + 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: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd + url: "https://pub.dev" + source: hosted + version: "0.7.4" + timezone: + dependency: transitive + description: + name: timezone + sha256: ffc9d5f4d1193534ef051f9254063fa53d588609418c84299956c3db9383587d + url: "https://pub.dev" + source: hosted + version: "0.10.0" + typed_data: + dependency: transitive + description: + name: typed_data + sha256: f9049c039ebfeb4cf7a7104a675823cd72dba8297f264b6637062516699fa006 + url: "https://pub.dev" + source: hosted + version: "1.4.0" + vector_math: + dependency: transitive + description: + name: vector_math + sha256: "80b3257d1492ce4d091729e3a67a60407d227c27241d6927be0130c98e741803" + url: "https://pub.dev" + source: hosted + version: "2.1.4" + vm_service: + dependency: transitive + description: + name: vm_service + sha256: "0968250880a6c5fe7edc067ed0a13d4bae1577fe2771dcf3010d52c4a9d3ca14" + url: "https://pub.dev" + source: hosted + version: "14.3.1" + web: + dependency: transitive + description: + name: web + sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" + url: "https://pub.dev" + source: hosted + version: "1.1.1" + win32: + dependency: transitive + description: + name: win32 + sha256: b89e6e24d1454e149ab20fbb225af58660f0c0bf4475544650700d8e2da54aef + url: "https://pub.dev" + source: hosted + version: "5.11.0" + 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: b015a8ad1c488f66851d762d3090a21c600e479dc75e68328c52774040cf9226 + url: "https://pub.dev" + source: hosted + version: "6.5.0" +sdks: + dart: ">=3.7.0 <4.0.0" + flutter: ">=3.27.0" diff --git a/flutter_application_1/pubspec.yaml b/flutter_application_1/pubspec.yaml new file mode 100644 index 000000000..447ff877f --- /dev/null +++ b/flutter_application_1/pubspec.yaml @@ -0,0 +1,132 @@ +name: flutter_application_1 +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.6.1 + +# 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 + intl: ^0.20.2 + flutter_spinkit: ^5.2.1 + bloc: ^9.0.0 + fl_chart: ^0.70.2 + flutter_bloc: ^9.1.0 + image_picker: ^1.1.2 + http: ^1.3.0 + file_picker: ^9.0.2 + firebase_core: ^3.12.1 + firebase_auth: ^5.5.1 + firebase_messaging: ^15.2.4 + flutter_local_notifications: ^18.0.1 + google_sign_in: ^6.3.0 + +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: ^5.0.0 + +# For information on the generic Dart part of this file, see the +# following page: https://dart.dev/tools/pub/pubspec + +# The following section is specific to Flutter 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 + assets: + - assets/images/ + + # 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: black + fonts: + - asset: assets/fonts/Satoshi-Black.ttf + + - family: blackItalic + fonts: + - asset: assets/fonts/Satoshi-BlackItalic.ttf + + - family: bold + fonts: + - asset: assets/fonts/Satoshi-Bold.ttf + + - family: boldItalic + fonts: + - asset: assets/fonts/Satoshi-BoldItalic.ttf + + - family: medium + fonts: + - asset: assets/fonts/Satoshi-Medium.ttf + + - family: mediumItalic + fonts: + - asset: assets/fonts/Satoshi-MediumItalic.ttf + + - family: regular + fonts: + - asset: assets/fonts/Satoshi-Regular.ttf + # 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/flutter_application_1/test/widget_test.dart b/flutter_application_1/test/widget_test.dart new file mode 100644 index 000000000..6cdbc9609 --- /dev/null +++ b/flutter_application_1/test/widget_test.dart @@ -0,0 +1,30 @@ +// // This is a basic Flutter widget test. +// // +// // To perform an interaction with a widget in your test, use the WidgetTester +// // utility in the flutter_test package. For example, you can send tap and scroll +// // gestures. You can also use WidgetTester to find child widgets in the widget +// // tree, read text, and verify that the values of widget properties are correct. +// +// import 'package:flutter/material.dart'; +// import 'package:flutter_test/flutter_test.dart'; +// +// import 'package:flutter_application_1/main.dart'; +// +// void main() { +// testWidgets('Counter increments smoke test', (WidgetTester tester) async { +// // Build our app and trigger a frame. +// await tester.pumpWidget(const MyApp()); +// +// // Verify that our counter starts at 0. +// expect(find.text('0'), findsOneWidget); +// expect(find.text('1'), findsNothing); +// +// // Tap the '+' icon and trigger a frame. +// await tester.tap(find.byIcon(Icons.add)); +// await tester.pump(); +// +// // Verify that our counter has incremented. +// expect(find.text('0'), findsNothing); +// expect(find.text('1'), findsOneWidget); +// }); +// } diff --git a/flutter_application_1/web/favicon.png b/flutter_application_1/web/favicon.png new file mode 100644 index 000000000..8aaa46ac1 Binary files /dev/null and b/flutter_application_1/web/favicon.png differ diff --git a/flutter_application_1/web/icons/Icon-192.png b/flutter_application_1/web/icons/Icon-192.png new file mode 100644 index 000000000..b749bfef0 Binary files /dev/null and b/flutter_application_1/web/icons/Icon-192.png differ diff --git a/flutter_application_1/web/icons/Icon-512.png b/flutter_application_1/web/icons/Icon-512.png new file mode 100644 index 000000000..88cfd48df Binary files /dev/null and b/flutter_application_1/web/icons/Icon-512.png differ diff --git a/flutter_application_1/web/icons/Icon-maskable-192.png b/flutter_application_1/web/icons/Icon-maskable-192.png new file mode 100644 index 000000000..eb9b4d76e Binary files /dev/null and b/flutter_application_1/web/icons/Icon-maskable-192.png differ diff --git a/flutter_application_1/web/icons/Icon-maskable-512.png b/flutter_application_1/web/icons/Icon-maskable-512.png new file mode 100644 index 000000000..d69c56691 Binary files /dev/null and b/flutter_application_1/web/icons/Icon-maskable-512.png differ diff --git a/flutter_application_1/web/index.html b/flutter_application_1/web/index.html new file mode 100644 index 000000000..34a4134c3 --- /dev/null +++ b/flutter_application_1/web/index.html @@ -0,0 +1,38 @@ + + + + + + + + + + + + + + + + + + + + flutter_application_1 + + + + + + diff --git a/flutter_application_1/web/manifest.json b/flutter_application_1/web/manifest.json new file mode 100644 index 000000000..dd8e066f4 --- /dev/null +++ b/flutter_application_1/web/manifest.json @@ -0,0 +1,35 @@ +{ + "name": "flutter_application_1", + "short_name": "flutter_application_1", + "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/flutter_application_1/windows/.gitignore b/flutter_application_1/windows/.gitignore new file mode 100644 index 000000000..d492d0d98 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/CMakeLists.txt b/flutter_application_1/windows/CMakeLists.txt new file mode 100644 index 000000000..8cc860b25 --- /dev/null +++ b/flutter_application_1/windows/CMakeLists.txt @@ -0,0 +1,108 @@ +# Project-level configuration. +cmake_minimum_required(VERSION 3.14) +project(flutter_application_1 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 "flutter_application_1") + +# 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/flutter_application_1/windows/flutter/CMakeLists.txt b/flutter_application_1/windows/flutter/CMakeLists.txt new file mode 100644 index 000000000..903f4899d --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/flutter/generated_plugin_registrant.cc b/flutter_application_1/windows/flutter/generated_plugin_registrant.cc new file mode 100644 index 000000000..5861e0f0e --- /dev/null +++ b/flutter_application_1/windows/flutter/generated_plugin_registrant.cc @@ -0,0 +1,20 @@ +// +// Generated file. Do not edit. +// + +// clang-format off + +#include "generated_plugin_registrant.h" + +#include +#include +#include + +void RegisterPlugins(flutter::PluginRegistry* registry) { + FileSelectorWindowsRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FileSelectorWindows")); + FirebaseAuthPluginCApiRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FirebaseAuthPluginCApi")); + FirebaseCorePluginCApiRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FirebaseCorePluginCApi")); +} diff --git a/flutter_application_1/windows/flutter/generated_plugin_registrant.h b/flutter_application_1/windows/flutter/generated_plugin_registrant.h new file mode 100644 index 000000000..dc139d85a --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/flutter/generated_plugins.cmake b/flutter_application_1/windows/flutter/generated_plugins.cmake new file mode 100644 index 000000000..ce851e9de --- /dev/null +++ b/flutter_application_1/windows/flutter/generated_plugins.cmake @@ -0,0 +1,26 @@ +# +# Generated file, do not edit. +# + +list(APPEND FLUTTER_PLUGIN_LIST + file_selector_windows + firebase_auth + firebase_core +) + +list(APPEND FLUTTER_FFI_PLUGIN_LIST +) + +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/flutter_application_1/windows/runner/CMakeLists.txt b/flutter_application_1/windows/runner/CMakeLists.txt new file mode 100644 index 000000000..394917c05 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/Runner.rc b/flutter_application_1/windows/runner/Runner.rc new file mode 100644 index 000000000..765011b4f --- /dev/null +++ b/flutter_application_1/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.example" "\0" + VALUE "FileDescription", "flutter_application_1" "\0" + VALUE "FileVersion", VERSION_AS_STRING "\0" + VALUE "InternalName", "flutter_application_1" "\0" + VALUE "LegalCopyright", "Copyright (C) 2025 com.example. All rights reserved." "\0" + VALUE "OriginalFilename", "flutter_application_1.exe" "\0" + VALUE "ProductName", "flutter_application_1" "\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/flutter_application_1/windows/runner/flutter_window.cpp b/flutter_application_1/windows/runner/flutter_window.cpp new file mode 100644 index 000000000..955ee3038 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/flutter_window.h b/flutter_application_1/windows/runner/flutter_window.h new file mode 100644 index 000000000..6da0652f0 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/main.cpp b/flutter_application_1/windows/runner/main.cpp new file mode 100644 index 000000000..bb2306021 --- /dev/null +++ b/flutter_application_1/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"flutter_application_1", 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/flutter_application_1/windows/runner/resource.h b/flutter_application_1/windows/runner/resource.h new file mode 100644 index 000000000..66a65d1e4 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/resources/app_icon.ico b/flutter_application_1/windows/runner/resources/app_icon.ico new file mode 100644 index 000000000..c04e20caf Binary files /dev/null and b/flutter_application_1/windows/runner/resources/app_icon.ico differ diff --git a/flutter_application_1/windows/runner/runner.exe.manifest b/flutter_application_1/windows/runner/runner.exe.manifest new file mode 100644 index 000000000..153653e8d --- /dev/null +++ b/flutter_application_1/windows/runner/runner.exe.manifest @@ -0,0 +1,14 @@ + + + + + PerMonitorV2 + + + + + + + + + diff --git a/flutter_application_1/windows/runner/utils.cpp b/flutter_application_1/windows/runner/utils.cpp new file mode 100644 index 000000000..3a0b46511 --- /dev/null +++ b/flutter_application_1/windows/runner/utils.cpp @@ -0,0 +1,65 @@ +#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(); + } + unsigned int target_length = ::WideCharToMultiByte( + CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, + -1, nullptr, 0, nullptr, nullptr) + -1; // remove the trailing null character + int input_length = (int)wcslen(utf16_string); + std::string utf8_string; + if (target_length == 0 || 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/flutter_application_1/windows/runner/utils.h b/flutter_application_1/windows/runner/utils.h new file mode 100644 index 000000000..3879d5475 --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/win32_window.cpp b/flutter_application_1/windows/runner/win32_window.cpp new file mode 100644 index 000000000..60608d0fe --- /dev/null +++ b/flutter_application_1/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/flutter_application_1/windows/runner/win32_window.h b/flutter_application_1/windows/runner/win32_window.h new file mode 100644 index 000000000..e901dde68 --- /dev/null +++ b/flutter_application_1/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_