90
.gitignore
vendored
Normal file
@ -0,0 +1,90 @@
|
||||
# Miscellaneous
|
||||
*.class
|
||||
*.log
|
||||
*.pyc
|
||||
*.swp
|
||||
.DS_Store
|
||||
.atom/
|
||||
.buildlog/
|
||||
.history
|
||||
.svn/
|
||||
migrate_working_dir/
|
||||
|
||||
# IntelliJ related
|
||||
*.iml
|
||||
*.ipr
|
||||
*.iws
|
||||
.idea/
|
||||
|
||||
# The .vscode folder contains launch configuration and tasks you configure in
|
||||
# VS Code which you may wish to be included in version control, so this line
|
||||
# is commented out by default.
|
||||
#.vscode/
|
||||
|
||||
# Flutter/Dart/Pub related
|
||||
**/doc/api/
|
||||
**/ios/Flutter/.last_build_id
|
||||
.dart_tool/
|
||||
.flutter-plugins
|
||||
.flutter-plugins-dependencies
|
||||
.packages
|
||||
.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
|
||||
|
||||
# iOS/XCode related
|
||||
**/ios/**/*.mode1v3
|
||||
**/ios/**/*.mode2v3
|
||||
**/ios/**/*.moved-aside
|
||||
**/ios/**/*.pbxuser
|
||||
**/ios/**/*.perspectivev3
|
||||
**/ios/**/*sync/
|
||||
**/ios/**/.sconsign.dblite
|
||||
**/ios/**/.tags*
|
||||
**/ios/**/.vagrant/
|
||||
**/ios/**/DerivedData/
|
||||
**/ios/**/Icon?
|
||||
**/ios/**/Pods/
|
||||
**/ios/**/.symlinks/
|
||||
**/ios/**/profile
|
||||
**/ios/**/xcuserdata
|
||||
**/ios/.generated/
|
||||
**/ios/Flutter/.last_build_id
|
||||
**/ios/Flutter/App.framework
|
||||
**/ios/Flutter/Flutter.framework
|
||||
**/ios/Flutter/Flutter.podspec
|
||||
**/ios/Flutter/Generated.xcconfig
|
||||
**/ios/Flutter/ephemeral
|
||||
**/ios/Flutter/app.flx
|
||||
**/ios/Flutter/app.zip
|
||||
**/ios/Flutter/flutter_assets/
|
||||
**/ios/Flutter/flutter_export_environment.sh
|
||||
**/ios/ServiceDefinitions.json
|
||||
**/ios/Runner/GeneratedPluginRegistrant.*
|
||||
|
||||
# macOS
|
||||
**/Flutter/ephemeral/
|
||||
**/Pods/
|
||||
**/macos/Flutter/GeneratedPluginRegistrant.swift
|
||||
**/macos/Flutter/ephemeral
|
||||
**/xcuserdata/
|
||||
|
||||
# Exceptions to above rules.
|
||||
!**/ios/**/default.mode1v3
|
||||
!**/ios/**/default.mode2v3
|
||||
!**/ios/**/default.pbxuser
|
||||
!**/ios/**/default.perspectivev3
|
||||
!/packages/flutter_tools/test/data/dart_dependencies_test/**/.packages
|
||||
!/dev/ci/**/Gemfile.lock
|
||||
!.vscode/settings.json
|
33
.metadata
Normal file
@ -0,0 +1,33 @@
|
||||
# This file tracks properties of this Flutter project.
|
||||
# Used by Flutter tool to assess capabilities and perform upgrades etc.
|
||||
#
|
||||
# This file should be version controlled and should not be manually edited.
|
||||
|
||||
version:
|
||||
revision: "6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e"
|
||||
channel: "stable"
|
||||
|
||||
project_type: app
|
||||
|
||||
# Tracks metadata for the flutter migrate command
|
||||
migration:
|
||||
platforms:
|
||||
- platform: root
|
||||
create_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
base_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
- platform: android
|
||||
create_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
base_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
- platform: ios
|
||||
create_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
base_revision: 6c4930c4ac86fb286f30e31d0ec8bffbcbb9953e
|
||||
|
||||
# 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'
|
28
analysis_options.yaml
Normal file
@ -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
|
13
android/.gitignore
vendored
Normal file
@ -0,0 +1,13 @@
|
||||
gradle-wrapper.jar
|
||||
/.gradle
|
||||
/captures/
|
||||
/gradlew
|
||||
/gradlew.bat
|
||||
/local.properties
|
||||
GeneratedPluginRegistrant.java
|
||||
|
||||
# Remember to never publicly share your keystore.
|
||||
# See https://flutter.dev/docs/deployment/android#reference-the-keystore-from-the-app
|
||||
key.properties
|
||||
**/*.keystore
|
||||
**/*.jks
|
76
android/app/build.gradle
Normal file
@ -0,0 +1,76 @@
|
||||
plugins {
|
||||
id "com.android.application"
|
||||
id "kotlin-android"
|
||||
id "dev.flutter.flutter-gradle-plugin"
|
||||
}
|
||||
|
||||
def localProperties = new Properties()
|
||||
def localPropertiesFile = rootProject.file('local.properties')
|
||||
if (localPropertiesFile.exists()) {
|
||||
localPropertiesFile.withReader('UTF-8') { reader ->
|
||||
localProperties.load(reader)
|
||||
}
|
||||
}
|
||||
|
||||
def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
|
||||
if (flutterVersionCode == null) {
|
||||
flutterVersionCode = '1'
|
||||
}
|
||||
|
||||
def flutterVersionName = localProperties.getProperty('flutter.versionName')
|
||||
if (flutterVersionName == null) {
|
||||
flutterVersionName = '1.0'
|
||||
}
|
||||
|
||||
android {
|
||||
namespace "com.shchoholiev.shopping_assistant_mobile_client"
|
||||
compileSdkVersion flutter.compileSdkVersion
|
||||
ndkVersion flutter.ndkVersion
|
||||
|
||||
compileOptions {
|
||||
sourceCompatibility JavaVersion.VERSION_1_8
|
||||
targetCompatibility JavaVersion.VERSION_1_8
|
||||
}
|
||||
|
||||
kotlinOptions {
|
||||
jvmTarget = '1.8'
|
||||
}
|
||||
|
||||
sourceSets {
|
||||
main.java.srcDirs += 'src/main/kotlin'
|
||||
}
|
||||
|
||||
defaultConfig {
|
||||
// TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
|
||||
applicationId "com.shchoholiev.shopping_assistant_mobile_client"
|
||||
// You can update the following values to match your application needs.
|
||||
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration.
|
||||
minSdkVersion flutter.minSdkVersion
|
||||
targetSdkVersion flutter.targetSdkVersion
|
||||
versionCode flutterVersionCode.toInteger()
|
||||
versionName flutterVersionName
|
||||
}
|
||||
|
||||
signingConfigs {
|
||||
release {
|
||||
keyAlias 'cartaid'
|
||||
keyPassword 'xxx'
|
||||
storeFile file('/Users/shchoholiev/Desktop/Cartaid/release-key.jks')
|
||||
storePassword 'xxx'
|
||||
}
|
||||
}
|
||||
|
||||
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.release
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
flutter {
|
||||
source '../..'
|
||||
}
|
||||
|
||||
dependencies {}
|
7
android/app/src/debug/AndroidManifest.xml
Normal file
@ -0,0 +1,7 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<!-- The INTERNET permission is required for development. Specifically,
|
||||
the Flutter tool needs it to communicate with the running application
|
||||
to allow setting breakpoints, to provide hot reload, etc.
|
||||
-->
|
||||
<uses-permission android:name="android.permission.INTERNET"/>
|
||||
</manifest>
|
34
android/app/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,34 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<application
|
||||
android:label="Cartaid"
|
||||
android:name="${applicationName}"
|
||||
android:icon="@mipmap/ic_launcher">
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:exported="true"
|
||||
android:launchMode="singleTop"
|
||||
android:theme="@style/LaunchTheme"
|
||||
android:configChanges="orientation|keyboardHidden|keyboard|screenSize|smallestScreenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
|
||||
android:hardwareAccelerated="true"
|
||||
android:windowSoftInputMode="adjustResize">
|
||||
<!-- Specifies an Android theme to apply to this Activity as soon as
|
||||
the Android process has started. This theme is visible to the user
|
||||
while the Flutter UI initializes. After that, this theme continues
|
||||
to determine the Window background behind the Flutter UI. -->
|
||||
<meta-data
|
||||
android:name="io.flutter.embedding.android.NormalTheme"
|
||||
android:resource="@style/NormalTheme"
|
||||
/>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN"/>
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
</intent-filter>
|
||||
</activity>
|
||||
<!-- Don't delete the meta-data below.
|
||||
This is used by the Flutter tool to generate GeneratedPluginRegistrant.java -->
|
||||
<meta-data
|
||||
android:name="flutterEmbedding"
|
||||
android:value="2" />
|
||||
</application>
|
||||
<uses-permission android:name="android.permission.INTERNET" />
|
||||
</manifest>
|
@ -0,0 +1,6 @@
|
||||
package com.shchoholiev.shopping_assistant_mobile_client
|
||||
|
||||
import io.flutter.embedding.android.FlutterActivity
|
||||
|
||||
class MainActivity: FlutterActivity() {
|
||||
}
|
12
android/app/src/main/res/drawable-v21/launch_background.xml
Normal file
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Modify this file to customize your launch splash screen -->
|
||||
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<item android:drawable="?android:colorBackground" />
|
||||
|
||||
<!-- You can insert your own image assets here -->
|
||||
<!-- <item>
|
||||
<bitmap
|
||||
android:gravity="center"
|
||||
android:src="@mipmap/launch_image" />
|
||||
</item> -->
|
||||
</layer-list>
|
12
android/app/src/main/res/drawable/launch_background.xml
Normal file
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Modify this file to customize your launch splash screen -->
|
||||
<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<item android:drawable="@android:color/white" />
|
||||
|
||||
<!-- You can insert your own image assets here -->
|
||||
<!-- <item>
|
||||
<bitmap
|
||||
android:gravity="center"
|
||||
android:src="@mipmap/launch_image" />
|
||||
</item> -->
|
||||
</layer-list>
|
BIN
android/app/src/main/res/ic_launcher.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
android/app/src/main/res/mipmap-hdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 5.2 KiB |
BIN
android/app/src/main/res/mipmap-mdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 3.4 KiB |
BIN
android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 7.2 KiB |
BIN
android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 15 KiB |
18
android/app/src/main/res/values-night/styles.xml
Normal file
@ -0,0 +1,18 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is on -->
|
||||
<style name="LaunchTheme" parent="@android:style/Theme.Black.NoTitleBar">
|
||||
<!-- Show a splash screen on the activity. Automatically removed when
|
||||
the Flutter engine draws its first frame -->
|
||||
<item name="android:windowBackground">@drawable/launch_background</item>
|
||||
</style>
|
||||
<!-- Theme applied to the Android Window as soon as the process has started.
|
||||
This theme determines the color of the Android Window while your
|
||||
Flutter UI initializes, as well as behind your Flutter UI while its
|
||||
running.
|
||||
|
||||
This Theme is only used starting with V2 of Flutter's Android embedding. -->
|
||||
<style name="NormalTheme" parent="@android:style/Theme.Black.NoTitleBar">
|
||||
<item name="android:windowBackground">?android:colorBackground</item>
|
||||
</style>
|
||||
</resources>
|
4
android/app/src/main/res/values/colors.xml
Normal file
@ -0,0 +1,4 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<color name="dark">#202124</color>
|
||||
</resources>
|
18
android/app/src/main/res/values/styles.xml
Normal file
@ -0,0 +1,18 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<!-- Theme applied to the Android Window while the process is starting when the OS's Dark Mode setting is off -->
|
||||
<style name="LaunchTheme" parent="@android:style/Theme.Light.NoTitleBar">
|
||||
<!-- Show a splash screen on the activity. Automatically removed when
|
||||
the Flutter engine draws its first frame -->
|
||||
<item name="android:windowBackground">@drawable/launch_background</item>
|
||||
</style>
|
||||
<!-- Theme applied to the Android Window as soon as the process has started.
|
||||
This theme determines the color of the Android Window while your
|
||||
Flutter UI initializes, as well as behind your Flutter UI while its
|
||||
running.
|
||||
|
||||
This Theme is only used starting with V2 of Flutter's Android embedding. -->
|
||||
<style name="NormalTheme" parent="@android:style/Theme.Light.NoTitleBar">
|
||||
<item name="android:windowBackground">?android:colorBackground</item>
|
||||
</style>
|
||||
</resources>
|
7
android/app/src/profile/AndroidManifest.xml
Normal file
@ -0,0 +1,7 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<!-- The INTERNET permission is required for development. Specifically,
|
||||
the Flutter tool needs it to communicate with the running application
|
||||
to allow setting breakpoints, to provide hot reload, etc.
|
||||
-->
|
||||
<uses-permission android:name="android.permission.INTERNET"/>
|
||||
</manifest>
|
BIN
android/app/src/profile/Icon-96.png
Normal file
After Width: | Height: | Size: 3.6 KiB |
31
android/build.gradle
Normal file
@ -0,0 +1,31 @@
|
||||
buildscript {
|
||||
ext.kotlin_version = '1.7.10'
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:7.3.0'
|
||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
|
||||
}
|
||||
}
|
||||
|
||||
allprojects {
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
}
|
||||
}
|
||||
|
||||
rootProject.buildDir = '../build'
|
||||
subprojects {
|
||||
project.buildDir = "${rootProject.buildDir}/${project.name}"
|
||||
}
|
||||
subprojects {
|
||||
project.evaluationDependsOn(':app')
|
||||
}
|
||||
|
||||
tasks.register("clean", Delete) {
|
||||
delete rootProject.buildDir
|
||||
}
|
3
android/gradle.properties
Normal file
@ -0,0 +1,3 @@
|
||||
org.gradle.jvmargs=-Xmx1536M
|
||||
android.useAndroidX=true
|
||||
android.enableJetifier=true
|
5
android/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,5 @@
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-all.zip
|
20
android/settings.gradle
Normal file
@ -0,0 +1,20 @@
|
||||
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
|
||||
}
|
||||
settings.ext.flutterSdkPath = flutterSdkPath()
|
||||
|
||||
includeBuild("${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle")
|
||||
|
||||
plugins {
|
||||
id "dev.flutter.flutter-gradle-plugin" version "1.0.0" apply false
|
||||
}
|
||||
}
|
||||
|
||||
include ":app"
|
||||
|
||||
apply from: "${settings.ext.flutterSdkPath}/packages/flutter_tools/gradle/app_plugin_loader.gradle"
|
3
assets/icons/add-products.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="30" height="30" viewBox="0 0 30 30" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M4 15H26M15 25V5" stroke="#009FFF" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
After Width: | Height: | Size: 215 B |
3
assets/icons/amazon.svg
Normal file
After Width: | Height: | Size: 8.1 KiB |
4
assets/icons/back.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg width="34" height="34" viewBox="0 0 34 34" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M10.1008 25.9391H21.4342C25.3442 25.9391 28.5175 22.7658 28.5175 18.8558C28.5175 14.9458 25.3442 11.7725 21.4342 11.7725H5.85083" stroke="#202124" stroke-width="2.5" stroke-miterlimit="10" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
<path d="M9.10909 15.3141L5.48242 11.6875L9.10909 8.06079" stroke="#202124" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
After Width: | Height: | Size: 494 B |
4
assets/icons/cart.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg width="21" height="20" viewBox="0 0 21 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M2.0551 12.257C1.3371 8.904 0.977103 7.227 1.8781 6.114C2.7781 5 4.4931 5 7.9221 5H13.0781C16.5081 5 18.2211 5 19.1221 6.114C20.0221 7.228 19.6631 8.904 18.9451 12.257L18.5161 14.257C18.0291 16.53 17.7861 17.666 16.9611 18.333C16.1361 19 14.9741 19 12.6501 19H8.3501C6.0261 19 4.8641 19 4.0401 18.333C3.2141 17.666 2.9701 16.53 2.4841 14.257L2.0551 12.257Z" stroke="white" stroke-width="1.5"/>
|
||||
<path d="M1.5 9H19.5M8.5 12H12.5M16.5 7L13.5 1M4.5 7L7.5 1" stroke="white" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
After Width: | Height: | Size: 649 B |
10
assets/icons/empty-star.svg
Normal file
@ -0,0 +1,10 @@
|
||||
<svg width="19" height="19" viewBox="0 0 19 19" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<g clip-path="url(#clip0_295_534)">
|
||||
<path d="M10.8696 2.77884L12.263 5.5655C12.453 5.95342 12.9596 6.3255 13.3871 6.39675L15.9126 6.81634C17.5276 7.0855 17.9075 8.25717 16.7438 9.413L14.7805 11.3763C14.448 11.7088 14.2659 12.3501 14.3688 12.8093L14.9309 15.2397C15.3742 17.1634 14.353 17.9076 12.6509 16.9022L10.2838 15.5009C9.8563 15.2476 9.15172 15.2476 8.7163 15.5009L6.34922 16.9022C4.65505 17.9076 3.62588 17.1555 4.06922 15.2397L4.6313 12.8093C4.73422 12.3501 4.55213 11.7088 4.21963 11.3763L2.2563 9.413C1.10047 8.25717 1.47255 7.0855 3.08755 6.81634L5.61297 6.39675C6.03255 6.3255 6.53922 5.95342 6.72922 5.5655L8.12255 2.77884C8.88255 1.26675 10.1176 1.26675 10.8696 2.77884Z" stroke="#FFC700" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="clip0_295_534">
|
||||
<rect width="19" height="19" fill="white"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
After Width: | Height: | Size: 980 B |
3
assets/icons/exit-cards.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="30" height="30" viewBox="0 0 30 30" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M15 18.75L18.75 15M18.75 15L15 11.25M18.75 15H5M5 9.06V9C5 7.6 5 6.9 5.2725 6.365C5.5125 5.89375 5.89375 5.5125 6.365 5.2725C6.9 5 7.6 5 9 5H21C22.4 5 23.1 5 23.6337 5.2725C24.105 5.5125 24.4875 5.89375 24.7275 6.365C25 6.89875 25 7.59875 25 8.99625V21.005C25 22.4025 25 23.1013 24.7275 23.635C24.4874 24.1055 24.1045 24.4879 23.6337 24.7275C23.1 25 22.4012 25 21.0037 25H8.99625C7.59875 25 6.89875 25 6.365 24.7275C5.89462 24.4878 5.51218 24.1054 5.2725 23.635C5 23.1 5 22.4 5 21V20.9375" stroke="#0165FF" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
After Width: | Height: | Size: 687 B |
9
assets/icons/half-star.svg
Normal file
@ -0,0 +1,9 @@
|
||||
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M7.47409 1.91065L8.43216 3.82679C8.56281 4.09353 8.9112 4.34937 9.20515 4.39837L10.9417 4.68688C12.0521 4.87196 12.3134 5.67761 11.5132 6.47237L10.1632 7.82238C9.93459 8.05101 9.80939 8.49194 9.88015 8.80766L10.2666 10.4788C10.5715 11.8016 9.86927 12.3133 8.6989 11.622L7.07127 10.6585C6.77731 10.4843 6.29284 10.4843 5.99344 10.6585L4.36581 11.622C3.20088 12.3133 2.49322 11.7962 2.79806 10.4788L3.18455 8.80766C3.25532 8.49194 3.13012 8.05101 2.90149 7.82238L1.55148 6.47237C0.756716 5.67761 1.01256 4.87196 2.12305 4.68688L3.85956 4.39837C4.14807 4.34937 4.49645 4.09353 4.6271 3.82679L5.58517 1.91065C6.10775 0.870929 6.95695 0.870929 7.47409 1.91065Z" fill="url(#paint0_linear_554_62)" stroke="#FFC700" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
<defs>
|
||||
<linearGradient id="paint0_linear_554_62" x1="1" y1="6" x2="12" y2="6" gradientUnits="userSpaceOnUse">
|
||||
<stop offset="0.53125" stop-color="#FFC700"/>
|
||||
<stop offset="0.557292" stop-color="#FFC700" stop-opacity="0"/>
|
||||
</linearGradient>
|
||||
</defs>
|
||||
</svg>
|
After Width: | Height: | Size: 1.1 KiB |
3
assets/icons/heart.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="28" height="26" viewBox="0 0 28 26" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M14.775 24.0125C14.35 24.1625 13.65 24.1625 13.225 24.0125C9.6 22.775 1.5 17.6125 1.5 8.8625C1.5 5 4.6125 1.875 8.45 1.875C10.725 1.875 12.7375 2.975 14 4.675C14.6422 3.80734 15.4787 3.10216 16.4425 2.61593C17.4063 2.1297 18.4705 1.87595 19.55 1.875C23.3875 1.875 26.5 5 26.5 8.8625C26.5 17.6125 18.4 22.775 14.775 24.0125Z" stroke="white" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</svg>
|
After Width: | Height: | Size: 520 B |
3
assets/icons/settings.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path fill-rule="evenodd" clip-rule="evenodd" d="M9.7911 1.2534C9.91165 0.53012 10.5374 0 11.2707 0H12.7293C13.4626 0 14.0884 0.530119 14.2089 1.2534L14.2724 1.6346C14.3382 2.02925 14.6344 2.34228 15.0159 2.46281C15.5698 2.63782 16.1029 2.85975 16.6105 3.12394C16.9655 3.30869 17.3963 3.29678 17.7219 3.0642L18.0371 2.83909C18.6337 2.41289 19.4511 2.48054 19.9696 2.99903L21.001 4.03043C21.5195 4.54892 21.5871 5.36627 21.1609 5.96295L20.9358 6.2781C20.7032 6.60371 20.6913 7.03455 20.8761 7.38949C21.1403 7.89707 21.3622 8.43016 21.5372 8.98409C21.6577 9.3656 21.9707 9.66179 22.3654 9.72757L22.7466 9.7911C23.4699 9.91165 24 10.5374 24 11.2707V12.7293C24 13.4626 23.4699 14.0884 22.7466 14.2089L22.3654 14.2724C21.9707 14.3382 21.6577 14.6344 21.5372 15.0159C21.3622 15.5698 21.1403 16.1029 20.8761 16.6105C20.6913 16.9654 20.7032 17.3963 20.9358 17.7219L21.1609 18.0371C21.5871 18.6337 21.5195 19.4511 21.001 19.9696L19.9696 21.001C19.4511 21.5195 18.6338 21.5871 18.0371 21.1609L17.7219 20.9358C17.3963 20.7032 16.9655 20.6913 16.6105 20.8761C16.1029 21.1403 15.5698 21.3622 15.0159 21.5372C14.6344 21.6577 14.3382 21.9707 14.2724 22.3654L14.2089 22.7466C14.0884 23.4699 13.4626 24 12.7293 24H11.2707C10.5374 24 9.91165 23.4699 9.7911 22.7466L9.72757 22.3654C9.66179 21.9707 9.36561 21.6577 8.98409 21.5372C8.43016 21.3622 7.89707 21.1403 7.38949 20.8761C7.03455 20.6913 6.60372 20.7032 6.27811 20.9358L5.96293 21.1609C5.36625 21.5871 4.5489 21.5195 4.03041 21.001L2.99901 19.9696C2.48052 19.4511 2.41287 18.6338 2.83907 18.0371L3.0642 17.7219C3.29678 17.3963 3.30869 16.9655 3.12394 16.6105C2.85975 16.1029 2.63782 15.5698 2.46281 15.0159C2.34228 14.6344 2.02925 14.3382 1.6346 14.2724L1.2534 14.2089C0.53012 14.0884 0 13.4626 0 12.7293V11.2707C0 10.5374 0.530119 9.91165 1.2534 9.7911L1.6346 9.72757C2.02925 9.66179 2.34228 9.3656 2.46281 8.98409C2.63782 8.43015 2.85975 7.89706 3.12394 7.38948C3.30869 7.03454 3.29678 6.60371 3.0642 6.2781L2.83909 5.96294C2.41289 5.36626 2.48054 4.54892 2.99903 4.03042L4.03043 2.99903C4.54892 2.48053 5.36627 2.41289 5.96295 2.83909L6.2781 3.0642C6.60371 3.29678 7.03455 3.30868 7.38949 3.12394C7.89707 2.85975 8.43016 2.63782 8.98409 2.46281C9.36561 2.34228 9.66179 2.02926 9.72757 1.6346L9.7911 1.2534ZM18.9291 13C18.4439 16.3923 15.5265 19 12 19C11.1568 19 10.3484 18.8509 9.5997 18.5776L11.6753 14.9826C11.7819 14.9941 11.8903 15 12 15C13.3062 15 14.4175 14.1652 14.8293 13H18.9291ZM18.9291 11H14.8293C14.4175 9.83481 13.3062 9 12 9C11.8903 9 11.782 9.00589 11.6754 9.01736L9.5998 5.42233C10.3484 5.14908 11.1568 5 12 5C15.5265 5 18.4439 7.60771 18.9291 11ZM7.83814 6.37104C6.11624 7.64626 5 9.69278 5 12C5 14.3072 6.1162 16.3537 7.83806 17.6289L9.86885 14.1115C9.33174 13.5694 9 12.8234 9 12C9 11.1765 9.33177 10.4306 9.86893 9.88848L7.83814 6.37104Z" fill="white"/>
|
||||
</svg>
|
After Width: | Height: | Size: 2.8 KiB |
10
assets/icons/star.svg
Normal file
@ -0,0 +1,10 @@
|
||||
<svg width="19" height="19" viewBox="0 0 19 19" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<g clip-path="url(#clip0_295_532)">
|
||||
<path d="M10.8696 2.77884L12.263 5.56551C12.453 5.95342 12.9596 6.3255 13.3871 6.39675L15.9125 6.81634C17.5276 7.0855 17.9075 8.25717 16.7438 9.413L14.7805 11.3763C14.448 11.7088 14.2659 12.3501 14.3688 12.8093L14.9309 15.2397C15.3742 17.1634 14.353 17.9076 12.6509 16.9022L10.2838 15.5009C9.8563 15.2476 9.15172 15.2476 8.7163 15.5009L6.34922 16.9022C4.65505 17.9076 3.62588 17.1555 4.06922 15.2397L4.6313 12.8093C4.73422 12.3501 4.55213 11.7088 4.21963 11.3763L2.2563 9.413C1.10047 8.25717 1.47255 7.0855 3.08755 6.81634L5.61297 6.39675C6.03255 6.3255 6.53922 5.95342 6.72922 5.56551L8.12255 2.77884C8.88255 1.26675 10.1175 1.26675 10.8696 2.77884Z" fill="#FFC700" stroke="#FFC700" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round"/>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="clip0_295_532">
|
||||
<rect width="19" height="19" fill="white"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
After Width: | Height: | Size: 997 B |
3
assets/icons/start-new-search.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="25" height="26" viewBox="0 0 25 26" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M22.9167 0.5H2.08333C1.5308 0.5 1.00089 0.719493 0.610194 1.11019C0.219493 1.50089 0 2.0308 0 2.58333V23.4167C0 23.9692 0.219493 24.4991 0.610194 24.8898C1.00089 25.2805 1.5308 25.5 2.08333 25.5H22.9167C23.4692 25.5 23.9991 25.2805 24.3898 24.8898C24.7805 24.4991 25 23.9692 25 23.4167V2.58333C25 2.0308 24.7805 1.50089 24.3898 1.11019C23.9991 0.719493 23.4692 0.5 22.9167 0.5ZM19.7917 14.0417H13.5417V20.2917C13.5417 20.5679 13.4319 20.8329 13.2366 21.0282C13.0412 21.2236 12.7763 21.3333 12.5 21.3333C12.2237 21.3333 11.9588 21.2236 11.7634 21.0282C11.5681 20.8329 11.4583 20.5679 11.4583 20.2917V14.0417H5.20833C4.93207 14.0417 4.66711 13.9319 4.47176 13.7366C4.27641 13.5412 4.16667 13.2763 4.16667 13C4.16667 12.7237 4.27641 12.4588 4.47176 12.2634C4.66711 12.0681 4.93207 11.9583 5.20833 11.9583H11.4583V5.70833C11.4583 5.43207 11.5681 5.16711 11.7634 4.97176C11.9588 4.77641 12.2237 4.66667 12.5 4.66667C12.7763 4.66667 13.0412 4.77641 13.2366 4.97176C13.4319 5.16711 13.5417 5.43207 13.5417 5.70833V11.9583H19.7917C20.0679 11.9583 20.3329 12.0681 20.5282 12.2634C20.7236 12.4588 20.8333 12.7237 20.8333 13C20.8333 13.2763 20.7236 13.5412 20.5282 13.7366C20.3329 13.9319 20.0679 14.0417 19.7917 14.0417Z" fill="white"/>
|
||||
</svg>
|
After Width: | Height: | Size: 1.3 KiB |
3
assets/icons/trash.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="18" height="20" viewBox="0 0 18 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M17 3C17.2652 3 17.5196 3.10536 17.7071 3.29289C17.8946 3.48043 18 3.73478 18 4C18 4.26522 17.8946 4.51957 17.7071 4.70711C17.5196 4.89464 17.2652 5 17 5H16L15.997 5.071L15.064 18.142C15.0281 18.6466 14.8023 19.1188 14.4321 19.4636C14.0619 19.8083 13.5749 20 13.069 20H4.93C4.42414 20 3.93707 19.8083 3.56688 19.4636C3.1967 19.1188 2.97092 18.6466 2.935 18.142L2.002 5.072C2.00048 5.04803 1.99982 5.02402 2 5H1C0.734784 5 0.48043 4.89464 0.292893 4.70711C0.105357 4.51957 0 4.26522 0 4C0 3.73478 0.105357 3.48043 0.292893 3.29289C0.48043 3.10536 0.734784 3 1 3H17ZM13.997 5H4.003L4.931 18H13.069L13.997 5ZM11 0C11.2652 0 11.5196 0.105357 11.7071 0.292893C11.8946 0.48043 12 0.734784 12 1C12 1.26522 11.8946 1.51957 11.7071 1.70711C11.5196 1.89464 11.2652 2 11 2H7C6.73478 2 6.48043 1.89464 6.29289 1.70711C6.10536 1.51957 6 1.26522 6 1C6 0.734784 6.10536 0.48043 6.29289 0.292893C6.48043 0.105357 6.73478 0 7 0H11Z" fill="white"/>
|
||||
</svg>
|
After Width: | Height: | Size: 1.0 KiB |
4
assets/icons/wishlists.svg
Normal file
@ -0,0 +1,4 @@
|
||||
<svg width="22" height="26" viewBox="0 0 22 26" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path fill-rule="evenodd" clip-rule="evenodd" d="M3.75 23V24C3.75 25.1046 4.64543 26 5.75 26H19.25C20.3546 26 21.25 25.1046 21.25 24V6C21.25 4.89543 20.3546 4 19.25 4H18.75V21C18.75 22.1046 17.8546 23 16.75 23H3.75Z" fill="white"/>
|
||||
<path fill-rule="evenodd" clip-rule="evenodd" d="M2.75 0C1.64543 0 0.75 0.89543 0.75 2V20C0.75 21.1046 1.64543 22 2.75 22H15.75C16.8546 22 17.75 21.1046 17.75 20V2C17.75 0.895431 16.8546 0 15.75 0H2.75ZM14.6141 9.26478C14.6077 9.20575 14.5939 9.07931 14.5762 8.96456C14.3212 7.57956 13.2812 6.59456 11.9812 6.50956C11.1062 6.44956 10.2462 6.76956 9.47625 7.43456L9.43625 7.46456L9.39625 7.43456C8.60625 6.75456 7.75625 6.43956 6.85125 6.50956C6.56625 6.52956 6.21125 6.62456 5.94125 6.74456C5.07625 7.14456 4.49625 7.93956 4.30625 8.97956C4.23125 9.33956 4.23125 9.90956 4.30625 10.2696C4.42625 10.9196 4.72625 11.5846 5.19125 12.2546C5.98625 13.3946 7.31625 14.6046 8.92625 15.6596L8.93583 15.6657C9.21727 15.8452 9.26325 15.8746 9.44125 15.8746C9.62125 15.8746 9.67125 15.8446 9.92625 15.6796C11.4562 14.6846 12.7012 13.5746 13.5362 12.4696C13.9262 11.9596 14.3062 11.2496 14.4612 10.7496C14.5362 10.4996 14.6062 10.0996 14.6162 9.96956C14.6362 9.74956 14.6362 9.54956 14.6162 9.28456C14.6157 9.27928 14.6149 9.27263 14.6141 9.26478Z" fill="white"/>
|
||||
</svg>
|
After Width: | Height: | Size: 1.4 KiB |
3
assets/icons/x.svg
Normal file
@ -0,0 +1,3 @@
|
||||
<svg width="30" height="30" viewBox="0 0 30 30" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M24.4324 22.4425C24.6966 22.7067 24.845 23.065 24.845 23.4386C24.845 23.8122 24.6966 24.1705 24.4324 24.4347C24.1682 24.6988 23.8099 24.8473 23.4363 24.8473C23.0627 24.8473 22.7044 24.6988 22.4402 24.4347L14.9999 16.9921L7.55737 24.4323C7.29319 24.6965 6.93488 24.8449 6.56128 24.8449C6.18767 24.8449 5.82936 24.6965 5.56518 24.4323C5.301 24.1681 5.15259 23.8098 5.15259 23.4362C5.15259 23.0626 5.301 22.7043 5.56518 22.4401L13.0078 14.9999L5.56753 7.55732C5.30335 7.29314 5.15493 6.93484 5.15493 6.56123C5.15493 6.18762 5.30335 5.82932 5.56753 5.56513C5.83171 5.30095 6.19001 5.15254 6.56362 5.15254C6.93723 5.15254 7.29553 5.30095 7.55971 5.56513L14.9999 13.0077L22.4425 5.56396C22.7067 5.29978 23.065 5.15137 23.4386 5.15137C23.8122 5.15137 24.1705 5.29978 24.4347 5.56396C24.6989 5.82814 24.8473 6.18645 24.8473 6.56006C24.8473 6.93366 24.6989 7.29197 24.4347 7.55615L16.9921 14.9999L24.4324 22.4425Z" fill="#0165FF"/>
|
||||
</svg>
|
After Width: | Height: | Size: 1.0 KiB |
BIN
assets/img/default-white.png
Normal file
After Width: | Height: | Size: 1.1 KiB |
34
ios/.gitignore
vendored
Normal file
@ -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
|
26
ios/Flutter/AppFrameworkInfo.plist
Normal file
@ -0,0 +1,26 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>en</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>App</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>io.flutter.flutter.app</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>App</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>FMWK</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>1.0</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>1.0</string>
|
||||
<key>MinimumOSVersion</key>
|
||||
<string>11.0</string>
|
||||
</dict>
|
||||
</plist>
|
2
ios/Flutter/Debug.xcconfig
Normal file
@ -0,0 +1,2 @@
|
||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"
|
||||
#include "Generated.xcconfig"
|
2
ios/Flutter/Release.xcconfig
Normal file
@ -0,0 +1,2 @@
|
||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"
|
||||
#include "Generated.xcconfig"
|
44
ios/Podfile
Normal file
@ -0,0 +1,44 @@
|
||||
# Uncomment this line to define a global platform for your project
|
||||
platform :ios, '15.0'
|
||||
|
||||
# CocoaPods analytics sends network stats synchronously affecting flutter build latency.
|
||||
ENV['COCOAPODS_DISABLE_STATS'] = 'true'
|
||||
|
||||
project 'Runner', {
|
||||
'Debug' => :debug,
|
||||
'Profile' => :release,
|
||||
'Release' => :release,
|
||||
}
|
||||
|
||||
def flutter_root
|
||||
generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'Generated.xcconfig'), __FILE__)
|
||||
unless File.exist?(generated_xcode_build_settings_path)
|
||||
raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure flutter pub get is executed first"
|
||||
end
|
||||
|
||||
File.foreach(generated_xcode_build_settings_path) do |line|
|
||||
matches = line.match(/FLUTTER_ROOT\=(.*)/)
|
||||
return matches[1].strip if matches
|
||||
end
|
||||
raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Generated.xcconfig, then run flutter pub get"
|
||||
end
|
||||
|
||||
require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root)
|
||||
|
||||
flutter_ios_podfile_setup
|
||||
|
||||
target 'Runner' do
|
||||
use_frameworks!
|
||||
use_modular_headers!
|
||||
|
||||
flutter_install_all_ios_pods File.dirname(File.realpath(__FILE__))
|
||||
target 'RunnerTests' do
|
||||
inherit! :search_paths
|
||||
end
|
||||
end
|
||||
|
||||
post_install do |installer|
|
||||
installer.pods_project.targets.each do |target|
|
||||
flutter_additional_ios_build_settings(target)
|
||||
end
|
||||
end
|
61
ios/Podfile.lock
Normal file
@ -0,0 +1,61 @@
|
||||
PODS:
|
||||
- connectivity_plus (0.0.1):
|
||||
- Flutter
|
||||
- ReachabilitySwift
|
||||
- Flutter (1.0.0)
|
||||
- FMDB (2.7.5):
|
||||
- FMDB/standard (= 2.7.5)
|
||||
- FMDB/standard (2.7.5)
|
||||
- path_provider_foundation (0.0.1):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- ReachabilitySwift (5.0.0)
|
||||
- shared_preferences_foundation (0.0.1):
|
||||
- Flutter
|
||||
- FlutterMacOS
|
||||
- sqflite (0.0.3):
|
||||
- Flutter
|
||||
- FMDB (>= 2.7.5)
|
||||
- url_launcher_ios (0.0.1):
|
||||
- Flutter
|
||||
|
||||
DEPENDENCIES:
|
||||
- connectivity_plus (from `.symlinks/plugins/connectivity_plus/ios`)
|
||||
- Flutter (from `Flutter`)
|
||||
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
|
||||
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
|
||||
- sqflite (from `.symlinks/plugins/sqflite/ios`)
|
||||
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`)
|
||||
|
||||
SPEC REPOS:
|
||||
trunk:
|
||||
- FMDB
|
||||
- ReachabilitySwift
|
||||
|
||||
EXTERNAL SOURCES:
|
||||
connectivity_plus:
|
||||
:path: ".symlinks/plugins/connectivity_plus/ios"
|
||||
Flutter:
|
||||
:path: Flutter
|
||||
path_provider_foundation:
|
||||
:path: ".symlinks/plugins/path_provider_foundation/darwin"
|
||||
shared_preferences_foundation:
|
||||
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
|
||||
sqflite:
|
||||
:path: ".symlinks/plugins/sqflite/ios"
|
||||
url_launcher_ios:
|
||||
:path: ".symlinks/plugins/url_launcher_ios/ios"
|
||||
|
||||
SPEC CHECKSUMS:
|
||||
connectivity_plus: bf0076dd84a130856aa636df1c71ccaff908fa1d
|
||||
Flutter: f04841e97a9d0b0a8025694d0796dd46242b2854
|
||||
FMDB: 2ce00b547f966261cd18927a3ddb07cb6f3db82a
|
||||
path_provider_foundation: 29f094ae23ebbca9d3d0cec13889cd9060c0e943
|
||||
ReachabilitySwift: 985039c6f7b23a1da463388634119492ff86c825
|
||||
shared_preferences_foundation: 5b919d13b803cadd15ed2dc053125c68730e5126
|
||||
sqflite: 31f7eba61e3074736dff8807a9b41581e4f7f15a
|
||||
url_launcher_ios: bf5ce03e0e2088bad9cc378ea97fa0ed5b49673b
|
||||
|
||||
PODFILE CHECKSUM: 9c46fd01abff66081b39f5fa5767b3f1d0b11d76
|
||||
|
||||
COCOAPODS: 1.12.1
|
724
ios/Runner.xcodeproj/project.pbxproj
Normal file
@ -0,0 +1,724 @@
|
||||
// !$*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 */; };
|
||||
56E0353746ED15182053ECD3 /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4A806F7CD09DA976CCAD3863 /* Pods_RunnerTests.framework */; };
|
||||
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 */; };
|
||||
B682E45354CC4BAB12178794 /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = FE92EE02B491BF6AD5FC98CD /* Pods_Runner.framework */; };
|
||||
/* 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 */
|
||||
029954CD834F78829E47247A /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; };
|
||||
0A4CEAE23127674AF3AB53BE /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; };
|
||||
0F8DC980862C7B0DDAB88DA6 /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; };
|
||||
127883CDBB2FC7B8C43C2347 /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; };
|
||||
1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; };
|
||||
1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; };
|
||||
331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; };
|
||||
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 = "<group>"; };
|
||||
4A806F7CD09DA976CCAD3863 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
6962B5FE101B87DBED54C956 /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; };
|
||||
74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = "<group>"; };
|
||||
74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; };
|
||||
7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; };
|
||||
9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; };
|
||||
9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; };
|
||||
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 = "<group>"; };
|
||||
97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
|
||||
97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; };
|
||||
97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
|
||||
9D611B8839E482A01080BA68 /* Pods-RunnerTests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.release.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.release.xcconfig"; sourceTree = "<group>"; };
|
||||
FE92EE02B491BF6AD5FC98CD /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
/* End PBXFileReference section */
|
||||
|
||||
/* Begin PBXFrameworksBuildPhase section */
|
||||
45D1A63F50B76A664F26E800 /* Frameworks */ = {
|
||||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
56E0353746ED15182053ECD3 /* Pods_RunnerTests.framework in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
97C146EB1CF9000F007C117D /* Frameworks */ = {
|
||||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
B682E45354CC4BAB12178794 /* Pods_Runner.framework in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXFrameworksBuildPhase section */
|
||||
|
||||
/* Begin PBXGroup section */
|
||||
331C8082294A63A400263BE5 /* RunnerTests */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
331C807B294A618700263BE5 /* RunnerTests.swift */,
|
||||
);
|
||||
path = RunnerTests;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
86B87DA936C01407FF532B09 /* Frameworks */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
FE92EE02B491BF6AD5FC98CD /* Pods_Runner.framework */,
|
||||
4A806F7CD09DA976CCAD3863 /* Pods_RunnerTests.framework */,
|
||||
);
|
||||
name = Frameworks;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
9740EEB11CF90186004384FC /* Flutter */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */,
|
||||
9740EEB21CF90195004384FC /* Debug.xcconfig */,
|
||||
7AFA3C8E1D35360C0083082E /* Release.xcconfig */,
|
||||
9740EEB31CF90195004384FC /* Generated.xcconfig */,
|
||||
);
|
||||
name = Flutter;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146E51CF9000F007C117D = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
9740EEB11CF90186004384FC /* Flutter */,
|
||||
97C146F01CF9000F007C117D /* Runner */,
|
||||
97C146EF1CF9000F007C117D /* Products */,
|
||||
331C8082294A63A400263BE5 /* RunnerTests */,
|
||||
FADF67E5D56A5B9DFE574E7A /* Pods */,
|
||||
86B87DA936C01407FF532B09 /* Frameworks */,
|
||||
);
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
97C146EF1CF9000F007C117D /* Products */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
97C146EE1CF9000F007C117D /* Runner.app */,
|
||||
331C8081294A63A400263BE5 /* RunnerTests.xctest */,
|
||||
);
|
||||
name = Products;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
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 = "<group>";
|
||||
};
|
||||
FADF67E5D56A5B9DFE574E7A /* Pods */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
127883CDBB2FC7B8C43C2347 /* Pods-Runner.debug.xcconfig */,
|
||||
0A4CEAE23127674AF3AB53BE /* Pods-Runner.release.xcconfig */,
|
||||
0F8DC980862C7B0DDAB88DA6 /* Pods-Runner.profile.xcconfig */,
|
||||
029954CD834F78829E47247A /* Pods-RunnerTests.debug.xcconfig */,
|
||||
9D611B8839E482A01080BA68 /* Pods-RunnerTests.release.xcconfig */,
|
||||
6962B5FE101B87DBED54C956 /* Pods-RunnerTests.profile.xcconfig */,
|
||||
);
|
||||
path = Pods;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
/* End PBXGroup section */
|
||||
|
||||
/* Begin PBXNativeTarget section */
|
||||
331C8080294A63A400263BE5 /* RunnerTests */ = {
|
||||
isa = PBXNativeTarget;
|
||||
buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */;
|
||||
buildPhases = (
|
||||
7D306219987D786916FB999D /* [CP] Check Pods Manifest.lock */,
|
||||
331C807D294A63A400263BE5 /* Sources */,
|
||||
331C807F294A63A400263BE5 /* Resources */,
|
||||
45D1A63F50B76A664F26E800 /* Frameworks */,
|
||||
);
|
||||
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 = (
|
||||
CA1CCCB9B316BC3A80D34A0D /* [CP] Check Pods Manifest.lock */,
|
||||
9740EEB61CF901F6004384FC /* Run Script */,
|
||||
97C146EA1CF9000F007C117D /* Sources */,
|
||||
97C146EB1CF9000F007C117D /* Frameworks */,
|
||||
97C146EC1CF9000F007C117D /* Resources */,
|
||||
9705A1C41CF9048500538489 /* Embed Frameworks */,
|
||||
3B06AD1E1E4923F5004D2608 /* Thin Binary */,
|
||||
E18ECFF3C9B5CAA9E2C1059F /* [CP] Embed Pods Frameworks */,
|
||||
);
|
||||
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 = 1430;
|
||||
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";
|
||||
};
|
||||
7D306219987D786916FB999D /* [CP] Check Pods Manifest.lock */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
||||
"${PODS_ROOT}/Manifest.lock",
|
||||
);
|
||||
name = "[CP] Check Pods Manifest.lock";
|
||||
outputFileListPaths = (
|
||||
);
|
||||
outputPaths = (
|
||||
"$(DERIVED_FILE_DIR)/Pods-RunnerTests-checkManifestLockResult.txt",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
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";
|
||||
};
|
||||
CA1CCCB9B316BC3A80D34A0D /* [CP] Check Pods Manifest.lock */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
||||
"${PODS_ROOT}/Manifest.lock",
|
||||
);
|
||||
name = "[CP] Check Pods Manifest.lock";
|
||||
outputFileListPaths = (
|
||||
);
|
||||
outputPaths = (
|
||||
"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
E18ECFF3C9B5CAA9E2C1059F /* [CP] Embed Pods Frameworks */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-input-files.xcfilelist",
|
||||
);
|
||||
name = "[CP] Embed Pods Frameworks";
|
||||
outputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-output-files.xcfilelist",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
/* 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 = "<group>";
|
||||
};
|
||||
97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = {
|
||||
isa = PBXVariantGroup;
|
||||
children = (
|
||||
97C147001CF9000F007C117D /* Base */,
|
||||
);
|
||||
name = LaunchScreen.storyboard;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
/* End PBXVariantGroup section */
|
||||
|
||||
/* Begin XCBuildConfiguration section */
|
||||
249021D3217E4FDB00AE95B9 /* Profile */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
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;
|
||||
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 = 11.0;
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
SDKROOT = iphoneos;
|
||||
SUPPORTED_PLATFORMS = iphoneos;
|
||||
TARGETED_DEVICE_FAMILY = "1,2";
|
||||
VALIDATE_PRODUCT = YES;
|
||||
};
|
||||
name = Profile;
|
||||
};
|
||||
249021D4217E4FDB00AE95B9 /* Profile */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||
DEVELOPMENT_TEAM = 64FG98G3D5;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.shchoholiev.ShoppingAssistant;
|
||||
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;
|
||||
baseConfigurationReference = 029954CD834F78829E47247A /* Pods-RunnerTests.debug.xcconfig */;
|
||||
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.shchoholiev.shoppingAssistantMobileClient.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;
|
||||
baseConfigurationReference = 9D611B8839E482A01080BA68 /* Pods-RunnerTests.release.xcconfig */;
|
||||
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.shchoholiev.shoppingAssistantMobileClient.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;
|
||||
baseConfigurationReference = 6962B5FE101B87DBED54C956 /* Pods-RunnerTests.profile.xcconfig */;
|
||||
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.shchoholiev.shoppingAssistantMobileClient.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;
|
||||
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;
|
||||
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 = 11.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;
|
||||
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;
|
||||
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 = 11.0;
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
SDKROOT = iphoneos;
|
||||
SUPPORTED_PLATFORMS = iphoneos;
|
||||
SWIFT_COMPILATION_MODE = wholemodule;
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-O";
|
||||
TARGETED_DEVICE_FAMILY = "1,2";
|
||||
VALIDATE_PRODUCT = YES;
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
97C147061CF9000F007C117D /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||
DEVELOPMENT_TEAM = 64FG98G3D5;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.shchoholiev.ShoppingAssistant;
|
||||
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)";
|
||||
DEVELOPMENT_TEAM = 64FG98G3D5;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.shchoholiev.ShoppingAssistant;
|
||||
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 */;
|
||||
}
|
7
ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata
generated
Normal file
@ -0,0 +1,7 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Workspace
|
||||
version = "1.0">
|
||||
<FileRef
|
||||
location = "self:">
|
||||
</FileRef>
|
||||
</Workspace>
|
@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>IDEDidComputeMac32BitWarning</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>PreviewsEnabled</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
98
ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme
Normal file
@ -0,0 +1,98 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Scheme
|
||||
LastUpgradeVersion = "1430"
|
||||
version = "1.3">
|
||||
<BuildAction
|
||||
parallelizeBuildables = "YES"
|
||||
buildImplicitDependencies = "YES">
|
||||
<BuildActionEntries>
|
||||
<BuildActionEntry
|
||||
buildForTesting = "YES"
|
||||
buildForRunning = "YES"
|
||||
buildForProfiling = "YES"
|
||||
buildForArchiving = "YES"
|
||||
buildForAnalyzing = "YES">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildActionEntry>
|
||||
</BuildActionEntries>
|
||||
</BuildAction>
|
||||
<TestAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES">
|
||||
<MacroExpansion>
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</MacroExpansion>
|
||||
<Testables>
|
||||
<TestableReference
|
||||
skipped = "NO"
|
||||
parallelizable = "YES">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "331C8080294A63A400263BE5"
|
||||
BuildableName = "RunnerTests.xctest"
|
||||
BlueprintName = "RunnerTests"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</TestableReference>
|
||||
</Testables>
|
||||
</TestAction>
|
||||
<LaunchAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
launchStyle = "0"
|
||||
useCustomWorkingDirectory = "NO"
|
||||
ignoresPersistentStateOnLaunch = "NO"
|
||||
debugDocumentVersioning = "YES"
|
||||
debugServiceExtension = "internal"
|
||||
allowLocationSimulation = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</LaunchAction>
|
||||
<ProfileAction
|
||||
buildConfiguration = "Profile"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES"
|
||||
savedToolIdentifier = ""
|
||||
useCustomWorkingDirectory = "NO"
|
||||
debugDocumentVersioning = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "97C146ED1CF9000F007C117D"
|
||||
BuildableName = "Runner.app"
|
||||
BlueprintName = "Runner"
|
||||
ReferencedContainer = "container:Runner.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</ProfileAction>
|
||||
<AnalyzeAction
|
||||
buildConfiguration = "Debug">
|
||||
</AnalyzeAction>
|
||||
<ArchiveAction
|
||||
buildConfiguration = "Release"
|
||||
revealArchiveInOrganizer = "YES">
|
||||
</ArchiveAction>
|
||||
</Scheme>
|
10
ios/Runner.xcworkspace/contents.xcworkspacedata
generated
Normal file
@ -0,0 +1,10 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Workspace
|
||||
version = "1.0">
|
||||
<FileRef
|
||||
location = "group:Runner.xcodeproj">
|
||||
</FileRef>
|
||||
<FileRef
|
||||
location = "group:Pods/Pods.xcodeproj">
|
||||
</FileRef>
|
||||
</Workspace>
|
@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>IDEDidComputeMac32BitWarning</key>
|
||||
<true/>
|
||||
</dict>
|
||||
</plist>
|
@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>PreviewsEnabled</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
13
ios/Runner/AppDelegate.swift
Normal file
@ -0,0 +1,13 @@
|
||||
import UIKit
|
||||
import Flutter
|
||||
|
||||
@UIApplicationMain
|
||||
@objc class AppDelegate: FlutterAppDelegate {
|
||||
override func application(
|
||||
_ application: UIApplication,
|
||||
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
|
||||
) -> Bool {
|
||||
GeneratedPluginRegistrant.register(with: self)
|
||||
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
|
||||
}
|
||||
}
|
122
ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json
Normal file
@ -0,0 +1,122 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"filename" : "Icon-40 2.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-60.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-29 1.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "1x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-58 1.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-87.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-80 1.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-120 1.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-120.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "60x60"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-180.png",
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "60x60"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-20.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "1x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-40.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "2x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-29.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "1x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-58.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "2x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-40 1.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "1x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-80.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "2x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-76.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "1x",
|
||||
"size" : "76x76"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-152.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "2x",
|
||||
"size" : "76x76"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-167.png",
|
||||
"idiom" : "ipad",
|
||||
"scale" : "2x",
|
||||
"size" : "83.5x83.5"
|
||||
},
|
||||
{
|
||||
"filename" : "Icon-App-1024x1024@1x.png",
|
||||
"idiom" : "ios-marketing",
|
||||
"scale" : "1x",
|
||||
"size" : "1024x1024"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-120 1.png
Normal file
After Width: | Height: | Size: 3.1 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-120.png
Normal file
After Width: | Height: | Size: 3.1 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-152.png
Normal file
After Width: | Height: | Size: 4.0 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-167.png
Normal file
After Width: | Height: | Size: 4.4 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-180.png
Normal file
After Width: | Height: | Size: 4.8 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-20.png
Normal file
After Width: | Height: | Size: 439 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-29 1.png
Normal file
After Width: | Height: | Size: 676 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-29.png
Normal file
After Width: | Height: | Size: 676 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-40 1.png
Normal file
After Width: | Height: | Size: 915 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-40 2.png
Normal file
After Width: | Height: | Size: 915 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-40.png
Normal file
After Width: | Height: | Size: 915 B |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-58 1.png
Normal file
After Width: | Height: | Size: 1.5 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-58.png
Normal file
After Width: | Height: | Size: 1.5 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-60.png
Normal file
After Width: | Height: | Size: 1.5 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-76.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-80 1.png
Normal file
After Width: | Height: | Size: 2.1 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-80.png
Normal file
After Width: | Height: | Size: 2.1 KiB |
BIN
ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-87.png
Normal file
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 11 KiB |
6
ios/Runner/Assets.xcassets/Contents.json
Normal file
@ -0,0 +1,6 @@
|
||||
{
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
23
ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json
vendored
Normal file
@ -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"
|
||||
}
|
||||
}
|
BIN
ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png
vendored
Normal file
After Width: | Height: | Size: 68 B |
BIN
ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png
vendored
Normal file
After Width: | Height: | Size: 68 B |
BIN
ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png
vendored
Normal file
After Width: | Height: | Size: 68 B |
5
ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md
vendored
Normal file
@ -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.
|
37
ios/Runner/Base.lproj/LaunchScreen.storyboard
Normal file
@ -0,0 +1,37 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="12121" systemVersion="16G29" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="12089"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--View Controller-->
|
||||
<scene sceneID="EHf-IW-A2E">
|
||||
<objects>
|
||||
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
|
||||
<layoutGuides>
|
||||
<viewControllerLayoutGuide type="top" id="Ydg-fD-yQy"/>
|
||||
<viewControllerLayoutGuide type="bottom" id="xbc-2k-c8Z"/>
|
||||
</layoutGuides>
|
||||
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<subviews>
|
||||
<imageView opaque="NO" clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="center" image="LaunchImage" translatesAutoresizingMaskIntoConstraints="NO" id="YRO-k0-Ey4">
|
||||
</imageView>
|
||||
</subviews>
|
||||
<color key="backgroundColor" red="1" green="1" blue="1" alpha="1" colorSpace="custom" customColorSpace="sRGB"/>
|
||||
<constraints>
|
||||
<constraint firstItem="YRO-k0-Ey4" firstAttribute="centerX" secondItem="Ze5-6b-2t3" secondAttribute="centerX" id="1a2-6s-vTC"/>
|
||||
<constraint firstItem="YRO-k0-Ey4" firstAttribute="centerY" secondItem="Ze5-6b-2t3" secondAttribute="centerY" id="4X2-HB-R7a"/>
|
||||
</constraints>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
<point key="canvasLocation" x="53" y="375"/>
|
||||
</scene>
|
||||
</scenes>
|
||||
<resources>
|
||||
<image name="LaunchImage" width="168" height="185"/>
|
||||
</resources>
|
||||
</document>
|
26
ios/Runner/Base.lproj/Main.storyboard
Normal file
@ -0,0 +1,26 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="10117" systemVersion="15F34" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" initialViewController="BYZ-38-t0r">
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="10085"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--Flutter View Controller-->
|
||||
<scene sceneID="tne-QT-ifu">
|
||||
<objects>
|
||||
<viewController id="BYZ-38-t0r" customClass="FlutterViewController" sceneMemberID="viewController">
|
||||
<layoutGuides>
|
||||
<viewControllerLayoutGuide type="top" id="y3c-jy-aDJ"/>
|
||||
<viewControllerLayoutGuide type="bottom" id="wfy-db-euE"/>
|
||||
</layoutGuides>
|
||||
<view key="view" contentMode="scaleToFill" id="8bC-Xf-vdC">
|
||||
<rect key="frame" x="0.0" y="0.0" width="600" height="600"/>
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="calibratedWhite"/>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="dkx-z0-nzr" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
</scene>
|
||||
</scenes>
|
||||
</document>
|
49
ios/Runner/Info.plist
Normal file
@ -0,0 +1,49 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CADisableMinimumFrameDurationOnPhone</key>
|
||||
<true/>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>Cartaid</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>shopping_assistant_mobile_client</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>APPL</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>$(FLUTTER_BUILD_NAME)</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(FLUTTER_BUILD_NUMBER)</string>
|
||||
<key>LSRequiresIPhoneOS</key>
|
||||
<true/>
|
||||
<key>UIApplicationSupportsIndirectInputEvents</key>
|
||||
<true/>
|
||||
<key>UILaunchStoryboardName</key>
|
||||
<string>LaunchScreen</string>
|
||||
<key>UIMainStoryboardFile</key>
|
||||
<string>Main</string>
|
||||
<key>UISupportedInterfaceOrientations</key>
|
||||
<array>
|
||||
<string>UIInterfaceOrientationPortrait</string>
|
||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||
</array>
|
||||
<key>UISupportedInterfaceOrientations~ipad</key>
|
||||
<array>
|
||||
<string>UIInterfaceOrientationPortrait</string>
|
||||
<string>UIInterfaceOrientationPortraitUpsideDown</string>
|
||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||
</array>
|
||||
</dict>
|
||||
</plist>
|
1
ios/Runner/Runner-Bridging-Header.h
Normal file
@ -0,0 +1 @@
|
||||
#import "GeneratedPluginRegistrant.h"
|
12
ios/RunnerTests/RunnerTests.swift
Normal file
@ -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.
|
||||
}
|
||||
|
||||
}
|
10
lib/constants/jwt_claims.dart
Normal file
@ -0,0 +1,10 @@
|
||||
class JwtClaims {
|
||||
|
||||
static const String id = 'http://schemas.xmlsoap.org/ws/2005/05/identity/claims/nameidentifier';
|
||||
|
||||
static const String email = 'http://schemas.xmlsoap.org/ws/2005/05/identity/claims/emailaddress';
|
||||
|
||||
static const String phone = 'http://schemas.xmlsoap.org/ws/2005/05/identity/claims/mobilephone';
|
||||
|
||||
static const String roles ='http://schemas.microsoft.com/ws/2008/06/identity/claims/role';
|
||||
}
|
143
lib/main.dart
Normal file
@ -0,0 +1,143 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_svg/flutter_svg.dart';
|
||||
import 'package:shopping_assistant_mobile_client/screens/chat.dart';
|
||||
import 'package:shopping_assistant_mobile_client/screens/settings.dart';
|
||||
import 'package:shopping_assistant_mobile_client/screens/wishlists.dart';
|
||||
|
||||
void main() {
|
||||
runApp(const MyApp());
|
||||
}
|
||||
|
||||
class MyApp extends StatefulWidget {
|
||||
const MyApp({super.key});
|
||||
|
||||
static const List<String> _pageNameOptions = <String>[
|
||||
'Wishlists',
|
||||
'New Chat',
|
||||
'Settings',
|
||||
];
|
||||
|
||||
static List<Widget> _widgetOptions = <Widget>[
|
||||
WishlistsScreen(),
|
||||
ChatScreen(wishlistId: '', wishlistName: 'New Chat', openedFromBottomBar: true),
|
||||
SettingsScreen(),
|
||||
];
|
||||
|
||||
static const Color _selectedColor = Color.fromRGBO(36, 36, 36, 1);
|
||||
static const Color _unselectedColor = Color.fromRGBO(144, 144, 144, 1);
|
||||
|
||||
State<MyApp> createState() => _MyAppState();
|
||||
}
|
||||
|
||||
class _MyAppState extends State<MyApp> {
|
||||
int _selectedIndex = 1;
|
||||
|
||||
void _onItemTapped(int index) {
|
||||
setState(() {
|
||||
_selectedIndex = index;
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return MaterialApp(
|
||||
theme: ThemeData(
|
||||
useMaterial3: true,
|
||||
appBarTheme: AppBarTheme(),
|
||||
textTheme: TextTheme(
|
||||
bodyMedium: TextStyle(
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
),
|
||||
),
|
||||
home: Scaffold(
|
||||
appBar: _selectedIndex == 1
|
||||
? null
|
||||
: AppBar(
|
||||
title: Text(MyApp._pageNameOptions[_selectedIndex]),
|
||||
centerTitle: true,
|
||||
bottom: PreferredSize(
|
||||
preferredSize: const Size.fromHeight(1),
|
||||
child: Container(
|
||||
color: Color.fromRGBO(234, 234, 234, 1),
|
||||
height: 1,
|
||||
),
|
||||
),
|
||||
),
|
||||
body: MyApp._widgetOptions[_selectedIndex],
|
||||
bottomNavigationBar: BottomNavigationBar(
|
||||
items: <BottomNavigationBarItem>[
|
||||
BottomNavigationBarItem(
|
||||
icon: SvgPicture.asset(
|
||||
'assets/icons/wishlists.svg',
|
||||
color: _selectedIndex == 0
|
||||
? MyApp._selectedColor
|
||||
: MyApp._unselectedColor,
|
||||
),
|
||||
label: 'Wishlists',
|
||||
),
|
||||
BottomNavigationBarItem(
|
||||
icon: SvgPicture.asset(
|
||||
'assets/icons/start-new-search.svg',
|
||||
color: _selectedIndex == 1
|
||||
? MyApp._selectedColor
|
||||
: MyApp._unselectedColor,
|
||||
),
|
||||
label: 'New Chat',
|
||||
),
|
||||
BottomNavigationBarItem(
|
||||
icon: SvgPicture.asset(
|
||||
'assets/icons/settings.svg',
|
||||
color: _selectedIndex == 2
|
||||
? MyApp._selectedColor
|
||||
: MyApp._unselectedColor,
|
||||
),
|
||||
label: 'Settings',
|
||||
),
|
||||
],
|
||||
selectedItemColor: MyApp._selectedColor,
|
||||
unselectedItemColor: MyApp._unselectedColor,
|
||||
selectedFontSize: 14,
|
||||
unselectedFontSize: 14,
|
||||
currentIndex: _selectedIndex,
|
||||
onTap: _onItemTapped,
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
// Use to seed wishlists for new user
|
||||
// final ApiClient client = ApiClient();
|
||||
//
|
||||
// const String startPersonalWishlistMutations = r'''
|
||||
// mutation startPersonalWishlist($dto: WishlistCreateDtoInput!) {
|
||||
// startPersonalWishlist(dto: $dto) {
|
||||
// createdById, id, name, type
|
||||
// }
|
||||
// }
|
||||
// ''';
|
||||
//
|
||||
// MutationOptions mutationOptions = MutationOptions(
|
||||
// document: gql(startPersonalWishlistMutations),
|
||||
// variables: const <String, dynamic>{
|
||||
// 'dto': {
|
||||
// 'firstMessageText': 'Gaming mechanical keyboard',
|
||||
// 'type': 'Product'
|
||||
// },
|
||||
// });
|
||||
//
|
||||
// var client = ApiClient();
|
||||
// // for (var i = 0; i < 5; i++) {
|
||||
// // client
|
||||
// // .mutate(mutationOptions)
|
||||
// // .then((result) => print(jsonEncode(result)));
|
||||
// // sleep(Duration(milliseconds: 100));
|
||||
// // }
|
||||
//
|
6
lib/models/enums/search_event_type.dart
Normal file
@ -0,0 +1,6 @@
|
||||
enum SearchEventType {
|
||||
wishlist,
|
||||
message,
|
||||
suggestion,
|
||||
product
|
||||
}
|
10
lib/models/global_instances/global_user.dart
Normal file
@ -0,0 +1,10 @@
|
||||
class GlobalUser {
|
||||
|
||||
static String? id;
|
||||
|
||||
static String? email;
|
||||
|
||||
static String? phone;
|
||||
|
||||
static List<String>? roles;
|
||||
}
|
33
lib/models/product.dart
Normal file
@ -0,0 +1,33 @@
|
||||
//import 'dart:ffi';
|
||||
|
||||
class Product {
|
||||
Product({
|
||||
required this.id,
|
||||
required this.name,
|
||||
required this.url,
|
||||
required this.imageUrls,
|
||||
required this.rating,
|
||||
required this.price,
|
||||
required this.description,
|
||||
required this.wasOpened,
|
||||
});
|
||||
|
||||
String id;
|
||||
String name;
|
||||
String url;
|
||||
List<String> imageUrls;
|
||||
double rating;
|
||||
double price;
|
||||
String description;
|
||||
bool wasOpened;
|
||||
|
||||
Product.fromJson(Map<String, dynamic> json)
|
||||
: id = json['id'] as String,
|
||||
name = json['name'] as String,
|
||||
url = json['url'] as String,
|
||||
imageUrls = (json['imageUrls'] is Null) ? [''] : json['imageUrls'] as List<String>,
|
||||
rating = (json['rating'] is int) ? (json['rating'] as int).toDouble() : json['rating'] as double,
|
||||
price = (json['price'] is int) ? (json['price'] as int).toDouble() : json['price'] as double,
|
||||
description = (json['description'] is Null) ? '' : json['description'] as String,
|
||||
wasOpened = (json['wasOpened'] is Null) ? false : json['wasOpened'] as bool;
|
||||
}
|
9
lib/models/server_sent_event.dart
Normal file
@ -0,0 +1,9 @@
|
||||
import 'package:shopping_assistant_mobile_client/models/enums/search_event_type.dart';
|
||||
|
||||
class ServerSentEvent {
|
||||
|
||||
SearchEventType event;
|
||||
String data;
|
||||
|
||||
ServerSentEvent(this.event, this.data);
|
||||
}
|
21
lib/models/wishlist.dart
Normal file
@ -0,0 +1,21 @@
|
||||
class Wishlist {
|
||||
Wishlist(
|
||||
{required this.id,
|
||||
required this.name,
|
||||
required this.type,
|
||||
required this.createdById});
|
||||
|
||||
String id;
|
||||
|
||||
String name;
|
||||
|
||||
String type;
|
||||
|
||||
String createdById;
|
||||
|
||||
Wishlist.fromJson(Map<String, dynamic> json)
|
||||
: id = json['id'] as String,
|
||||
name = json['name'] as String,
|
||||
type = json['type'] as String,
|
||||
createdById = json['createdById'] as String;
|
||||
}
|
98
lib/network/api_client.dart
Normal file
@ -0,0 +1,98 @@
|
||||
import 'dart:async';
|
||||
import 'dart:convert';
|
||||
import 'dart:io';
|
||||
import 'package:graphql/client.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
import 'package:shopping_assistant_mobile_client/models/enums/search_event_type.dart';
|
||||
import 'package:shopping_assistant_mobile_client/models/global_instances/global_user.dart';
|
||||
import 'package:shopping_assistant_mobile_client/models/server_sent_event.dart';
|
||||
import 'package:shopping_assistant_mobile_client/network/authentication_service.dart';
|
||||
|
||||
class ApiClient {
|
||||
final String _apiBaseUrl = 'https://shopping-assistant-api-dev.azurewebsites.net/';
|
||||
late String _accessToken;
|
||||
|
||||
final AuthenticationService _authenticationService = AuthenticationService();
|
||||
|
||||
late GraphQLClient _graphqlClient;
|
||||
final http.Client _httpClient = http.Client();
|
||||
|
||||
Future<Map<String, dynamic>?> query(QueryOptions options) async {
|
||||
await _setAuthentication();
|
||||
|
||||
final QueryResult result = await _graphqlClient.query(options);
|
||||
|
||||
if (result.hasException) {
|
||||
print(result.exception.toString());
|
||||
}
|
||||
|
||||
return result.data;
|
||||
}
|
||||
|
||||
Future<Map<String, dynamic>?> mutate(MutationOptions options) async {
|
||||
await _setAuthentication();
|
||||
|
||||
final QueryResult result = await _graphqlClient.mutate(options);
|
||||
|
||||
if (result.hasException) {
|
||||
print(result.exception.toString());
|
||||
}
|
||||
|
||||
return result.data;
|
||||
}
|
||||
|
||||
Stream<ServerSentEvent> getServerSentEventStream(String urlPath, Map<dynamic, dynamic> requestBody) async* {
|
||||
await _setAuthentication();
|
||||
|
||||
final url = Uri.parse('$_apiBaseUrl$urlPath');
|
||||
final request = http.Request('POST', url);
|
||||
request.body = jsonEncode(requestBody);
|
||||
request.headers.addAll({
|
||||
HttpHeaders.authorizationHeader: 'Bearer $_accessToken',
|
||||
HttpHeaders.contentTypeHeader: 'application/json'
|
||||
});
|
||||
final response = await _httpClient.send(request);
|
||||
|
||||
var eventType = SearchEventType.message;
|
||||
await for (var line in response.stream.transform(utf8.decoder).transform(const LineSplitter())) {
|
||||
if (line.startsWith('event: ')) {
|
||||
var type = line.substring('event: '.length);
|
||||
switch (type) {
|
||||
case 'Message':
|
||||
eventType = SearchEventType.message;
|
||||
break;
|
||||
case 'Suggestion':
|
||||
eventType = SearchEventType.suggestion;
|
||||
break;
|
||||
case 'Product':
|
||||
eventType = SearchEventType.product;
|
||||
break;
|
||||
case 'Wishlist':
|
||||
eventType = SearchEventType.wishlist;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (line.startsWith('data: ')) {
|
||||
yield ServerSentEvent(eventType, line.substring('data: '.length));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Future _setAuthentication() async {
|
||||
_accessToken = await _authenticationService.getAccessToken();
|
||||
|
||||
GlobalUser.id = _authenticationService.getIdFromAccessToken(_accessToken);
|
||||
GlobalUser.email = _authenticationService.getEmailFromAccessToken(_accessToken);
|
||||
GlobalUser.phone = _authenticationService.getPhoneFromAccessToken(_accessToken);
|
||||
// GlobalUser.roles = _authenticationService.getRolesFromAccessToken(_accessToken);
|
||||
|
||||
final httpLink = HttpLink('${_apiBaseUrl}graphql/', defaultHeaders: {
|
||||
HttpHeaders.authorizationHeader: 'Bearer $_accessToken'
|
||||
});
|
||||
|
||||
_graphqlClient = GraphQLClient(
|
||||
cache: GraphQLCache(),
|
||||
link: httpLink
|
||||
);
|
||||
}
|
||||
}
|
169
lib/network/authentication_service.dart
Normal file
@ -0,0 +1,169 @@
|
||||
import 'package:graphql/client.dart';
|
||||
import 'package:jwt_decoder/jwt_decoder.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
import 'package:shopping_assistant_mobile_client/constants/jwt_claims.dart';
|
||||
import 'package:uuid/uuid.dart';
|
||||
|
||||
class AuthenticationService {
|
||||
final GraphQLClient client = GraphQLClient(
|
||||
cache: GraphQLCache(),
|
||||
link: HttpLink('https://shopping-assistant-api-dev.azurewebsites.net/graphql'),
|
||||
);
|
||||
|
||||
late SharedPreferences prefs;
|
||||
|
||||
Future<String> getAccessToken() async {
|
||||
prefs = await SharedPreferences.getInstance();
|
||||
var accessToken = prefs.getString('accessToken');
|
||||
var refreshToken = prefs.getString('refreshToken');
|
||||
|
||||
if (accessToken == null && refreshToken != null) {
|
||||
print('WTF??');
|
||||
} else if (accessToken == null && refreshToken == null) {
|
||||
accessToken = await _accessGuest();
|
||||
print('Got new access token $accessToken');
|
||||
} else if (JwtDecoder.isExpired(accessToken!)) {
|
||||
accessToken = await _refreshAccessToken();
|
||||
print('Refreshed access token $accessToken');
|
||||
}
|
||||
|
||||
print('Returned access token $accessToken');
|
||||
return accessToken!;
|
||||
}
|
||||
|
||||
Future login(String? email, String? phone, String password) async {
|
||||
prefs = await SharedPreferences.getInstance();
|
||||
const String loginQuery = r'''
|
||||
mutation Login($login: AccessUserModelInput!) {
|
||||
login(login: $login) {
|
||||
accessToken
|
||||
refreshToken
|
||||
}
|
||||
}
|
||||
''';
|
||||
|
||||
final MutationOptions options = MutationOptions(
|
||||
document: gql(loginQuery),
|
||||
variables: <String, dynamic>{
|
||||
'login': {
|
||||
'email': email,
|
||||
'phone': phone,
|
||||
'password': password,
|
||||
},
|
||||
},
|
||||
);
|
||||
|
||||
final QueryResult result = await client.mutate(options);
|
||||
|
||||
if (result.hasException) {
|
||||
print(result.exception.toString());
|
||||
}
|
||||
|
||||
final accessToken = result.data?['login']['accessToken'] as String;
|
||||
final refreshToken = result.data?['login']['refreshToken'] as String;
|
||||
|
||||
prefs.setString('accessToken', accessToken);
|
||||
prefs.setString('refreshToken', refreshToken);
|
||||
}
|
||||
|
||||
Future<String> _accessGuest() async {
|
||||
prefs = await SharedPreferences.getInstance();
|
||||
String? guestId = prefs.getString('guestId');
|
||||
guestId ??= const Uuid().v4();
|
||||
prefs.setString('guestId', guestId);
|
||||
|
||||
const String accessGuestMutation = r'''
|
||||
mutation AccessGuest($guest: AccessGuestModelInput!) {
|
||||
accessGuest(guest: $guest) {
|
||||
accessToken
|
||||
refreshToken
|
||||
}
|
||||
}
|
||||
''';
|
||||
|
||||
final MutationOptions options = MutationOptions(
|
||||
document: gql(accessGuestMutation),
|
||||
variables: <String, dynamic>{
|
||||
'guest': {'guestId': guestId},
|
||||
},
|
||||
);
|
||||
|
||||
final QueryResult result = await client.mutate(options);
|
||||
|
||||
if (result.hasException) {
|
||||
print(result.exception.toString());
|
||||
}
|
||||
|
||||
final String accessToken =
|
||||
result.data?['accessGuest']['accessToken'] as String;
|
||||
final String refreshToken =
|
||||
result.data?['accessGuest']['refreshToken'] as String;
|
||||
|
||||
prefs.setString('accessToken', accessToken);
|
||||
prefs.setString('refreshToken', refreshToken);
|
||||
|
||||
return accessToken;
|
||||
}
|
||||
|
||||
Future<String> _refreshAccessToken() async {
|
||||
var accessToken = prefs.getString('accessToken');
|
||||
var refreshToken = prefs.getString('refreshToken');
|
||||
|
||||
const String refreshAccessTokenMutation = r'''
|
||||
mutation RefreshAccessToken($model: TokensModelInput!) {
|
||||
refreshAccessToken(model: $model) {
|
||||
accessToken
|
||||
refreshToken
|
||||
}
|
||||
}
|
||||
''';
|
||||
|
||||
final QueryOptions options = QueryOptions(
|
||||
document: gql(refreshAccessTokenMutation),
|
||||
variables: <String, dynamic>{
|
||||
'model': {
|
||||
'accessToken': accessToken,
|
||||
'refreshToken': refreshToken,
|
||||
},
|
||||
},
|
||||
);
|
||||
|
||||
final QueryResult result = await client.query(options);
|
||||
|
||||
if (result.hasException) {
|
||||
print(result.exception.toString());
|
||||
}
|
||||
|
||||
accessToken = result.data?['refreshAccessToken']['accessToken'] as String;
|
||||
refreshToken = result.data?['refreshAccessToken']['refreshToken'] as String;
|
||||
|
||||
prefs.setString('accessToken', accessToken);
|
||||
prefs.setString('refreshToken', refreshToken);
|
||||
|
||||
return accessToken;
|
||||
}
|
||||
|
||||
String getIdFromAccessToken(String accessToken) {
|
||||
var decodedToken = JwtDecoder.decode(accessToken);
|
||||
return decodedToken[JwtClaims.id];
|
||||
}
|
||||
|
||||
String getEmailFromAccessToken(String accessToken) {
|
||||
var decodedToken = JwtDecoder.decode(accessToken);
|
||||
return decodedToken[JwtClaims.email];
|
||||
}
|
||||
|
||||
String getPhoneFromAccessToken(String accessToken) {
|
||||
var decodedToken = JwtDecoder.decode(accessToken);
|
||||
return decodedToken[JwtClaims.phone];
|
||||
}
|
||||
|
||||
// List<String> getRolesFromAccessToken(String accessToken) {
|
||||
// var decodedToken = JwtDecoder.decode(accessToken);
|
||||
// List<String> roles = [];
|
||||
// for (var role in decodedToken[JwtClaims.roles] as List<dynamic>) {
|
||||
// roles.add(role);
|
||||
// }
|
||||
// return roles;
|
||||
// }
|
||||
}
|
31
lib/network/product_service.dart
Normal file
@ -0,0 +1,31 @@
|
||||
import 'package:graphql/client.dart';
|
||||
import 'package:shopping_assistant_mobile_client/models/product.dart';
|
||||
import 'package:shopping_assistant_mobile_client/network/api_client.dart';
|
||||
|
||||
class ProductService {
|
||||
final ApiClient client = ApiClient();
|
||||
|
||||
Future<void> addProductToPersonalWishlist(Product product, String wishlisId) async {
|
||||
final options = MutationOptions(
|
||||
document: gql('''
|
||||
mutation AddProductToPersonalWishlist(\$wishlistId: String!, \$dto: ProductCreateDtoInput!) {
|
||||
addProductToPersonalWishlist(wishlistId: \$wishlistId, dto: \$dto) {
|
||||
|
||||
}
|
||||
}
|
||||
'''),
|
||||
variables: {'wishlistId': wishlisId,
|
||||
'dto': {
|
||||
'wasOpened': product.wasOpened,
|
||||
'url': product.url,
|
||||
'rating': product.rating,
|
||||
'price': product.price,
|
||||
'name': product.name,
|
||||
'imagesUrls': product.imageUrls,
|
||||
'description': product.description,
|
||||
}},
|
||||
);
|
||||
|
||||
final result = await client.mutate(options);
|
||||
}
|
||||
}
|
152
lib/network/search_service.dart
Normal file
@ -0,0 +1,152 @@
|
||||
// search_service.dart
|
||||
import 'dart:async';
|
||||
import 'package:logger/logger.dart';
|
||||
import 'package:graphql_flutter/graphql_flutter.dart';
|
||||
import '../models/enums/search_event_type.dart';
|
||||
import '../models/server_sent_event.dart';
|
||||
import '../network/api_client.dart';
|
||||
import '../screens/chat.dart';
|
||||
|
||||
const String startPersonalWishlistMutations = r'''
|
||||
mutation startPersonalWishlist($dto: WishlistCreateDtoInput!) {
|
||||
startPersonalWishlist(dto: $dto) {
|
||||
createdById, id, name, type
|
||||
}
|
||||
}
|
||||
''';
|
||||
|
||||
var logger = Logger();
|
||||
|
||||
SearchEventType type = SearchEventType.message;
|
||||
|
||||
class SearchService {
|
||||
final ApiClient client = ApiClient();
|
||||
|
||||
List<String> products = [];
|
||||
|
||||
late final _sseController = StreamController<ServerSentEvent>();
|
||||
|
||||
Stream<ServerSentEvent> get sseStream => _sseController.stream;
|
||||
|
||||
bool checkerForProduct() {
|
||||
return type == SearchEventType.product;
|
||||
}
|
||||
|
||||
bool checkerForSuggestion() {
|
||||
return type == SearchEventType.suggestion;
|
||||
}
|
||||
|
||||
String? wishlistId;
|
||||
|
||||
Future<String?> generateNameForPersonalWishlist(String wishlistId) async {
|
||||
final options = MutationOptions(
|
||||
document: gql('''
|
||||
mutation GenerateNameForPersonalWishlist(\$wishlistId: String!) {
|
||||
generateNameForPersonalWishlist(wishlistId: \$wishlistId) {
|
||||
id
|
||||
name
|
||||
}
|
||||
}
|
||||
'''),
|
||||
variables: {'wishlistId': wishlistId},
|
||||
);
|
||||
|
||||
final result = await client.mutate(options);
|
||||
|
||||
if (result != null && result.containsKey('generateNameForPersonalWishlist')) {
|
||||
final name = result['generateNameForPersonalWishlist']['name'];
|
||||
return name;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
Future<String> startPersonalWishlist(String message) async {
|
||||
logger.d('WISHLIST ID: $wishlistId');
|
||||
if (wishlistId == null) {
|
||||
final options = MutationOptions(
|
||||
document: gql(startPersonalWishlistMutations),
|
||||
variables: <String, dynamic>{
|
||||
'dto': {'firstMessageText': "What are you looking for?", 'type': 'Product'},
|
||||
},
|
||||
);
|
||||
|
||||
final result = await client.mutate(options);
|
||||
|
||||
if (result != null && result.containsKey('startPersonalWishlist')) {
|
||||
wishlistId = result['startPersonalWishlist']['id'];
|
||||
}
|
||||
}
|
||||
return wishlistId.toString();
|
||||
}
|
||||
|
||||
Future<void> sendMessages(String message) async {
|
||||
logger.d('WISHLIST ID: $wishlistId');
|
||||
if (wishlistId != null) {
|
||||
final sseStream = client.getServerSentEventStream(
|
||||
'api/productssearch/search/$wishlistId',
|
||||
{'text': message},
|
||||
);
|
||||
|
||||
await for (final chunk in sseStream) {
|
||||
print("Original chunk.data: ${chunk.event}");
|
||||
final cleanedMessage = chunk.data.replaceAll(RegExp(r'(^"|"$)'), '');
|
||||
|
||||
final event = ServerSentEvent(chunk.event, cleanedMessage);
|
||||
type = chunk.event;
|
||||
if(type == SearchEventType.product) {
|
||||
String pattern = r'[\\\"]';
|
||||
String product = event.data.replaceAll(RegExp(pattern), '');
|
||||
products.add(product);
|
||||
}
|
||||
_sseController.add(event);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Future<List<Message>> getMessagesFromPersonalWishlist(String wishlistIdPar, int pageNumber, int pageSize) async {
|
||||
final options = QueryOptions(
|
||||
document: gql('''
|
||||
query MessagesPageFromPersonalWishlist(\$wishlistId: String!, \$pageNumber: Int!, \$pageSize: Int!) {
|
||||
messagesPageFromPersonalWishlist(wishlistId: \$wishlistId, pageNumber: \$pageNumber, pageSize: \$pageSize) {
|
||||
items {
|
||||
id
|
||||
text
|
||||
role
|
||||
createdById
|
||||
}
|
||||
}
|
||||
}
|
||||
'''),
|
||||
variables: {
|
||||
'wishlistId': wishlistIdPar,
|
||||
'pageNumber': pageNumber,
|
||||
'pageSize': pageSize,
|
||||
},
|
||||
);
|
||||
|
||||
logger.d("DOCUMENT: ${options.document}");
|
||||
|
||||
final result = await client.query(options);
|
||||
|
||||
print("RESULT: ${result}");
|
||||
print(result);
|
||||
if (result != null &&
|
||||
result.containsKey('messagesPageFromPersonalWishlist') &&
|
||||
result['messagesPageFromPersonalWishlist'] != null &&
|
||||
result['messagesPageFromPersonalWishlist']['items'] != null) {
|
||||
final List<dynamic> items = result['messagesPageFromPersonalWishlist']['items'];
|
||||
|
||||
final List<Message> messages = items.map((item) {
|
||||
return Message(
|
||||
text: item['text'],
|
||||
role: item['role'],
|
||||
isProduct: false,
|
||||
);
|
||||
}).toList();
|
||||
|
||||
return messages;
|
||||
}
|
||||
return [];
|
||||
}
|
||||
}
|
287
lib/screens/cards.dart
Normal file
@ -0,0 +1,287 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_svg/svg.dart';
|
||||
import 'package:shopping_assistant_mobile_client/models/product.dart';
|
||||
import 'package:shopping_assistant_mobile_client/network/product_service.dart';
|
||||
|
||||
|
||||
class Cards extends StatefulWidget {
|
||||
final String wishlistId;
|
||||
final String wishlistName;
|
||||
List<String> inputProducts = [];
|
||||
List<Product> products = [];
|
||||
|
||||
Cards({required this.wishlistName, required this.inputProducts, required this.wishlistId});
|
||||
|
||||
@override
|
||||
_CardsState createState() => _CardsState();
|
||||
}
|
||||
|
||||
class _CardsState extends State<Cards> {
|
||||
int currentProduct = 0;
|
||||
ProductService productService = ProductService();
|
||||
List<Product> cart = [];
|
||||
|
||||
@override
|
||||
void initState(){
|
||||
for(String productName in this.widget.inputProducts){
|
||||
this.widget.products.add(
|
||||
Product(
|
||||
id: '',
|
||||
name: productName,
|
||||
url: 'link',
|
||||
imageUrls: [],
|
||||
rating: 0.0,
|
||||
price: 0.0,
|
||||
description: '',
|
||||
wasOpened: false,
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Widget buildRatingStars(double rating) {
|
||||
int whole = rating.floor();
|
||||
double fractal = rating - whole;
|
||||
|
||||
List<Widget> stars = [];
|
||||
|
||||
for (int i = 0; i < 5; i++) {
|
||||
if (i < whole) {
|
||||
// Whole star
|
||||
stars.add(SvgPicture.asset(
|
||||
'assets/icons/star.svg',
|
||||
width: 19,
|
||||
height: 19,
|
||||
));
|
||||
} else if (fractal != 0.0) {
|
||||
// Half star
|
||||
stars.add(SvgPicture.asset(
|
||||
'assets/icons/half-star.svg',
|
||||
width: 19,
|
||||
height: 19,
|
||||
));
|
||||
fractal -= fractal;
|
||||
} else {
|
||||
// Empty star
|
||||
stars.add(SvgPicture.asset(
|
||||
'assets/icons/empty-star.svg',
|
||||
width: 19,
|
||||
height: 19,
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
return Row(
|
||||
children: stars,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool hasCards = widget.products.isNotEmpty;
|
||||
bool isLastProduct = currentProduct == widget.products.length;
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(widget.wishlistName),
|
||||
centerTitle: true,
|
||||
),
|
||||
body: Center(
|
||||
child: Stack(
|
||||
children: [
|
||||
if (currentProduct < widget.products.length - 1)
|
||||
Positioned(
|
||||
child: Transform.rotate(
|
||||
angle: -5 * 3.1415926535 / 180,
|
||||
child: Container(
|
||||
width: 300,
|
||||
height: 600,
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
color: Colors.white,
|
||||
border: Border.all(
|
||||
color: Color(0xFF009FFF),
|
||||
),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Color(0xFF0165FF),
|
||||
blurRadius: 8.0,
|
||||
spreadRadius: 0.0,
|
||||
offset: Offset(0, 0),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
Positioned(
|
||||
child: Container(
|
||||
width: 300,
|
||||
height: 600,
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
color: Colors.white,
|
||||
border: Border.all(
|
||||
color: Color(0xFF009FFF),
|
||||
),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Color(0xFF0165FF),
|
||||
blurRadius: 8.0,
|
||||
spreadRadius: 0.0,
|
||||
offset: Offset(0, 0),
|
||||
),
|
||||
],
|
||||
),
|
||||
child: Column(
|
||||
children: [
|
||||
Container(
|
||||
height: 300,
|
||||
decoration: BoxDecoration(
|
||||
image: DecorationImage(
|
||||
image: AssetImage('assets/img/default-white.png'), // Replace with your image path
|
||||
fit: BoxFit.scaleDown,
|
||||
),
|
||||
),
|
||||
),
|
||||
SizedBox(height: 10),
|
||||
Text(
|
||||
currentProduct < widget.products.length
|
||||
? widget.products[currentProduct].name
|
||||
: "The cards ended.",
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 20,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
),
|
||||
if (currentProduct == widget.products.length)
|
||||
Text(
|
||||
"Swipe right to show more or left to exit.",
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 17,
|
||||
fontWeight: FontWeight.w400,
|
||||
),
|
||||
),
|
||||
SizedBox(height: 10),
|
||||
if (!isLastProduct)
|
||||
Column(
|
||||
children: [
|
||||
Text(
|
||||
currentProduct < widget.products.length
|
||||
? widget.products[currentProduct].description
|
||||
: "Product description not available.",
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 15,
|
||||
fontWeight: FontWeight.w400,
|
||||
),
|
||||
),
|
||||
SizedBox(height: 10),
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
buildRatingStars(currentProduct < widget.products.length
|
||||
? widget.products[currentProduct].rating
|
||||
: 0.0),
|
||||
SizedBox(width: 20),
|
||||
Text(
|
||||
'\$${currentProduct < widget.products.length ? widget.products[currentProduct].price : "-"}',
|
||||
style: TextStyle(
|
||||
fontSize: 15,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
SizedBox(height: 50),
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||
children: [
|
||||
IconButton(
|
||||
icon: currentProduct < widget.products.length
|
||||
? SvgPicture.asset(
|
||||
'assets/icons/x.svg',
|
||||
width: 30,
|
||||
height: 30,
|
||||
)
|
||||
: SvgPicture.asset(
|
||||
'assets/icons/exit-cards.svg',
|
||||
width: 30,
|
||||
height: 30,
|
||||
),
|
||||
onPressed: () {
|
||||
if (currentProduct < widget.products.length) {
|
||||
showNextProduct();
|
||||
} else {
|
||||
Navigator.of(context).pop();
|
||||
}
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: SvgPicture.asset(
|
||||
'assets/icons/back.svg',
|
||||
width: 30,
|
||||
height: 30,
|
||||
),
|
||||
onPressed: () {
|
||||
showPreviousProduct();
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: currentProduct < widget.products.length
|
||||
? SvgPicture.asset(
|
||||
'assets/icons/heart.svg',
|
||||
width: 30,
|
||||
height: 30,
|
||||
color: Colors.blue,
|
||||
)
|
||||
: SvgPicture.asset(
|
||||
'assets/icons/add-products.svg',
|
||||
width: 30,
|
||||
height: 30,
|
||||
),
|
||||
onPressed: () async {
|
||||
if (currentProduct < widget.products.length) {
|
||||
if (!cart.contains(widget.products[currentProduct])) {
|
||||
await productService.addProductToPersonalWishlist(
|
||||
widget.products[currentProduct],
|
||||
widget.wishlistId,
|
||||
);
|
||||
cart.add(widget.products[currentProduct]);
|
||||
}
|
||||
showNextProduct();
|
||||
}
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
void showNextProduct() {
|
||||
setState(() {
|
||||
if(currentProduct < widget.products.length) {
|
||||
++currentProduct;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void showPreviousProduct() {
|
||||
setState(() {
|
||||
if(currentProduct > 0 ) {
|
||||
--currentProduct;
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
230
lib/screens/cart.dart
Normal file
@ -0,0 +1,230 @@
|
||||
import 'package:cached_network_image/cached_network_image.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_svg/svg.dart';
|
||||
import 'package:graphql/client.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
import 'package:shopping_assistant_mobile_client/models/product.dart';
|
||||
import 'package:shopping_assistant_mobile_client/network/api_client.dart';
|
||||
|
||||
|
||||
class CartScreen extends StatefulWidget {
|
||||
CartScreen({super.key, required this.wishlistId});
|
||||
|
||||
final String wishlistId;
|
||||
|
||||
@override
|
||||
State<CartScreen> createState() => _CartScreenState(wishlistId: wishlistId);
|
||||
}
|
||||
|
||||
class _CartScreenState extends State<CartScreen> {
|
||||
_CartScreenState({required this.wishlistId});
|
||||
|
||||
var client = ApiClient();
|
||||
|
||||
final String wishlistId;
|
||||
|
||||
late Future _productsFuture;
|
||||
late List<Product> _products;
|
||||
|
||||
@override
|
||||
void initState(){
|
||||
super.initState();
|
||||
_productsFuture = _fetchProducts();
|
||||
}
|
||||
|
||||
Future _fetchProducts() async {
|
||||
const String productsPageFromPersonalWishlistQuery = r'''
|
||||
query ProductsPageFromPersonalWishlist($wishlistId: String!, $pageNumber: Int!, $pageSize: Int!) {
|
||||
productsPageFromPersonalWishlist(
|
||||
wishlistId: $wishlistId,
|
||||
pageNumber: $pageNumber,
|
||||
pageSize: $pageSize
|
||||
) {
|
||||
items {
|
||||
id
|
||||
url
|
||||
name
|
||||
rating
|
||||
price
|
||||
imagesUrls
|
||||
}
|
||||
}
|
||||
}''';
|
||||
|
||||
QueryOptions queryOptions = QueryOptions(
|
||||
document: gql(productsPageFromPersonalWishlistQuery),
|
||||
variables: <String, dynamic>{
|
||||
'wishlistId': wishlistId,
|
||||
'pageNumber': 1,
|
||||
'pageSize': 10,
|
||||
});
|
||||
|
||||
var result = await client.query(queryOptions);
|
||||
print(result);
|
||||
|
||||
_products = List<Map<String, dynamic>>.from(
|
||||
result?['productsPageFromPersonalWishlist']['items'])
|
||||
.map((e) => Product.fromJson(e))
|
||||
.toList();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context){
|
||||
return FutureBuilder(
|
||||
future: _productsFuture,
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.hasError) {
|
||||
return Center(
|
||||
child: Text('Error: ${snapshot.error}'),
|
||||
);
|
||||
} else if (snapshot.connectionState == ConnectionState.done) {
|
||||
// Data loaded successfully, display the widget
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text("Cart"),
|
||||
centerTitle: true,
|
||||
leading: IconButton(
|
||||
icon: Icon(Icons.arrow_back),
|
||||
onPressed: () {
|
||||
Navigator.pop(context);
|
||||
},
|
||||
),
|
||||
),
|
||||
body: _products.length == 0 ?
|
||||
Center(child: Text("The cart is empty", style: TextStyle(fontSize: 18),),)
|
||||
: ListView.builder(
|
||||
padding: EdgeInsets.symmetric(vertical: 30),
|
||||
itemCount: _products.length,
|
||||
itemBuilder: (context, index){
|
||||
return CartItem(product: _products[index]);
|
||||
}
|
||||
),
|
||||
backgroundColor: Colors.white,
|
||||
);
|
||||
};
|
||||
|
||||
return Center(
|
||||
child: CircularProgressIndicator(),
|
||||
);
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class CartItem extends StatelessWidget{
|
||||
CartItem({
|
||||
super.key,
|
||||
required Product product,
|
||||
}) : _product = product;
|
||||
|
||||
final Product _product;
|
||||
|
||||
|
||||
Widget _buildRatingStar(int index) {
|
||||
int whole = _product.rating.floor().toInt();
|
||||
double fractional = _product.rating - whole;
|
||||
|
||||
if (index < whole) {
|
||||
return Icon(Icons.star, color: Colors.yellow[600], size: 20);
|
||||
}
|
||||
if (fractional >= 0.25 && fractional <= 0.75) {
|
||||
return Icon(Icons.star_half, color: Colors.yellow[600], size: 20);
|
||||
}
|
||||
if (fractional > 0.75) {
|
||||
return Icon(Icons.star, color: Colors.yellow[600], size: 20);
|
||||
}else {
|
||||
return Icon(Icons.star_border, color: Colors.grey, size: 20);
|
||||
}
|
||||
}
|
||||
|
||||
List<Widget> _buildRatingStars() {
|
||||
List<Widget> stars = [];
|
||||
for (int i = 0; i < 5; i++) {
|
||||
stars.add(_buildRatingStar(i));
|
||||
}
|
||||
return stars;
|
||||
}
|
||||
|
||||
|
||||
Future<void> _launchUrl(String url) async {
|
||||
final Uri uri = Uri.parse(url);
|
||||
if (!await launchUrl(uri)) throw 'Could not launch $url';
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Container(
|
||||
width: double.infinity,
|
||||
height: 140,
|
||||
margin: EdgeInsets.all(10),
|
||||
padding: EdgeInsets.all(10),
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.white,
|
||||
borderRadius: BorderRadius.all(Radius.circular(10)),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.grey.withOpacity(0.5),
|
||||
spreadRadius: 2,
|
||||
blurRadius: 2,
|
||||
offset: Offset(1, 2),
|
||||
)
|
||||
]
|
||||
),
|
||||
child: Row(
|
||||
crossAxisAlignment: CrossAxisAlignment.end,
|
||||
children: [
|
||||
Container(
|
||||
width: 100,
|
||||
alignment: Alignment.center,
|
||||
child: CachedNetworkImage(
|
||||
imageUrl: _product.imageUrls[0],
|
||||
placeholder: (context, url) => CircularProgressIndicator(),
|
||||
errorWidget: (context, url, error) => Container(color: Colors.white,),
|
||||
),
|
||||
),
|
||||
SizedBox(width: 20),
|
||||
Expanded(
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
_product.name,
|
||||
style: const TextStyle(fontSize: 13),
|
||||
overflow: TextOverflow.ellipsis,
|
||||
maxLines: 3,
|
||||
),
|
||||
Row(
|
||||
children: [
|
||||
Row(
|
||||
children: <Widget>[
|
||||
Text(_product.rating.toStringAsFixed(1), style: TextStyle(fontSize: 14))
|
||||
] + _buildRatingStars(),
|
||||
),
|
||||
Text("\$" + _product.price.toStringAsFixed(2), style: TextStyle(fontSize: 14)),
|
||||
],
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
),
|
||||
Container(
|
||||
width: double.infinity,
|
||||
height: 35,
|
||||
child: ElevatedButton.icon(
|
||||
onPressed: () => _launchUrl(_product.url),
|
||||
style: ElevatedButton.styleFrom(
|
||||
backgroundColor: Colors.blue,
|
||||
padding: EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||
),
|
||||
icon: SvgPicture.asset("assets/icons/amazon.svg", height: 15),
|
||||
label: Text(""),
|
||||
),
|
||||
)
|
||||
],
|
||||
)
|
||||
)
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
489
lib/screens/chat.dart
Normal file
@ -0,0 +1,489 @@
|
||||
// search_screen.dart
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_spinkit/flutter_spinkit.dart';
|
||||
import 'package:logger/logger.dart';
|
||||
import 'package:shopping_assistant_mobile_client/network/search_service.dart';
|
||||
import 'package:shopping_assistant_mobile_client/screens/cards.dart';
|
||||
|
||||
class Message {
|
||||
final String text;
|
||||
final String role;
|
||||
bool isProduct;
|
||||
bool isSuggestion;
|
||||
|
||||
Message({required this.text, this.role = "", this.isProduct = false, this.isSuggestion = false});
|
||||
}
|
||||
|
||||
class MessageBubble extends StatelessWidget {
|
||||
final String message;
|
||||
final bool isOutgoing;
|
||||
final bool isProduct;
|
||||
|
||||
MessageBubble({required this.message, this.isOutgoing = true, this.isProduct = false});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Align(
|
||||
alignment: isOutgoing ? Alignment.centerRight : Alignment.centerLeft,
|
||||
child: Container(
|
||||
margin: const EdgeInsets.all(8.0),
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
constraints: BoxConstraints(
|
||||
maxWidth: 300.0,
|
||||
),
|
||||
decoration: BoxDecoration(
|
||||
color: isOutgoing ? Colors.blue : Colors.grey[200],
|
||||
borderRadius: BorderRadius.circular(10.0),
|
||||
),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
message.trim(),
|
||||
style: TextStyle(
|
||||
color: isOutgoing ? Colors.white : Colors.black,
|
||||
fontSize: 18.0,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class ChatScreen extends StatefulWidget {
|
||||
String wishlistId;
|
||||
String wishlistName;
|
||||
bool openedFromBottomBar;
|
||||
|
||||
ChatScreen({Key? key, required this.wishlistId, required this.wishlistName, required this.openedFromBottomBar}) : super(key: key);
|
||||
|
||||
@override
|
||||
State createState() => ChatScreenState();
|
||||
}
|
||||
|
||||
class ChatScreenState extends State<ChatScreen> {
|
||||
var logger = Logger();
|
||||
SearchService _searchService = SearchService();
|
||||
List<Message> messages = [];
|
||||
TextEditingController _messageController = TextEditingController();
|
||||
List<String> suggestions = [];
|
||||
bool showBackButton = false;
|
||||
bool buttonsVisible = true;
|
||||
bool isSendButtonEnabled = false;
|
||||
bool showButtonsContainer = true;
|
||||
bool isWaitingForResponse = false;
|
||||
ScrollController _scrollController = ScrollController();
|
||||
late Widget appBarTitle;
|
||||
|
||||
void initState() {
|
||||
super.initState();
|
||||
if (widget.openedFromBottomBar) {
|
||||
_resetState();
|
||||
}
|
||||
appBarTitle = Text('New Chat', style: TextStyle(fontSize: 18.0));
|
||||
_searchService.sseStream.listen((event) {
|
||||
_handleSSEMessage(Message(text: '${event.data}'));
|
||||
});
|
||||
Future.delayed(Duration(milliseconds: 2000));
|
||||
if(!widget.wishlistId.isEmpty)
|
||||
{
|
||||
_loadPreviousMessages();
|
||||
showBackButton = true;
|
||||
showButtonsContainer = false;
|
||||
buttonsVisible = false;
|
||||
}
|
||||
}
|
||||
|
||||
void _resetState() {
|
||||
widget.wishlistId = '';
|
||||
widget.wishlistName = '';
|
||||
_searchService = SearchService();
|
||||
messages = [];
|
||||
_messageController = TextEditingController();
|
||||
showBackButton = false;
|
||||
buttonsVisible = true;
|
||||
isSendButtonEnabled = false;
|
||||
showButtonsContainer = true;
|
||||
isWaitingForResponse = false;
|
||||
_scrollController = ScrollController();
|
||||
appBarTitle = const Text('New Chat', style: TextStyle(fontSize: 18.0));
|
||||
}
|
||||
|
||||
Future<void> _loadPreviousMessages() async {
|
||||
final pageNumber = 1;
|
||||
final pageSize = 200;
|
||||
logger.d('WISH LIST ID: $widget.wishlistId');
|
||||
appBarTitle = Text(widget.wishlistName, style: TextStyle(fontSize: 18.0));
|
||||
try {
|
||||
final previousMessages = await _searchService.getMessagesFromPersonalWishlist(widget.wishlistId, pageNumber, pageSize);
|
||||
final reversedMessages = previousMessages.reversed.toList();
|
||||
setState(() {
|
||||
messages.addAll(reversedMessages);
|
||||
});
|
||||
logger.d('Previous Messages: $previousMessages');
|
||||
|
||||
for(final message in messages)
|
||||
{
|
||||
logger.d("MESSAGES TEXT: ${message.text}");
|
||||
logger.d("MESSAGES ROLE: ${message.role}");
|
||||
}
|
||||
} catch (error) {
|
||||
logger.d('Error loading previous messages: $error');
|
||||
}
|
||||
}
|
||||
|
||||
void _handleSSEMessage(Message message) {
|
||||
setState(() {
|
||||
isWaitingForResponse = true;
|
||||
final lastMessage = messages.isNotEmpty ? messages.last : null;
|
||||
message.isProduct = _searchService.checkerForProduct();
|
||||
message.isSuggestion = _searchService.checkerForSuggestion();
|
||||
if(message.isSuggestion){
|
||||
suggestions.add(message.text);
|
||||
}
|
||||
logger.d("Product status: ${message.isProduct}");
|
||||
logger.d("Suggestion status: ${message.isSuggestion}");
|
||||
logger.d("Message text: ${message.text}");
|
||||
if (lastMessage != null && lastMessage.role != "User" && message.role != "User" && !message.isSuggestion) {
|
||||
String fullMessageText = lastMessage.text + message.text;
|
||||
fullMessageText = fullMessageText.replaceAll("\\n", "");
|
||||
logger.d("fullMessageText: $fullMessageText");
|
||||
final updatedMessage = Message(
|
||||
text: fullMessageText,
|
||||
role: "Application",
|
||||
isProduct: message.isProduct);
|
||||
messages.removeLast();
|
||||
messages.add(updatedMessage);
|
||||
} else {
|
||||
String messageText = message.text.replaceAll("\\n", "");
|
||||
if (!message.isSuggestion) {
|
||||
messages.add(Message(text: messageText, role: message.role, isProduct: message.isProduct, isSuggestion: message.isSuggestion));
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
setState(() {
|
||||
isWaitingForResponse = false;
|
||||
});
|
||||
_scrollToBottom();
|
||||
}
|
||||
|
||||
Future<void> updateChatTitle(String wishlistId) async {
|
||||
final wishlistName = await _searchService.generateNameForPersonalWishlist(wishlistId);
|
||||
if (wishlistName != null) {
|
||||
setState(() {
|
||||
appBarTitle = Text(wishlistName, style: TextStyle(fontSize: 18.0));
|
||||
this.widget.wishlistName = wishlistName;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _startPersonalWishlist(String message) async {
|
||||
setState(() {
|
||||
buttonsVisible = false;
|
||||
showButtonsContainer = false;
|
||||
isWaitingForResponse = true;
|
||||
});
|
||||
widget.wishlistId = await _searchService.startPersonalWishlist(message);
|
||||
await _sendMessageToAPI(message);
|
||||
await updateChatTitle(_searchService.wishlistId.toString());
|
||||
_scrollToBottom();
|
||||
|
||||
setState(() {
|
||||
isWaitingForResponse = false;
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> _sendMessageToAPI(String message)async {
|
||||
setState(() {
|
||||
buttonsVisible = false;
|
||||
showButtonsContainer = false;
|
||||
isWaitingForResponse = true;
|
||||
});
|
||||
await _searchService.sendMessages(message);
|
||||
_scrollToBottom();
|
||||
|
||||
setState(() {
|
||||
if(_searchService.checkerForProduct()){
|
||||
messages.removeLast();
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => Cards(wishlistName: this.widget.wishlistName, inputProducts: this._searchService.products, wishlistId: this.widget.wishlistId,),
|
||||
),
|
||||
);
|
||||
}
|
||||
isWaitingForResponse = false;
|
||||
});
|
||||
}
|
||||
|
||||
void _sendMessage() {
|
||||
final message = _messageController.text;
|
||||
logger.d('WISH LIST ID: $widget.wishlistId');
|
||||
if (widget.wishlistId.isEmpty) {
|
||||
setState(() {
|
||||
messages.add(Message(text: "What are you looking for?", role: "Application"));
|
||||
messages.add(Message(text: message, role: "User"));
|
||||
});
|
||||
_startPersonalWishlist(message);
|
||||
} else {
|
||||
setState(() {
|
||||
messages.add(Message(text: message, role: "User"));
|
||||
});
|
||||
_searchService.wishlistId = widget.wishlistId.toString();
|
||||
_sendMessageToAPI(message);
|
||||
}
|
||||
|
||||
_messageController.clear();
|
||||
suggestions.clear();
|
||||
_scrollToBottom();
|
||||
}
|
||||
|
||||
void _handleSuggestion(String suggestion) {
|
||||
_messageController.text = suggestion;
|
||||
_sendMessage();
|
||||
suggestions.clear();
|
||||
}
|
||||
|
||||
|
||||
void _scrollToBottom() {
|
||||
_scrollController.animateTo(
|
||||
_scrollController.position.maxScrollExtent,
|
||||
duration: Duration(milliseconds: 300),
|
||||
curve: Curves.easeOut,
|
||||
);
|
||||
}
|
||||
|
||||
void _showGiftNotAvailable() {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
title: Text('Gift Functionality'),
|
||||
content: Text('This function is currently unavailable.'),
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
child: Text('OK'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
],
|
||||
);
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
Widget _generateSuggestionButtons() {
|
||||
return Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Padding(
|
||||
padding: EdgeInsets.symmetric(horizontal: 16, vertical: 8),
|
||||
child: Text(
|
||||
'Several possible options',
|
||||
style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold, color: Colors.grey),
|
||||
),
|
||||
),
|
||||
SingleChildScrollView(
|
||||
scrollDirection: Axis.horizontal,
|
||||
child: Row(
|
||||
children: suggestions.map((suggestion) {
|
||||
return Padding(
|
||||
padding: EdgeInsets.symmetric(horizontal: 8, vertical: 5),
|
||||
child: ElevatedButton(
|
||||
onPressed: () {
|
||||
_handleSuggestion(suggestion);
|
||||
},
|
||||
style: ElevatedButton.styleFrom(
|
||||
padding: EdgeInsets.symmetric(horizontal: 30, vertical: 16),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
primary: Colors.white,
|
||||
onPrimary: Colors.blue,
|
||||
side: BorderSide(color: Colors.blue, width: 2.0),
|
||||
),
|
||||
child: Text(suggestion, style: TextStyle(color: Colors.black)),
|
||||
),
|
||||
);
|
||||
}).toList(),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: appBarTitle,
|
||||
centerTitle: true,
|
||||
leading: showBackButton
|
||||
? IconButton(
|
||||
icon: Icon(Icons.arrow_back),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
)
|
||||
: null,
|
||||
),
|
||||
body: Column(
|
||||
children: <Widget>[
|
||||
Visibility(
|
||||
visible: buttonsVisible,
|
||||
child: Align(
|
||||
alignment: Alignment.topCenter,
|
||||
child: Column(
|
||||
children: [
|
||||
Text(
|
||||
'What are you looking for?',
|
||||
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||
),
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: <Widget>[
|
||||
ElevatedButton(
|
||||
onPressed: () {
|
||||
print('Product button pressed');
|
||||
},
|
||||
style: ElevatedButton.styleFrom(
|
||||
padding: EdgeInsets.symmetric(
|
||||
horizontal: 30, vertical: 16),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
primary: Colors.blue,
|
||||
onPrimary: Colors.white,
|
||||
),
|
||||
child: Text('Product'),
|
||||
),
|
||||
SizedBox(width: 16.0),
|
||||
ElevatedButton(
|
||||
onPressed: _showGiftNotAvailable,
|
||||
style: ElevatedButton.styleFrom(
|
||||
padding: EdgeInsets.symmetric(
|
||||
horizontal: 30, vertical: 16),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
primary: Colors.white,
|
||||
onPrimary: Colors.black,
|
||||
),
|
||||
child: Text('Gift'),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
SizedBox(height: 16.0),
|
||||
Visibility(
|
||||
visible: showButtonsContainer,
|
||||
child: Container(
|
||||
margin: const EdgeInsets.all(8.0),
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: <Widget>[
|
||||
Container(
|
||||
margin: EdgeInsets.all(8),
|
||||
child: ElevatedButton(
|
||||
onPressed: () {
|
||||
_messageController.text = 'Christmas gift🎁';
|
||||
_sendMessage();
|
||||
},
|
||||
style: ElevatedButton.styleFrom(
|
||||
padding: EdgeInsets.symmetric(horizontal: 30, vertical: 16),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
primary: Colors.white,
|
||||
onPrimary: Colors.blue,
|
||||
side: BorderSide(color: Colors.blue, width: 2.0),
|
||||
),
|
||||
child: Text('Christmas gift🎁', style: TextStyle(color: Colors.grey)),
|
||||
),
|
||||
),
|
||||
Container(
|
||||
margin: EdgeInsets.all(8),
|
||||
child: ElevatedButton(
|
||||
onPressed: () {
|
||||
_messageController.text = 'Birthday gift🎉';
|
||||
_sendMessage();
|
||||
},
|
||||
style: ElevatedButton.styleFrom(
|
||||
padding: EdgeInsets.symmetric(horizontal: 30, vertical: 16),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
primary: Colors.white,
|
||||
onPrimary: Colors.blue,
|
||||
side: BorderSide(color: Colors.blue, width: 2.0),
|
||||
),
|
||||
child: Text('Birthday gift🎉', style: TextStyle(color: Colors.grey)),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
child: ListView.builder(
|
||||
controller: _scrollController,
|
||||
reverse: false,
|
||||
itemCount: messages.length,
|
||||
itemBuilder: (context, index) {
|
||||
final message = messages[index];
|
||||
return MessageBubble(
|
||||
message: message.text,
|
||||
isOutgoing: message.role == "User",
|
||||
isProduct: message.isProduct,
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
if (isWaitingForResponse)
|
||||
SpinKitFadingCircle(
|
||||
color: Colors.blue,
|
||||
size: 25.0,
|
||||
),
|
||||
if (suggestions.isNotEmpty)
|
||||
_generateSuggestionButtons(),
|
||||
Container(
|
||||
margin: const EdgeInsets.all(8.0),
|
||||
child: Row(
|
||||
children: <Widget>[
|
||||
Expanded(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 15.0), // Adjust the left padding as needed
|
||||
child: TextField(
|
||||
controller: _messageController,
|
||||
onChanged: (text) {
|
||||
setState(() {
|
||||
isSendButtonEnabled = text.isNotEmpty;
|
||||
});
|
||||
},
|
||||
decoration: InputDecoration(
|
||||
hintText: 'Enter your message...',
|
||||
contentPadding: EdgeInsets.symmetric(vertical: 20.0),
|
||||
),
|
||||
),
|
||||
|
||||
),
|
||||
),
|
||||
IconButton(
|
||||
icon: Icon(Icons.send),
|
||||
onPressed: isSendButtonEnabled ? _sendMessage : null,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|