diff --git a/evently/ios/Podfile.lock b/evently/ios/Podfile.lock new file mode 100644 index 0000000000..b910d3869b --- /dev/null +++ b/evently/ios/Podfile.lock @@ -0,0 +1,118 @@ +PODS: + - DKImagePickerController/Core (4.3.9): + - DKImagePickerController/ImageDataManager + - DKImagePickerController/Resource + - DKImagePickerController/ImageDataManager (4.3.9) + - DKImagePickerController/PhotoGallery (4.3.9): + - DKImagePickerController/Core + - DKPhotoGallery + - DKImagePickerController/Resource (4.3.9) + - DKPhotoGallery (0.0.19): + - DKPhotoGallery/Core (= 0.0.19) + - DKPhotoGallery/Model (= 0.0.19) + - DKPhotoGallery/Preview (= 0.0.19) + - DKPhotoGallery/Resource (= 0.0.19) + - SDWebImage + - SwiftyGif + - DKPhotoGallery/Core (0.0.19): + - DKPhotoGallery/Model + - DKPhotoGallery/Preview + - SDWebImage + - SwiftyGif + - DKPhotoGallery/Model (0.0.19): + - SDWebImage + - SwiftyGif + - DKPhotoGallery/Preview (0.0.19): + - DKPhotoGallery/Model + - DKPhotoGallery/Resource + - SDWebImage + - SwiftyGif + - DKPhotoGallery/Resource (0.0.19): + - SDWebImage + - SwiftyGif + - file_picker (0.0.1): + - DKImagePickerController/PhotoGallery + - Flutter + - Flutter (1.0.0) + - image_cropper (0.0.4): + - Flutter + - TOCropViewController (~> 2.6.1) + - network_info_plus (0.0.1): + - Flutter + - path_provider_foundation (0.0.1): + - Flutter + - FlutterMacOS + - SDWebImage (5.19.2): + - SDWebImage/Core (= 5.19.2) + - SDWebImage/Core (5.19.2) + - shared_preferences_foundation (0.0.1): + - Flutter + - FlutterMacOS + - sqflite (0.0.3): + - Flutter + - FlutterMacOS + - SwiftyGif (5.4.5) + - TOCropViewController (2.6.1) + - uni_links (0.0.1): + - Flutter + - url_launcher_ios (0.0.1): + - Flutter + +DEPENDENCIES: + - file_picker (from `.symlinks/plugins/file_picker/ios`) + - Flutter (from `Flutter`) + - image_cropper (from `.symlinks/plugins/image_cropper/ios`) + - network_info_plus (from `.symlinks/plugins/network_info_plus/ios`) + - 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/darwin`) + - uni_links (from `.symlinks/plugins/uni_links/ios`) + - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) + +SPEC REPOS: + trunk: + - DKImagePickerController + - DKPhotoGallery + - SDWebImage + - SwiftyGif + - TOCropViewController + +EXTERNAL SOURCES: + file_picker: + :path: ".symlinks/plugins/file_picker/ios" + Flutter: + :path: Flutter + image_cropper: + :path: ".symlinks/plugins/image_cropper/ios" + network_info_plus: + :path: ".symlinks/plugins/network_info_plus/ios" + 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/darwin" + uni_links: + :path: ".symlinks/plugins/uni_links/ios" + url_launcher_ios: + :path: ".symlinks/plugins/url_launcher_ios/ios" + +SPEC CHECKSUMS: + DKImagePickerController: 946cec48c7873164274ecc4624d19e3da4c1ef3c + DKPhotoGallery: b3834fecb755ee09a593d7c9e389d8b5d6deed60 + file_picker: ce3938a0df3cc1ef404671531facef740d03f920 + Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 + image_cropper: a3291c624a953049bc6a02e1f8c8ceb162a24b25 + network_info_plus: 9d930145451916919786087c4173226363616071 + path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46 + SDWebImage: dfe95b2466a9823cf9f0c6d01217c06550d7b29a + shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78 + sqflite: 673a0e54cc04b7d6dba8d24fb8095b31c3a99eec + SwiftyGif: 706c60cf65fa2bc5ee0313beece843c8eb8194d4 + TOCropViewController: edfd4f25713d56905ad1e0b9f5be3fbe0f59c863 + uni_links: d97da20c7701486ba192624d99bffaaffcfc298a + url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe + +PODFILE CHECKSUM: 819463e6a0290f5a72f145ba7cde16e8b6ef0796 + +COCOAPODS: 1.15.2 diff --git a/evently/ios/Runner.xcodeproj/project.pbxproj b/evently/ios/Runner.xcodeproj/project.pbxproj index 39d6832f04..6d15a2b043 100644 --- a/evently/ios/Runner.xcodeproj/project.pbxproj +++ b/evently/ios/Runner.xcodeproj/project.pbxproj @@ -10,7 +10,9 @@ 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 */; }; + 654D85D1376A41A40FD0D375 /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 7B7863E15540969EA0830F56 /* Pods_RunnerTests.framework */; }; 74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; + 7D5647AC50CD435BD3697028 /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 1ECE69C8E6FF389C86529C4E /* Pods_Runner.framework */; }; 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 */; }; @@ -40,14 +42,19 @@ /* End PBXCopyFilesBuildPhase section */ /* Begin PBXFileReference section */ + 0E68AA088CC54A32B209E5A9 /* 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 = ""; }; 1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = ""; }; 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = ""; }; + 1ECE69C8E6FF389C86529C4E /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; }; 331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = ""; }; 331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = ""; }; + 3EBDA4956A7A8415A5A9EC21 /* 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 = ""; }; + 4D2E1B9CE910E7CA4F2E7824 /* 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 = ""; }; 74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = ""; }; 74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; 7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = ""; }; + 7B7863E15540969EA0830F56 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; }; 9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = ""; }; 9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = ""; }; 97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; @@ -55,19 +62,45 @@ 97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; 97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; 97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; + B3AEAC0457BAA712515513DB /* 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 = ""; }; + D74402E1D255FC3822687365 /* 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 = ""; }; + D788EAF91ED3ED0E558E835C /* 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 = ""; }; /* End PBXFileReference section */ /* Begin PBXFrameworksBuildPhase section */ + 8FC1627F69027DB10219A05D /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + 654D85D1376A41A40FD0D375 /* Pods_RunnerTests.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; 97C146EB1CF9000F007C117D /* Frameworks */ = { isa = PBXFrameworksBuildPhase; buildActionMask = 2147483647; files = ( + 7D5647AC50CD435BD3697028 /* Pods_Runner.framework in Frameworks */, ); runOnlyForDeploymentPostprocessing = 0; }; /* End PBXFrameworksBuildPhase section */ /* Begin PBXGroup section */ + 0AA7387F9F13BD48F8372D87 /* Pods */ = { + isa = PBXGroup; + children = ( + D74402E1D255FC3822687365 /* Pods-Runner.debug.xcconfig */, + 3EBDA4956A7A8415A5A9EC21 /* Pods-Runner.release.xcconfig */, + D788EAF91ED3ED0E558E835C /* Pods-Runner.profile.xcconfig */, + B3AEAC0457BAA712515513DB /* Pods-RunnerTests.debug.xcconfig */, + 0E68AA088CC54A32B209E5A9 /* Pods-RunnerTests.release.xcconfig */, + 4D2E1B9CE910E7CA4F2E7824 /* Pods-RunnerTests.profile.xcconfig */, + ); + name = Pods; + path = Pods; + sourceTree = ""; + }; 331C8082294A63A400263BE5 /* RunnerTests */ = { isa = PBXGroup; children = ( @@ -94,6 +127,8 @@ 97C146F01CF9000F007C117D /* Runner */, 97C146EF1CF9000F007C117D /* Products */, 331C8082294A63A400263BE5 /* RunnerTests */, + 0AA7387F9F13BD48F8372D87 /* Pods */, + AD83C07F807675525BA892D2 /* Frameworks */, ); sourceTree = ""; }; @@ -121,6 +156,15 @@ path = Runner; sourceTree = ""; }; + AD83C07F807675525BA892D2 /* Frameworks */ = { + isa = PBXGroup; + children = ( + 1ECE69C8E6FF389C86529C4E /* Pods_Runner.framework */, + 7B7863E15540969EA0830F56 /* Pods_RunnerTests.framework */, + ); + name = Frameworks; + sourceTree = ""; + }; /* End PBXGroup section */ /* Begin PBXNativeTarget section */ @@ -128,8 +172,10 @@ isa = PBXNativeTarget; buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; buildPhases = ( + 4D5B70523743B6F37D4C08E3 /* [CP] Check Pods Manifest.lock */, 331C807D294A63A400263BE5 /* Sources */, 331C807F294A63A400263BE5 /* Resources */, + 8FC1627F69027DB10219A05D /* Frameworks */, ); buildRules = ( ); @@ -145,12 +191,14 @@ isa = PBXNativeTarget; buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; buildPhases = ( + D8567D0A2B55FC5810699719 /* [CP] Check Pods Manifest.lock */, 9740EEB61CF901F6004384FC /* Run Script */, 97C146EA1CF9000F007C117D /* Sources */, 97C146EB1CF9000F007C117D /* Frameworks */, 97C146EC1CF9000F007C117D /* Resources */, 9705A1C41CF9048500538489 /* Embed Frameworks */, 3B06AD1E1E4923F5004D2608 /* Thin Binary */, + E29BD6672734A1A8E7644EF7 /* [CP] Embed Pods Frameworks */, ); buildRules = ( ); @@ -238,6 +286,28 @@ shellPath = /bin/sh; shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" embed_and_thin"; }; + 4D5B70523743B6F37D4C08E3 /* [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; @@ -253,6 +323,45 @@ shellPath = /bin/sh; shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; }; + D8567D0A2B55FC5810699719 /* [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; + }; + E29BD6672734A1A8E7644EF7 /* [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 */ @@ -379,6 +488,7 @@ }; 331C8088294A63A400263BE5 /* Debug */ = { isa = XCBuildConfiguration; + baseConfigurationReference = B3AEAC0457BAA712515513DB /* Pods-RunnerTests.debug.xcconfig */; buildSettings = { BUNDLE_LOADER = "$(TEST_HOST)"; CODE_SIGN_STYLE = Automatic; @@ -396,6 +506,7 @@ }; 331C8089294A63A400263BE5 /* Release */ = { isa = XCBuildConfiguration; + baseConfigurationReference = 0E68AA088CC54A32B209E5A9 /* Pods-RunnerTests.release.xcconfig */; buildSettings = { BUNDLE_LOADER = "$(TEST_HOST)"; CODE_SIGN_STYLE = Automatic; @@ -411,6 +522,7 @@ }; 331C808A294A63A400263BE5 /* Profile */ = { isa = XCBuildConfiguration; + baseConfigurationReference = 4D2E1B9CE910E7CA4F2E7824 /* Pods-RunnerTests.profile.xcconfig */; buildSettings = { BUNDLE_LOADER = "$(TEST_HOST)"; CODE_SIGN_STYLE = Automatic; diff --git a/evently/ios/Runner.xcworkspace/contents.xcworkspacedata b/evently/ios/Runner.xcworkspace/contents.xcworkspacedata index 1d526a16ed..21a3cc14c7 100644 --- a/evently/ios/Runner.xcworkspace/contents.xcworkspacedata +++ b/evently/ios/Runner.xcworkspace/contents.xcworkspacedata @@ -4,4 +4,7 @@ + + diff --git a/evently/pubspec.lock b/evently/pubspec.lock index af0ac58ecf..75f35224ec 100644 --- a/evently/pubspec.lock +++ b/evently/pubspec.lock @@ -21,10 +21,10 @@ packages: dependency: transitive description: name: args - sha256: "7cf60b9f0cc88203c5a190b4cd62a99feea42759a7fa695010eb5de1c0b2252a" + sha256: bf9f5caeea8d8fe6721a9c358dd8a5c1947b27f1cfaa18b39c301273594919e6 url: "https://pub.dev" source: hosted - version: "2.5.0" + version: "2.6.0" async: dependency: transitive description: @@ -61,10 +61,10 @@ packages: dependency: transitive description: name: build_daemon - sha256: "0343061a33da9c5810b2d6cee51945127d8f4c060b7fbdd9d54917f0a3feaaa1" + sha256: "79b2aef6ac2ed00046867ed354c88778c9c0f029df8a20fe10b5436826721ef9" url: "https://pub.dev" source: hosted - version: "4.0.1" + version: "4.0.2" build_resolvers: dependency: transitive description: @@ -77,18 +77,18 @@ packages: dependency: "direct dev" description: name: build_runner - sha256: "3ac61a79bfb6f6cc11f693591063a7f19a7af628dc52f141743edac5c16e8c22" + sha256: "644dc98a0f179b872f612d3eb627924b578897c629788e858157fa5e704ca0c7" url: "https://pub.dev" source: hosted - version: "2.4.9" + version: "2.4.11" build_runner_core: dependency: transitive description: name: build_runner_core - sha256: "4ae8ffe5ac758da294ecf1802f2aff01558d8b1b00616aa7538ea9a8a5d50799" + sha256: e3c79f69a64bdfcd8a776a3c28db4eb6e3fb5356d013ae5eb2e52007706d5dbe url: "https://pub.dev" source: hosted - version: "7.3.0" + version: "7.3.1" built_collection: dependency: transitive description: @@ -181,18 +181,18 @@ packages: dependency: transitive description: name: convert - sha256: "0f08b14755d163f6e2134cb58222dd25ea2a2ee8a195e53983d57c075324d592" + sha256: b30acd5944035672bc15c6b7a8b47d773e41e2f17de064350988c5d02adb1c68 url: "https://pub.dev" source: hosted - version: "3.1.1" + version: "3.1.2" crypto: dependency: transitive description: name: crypto - sha256: ff625774173754681d66daaf4a448684fb04b78f902da9cb3d308c19cc5e8bab + sha256: "1e445881f28f22d6140f181e07737b22f1e099a5e1ff94b0af2f9e4a463f4855" url: "https://pub.dev" source: hosted - version: "3.0.3" + version: "3.0.6" cupertino_icons: dependency: "direct main" description: @@ -237,18 +237,26 @@ packages: dependency: transitive description: name: dev_build - sha256: "2fa3bf81b5e92504f8d7a412df2c69b61a24ceca468466e5e777cbb59c30af96" + sha256: f526d1fbe68875f6119ffc333f114dfe6aa93ad04439276d53968f7977cc410e url: "https://pub.dev" source: hosted - version: "0.16.5" + version: "1.0.0+11" dio: dependency: "direct main" description: name: dio - sha256: "11e40df547d418cc0c4900a9318b26304e665da6fa4755399a9ff9efd09034b5" + sha256: "5598aa796bbf4699afd5c67c0f5f6e2ed542afc956884b9cd58c306966efc260" + url: "https://pub.dev" + source: hosted + version: "5.7.0" + dio_web_adapter: + dependency: transitive + description: + name: dio_web_adapter + sha256: "33259a9276d6cea88774a0000cfae0d861003497755969c92faa223108620dc8" url: "https://pub.dev" source: hosted - version: "5.4.3+1" + version: "2.0.0" dotted_border: dependency: "direct main" description: @@ -261,10 +269,10 @@ packages: dependency: "direct main" description: name: easy_localization - sha256: "432698c31a488dd64c56d4759f20d04844baba5e9e4f2cb1abb9676257918b17" + sha256: fa59bcdbbb911a764aa6acf96bbb6fa7a5cf8234354fc45ec1a43a0349ef0201 url: "https://pub.dev" source: hosted - version: "3.0.6" + version: "3.0.7" easy_logger: dependency: transitive description: @@ -293,18 +301,18 @@ packages: dependency: transitive description: name: ffi - sha256: "493f37e7df1804778ff3a53bd691d8692ddf69702cf4c1c1096a2e41b4779e21" + sha256: "16ed7b077ef01ad6170a3d0c57caa4a112a38d7a2ed5602e0aca9ca6f3d98da6" url: "https://pub.dev" source: hosted - version: "2.1.2" + version: "2.1.3" file: dependency: transitive description: name: file - sha256: "5fc22d7c25582e38ad9a8515372cd9a93834027aacf1801cf01164dac0ffa08c" + sha256: a3b4f84adafef897088c160faf7dfffb7696046cb13ae90b508c2cbc95d3b8d4 url: "https://pub.dev" source: hosted - version: "7.0.0" + version: "7.0.1" file_picker: dependency: "direct main" description: @@ -317,10 +325,10 @@ packages: dependency: transitive description: name: fixnum - sha256: "25517a4deb0c03aa0f32fd12db525856438902d9c16536311e76cdc57b31d7d1" + sha256: b6dc7065e46c974bc7c5f143080a6764ec7a4be6da1285ececdc37be96de53be url: "https://pub.dev" source: hosted - version: "1.1.0" + version: "1.1.1" floor: dependency: "direct main" description: @@ -383,26 +391,26 @@ packages: dependency: transitive description: name: flutter_plugin_android_lifecycle - sha256: "8cf40eebf5dec866a6d1956ad7b4f7016e6c0cc69847ab946833b7d43743809f" + sha256: "9ee02950848f61c4129af3d6ec84a1cfc0e47931abc746b03e7a3bc3e8ff6eda" url: "https://pub.dev" source: hosted - version: "2.0.19" + version: "2.0.22" flutter_screenutil: dependency: "direct main" description: name: flutter_screenutil - sha256: "8cf100b8e4973dc570b6415a2090b0bfaa8756ad333db46939efc3e774ee100d" + sha256: "8239210dd68bee6b0577aa4a090890342d04a136ce1c81f98ee513fc0ce891de" url: "https://pub.dev" source: hosted - version: "5.9.0" + version: "5.9.3" flutter_slidable: dependency: "direct main" description: name: flutter_slidable - sha256: "673403d2eeef1f9e8483bd6d8d92aae73b1d8bd71f382bc3930f699c731bc27c" + sha256: "2c5611c0b44e20d180e4342318e1bbc28b0a44ad2c442f5df16962606fd3e8e3" url: "https://pub.dev" source: hosted - version: "3.1.0" + version: "3.1.1" flutter_svg: dependency: "direct main" description: @@ -473,10 +481,10 @@ packages: dependency: transitive description: name: graphs - sha256: aedc5a15e78fc65a6e23bcd927f24c64dd995062bcd1ca6eda65a3cff92a4d19 + sha256: "741bbf84165310a68ff28fe9e727332eef1407342fca52759cb21ad8177bb8d0" url: "https://pub.dev" source: hosted - version: "2.3.1" + version: "2.3.2" http: dependency: transitive description: @@ -529,18 +537,18 @@ packages: dependency: "direct main" description: name: injectable - sha256: "3c8355a29d11ff28c0311bed754649761f345ef7a13ff66a714380954af51226" + sha256: "5e1556ea1d374fe44cbe846414d9bab346285d3d8a1da5877c01ad0774006068" url: "https://pub.dev" source: hosted - version: "2.4.2" + version: "2.5.0" injectable_generator: dependency: "direct main" description: name: injectable_generator - sha256: "2ca3ada337eac0ef6b82f8049c970ddb63947738fdf32ac6cbef8d1567d7ba05" + sha256: af403d76c7b18b4217335e0075e950cd0579fd7f8d7bd47ee7c85ada31680ba1 url: "https://pub.dev" source: hosted - version: "2.6.1" + version: "2.6.2" intl: dependency: transitive description: @@ -617,10 +625,10 @@ packages: dependency: transitive description: name: logging - sha256: "623a88c9594aa774443aa3eb2d41807a48486b5613e67599fb4c41c0ad47c340" + sha256: c8245ada5f1717ed44271ed1c26b8ce85ca3228fd2ffdb75468ab01979309d61 url: "https://pub.dev" source: hosted - version: "1.2.0" + version: "1.3.0" matcher: dependency: transitive description: @@ -649,10 +657,10 @@ packages: dependency: transitive description: name: mime - sha256: "2e123074287cc9fd6c09de8336dae606d1ddb88d9ac47358826db698c176a1f2" + sha256: "801fd0b26f14a4a58ccb09d5892c3fbdeff209594300a542492cf13fba9d247a" url: "https://pub.dev" source: hosted - version: "1.0.5" + version: "1.0.6" nested: dependency: transitive description: @@ -673,10 +681,10 @@ packages: dependency: transitive description: name: network_info_plus_platform_interface - sha256: "2e193d61d3072ac17824638793d3b89c6d581ce90c11604f4ca87311b42f2706" + sha256: b7f35f4a7baef511159e524499f3c15464a49faa5ec10e92ee0bce265e664906 url: "https://pub.dev" source: hosted - version: "2.0.0" + version: "2.0.1" nm: dependency: transitive description: @@ -689,10 +697,10 @@ packages: dependency: transitive description: name: octo_image - sha256: "45b40f99622f11901238e18d48f5f12ea36426d8eced9f4cbf58479c7aa2430d" + sha256: "34faa6639a78c7e3cbe79be6f9f96535867e879748ade7d17c9b1ae7536293bd" url: "https://pub.dev" source: hosted - version: "2.0.0" + version: "2.1.0" package_config: dependency: transitive description: @@ -721,34 +729,34 @@ packages: dependency: transitive description: name: path_parsing - sha256: e3e67b1629e6f7e8100b367d3db6ba6af4b1f0bb80f64db18ef1fbabd2fa9ccf + sha256: "883402936929eac138ee0a45da5b0f2c80f89913e6dc3bf77eb65b84b409c6ca" url: "https://pub.dev" source: hosted - version: "1.0.1" + version: "1.1.0" path_provider: dependency: transitive description: name: path_provider - sha256: c9e7d3a4cd1410877472158bee69963a4579f78b68c65a2b7d40d1a7a88bb161 + sha256: "50c5dd5b6e1aaf6fb3a78b33f6aa3afca52bf903a8a5298f53101fdaee55bbcd" url: "https://pub.dev" source: hosted - version: "2.1.3" + version: "2.1.5" path_provider_android: dependency: transitive description: name: path_provider_android - sha256: a248d8146ee5983446bf03ed5ea8f6533129a12b11f12057ad1b4a67a2b3b41d + sha256: "6f01f8e37ec30b07bc424b4deabac37cacb1bc7e2e515ad74486039918a37eb7" url: "https://pub.dev" source: hosted - version: "2.2.4" + version: "2.2.10" path_provider_foundation: dependency: transitive description: name: path_provider_foundation - sha256: "5a7999be66e000916500be4f15a3633ebceb8302719b47b9cc49ce924125350f" + sha256: f234384a3fdd67f989b4d54a5d73ca2a6c422fa55ae694381ae0f4375cd1ea16 url: "https://pub.dev" source: hosted - version: "2.3.2" + version: "2.4.0" path_provider_linux: dependency: transitive description: @@ -769,10 +777,10 @@ packages: dependency: transitive description: name: path_provider_windows - sha256: "8bc9f22eee8690981c22aa7fc602f5c85b497a6fb2ceb35ee5a5e5ed85ad8170" + sha256: bd6f00dbd873bfb70d0761682da2b3a2c2fccc2b9e84c495821639601d81afe7 url: "https://pub.dev" source: hosted - version: "2.2.1" + version: "2.3.0" petitparser: dependency: transitive description: @@ -785,10 +793,10 @@ packages: dependency: transitive description: name: platform - sha256: "12220bb4b65720483f8fa9450b4332347737cf8213dd2840d8b2c823e47243ec" + sha256: "5d6b1b0036a5f331ebc77c850ebc8506cbc1e9416c27e59b439f917a902a4984" url: "https://pub.dev" source: hosted - version: "3.1.4" + version: "3.1.6" plugin_platform_interface: dependency: transitive description: @@ -809,10 +817,10 @@ packages: dependency: transitive description: name: process_run - sha256: "8d9c6198b98fbbfb511edd42e7364e24d85c163e47398919871b952dc86a423e" + sha256: c917dfb5f7afad4c7485bc00a4df038621248fce046105020cea276d1a87c820 url: "https://pub.dev" source: hosted - version: "0.14.2" + version: "1.1.0" protobuf: dependency: transitive description: @@ -841,10 +849,10 @@ packages: dependency: transitive description: name: pubspec_parse - sha256: c63b2876e58e194e4b0828fcb080ad0e06d051cb607a6be51a9e084f47cb9367 + sha256: c799b721d79eb6ee6fa56f00c04b472dcd44a30d258fac2174a6ec57302678f8 url: "https://pub.dev" source: hosted - version: "1.2.3" + version: "1.3.0" pylons_sdk: dependency: "direct main" description: @@ -856,10 +864,10 @@ packages: dependency: transitive description: name: rational - sha256: ba58e9e18df9abde280e8b10051e4bce85091e41e8e7e411b6cde2e738d357cf + sha256: cb808fb6f1a839e6fc5f7d8cb3b0a10e1db48b3be102de73938c627f0b636336 url: "https://pub.dev" source: hosted - version: "2.2.2" + version: "2.2.3" recase: dependency: transitive description: @@ -880,58 +888,58 @@ packages: dependency: "direct main" description: name: shared_preferences - sha256: d3bbe5553a986e83980916ded2f0b435ef2e1893dfaa29d5a7a790d0eca12180 + sha256: "95f9997ca1fb9799d494d0cb2a780fd7be075818d59f00c43832ed112b158a82" url: "https://pub.dev" source: hosted - version: "2.2.3" + version: "2.3.3" shared_preferences_android: dependency: transitive description: name: shared_preferences_android - sha256: "1ee8bf911094a1b592de7ab29add6f826a7331fb854273d55918693d5364a1f2" + sha256: "480ba4345773f56acda9abf5f50bd966f581dac5d514e5fc4a18c62976bbba7e" url: "https://pub.dev" source: hosted - version: "2.2.2" + version: "2.3.2" shared_preferences_foundation: dependency: transitive description: name: shared_preferences_foundation - sha256: "0a8a893bf4fd1152f93fec03a415d11c27c74454d96e2318a7ac38dd18683ab7" + sha256: "07e050c7cd39bad516f8d64c455f04508d09df104be326d8c02551590a0d513d" url: "https://pub.dev" source: hosted - version: "2.4.0" + version: "2.5.3" shared_preferences_linux: dependency: transitive description: name: shared_preferences_linux - sha256: "9f2cbcf46d4270ea8be39fa156d86379077c8a5228d9dfdb1164ae0bb93f1faa" + sha256: "580abfd40f415611503cae30adf626e6656dfb2f0cee8f465ece7b6defb40f2f" url: "https://pub.dev" source: hosted - version: "2.3.2" + version: "2.4.1" shared_preferences_platform_interface: dependency: transitive description: name: shared_preferences_platform_interface - sha256: "22e2ecac9419b4246d7c22bfbbda589e3acf5c0351137d87dd2939d984d37c3b" + sha256: "57cbf196c486bc2cf1f02b85784932c6094376284b3ad5779d1b1c6c6a816b80" url: "https://pub.dev" source: hosted - version: "2.3.2" + version: "2.4.1" shared_preferences_web: dependency: transitive description: name: shared_preferences_web - sha256: "9aee1089b36bd2aafe06582b7d7817fd317ef05fc30e6ba14bff247d0933042a" + sha256: d2ca4132d3946fec2184261726b355836a82c33d7d5b67af32692aff18a4684e url: "https://pub.dev" source: hosted - version: "2.3.0" + version: "2.4.2" shared_preferences_windows: dependency: transitive description: name: shared_preferences_windows - sha256: "841ad54f3c8381c480d0c9b508b89a34036f512482c407e6df7a9c4aa2ef8f59" + sha256: "94ef0f72b2d71bc3e700e025db3710911bd51a71cefb65cc609dd0d9a982e3c1" url: "https://pub.dev" source: hosted - version: "2.3.2" + version: "2.4.1" shelf: dependency: transitive description: @@ -944,10 +952,10 @@ packages: dependency: transitive description: name: shelf_web_socket - sha256: "9ca081be41c60190ebcb4766b2486a7d50261db7bd0f5d9615f2d653637a84c1" + sha256: "073c147238594ecd0d193f3456a5fe91c4b0abbcc68bf5cd95b36c4e194ac611" url: "https://pub.dev" source: hosted - version: "1.0.4" + version: "2.0.0" shimmer_animation: dependency: "direct main" description: @@ -1013,18 +1021,18 @@ packages: dependency: transitive description: name: sqflite_common_ffi_web - sha256: cfc9d1c61a3e06e5b2e96994a44b11125b4f451fee95b9fad8bd473b4613d592 + sha256: e9d1cb35a5ff7c43072968ed734e0a1a859564fd2b2c8654e0c6244a57dc82a8 url: "https://pub.dev" source: hosted - version: "0.4.3+1" + version: "0.4.4" sqlite3: dependency: transitive description: name: sqlite3 - sha256: b384f598b813b347c5a7e5ffad82cbaff1bec3d1561af267041e66f6f0899295 + sha256: bb174b3ec2527f9c5f680f73a89af8149dd99782fbb56ea88ad0807c5638f2ed url: "https://pub.dev" source: hosted - version: "2.4.3" + version: "2.4.7" sqlparser: dependency: transitive description: @@ -1077,10 +1085,10 @@ packages: dependency: transitive description: name: strings - sha256: b33f40c4dd3e597bf6d9e7f4f4dc282dad0f19b07d9f320cb5c2183859cbccf5 + sha256: "052836499f03897d3860a603b330c1ea3c8a14177b21f34b15a1295f36024aae" url: "https://pub.dev" source: hosted - version: "3.1.1" + version: "3.1.2" synchronized: dependency: transitive description: @@ -1157,42 +1165,42 @@ packages: dependency: transitive description: name: url_launcher - sha256: "6ce1e04375be4eed30548f10a315826fd933c1e493206eab82eed01f438c8d2e" + sha256: "9d06212b1362abc2f0f0d78e6f09f726608c74e3b9462e8368bb03314aa8d603" url: "https://pub.dev" source: hosted - version: "6.2.6" + version: "6.3.1" url_launcher_android: dependency: transitive description: name: url_launcher_android - sha256: "360a6ed2027f18b73c8d98e159dda67a61b7f2e0f6ec26e86c3ada33b0621775" + sha256: f0c73347dfcfa5b3db8bc06e1502668265d39c08f310c29bff4e28eea9699f79 url: "https://pub.dev" source: hosted - version: "6.3.1" + version: "6.3.9" url_launcher_ios: dependency: transitive description: name: url_launcher_ios - sha256: "9149d493b075ed740901f3ee844a38a00b33116c7c5c10d7fb27df8987fb51d5" + sha256: e43b677296fadce447e987a2f519dcf5f6d1e527dc35d01ffab4fff5b8a7063e url: "https://pub.dev" source: hosted - version: "6.2.5" + version: "6.3.1" url_launcher_linux: dependency: transitive description: name: url_launcher_linux - sha256: ab360eb661f8879369acac07b6bb3ff09d9471155357da8443fd5d3cf7363811 + sha256: e2b9622b4007f97f504cd64c0128309dfb978ae66adbe944125ed9e1750f06af url: "https://pub.dev" source: hosted - version: "3.1.1" + version: "3.2.0" url_launcher_macos: dependency: transitive description: name: url_launcher_macos - sha256: b7244901ea3cf489c5335bdacda07264a6e960b1c1b1a9f91e4bc371d9e68234 + sha256: "769549c999acdb42b8bcfa7c43d72bf79a382ca7441ab18a808e101149daf672" url: "https://pub.dev" source: hosted - version: "3.1.0" + version: "3.2.1" url_launcher_platform_interface: dependency: transitive description: @@ -1205,26 +1213,26 @@ packages: dependency: transitive description: name: url_launcher_web - sha256: "8d9e750d8c9338601e709cd0885f95825086bd8b642547f26bda435aade95d8a" + sha256: "772638d3b34c779ede05ba3d38af34657a05ac55b06279ea6edd409e323dca8e" url: "https://pub.dev" source: hosted - version: "2.3.1" + version: "2.3.3" url_launcher_windows: dependency: transitive description: name: url_launcher_windows - sha256: ecf9725510600aa2bb6d7ddabe16357691b6d2805f66216a97d1b881e21beff7 + sha256: "44cf3aabcedde30f2dba119a9dea3b0f2672fbe6fa96e85536251d678216b3c4" url: "https://pub.dev" source: hosted - version: "3.1.1" + version: "3.1.3" uuid: dependency: transitive description: name: uuid - sha256: "814e9e88f21a176ae1359149021870e87f7cddaf633ab678a5d2b0bff7fd1ba8" + sha256: a5be9ef6618a7ac1e964353ef476418026db906c4facdedaa299b7a2e71690ff url: "https://pub.dev" source: hosted - version: "4.4.0" + version: "4.5.1" vector_math: dependency: transitive description: @@ -1261,34 +1269,42 @@ packages: dependency: transitive description: name: web - sha256: "97da13628db363c635202ad97068d47c5b8aa555808e7a9411963c533b449b27" + sha256: cd3543bd5798f6ad290ea73d210f423502e71900302dde696f8bff84bf89a1cb url: "https://pub.dev" source: hosted - version: "0.5.1" + version: "1.1.0" + web_socket: + dependency: transitive + description: + name: web_socket + sha256: "3c12d96c0c9a4eec095246debcea7b86c0324f22df69893d538fcc6f1b8cce83" + url: "https://pub.dev" + source: hosted + version: "0.1.6" web_socket_channel: dependency: transitive description: name: web_socket_channel - sha256: "58c6666b342a38816b2e7e50ed0f1e261959630becd4c879c4f26bfa14aa5a42" + sha256: "9f187088ed104edd8662ca07af4b124465893caf063ba29758f97af57e61da8f" url: "https://pub.dev" source: hosted - version: "2.4.5" + version: "3.0.1" win32: dependency: transitive description: name: win32 - sha256: "0eaf06e3446824099858367950a813472af675116bf63f008a4c2a75ae13e9cb" + sha256: "68d1e89a91ed61ad9c370f9f8b6effed9ae5e0ede22a270bdfa6daf79fc2290a" url: "https://pub.dev" source: hosted - version: "5.5.0" + version: "5.5.4" xdg_directories: dependency: transitive description: name: xdg_directories - sha256: faea9dee56b520b55a566385b84f2e8de55e7496104adada9962e0bd11bcff1d + sha256: "7a3f37b05d989967cdddcbb571f1ea834867ae2faa29725fd085180e0883aa15" url: "https://pub.dev" source: hosted - version: "1.0.4" + version: "1.1.0" xml: dependency: transitive description: @@ -1306,5 +1322,5 @@ packages: source: hosted version: "3.1.2" sdks: - dart: ">=3.3.4 <4.0.0" - flutter: ">=3.19.0" + dart: ">=3.4.0 <4.0.0" + flutter: ">=3.22.0" diff --git a/wallet/.gitignore b/wallet/.gitignore index 2bda0d8f36..ff040f4592 100644 --- a/wallet/.gitignore +++ b/wallet/.gitignore @@ -55,4 +55,6 @@ env/.dev_env env/.local_env env/.prod_env +google-services.json + .vscode/launch.json diff --git a/wallet/android/app/src/main/kotlin/tech/pylons/wallet/MainActivity.kt b/wallet/android/app/src/main/kotlin/tech/pylons/wallet/MainActivity.kt index 712dad89a4..ed4cd4ee5a 100644 --- a/wallet/android/app/src/main/kotlin/tech/pylons/wallet/MainActivity.kt +++ b/wallet/android/app/src/main/kotlin/tech/pylons/wallet/MainActivity.kt @@ -3,7 +3,6 @@ package tech.pylons.wallet import io.flutter.embedding.android.FlutterFragmentActivity import androidx.annotation.NonNull import com.google.firebase.FirebaseApp -import com.google.firebase.appcheck.FirebaseAppCheck import com.google.firebase.appcheck.debug.DebugAppCheckProviderFactory import io.flutter.embedding.engine.FlutterEngine import io.flutter.plugin.common.MethodChannel @@ -17,8 +16,6 @@ class MainActivity : FlutterFragmentActivity() { ).setMethodCallHandler { call, result -> if (call.method == "getFirebaseAppCheckDebugToken") { FirebaseApp.initializeApp(this) - val firebaseAppCheck = FirebaseAppCheck.getInstance() - firebaseAppCheck.installAppCheckProviderFactory(DebugAppCheckProviderFactory.getInstance()) result.success(true) } } diff --git a/wallet/env/pubspec.yaml b/wallet/env/pubspec.yaml new file mode 100644 index 0000000000..1173c6a9fc --- /dev/null +++ b/wallet/env/pubspec.yaml @@ -0,0 +1,154 @@ +name: pylons_wallet +description: Pylons Wallet +publish_to: "none" +version: 1.0.1+176 + +environment: + sdk: ">=2.17.0 <3.0.0" + +dependencies: + advance_pdf_viewer: + path: pkgs/pdf_viewer-master + alan: + path: pkgs/alan + audio_video_progress_bar: ^2.0.3 + auto_size_text: ^3.0.0 + backdrop: ^0.9.0 + bottom_drawer: ^0.0.3 + cached_network_image: + cloud_firestore: ^5.0.2 + cupertino_icons: ^1.0.3 + dartz: ^0.10.1 + decimal: ^2.3.0 + detectable_text_field: ^3.0.2 + dotted_border: ^2.0.0+2 + easy_localization: ^3.0.1 + equatable: ^2.0.5 + expandable: ^5.0.1 + firebase_analytics: + firebase_app_check: ^0.3.0+2 + firebase_core: ^3.1.1 + firebase_crashlytics: ^4.0.2 + firebase_dynamic_links: ^6.0.2 + firebase_messaging: ^15.0.2 + firebase_remote_config: ^5.0.2 + fixnum: ^1.0.0 + floor: ^1.3.0 + flutter: + sdk: flutter + flutter_dotenv: ^5.0.2 + flutter_downloader: ^1.9.1 + flutter_inappwebview: ^6.0.0 + flutter_local_notifications: ^17.2.1 + flutter_localizations: + sdk: flutter + flutter_mobx: ^2.0.1 + flutter_screenutil: ^5.5.4 + flutter_secure_storage: ^8.0.0 + flutter_staggered_grid_view: ^0.7.0 + flutter_sticky_header: ^0.6.0 + flutter_stripe: + flutter_svg: + flutter_svg_provider: + git: + url: https://github.com/arigilder/flutter_svg_provider + ref: patch-1 + focus_detector: ^2.0.1 + get_it: ^7.2.0 + google_sign_in: ^6.0.2 + googleapis: ^13.2.0 + grpc: ^3.0.2 + home_widget: ^0.6.0 + http: + icloud_storage: ^2.0.0 + image: ^4.0.12 + image_cropper: + image_picker: ^1.1.2 + in_app_purchase: ^3.0.7 + internet_connection_checker: ^1.0.0+1 + intl: + just_audio: ^0.9.29 + just_the_tooltip: 0.0.12 + local_auth: ^2.1.2 + mobx: ^2.0.3 + modal_bottom_sheet: ^3.0.0 + model_viewer_plus: + path: pkgs/model_viewer_plus.dart-master + path_provider: ^2.0.11 + permission_handler: ^11.3.1 + provider: ^6.0.3 + qr_flutter: ^4.0.0 + + retry: ^3.1.2 + share_plus: ^6.0.0 + shared_preferences: ^2.0.12 + shimmer_animation: ^2.1.0+1 + smooth_page_indicator: ^1.0.0+2 + sprintf: ^7.0.0 + transaction_signing_gateway: + path: pkgs/transaction_signing_gateway + transparent_image: ^2.0.0 + tuple: ^2.0.1 + uni_links: ^0.5.1 + url_launcher: ^6.1.5 + uuid: ^3.0.6 + video_player: ^2.4.7 + video_thumbnail: ^0.5.3 + wallpaper: ^1.1.0 + + +dev_dependencies: + build_runner: ^2.3.2 + flutter_gen_runner: + # # Revert it for making autogenerated code for database + # floor_generator: 1.3.0 + flutter_test: + sdk: flutter + lint: ^2.0.1 + mockito: + + +flutter: + uses-material-design: true + assets: + - assets/images/ + - assets/images/splash/ + - assets/images/icons/ + - assets/images/masks/ + - assets/images/onboarding/ + - assets/images/pages/payment/ + - assets/images/sample_content/ + - assets/images/svg/ + - assets/images/gifs/ + - assets/ + - env/ + - i18n/en.json + - i18n/ru.json + - i18n/id.json + - i18n/de.json + - i18n/ko.json + - i18n/ja.json + - i18n/es.json + - i18n/vi.json + + fonts: + - family: Inter + fonts: + - asset: assets/fonts/Inter-Black.ttf + - asset: assets/fonts/Inter-Bold.ttf + - asset: assets/fonts/Inter-ExtraBold.ttf + - asset: assets/fonts/Inter-ExtraLight.ttf + - asset: assets/fonts/Inter-Light.ttf + - asset: assets/fonts/Inter-Medium.ttf + - asset: assets/fonts/Inter-Regular.ttf + - asset: assets/fonts/Inter-SemiBold.ttf + - asset: assets/fonts/Inter-Thin.ttf + - family: PylonsV3 + fonts: + - asset: assets/fonts/PylonsV3-Condensed.otf + - family: UniversalSans + fonts: + - asset: assets/fonts/UniversalSans-600.ttf + weight: 600 + - asset: assets/fonts/UniversalSans-750.ttf + weight: 700 \ No newline at end of file diff --git a/wallet/ios/Runner.xcodeproj/project.pbxproj b/wallet/ios/Runner.xcodeproj/project.pbxproj index e9eaeb3bd3..39f4c2bad6 100644 --- a/wallet/ios/Runner.xcodeproj/project.pbxproj +++ b/wallet/ios/Runner.xcodeproj/project.pbxproj @@ -13,7 +13,6 @@ 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 */; }; - A4A8D90A284E2F3E00DFDE59 /* StoreKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = A4A8D909284E2F3D00DFDE59 /* StoreKit.framework */; }; F0141E24A6BA0796DB8F855D /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 51AA264E01C28B508FFE1809 /* Pods_Runner.framework */; }; /* End PBXBuildFile section */ @@ -52,7 +51,6 @@ 97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; 97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; 97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; - A4A8D909284E2F3D00DFDE59 /* StoreKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = StoreKit.framework; path = System/Library/Frameworks/StoreKit.framework; sourceTree = SDKROOT; }; A4D3E8432768CB44009FC1B3 /* Runner.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = Runner.entitlements; sourceTree = ""; }; F0345B6003A65B27B0F42AB7 /* Pods-Runner.release-prod.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release-prod.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release-prod.xcconfig"; sourceTree = ""; }; /* End PBXFileReference section */ @@ -63,7 +61,6 @@ buildActionMask = 2147483647; files = ( F0141E24A6BA0796DB8F855D /* Pods_Runner.framework in Frameworks */, - A4A8D90A284E2F3E00DFDE59 /* StoreKit.framework in Frameworks */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -73,7 +70,6 @@ 40BD3CB917AE99236BF7F951 /* Frameworks */ = { isa = PBXGroup; children = ( - A4A8D909284E2F3D00DFDE59 /* StoreKit.framework */, 51AA264E01C28B508FFE1809 /* Pods_Runner.framework */, ); name = Frameworks; @@ -431,7 +427,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = ""; INFOPLIST_FILE = Runner/Info.plist; @@ -441,7 +437,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -569,7 +565,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; "EXCLUDED_ARCHS[sdk=iphonesimulator*]" = i386; GOOGLE_SERVICE_PATH = ""; @@ -580,7 +576,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -602,7 +598,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = ""; INFOPLIST_FILE = Runner/Info.plist; @@ -612,7 +608,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -688,7 +684,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = "${SRCROOT}/Runner/Firebase/prod/GoogleService-Info.plist"; INFOPLIST_FILE = Runner/Info.plist; @@ -698,7 +694,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -772,7 +768,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = "${SRCROOT}/Runner/Firebase/prod/GoogleService-Info.plist"; INFOPLIST_FILE = Runner/Info.plist; @@ -782,7 +778,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -858,7 +854,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = "${SRCROOT}/Runner/Firebase/development/GoogleService-Info.plist"; INFOPLIST_FILE = Runner/Info.plist; @@ -868,7 +864,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.development; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; @@ -942,7 +938,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 4; - DEVELOPMENT_TEAM = QV67PMQ8H3; + DEVELOPMENT_TEAM = VBTGJ398PV; ENABLE_BITCODE = NO; GOOGLE_SERVICE_PATH = "${SRCROOT}/Runner/Firebase/development/GoogleService-Info.plist"; INFOPLIST_FILE = Runner/Info.plist; @@ -952,7 +948,7 @@ "@executable_path/Frameworks", ); MARKETING_VERSION = 1.2.0; - PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.development; + PRODUCT_BUNDLE_IDENTIFIER = xyz.pylons.wallet.blah; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; diff --git a/wallet/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist b/wallet/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist index 9ab4e6b654..18d981003d 100644 --- a/wallet/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist +++ b/wallet/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist @@ -5,4 +5,4 @@ IDEDidComputeMac32BitWarning - \ No newline at end of file + diff --git a/wallet/ios/Runner/Info.plist b/wallet/ios/Runner/Info.plist index 114e27979d..ced19d55cd 100644 --- a/wallet/ios/Runner/Info.plist +++ b/wallet/ios/Runner/Info.plist @@ -50,6 +50,8 @@ CFBundleVersion $(CURRENT_PROJECT_VERSION) + FDMaximumConcurrentTasks + 5 FirebaseDynamicLinksCustomDomains https://wallet.pylons.tech @@ -64,6 +66,21 @@ LSRequiresIPhoneOS + NSAppTransportSecurity + + NSExceptionDomains + + proxy.pylons.tech + + NSIncludesSubdomains + + NSTemporaryExceptionAllowsInsecureHTTPLoads + + NSTemporaryExceptionMinimumTLSVersion + TLSv1.1 + + + NSCameraUsageDescription Pylons Wallet needs to use your camera to capture images for minting and user profile NSFaceIDUsageDescription @@ -72,6 +89,8 @@ Pylons Wallet needs to use your microphone to record audios for minting purposes NSPhotoLibraryUsageDescription Pylons Wallet needs to access your photo library for minting and user profile + UIApplicationSupportsIndirectInputEvents + UIBackgroundModes fetch @@ -95,25 +114,5 @@ io.flutter.embedded_views_preview - NSAppTransportSecurity - - NSExceptionDomains - - proxy.pylons.tech - - NSIncludesSubdomains - - NSTemporaryExceptionAllowsInsecureHTTPLoads - - NSTemporaryExceptionMinimumTLSVersion - TLSv1.1 - - - - - FDMaximumConcurrentTasks - 5 - UIApplicationSupportsIndirectInputEvents - diff --git a/wallet/ios/Runner/Runner.entitlements b/wallet/ios/Runner/Runner.entitlements index 3b41029176..0c67376eba 100644 --- a/wallet/ios/Runner/Runner.entitlements +++ b/wallet/ios/Runner/Runner.entitlements @@ -1,25 +1,5 @@ - - aps-environment - development - com.apple.developer.associated-domains - - applinks:wallet.pylons.tech - applinks:pylons.page.link - - com.apple.developer.icloud-container-identifiers - - iCloud.pylonsStorage - - com.apple.developer.icloud-services - - CloudDocuments - - com.apple.developer.ubiquity-container-identifiers - - iCloud.pylonsStorage - - + diff --git a/wallet/lib/model/event.dart b/wallet/lib/model/event.dart index 86dbdf7a45..031aefffa3 100644 --- a/wallet/lib/model/event.dart +++ b/wallet/lib/model/event.dart @@ -5,7 +5,6 @@ import 'package:pylons_wallet/pages/home/currency_screen/model/ibc_coins.dart'; import 'package:pylons_wallet/stores/wallet_store.dart'; import '../modules/Pylonstech.pylons.pylons/module/client/cosmos/base/v1beta1/coin.pb.dart'; import '../modules/Pylonstech.pylons.pylons/module/client/pylons/recipe.pb.dart'; - enum FreeDrop { yes, no, unselected } class Events extends Equatable { @@ -22,6 +21,7 @@ class Events extends Equatable { final String description; final String numberOfTickets; final String price; + final String? challenge; final List? listOfPerks; final String isFreeDrops; final String cookbookID; @@ -58,6 +58,7 @@ class Events extends Equatable { ///* other this.cookbookID = '', this.recipeID = '', + this.challenge, ///* for tracking where its save as draft this.step = '', @@ -89,6 +90,7 @@ class Events extends Equatable { map['price'] = price; map['isFreeDrops'] = isFreeDrops; map['cookbookID'] = cookbookID; + map['challenge'] = challenge ?? ''; map['step'] = step; map['denom'] = denom.toString(); map['listOfPerks'] = perks; @@ -116,6 +118,7 @@ class Events extends Equatable { price: json['price'] as String, isFreeDrops: json['isFreeDrops'] as String, cookbookID: json['cookbookID'] as String, + challenge: json['challenge'] as String?, step: json['step'] as String, listOfPerks: listOfPerks, denom: json['denom'].toString().toIBCCoinsEnumforEvent(), @@ -160,6 +163,7 @@ class Events extends Equatable { listOfPerks: listOfPerks, cookbookID: map[kCookBookId]!, recipeID: map[kRecipeId]!, + challenge: map[kChallenge], denom: denom.isEmpty ? IBCCoins.upylon : denom.toIBCCoinsEnum(), ); } @@ -182,6 +186,7 @@ class Events extends Equatable { case kPerks: case kFreeDrop: case kCookBookId: + case kChallenge: case kRecipeId: attributeValues[attribute.key] = attribute.value; break; @@ -202,11 +207,11 @@ class Events extends Equatable { @override List get props => - [eventName, hostName, thumbnail, startDate, endDate, startTime, endTime, location, description, numberOfTickets, price, listOfPerks, isFreeDrops, cookbookID, recipeID, step]; + [eventName, hostName, thumbnail, startDate, endDate, startTime, endTime, location, description, numberOfTickets, price, listOfPerks, isFreeDrops, cookbookID, recipeID, step, challenge]; @override String toString() { - return 'Event{eventName: $eventName, hostName: $hostName, thumbnail: $thumbnail, startDate: $startDate, endDate: $endDate, startTime: $startTime, endTime: $endTime, location: $location, description: $description, numberOfTickets: $numberOfTickets, price: $price, listOfPerks: $listOfPerks, isFreeDrop: $isFreeDrops, cookbookID: $cookbookID, recipeID: $recipeID, step: $step}'; + return 'Event{eventName: $eventName, hostName: $hostName, thumbnail: $thumbnail, startDate: $startDate, endDate: $endDate, startTime: $startTime, endTime: $endTime, location: $location, description: $description, numberOfTickets: $numberOfTickets, price: $price, listOfPerks: $listOfPerks, isFreeDrop: $isFreeDrops, cookbookID: $cookbookID, recipeID: $recipeID, step: $step, challenge: $challenge}'; } static Future eventFromRecipeId(String cookbookId, String recipeId) async { @@ -237,6 +242,7 @@ const kPrice = "kPrice"; const kFreeDrop = "kFreeDrop"; const kRecipeId = "kRecipeId"; const kCookBookId = "kCookBookId"; +const kChallenge = "kChallenge"; const kVersion = "v0.2.0"; const kUpylon = "upylon"; const kPylonSymbol = 'upylon'; diff --git a/wallet/lib/model/update_recipe_model.dart b/wallet/lib/model/update_recipe_model.dart index a8d6c64e19..05437cc84d 100644 --- a/wallet/lib/model/update_recipe_model.dart +++ b/wallet/lib/model/update_recipe_model.dart @@ -7,6 +7,7 @@ class UpdateRecipeModel { String nftPrice = ""; String denom = ""; int quantity = 0; + String challenge = ""; UpdateRecipeModel({ required this.recipe, @@ -15,10 +16,11 @@ class UpdateRecipeModel { required this.nftPrice, required this.denom, required this.quantity, + required this.challenge, }); @override String toString() { - return 'UpdateRecipeModel{recipe: $recipe, publicAddress: $publicAddress, enabledStatus: $enabledStatus, nftPrice: $nftPrice, denom: $denom, quantity: $quantity}'; + return 'UpdateRecipeModel{recipe: $recipe, publicAddress: $publicAddress, enabledStatus: $enabledStatus, nftPrice: $nftPrice, denom: $denom, quantity: $quantity, challenge: $challenge}'; } } diff --git a/wallet/lib/pages/detailed_asset_view/owner_view_view_model.dart b/wallet/lib/pages/detailed_asset_view/owner_view_view_model.dart index 743ac368bb..d5e2a47545 100644 --- a/wallet/lib/pages/detailed_asset_view/owner_view_view_model.dart +++ b/wallet/lib/pages/detailed_asset_view/owner_view_view_model.dart @@ -506,6 +506,7 @@ class OwnerViewViewModel extends ChangeNotifier { notifyListeners(); } + Future cancelTrade({required String tradeId, required String address}) async { final tradeResponse = await repository.cancelTrade( tradeId: TradeId(Int64(int.parse(tradeId))), @@ -517,6 +518,28 @@ class OwnerViewViewModel extends ChangeNotifier { } } + Future stampTicket({ + required String cookbookId, + required String recipeId, + required Address creatorAddress, + required String challenge, + }) async { + final response = await repository.stampTicket( + cookBookId: CookbookId(cookbookId), + recipeId: RecipeId(recipeId), + creatorAddress: creatorAddress, + challenge: challenge, + ); + + if (response.isLeft()) { + throw response.getLeft(); + } + + notifyListeners(); + } + + + void logEvent() { repository.logUserJourney(screenName: AnalyticsScreenEvents.ownerView); } diff --git a/wallet/lib/pages/events/event_qr_code_screen.dart b/wallet/lib/pages/events/event_qr_code_screen.dart index b75d18e23a..c2413735a6 100644 --- a/wallet/lib/pages/events/event_qr_code_screen.dart +++ b/wallet/lib/pages/events/event_qr_code_screen.dart @@ -4,6 +4,7 @@ import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_screenutil/flutter_screenutil.dart'; import 'package:flutter_svg/flutter_svg.dart'; +import 'package:provider/provider.dart'; import 'package:pylons_wallet/components/buttons/custom_paint_button.dart'; import 'package:pylons_wallet/gen/assets.gen.dart'; import 'package:pylons_wallet/generated/locale_keys.g.dart'; @@ -12,6 +13,8 @@ import 'package:pylons_wallet/pages/detailed_asset_view/widgets/nft_image_asset. import 'package:pylons_wallet/utils/constants.dart'; import 'package:qr_flutter/qr_flutter.dart'; +import '../../providers/account_provider.dart'; + class EventQrCodeScreen extends StatefulWidget { const EventQrCodeScreen({ super.key, @@ -25,67 +28,97 @@ class EventQrCodeScreen extends StatefulWidget { } class _EventQrCodeScreenState extends State { + String qrdata = ""; GlobalKey renderObjectKey = GlobalKey(); + @override - Widget build(BuildContext context) { - return Material( - color: AppColors.kBlack, - child: Stack( - children: [ - NftImageWidget( - url: widget.events.thumbnail, - opacity: 0.5, - ), - Padding( - padding: EdgeInsets.only(left: 23.w, top: MediaQuery.of(context).viewPadding.top + 13.h), - child: GestureDetector( - onTap: () async { - Navigator.pop(context); - }, - child: SvgPicture.asset( - Assets.images.icons.back, - height: 25.h, - ), + void initState() { + super.initState(); + + createLink(); + } + + void createLink() { + final wallet = context + .read() + .accountPublicInfo; + + if (wallet == null) { + return; + } + qrdata = jsonEncode({ + 'cookbookId': widget.events.cookbookID, + 'recipeId': widget.events.recipeID, + 'challenge': widget.events.challenge, + }); + } + + @override + Widget build(BuildContext context) { + return Material( + color: AppColors.kBlack, + child: Stack( + children: [ + NftImageWidget( + url: widget.events.thumbnail, + opacity: 0.5, ), - ), - ColoredBox( - color: AppColors.kBlack.withOpacity(0.5), - child: Align( - child: RepaintBoundary( - key: renderObjectKey, - child: QrImageView( - padding: EdgeInsets.zero, - data: jsonEncode(widget.events.toJson()), - size: 200, - dataModuleStyle: const QrDataModuleStyle(color: AppColors.kWhite), - eyeStyle: const QrEyeStyle(eyeShape: QrEyeShape.square, color: AppColors.kWhite), + Padding( + padding: EdgeInsets.only(left: 23.w, top: MediaQuery + .of(context) + .viewPadding + .top + 13.h), + child: GestureDetector( + onTap: () async { + Navigator.pop(context); + }, + child: SvgPicture.asset( + Assets.images.icons.back, + height: 25.h, ), ), ), - ), - Align( - child: Padding( - padding: EdgeInsets.symmetric(horizontal: 20.w), - child: SvgPicture.asset(Assets.images.svg.qrSideBorder), + ColoredBox( + color: AppColors.kBlack.withOpacity(0.5), + child: Align( + child: RepaintBoundary( + key: renderObjectKey, + child: QrImageView( + padding: EdgeInsets.zero, + data: qrdata, + size: 200, + dataModuleStyle: const QrDataModuleStyle( + color: AppColors.kWhite), + eyeStyle: const QrEyeStyle( + eyeShape: QrEyeShape.square, color: AppColors.kWhite), + ), + ), + ), ), - ), - Align( - alignment: Alignment.bottomCenter, - child: Padding( - padding: EdgeInsets.only(bottom: 30.h), - child: CustomPaintButton( - title: LocaleKeys.done.tr(), - bgColor: AppColors.kWhite.withOpacity(0.3), - width: 280.w, - onPressed: () { - Navigator.pop(context); - }, + Align( + child: Padding( + padding: EdgeInsets.symmetric(horizontal: 20.w), + child: SvgPicture.asset(Assets.images.svg.qrSideBorder), ), ), - ) - ], - ), - ); + Align( + alignment: Alignment.bottomCenter, + child: Padding( + padding: EdgeInsets.only(bottom: 30.h), + child: CustomPaintButton( + title: LocaleKeys.done.tr(), + bgColor: AppColors.kWhite.withOpacity(0.3), + width: 280.w, + onPressed: () { + Navigator.pop(context); + }, + ), + ), + ) + ], + ), + ); + } } -} + diff --git a/wallet/lib/pages/events/mobile_scanner.dart b/wallet/lib/pages/events/mobile_scanner.dart index dcf11ccaf8..049c1c44bd 100644 --- a/wallet/lib/pages/events/mobile_scanner.dart +++ b/wallet/lib/pages/events/mobile_scanner.dart @@ -1,5 +1,9 @@ import 'package:flutter/material.dart'; +import 'package:get_it/get_it.dart'; import 'package:mobile_scanner/mobile_scanner.dart'; +import 'package:pylons_wallet/pages/events/stamping_screen.dart'; +import '../../model/event.dart'; +import '../detailed_asset_view/owner_view_view_model.dart'; class MobileQrScanner extends StatefulWidget { const MobileQrScanner({super.key}); @@ -9,53 +13,93 @@ class MobileQrScanner extends StatefulWidget { } class _MobileQrScannerState extends State { + final OwnerViewViewModel ownerViewViewModel = GetIt.I.get(); Barcode? _barcode; + bool isScanning = true; - Widget _buildBarcode(Barcode? value) { - if (value == null) { + Widget _buildBarcodeDisplay(Barcode? barcode) { + if (barcode == null) { return const Text( - 'Scan something!', - overflow: TextOverflow.fade, - style: TextStyle(color: Colors.white), + 'Scan Ticket!', + overflow: TextOverflow.ellipsis, + style: TextStyle(color: Colors.white, fontSize: 18, fontWeight: FontWeight.bold), ); } - return Text( - value.displayValue ?? 'No display value.', - overflow: TextOverflow.fade, - style: const TextStyle(color: Colors.white), + barcode.displayValue ?? 'No display value.', + overflow: TextOverflow.ellipsis, + style: const TextStyle(color: Colors.white, fontSize: 18, fontWeight: FontWeight.bold), ); } - void _handleBarcode(BarcodeCapture barcodes) { - if (mounted) { - setState(() { - _barcode = barcodes.barcodes.firstOrNull; - }); - } + void _handleBarcode(BarcodeCapture barcodeCapture) { + if (!isScanning) return; + + setState(() { + if (barcodeCapture.barcodes.isNotEmpty) { + _barcode = barcodeCapture.barcodes[0]; + final qrData = _barcode?.displayValue ?? ''; + final dataParts = qrData.split(','); + + if (dataParts.length >= 2) { + final cookbookId = dataParts[0]; + final recipeId = dataParts[1]; + final challenge = dataParts.length > 2 ? dataParts[2] : ''; + + Navigator.push( + context, + MaterialPageRoute( + builder: (context) => StampingScreen( + cookbookId: cookbookId, + recipeId: recipeId, + challenge: challenge, + ownerViewViewModel: ownerViewViewModel, + event: Events(eventName: 'Event Name', thumbnail: 'Thumbnail URL', description: 'Description'), + ), + ), + ); + } else { + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar(content: Text('Invalid QR code data.')), + ); + } + } else { + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar(content: Text('No barcodes detected.')), + ); + } + }); } @override Widget build(BuildContext context) { return Scaffold( - appBar: AppBar(title: const Text('Simple scanner')), + appBar: AppBar( + title: const Text('Ticket Scanner'), + backgroundColor: Colors.blueGrey[900], + ), backgroundColor: Colors.black, body: Stack( children: [ MobileScanner( onDetect: _handleBarcode, + scanWindow: Rect.fromLTWH( + 0, + 0, + MediaQuery.of(context).size.width, + MediaQuery.of(context).size.height, + ), ), - Align( - alignment: Alignment.bottomCenter, + Positioned( + bottom: 0, + left: 0, + right: 0, child: Container( - alignment: Alignment.bottomCenter, + alignment: Alignment.center, height: 100, - color: Colors.black.withOpacity(0.4), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceEvenly, - children: [ - Expanded(child: Center(child: _buildBarcode(_barcode))), - ], + color: Colors.black.withOpacity(0.5), + child: Center( + child: _buildBarcodeDisplay(_barcode), ), ), ), diff --git a/wallet/lib/pages/events/stamping_screen.dart b/wallet/lib/pages/events/stamping_screen.dart new file mode 100644 index 0000000000..54bf502650 --- /dev/null +++ b/wallet/lib/pages/events/stamping_screen.dart @@ -0,0 +1,169 @@ +import 'package:flutter/material.dart'; +import 'package:cached_network_image/cached_network_image.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import 'package:get_it/get_it.dart'; +import 'package:pylons_wallet/pages/detailed_asset_view/owner_view_view_model.dart'; +import '../../model/common.dart'; +import '../../model/event.dart'; +import '../../stores/wallet_store.dart'; + +class StampingScreen extends StatefulWidget { + final String cookbookId; + final String recipeId; + final String challenge; + final OwnerViewViewModel ownerViewViewModel; + final Events event; + + const StampingScreen({ + super.key, + required this.cookbookId, + required this.recipeId, + required this.challenge, + required this.ownerViewViewModel, + required this.event, + }); + + @override + _StampingScreenState createState() => _StampingScreenState(); +} + +class _StampingScreenState extends State { + Events? event; + late OwnerViewViewModel ownerViewViewModel; + bool isLoading = true; + bool isAlreadyStamped = false; + + @override + void initState() { + super.initState(); + ownerViewViewModel = GetIt.I.get(); + fetchEvent(); + } + + Future fetchEvent() async { + try { + debugPrint(widget.cookbookId); + debugPrint(widget.recipeId); + + final fetchedEvent = await eventFromRecipeId(widget.cookbookId, widget.recipeId); + + if (mounted) { + setState(() { + event = fetchedEvent; + isLoading = false; + isAlreadyStamped = widget.challenge.isNotEmpty; + }); + } + } catch (e) { + if (mounted) { + setState(() { + isLoading = false; + }); + } + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('Failed to fetch event: $e')), + ); + } + } + + Future _stampTicket(BuildContext context) async { + try { + if (isAlreadyStamped) { + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar(content: Text('This ticket is already stamped.')), + ); + return; + } + + await ownerViewViewModel.stampTicket( + cookbookId: widget.cookbookId, + recipeId: widget.recipeId, + creatorAddress: Address(widget.ownerViewViewModel.owner), + challenge: widget.challenge, + ); + + final updatedEvent = await eventFromRecipeId(widget.cookbookId, widget.recipeId); + + if (mounted) { + setState(() { + event = updatedEvent; + isAlreadyStamped = true; + }); + } + + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('Ticket with challenge ${widget.challenge} stamped successfully!')), + ); + } catch (e) { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('Failed to stamp ticket: $e')), + ); + } + } + + Future eventFromRecipeId(String cookbookId, String recipeId) async { + final walletsStore = GetIt.I.get(); + final recipeEither = await walletsStore.getRecipe(cookbookId, recipeId); + + if (recipeEither.isLeft()) { + return null; + } + + return Events.fromRecipe(recipeEither.toOption().toNullable()!); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar(title: const Text('Stamp Ticket')), + body: isLoading + ? const Center(child: CircularProgressIndicator()) + : event == null + ? const Center(child: Text('No event found for the provided IDs.')) + : SingleChildScrollView( + padding: const EdgeInsets.all(16.0), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + CachedNetworkImage( + imageUrl: event!.thumbnail, + height: 200.h, + width: double.infinity, + fit: BoxFit.cover, + placeholder: (context, url) => const CircularProgressIndicator(), + errorWidget: (context, url, error) => const Icon(Icons.error), + ), + const SizedBox(height: 16), + Text( + event!.eventName, + style: TextStyle( + fontSize: 24.sp, + fontWeight: FontWeight.bold, + ), + ), + const SizedBox(height: 8), + Text( + event!.description, + style: TextStyle( + fontSize: 16.sp, + color: Colors.grey[700], + ), + ), + const SizedBox(height: 16), + Text( + 'Cookbook ID: ${widget.cookbookId}\nRecipe ID: ${widget.recipeId}\nChallenge: ${widget.challenge}', + style: TextStyle(fontSize: 18.sp), + ), + const SizedBox(height: 20), + Center( + child: ElevatedButton( + onPressed: () => _stampTicket(context), + child: const Text('Stamp Ticket'), + ), + ), + ], + ), + ), + ); + } +} diff --git a/wallet/lib/pages/home/home.dart b/wallet/lib/pages/home/home.dart index a8cd050ca1..e92acd3a12 100644 --- a/wallet/lib/pages/home/home.dart +++ b/wallet/lib/pages/home/home.dart @@ -309,6 +309,16 @@ class HomeScreenState extends State with SingleTickerProviderStateMi ), ), ), + Positioned( + top: 0.06.sh, + right: 0.25.sw, + child: ElevatedButton( + onPressed: () { + Navigator.of(context).pushNamed(Routes.mobileQrScanner.name); + }, + child: const Text('Open Scanner'), + ), + ), Positioned( top: 0.06.sh, left: 0.09.sw, diff --git a/wallet/lib/services/data_stores/remote_data_store.dart b/wallet/lib/services/data_stores/remote_data_store.dart index f66686523a..1b5a6e5675 100644 --- a/wallet/lib/services/data_stores/remote_data_store.dart +++ b/wallet/lib/services/data_stores/remote_data_store.dart @@ -1,13 +1,13 @@ import 'dart:async'; import 'dart:convert'; import 'dart:developer'; +import 'dart:math' as math; import 'package:alan/proto/cosmos/bank/v1beta1/export.dart' as bank; import 'package:cosmos_utils/cosmos_utils.dart'; import 'package:dartz/dartz.dart'; import 'package:decimal/decimal.dart'; import 'package:easy_localization/easy_localization.dart'; -import 'package:firebase_app_check/firebase_app_check.dart'; import 'package:firebase_dynamic_links/firebase_dynamic_links.dart'; import 'package:fixnum/fixnum.dart'; import 'package:get_it/get_it.dart'; @@ -27,8 +27,10 @@ import 'package:pylons_wallet/model/pylon_items.dart'; import 'package:pylons_wallet/model/stripe_get_login_based_address.dart'; import 'package:pylons_wallet/model/transaction.dart'; import 'package:pylons_wallet/model/wallet_creation_model.dart'; -import 'package:pylons_wallet/modules/Pylonstech.pylons.pylons/module/export.dart' as pylons; -import 'package:pylons_wallet/modules/cosmos.tx.v1beta1/module/export.dart' as cosmos_tx; +import 'package:pylons_wallet/modules/Pylonstech.pylons.pylons/module/export.dart' + as pylons; +import 'package:pylons_wallet/modules/cosmos.tx.v1beta1/module/export.dart' + as cosmos_tx; import 'package:pylons_wallet/pages/detailed_asset_view/widgets/create_trade_bottom_sheet.dart'; import 'package:pylons_wallet/pages/home/currency_screen/model/ibc_coins.dart'; import 'package:pylons_wallet/pages/home/currency_screen/model/ibc_trace_model.dart'; @@ -37,6 +39,7 @@ import 'package:pylons_wallet/services/third_party_services/firestore_helper.dar import 'package:pylons_wallet/services/third_party_services/store_payment_service.dart'; import 'package:pylons_wallet/stores/models/transaction_response.dart'; import 'package:pylons_wallet/utils/base_env.dart'; +import 'package:pylons_wallet/utils/constants.dart' as constants; import 'package:pylons_wallet/utils/constants.dart'; import 'package:pylons_wallet/utils/custom_transaction_signing_gateaway/custom_transaction_signing_gateway.dart'; import 'package:pylons_wallet/utils/dependency_injection/dependency_injection.dart'; @@ -52,44 +55,49 @@ import '../../model/common.dart'; import '../../model/update_recipe_model.dart'; import '../../modules/Pylonstech.pylons.pylons/module/client/pylons/tx.pb.dart'; import '../../utils/types.dart'; -import 'package:pylons_wallet/utils/constants.dart' as constants; abstract class RemoteDataStore { /// This method is used to generate the stripe registration token /// Input: [Address] the address of the account /// Output : [StripeGenerateRegistrationTokenResponse] contains token of the account /// else will through error - Future generateStripeRegistrationToken({required String address}); + Future + generateStripeRegistrationToken({required String address}); /// This method is used to generate the stripe registration token /// Input: [Address] the address of the account /// Output : [StripeGenerateUpdateTokenResponse] contains token of the account /// else will through error - Future generateUpdateToken({required String address}); + Future generateUpdateToken( + {required String address}); /// This method is used to update the user stripe account /// Input: [StripeUpdateAccountRequest] the request parameters that are required for the update /// Output : [StripeUpdateAccountResponse] contains token of the account /// else will through error - Future updateStripeAccount({required StripeUpdateAccountRequest req}); + Future updateStripeAccount( + {required StripeUpdateAccountRequest req}); /// This method is used to get the account link and account id based on update token /// Input: [StripeUpdateAccountRequest] the request parameters that are required for the update /// Output : [StripeUpdateAccountResponse] contains account id and account link of the user stripe account /// else will through error - Future getAccountLinkBasedOnUpdateToken({required StripeUpdateAccountRequest req}); + Future getAccountLinkBasedOnUpdateToken( + {required StripeUpdateAccountRequest req}); /// This method is used to get the account link and account id based on update token /// Input: [StripeUpdateAccountRequest] the request parameters that are required for the update /// Output : [StripeUpdateAccountResponse] contains account id and account link of the user stripe account /// else will through error - Future registerAccount({required StripeRegisterAccountRequest req}); + Future registerAccount( + {required StripeRegisterAccountRequest req}); /// This method is used to get the account link or onboarding link if the account is not set up but user exists in system /// Input: [StripeGetLoginBasedOnAddressRequest] the request parameters that are required for the login link /// Output : [StripeGetLoginBasedOnAddressResponse] contains the response for the login link /// else will through error - Future getLoginLinkBasedOnAddress({required StripeGetLoginBasedOnAddressRequest req}); + Future getLoginLinkBasedOnAddress( + {required StripeGetLoginBasedOnAddressRequest req}); /// This method will return the ibc coin info /// Input: [ibcHash] hash of the ibc coin @@ -110,14 +118,16 @@ abstract class RemoteDataStore { /// Input: [address] the address of the user /// Output: returns [List][TransactionHistory] if success /// else will through error - Future> getTransactionHistory({required String address}); + Future> getTransactionHistory( + {required String address}); /// This method will return list of notification messages /// Input: [walletAddress] of the user ,[limit] the maximum number of records to show at a time /// and [offset] is the last item's position /// Output: if successful will return [List][NotificationMessage] the list of the NotificationMessage /// else will throw error - Future> getAllNotificationMessages({required String walletAddress, required int limit, required int offset}); + Future> getAllNotificationMessages( + {required String walletAddress, required int limit, required int offset}); /// This method will update the recipe in the chain /// Input: [updateRecipeModel] contains the info regarding the recipe update @@ -128,19 +138,22 @@ abstract class RemoteDataStore { /// This method is used to get likes count of NFT /// Input: [recipeId],[cookBookID] and [walletAddress] of the given NFT /// Output : [int] will contain the likes count of the NFT - Future getLikesCount({required RecipeId recipeId, required CookbookId cookBookID}); + Future getLikesCount( + {required RecipeId recipeId, required CookbookId cookBookID}); /// This method is used to get history of nft owners /// Input: [itemId] and [cookBookID] of the NFT /// Output : [List][NftOwnershipHistory] will contain the list of NftOwnershipHistory data if success /// else will throw error - Future> getNftOwnershipHistory({required String itemId, required String cookBookId}); + Future> getNftOwnershipHistory( + {required String itemId, required String cookBookId}); /// This method is used to get history of nft owners /// Input: [cookBookId] and [recipeId] of the NFT /// Output : [List][NftOwnershipHistory] will contain the list of NftOwnershipHistory data if success /// else will throw error - Future> getNftOwnershipHistoryByCookbookIdAndRecipeId({ + Future> + getNftOwnershipHistoryByCookbookIdAndRecipeId({ required CookbookId cookBookId, required RecipeId recipeId, }); @@ -148,7 +161,8 @@ abstract class RemoteDataStore { /// This method is used to get views count of NFT /// Input: [recipeId],[cookBookID] and [walletAddress] of the given NFT /// Output : [int] will contain the views count of the NFT - Future getViewsCount({required RecipeId recipeId, required CookbookId cookBookID}); + Future getViewsCount( + {required RecipeId recipeId, required CookbookId cookBookID}); /// This method is used to increment a view for an NFT when a user view an NFT /// Input: [recipeId],[cookBookID] and [walletAddress] of the given NFT @@ -178,7 +192,8 @@ abstract class RemoteDataStore { /// This method is used get the recipe object based on CookbookId and RecipeId /// Input: [recipeId] and [cookBookId] of the given Recipe /// Output : [pylons.Recipe] will be the recipe object based on CookbookId and RecipeId - Future getRecipe({required CookbookId cookBookId, required RecipeId recipeId}); + Future getRecipe( + {required CookbookId cookBookId, required RecipeId recipeId}); /// This method is used get the username based on the wallet address /// Input: [address] wll be the public address of the user's wallet @@ -188,7 +203,8 @@ abstract class RemoteDataStore { /// This method is used get the list of recipes object based on [cookBookId] /// Input: [cookBookId] of the given Recipe /// Output : [pylons.Recipe] will be the list of recipes object based on CookbookId - Future> getRecipesBasedOnCookBookId({required String cookBookId}); + Future> getRecipesBasedOnCookBookId( + {required String cookBookId}); /// This method is used get the cookbook object based on [cookBookId] /// Input: [cookBookId] of the given Cookbook @@ -203,12 +219,14 @@ abstract class RemoteDataStore { /// This method is used get the list of executions based on [cookBookId] and [recipeId] /// Input: [cookBookId] and [recipeId] of the given Recipe /// Output : [ExecutionListByRecipeResponse] will be the list of recipes executions - Future getExecutionsByRecipeId({required String cookBookId, required String recipeId}); + Future getExecutionsByRecipeId( + {required String cookBookId, required String recipeId}); /// This method is used get the item based on [cookBookId] and [itemId] /// Input: [cookBookId] and [itemId] of the given Recipe /// Output : [pylons.Item] will be the item object - Future getItem({required String cookBookId, required String itemId}); + Future getItem( + {required String cookBookId, required String itemId}); /// This method is used get the list if items based on [owner] /// Input: [owner] will be the owner name to get the list of his items @@ -223,25 +241,29 @@ abstract class RemoteDataStore { /// This method is used get the list of trades based on its [creator] /// Input: [creator] will be the creator name to get trades /// Output : [pylons.Trade] will be the the list of trades based on its [creator] - Future> getTradesBasedOnCreator({required Address creator}); + Future> getTradesBasedOnCreator( + {required Address creator}); /// This method is used get the list of cookbooks based on its wallet [address] /// Input: [address] will the public address of user's wallet /// Output : [pylons.Trade] will be the the list of trades based on its [creator] - Future> getCookbooksByCreator({required String address}); + Future> getCookbooksByCreator( + {required String address}); /// This method is used get the trade based on its [id] /// Input: [id] will the the trade id to get complete trade /// Output : [pylons.Trade] will be the trade based on its [id] Future getTradeByID({required Int64 id}); - Future sendGoogleInAppPurchaseCoinsRequest(GoogleInAppPurchaseModel googleInAppPurchaseModel); + Future sendGoogleInAppPurchaseCoinsRequest( + GoogleInAppPurchaseModel googleInAppPurchaseModel); /// This method will send apple in app purchase request to the chain /// Input: [AppleInAppPurchaseModel] contains the input data for the api. /// Output: if successful will return the [String] hash of the transaction /// else will throw the error - Future sendAppleInAppPurchaseCoinsRequest(AppleInAppPurchaseModel googleInAppPurchaseModel); + Future sendAppleInAppPurchaseCoinsRequest( + AppleInAppPurchaseModel googleInAppPurchaseModel); /// This method will get the products For Sale from the underling store /// Input: [itemId] the id of the item @@ -266,12 +288,16 @@ abstract class RemoteDataStore { /// This method will update fcm token against an address. /// Input: [address] the address whose fcm needs to be updated, [fcmToken] the user fcm token , [appCheckToken] the firebase app check token /// Output: [bool] returns true if successful else will throw error. - Future updateFcmToken({required String address, required String fcmToken, required String appCheckToken}); + Future updateFcmToken( + {required String address, + required String fcmToken, + required String appCheckToken}); /// This method will mark notification as read. /// Input: [List][id] the list containing ids of notifications which needed to be mark as read, [appCheckToken] the firebase app check token /// Output: [bool] returns true if successful else will throw error. - Future markNotificationAsRead({required List idsList, required String appCheckToken}); + Future markNotificationAsRead( + {required List idsList, required String appCheckToken}); /// This method will create dynamic link for the user invite /// Input : [address] the address against which the invite link to be generated @@ -281,17 +307,22 @@ abstract class RemoteDataStore { /// This method will create dynamic link for the nft share recipe /// Input : [address] the address & [NFT] against which the invite link to be generated /// Output: [String] return the generated dynamic link else will throw error - Future createDynamicLinkForRecipeNftShare({required String address, required NFT nft}); + Future createDynamicLinkForRecipeNftShare( + {required String address, required NFT nft}); /// This method will create dynamic link for the nft share trade /// Input : [address] the address & [tradeId] against which the invite link to be generated /// Output: [String] return the generated dynamic link else will throw error - Future createDynamicLinkForTradeNftShare({required String address, required String tradeId}); + Future createDynamicLinkForTradeNftShare( + {required String address, required String tradeId}); /// This method will create dynamic link for the nft share purchase /// Input : [address] the address & [itemId] the id of the item& [cookbookId] against which the invite link to be generated /// Output: [String] return the generated dynamic link else will throw error - Future createDynamicLinkForItemNftShare({required String address, required String itemId, required String cookbookId}); + Future createDynamicLinkForItemNftShare( + {required String address, + required String itemId, + required String cookbookId}); /// This method will create User account based on account public info /// Input: [publicInfo] contains info related to user chain address, [walletCreationModel] contains user entered data, [appCheckToken] the app specific token, [referralToken] the invitee user address @@ -306,7 +337,10 @@ abstract class RemoteDataStore { /// This method will save users feedback to firebase based on its wallet address /// Input : [walletAddress], [subject] and [feedback] the address against which the feedbacks needs to be stored /// Output : [bool] It will return true if the saving feedback is successful otherwise false - Future saveUserFeedback({required String walletAddress, required String subject, required String feedback}); + Future saveUserFeedback( + {required String walletAddress, + required String subject, + required String feedback}); /// This method will set the app level user identifier in the analytics /// Input: [address] the address of the user @@ -326,7 +360,8 @@ abstract class RemoteDataStore { /// Get a tuple of the native Cosmos TX struct and (if it exists) the TxResponse struct. /// Input: [hash] hash of the transaction to query. /// Output: [Tuple2] of the [cosmos_tx.Tx] and (if it exists) the [cosmos_tx.TxResponse]. - Future> getTx({required String hash}); + Future> getTx( + {required String hash}); Future logAddToCart({ required RecipeId recipeId, @@ -351,16 +386,26 @@ abstract class RemoteDataStore { required Address creatorAddress, }); - Future cancelTrade({required TradeId tradeId, required Address address}); + Future cancelTrade( + {required TradeId tradeId, required Address address}); - Future createTrade({required pylons.MsgCreateTrade msgCreateTrade}); + Future createTrade( + {required pylons.MsgCreateTrade msgCreateTrade}); Future> getPylonItem({required Address address}); /// This method will create dynamic link for the event share recipe /// Input : [address] the address & [Event] against which the invite link to be generated /// Output: [String] return the generated dynamic link else will throw error - Future createDynamicLinkForRecipeEventShare({required String address, required Events events}); + Future createDynamicLinkForRecipeEventShare( + {required String address, required Events events}); + + Future stampTicket({ + required CookbookId cookBookId, + required RecipeId recipeId, + required Address creatorAddress, + required String challenge, + }); } class RemoteDataStoreImp implements RemoteDataStore { @@ -372,7 +417,6 @@ class RemoteDataStoreImp implements RemoteDataStore { required this.firebaseHelper, required this.analyticsHelper, required this.storePaymentService, - required this.firebaseAppCheck, required this.onLogError, }); @@ -385,7 +429,8 @@ class RemoteDataStoreImp implements RemoteDataStore { final baseApiUrl = getBaseEnv().baseMongoUrl; final body = {userIdKey: walletAddress}; - final uri = Uri.parse("$baseApiUrl/api/actions/likes/$cookBookID/$recipeId"); + final uri = + Uri.parse("$baseApiUrl/api/actions/likes/$cookBookID/$recipeId"); final response = await httpClient.post(uri, body: body).timeout( timeOutDuration, @@ -406,10 +451,12 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future getLikesCount({required RecipeId recipeId, required CookbookId cookBookID}) async { + Future getLikesCount( + {required RecipeId recipeId, required CookbookId cookBookID}) async { final baseApiUrl = GetIt.I.get().baseMongoUrl; - final uri = Uri.parse("$baseApiUrl/api/actions/likes/$cookBookID/$recipeId"); + final uri = + Uri.parse("$baseApiUrl/api/actions/likes/$cookBookID/$recipeId"); log("$baseApiUrl/api/actions/likes/$cookBookID/$recipeId"); @@ -445,7 +492,8 @@ class RemoteDataStoreImp implements RemoteDataStore { final body = {userIdKey: walletAddress.toString()}; - final uri = Uri.parse("$baseApiUrl/api/actions/views/$cookBookID/$recipeId"); + final uri = + Uri.parse("$baseApiUrl/api/actions/views/$cookBookID/$recipeId"); final response = await httpClient.post(uri, body: body).timeout( timeOutDuration, @@ -466,10 +514,12 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future getViewsCount({required RecipeId recipeId, required CookbookId cookBookID}) async { + Future getViewsCount( + {required RecipeId recipeId, required CookbookId cookBookID}) async { final baseApiUrl = GetIt.I.get().baseMongoUrl; - final uri = Uri.parse("$baseApiUrl/api/actions/views/$cookBookID/$recipeId"); + final uri = + Uri.parse("$baseApiUrl/api/actions/views/$cookBookID/$recipeId"); final response = await httpClient.get(uri).timeout( timeOutDuration, @@ -501,7 +551,8 @@ class RemoteDataStoreImp implements RemoteDataStore { }) async { final baseApiUrl = GetIt.I.get().baseMongoUrl; - final uri = Uri.parse("$baseApiUrl/api/actions/likes/$walletAddress/$cookBookID/$recipeId"); + final uri = Uri.parse( + "$baseApiUrl/api/actions/likes/$walletAddress/$cookBookID/$recipeId"); final response = await httpClient.get(uri).timeout( timeOutDuration, @@ -526,19 +577,25 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future generateStripeRegistrationToken({required String address}) async { + Future + generateStripeRegistrationToken({required String address}) async { final baseEnv = getBaseEnv(); - final response = await httpClient.get(Uri.parse("${baseEnv.baseStripeUrl}/generate-registration-token?address=$address")).timeout( + final response = await httpClient + .get(Uri.parse( + "${baseEnv.baseStripeUrl}/generate-registration-token?address=$address")) + .timeout( timeOutDuration, ); log(response.body, name: "Stripe | generateStripeRegistrationToken"); if (response.statusCode == API_SUCCESS_CODE) { - return StripeGenerateRegistrationTokenResponse.fromJson(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeGenerateRegistrationTokenResponse.fromJson( + json.decode(utf8.decode(response.bodyBytes)) as Map); } - if (response.statusCode == API_ERROR_CODE && response.body == 'account already exists\n') { + if (response.statusCode == API_ERROR_CODE && + response.body == 'account already exists\n') { throw const AccountAlreadyExistsFailure('Account Already exists'); } @@ -546,16 +603,19 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future generateUpdateToken({required String address}) async { + Future generateUpdateToken( + {required String address}) async { final baseEnv = getBaseEnv(); final response = await httpClient .get( - Uri.parse("${baseEnv.baseStripeUrl}/generate-update-token?address=$address"), + Uri.parse( + "${baseEnv.baseStripeUrl}/generate-update-token?address=$address"), ) .timeout(timeOutDuration); if (response.statusCode == API_SUCCESS_CODE) { - return StripeGenerateUpdateTokenResponse.fromJson(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeGenerateUpdateTokenResponse.fromJson( + json.decode(utf8.decode(response.bodyBytes)) as Map); } log(response.body, name: "Stripe | generateUpdateToken"); @@ -564,7 +624,8 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future updateStripeAccount({required StripeUpdateAccountRequest req}) async { + Future updateStripeAccount( + {required StripeUpdateAccountRequest req}) async { final baseEnv = getBaseEnv(); final response = await httpClient .post( @@ -577,14 +638,16 @@ class RemoteDataStoreImp implements RemoteDataStore { ); if (response.statusCode == API_SUCCESS_CODE) { - return StripeUpdateAccountResponse.fromJson(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeUpdateAccountResponse.fromJson( + json.decode(utf8.decode(response.bodyBytes)) as Map); } throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; } @override - Future getAccountLinkBasedOnUpdateToken({required StripeUpdateAccountRequest req}) async { + Future getAccountLinkBasedOnUpdateToken( + {required StripeUpdateAccountRequest req}) async { final baseEnv = getBaseEnv(); final response = await httpClient .post( @@ -599,7 +662,8 @@ class RemoteDataStoreImp implements RemoteDataStore { log(response.body, name: "Stripe | getAccountLinkBasedOnUpdateToken"); if (response.statusCode == API_SUCCESS_CODE) { - return StripeUpdateAccountResponse.fromJson(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeUpdateAccountResponse.fromJson( + json.decode(utf8.decode(response.bodyBytes)) as Map); } throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; @@ -615,14 +679,16 @@ class RemoteDataStoreImp implements RemoteDataStore { .timeout(timeOutDuration); if (response.statusCode == API_SUCCESS_CODE) { - return IBCTraceModel.fromJson(jsonDecode(response.body) as Map, ibcHash); + return IBCTraceModel.fromJson( + jsonDecode(response.body) as Map, ibcHash); } throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; } @override - Future getLoginLinkBasedOnAddress({required StripeGetLoginBasedOnAddressRequest req}) async { + Future getLoginLinkBasedOnAddress( + {required StripeGetLoginBasedOnAddressRequest req}) async { final baseEnv = getBaseEnv(); final response = await httpClient .post( @@ -634,14 +700,16 @@ class RemoteDataStoreImp implements RemoteDataStore { log(response.body, name: "Stripe | getLoginLinkBasedOnAddress"); if (response.statusCode == API_SUCCESS_CODE) { - return StripeGetLoginBasedOnAddressResponse.from(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeGetLoginBasedOnAddressResponse.from( + json.decode(utf8.decode(response.bodyBytes)) as Map); } throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; } @override - Future registerAccount({required StripeRegisterAccountRequest req}) async { + Future registerAccount( + {required StripeRegisterAccountRequest req}) async { final baseEnv = getBaseEnv(); final response = await httpClient .post( @@ -653,10 +721,12 @@ class RemoteDataStoreImp implements RemoteDataStore { log(response.body, name: "Stripe | registerAccount"); if (response.statusCode == API_SUCCESS_CODE) { - return StripeRegisterAccountResponse.from(json.decode(utf8.decode(response.bodyBytes)) as Map); + return StripeRegisterAccountResponse.from( + json.decode(utf8.decode(response.bodyBytes)) as Map); } - if (response.statusCode == API_ERROR_CODE && response.body == 'account already setup\n') { + if (response.statusCode == API_ERROR_CODE && + response.body == 'account already setup\n') { throw const AccountAlreadyExistsFailure('Account Already exists'); } @@ -684,16 +754,21 @@ class RemoteDataStoreImp implements RemoteDataStore { Future> getBalance(String walletAddress) async { final bank.QueryClient bankQueryClient = getBankQueryClient(); - final queryAllBalancesRequest = bank.QueryAllBalancesRequest.create()..address = walletAddress; + final queryAllBalancesRequest = bank.QueryAllBalancesRequest.create() + ..address = walletAddress; final response = await bankQueryClient.allBalances(queryAllBalancesRequest); final balances = []; - if (response.balances.isEmpty || response.balances.indexWhere((element) => element.denom == 'upylon') == -1) { + if (response.balances.isEmpty || + response.balances.indexWhere((element) => element.denom == 'upylon') == + -1) { balances.add(Balance(denom: "upylon", amount: Amount(Decimal.zero))); } - if (response.balances.indexWhere((element) => element.denom == 'ustripeusd') == -1) { + if (response.balances + .indexWhere((element) => element.denom == 'ustripeusd') == + -1) { balances.add(Balance(denom: "ustripeusd", amount: Amount(Decimal.zero))); } @@ -701,35 +776,45 @@ class RemoteDataStoreImp implements RemoteDataStore { for (final balance in response.balances) { if (balance.denom.contains("ibc")) { - futureIBCHashList.add(getIBCHashTrace(ibcHash: balance.denom.replaceAll("ibc/", ''))); + futureIBCHashList.add( + getIBCHashTrace(ibcHash: balance.denom.replaceAll("ibc/", ''))); continue; } - balances.add(Balance(denom: balance.denom, amount: Amount(Decimal.parse(balance.amount)))); + balances.add(Balance( + denom: balance.denom, amount: Amount(Decimal.parse(balance.amount)))); } final ibcHashResults = await Future.wait(futureIBCHashList); for (final ibcHashResult in ibcHashResults) { - final relatedBalance = response.balances.firstWhere((element) => element.denom == "ibc/${ibcHashResult.ibcHash}"); - balances.add(Balance(denom: ibcHashResult.denomTrace.baseDenom.name, amount: Amount(Decimal.parse(relatedBalance.amount)))); + final relatedBalance = response.balances.firstWhere( + (element) => element.denom == "ibc/${ibcHashResult.ibcHash}"); + balances.add(Balance( + denom: ibcHashResult.denomTrace.baseDenom.name, + amount: Amount(Decimal.parse(relatedBalance.amount)))); } return balances; } @override - Future> getTransactionHistory({required String address}) async { + Future> getTransactionHistory( + {required String address}) async { final baseEnv = getBaseEnv(); final transactionList = []; - final transactionUri = '${baseEnv.baseApiUrl}/pylons/tx?address=%27$address%27'; + final transactionUri = + '${baseEnv.baseApiUrl}/pylons/tx?address=%27$address%27'; - final transactionResponse = await httpClient.get(Uri.parse(transactionUri)).timeout(timeOutDuration); + final transactionResponse = await httpClient + .get(Uri.parse(transactionUri)) + .timeout(timeOutDuration); final transactionMap = jsonDecode(transactionResponse.body); - if (transactionMap == null || transactionResponse.body.length == 2) return transactionList; + if (transactionMap == null || transactionResponse.body.length == 2) + return transactionList; transactionList.addAll((transactionMap as List) .map( (e) => TransactionHistory.fromJson(e as Map), @@ -740,12 +825,17 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getAllNotificationMessages({required String walletAddress, required int limit, required int offset}) async { + Future> getAllNotificationMessages( + {required String walletAddress, + required int limit, + required int offset}) async { final baseApiUrl = getBaseEnv().baseMongoUrl; - final uri = Uri.parse("$baseApiUrl/api/notifications/getAllNotifications/$walletAddress/$limit/$offset"); + final uri = Uri.parse( + "$baseApiUrl/api/notifications/getAllNotifications/$walletAddress/$limit/$offset"); - final notificationResponse = await httpClient.get(uri).timeout(timeOutDuration); + final notificationResponse = + await httpClient.get(uri).timeout(timeOutDuration); if (notificationResponse.statusCode != API_SUCCESS_CODE) { throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; @@ -766,9 +856,11 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getNftOwnershipHistory({required String itemId, required String cookBookId}) async { + Future> getNftOwnershipHistory( + {required String itemId, required String cookBookId}) async { final baseApiUrl = getBaseEnv().baseApiUrl; - final uri = Uri.parse("$baseApiUrl/pylons/item_history/$cookBookId/$itemId"); + final uri = + Uri.parse("$baseApiUrl/pylons/item_history/$cookBookId/$itemId"); final List historyList = []; final historyResponse = await httpClient.get(uri).timeout(timeOutDuration); @@ -792,13 +884,15 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getNftOwnershipHistoryByCookbookIdAndRecipeId({ + Future> + getNftOwnershipHistoryByCookbookIdAndRecipeId({ required CookbookId cookBookId, required RecipeId recipeId, }) async { final baseApiUrl = getBaseEnv().baseApiUrl; // ignore: noop_primitive_operations - final uri = Uri.parse("$baseApiUrl/pylons/get_recipe_history/${cookBookId.toString()}/${recipeId.toString()}"); + final uri = Uri.parse( + "$baseApiUrl/pylons/get_recipe_history/${cookBookId.toString()}/${recipeId.toString()}"); final List historyList = []; @@ -813,7 +907,8 @@ class RemoteDataStoreImp implements RemoteDataStore { throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; } historyMap[kHistory].map((e) { - final nft = NftOwnershipHistory.fromCookBookAndRecipeJson(e as Map); + final nft = NftOwnershipHistory.fromCookBookAndRecipeJson( + e as Map); historyList.add(nft); }).toList(); @@ -824,10 +919,12 @@ class RemoteDataStoreImp implements RemoteDataStore { Future _signAndBroadcast(GeneratedMessage message) async { final unsignedTransaction = UnsignedAlanTransaction(messages: [message]); - final customTransactionSigningGateway = getCustomTransactionSigningGateway(); + final customTransactionSigningGateway = + getCustomTransactionSigningGateway(); final transactionSigningGateway = getTransactionSigningGateway(); - final walletsResultEither = await customTransactionSigningGateway.getWalletsList(); + final walletsResultEither = + await customTransactionSigningGateway.getWalletsList(); if (walletsResultEither.isLeft()) { onLogError(walletsResultEither.getLeft().toString(), fatal: true); @@ -890,17 +987,21 @@ class RemoteDataStoreImp implements RemoteDataStore { final recipeProto3Json = updateRecipeModel.recipe.toProto3Json()! as Map; recipeProto3Json.remove(kCreatedAtCamelCase); recipeProto3Json.remove(kUpdatedAtCamelCase); - final msgUpdateRecipe = pylons.MsgUpdateRecipe.create()..mergeFromProto3Json(recipeProto3Json); + final msgUpdateRecipe = pylons.MsgUpdateRecipe.create() + ..mergeFromProto3Json(recipeProto3Json); msgUpdateRecipe.enabled = updateRecipeModel.enabledStatus; msgUpdateRecipe.creator = updateRecipeModel.publicAddress; - msgUpdateRecipe.coinInputs.first.coins.first.amount = updateRecipeModel.nftPrice; - msgUpdateRecipe.coinInputs.first.coins.first.denom = updateRecipeModel.denom; + msgUpdateRecipe.coinInputs.first.coins.first.amount = + updateRecipeModel.nftPrice; + msgUpdateRecipe.coinInputs.first.coins.first.denom = + updateRecipeModel.denom; msgUpdateRecipe.version = msgUpdateRecipe.version.incrementRecipeVersion(); return _signAndBroadcast(msgUpdateRecipe); } @override - Future getRecipe({required CookbookId cookBookId, required RecipeId recipeId}) async { + Future getRecipe( + {required CookbookId cookBookId, required RecipeId recipeId}) async { try { const retry = RetryOptions(); final response = await retry.retry( @@ -929,7 +1030,8 @@ class RemoteDataStoreImp implements RemoteDataStore { Future getUsername({required Address address}) async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryGetUsernameByAddressRequest.create()..address = address.toString(); + final request = pylons.QueryGetUsernameByAddressRequest.create() + ..address = address.toString(); final response = await queryClient.usernameByAddress(request); @@ -940,8 +1042,10 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getTx({required String hash}) async { - final cosmos_tx.ServiceClient serviceClient = cosmos_tx.ServiceClient(sl.get().networkInfo.gRPCChannel); + Future> getTx( + {required String hash}) async { + final cosmos_tx.ServiceClient serviceClient = + cosmos_tx.ServiceClient(sl.get().networkInfo.gRPCChannel); final request = cosmos_tx.GetTxRequest.create()..hash = hash; final response = await serviceClient.getTx(request); @@ -957,9 +1061,11 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getRecipesBasedOnCookBookId({required String cookBookId}) async { + Future> getRecipesBasedOnCookBookId( + {required String cookBookId}) async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryListRecipesByCookbookRequest.create()..cookbookId = cookBookId; + final request = pylons.QueryListRecipesByCookbookRequest.create() + ..cookbookId = cookBookId; final response = await queryClient.listRecipesByCookbook(request); @@ -967,14 +1073,16 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future getCookbookBasedOnId({required String cookBookId}) async { + Future getCookbookBasedOnId( + {required String cookBookId}) async { try { const retry = RetryOptions(); final response = await retry.retry( () async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryGetCookbookRequest.create()..id = cookBookId; + final request = pylons.QueryGetCookbookRequest.create() + ..id = cookBookId; final response = await queryClient.cookbook(request); return response; @@ -996,7 +1104,8 @@ class RemoteDataStoreImp implements RemoteDataStore { Future getAddressBasedOnUsername(String username) async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryGetAddressByUsernameRequest.create()..username = username; + final request = pylons.QueryGetAddressByUsernameRequest.create() + ..username = username; final response = await queryClient.addressByUsername(request); @@ -1008,18 +1117,24 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future getExecutionsByRecipeId({required String cookBookId, required String recipeId}) async { + Future getExecutionsByRecipeId( + {required String cookBookId, required String recipeId}) async { final pylons.QueryClient queryClient = getQueryClient(); - final queryExecutionListByRecipe = pylons.QueryListExecutionsByRecipeRequest() - ..cookbookId = cookBookId - ..recipeId = recipeId; - final response = await queryClient.listExecutionsByRecipe(queryExecutionListByRecipe); + final queryExecutionListByRecipe = + pylons.QueryListExecutionsByRecipeRequest() + ..cookbookId = cookBookId + ..recipeId = recipeId; + final response = + await queryClient.listExecutionsByRecipe(queryExecutionListByRecipe); - return ExecutionListByRecipeResponse(completedExecutions: response.completedExecutions, pendingExecutions: response.pendingExecutions); + return ExecutionListByRecipeResponse( + completedExecutions: response.completedExecutions, + pendingExecutions: response.pendingExecutions); } @override - Future getItem({required String cookBookId, required String itemId}) async { + Future getItem( + {required String cookBookId, required String itemId}) async { final pylons.QueryClient queryClient = getQueryClient(); final queryGetItemRequest = pylons.QueryGetItemRequest() ..cookbookId = cookBookId @@ -1037,7 +1152,8 @@ class RemoteDataStoreImp implements RemoteDataStore { @override Future> getListItemByOwner({required Address owner}) async { final pylons.QueryClient queryClient = getQueryClient(); - final queryListItemByOwner = pylons.QueryListItemByOwnerRequest()..owner = owner.toString(); + final queryListItemByOwner = pylons.QueryListItemByOwnerRequest() + ..owner = owner.toString(); final response = await queryClient.listItemByOwner(queryListItemByOwner); @@ -1059,7 +1175,10 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future updateFcmToken({required String address, required String fcmToken, required String appCheckToken}) async { + Future updateFcmToken( + {required String address, + required String fcmToken, + required String appCheckToken}) async { final baseApiUrl = getBaseEnv().baseMongoUrl; final uri = Uri.parse("$baseApiUrl/api/fcmtoken/update/$address/$fcmToken"); @@ -1078,12 +1197,18 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future markNotificationAsRead({required List idsList, required String appCheckToken}) async { + Future markNotificationAsRead( + {required List idsList, required String appCheckToken}) async { final baseApiUrl = getBaseEnv().baseMongoUrl; final uri = Uri.parse("$baseApiUrl/api/notifications/markread"); - final response = await httpClient.post(uri, headers: {FIREBASE_APP_CHECK_HEADER: appCheckToken, 'Content-type': 'application/json'}, body: jsonEncode({kNotificationsIds: idsList})); + final response = await httpClient.post(uri, + headers: { + FIREBASE_APP_CHECK_HEADER: appCheckToken, + 'Content-type': 'application/json' + }, + body: jsonEncode({kNotificationsIds: idsList})); if (response.statusCode == API_SUCCESS_CODE) { final historyMap = jsonDecode(response.body); @@ -1097,9 +1222,11 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getTradesBasedOnCreator({required Address creator}) async { + Future> getTradesBasedOnCreator( + {required Address creator}) async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryListTradesByCreatorRequest.create()..creator = creator.toString(); + final request = pylons.QueryListTradesByCreatorRequest.create() + ..creator = creator.toString(); final response = await queryClient.listTradesByCreator(request); @@ -1110,13 +1237,17 @@ class RemoteDataStoreImp implements RemoteDataStore { throw TradeNotFoundFailure(LocaleKeys.trade_not_found.tr()); } + String generateRandomString(int length) { + const characters = + 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'; + final math.Random random = math.Random(); + return String.fromCharCodes(Iterable.generate(length, + (_) => characters.codeUnitAt(random.nextInt(characters.length)))); + } + @override Future getAppCheckToken() async { - final String? token = await firebaseAppCheck.getToken(); - - if (token == null || token.isEmpty) { - throw const AppCheckTokenFailure(HandlerFactory.ERR_SOMETHING_WENT_WRONG); - } + final Future token = Future.value(generateRandomString(10)); return token; } @@ -1126,9 +1257,11 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future> getCookbooksByCreator({required String address}) async { + Future> getCookbooksByCreator( + {required String address}) async { final pylons.QueryClient queryClient = getQueryClient(); - final request = pylons.QueryListCookbooksByCreatorRequest.create()..creator = address; + final request = pylons.QueryListCookbooksByCreatorRequest.create() + ..creator = address; final response = await queryClient.listCookbooksByCreator(request); return response.cookbooks; @@ -1149,12 +1282,16 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future sendGoogleInAppPurchaseCoinsRequest(GoogleInAppPurchaseModel googleInAppPurchaseModel) async { - final msgGoogleInAPPPurchase = pylons.MsgGoogleInAppPurchaseGetCoins()..mergeFromProto3Json(googleInAppPurchaseModel.toJson()); + Future sendGoogleInAppPurchaseCoinsRequest( + GoogleInAppPurchaseModel googleInAppPurchaseModel) async { + final msgGoogleInAPPPurchase = pylons.MsgGoogleInAppPurchaseGetCoins() + ..mergeFromProto3Json(googleInAppPurchaseModel.toJson()); - final customTransactionSigningGateway = getCustomTransactionSigningGateway(); + final customTransactionSigningGateway = + getCustomTransactionSigningGateway(); - final walletsResultEither = await customTransactionSigningGateway.getWalletsList(); + final walletsResultEither = + await customTransactionSigningGateway.getWalletsList(); if (walletsResultEither.isLeft()) { onLogError(walletsResultEither.getLeft().message, fatal: true); throw const TransactionSigningFailure( @@ -1178,8 +1315,10 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future sendAppleInAppPurchaseCoinsRequest(AppleInAppPurchaseModel appleInAppPurchaseModel) { - final msgAppleInAppPurchases = pylons.MsgAppleIap()..mergeFromProto3Json(appleInAppPurchaseModel.toJson()); + Future sendAppleInAppPurchaseCoinsRequest( + AppleInAppPurchaseModel appleInAppPurchaseModel) { + final msgAppleInAppPurchases = pylons.MsgAppleIap() + ..mergeFromProto3Json(appleInAppPurchaseModel.toJson()); return _signAndBroadcast(msgAppleInAppPurchases); } @@ -1200,11 +1339,13 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future createDynamicLinkForUserInvite({required String address}) async { + Future createDynamicLinkForUserInvite( + {required String address}) async { final dynamicLinkParams = DynamicLinkParameters( link: Uri.parse("https://wallet.pylons.tech/invite/$address"), uriPrefix: "https://pylons.page.link/", - androidParameters: const AndroidParameters(packageName: "tech.pylons.wallet"), + androidParameters: + const AndroidParameters(packageName: "tech.pylons.wallet"), iosParameters: const IOSParameters( bundleId: "xyz.pylons.wallet", ), @@ -1215,27 +1356,33 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future createDynamicLinkForRecipeNftShare({required String address, required NFT nft}) async { + Future createDynamicLinkForRecipeNftShare( + {required String address, required NFT nft}) async { final updateText = nft.ibcCoins.getAbbrev() == constants.kPYLN_ABBREVATION ? "\$${nft.ibcCoins.pylnToCredit(nft.ibcCoins.getCoinWithProperDenomination(nft.price))}" : "${nft.ibcCoins.getCoinWithProperDenomination(nft.price)} ${nft.ibcCoins.getAbbrev()}"; final dynamicLinkParams = DynamicLinkParameters( - link: Uri.parse("$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), + link: Uri.parse( + "$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), uriPrefix: kDeepLink, androidParameters: AndroidParameters( packageName: packageName, - fallbackUrl: Uri.parse("$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), ), iosParameters: IOSParameters( bundleId: bundleId, - fallbackUrl: Uri.parse("$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?recipe_id=${nft.recipeID}&cookbook_id=${nft.cookbookID}&address=$address"), ), - navigationInfoParameters: const NavigationInfoParameters(forcedRedirectEnabled: true), + navigationInfoParameters: + const NavigationInfoParameters(forcedRedirectEnabled: true), socialMetaTagParameters: SocialMetaTagParameters( title: nft.name, imageUrl: Uri.parse(nft.url), - description: '${nft.description} Price:${nft.price == "0" ? LocaleKeys.free.tr() : updateText}', + description: + '${nft.description} Price:${nft.price == "0" ? LocaleKeys.free.tr() : updateText}', )); final link = await dynamicLinksGenerator.buildShortLink( @@ -1257,7 +1404,8 @@ class RemoteDataStoreImp implements RemoteDataStore { required String appCheckToken, required String referralToken, }) async { - final customTransactionSigningGateway = getCustomTransactionSigningGateway(); + final customTransactionSigningGateway = + getCustomTransactionSigningGateway(); final walletLookupKey = createWalletLookUp(publicInfo); final msgObj = pylons.MsgCreateAccount( @@ -1268,7 +1416,10 @@ class RemoteDataStoreImp implements RemoteDataStore { final unsignedTransaction = UnsignedAlanTransaction(messages: [msgObj]); - final result = await customTransactionSigningGateway.signTransaction(transaction: unsignedTransaction, walletLookupKey: walletLookupKey).mapError((error) { + final result = await customTransactionSigningGateway + .signTransaction( + transaction: unsignedTransaction, walletLookupKey: walletLookupKey) + .mapError((error) { throw error; }).flatMap( (signed) => customTransactionSigningGateway.broadcastTransaction( @@ -1297,7 +1448,8 @@ class RemoteDataStoreImp implements RemoteDataStore { recipeProto3Json.remove(kCreatedAtCamelCase); recipeProto3Json.remove(kUpdatedAtCamelCase); - final msgUpdateRecipe = pylons.MsgUpdateRecipe.create()..mergeFromProto3Json(recipeProto3Json); + final msgUpdateRecipe = pylons.MsgUpdateRecipe.create() + ..mergeFromProto3Json(recipeProto3Json); msgUpdateRecipe.enabled = enabled; msgUpdateRecipe.version = msgUpdateRecipe.version.incrementRecipeVersion(); msgUpdateRecipe.creator = creatorAddress.toString(); @@ -1305,6 +1457,29 @@ class RemoteDataStoreImp implements RemoteDataStore { return _signAndBroadcast(msgUpdateRecipe); } + @override + Future stampTicket({ + required CookbookId cookBookId, + required RecipeId recipeId, + required Address creatorAddress, + required String challenge, + }) async { + final recipe = await getRecipe(cookBookId: cookBookId, recipeId: recipeId); + + final recipeProto3Json = recipe.toProto3Json()! as Map; + recipeProto3Json.remove(kCreatedAtCamelCase); + recipeProto3Json.remove(kUpdatedAtCamelCase); + + final msgUpdateRecipe = pylons.MsgUpdateRecipe.create() + ..mergeFromProto3Json(recipeProto3Json); + + msgUpdateRecipe.extraInfo = challenge; + msgUpdateRecipe.version = msgUpdateRecipe.version.incrementRecipeVersion(); + msgUpdateRecipe.creator = creatorAddress.toString(); + + return _signAndBroadcast(msgUpdateRecipe); + } + @override Future createDynamicLinkForItemNftShare({ required String address, @@ -1312,9 +1487,13 @@ class RemoteDataStoreImp implements RemoteDataStore { required String cookbookId, }) async { final dynamicLinkParams = DynamicLinkParameters( - link: Uri.parse("$bigDipperBaseLink?item_id=$itemId&cookbook_id=$cookbookId&address=$address"), + link: Uri.parse( + "$bigDipperBaseLink?item_id=$itemId&cookbook_id=$cookbookId&address=$address"), uriPrefix: kDeepLink, - androidParameters: AndroidParameters(packageName: packageName, fallbackUrl: Uri.parse("$bigDipperBaseLink?item_id=$itemId&cookbook_id=$cookbookId&address=$address")), + androidParameters: AndroidParameters( + packageName: packageName, + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?item_id=$itemId&cookbook_id=$cookbookId&address=$address")), iosParameters: const IOSParameters(bundleId: bundleId), ); @@ -1323,11 +1502,15 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future createDynamicLinkForTradeNftShare({required String address, required String tradeId}) async { + Future createDynamicLinkForTradeNftShare( + {required String address, required String tradeId}) async { final dynamicLinkParams = DynamicLinkParameters( link: Uri.parse("$bigDipperBaseLink?trade_id=$tradeId&address=$address"), uriPrefix: kDeepLink, - androidParameters: AndroidParameters(packageName: packageName, fallbackUrl: Uri.parse("$bigDipperBaseLink?trade_id=$tradeId&address=$address")), + androidParameters: AndroidParameters( + packageName: packageName, + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?trade_id=$tradeId&address=$address")), iosParameters: const IOSParameters(bundleId: bundleId), ); @@ -1336,8 +1519,12 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future saveUserFeedback({required String walletAddress, required String subject, required String feedback}) { - return firebaseHelper.saveUserFeedback(walletAddress: walletAddress, subject: subject, feedback: feedback); + Future saveUserFeedback( + {required String walletAddress, + required String subject, + required String feedback}) { + return firebaseHelper.saveUserFeedback( + walletAddress: walletAddress, subject: subject, feedback: feedback); } @override @@ -1393,14 +1580,16 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future cancelTrade({required TradeId tradeId, required Address address}) async { + Future cancelTrade( + {required TradeId tradeId, required Address address}) async { final msgObj = MsgCancelTrade(creator: address.toString(), id: tradeId.id); final txHash = await _signAndBroadcast(msgObj); return TransactionResponse(hash: txHash); } @override - Future createTrade({required pylons.MsgCreateTrade msgCreateTrade}) async { + Future createTrade( + {required pylons.MsgCreateTrade msgCreateTrade}) async { final txHash = await _signAndBroadcast(msgCreateTrade); return TransactionResponse(hash: txHash); } @@ -1428,7 +1617,6 @@ class RemoteDataStoreImp implements RemoteDataStore { final http.Client httpClient; final StorePaymentService storePaymentService; - final FirebaseAppCheck firebaseAppCheck; final FirebaseDynamicLinks dynamicLinksGenerator; final FirestoreHelper firebaseHelper; final AnalyticsHelper analyticsHelper; @@ -1440,7 +1628,8 @@ class RemoteDataStoreImp implements RemoteDataStore { final uri = Uri.parse("$baseApiUrl/pylons/items/${address.id}"); - final pylonItemsResponse = await httpClient.get(uri).timeout(timeOutDuration); + final pylonItemsResponse = + await httpClient.get(uri).timeout(timeOutDuration); if (pylonItemsResponse.statusCode != API_SUCCESS_CODE) { throw HandlerFactory.ERR_SOMETHING_WENT_WRONG; @@ -1459,27 +1648,33 @@ class RemoteDataStoreImp implements RemoteDataStore { } @override - Future createDynamicLinkForRecipeEventShare({required String address, required Events events}) async { + Future createDynamicLinkForRecipeEventShare( + {required String address, required Events events}) async { final updateText = events.denom.getAbbrev() == constants.kPYLN_ABBREVATION ? "\$${events.denom.pylnToCredit(events.denom.getCoinWithProperDenomination(events.price))}" : "${events.denom.getCoinWithProperDenomination(events.price)} ${events.denom.getAbbrev()}"; final dynamicLinkParams = DynamicLinkParameters( - link: Uri.parse("$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), + link: Uri.parse( + "$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), uriPrefix: kDeepLink, androidParameters: AndroidParameters( packageName: packageName, - fallbackUrl: Uri.parse("$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), ), iosParameters: IOSParameters( bundleId: bundleId, - fallbackUrl: Uri.parse("$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), + fallbackUrl: Uri.parse( + "$bigDipperBaseLink?recipe_id=${events.recipeID}&cookbook_id=${events.cookbookID}&address=$address"), ), - navigationInfoParameters: const NavigationInfoParameters(forcedRedirectEnabled: true), + navigationInfoParameters: + const NavigationInfoParameters(forcedRedirectEnabled: true), socialMetaTagParameters: SocialMetaTagParameters( title: events.eventName, imageUrl: Uri.parse(events.thumbnail), - description: '${events.description} Price:${events.price == "0" ? LocaleKeys.free.tr() : updateText}', + description: + '${events.description} Price:${events.price == "0" ? LocaleKeys.free.tr() : updateText}', )); final link = await dynamicLinksGenerator.buildShortLink( @@ -1509,7 +1704,12 @@ class AppleInAppPurchaseModel { receiptData = json['receiptDataBase64'] as String, creator = json['creator'] as String; - Map toJson() => {"productId": productID, "purchaseId": purchaseID, "receiptDataBase64": receiptData, "creator": creator}; + Map toJson() => { + "productId": productID, + "purchaseId": purchaseID, + "receiptDataBase64": receiptData, + "creator": creator + }; } class GoogleInAppPurchaseModel { @@ -1534,9 +1734,21 @@ class GoogleInAppPurchaseModel { signature = json['signature'] as String, creator = json['creator'] as String; - Map toJson() => {"product_id": productID, "purchase_token": purchaseToken, "receipt_data_base64": getReceiptDataInBase64(), "signature": signature, "creator": creator}; - - Map toJsonLocalRetry() => {"product_id": productID, "purchase_token": purchaseToken, "receipt_data_base64": receiptData, "signature": signature, "creator": creator}; + Map toJson() => { + "product_id": productID, + "purchase_token": purchaseToken, + "receipt_data_base64": getReceiptDataInBase64(), + "signature": signature, + "creator": creator + }; + + Map toJsonLocalRetry() => { + "product_id": productID, + "purchase_token": purchaseToken, + "receipt_data_base64": receiptData, + "signature": signature, + "creator": creator + }; String getReceiptDataInBase64() { return base64Url.encode(utf8.encode(jsonEncode(receiptData))); diff --git a/wallet/lib/services/repository/repository.dart b/wallet/lib/services/repository/repository.dart index 3deae3e142..3bda2ea60b 100644 --- a/wallet/lib/services/repository/repository.dart +++ b/wallet/lib/services/repository/repository.dart @@ -577,6 +577,13 @@ abstract class Repository { required Address creatorAddress, }); + Future> stampTicket({ + required CookbookId cookBookId, + required RecipeId recipeId, + required Address creatorAddress, + required String challenge, + }); + Future> createTrade({required pylons.MsgCreateTrade msgCreateTrade}); Future> cancelTrade({required TradeId tradeId, required Address address}); @@ -2422,6 +2429,28 @@ class RepositoryImp implements Repository { } } + @override + Future> stampTicket({ + required CookbookId cookBookId, + required RecipeId recipeId, + required Address creatorAddress, + required String challenge, + }) async { + try { + await remoteDataStore.stampTicket( + cookBookId: cookBookId, + recipeId: recipeId, + creatorAddress: creatorAddress, + challenge: challenge, + ); + return const Right(null); + } on Exception catch (e) { + recordErrorInCrashlytics(e); + return Left(ServerFailure(e.toString())); + } + } + + @override Future> createTrade({required pylons.MsgCreateTrade msgCreateTrade}) async { try { diff --git a/wallet/lib/utils/dependency_injection/dependency_injection.dart b/wallet/lib/utils/dependency_injection/dependency_injection.dart index 90a8492861..22b30f8787 100644 --- a/wallet/lib/utils/dependency_injection/dependency_injection.dart +++ b/wallet/lib/utils/dependency_injection/dependency_injection.dart @@ -1,6 +1,5 @@ import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:firebase_analytics/firebase_analytics.dart'; -import 'package:firebase_app_check/firebase_app_check.dart'; import 'package:firebase_crashlytics/firebase_crashlytics.dart'; import 'package:firebase_dynamic_links/firebase_dynamic_links.dart'; import 'package:firebase_messaging/firebase_messaging.dart'; @@ -32,9 +31,9 @@ import 'package:pylons_wallet/services/data_stores/remote_data_store.dart'; import 'package:pylons_wallet/services/repository/repository.dart'; import 'package:pylons_wallet/services/third_party_services/analytics_helper.dart'; import 'package:pylons_wallet/services/third_party_services/audio_player_helper.dart'; +import 'package:pylons_wallet/services/third_party_services/connectivity_info.dart'; import 'package:pylons_wallet/services/third_party_services/crashlytics_helper.dart'; import 'package:pylons_wallet/services/third_party_services/firestore_helper.dart'; -import 'package:pylons_wallet/services/third_party_services/connectivity_info.dart'; import 'package:pylons_wallet/services/third_party_services/remote_config_service/remote_config_service.dart'; import 'package:pylons_wallet/services/third_party_services/remote_notifications_service.dart'; import 'package:pylons_wallet/services/third_party_services/share_helper.dart'; @@ -80,9 +79,11 @@ Future init({ }) async { /// Services sl.registerLazySingleton( - () => InternetConnectionChecker.createInstance(checkTimeout: const Duration(seconds: 20)), + () => InternetConnectionChecker.createInstance( + checkTimeout: const Duration(seconds: 20)), ); - sl.registerLazySingleton(() => ConnectivityInfoImpl(sl())); + sl.registerLazySingleton( + () => ConnectivityInfoImpl(sl())); sl.registerLazySingleton(() => IPCEngine( repository: sl(), walletsStore: sl(), @@ -90,51 +91,68 @@ Future init({ onLogEvent: onLogEvent, onLogError: onLogError, )); - sl.registerLazySingleton(() => LocalServer(sl())); + sl.registerLazySingleton( + () => LocalServer(sl())); sl.registerFactory(() => AudioPlayerHelperImpl(sl())); sl.registerFactory(() => VideoPlayerHelperImp(sl())); sl.registerFactory(() => ThumbnailHelperImp()); - sl.registerLazySingleton(() => CrashlyticsHelperImpl(sl())); + sl.registerLazySingleton( + () => CrashlyticsHelperImpl(sl())); sl.registerFactory(() => ShareHelperImpl()); - sl.registerLazySingleton(() => RemoteNotificationsProvider( - firebaseMessaging: sl(), - flutterLocalNotificationsPlugin: sl(), - repository: sl(), - )); - sl.registerLazySingleton(() => FirestoreHelperImp(mainFeedbacksCollection: sl())); - sl.registerLazySingleton(() => AnalyticsHelperImpl(firebaseAnalytics: sl())); + sl.registerLazySingleton( + () => RemoteNotificationsProvider( + firebaseMessaging: sl(), + flutterLocalNotificationsPlugin: sl(), + repository: sl(), + )); + sl.registerLazySingleton( + () => FirestoreHelperImp(mainFeedbacksCollection: sl())); + sl.registerLazySingleton( + () => AnalyticsHelperImpl(firebaseAnalytics: sl())); /// External Dependencies - sl.registerLazySingleton(() => FirebaseAppCheck.instance); - sl.registerLazySingleton(() => FlutterLocalNotificationsPlugin()); + sl.registerLazySingleton( + () => FlutterLocalNotificationsPlugin()); sl.registerLazySingleton(() => FirebaseMessaging.instance); sl.registerLazySingleton(() => ImagePicker()); sl.registerLazySingleton(() => LocalAuthentication()); - sl.registerSingletonAsync(() => SharedPreferences.getInstance()); + sl.registerSingletonAsync( + () => SharedPreferences.getInstance()); sl.registerLazySingleton(() => GoogleDriveApiImpl()); sl.registerLazySingleton(() => ICloudDriverApiImpl()); sl.registerLazySingleton(() => FirebaseRemoteConfig.instance); sl.registerLazySingleton(() => const FlutterSecureStorage()); sl.registerFactory(() => AudioPlayer()); - sl.registerFactory(() => VideoPlayerController.networkUrl(Uri(host: ""))); + sl.registerFactory( + () => VideoPlayerController.networkUrl(Uri(host: ""))); sl.registerLazySingleton(() => http.Client()); - sl.registerLazySingleton(() => AlanTransactionSigner(sl.get().networkInfo)); - sl.registerLazySingleton(() => AlanTransactionBroadcaster(sl.get().networkInfo)); - sl.registerLazySingleton(() => SharedPrefsPlainDataStore()); - sl.registerLazySingleton(() => NoOpTransactionSummaryUI()); - sl.registerLazySingleton(() => CustomTransactionSigner(sl.get().networkInfo)); + sl.registerLazySingleton( + () => AlanTransactionSigner(sl.get().networkInfo)); + sl.registerLazySingleton( + () => AlanTransactionBroadcaster(sl.get().networkInfo)); + sl.registerLazySingleton( + () => SharedPrefsPlainDataStore()); + sl.registerLazySingleton( + () => NoOpTransactionSummaryUI()); + sl.registerLazySingleton( + () => CustomTransactionSigner(sl.get().networkInfo)); sl.registerLazySingleton( () => CustomTransactionBroadcasterImp(sl.get().networkInfo)); sl.registerLazySingleton(() => FirebaseCrashlytics.instance); sl.registerLazySingleton(() => AlanAccountDerivator()); - sl.registerLazySingleton(() => FirebaseDynamicLinks.instance); - sl.registerLazySingleton(() => FirebaseFirestore.instance.collection(kFeedbacks)); + sl.registerLazySingleton( + () => FirebaseDynamicLinks.instance); + sl.registerLazySingleton( + () => FirebaseFirestore.instance.collection(kFeedbacks)); sl.registerLazySingleton(() => FirebaseAnalytics.instance); - final database = await $FloorAppDatabase.databaseBuilder('app_database.db').build(); + final database = + await $FloorAppDatabase.databaseBuilder('app_database.db').build(); sl.registerSingleton(database); - sl.registerLazySingleton(() => FlutterSecureStorageDataStore(storage: sl())); - sl.registerLazySingleton(() => AlanCredentialsSerializer()); + sl.registerLazySingleton( + () => FlutterSecureStorageDataStore(storage: sl())); + sl.registerLazySingleton( + () => AlanCredentialsSerializer()); sl.registerLazySingleton(() => CosmosKeyInfoStorage( serializers: [sl.get()], @@ -186,7 +204,8 @@ Future init({ await sl.isReady(); - final remoteConfigService = RemoteConfigServiceImpl(firebaseRemoteConfig: sl(), crashlyticsHelper: sl()); + final remoteConfigService = RemoteConfigServiceImpl( + firebaseRemoteConfig: sl(), crashlyticsHelper: sl()); await remoteConfigService.init(); sl.registerLazySingleton( @@ -200,14 +219,14 @@ Future init({ sl.registerLazySingleton(() => RemoteDataStoreImp( httpClient: sl(), storePaymentService: sl(), - firebaseAppCheck: sl(), dynamicLinksGenerator: sl(), firebaseHelper: sl(), analyticsHelper: sl(), onLogError: onLogError, )); - sl.registerLazySingleton(() => StorePaymentServiceImpl()); + sl.registerLazySingleton( + () => StorePaymentServiceImpl()); sl.registerLazySingleton(() => QueryHelper(httpClient: sl())); @@ -229,7 +248,10 @@ Future init({ /// ViewModels sl.registerLazySingleton( () => WalletsStoreImp( - repository: sl(), crashlyticsHelper: sl(), accountProvider: sl(), remoteNotificationProvider: sl()), + repository: sl(), + crashlyticsHelper: sl(), + accountProvider: sl(), + remoteNotificationProvider: sl()), ); sl.registerFactory( () => PurchaseItemViewModel(sl(), @@ -239,14 +261,19 @@ Future init({ shareHelper: sl(), accountPublicInfo: sl().accountPublicInfo), ); - sl.registerLazySingleton( - () => StripeHandler(walletsStore: sl(), localDataSource: sl(), repository: sl(), accountProvider: sl())); - sl.registerLazySingleton( - () => HomeProvider(repository: sl(), accountPublicInfo: sl().accountPublicInfo!)); + sl.registerLazySingleton(() => StripeHandler( + walletsStore: sl(), + localDataSource: sl(), + repository: sl(), + accountProvider: sl())); + sl.registerLazySingleton(() => HomeProvider( + repository: sl(), + accountPublicInfo: sl().accountPublicInfo!)); sl.registerLazySingleton(() => GeneralScreenViewModel()); - sl.registerLazySingleton(() => UserInfoProvider(sl(), sl())); sl.registerLazySingleton( - () => GeneralScreenLocalizationViewModel(shareHelper: sl(), repository: sl(), walletStore: sl())); + () => UserInfoProvider(sl(), sl())); + sl.registerLazySingleton(() => GeneralScreenLocalizationViewModel( + shareHelper: sl(), repository: sl(), walletStore: sl())); sl.registerLazySingleton(() => PracticeTestViewModel(sl())); sl.registerLazySingleton(() => FailureManagerViewModel(repository: sl())); sl.registerFactory(() => OwnerViewViewModel( @@ -259,10 +286,9 @@ Future init({ )); sl.registerLazySingleton(() => UserBannerViewModel()); sl.registerLazySingleton(() => CollectionsTabProvider()); - sl.registerLazySingleton(() => AcceptPolicyViewModel(repository: sl(), walletsStore: sl())); + sl.registerLazySingleton( + () => AcceptPolicyViewModel(repository: sl(), walletsStore: sl())); /// Configurations sl.registerLazySingleton(() => remoteConfigService.getBaseEnv()); - - } diff --git a/wallet/lib/utils/route_util.dart b/wallet/lib/utils/route_util.dart index 313e12ca85..2c6c039c88 100644 --- a/wallet/lib/utils/route_util.dart +++ b/wallet/lib/utils/route_util.dart @@ -3,9 +3,9 @@ import 'package:flutter/material.dart'; import 'package:pylons_wallet/pages/events/event_purchase_view.dart'; import 'package:pylons_wallet/pages/events/events_owner_view.dart'; +import 'package:pylons_wallet/pages/events/mobile_scanner.dart'; import 'package:pylons_wallet/pages/settings/screens/general_screen/general_screen.dart'; import 'package:pylons_wallet/pages/settings/screens/recovery_screen/recovery_screen.dart'; - import '../model/event.dart'; import '../model/nft.dart'; import '../model/transaction_failure_model.dart'; @@ -32,12 +32,16 @@ import '../pages/settings/settings_screen.dart'; import '../pages/transaction_failure_manager/local_transaction_detail_screen.dart'; import '../pages/transaction_failure_manager/local_transactions_screen.dart'; import 'dependency_injection/dependency_injection.dart'; +import 'package:mobile_scanner/mobile_scanner.dart'; class RouteUtil { RouteUtil(); static Route? onGenerateRoute(RouteSettings settings) { final route = Routes.getAppRouteFromString(settings.name ?? ""); + switch (route) { + case Routes.mobileQrScanner: + return createRoute(const MobileQrScanner()); case Routes.initial: return createRoute(const SplashScreen()); case Routes.home: @@ -170,10 +174,13 @@ enum Routes { acceptPolicy, fallback, eventOwnerView, - eventPurchaseView; + eventPurchaseView, + mobileQrScanner; static Routes getAppRouteFromString(String routeName) { switch (routeName) { + case'mobileQrScanner': + return mobileQrScanner; case '/': return initial; case 'home': diff --git a/wallet/lib/utils/string_utils.dart b/wallet/lib/utils/string_utils.dart new file mode 100644 index 0000000000..9dfef46148 --- /dev/null +++ b/wallet/lib/utils/string_utils.dart @@ -0,0 +1,15 @@ +import 'dart:math'; + +class StringUtils { + static String generateRandomString(int length) { + const String _chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; + Random _rnd = Random(); + + return String.fromCharCodes( + Iterable.generate( + length, + (_) => _chars.codeUnitAt(_rnd.nextInt(_chars.length)), + ), + ); + } +} diff --git a/wallet/macos/Flutter/GeneratedPluginRegistrant.swift b/wallet/macos/Flutter/GeneratedPluginRegistrant.swift index b2bb0501ea..c23bb4108c 100644 --- a/wallet/macos/Flutter/GeneratedPluginRegistrant.swift +++ b/wallet/macos/Flutter/GeneratedPluginRegistrant.swift @@ -15,6 +15,7 @@ import firebase_core import firebase_crashlytics import firebase_messaging import firebase_remote_config +import flutter_inappwebview_macos import flutter_local_notifications import flutter_secure_storage_macos import google_sign_in_ios @@ -41,6 +42,7 @@ func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { FLTFirebaseCrashlyticsPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseCrashlyticsPlugin")) FLTFirebaseMessagingPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseMessagingPlugin")) FLTFirebaseRemoteConfigPlugin.register(with: registry.registrar(forPlugin: "FLTFirebaseRemoteConfigPlugin")) + InAppWebViewFlutterPlugin.register(with: registry.registrar(forPlugin: "InAppWebViewFlutterPlugin")) FlutterLocalNotificationsPlugin.register(with: registry.registrar(forPlugin: "FlutterLocalNotificationsPlugin")) FlutterSecureStoragePlugin.register(with: registry.registrar(forPlugin: "FlutterSecureStoragePlugin")) FLTGoogleSignInPlugin.register(with: registry.registrar(forPlugin: "FLTGoogleSignInPlugin")) diff --git a/wallet/test/unit_testing/services/data_stores/remote_data_store_test.dart b/wallet/test/unit_testing/services/data_stores/remote_data_store_test.dart index afab1cceb2..e5ed800419 100644 --- a/wallet/test/unit_testing/services/data_stores/remote_data_store_test.dart +++ b/wallet/test/unit_testing/services/data_stores/remote_data_store_test.dart @@ -1,9 +1,10 @@ import 'dart:convert'; + import 'package:flutter_dotenv/flutter_dotenv.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:get_it/get_it.dart'; -import 'package:http/http.dart'; import 'package:http/http.dart' as http; +import 'package:http/http.dart'; import 'package:http/testing.dart'; import 'package:pylons_wallet/model/export.dart'; import 'package:pylons_wallet/services/data_stores/remote_data_store.dart'; @@ -11,19 +12,21 @@ import 'package:pylons_wallet/services/third_party_services/analytics_helper.dar import 'package:pylons_wallet/services/third_party_services/firestore_helper.dart'; import 'package:pylons_wallet/services/third_party_services/store_payment_service.dart'; +import '../../../mocks/mock_analytics_helper.dart'; import '../../../mocks/mock_constants.dart'; import '../../../mocks/mock_firebase_dynamic_link.dart'; -import '../../../mocks/mock_store_payment_service.dart'; import '../../../mocks/mock_firestore_helper.dart'; -import '../../../mocks/mock_analytics_helper.dart'; -import '../../../mocks/test_mocks.mocks.dart'; +import '../../../mocks/mock_store_payment_service.dart'; void main() { - test('should get account link and account id on getAccountLinkBasedOnUpdateToken', () async { + test( + 'should get account link and account id on getAccountLinkBasedOnUpdateToken', + () async { dotenv.testLoad(fileInput: '''ENV=true'''); final StorePaymentService storePaymentService = MockStripePaymentService(); - final MockFirebaseDynamicLinks mockFirebaseDynamicLinks = MockFirebaseDynamicLinks(); + final MockFirebaseDynamicLinks mockFirebaseDynamicLinks = + MockFirebaseDynamicLinks(); final AnalyticsHelper analyticsHelper = MockAnalyticsHelper(); final FirestoreHelper firestoreHelper = MockFirestoreHelper(); @@ -33,7 +36,6 @@ void main() { final remoteDataStore = RemoteDataStoreImp( httpClient: client, storePaymentService: storePaymentService, - firebaseAppCheck: MockFirebaseAppCheck(), dynamicLinksGenerator: mockFirebaseDynamicLinks, firebaseHelper: firestoreHelper, analyticsHelper: analyticsHelper, @@ -60,5 +62,7 @@ Future requestHandler(Request request) async { expect(mapBody["token"], MOCK_TOKEN); expect(mapBody["signature"], SIGNATURE); - return http.Response(jsonEncode({"accountlink": MOCK_ACCOUNT_LINK, "account": MOCK_ACCOUNT}), 200); + return http.Response( + jsonEncode({"accountlink": MOCK_ACCOUNT_LINK, "account": MOCK_ACCOUNT}), + 200); }