diff --git a/.DS_Store b/.DS_Store index 27bf70574c2297bd84bf4f461e9c378f8489fea5..be4a7c6ed08093a76031a3135bdf6f7e0dcd86ad 100644 Binary files a/.DS_Store and b/.DS_Store differ diff --git a/TraccarClient.xcodeproj/project.pbxproj b/TraccarClient.xcodeproj/project.pbxproj index 2f1574112a659ec574b01779120ed09d6b9a56e3..ea4f213af996a5fb4f3083317de4aa46ffe94fa0 100644 --- a/TraccarClient.xcodeproj/project.pbxproj +++ b/TraccarClient.xcodeproj/project.pbxproj @@ -7,6 +7,18 @@ objects = { /* Begin PBXBuildFile section */ + 53157EC32AEE3FF9003C9B6A /* API+Auth.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157EC22AEE3FF9003C9B6A /* API+Auth.swift */; }; + 53157EC62AEE4076003C9B6A /* UserModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157EC52AEE4076003C9B6A /* UserModel.swift */; }; + 53157EC82AEE4D6E003C9B6A /* SignInModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157EC72AEE4D6E003C9B6A /* SignInModel.swift */; }; + 53157ECA2AEE4DFE003C9B6A /* DeviceModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157EC92AEE4DFE003C9B6A /* DeviceModel.swift */; }; + 53157ECC2AEE55AB003C9B6A /* KNViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157ECB2AEE55AB003C9B6A /* KNViewController.swift */; }; + 53157ED02AEE563C003C9B6A /* KNSpinnerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157ECE2AEE563C003C9B6A /* KNSpinnerView.swift */; }; + 53157ED12AEE563C003C9B6A /* KNSpinnerView.xib in Resources */ = {isa = PBXBuildFile; fileRef = 53157ECF2AEE563C003C9B6A /* KNSpinnerView.xib */; }; + 53157ED32AEE56FE003C9B6A /* Globals.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157ED22AEE56FE003C9B6A /* Globals.swift */; }; + 53157ED72AEE5751003C9B6A /* KNAlertViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = 53157ED52AEE5751003C9B6A /* KNAlertViewController.xib */; }; + 53157ED82AEE5751003C9B6A /* KNAlertViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157ED62AEE5751003C9B6A /* KNAlertViewController.swift */; }; + 53157EDA2AEE57FA003C9B6A /* UITextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157ED92AEE57FA003C9B6A /* UITextView.swift */; }; + 53157EDC2AEE5841003C9B6A /* UIWindow.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53157EDB2AEE5841003C9B6A /* UIWindow.swift */; }; 5326833C2AE9175900A364C0 /* LoginViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5326833A2AE9175900A364C0 /* LoginViewController.swift */; }; 5326833D2AE9175900A364C0 /* LoginViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = 5326833B2AE9175900A364C0 /* LoginViewController.xib */; }; 532683512AE91A6F00A364C0 /* Montserrat-ExtraLightItalic.ttf in Resources */ = {isa = PBXBuildFile; fileRef = 5326833F2AE91A6E00A364C0 /* Montserrat-ExtraLightItalic.ttf */; }; @@ -70,6 +82,24 @@ 532683C12AEA50FA00A364C0 /* LoginViewControllerProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 532683C02AEA50FA00A364C0 /* LoginViewControllerProtocol.swift */; }; 532683C32AEA537F00A364C0 /* UITableView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 532683C22AEA537F00A364C0 /* UITableView.swift */; }; 532683C82AEA631800A364C0 /* IQKeyboardManagerSwift in Frameworks */ = {isa = PBXBuildFile; productRef = 532683C72AEA631800A364C0 /* IQKeyboardManagerSwift */; }; + 53554ADF2AED1B480018BAEE /* KNButtonView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53554AD92AED1B480018BAEE /* KNButtonView.swift */; }; + 53554AE02AED1B480018BAEE /* KNButtonViewHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53554ADA2AED1B480018BAEE /* KNButtonViewHelper.swift */; }; + 53554AE12AED1B480018BAEE /* ButtonTableViewCell.xib in Resources */ = {isa = PBXBuildFile; fileRef = 53554ADC2AED1B480018BAEE /* ButtonTableViewCell.xib */; }; + 53554AE22AED1B480018BAEE /* ButtonTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53554ADD2AED1B480018BAEE /* ButtonTableViewCell.swift */; }; + 53554AE32AED1B480018BAEE /* KNButtonView.xib in Resources */ = {isa = PBXBuildFile; fileRef = 53554ADE2AED1B480018BAEE /* KNButtonView.xib */; }; + 53554AE52AED1BF10018BAEE /* UIButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53554AE42AED1BF10018BAEE /* UIButton.swift */; }; + 5370B4702AEFB8A900AE08CC /* splash.json in Resources */ = {isa = PBXBuildFile; fileRef = 5370B46F2AEFB8A900AE08CC /* splash.json */; }; + 53D62E3F2AEFA31200C80BAC /* InitialViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53D62E3D2AEFA31200C80BAC /* InitialViewController.swift */; }; + 53D62E402AEFA31200C80BAC /* InitialViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = 53D62E3E2AEFA31200C80BAC /* InitialViewController.xib */; }; + 53D62E432AEFA4DC00C80BAC /* Lottie in Frameworks */ = {isa = PBXBuildFile; productRef = 53D62E422AEFA4DC00C80BAC /* Lottie */; }; + 53F10AE22AF05C57004D0529 /* KNNavigationController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AE12AF05C57004D0529 /* KNNavigationController.swift */; }; + 53F10AE42AF05E68004D0529 /* AuthenticationStep.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AE32AF05E68004D0529 /* AuthenticationStep.swift */; }; + 53F10AE62AF05E97004D0529 /* AuthenticationCoordinator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AE52AF05E97004D0529 /* AuthenticationCoordinator.swift */; }; + 53F10AE82AF05EB5004D0529 /* Coordinatable.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AE72AF05EB5004D0529 /* Coordinatable.swift */; }; + 53F10AEA2AF0602C004D0529 /* UINavigationController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AE92AF0602C004D0529 /* UINavigationController.swift */; }; + 53F10AEC2AF06556004D0529 /* UIImage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AEB2AF06556004D0529 /* UIImage.swift */; }; + 53F10AEE2AF06785004D0529 /* UITextField.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AED2AF06785004D0529 /* UITextField.swift */; }; + 53F10AF02AF06906004D0529 /* UIImageView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53F10AEF2AF06906004D0529 /* UIImageView.swift */; }; 5E394EBE28A9CC7600396F33 /* BatteryStatus.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5E394EBD28A9CC7600396F33 /* BatteryStatus.swift */; }; 5E716A271F63A0B100A2DBC3 /* DistanceCalculator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5E716A261F63A0B100A2DBC3 /* DistanceCalculator.swift */; }; 5E716A291F63A45A00A2DBC3 /* RequestManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5E716A281F63A45A00A2DBC3 /* RequestManager.swift */; }; @@ -128,6 +158,18 @@ /* End PBXCopyFilesBuildPhase section */ /* Begin PBXFileReference section */ + 53157EC22AEE3FF9003C9B6A /* API+Auth.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "API+Auth.swift"; sourceTree = "<group>"; }; + 53157EC52AEE4076003C9B6A /* UserModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserModel.swift; sourceTree = "<group>"; }; + 53157EC72AEE4D6E003C9B6A /* SignInModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SignInModel.swift; sourceTree = "<group>"; }; + 53157EC92AEE4DFE003C9B6A /* DeviceModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeviceModel.swift; sourceTree = "<group>"; }; + 53157ECB2AEE55AB003C9B6A /* KNViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = KNViewController.swift; sourceTree = "<group>"; }; + 53157ECE2AEE563C003C9B6A /* KNSpinnerView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = KNSpinnerView.swift; sourceTree = "<group>"; }; + 53157ECF2AEE563C003C9B6A /* KNSpinnerView.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = KNSpinnerView.xib; sourceTree = "<group>"; }; + 53157ED22AEE56FE003C9B6A /* Globals.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Globals.swift; sourceTree = "<group>"; }; + 53157ED52AEE5751003C9B6A /* KNAlertViewController.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = KNAlertViewController.xib; sourceTree = "<group>"; }; + 53157ED62AEE5751003C9B6A /* KNAlertViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = KNAlertViewController.swift; sourceTree = "<group>"; }; + 53157ED92AEE57FA003C9B6A /* UITextView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UITextView.swift; sourceTree = "<group>"; }; + 53157EDB2AEE5841003C9B6A /* UIWindow.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UIWindow.swift; sourceTree = "<group>"; }; 5326833A2AE9175900A364C0 /* LoginViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LoginViewController.swift; sourceTree = "<group>"; }; 5326833B2AE9175900A364C0 /* LoginViewController.xib */ = {isa = PBXFileReference; lastKnownFileType = file.xib; path = LoginViewController.xib; sourceTree = "<group>"; }; 5326833F2AE91A6E00A364C0 /* Montserrat-ExtraLightItalic.ttf */ = {isa = PBXFileReference; lastKnownFileType = file; path = "Montserrat-ExtraLightItalic.ttf"; sourceTree = "<group>"; }; @@ -188,6 +230,23 @@ 532683BE2AEA50EF00A364C0 /* LoginDataProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LoginDataProvider.swift; sourceTree = "<group>"; }; 532683C02AEA50FA00A364C0 /* LoginViewControllerProtocol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LoginViewControllerProtocol.swift; sourceTree = "<group>"; }; 532683C22AEA537F00A364C0 /* UITableView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UITableView.swift; sourceTree = "<group>"; }; + 53554AD92AED1B480018BAEE /* KNButtonView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = KNButtonView.swift; sourceTree = "<group>"; }; + 53554ADA2AED1B480018BAEE /* KNButtonViewHelper.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = KNButtonViewHelper.swift; sourceTree = "<group>"; }; + 53554ADC2AED1B480018BAEE /* ButtonTableViewCell.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = ButtonTableViewCell.xib; sourceTree = "<group>"; }; + 53554ADD2AED1B480018BAEE /* ButtonTableViewCell.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ButtonTableViewCell.swift; sourceTree = "<group>"; }; + 53554ADE2AED1B480018BAEE /* KNButtonView.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = KNButtonView.xib; sourceTree = "<group>"; }; + 53554AE42AED1BF10018BAEE /* UIButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UIButton.swift; sourceTree = "<group>"; }; + 5370B46F2AEFB8A900AE08CC /* splash.json */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.json; path = splash.json; sourceTree = "<group>"; }; + 53D62E3D2AEFA31200C80BAC /* InitialViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = InitialViewController.swift; sourceTree = "<group>"; }; + 53D62E3E2AEFA31200C80BAC /* InitialViewController.xib */ = {isa = PBXFileReference; lastKnownFileType = file.xib; path = InitialViewController.xib; sourceTree = "<group>"; }; + 53F10AE12AF05C57004D0529 /* KNNavigationController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = KNNavigationController.swift; sourceTree = "<group>"; }; + 53F10AE32AF05E68004D0529 /* AuthenticationStep.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AuthenticationStep.swift; sourceTree = "<group>"; }; + 53F10AE52AF05E97004D0529 /* AuthenticationCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AuthenticationCoordinator.swift; sourceTree = "<group>"; }; + 53F10AE72AF05EB5004D0529 /* Coordinatable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Coordinatable.swift; sourceTree = "<group>"; }; + 53F10AE92AF0602C004D0529 /* UINavigationController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UINavigationController.swift; sourceTree = "<group>"; }; + 53F10AEB2AF06556004D0529 /* UIImage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UIImage.swift; sourceTree = "<group>"; }; + 53F10AED2AF06785004D0529 /* UITextField.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UITextField.swift; sourceTree = "<group>"; }; + 53F10AEF2AF06906004D0529 /* UIImageView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UIImageView.swift; sourceTree = "<group>"; }; 5E394EBD28A9CC7600396F33 /* BatteryStatus.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BatteryStatus.swift; sourceTree = "<group>"; }; 5E716A261F63A0B100A2DBC3 /* DistanceCalculator.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DistanceCalculator.swift; sourceTree = "<group>"; }; 5E716A281F63A45A00A2DBC3 /* RequestManager.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RequestManager.swift; sourceTree = "<group>"; }; @@ -255,6 +314,7 @@ buildActionMask = 2147483647; files = ( CBDEEB8A1B8EB6A4006BC126 /* SystemConfiguration.framework in Frameworks */, + 53D62E432AEFA4DC00C80BAC /* Lottie in Frameworks */, 5326839F2AE9489C00A364C0 /* Reachability in Frameworks */, CBCE82EF1B8D253600A7318B /* CoreData.framework in Frameworks */, 5EE3CA342998948F002C86E4 /* FirebaseAnalytics in Frameworks */, @@ -280,9 +340,38 @@ /* End PBXFrameworksBuildPhase section */ /* Begin PBXGroup section */ + 53157EC42AEE4059003C9B6A /* Models */ = { + isa = PBXGroup; + children = ( + 53157EC52AEE4076003C9B6A /* UserModel.swift */, + 53157EC72AEE4D6E003C9B6A /* SignInModel.swift */, + 53157EC92AEE4DFE003C9B6A /* DeviceModel.swift */, + ); + name = Models; + sourceTree = "<group>"; + }; + 53157ECD2AEE563C003C9B6A /* KNSpinner */ = { + isa = PBXGroup; + children = ( + 53157ECE2AEE563C003C9B6A /* KNSpinnerView.swift */, + 53157ECF2AEE563C003C9B6A /* KNSpinnerView.xib */, + ); + path = KNSpinner; + sourceTree = "<group>"; + }; + 53157ED42AEE5751003C9B6A /* KNAlert */ = { + isa = PBXGroup; + children = ( + 53157ED52AEE5751003C9B6A /* KNAlertViewController.xib */, + 53157ED62AEE5751003C9B6A /* KNAlertViewController.swift */, + ); + path = KNAlert; + sourceTree = "<group>"; + }; 532683392AE9174700A364C0 /* Views */ = { isa = PBXGroup; children = ( + 53157ED42AEE5751003C9B6A /* KNAlert */, 532683C52AEA5DDF00A364C0 /* Custom */, CED4873317DB1BF6007FCF57 /* MainStoryboard.storyboard */, CEF643241B919FFA00195CEA /* LaunchScreen.xib */, @@ -318,7 +407,6 @@ 532683632AE91CAC00A364C0 /* helpers */ = { isa = PBXGroup; children = ( - 5326838E2AE947D900A364C0 /* Network */, 532683642AE91D0800A364C0 /* UIFont.swift */, 5326838C2AE92D9900A364C0 /* CGFLoat.swift */, 532683862AE9295500A364C0 /* Nameable.swift */, @@ -333,6 +421,15 @@ 532683B02AE94C9400A364C0 /* Dictionary.swift */, 532683B22AE98D0000A364C0 /* UILabel.swift */, 532683C22AEA537F00A364C0 /* UITableView.swift */, + 53554AE42AED1BF10018BAEE /* UIButton.swift */, + 53157ED22AEE56FE003C9B6A /* Globals.swift */, + 53157ED92AEE57FA003C9B6A /* UITextView.swift */, + 53157EDB2AEE5841003C9B6A /* UIWindow.swift */, + 53F10AE72AF05EB5004D0529 /* Coordinatable.swift */, + 53F10AE92AF0602C004D0529 /* UINavigationController.swift */, + 53F10AEB2AF06556004D0529 /* UIImage.swift */, + 53F10AED2AF06785004D0529 /* UITextField.swift */, + 53F10AEF2AF06906004D0529 /* UIImageView.swift */, ); name = helpers; sourceTree = "<group>"; @@ -383,6 +480,7 @@ 532683942AE9482300A364C0 /* NetworkRequest.swift */, 5326838A2AE92CA500A364C0 /* JSONField.swift */, 532683B42AE98FB600A364C0 /* UIString.swift */, + 53157EC22AEE3FF9003C9B6A /* API+Auth.swift */, ); name = Network; sourceTree = "<group>"; @@ -404,6 +502,8 @@ 532683BC2AEA50C600A364C0 /* LoginModelController.swift */, 532683BE2AEA50EF00A364C0 /* LoginDataProvider.swift */, 532683C02AEA50FA00A364C0 /* LoginViewControllerProtocol.swift */, + 53F10AE32AF05E68004D0529 /* AuthenticationStep.swift */, + 53F10AE52AF05E97004D0529 /* AuthenticationCoordinator.swift */, ); path = Model; sourceTree = "<group>"; @@ -427,7 +527,10 @@ 532683C42AEA5DB000A364C0 /* Controllers */ = { isa = PBXGroup; children = ( + 53D62E3C2AEFA2F800C80BAC /* InitialViewController */, 532683B62AEA505C00A364C0 /* LoginViewController */, + 53157ECB2AEE55AB003C9B6A /* KNViewController.swift */, + 53F10AE12AF05C57004D0529 /* KNNavigationController.swift */, ); name = Controllers; sourceTree = "<group>"; @@ -435,13 +538,44 @@ 532683C52AEA5DDF00A364C0 /* Custom */ = { isa = PBXGroup; children = ( + 53157ECD2AEE563C003C9B6A /* KNSpinner */, 532683842AE9293200A364C0 /* KNComponentView.swift */, 532683822AE928F900A364C0 /* KNTableViewCell.swift */, + 53554AD82AED1B480018BAEE /* KNButtonView */, 532683662AE923A500A364C0 /* BMTextFieldView */, ); path = Custom; sourceTree = "<group>"; }; + 53554AD82AED1B480018BAEE /* KNButtonView */ = { + isa = PBXGroup; + children = ( + 53554AD92AED1B480018BAEE /* KNButtonView.swift */, + 53554ADA2AED1B480018BAEE /* KNButtonViewHelper.swift */, + 53554ADB2AED1B480018BAEE /* Cell */, + 53554ADE2AED1B480018BAEE /* KNButtonView.xib */, + ); + path = KNButtonView; + sourceTree = "<group>"; + }; + 53554ADB2AED1B480018BAEE /* Cell */ = { + isa = PBXGroup; + children = ( + 53554ADC2AED1B480018BAEE /* ButtonTableViewCell.xib */, + 53554ADD2AED1B480018BAEE /* ButtonTableViewCell.swift */, + ); + path = Cell; + sourceTree = "<group>"; + }; + 53D62E3C2AEFA2F800C80BAC /* InitialViewController */ = { + isa = PBXGroup; + children = ( + 53D62E3D2AEFA31200C80BAC /* InitialViewController.swift */, + 53D62E3E2AEFA31200C80BAC /* InitialViewController.xib */, + ); + path = InitialViewController; + sourceTree = "<group>"; + }; CED4870F17DB1BF6007FCF57 = { isa = PBXGroup; children = ( @@ -485,6 +619,8 @@ CEF643371B91A94600195CEA /* Localizable.strings */, 532683392AE9174700A364C0 /* Views */, CED4872217DB1BF6007FCF57 /* Supporting Files */, + 53157EC42AEE4059003C9B6A /* Models */, + 5326838E2AE947D900A364C0 /* Network */, CB4197921F674A3E008F301C /* DatabaseHelper.swift */, 5E716A261F63A0B100A2DBC3 /* DistanceCalculator.swift */, CBAA0F7E1F68E807008BBBBE /* MainViewController.swift */, @@ -505,6 +641,7 @@ children = ( 5326833E2AE91A6200A364C0 /* Fonts */, CED4872317DB1BF6007FCF57 /* TraccarClient-Info.plist */, + 5370B46F2AEFB8A900AE08CC /* splash.json */, CED4872917DB1BF6007FCF57 /* TraccarClient-Prefix.pch */, CBCE82F01B8D265800A7318B /* TraccarClient.xcdatamodeld */, CED4879A17DB1E61007FCF57 /* InAppSettings.bundle */, @@ -557,6 +694,7 @@ 532683802AE923DE00A364C0 /* Alamofire */, 5326839E2AE9489C00A364C0 /* Reachability */, 532683C72AEA631800A364C0 /* IQKeyboardManagerSwift */, + 53D62E422AEFA4DC00C80BAC /* Lottie */, ); productName = TraccarClient; productReference = CED4871817DB1BF6007FCF57 /* TraccarClient.app */; @@ -626,6 +764,7 @@ 5326837F2AE923DE00A364C0 /* XCRemoteSwiftPackageReference "Alamofire" */, 5326839D2AE9489C00A364C0 /* XCRemoteSwiftPackageReference "Reachability" */, 532683C62AEA631800A364C0 /* XCRemoteSwiftPackageReference "IQKeyboardManager" */, + 53D62E412AEFA4DC00C80BAC /* XCRemoteSwiftPackageReference "lottie-ios" */, ); productRefGroup = CED4871917DB1BF6007FCF57 /* Products */; projectDirPath = ""; @@ -646,8 +785,10 @@ 532683572AE91A6F00A364C0 /* Montserrat-BoldItalic.ttf in Resources */, CEF6433C1B91AA9400195CEA /* MainStoryboard.storyboard in Resources */, 5326835B2AE91A6F00A364C0 /* Montserrat-BlackItalic.ttf in Resources */, + 53554AE12AED1B480018BAEE /* ButtonTableViewCell.xib in Resources */, CEF6433D1B91AA9400195CEA /* Localizable.strings in Resources */, 532683552AE91A6F00A364C0 /* Montserrat-Italic.ttf in Resources */, + 53D62E402AEFA31200C80BAC /* InitialViewController.xib in Resources */, 5326833D2AE9175900A364C0 /* LoginViewController.xib in Resources */, CE5899C81B115C9100ED70D2 /* Images.xcassets in Resources */, 5326835F2AE91A6F00A364C0 /* Montserrat-Black.ttf in Resources */, @@ -662,11 +803,15 @@ 532683612AE91A6F00A364C0 /* Montserrat-ExtraLight.ttf in Resources */, 532683782AE923A500A364C0 /* FieldTableViewCell.xib in Resources */, 5326835D2AE91A6F00A364C0 /* Montserrat-Bold.ttf in Resources */, + 53554AE32AED1B480018BAEE /* KNButtonView.xib in Resources */, 532683542AE91A6F00A364C0 /* Montserrat-LightItalic.ttf in Resources */, 5326835C2AE91A6F00A364C0 /* Montserrat-ThinItalic.ttf in Resources */, + 53157ED12AEE563C003C9B6A /* KNSpinnerView.xib in Resources */, 532683562AE91A6F00A364C0 /* Montserrat-MediumItalic.ttf in Resources */, 5326835A2AE91A6F00A364C0 /* Montserrat-Thin.ttf in Resources */, 532683602AE91A6F00A364C0 /* Montserrat-Medium.ttf in Resources */, + 53157ED72AEE5751003C9B6A /* KNAlertViewController.xib in Resources */, + 5370B4702AEFB8A900AE08CC /* splash.json in Resources */, 532683512AE91A6F00A364C0 /* Montserrat-ExtraLightItalic.ttf in Resources */, ); runOnlyForDeploymentPostprocessing = 0; @@ -703,42 +848,62 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + 53157EC32AEE3FF9003C9B6A /* API+Auth.swift in Sources */, 5326833C2AE9175900A364C0 /* LoginViewController.swift in Sources */, 532683962AE9482300A364C0 /* Network+Endpoint.swift in Sources */, CB7ED0801F6602CD00A33FCF /* Position.swift in Sources */, 532683972AE9482300A364C0 /* Network+Result.swift in Sources */, 532683BB2AEA50B800A364C0 /* LoginDataSource.swift in Sources */, + 53157ED82AEE5751003C9B6A /* KNAlertViewController.swift in Sources */, 5326837A2AE923A500A364C0 /* KNFieldDropDownAddressObjectProtocol.swift in Sources */, CBAA0F7F1F68E807008BBBBE /* MainViewController.swift in Sources */, 532683B52AE98FB600A364C0 /* UIString.swift in Sources */, 5326839C2AE9482300A364C0 /* Network+Error.swift in Sources */, 532683AF2AE94C2F00A364C0 /* NSLayoutConstraints.swift in Sources */, CB7ED0821F661B4F00A33FCF /* PositionProvider.swift in Sources */, + 53554AE52AED1BF10018BAEE /* UIButton.swift in Sources */, + 53554AE02AED1B480018BAEE /* KNButtonViewHelper.swift in Sources */, + 53157EDC2AEE5841003C9B6A /* UIWindow.swift in Sources */, + 53554AE22AED1B480018BAEE /* ButtonTableViewCell.swift in Sources */, 532683832AE928F900A364C0 /* KNTableViewCell.swift in Sources */, 5326837C2AE923A500A364C0 /* KNField.swift in Sources */, + 53157ECA2AEE4DFE003C9B6A /* DeviceModel.swift in Sources */, 532683AB2AE94BA500A364C0 /* UIView.swift in Sources */, 5326839A2AE9482300A364C0 /* Network+Types.swift in Sources */, CB4197991F67724F008F301C /* NetworkManager.swift in Sources */, 532683B32AE98D0000A364C0 /* UILabel.swift in Sources */, 5326838D2AE92D9900A364C0 /* CGFLoat.swift in Sources */, CB7ED0841F662BAF00A33FCF /* AppDelegate.swift in Sources */, + 53F10AE82AF05EB5004D0529 /* Coordinatable.swift in Sources */, + 53D62E3F2AEFA31200C80BAC /* InitialViewController.swift in Sources */, 532683AD2AE94BFC00A364C0 /* Logs.swift in Sources */, 5E394EBE28A9CC7600396F33 /* BatteryStatus.swift in Sources */, + 53157ED32AEE56FE003C9B6A /* Globals.swift in Sources */, + 53F10AF02AF06906004D0529 /* UIImageView.swift in Sources */, 532683872AE9295500A364C0 /* Nameable.swift in Sources */, 5326837E2AE923A500A364C0 /* KNFieldDropdownObjectProtocol.swift in Sources */, + 53F10AEA2AF0602C004D0529 /* UINavigationController.swift in Sources */, 532683752AE923A500A364C0 /* KNTextFieldView+Protocols.swift in Sources */, 532683A52AE94A3500A364C0 /* KNTextField.swift in Sources */, + 53157EDA2AEE57FA003C9B6A /* UITextView.swift in Sources */, 5E716A2B1F63A60800A2DBC3 /* ProtocolFormatter.swift in Sources */, + 53F10AE62AF05E97004D0529 /* AuthenticationCoordinator.swift in Sources */, 532683772AE923A500A364C0 /* KNTextFieldView.swift in Sources */, + 53157ECC2AEE55AB003C9B6A /* KNViewController.swift in Sources */, + 53554ADF2AED1B480018BAEE /* KNButtonView.swift in Sources */, + 53157EC62AEE4076003C9B6A /* UserModel.swift in Sources */, 532683652AE91D0800A364C0 /* UIFont.swift in Sources */, CBCE82F21B8D265800A7318B /* TraccarClient.xcdatamodeld in Sources */, 532683762AE923A500A364C0 /* KNTextFieldViewHelper.swift in Sources */, CBAA0F7D1F68E14C008BBBBE /* StatusViewController.swift in Sources */, + 53F10AEC2AF06556004D0529 /* UIImage.swift in Sources */, 532683A72AE94ADC00A364C0 /* KNObject.swift in Sources */, 532683A12AE948FF00A364C0 /* AppManager.swift in Sources */, + 53157EC82AEE4D6E003C9B6A /* SignInModel.swift in Sources */, 532683C12AEA50FA00A364C0 /* LoginViewControllerProtocol.swift in Sources */, 532683A32AE9497100A364C0 /* Keys.swift in Sources */, 532683A92AE94B1300A364C0 /* UIInt.swift in Sources */, + 53157ED02AEE563C003C9B6A /* KNSpinnerView.swift in Sources */, 5326837D2AE923A500A364C0 /* KNFieldBuilder.swift in Sources */, 532683852AE9293200A364C0 /* KNComponentView.swift in Sources */, CBAA0F811F68EB62008BBBBE /* TrackingController.swift in Sources */, @@ -747,7 +912,10 @@ 532683792AE923A500A364C0 /* FieldTableViewCell.swift in Sources */, 5326838B2AE92CA500A364C0 /* JSONField.swift in Sources */, 532683BF2AEA50EF00A364C0 /* LoginDataProvider.swift in Sources */, + 53F10AEE2AF06785004D0529 /* UITextField.swift in Sources */, 532683892AE92B6800A364C0 /* Observable.swift in Sources */, + 53F10AE42AF05E68004D0529 /* AuthenticationStep.swift in Sources */, + 53F10AE22AF05C57004D0529 /* KNNavigationController.swift in Sources */, 532683BD2AEA50C600A364C0 /* LoginModelController.swift in Sources */, 5326837B2AE923A500A364C0 /* KNFieldType.swift in Sources */, 532683B12AE94C9400A364C0 /* Dictionary.swift in Sources */, @@ -1247,6 +1415,14 @@ minimumVersion = 6.5.16; }; }; + 53D62E412AEFA4DC00C80BAC /* XCRemoteSwiftPackageReference "lottie-ios" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/airbnb/lottie-ios.git"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 4.3.3; + }; + }; 5EE3CA322998948F002C86E4 /* XCRemoteSwiftPackageReference "firebase-ios-sdk" */ = { isa = XCRemoteSwiftPackageReference; repositoryURL = "https://github.com/firebase/firebase-ios-sdk"; @@ -1281,6 +1457,11 @@ package = 532683C62AEA631800A364C0 /* XCRemoteSwiftPackageReference "IQKeyboardManager" */; productName = IQKeyboardManagerSwift; }; + 53D62E422AEFA4DC00C80BAC /* Lottie */ = { + isa = XCSwiftPackageProductDependency; + package = 53D62E412AEFA4DC00C80BAC /* XCRemoteSwiftPackageReference "lottie-ios" */; + productName = Lottie; + }; 5EE3CA332998948F002C86E4 /* FirebaseAnalytics */ = { isa = XCSwiftPackageProductDependency; package = 5EE3CA322998948F002C86E4 /* XCRemoteSwiftPackageReference "firebase-ios-sdk" */; diff --git a/TraccarClient.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/TraccarClient.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index c2746fdab1021ccf7053be9a7e6e8c138e119a8c..de2080aad6f15279f3c40bb73218914b4af8ec6a 100644 --- a/TraccarClient.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/TraccarClient.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -108,6 +108,15 @@ "version" : "1.22.2" } }, + { + "identity" : "lottie-ios", + "kind" : "remoteSourceControl", + "location" : "https://github.com/airbnb/lottie-ios.git", + "state" : { + "revision" : "45517c3cfec9469bbdd4f86e32393c28ae9df0bc", + "version" : "4.3.3" + } + }, { "identity" : "nanopb", "kind" : "remoteSourceControl", diff --git a/TraccarClient.xcodeproj/project.xcworkspace/xcuserdata/g.makhoul.xcuserdatad/UserInterfaceState.xcuserstate b/TraccarClient.xcodeproj/project.xcworkspace/xcuserdata/g.makhoul.xcuserdatad/UserInterfaceState.xcuserstate index c7d0ddfdb329b1dcd14adf5d29c7f4fc41fe0645..ed8e56072da3416147a9f55a7d4db96802839e7f 100644 Binary files a/TraccarClient.xcodeproj/project.xcworkspace/xcuserdata/g.makhoul.xcuserdatad/UserInterfaceState.xcuserstate and b/TraccarClient.xcodeproj/project.xcworkspace/xcuserdata/g.makhoul.xcuserdatad/UserInterfaceState.xcuserstate differ diff --git a/TraccarClient/.DS_Store b/TraccarClient/.DS_Store new file mode 100644 index 0000000000000000000000000000000000000000..686a648a6a50c5bbc9ee4436a0b2d1044d78c75b Binary files /dev/null and b/TraccarClient/.DS_Store differ diff --git a/TraccarClient/API+Auth.swift b/TraccarClient/API+Auth.swift new file mode 100644 index 0000000000000000000000000000000000000000..8e4c37669b071003684b36e1dfe9516e961c87a4 --- /dev/null +++ b/TraccarClient/API+Auth.swift @@ -0,0 +1,23 @@ +// +// API+Auth.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// +import Alamofire +import Foundation + +extension NetworkRequest { + // MARK: - Signin + func signin(params: Parameters, _ completion: @escaping ResultClosure) { + request(endpoint: endpoint(.signin), method: .post, params: params) { result in + let saveUser: Bool = result.success + let user = SignInModel(result.object, saveUser: saveUser) + result.data = user + print(result) + completion(result) + } + } +} + diff --git a/TraccarClient/AppDelegate.swift b/TraccarClient/AppDelegate.swift index e5271e65f1cb45778c71a929097ecc1a72f41bef..2e6c6848aa29ab1cff2a37b3745bb8e3061d9c43 100644 --- a/TraccarClient/AppDelegate.swift +++ b/TraccarClient/AppDelegate.swift @@ -36,18 +36,22 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate } func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey : Any]?) -> Bool { - #if FIREBASE +#if FIREBASE FirebaseApp.configure() - #endif - +#endif + IQKeyboardManager.shared.enable = true + IQKeyboardManager.shared.shouldResignOnTouchOutside = true + IQKeyboardManager.shared.previousNextDisplayMode = .alwaysShow + IQKeyboardManager.shared.toolbarTintColor = .purple + UIDevice.current.isBatteryMonitoringEnabled = true - + let userDefaults = UserDefaults.standard if userDefaults.string(forKey: "device_id_preference") == nil { let identifier = "\(Int.random(in: 100000..<1000000))" userDefaults.setValue(identifier, forKey: "device_id_preference") } - + registerDefaultsFromSettingsBundle() migrateLegacyDefaults() @@ -62,24 +66,22 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate managedObjectContext = NSManagedObjectContext.init(concurrencyType: .mainQueueConcurrencyType) managedObjectContext?.persistentStoreCoordinator = persistentStoreCoordinator - - if userDefaults.bool(forKey: "service_status_preference") { - StatusViewController.addMessage(NSLocalizedString("Service created", comment: "")) - trackingController = TrackingController() - trackingController?.start() - } + // + // if userDefaults.bool(forKey: "service_status_preference") { + // StatusViewController.addMessage(NSLocalizedString("Service created", comment: "")) + // trackingController = TrackingController() + // trackingController?.start() + // } + + // } + + initialViewController() - IQKeyboardManager.shared.enable = true - IQKeyboardManager.shared.shouldResignOnTouchOutside = true - IQKeyboardManager.shared.previousNextDisplayMode = .alwaysShow - IQKeyboardManager.shared.toolbarTintColor = .purple -// setRootNavigation() - return true } - func setRootNavigation() { + func login() { window = UIWindow(frame: UIScreen.main.bounds) let vc = LoginViewController() @@ -89,6 +91,16 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate window?.makeKeyAndVisible() } + func initialViewController() { + window = UIWindow(frame: UIScreen.main.bounds) + + let vc = InitialViewController() + vc.window = window + + window?.rootViewController = vc + window?.makeKeyAndVisible() + } + func application(_ application: UIApplication, performActionFor shortcutItem: UIApplicationShortcutItem, completionHandler: @escaping (Bool) -> Void) { let userDefaults = UserDefaults.standard @@ -122,10 +134,10 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate } func didUpdate(position: Position) { - + positionProvider?.stopUpdates() positionProvider = nil - + let userDefaults = UserDefaults.standard if let request = ProtocolFormatter.formatPostion(position, url: userDefaults.string(forKey: "server_url_preference")!, alarm: "sos") { @@ -148,6 +160,9 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate } func applicationWillTerminate(_ application: UIApplication) { + let userDefaults = UserDefaults.standard + userDefaults.setValue(false, forKey: "service_status_preference") + if let context = managedObjectContext { if context.hasChanges { try! context.save() @@ -190,5 +205,5 @@ class AppDelegate: UIResponder, UIApplicationDelegate, PositionProviderDelegate userDefaults.removeObject(forKey: "secure_preference") } } - + } diff --git a/TraccarClient/AppManager.swift b/TraccarClient/AppManager.swift index 182c0601b3b98fa604ca7ca3835ced6d3de111a3..c6b751cecf53048e4d88654cd0266c84a0f299e3 100644 --- a/TraccarClient/AppManager.swift +++ b/TraccarClient/AppManager.swift @@ -35,28 +35,7 @@ final class AppManager { var useShortLogsForRequests = false //true var logRequest: [Endpoint] = [ .all, - .signin, - .signup, - .companySignup, - .signupPassword, - .companySignupPassword, - .signupVerifyCode, - .addresses, - .bankAccount, - .contactus, - .upload_Media, - .product_builder_step_information_media, - .documents, - .favorite, - .product_builder_step_shipping, - .variant, - .salesOrders, - .salesOrderDetails, - .checkout_shipping_methods, - .sales_summary, - .checkout_payment_methods, - .checkout_add_coupon, - .checkout_remove_coupon + .signin ] // MARK: - Data @@ -66,9 +45,6 @@ final class AppManager { var isLoggedIn: Bool { if let _ = UserDefaults.standard.string(forKey: Keys.User.access_token) { return true - - } else if let _ = UserDefaults.standard.string(forKey: Keys.guestToken) { - return true } return false } diff --git a/TraccarClient/Base.lproj/MainStoryboard.storyboard b/TraccarClient/Base.lproj/MainStoryboard.storyboard index f5ea5bcf7da3a57c47628446d9b50eeb04e6a12e..c0daa302d746771589d9d60df0011bc2df702feb 100644 --- a/TraccarClient/Base.lproj/MainStoryboard.storyboard +++ b/TraccarClient/Base.lproj/MainStoryboard.storyboard @@ -1,9 +1,9 @@ <?xml version="1.0" encoding="UTF-8"?> -<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="21507" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" colorMatched="YES" initialViewController="4Uy-y2-7bi"> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="22155" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" colorMatched="YES" initialViewController="4Uy-y2-7bi"> <device id="retina4_7" orientation="portrait" appearance="light"/> <dependencies> <deployment identifier="iOS"/> - <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="21505"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="22131"/> <capability name="Named colors" minToolsVersion="9.0"/> <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> </dependencies> @@ -11,7 +11,7 @@ <!--Root View Controller--> <scene sceneID="Ejt-ia-q1N"> <objects> - <tableViewController id="ZiV-Qa-hnE" customClass="MainViewController" customModule="TraccarClient" customModuleProvider="target" sceneMemberID="viewController"> + <tableViewController storyboardIdentifier="MainViewController" id="ZiV-Qa-hnE" customClass="MainViewController" customModule="TraccarClient" customModuleProvider="target" sceneMemberID="viewController"> <tableView key="view" opaque="NO" clipsSubviews="YES" clearsContextBeforeDrawing="NO" contentMode="scaleToFill" alwaysBounceVertical="YES" dataMode="prototypes" style="grouped" separatorStyle="default" rowHeight="44" sectionHeaderHeight="10" sectionFooterHeight="10" id="KcQ-JP-xCp"> <rect key="frame" x="0.0" y="0.0" width="375" height="667"/> <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> @@ -24,6 +24,7 @@ </connections> </barButtonItem> </navigationItem> + <nil key="simulatedTopBarMetrics"/> </tableViewController> <placeholder placeholderIdentifier="IBFirstResponder" id="68Y-02-qka" userLabel="First Responder" sceneMemberID="firstResponder"/> </objects> @@ -69,7 +70,7 @@ <objects> <navigationController definesPresentationContext="YES" id="4Uy-y2-7bi" sceneMemberID="viewController"> <navigationBar key="navigationBar" contentMode="scaleToFill" id="aeM-3M-Z4b"> - <rect key="frame" x="0.0" y="0.0" width="375" height="44"/> + <rect key="frame" x="0.0" y="20" width="375" height="44"/> <autoresizingMask key="autoresizingMask"/> <color key="tintColor" white="1" alpha="1" colorSpace="calibratedWhite"/> <color key="barTintColor" name="Brand"/> diff --git a/TraccarClient/Coordinatable.swift b/TraccarClient/Coordinatable.swift new file mode 100644 index 0000000000000000000000000000000000000000..0fb0f16dc1e03616e14658f2364091958d0c7306 --- /dev/null +++ b/TraccarClient/Coordinatable.swift @@ -0,0 +1,15 @@ +// +// Coordinatable.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +protocol Coordinatable { + var navigationController: UINavigationController { get set } + + func start() +} diff --git a/TraccarClient/Custom/BMTextFieldView/KNTextFieldView.swift b/TraccarClient/Custom/BMTextFieldView/KNTextFieldView.swift index 3002cef10bc85e5b1fa207a303ec23bb059062b1..b721959add6d70ed192eb124b640c6b4b8f3d6a8 100644 --- a/TraccarClient/Custom/BMTextFieldView/KNTextFieldView.swift +++ b/TraccarClient/Custom/BMTextFieldView/KNTextFieldView.swift @@ -171,17 +171,17 @@ final class KNTextFieldView: KNComponentView { private func updateUI() { guard let field = field else { return } // helper - + helper = KNTextFieldViewHelper(field: field) guard let helper = helper else { return } // borderView.addBorder(radius: 13, width: 1, color: .purple) - textField.textAlignment = .left - textField.semanticContentAttribute = .forceLeftToRight + textField.textAlignment = .left + textField.semanticContentAttribute = .forceLeftToRight - titleLabel.text = field.title == "" ? helper.title : field.title + titleLabel.text = field.title == "" ? helper.title : field.title titleLabel_bottom.set(8) @@ -196,61 +196,16 @@ final class KNTextFieldView: KNComponentView { if field.field == .iban { textField.autocapitalizationType = .allCharacters } - - if field.field == .price || field.field == .sale_price { - textField.keyboardType = .decimalPad - } -// -// if field.hasPlaceholder { -// if isUnitMeasurement.dropdown { -// //textField.placeholder = field.unitMeasurement?.display ?? "" -// } else { -// if field.isArabic { -// textField.setForArabic(localizedPlaceholder: field.placeholder != "" ? field.placeholder.localized : helper.placeholder, color: .placeholder, font: .regular(16)) -// } else if field.isEnglish { -// textField.setForEnglighs(localizedPlaceholder: field.placeholder != "" ? field.placeholder.localized : helper.placeholder, color: .placeholder, font: .regular(16)) -// } else { -// if field.placeholder == "" { -// textField.set(localizedPlaceholder: field.placeholder != "" ? field.placeholder.localized : helper.placeholder, color: .placeholder, font: .regular(16)) -// } else { -// textField.set(localizedPlaceholder: field.placeholder != "" ? field.placeholder.localized : helper.placeholder, color: .placeholder, font: .regular(16)) -// } -// } -// } -// } else { -// textField.placeholder = "" -// } - - textField.show() - unitValueLabel.hide() - textField.text = field.value.value - // if field.isArabic { - // textField.textAlignment = lang == .en ? .right : .left - // textFieldView.semanticContentAttribute = lang == .en ? .forceRightToLeft : .forceLeftToRight - // } else { - // textField.textAlignment = lang == .en ? .left : .right - // textFieldView.semanticContentAttribute = lang == .en ? .forceLeftToRight : .forceRightToLeft - // - // } - -// field.unitValue.observe { [weak self] unit in -// guard let strongSelf = self else { return } -// -// strongSelf.borderView.addBorder(radius: 13, -// width: 1, -// color: .border) -// -// //strongSelf.textField.set(text: unit, -//// color: .placeholder, -//// font: .regular(16)) -// strongSelf.unitValueLabel.show() -// strongSelf.unitValueLabel.set(text: unit, -// color: .red, -// font: .regular(16)) -// print("putin", self?.field?.isUnitMeasurement, self?.isUnitMeasurement) -// print(strongSelf.unitValueField?.field?.value.value) -// print(strongSelf.unitValueField?.field?.unitValue.value) -// }.disposed(by: disposeBag) + + if field.field == .price || field.field == .sale_price { + textField.keyboardType = .decimalPad + } + + textField.setForEnglighs(localizedPlaceholder: field.placeholder != "" ? field.placeholder : helper.placeholder, color: .lightGray, font: .regular(16)) + + textField.show() + unitValueLabel.hide() + textField.text = field.value.value if !isUnitMeasurement.dropdown { unitValueLabel.hide() @@ -258,23 +213,23 @@ final class KNTextFieldView: KNComponentView { field.value.observe { [weak self] text in guard let strongSelf = self else { return } - strongSelf.textField.text = text + strongSelf.textField.text = text - if text.count > 0 { - strongSelf.borderView.addBorder(radius: 13, - width: 2, - color: .purple) - - if strongSelf.field?.field != .email || strongSelf.field?.field != .shipping_address_email || strongSelf.field?.field != .billing_address_email { - strongSelf.field?.error.value = "" - } - } else { - //@TODO: check the reset thign again(no reload/updates?) - strongSelf.resetTextField() + if text.count > 0 { + strongSelf.borderView.addBorder(radius: 13, + width: 2, + color: .purple) + + if strongSelf.field?.field != .email || strongSelf.field?.field != .shipping_address_email || strongSelf.field?.field != .billing_address_email { + strongSelf.field?.error.value = "" } + } else { + //@TODO: check the reset thign again(no reload/updates?) + strongSelf.resetTextField() + } }.disposed(by: disposeBag) - + textField.bind { [weak self] binded in guard let strongSelf = self else { return } @@ -303,12 +258,10 @@ final class KNTextFieldView: KNComponentView { } } - // ////////////////////////////////// // - - // accessory - if field.field == .password || field.field == .confirm_password || field.field == .old_password { + if field.field == .password { + accessoryViewIcon.set(icon: helper.icon) accessoryView_width.set(50) accessoryView.show() accessoryViewIcon.show() @@ -323,9 +276,9 @@ final class KNTextFieldView: KNComponentView { set(error: field.error.value) } else { - - infoLabel_top.set(0) - infoLabel.text = "" + + infoLabel_top.set(0) + infoLabel.text = "" } field.error.observe { [weak self] text in @@ -370,21 +323,22 @@ final class KNTextFieldView: KNComponentView { } textField.textAlignment = .left textField.semanticContentAttribute = .forceLeftToRight - + } // MARK: - Handler @objc private func buttonTapped() { - + } @objc private func prefixTapped() { - + } @objc private func accessoryButtonTapped() { - + textField.isSecureTextEntry.toggle() + accessoryViewIcon.set(icon: textField.isSecureTextEntry ? .field_show_password : .field_hide_password) } //@TODO @bibi add network error field @@ -424,11 +378,11 @@ extension KNTextFieldView { private func resetTextField() { if isUnitMeasurement.dropdown { return } guard let field = field else { return } - textField.textAlignment = .left - textField.semanticContentAttribute = .forceLeftToRight -// textField.textAlignment = lang == .en ? .left : .right -// textField.semanticContentAttribute = lang == .en ? .forceLeftToRight : .forceRightToLeft -// + textField.textAlignment = .left + textField.semanticContentAttribute = .forceLeftToRight + // textField.textAlignment = lang == .en ? .left : .right + // textField.semanticContentAttribute = lang == .en ? .forceLeftToRight : .forceRightToLeft + // set(field: field) } } @@ -453,13 +407,13 @@ extension KNTextFieldView: UITextFieldDelegate { let nsString = NSString(string: textFieldString) let updatedString = nsString.replacingCharacters(in:range, with:string) textField.text = updatedString - + //Setting the cursor at the right place let selectedRange = NSMakeRange(range.location + string.count, 0) let from = textField.position(from: textField.beginningOfDocument, offset:selectedRange.location) let to = textField.position(from: from!, offset:selectedRange.length) textField.selectedTextRange = textField.textRange(from: from!, to: to!) - + //Sending an action textField.sendActions(for: .editingChanged) diff --git a/TraccarClient/Custom/BMTextFieldView/KNTextFieldViewHelper.swift b/TraccarClient/Custom/BMTextFieldView/KNTextFieldViewHelper.swift index 0b898ad4284d41d19eae8b2dcd64e01cb0387dea..ad30934404a703fb3ac888b5645846b59288f19b 100644 --- a/TraccarClient/Custom/BMTextFieldView/KNTextFieldViewHelper.swift +++ b/TraccarClient/Custom/BMTextFieldView/KNTextFieldViewHelper.swift @@ -25,19 +25,45 @@ final class KNTextFieldViewHelper { // MARK: - Localization var title: String { - return "" + switch field { + case .password: + return "Password" + case .domain: + return "Domain" + case .username: + return "Username" + default: + return "" + } } var placeholder: String { - return "" + switch field { + case .password: + return "********" + case .domain: + return "Domain" + case .username: + return "Username" + default: + return "" + } } var error: String { return "" } - - + var icon: UIImage { + + if field == .password || field == .confirm_password || field == .old_password{ + return .field_show_password + } + + return UIImage() + } + + // keyboard type var keyboard: UIKeyboardType { diff --git a/TraccarClient/Custom/BMTextFieldView/Model/KNField.swift b/TraccarClient/Custom/BMTextFieldView/Model/KNField.swift index 24d8be43f7ebcf257e6d1a1779403e354a482927..6a7cdab98b7bfada7aaca014525764f5ef28509c 100644 --- a/TraccarClient/Custom/BMTextFieldView/Model/KNField.swift +++ b/TraccarClient/Custom/BMTextFieldView/Model/KNField.swift @@ -20,6 +20,9 @@ final class KNField { var value: Observable<String> = Observable("") var info: Observable<String> = Observable("") var error: Observable<String> = Observable("") + + var action: KNButtonViewAction = .tap + var style: KNButtonViewStyle = .simple var title: String = "" @@ -32,7 +35,8 @@ final class KNField { var labelStyle: KNLableStyle = .simple var allignment: NSTextAlignment = .natural - + var placeholder: String = "" + // MARK: - Init init() { } @@ -47,6 +51,8 @@ final class KNField { error.value = builder.error info.value = builder.info + action = builder.action + style = builder.style height = builder.height title = builder.title @@ -57,6 +63,7 @@ final class KNField { labelStyle = builder.labelStyle allignment = builder.allignment hidden = builder.hidden + placeholder = builder.placeholder } convenience init(_ block: KNFieldBuilderClosure) { @@ -81,6 +88,11 @@ final class KNField { if !value.value.isEmpty { json.add(key: field, value.value) } + default: + if !value.value.isEmpty { + json.add(key: field, value.value) + } + } return json diff --git a/TraccarClient/Custom/BMTextFieldView/Model/KNFieldBuilder.swift b/TraccarClient/Custom/BMTextFieldView/Model/KNFieldBuilder.swift index 7b04788d249d85c977bcd78f1dc4e13df76a45dd..2cc23f316c06102ef87e16a3cc15a73474d0df62 100644 --- a/TraccarClient/Custom/BMTextFieldView/Model/KNFieldBuilder.swift +++ b/TraccarClient/Custom/BMTextFieldView/Model/KNFieldBuilder.swift @@ -25,7 +25,9 @@ final class KNFieldBuilder { var value: String = "" var error: String = "" var info: String = "" - + var action: KNButtonViewAction = .tap + var style: KNButtonViewStyle = .simple + var height: CGFloat = .automatic var enabled: Bool = true @@ -35,7 +37,8 @@ final class KNFieldBuilder { var labelStyle: KNLableStyle = .simple var allignment: NSTextAlignment = .natural - + var placeholder: String = "" + // MARK: - Init init() { diff --git a/TraccarClient/Custom/BMTextFieldView/Model/KNFieldType.swift b/TraccarClient/Custom/BMTextFieldView/Model/KNFieldType.swift index 733004711baefe1fb72fb533b6ee5e4ffc6b555c..2333af8f232b4424b7cdb4b2d436251c948d7afd 100644 --- a/TraccarClient/Custom/BMTextFieldView/Model/KNFieldType.swift +++ b/TraccarClient/Custom/BMTextFieldView/Model/KNFieldType.swift @@ -15,4 +15,5 @@ enum KNFieldTag { enum KNFieldType { case empty case text + case button } diff --git a/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.swift b/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.swift new file mode 100644 index 0000000000000000000000000000000000000000..4553e6d5e715448b9fe4d2e9bb0c2cad41d096c5 --- /dev/null +++ b/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.swift @@ -0,0 +1,58 @@ +// +// ButtonTableViewCell.swift +// opa +// +// Created by iOS Developer on 2/7/22. +// + +import UIKit + +protocol ButtonTableViewCellDelegate: AnyObject { + func cellButtonTapped(action: KNButtonViewAction) +} + +final class ButtonTableViewCell: KNTableViewCell { + + // MARK: - Outlets + @IBOutlet private weak var buttonView: KNButtonView! + + // MARK: - Properties + private var indexPath: IndexPath? + private var field: KNField? + + weak var delegate: ButtonTableViewCellDelegate? + + // MARK: - Lifecycle + override func awakeFromNib() { + super.awakeFromNib() + + setupUI() + } + + private func setupUI() { + contentView.backgroundColor = .clear + layer.cornerRadius = 10 + self.contentView.layer.cornerRadius = 10 + } + + private func updateUI() { + guard let field = field else { return } + buttonView.layer.cornerRadius = 10 + buttonView.primary(title: field.title, style: field.style, action: field.action, delegate: self) + } + + // MARK: - Setup + func set(field: KNField?, indexPath: IndexPath?) { + self.indexPath = indexPath + self.field = field + + updateUI() + } +} + +// MARK: - Button Delegate +extension ButtonTableViewCell: KNButtonViewDelegate { + func buttonTapped(with action: KNButtonViewAction) { + delegate?.cellButtonTapped(action: action) + } +} diff --git a/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.xib b/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.xib new file mode 100644 index 0000000000000000000000000000000000000000..a04499d38dd36f01a9e7e87a54398ac1958d3b07 --- /dev/null +++ b/TraccarClient/Custom/KNButtonView/Cell/ButtonTableViewCell.xib @@ -0,0 +1,39 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="19162" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES"> + <device id="retina6_1" orientation="portrait" appearance="light"/> + <dependencies> + <deployment identifier="iOS"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="19144"/> + <capability name="Safe area layout guides" minToolsVersion="9.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <objects> + <placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner"/> + <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> + <tableViewCell contentMode="scaleToFill" selectionStyle="default" indentationWidth="10" reuseIdentifier="ButtonTableViewCell" rowHeight="454" id="KGk-i7-Jjw" customClass="ButtonTableViewCell" customModule="opa" customModuleProvider="target"> + <rect key="frame" x="0.0" y="0.0" width="499" height="454"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMaxY="YES"/> + <tableViewCellContentView key="contentView" opaque="NO" clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="center" tableViewCell="KGk-i7-Jjw" id="H2p-sc-9uM"> + <rect key="frame" x="0.0" y="0.0" width="499" height="454"/> + <autoresizingMask key="autoresizingMask"/> + <subviews> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="vGc-2X-rhu" customClass="KNButtonView" customModule="opa" customModuleProvider="target"> + <rect key="frame" x="16" y="0.0" width="467" height="454"/> + <color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + </view> + </subviews> + <constraints> + <constraint firstAttribute="bottom" secondItem="vGc-2X-rhu" secondAttribute="bottom" id="2hC-us-axl"/> + <constraint firstAttribute="trailing" secondItem="vGc-2X-rhu" secondAttribute="trailing" constant="16" id="Osn-ZS-lkn"/> + <constraint firstItem="vGc-2X-rhu" firstAttribute="leading" secondItem="H2p-sc-9uM" secondAttribute="leading" constant="16" id="XaI-ap-r49"/> + <constraint firstItem="vGc-2X-rhu" firstAttribute="top" secondItem="H2p-sc-9uM" secondAttribute="top" id="vEN-55-PfX"/> + </constraints> + </tableViewCellContentView> + <viewLayoutGuide key="safeArea" id="njF-e1-oar"/> + <connections> + <outlet property="buttonView" destination="vGc-2X-rhu" id="xkP-uC-kY5"/> + </connections> + <point key="canvasLocation" x="261.59420289855075" y="141.29464285714286"/> + </tableViewCell> + </objects> +</document> diff --git a/TraccarClient/Custom/KNButtonView/KNButtonView.swift b/TraccarClient/Custom/KNButtonView/KNButtonView.swift new file mode 100644 index 0000000000000000000000000000000000000000..abd50edb31759c0b3da10a4bf34a988212dfe810 --- /dev/null +++ b/TraccarClient/Custom/KNButtonView/KNButtonView.swift @@ -0,0 +1,251 @@ +// +// KNButtonView.swift +// opa +// +// Created by iOS Developer on 2/2/22. +// + +import Foundation +import UIKit + +final class KNButtonView: KNComponentView { + + // MARK: - Outlets + @IBOutlet private weak var view: UIView! + @IBOutlet private weak var boxView: UIView! + @IBOutlet private weak var button: UIButton! + + // MARK: - Properties + private var color: UIColor = .purple + private var title: String = "" + private var cornerRadius: CGFloat = 10.0 + private var titleColor: UIColor = .white + private var titleFont: UIFont = .semibold(17) + private var icon: UIImage = UIImage() + private var type: KNButtonViewType = .primary + private var style: KNButtonViewStyle = .simple + private var action: KNButtonViewAction = .tap + + weak var delegate: KNButtonViewDelegate? + + // MARK: - Lifecycle + override func setupNib() { + super.setupNib() + + view.frame = bounds + view.autoresizingMask = [.flexibleHeight, .flexibleWidth] + + setupUI() + } + + // MARK: - UI + private func setupUI() { + view.backgroundColor = .clear + boxView.backgroundColor = .clear + + button.addTarget(self, action: #selector(buttonTapped), for: .touchUpInside) + } + + private func adjustSpaceBetweenTitleAndIcon() { + var padding: CGFloat = 5 + + if style != .withIcon { return } + + + button.titleEdgeInsets = UIEdgeInsets(top: 0, left: padding, bottom: 0, right: 0) + button.imageEdgeInsets = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: padding) + } + + private func adjustIconPlacement() { + if style != .withIconAfterText { return } + + let padding: CGFloat = 5 + + button.semanticContentAttribute = .forceLeftToRight + + button.titleEdgeInsets = UIEdgeInsets(top: 0, left: padding, bottom: 0, right: 0) + button.imageEdgeInsets = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: padding) + + //@TODO @george clarify + button.transform = CGAffineTransform(scaleX: -1.0, y: 1.0) + button.titleLabel?.transform = CGAffineTransform(scaleX: -1.0, y: 1.0) + button.imageView?.transform = CGAffineTransform(scaleX: -1.0, y: 1.0) + } + + // MARK: - Logic + @objc private func buttonTapped() { + delegate?.buttonTapped(with: action) + } + + func primary(color: UIColor, title: String, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.color = color + self.title = title + self.type = .primary + self.style = style + self.action = action + self.delegate = delegate + + updatePrimary() + } + + func primary(title: String, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.type = .primary + self.style = style + self.action = action + self.delegate = delegate + + updatePrimary() + } + + func primary(title: String, icon: UIImage, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.icon = icon + self.type = .primary + self.style = style + self.action = action + self.delegate = delegate + + button.set(icon: icon) + + updatePrimary() + } + + func primary(color: UIColor, cornerRadius: CGFloat, title: String, titleFont: UIFont, titleColor: UIColor, icon: UIImage, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.color = color + self.cornerRadius = cornerRadius + self.title = title + self.titleFont = titleFont + self.titleColor = titleColor + self.icon = icon + self.type = .primary + self.style = style + self.action = action + self.delegate = delegate + + button.set(icon: icon) + + updatePrimary() + } + + func secondary(title: String, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.type = .secondary + self.style = style + self.action = action + self.delegate = delegate + + updateSecondary() + } + + func secondary(title: String, icon: UIImage, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.icon = icon + self.type = .secondary + self.style = style + self.action = action + self.delegate = delegate + + button.set(icon: icon) + + updateSecondary() + } + + func tertiary(title: String, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.type = .tertiary + self.style = style + self.action = action + self.delegate = delegate + + updateTertiary() + } + + func tertiary(title: String, icon: UIImage, style: KNButtonViewStyle, action: KNButtonViewAction, delegate: KNButtonViewDelegate?) { + self.title = title + self.icon = icon + self.type = .tertiary + self.style = style + self.action = action + self.delegate = delegate + + button.set(icon: icon) + + updateTertiary() + } + + private func updatePrimary() { + boxView.addCorners(cornerRadius) + boxView.backgroundColor = color + titleColor = .white + + button.set(localized: title, color: titleColor, font: titleFont) + } + + private func updateSecondary() { + boxView.addCorners(10) + + if action == .retryPayment || action == .trackOrder { + boxView.backgroundColor = .purple + + } else { + boxView.backgroundColor = .purple + } + + + var color: UIColor = .white + var font: UIFont = .semibold(17) + + if style == .transparent { + boxView.backgroundColor = .purple + + color = .purple + font = .semibold(14) + }else if style == .transperantExtraBlue { + boxView.backgroundColor = .purple + + titleColor = .white + titleFont = .semibold(14) + } + + adjustSpaceBetweenTitleAndIcon() + adjustIconPlacement() + + button.set(localized: title, color: color, font: font) + } + + private func updateTertiary() { + boxView.addCorners(10) + boxView.backgroundColor = .purple + + adjustSpaceBetweenTitleAndIcon() + adjustIconPlacement() + + button.set(localized: title, color: .white, font: .semibold(17)) + } + + func setAllignment(_ side :UIControl.ContentHorizontalAlignment) { + button.contentHorizontalAlignment = side + boxView.addCorners(side == .center ? 10 : 0) + } + + + func secondaryGrey(title: String, + style: KNButtonViewStyle, + action: KNButtonViewAction, + delegate: KNButtonViewDelegate?) { + self.title = title + self.type = .secondaryGrey + self.style = style + self.action = action + self.delegate = delegate + + boxView.addCorners(cornerRadius) + boxView.backgroundColor = .purple + + titleColor = .white + titleFont = .semibold(17) + + button.set(localized: title, color: titleColor, font: titleFont) + } +} diff --git a/TraccarClient/Custom/KNButtonView/KNButtonView.xib b/TraccarClient/Custom/KNButtonView/KNButtonView.xib new file mode 100644 index 0000000000000000000000000000000000000000..2bd2a80415e04a9e0670396496fba5adb4003955 --- /dev/null +++ b/TraccarClient/Custom/KNButtonView/KNButtonView.xib @@ -0,0 +1,58 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="17701" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES"> + <device id="retina6_1" orientation="portrait" appearance="light"/> + <dependencies> + <deployment identifier="iOS"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="17703"/> + <capability name="Safe area layout guides" minToolsVersion="9.0"/> + <capability name="System colors in document resources" minToolsVersion="11.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <objects> + <placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner" customClass="KNButtonView" customModule="opa" customModuleProvider="target"> + <connections> + <outlet property="boxView" destination="1bM-jL-Zov" id="IDg-oc-lDD"/> + <outlet property="button" destination="ud2-KH-K4K" id="pnY-G5-8NP"/> + <outlet property="view" destination="iN0-l3-epB" id="acd-0z-aGn"/> + </connections> + </placeholder> + <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> + <view contentMode="scaleToFill" id="iN0-l3-epB"> + <rect key="frame" x="0.0" y="0.0" width="701" height="861"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="1bM-jL-Zov"> + <rect key="frame" x="0.0" y="44" width="701" height="783"/> + <subviews> + <button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="ud2-KH-K4K"> + <rect key="frame" x="0.0" y="0.0" width="701" height="783"/> + <state key="normal" title=" "/> + </button> + </subviews> + <color key="backgroundColor" systemColor="systemBackgroundColor"/> + <constraints> + <constraint firstItem="ud2-KH-K4K" firstAttribute="leading" secondItem="1bM-jL-Zov" secondAttribute="leading" id="IOR-b1-C0e"/> + <constraint firstAttribute="bottom" secondItem="ud2-KH-K4K" secondAttribute="bottom" id="KS4-Uv-UR6"/> + <constraint firstItem="ud2-KH-K4K" firstAttribute="top" secondItem="1bM-jL-Zov" secondAttribute="top" id="agC-PT-oRP"/> + <constraint firstAttribute="trailing" secondItem="ud2-KH-K4K" secondAttribute="trailing" id="hbj-aU-62J"/> + </constraints> + </view> + </subviews> + <viewLayoutGuide key="safeArea" id="vUN-kp-3ea"/> + <color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + <constraints> + <constraint firstItem="1bM-jL-Zov" firstAttribute="top" secondItem="vUN-kp-3ea" secondAttribute="top" id="26D-iR-c1O"/> + <constraint firstItem="vUN-kp-3ea" firstAttribute="trailing" secondItem="1bM-jL-Zov" secondAttribute="trailing" id="6St-5y-9c3"/> + <constraint firstItem="vUN-kp-3ea" firstAttribute="bottom" secondItem="1bM-jL-Zov" secondAttribute="bottom" id="ncE-53-Dw9"/> + <constraint firstItem="1bM-jL-Zov" firstAttribute="leading" secondItem="vUN-kp-3ea" secondAttribute="leading" id="voN-Hy-OfY"/> + </constraints> + <freeformSimulatedSizeMetrics key="simulatedDestinationMetrics"/> + <point key="canvasLocation" x="-25" y="79"/> + </view> + </objects> + <resources> + <systemColor name="systemBackgroundColor"> + <color white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + </systemColor> + </resources> +</document> diff --git a/TraccarClient/Custom/KNButtonView/KNButtonViewHelper.swift b/TraccarClient/Custom/KNButtonView/KNButtonViewHelper.swift new file mode 100644 index 0000000000000000000000000000000000000000..654f7c802dfbd922f7aa10e61aaf8bfb8ce8b2be --- /dev/null +++ b/TraccarClient/Custom/KNButtonView/KNButtonViewHelper.swift @@ -0,0 +1,73 @@ +// +// KNButtonViewHelper.swift +// opa +// +// Created by iOS Developer on 2/2/22. +// + +import Foundation + +protocol KNButtonViewDelegate: AnyObject { + func buttonTapped(with action: KNButtonViewAction) +} + +enum KNButtonViewAction { + case upload + case remove + case new + case addToCart + case buyNow + case message + case call + case back + case next + case tap + case forgotPassword + case signin + case createAccount + case searchAccount + case verify + case didNotReceiveCode + case resetPassword + case resentCode + case changeLanguage + case changeCurrency + case add_address + case addAttachment + case addVariant + case variantsDone + case viewAttachment + case userLocation + case submit + case personalInformationSettings + case accountSettings + case addLocation + case showPicker + case proceedToCheckout + case done + case cancel + case trackOrder + case continueShopping + case submitReport + case guest + case retryPayment + case addSizeVariantField + case addColorVariantField +} + +enum KNButtonViewStyle { + case transparent + case transperantExtraBlue + case simple + case withIcon + case withBorder + case withIconAfterText +} + +enum KNButtonViewType { + case primary + case secondary + case tertiary + + case secondaryGrey +} diff --git a/TraccarClient/Custom/KNSpinner/KNSpinnerView.swift b/TraccarClient/Custom/KNSpinner/KNSpinnerView.swift new file mode 100644 index 0000000000000000000000000000000000000000..f5a787ac871f0eace8dd843b1100708a1e727b87 --- /dev/null +++ b/TraccarClient/Custom/KNSpinner/KNSpinnerView.swift @@ -0,0 +1,41 @@ +// +// KNSpinnerView.swift +// opa +// +// Created by George Makhoul on 2/22/22. +// + +import Foundation +import UIKit + +final class KNSpinnerView: KNComponentView { + + // MARK: - Outlets + @IBOutlet private weak var view: UIView! + + // MARK: - Properties + static let tag: Int = 130_220_310_000 + var requestTag: Endpoint? + + // MARK: - Lifecycle + override func setupNib() { + super.setupNib() + + view.frame = bounds + view.autoresizingMask = [.flexibleWidth, .flexibleHeight] + + setupUI() + } + + private func setupUI() { + tag = KNSpinnerView.tag + + backgroundColor = .clear + view.backgroundColor = .clear + + } + + func stop() { + removeFromSuperview() + } +} diff --git a/TraccarClient/Custom/KNSpinner/KNSpinnerView.xib b/TraccarClient/Custom/KNSpinner/KNSpinnerView.xib new file mode 100644 index 0000000000000000000000000000000000000000..b49dc8975bffc80b197cc165ed2ac7e275cf9a5d --- /dev/null +++ b/TraccarClient/Custom/KNSpinner/KNSpinnerView.xib @@ -0,0 +1,42 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="22155" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES"> + <device id="retina4_0" orientation="portrait" appearance="light"/> + <dependencies> + <deployment identifier="iOS"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="22131"/> + <capability name="Safe area layout guides" minToolsVersion="9.0"/> + <capability name="System colors in document resources" minToolsVersion="11.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <objects> + <placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner" customClass="KNSpinnerView" customModule="TraccarClient" customModuleProvider="target"> + <connections> + <outlet property="view" destination="iN0-l3-epB" id="gWS-OW-eEB"/> + </connections> + </placeholder> + <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> + <view contentMode="scaleToFill" id="iN0-l3-epB"> + <rect key="frame" x="0.0" y="0.0" width="530" height="569"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <activityIndicatorView opaque="NO" contentMode="scaleToFill" horizontalHuggingPriority="750" verticalHuggingPriority="750" animating="YES" style="large" translatesAutoresizingMaskIntoConstraints="NO" id="U6U-1g-XhU"> + <rect key="frame" x="246.5" y="266" width="37" height="37"/> + <color key="color" systemColor="systemPurpleColor"/> + </activityIndicatorView> + </subviews> + <viewLayoutGuide key="safeArea" id="vUN-kp-3ea"/> + <color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + <constraints> + <constraint firstItem="U6U-1g-XhU" firstAttribute="centerX" secondItem="iN0-l3-epB" secondAttribute="centerX" id="cfp-i8-iwL"/> + <constraint firstItem="U6U-1g-XhU" firstAttribute="centerY" secondItem="iN0-l3-epB" secondAttribute="centerY" id="kH9-mm-lIs"/> + </constraints> + <freeformSimulatedSizeMetrics key="simulatedDestinationMetrics"/> + <point key="canvasLocation" x="-41" y="154"/> + </view> + </objects> + <resources> + <systemColor name="systemPurpleColor"> + <color red="0.68627450980392157" green="0.32156862745098042" blue="0.87058823529411766" alpha="1" colorSpace="custom" customColorSpace="sRGB"/> + </systemColor> + </resources> +</document> diff --git a/TraccarClient/DeviceModel.swift b/TraccarClient/DeviceModel.swift new file mode 100644 index 0000000000000000000000000000000000000000..371b245355cfaa7955edbf74ad36c271ad8d6653 --- /dev/null +++ b/TraccarClient/DeviceModel.swift @@ -0,0 +1,64 @@ +// +// DeviceModel.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +final class DeviceModel: KNObject { + + // MARK: - Properties + var name: String = "" + var uniqueId: String = "" + var status: String = "" + + // MARK: - Init + override init() { + super.init() + } + + init(_ dict: JSON, saveUser: Bool = false) { + super.init() + name = dict["name"] as? String ?? "" + uniqueId = dict["uniqueId"] as? String ?? "" + status = dict["status"] as? String ?? "" + + if saveUser { + saveToUserDefaults() + } + } + //MARK: - Methods + private func saveToUserDefaults() { + let ud = UserDefaults.standard + ud.set(name, forKey: Keys.User.deviceName) + ud.set(uniqueId, forKey: Keys.User.uniqueId) + ud.set(status, forKey: Keys.User.status) + } + + class func loadFromUserDefaults() -> DeviceModel { + let ud = UserDefaults.standard + var params: PARAMS = [:] + + if let deviceName = ud.value(forKey: Keys.User.deviceName) as? String { + params.add(key: .name, deviceName) + } + if let uniqueId = ud.value(forKey: Keys.User.uniqueId) as? String { + params.add(key: .uniqueId, uniqueId) + } + if let status = ud.value(forKey: Keys.User.status) as? String { + params.add(key: .status, status) + } + return DeviceModel(params) + } + + class func clearUserFromUserDefaults() { + let ud = UserDefaults.standard + + ud.removeObject(forKey: Keys.User.deviceName) + ud.removeObject(forKey: Keys.User.uniqueId) + ud.removeObject(forKey: Keys.User.status) + + } +} diff --git a/TraccarClient/Globals.swift b/TraccarClient/Globals.swift new file mode 100644 index 0000000000000000000000000000000000000000..af5e047f604290fdca27a07bff281fed6bd5269d --- /dev/null +++ b/TraccarClient/Globals.swift @@ -0,0 +1,176 @@ +// +// Globals.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +import UIKit + + +// MARK: - Delay with Closure +func delay(_ by: Double = 2.0, completion: @escaping () -> ()) { + DispatchQueue.main.asyncAfter(deadline: .now() + by) { + completion() + } +} + +// MARK: - Main Thread +func main(_ completion: @escaping () -> ()) { + DispatchQueue.main.async() { + completion() + } +} + +// MARK: - Global Custom AlertView +typealias AlertViewClosure = () -> () + +// use this when you only need to display the error without any actions +func ok(_ message: String, viewController:KNViewController? = nil) { + let vc = KNAlertViewController(type: .error, message: message) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + if let viewController = viewController { + viewController.present(vc, animated: false, completion: nil) + }else{ + topViewController()?.present(vc, animated: false, completion: nil) + } +} + +func ok(_ message: String, _ completion: AlertViewClosure?) { + let vc = KNAlertViewController(type: .error, message: message) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + vc.confirmClosure = completion + + topViewController()?.present(vc, animated: false, completion: nil) +} + +func ok(_ result: NetworkResultModel) { + guard result.haveFieldsErrors == false else { return } + log("error", result.displayError) + var message: String = "" + message = result.displayError + + var codeWithMessage = "" + if app.environment != .production { + codeWithMessage = "[\(result.statusCode)]\n\n" + message + } else { + codeWithMessage = message + } + + let vc = KNAlertViewController(type: .error, message: codeWithMessage) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + + topViewController()?.present(vc, animated: false, completion: nil) +} + +// use this when you need to have the action from the button +func ok(_ result: NetworkResultModel, _ completion: AlertViewClosure?){ + guard result.displayError.isEmpty else { return } + + var message: String = "" + + if result.displayError.isEmpty { + message = "" + } else { + message = result.displayError + } + + var codeWithMessage = "" + if app.environment != .production { + codeWithMessage = "[\(result.statusCode)]\n\n" + message + } else { + codeWithMessage = message + } + log("error", message) + let vc = KNAlertViewController(type: .error, message: codeWithMessage) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + vc.confirmClosure = completion + + topViewController()?.present(vc, animated: false, completion: nil) +} + +@discardableResult +func alert(confirm: String, + destructive: String, + message: String, + vcToPresent: UIViewController? = nil, + _ completion: AlertViewClosure?, + _ cancelCompletion: AlertViewClosure? = nil) -> KNAlertViewController { + + let vc = KNAlertViewController(type: .decision, + message: message, + confirm: confirm, + destructive: destructive) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + vc.confirmClosure = completion + vc.destructiveClosure = cancelCompletion + + // Todo to be removed when detecting the error of twice appearing in the pop-up. + + if let vcToPresent = vcToPresent { + vcToPresent.present(vc, animated: false, completion: nil) + + } else { + if let topV = topViewController() as? KNAlertViewController { + return topV + } + topViewController()?.present(vc, animated: false, completion: nil) + } + + + + return vc +} + +@discardableResult +func alert(confirm: String, message: String, _ completion: AlertViewClosure?) -> KNAlertViewController { + let vc = KNAlertViewController(type: .error, + message: message, + confirm: confirm, destructive: "") + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + vc.confirmClosure = completion + + // Todo to be removed when detecting the error of twice appearing in the pop-up. + //@TODO what is this @bibi. + if let topV = topViewController() as? KNAlertViewController { + return topV + } + topViewController()?.present(vc, animated: false, completion: nil) + + return vc +} +@discardableResult + +func alert(presentingBy controller: KNViewController, + confirm: String, + destructive: String, + message: String, + _ completion: AlertViewClosure?, + _ cancelCompletion: AlertViewClosure? = nil) -> KNAlertViewController { + + let vc = KNAlertViewController(type: .decision, + message: message, + confirm: confirm, + destructive: destructive) + vc.modalPresentationStyle = .overFullScreen + vc.modalTransitionStyle = .crossDissolve + vc.confirmClosure = completion + vc.destructiveClosure = cancelCompletion + + // Todo to be removed when detecting the error of twice appearing in the pop-up. + if let topV = topViewController() as? KNAlertViewController { + return topV + } + controller.present(vc, animated: false, completion: nil) + + return vc +} + diff --git a/TraccarClient/Images.xcassets/.DS_Store b/TraccarClient/Images.xcassets/.DS_Store new file mode 100644 index 0000000000000000000000000000000000000000..cf59f693fc5726c57280ef274cc28e817a562ed0 Binary files /dev/null and b/TraccarClient/Images.xcassets/.DS_Store differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/100.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/100.png deleted file mode 100644 index c2fd176693fa7fb61ea3a98bedff3d287d8b26da..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/100.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/1024.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/1024.png index 2b31dd5db33453fa95d6fcf21e3a53c9d8ce61e7..e99e2a635dcb2fe151c3bc4cf43f6c7c73556cd2 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/1024.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/1024.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/114.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/114.png index b590b7608aadbbd0287265e820ab68ecfb2eef70..d149871669a351d62041eec106fd23503ab407ee 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/114.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/114.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/120.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/120.png index ff47ff513bd7dd3b14e765f59d6b05a844acbce0..d95a99b26225e4deb783c664856fb9c5d2ffd242 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/120.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/120.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/144.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/144.png deleted file mode 100644 index c7dc8fd745e0649bf030ba506a678422ecbc12bc..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/144.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/152.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/152.png deleted file mode 100644 index a352b16e19f007dd26d6e53cd6b49ceccc33826f..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/152.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/167.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/167.png deleted file mode 100644 index 775d0f13a7a0dd1df91e27ce450b4a00aba9a65f..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/167.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/180.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/180.png index 2da8eb5371d8049cae1cdc3d634e3d63fb999b42..9c74097a8d8a7bc5abd577dcb65dd31353dea94f 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/180.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/180.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/20.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/20.png deleted file mode 100644 index a02a3585dda02fea50c666c7780b8fc7d1adb4bd..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/20.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/29.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/29.png index 9332bbfbebe3a53fd8331ba4594b5513619d9ca2..ba88e87cbbbae7d2b596c7c8c21f5e2eb4f2ff6e 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/29.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/29.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/40.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/40.png index 0ad24730ab6915664a8aa18a40d4856a13c59e94..29cb59ce3605ee78fb03cb7d888310cc153e7da4 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/40.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/40.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/50.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/50.png deleted file mode 100644 index f1d873799721bafa570d571928cb82f24243e3c5..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/50.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/57.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/57.png index 27d7e8599f82bec5a73d750fa766450e5e123a2e..21cde5c02fc8016cf9235e74f902e8a3e2b1788d 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/57.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/57.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/58.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/58.png index 71395ffb8e585642c602bcab0d2ff455ce62f0e1..b975349bf7c705f6e9b52ddc74923643fb73803a 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/58.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/58.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/60.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/60.png index 8e12676f4f3bcb1190676377db77be2f213f30c9..eef25887c92ec4fe0f4b10e3b712440f40aec68c 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/60.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/60.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/72.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/72.png deleted file mode 100644 index 0a3720e633247927765e8dbdcaf1d49366ff0264..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/72.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/76.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/76.png deleted file mode 100644 index 93a8c0dca2cecd5cf724fcd56a88347cb87e9141..0000000000000000000000000000000000000000 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/76.png and /dev/null differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/80.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/80.png index 627736770128d5e760cce2e28ac18357e1f74277..0a635a93c9142f8c47d33ea65db62a0bde857765 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/80.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/80.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/87.png b/TraccarClient/Images.xcassets/AppIcon.appiconset/87.png index c25052d78fdde19b8ec4f62abb6b4f75a30d0bf2..a1c42a757ee62b35b8346a2cc5a120858ddeb54b 100644 Binary files a/TraccarClient/Images.xcassets/AppIcon.appiconset/87.png and b/TraccarClient/Images.xcassets/AppIcon.appiconset/87.png differ diff --git a/TraccarClient/Images.xcassets/AppIcon.appiconset/Contents.json b/TraccarClient/Images.xcassets/AppIcon.appiconset/Contents.json index 65b74d7ef11fa59fafa829e681ac90906f3ac8b2..73d3b7f6d10839e48787d4ed581df4a1194b51fb 100644 --- a/TraccarClient/Images.xcassets/AppIcon.appiconset/Contents.json +++ b/TraccarClient/Images.xcassets/AppIcon.appiconset/Contents.json @@ -1 +1 @@ -{"images":[{"size":"60x60","expected-size":"180","filename":"180.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"40x40","expected-size":"80","filename":"80.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"40x40","expected-size":"120","filename":"120.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"60x60","expected-size":"120","filename":"120.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"57x57","expected-size":"57","filename":"57.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"1x"},{"size":"29x29","expected-size":"58","filename":"58.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"29x29","expected-size":"29","filename":"29.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"1x"},{"size":"29x29","expected-size":"87","filename":"87.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"57x57","expected-size":"114","filename":"114.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"20x20","expected-size":"40","filename":"40.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"20x20","expected-size":"60","filename":"60.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"1024x1024","filename":"1024.png","expected-size":"1024","idiom":"ios-marketing","folder":"Assets.xcassets/AppIcon.appiconset/","scale":"1x"},{"size":"40x40","expected-size":"80","filename":"80.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"72x72","expected-size":"72","filename":"72.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"76x76","expected-size":"152","filename":"152.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"50x50","expected-size":"100","filename":"100.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"29x29","expected-size":"58","filename":"58.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"76x76","expected-size":"76","filename":"76.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"29x29","expected-size":"29","filename":"29.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"50x50","expected-size":"50","filename":"50.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"72x72","expected-size":"144","filename":"144.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"40x40","expected-size":"40","filename":"40.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"83.5x83.5","expected-size":"167","filename":"167.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"},{"size":"20x20","expected-size":"20","filename":"20.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"1x"},{"size":"20x20","expected-size":"40","filename":"40.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"ipad","scale":"2x"}]} \ No newline at end of file +{"images":[{"size":"60x60","expected-size":"180","filename":"180.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"40x40","expected-size":"80","filename":"80.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"40x40","expected-size":"120","filename":"120.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"60x60","expected-size":"120","filename":"120.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"57x57","expected-size":"57","filename":"57.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"1x"},{"size":"29x29","expected-size":"58","filename":"58.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"29x29","expected-size":"29","filename":"29.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"1x"},{"size":"29x29","expected-size":"87","filename":"87.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"57x57","expected-size":"114","filename":"114.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"20x20","expected-size":"40","filename":"40.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"2x"},{"size":"20x20","expected-size":"60","filename":"60.png","folder":"Assets.xcassets/AppIcon.appiconset/","idiom":"iphone","scale":"3x"},{"size":"1024x1024","filename":"1024.png","expected-size":"1024","idiom":"ios-marketing","folder":"Assets.xcassets/AppIcon.appiconset/","scale":"1x"}]} \ No newline at end of file diff --git a/TraccarClient/Images.xcassets/field_hide_password.imageset/Contents.json b/TraccarClient/Images.xcassets/field_hide_password.imageset/Contents.json new file mode 100644 index 0000000000000000000000000000000000000000..cb6b796581f48250bf162eabd8a8e29786d9956f --- /dev/null +++ b/TraccarClient/Images.xcassets/field_hide_password.imageset/Contents.json @@ -0,0 +1,23 @@ +{ + "images" : [ + { + "filename" : "field_hide_password.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "field_hide_password@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "filename" : "field_hide_password@3x.png", + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password.png b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password.png new file mode 100644 index 0000000000000000000000000000000000000000..19c7e7f8434190a8496b0663101dd8460cdb2927 Binary files /dev/null and b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password.png differ diff --git a/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@2x.png b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..3ebc5e41d7aefe8860a1639c7cafe716a9563471 Binary files /dev/null and b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@2x.png differ diff --git a/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@3x.png b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..db3dcde47f2ad8288bf119a5597e31dc327fafbc Binary files /dev/null and b/TraccarClient/Images.xcassets/field_hide_password.imageset/field_hide_password@3x.png differ diff --git a/TraccarClient/Images.xcassets/field_show_password.imageset/Contents.json b/TraccarClient/Images.xcassets/field_show_password.imageset/Contents.json new file mode 100644 index 0000000000000000000000000000000000000000..696fb73985fb6fda04cbf98f647a34d5065ccf7e --- /dev/null +++ b/TraccarClient/Images.xcassets/field_show_password.imageset/Contents.json @@ -0,0 +1,23 @@ +{ + "images" : [ + { + "filename" : "field_show_password.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "field_show_password@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "filename" : "field_show_password@3x.png", + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password.png b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password.png new file mode 100644 index 0000000000000000000000000000000000000000..549ec25cd15f713e8357a5a435a2b92345c99f31 Binary files /dev/null and b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password.png differ diff --git a/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@2x.png b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..9a3755fa7af9f5c97b0c7eb4b4141a2168268026 Binary files /dev/null and b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@2x.png differ diff --git a/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@3x.png b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..ad93d5a915ac27e0cde7961abc0298c2c3f877cb Binary files /dev/null and b/TraccarClient/Images.xcassets/field_show_password.imageset/field_show_password@3x.png differ diff --git a/TraccarClient/InitialViewController/InitialViewController.swift b/TraccarClient/InitialViewController/InitialViewController.swift new file mode 100644 index 0000000000000000000000000000000000000000..6f21ce423b7c62c3c89a009a0f6ea006b3cff9c2 --- /dev/null +++ b/TraccarClient/InitialViewController/InitialViewController.swift @@ -0,0 +1,65 @@ +// +// InitialViewController.swift +// TraccarClient +// +// Created by George Makhoul on 30/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit +import Lottie + +final class InitialViewController: KNViewController { + // MARK: - Outlets + @IBOutlet weak var animationView: LottieAnimationView! + + // MARK: - Properties + private let animationFileName: String = "splash" + var window: UIWindow? + + // MARK: - LifeCycle + override func viewDidLoad() { + super.viewDidLoad() + view.backgroundColor = .white + animationView.backgroundColor = .white + animationView.contentMode = .scaleAspectFill + animationView.loopMode = .repeat(1) + animationView.translatesAutoresizingMaskIntoConstraints = false + + animationView.play {_ in + self.navigateTo() + } + + } + + override func viewDidLayoutSubviews() { + super.viewDidLayoutSubviews() + } +} +// MARK: - Methods +extension InitialViewController { + func navigateTo() { + if !app.isLoggedIn { + let login = LoginViewController() + let navigation = KNNavigationController(rootViewController: login) + navigation.modalPresentationStyle = .overFullScreen + + let coordinator = AuthenticationCoordinator(navigationController: navigation) + coordinator.step = .signin + coordinator.start() + + window?.rootViewController = navigation + window?.makeKeyAndVisible() + + } else { + let storyboard = UIStoryboard(name: "MainStoryboard", bundle: nil) + if let mainViewController = storyboard.instantiateViewController(withIdentifier: "MainViewController") as? MainViewController { + let navigation = KNNavigationController(rootViewController: mainViewController) + navigation.modalPresentationStyle = .overFullScreen + window?.rootViewController = navigation + window?.makeKeyAndVisible() + + } + } + } +} diff --git a/TraccarClient/InitialViewController/InitialViewController.xib b/TraccarClient/InitialViewController/InitialViewController.xib new file mode 100644 index 0000000000000000000000000000000000000000..507ca0fcd40b6cfa3f31497ebde54a68316e2a10 --- /dev/null +++ b/TraccarClient/InitialViewController/InitialViewController.xib @@ -0,0 +1,42 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="22155" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES"> + <device id="retina6_12" orientation="portrait" appearance="light"/> + <dependencies> + <deployment identifier="iOS"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="22131"/> + <capability name="Safe area layout guides" minToolsVersion="9.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <objects> + <placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner" customClass="InitialViewController" customModule="TraccarClient" customModuleProvider="target"> + <connections> + <outlet property="animationView" destination="Ts7-dn-NYa" id="aG1-VB-698"/> + <outlet property="view" destination="i5M-Pr-FkT" id="sfx-zR-JGt"/> + </connections> + </placeholder> + <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> + <view clearsContextBeforeDrawing="NO" contentMode="scaleToFill" insetsLayoutMarginsFromSafeArea="NO" id="i5M-Pr-FkT"> + <rect key="frame" x="0.0" y="0.0" width="393" height="852"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="Ts7-dn-NYa" customClass="LottieAnimationView" customModule="Lottie"> + <rect key="frame" x="0.0" y="0.0" width="393" height="852"/> + <color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + <userDefinedRuntimeAttributes> + <userDefinedRuntimeAttribute type="string" keyPath="animationName" value="splash"/> + </userDefinedRuntimeAttributes> + </view> + </subviews> + <viewLayoutGuide key="safeArea" id="fnl-2z-Ty3"/> + <color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + <constraints> + <constraint firstItem="Ts7-dn-NYa" firstAttribute="top" secondItem="i5M-Pr-FkT" secondAttribute="top" id="3j6-eK-FTm"/> + <constraint firstItem="Ts7-dn-NYa" firstAttribute="leading" secondItem="fnl-2z-Ty3" secondAttribute="leading" id="8qA-EO-VKP"/> + <constraint firstItem="fnl-2z-Ty3" firstAttribute="trailing" secondItem="Ts7-dn-NYa" secondAttribute="trailing" id="bR4-Me-UrT"/> + <constraint firstAttribute="bottom" secondItem="Ts7-dn-NYa" secondAttribute="bottom" id="xiZ-xZ-UW6"/> + </constraints> + <nil key="simulatedTopBarMetrics"/> + <point key="canvasLocation" x="990" y="63"/> + </view> + </objects> +</document> diff --git a/TraccarClient/JSONField.swift b/TraccarClient/JSONField.swift index 0ca1fc2a2ea927afcbccb31ff9e26b296addf858..b0fe04295250b13609d2d623a0a199ca2597eef8 100644 --- a/TraccarClient/JSONField.swift +++ b/TraccarClient/JSONField.swift @@ -13,16 +13,16 @@ enum JSONField: String { case none case app_errors = "app_errors" - case field_errors = "field_errors" + case field_errors = "errors" case display_errors = "display_errors" - case access_token = "access_token" + case access_token = "token" case refresh_token = "refresh_token" case picture = "picture" - case id case results - + case user + case device case email case password case name @@ -30,6 +30,8 @@ enum JSONField: String { case prefix case user_type case account_type + case uniqueId + case status case country case governorate @@ -40,7 +42,9 @@ enum JSONField: String { case height case bio case website - + case avatar + case cover + case designation_id case product_name_en = "name_en" case product_name_ar = "name_ar" case product_type = "type" @@ -80,6 +84,7 @@ enum JSONField: String { case option case confirm_password case code + case domain case username = "username" case business = "business_name" case dynamicSpace diff --git a/TraccarClient/KNAlert/KNAlertViewController.swift b/TraccarClient/KNAlert/KNAlertViewController.swift new file mode 100644 index 0000000000000000000000000000000000000000..7fb23562c59c89ddab15e0971e44dea1eaf6732c --- /dev/null +++ b/TraccarClient/KNAlert/KNAlertViewController.swift @@ -0,0 +1,281 @@ +// +// KNAlertViewController.swift +// opa +// +// Created by George Makhoul on 2/23/22. +// + +import UIKit + +final class KNAlertViewController: KNViewController { + + enum KNAlertViewType { + case none + + case success + case error + case decision + } + + // MARK: - Outlets + @IBOutlet private weak var dimView: UIView! + + @IBOutlet private weak var alertView: UIView! + @IBOutlet private weak var alertView_height: NSLayoutConstraint! + + @IBOutlet private weak var confirmationButton: UIButton! + @IBOutlet private weak var destructiveButton: UIButton! + + @IBOutlet private weak var buttonsView_height: NSLayoutConstraint! + + @IBOutlet private weak var messageTextView: UITextView! + @IBOutlet private weak var buttonStackView: UIStackView! + @IBOutlet private weak var buttonView_leading: NSLayoutConstraint! + @IBOutlet private weak var buttonsView_trailing: NSLayoutConstraint! + + // MARK: - Properties + + private let dimViewColor: UIColor = UIColor.purple.withAlphaComponent(0.6) + private var type: KNAlertViewType = .none + private var content_height: CGFloat = 150 + + private var message: String = "" + + private var confirmButtonTitle: String = "yes" + private var destructiveButtonTitle: String = "no" + + var shouldDismissAndLogin: Bool = false + + var confirmClosure: AlertViewClosure? + var destructiveClosure: AlertViewClosure? + + private let BUTTON_HEIGHT: CGFloat = 52 + private let BUTTON_WIDTH: CGFloat = 128 + + private var buttonsView_Spacing: CGFloat = 26 + + // MARK: - Init + init(type: KNAlertViewType, message: String) { + super.init() + + self.type = type + self.message = message + } + + init(type: KNAlertViewType, message: String, confirm: String, destructive: String) { + super.init() + + self.type = type + self.message = message + + confirmButtonTitle = confirm + destructiveButtonTitle = destructive + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + // MARK: - Lifecycle + override func viewDidLoad() { + super.viewDidLoad() + + view.endEditing(true) + + setupUI() + updateUI() + updateHeight() + } + + override func viewWillAppear(_ animated: Bool) { + super.viewWillAppear(animated) + + showPopup() + } + + override func viewDidAppear(_ animated: Bool) { + super.viewDidAppear(animated) + + } +} + +// MARK: - UI +extension KNAlertViewController { + private func setupUI() { + view.backgroundColor = .clear + dimView.backgroundColor = dimViewColor + + // alert + alertView.addBorder(radius: 16, width: 0, color: .white) + + // text + messageTextView.textAlignment = .center + messageTextView.textColor = .purple + messageTextView.font = .medium(16) + messageTextView.text = "" + messageTextView.isEditable = false + messageTextView.isSelectable = false + messageTextView.textContainerInset = .zero + messageTextView.textContainer.lineFragmentPadding = 0 + + // buttons stack + buttonStackView.spacing = 27 + buttonStackView.backgroundColor = .clear + + // buttons + confirmationButton.set(localized: confirmButtonTitle, color: .white, font: .semibold(17)) + confirmationButton.addCorners(20) + confirmationButton.backgroundColor = .purple + + confirmationButton.addTarget(self, action: #selector(confirmationButtonTapped), for: .touchUpInside) + + destructiveButton.set(localized: destructiveButtonTitle, color: .purple, font: .semibold(17)) + destructiveButton.addCorners(20) + destructiveButton.backgroundColor = .white + + destructiveButton.addTarget(self, action: #selector(destructiveButtonTapped), for: .touchUpInside) + + buttonView_leading.set(buttonsView_Spacing) + buttonsView_trailing.set(buttonsView_Spacing) + + destructiveButton.hide() + + confirmationButton.hide() + + buttonsView_height.set(0) + } + + private func updateHeight() { + var height: CGFloat = 0 + + height += 28 + height += 20 + height += 28 + height += buttonsView_height.constant + //@TODO @LOW dynamic height? using a tableView? + let message_height = messageTextView.height() + + if message_height > 500 { + height += 500 + messageTextView.isScrollEnabled = true + } else { + height += message_height + messageTextView.isScrollEnabled = false + } + + if height < 130 { + content_height = 130 + } else { + content_height = height + 20 + } + } + + private func updateUI() { + messageTextView.text = message + + destructiveButton.hide() + + confirmationButton.hide() + + buttonsView_height.set(0) + + switch type { + case .success, .error: + let alert_width = alertView.frame.width + + buttonsView_Spacing = (alert_width - BUTTON_WIDTH) / 2 + + buttonView_leading.set(buttonsView_Spacing) + buttonsView_trailing.set(buttonsView_Spacing) + + confirmationButton.show() + confirmationButton.set(title: "Ok") + buttonsView_height.set(BUTTON_HEIGHT) + + case .decision: + buttonsView_Spacing = 26 + + buttonView_leading.set(buttonsView_Spacing) + buttonsView_trailing.set(buttonsView_Spacing) + + destructiveButton.show() + + confirmationButton.show() + + confirmationButton.set(title: "Yes") + + buttonsView_height.set(BUTTON_HEIGHT) + + if !destructiveButtonTitle.isEmpty && !confirmButtonTitle.isEmpty { + confirmationButton.set(localized: confirmButtonTitle, color: .white, font: .semibold(17)) + confirmationButton.addCorners(20) + + confirmationButton.backgroundColor = .purple + + destructiveButton.set(title: destructiveButtonTitle, color: .purple, font: .semibold(17)) + destructiveButton.addCorners(20) + destructiveButton.backgroundColor = .white + } + default: + break + } + } + + private func showPopup() { + var transform = CGAffineTransform.identity + transform = transform.translatedBy(x: 0, y: UIScreen.main.bounds.height) + transform = transform.scaledBy(x: 0, y: 0) + alertView.transform = transform + + dimView.alpha = 0 + UIView.animate(withDuration: 0.3) { [weak self] in + self?.alertView_height.constant = (self?.content_height ?? 0) + self?.alertView.transform = .identity + self?.dimView.alpha = 1.0 + + self?.view.layoutIfNeeded() + } + } + + @objc private func dismissPopup() { + UIView.animate(withDuration: 0.3) { [weak self] in + var transform = CGAffineTransform.identity + transform = transform.translatedBy(x: 0, y: UIScreen.main.bounds.height) + transform = transform.scaledBy(x: 0, y: 0) + self?.alertView.transform = transform + + self?.dimView.alpha = 0 + self?.view.layoutIfNeeded() + } completion: { [weak self] finished in + if finished { + self?.dismiss(animated: false) { + // + } + } + } + } +} + +// MARK: - Delegate +extension KNAlertViewController { + @objc func destructiveButtonTapped() { + if let closure = destructiveClosure { + closure() + } else { + dismissPopup() + } + } + + @objc func confirmationButtonTapped() { + if let closure = confirmClosure { + closure() + } else { + dismissPopup() + } + } + + func hideEverything() { + view.hide() + } +} + diff --git a/TraccarClient/KNAlert/KNAlertViewController.xib b/TraccarClient/KNAlert/KNAlertViewController.xib new file mode 100644 index 0000000000000000000000000000000000000000..2571e47d9fc5f868cea9f346566299da3afa5033 --- /dev/null +++ b/TraccarClient/KNAlert/KNAlertViewController.xib @@ -0,0 +1,111 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="21507" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES"> + <device id="retina6_1" orientation="portrait" appearance="light"/> + <dependencies> + <deployment identifier="iOS"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="21505"/> + <capability name="Safe area layout guides" minToolsVersion="9.0"/> + <capability name="System colors in document resources" minToolsVersion="11.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <objects> + <placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner" customClass="KNAlertViewController" customModule="opa" customModuleProvider="target"> + <connections> + <outlet property="alertView" destination="HEa-7L-mBd" id="MYI-il-dxf"/> + <outlet property="alertView_height" destination="bda-Ts-u7O" id="0oR-a5-EyR"/> + <outlet property="buttonStackView" destination="mFa-hh-JzT" id="05H-L9-JXx"/> + <outlet property="buttonView_leading" destination="GIS-Wm-udf" id="2DG-Cf-wnA"/> + <outlet property="buttonsView_height" destination="H5D-nn-1fO" id="BRt-Rc-rfw"/> + <outlet property="buttonsView_trailing" destination="vzh-SN-Ztf" id="Brk-ob-y0k"/> + <outlet property="confirmationButton" destination="6RT-mM-MZO" id="DAx-Kd-q0W"/> + <outlet property="destructiveButton" destination="M8l-cs-Sk1" id="RMQ-kp-MLt"/> + <outlet property="dimView" destination="Ke8-mm-O5l" id="m3R-JB-7UI"/> + <outlet property="messageTextView" destination="KRY-Mp-Cgi" id="PrE-JR-vbV"/> + <outlet property="view" destination="i5M-Pr-FkT" id="sfx-zR-JGt"/> + </connections> + </placeholder> + <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> + <view clearsContextBeforeDrawing="NO" contentMode="scaleToFill" id="i5M-Pr-FkT"> + <rect key="frame" x="0.0" y="0.0" width="414" height="896"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="Ke8-mm-O5l" userLabel="dimView"> + <rect key="frame" x="0.0" y="0.0" width="414" height="896"/> + <color key="backgroundColor" white="0.0" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + </view> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="HEa-7L-mBd" userLabel="alertView"> + <rect key="frame" x="16" y="330" width="382" height="250"/> + <subviews> + <textView clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="scaleToFill" text=" " textAlignment="natural" translatesAutoresizingMaskIntoConstraints="NO" id="KRY-Mp-Cgi"> + <rect key="frame" x="28" y="28" width="326" height="122"/> + <color key="textColor" systemColor="labelColor"/> + <fontDescription key="fontDescription" type="system" pointSize="14"/> + <textInputTraits key="textInputTraits" autocapitalizationType="sentences"/> + </textView> + <view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="0Iw-HP-6vl" userLabel="buttonsView"> + <rect key="frame" x="26" y="170" width="330" height="52"/> + <subviews> + <stackView opaque="NO" contentMode="scaleToFill" distribution="fillEqually" translatesAutoresizingMaskIntoConstraints="NO" id="mFa-hh-JzT"> + <rect key="frame" x="0.0" y="0.0" width="330" height="52"/> + <subviews> + <button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="M8l-cs-Sk1" userLabel="destructiveButton"> + <rect key="frame" x="0.0" y="0.0" width="165" height="52"/> + <state key="normal" title=" "/> + </button> + <button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="6RT-mM-MZO" userLabel="confirmButton"> + <rect key="frame" x="165" y="0.0" width="165" height="52"/> + <constraints> + <constraint firstAttribute="height" constant="366" id="cUE-2O-C4X"/> + </constraints> + <state key="normal" title=" "/> + </button> + </subviews> + </stackView> + </subviews> + <color key="backgroundColor" systemColor="systemBackgroundColor"/> + <constraints> + <constraint firstItem="mFa-hh-JzT" firstAttribute="top" secondItem="0Iw-HP-6vl" secondAttribute="top" id="02x-AD-Zhw"/> + <constraint firstItem="mFa-hh-JzT" firstAttribute="leading" secondItem="0Iw-HP-6vl" secondAttribute="leading" id="DR3-dd-o3o"/> + <constraint firstAttribute="trailing" secondItem="mFa-hh-JzT" secondAttribute="trailing" id="EvO-ip-xRD"/> + <constraint firstAttribute="height" constant="52" id="H5D-nn-1fO"/> + <constraint firstAttribute="bottom" secondItem="mFa-hh-JzT" secondAttribute="bottom" id="heB-PX-nNZ"/> + </constraints> + </view> + </subviews> + <color key="backgroundColor" systemColor="systemBackgroundColor"/> + <constraints> + <constraint firstItem="KRY-Mp-Cgi" firstAttribute="leading" secondItem="HEa-7L-mBd" secondAttribute="leading" constant="28" id="7se-55-EI4"/> + <constraint firstItem="KRY-Mp-Cgi" firstAttribute="top" secondItem="HEa-7L-mBd" secondAttribute="top" constant="28" id="Fsl-r8-vIk"/> + <constraint firstItem="0Iw-HP-6vl" firstAttribute="leading" secondItem="HEa-7L-mBd" secondAttribute="leading" constant="26" id="GIS-Wm-udf"/> + <constraint firstAttribute="trailing" secondItem="KRY-Mp-Cgi" secondAttribute="trailing" constant="28" id="Kjz-ul-BMl"/> + <constraint firstAttribute="height" constant="250" id="bda-Ts-u7O"/> + <constraint firstItem="0Iw-HP-6vl" firstAttribute="top" secondItem="KRY-Mp-Cgi" secondAttribute="bottom" constant="20" id="uzr-4g-H6v"/> + <constraint firstAttribute="trailing" secondItem="0Iw-HP-6vl" secondAttribute="trailing" constant="26" id="vzh-SN-Ztf"/> + <constraint firstAttribute="bottom" secondItem="0Iw-HP-6vl" secondAttribute="bottom" constant="28" id="zIz-7A-Fhl"/> + </constraints> + </view> + </subviews> + <viewLayoutGuide key="safeArea" id="fnl-2z-Ty3"/> + <color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + <constraints> + <constraint firstItem="fnl-2z-Ty3" firstAttribute="leading" secondItem="HEa-7L-mBd" secondAttribute="leading" constant="-16" id="J59-Wx-NLp"/> + <constraint firstItem="Ke8-mm-O5l" firstAttribute="bottom" secondItem="i5M-Pr-FkT" secondAttribute="bottom" id="OO0-29-GbT"/> + <constraint firstAttribute="trailing" secondItem="Ke8-mm-O5l" secondAttribute="trailing" id="XEW-zk-P9r"/> + <constraint firstItem="Ke8-mm-O5l" firstAttribute="top" secondItem="i5M-Pr-FkT" secondAttribute="top" id="gnh-Wf-UUa"/> + <constraint firstAttribute="leading" secondItem="Ke8-mm-O5l" secondAttribute="leading" id="jUn-qr-3a3"/> + <constraint firstItem="HEa-7L-mBd" firstAttribute="centerY" secondItem="fnl-2z-Ty3" secondAttribute="centerY" id="l3t-nt-x4P"/> + <constraint firstItem="fnl-2z-Ty3" firstAttribute="trailing" secondItem="HEa-7L-mBd" secondAttribute="trailing" constant="16" id="wTo-Vm-Gmy"/> + </constraints> + <freeformSimulatedSizeMetrics key="simulatedDestinationMetrics"/> + <point key="canvasLocation" x="131.8840579710145" y="131.91964285714286"/> + </view> + </objects> + <resources> + <systemColor name="labelColor"> + <color red="0.0" green="0.0" blue="0.0" alpha="1" colorSpace="custom" customColorSpace="sRGB"/> + </systemColor> + <systemColor name="systemBackgroundColor"> + <color white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/> + </systemColor> + </resources> +</document> diff --git a/TraccarClient/KNNavigationController.swift b/TraccarClient/KNNavigationController.swift new file mode 100644 index 0000000000000000000000000000000000000000..49b277a8fbb290fa542df87706b0503db3f9a5a0 --- /dev/null +++ b/TraccarClient/KNNavigationController.swift @@ -0,0 +1,57 @@ +// +// KNNavigationController.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +final class KNNavigationController: UINavigationController { + + // MARK: - Properties + let textAttributes = [ + NSAttributedString.Key.foregroundColor: UIColor.black, + NSAttributedString.Key.font: UIFont.medium(17) + ] + + // MARK: - Lifecycle + override func viewDidLoad() { + super.viewDidLoad() + + setupUI() + } + + // MARK: - UI + private func setupUI() { + navigationBar.setBackgroundImage(UIImage(), for: .default) + navigationBar.shadowImage = UIImage() + navigationBar.isTranslucent = false + navigationBar.barTintColor = .white + navigationBar.titleTextAttributes = textAttributes + + if #available(iOS 13.0, *) { + let appearance = UINavigationBarAppearance() + appearance.configureWithOpaqueBackground() + appearance.backgroundColor = .white + appearance.shadowColor = .clear + appearance.titleTextAttributes = textAttributes + + navigationBar.standardAppearance = appearance + navigationBar.scrollEdgeAppearance = appearance + } + } + + private func setupStatusBar() { + let window = UIApplication.shared.windows.filter {$0.isKeyWindow}.first + if let statusBarFrame = window?.windowScene?.statusBarManager?.statusBarFrame { + let statusBarView = UIView(frame: statusBarFrame) + statusBarView.backgroundColor = .clear + + view.addSubview(statusBarView) + } + } +} + + diff --git a/TraccarClient/KNViewController.swift b/TraccarClient/KNViewController.swift new file mode 100644 index 0000000000000000000000000000000000000000..e6d629f2f3f9832a5f9e9953fbb0d1d3d6f2d266 --- /dev/null +++ b/TraccarClient/KNViewController.swift @@ -0,0 +1,148 @@ +// +// KNViewController.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +class KNViewController: UIViewController, Nameable { + + // MARK: - Properties + + var screenWidth: CGFloat { + view.frame.width + } + + var isPresented: Bool { + if let navigationController = navigationController { + return navigationController.viewControllers.firstIndex(of: self) == 0 + } else { + return presentingViewController != nil + } + } + + var isNavigationBarHidden: Bool { + navigationController?.navigationBar.isHidden ?? false + } + + var isBackButtonHidden: Bool = false { + didSet { + navigationItem.hidesBackButton = isBackButtonHidden + } + } + + var hasCustomNavigation: Bool = false { + didSet { + navigationController?.setNavigationBarHidden(hasCustomNavigation, animated: true) + } + } + + + + var isTabbarHidden: Bool = false { + didSet { + tabBarController?.tabBar.isHidden = isTabbarHidden + } + } + + var statusBarStyle: UIStatusBarStyle = .default // default is black + + // MARK: - Init + init() { + let nibName = String(describing: type(of: self)) + super.init(nibName: nibName, bundle: nil) + } + + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + // MARK: - Lifecycle + deinit { + log("💥", Self.name) + } + + override func viewDidLoad() { + super.viewDidLoad() + // + } + + override func viewWillAppear(_ animated: Bool) { + super.viewWillAppear(animated) + setNeedsStatusBarAppearanceUpdate() + } + + override var preferredStatusBarStyle: UIStatusBarStyle { + statusBarStyle + } +} + +// MARK: - Spinner +extension KNViewController { + fileprivate func centerPoint() -> CGPoint { + let x: CGFloat = (UIScreen.main.bounds.width / 2) + var y: CGFloat = (UIScreen.main.bounds.height / 2) + + return CGPoint(x: x, y: y) + } + + fileprivate func spinnerPosition() -> CGRect { + let width: CGFloat = 100 + let height: CGFloat = 100 + + let point = centerPoint() + + let frame = CGRect(x: point.x, y: point.y - (height / 2), width: width, height: height) + + return frame + } + + @discardableResult + func spin() -> KNSpinnerView { + let spinner = KNSpinnerView(frame: spinnerPosition()) + spinner.backgroundColor = UIColor.clear + spinner.alpha = 0 + + view.addSubview(spinner) + view.bringSubviewToFront(spinner) + spinner.center = CGPoint(x: centerPoint().x, y: centerPoint().y) + + UIView.animate(withDuration: 0.3) { + spinner.alpha = 1 + } + + return spinner + } + + func customSpin() -> KNSpinnerView { + let spinner = KNSpinnerView(frame: spinnerPosition()) + spinner.backgroundColor = .clear + spinner.alpha = 0 + + return spinner + } + + func addSpinner(spinner: KNSpinnerView) { + view.addSubview(spinner) + view.bringSubviewToFront(spinner) + spinner.center = CGPoint(x: centerPoint().x, y: centerPoint().y) + + UIView.animate(withDuration: 0.3) { + spinner.alpha = 1 + } + } + + func addSpinnerToView(spinner: KNSpinnerView, displayView: UIView) { + displayView.addSubview(spinner) + displayView.bringSubviewToFront(spinner) + spinner.center = CGPoint(x: centerPoint().x, y: centerPoint().y) + + UIView.animate(withDuration: 0.3) { + spinner.alpha = 1 + } + } +} + diff --git a/TraccarClient/Keys.swift b/TraccarClient/Keys.swift index f8cf3dbe4f1bd70a3448521427f80df9b7eb847c..85b21aa9cc28dca855df4bd63f5ad4af3c31b14d 100644 --- a/TraccarClient/Keys.swift +++ b/TraccarClient/Keys.swift @@ -36,6 +36,16 @@ struct Keys { static let phone = tag + "_phone" static let prefix = tag + "_prefix" static let account_type = tag + "_account_type" + static let avatar = tag + "_avatar" + static let cover = tag + "_cover" + static let designation_id = tag + "_designation_id" + static let name = tag + "_name" + static let user = tag + "_user" + static let uniqueId = tag + "_uniqueId" + static let status = tag + "_status" + static let deviceName = tag + "_device_name" + static let device = tag + "_device" + } struct Badge { diff --git a/TraccarClient/LaunchScreen.xib b/TraccarClient/LaunchScreen.xib index 9922a47edd9230eabbdedfede3fbf1537dfcfc5c..e7414a655f978524e3c56993d24188e772fba49a 100644 --- a/TraccarClient/LaunchScreen.xib +++ b/TraccarClient/LaunchScreen.xib @@ -1,9 +1,9 @@ <?xml version="1.0" encoding="UTF-8"?> -<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="15505" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" colorMatched="YES"> +<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="22155" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" colorMatched="YES"> <device id="retina4_7" orientation="portrait" appearance="light"/> <dependencies> <deployment identifier="iOS"/> - <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="15510"/> + <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="22131"/> <capability name="Named colors" minToolsVersion="9.0"/> <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> </dependencies> @@ -21,19 +21,32 @@ </constraints> <color key="barTintColor" name="Brand"/> </navigationBar> + <imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFit" horizontalHuggingPriority="251" verticalHuggingPriority="251" image="DeliveryIcon" translatesAutoresizingMaskIntoConstraints="NO" id="chw-lY-qlJ"> + <rect key="frame" x="67.5" y="213.5" width="240" height="240"/> + <constraints> + <constraint firstAttribute="height" constant="240" id="Rtp-MQ-ir8"/> + <constraint firstAttribute="width" constant="240" id="ySq-lc-u6B"/> + </constraints> + </imageView> </subviews> - <color key="backgroundColor" cocoaTouchSystemColor="groupTableViewBackgroundColor"/> + <color key="backgroundColor" systemColor="groupTableViewBackgroundColor"/> <constraints> <constraint firstAttribute="trailing" secondItem="jyO-lQ-xCZ" secondAttribute="trailing" id="BVV-Jp-FLs"/> <constraint firstItem="jyO-lQ-xCZ" firstAttribute="top" secondItem="iN0-l3-epB" secondAttribute="top" id="Ra4-bu-wwa"/> <constraint firstItem="jyO-lQ-xCZ" firstAttribute="leading" secondItem="iN0-l3-epB" secondAttribute="leading" id="frB-PI-Vg3"/> + <constraint firstItem="chw-lY-qlJ" firstAttribute="centerY" secondItem="iN0-l3-epB" secondAttribute="centerY" id="oZD-Ui-4L5"/> + <constraint firstItem="chw-lY-qlJ" firstAttribute="centerX" secondItem="iN0-l3-epB" secondAttribute="centerX" id="ypL-5B-XCA"/> </constraints> <point key="canvasLocation" x="499" y="433"/> </view> </objects> <resources> + <image name="DeliveryIcon" width="160" height="160"/> <namedColor name="Brand"> - <color red="0.2627450980392157" green="0.62745098039215685" blue="0.27843137254901962" alpha="1" colorSpace="custom" customColorSpace="sRGB"/> + <color red="0.1803921568627451" green="0.49019607843137253" blue="0.19607843137254902" alpha="1" colorSpace="custom" customColorSpace="sRGB"/> </namedColor> + <systemColor name="groupTableViewBackgroundColor"> + <color red="0.94901960784313721" green="0.94901960784313721" blue="0.96862745098039216" alpha="1" colorSpace="custom" customColorSpace="sRGB"/> + </systemColor> </resources> </document> diff --git a/TraccarClient/LoginViewController/Controller/LoginViewController.swift b/TraccarClient/LoginViewController/Controller/LoginViewController.swift index 00085c326ed36f200cf5a4b5bec9dd524f823feb..c1408df3ad49d61dc41da2f0fa3f593e97757a81 100644 --- a/TraccarClient/LoginViewController/Controller/LoginViewController.swift +++ b/TraccarClient/LoginViewController/Controller/LoginViewController.swift @@ -8,12 +8,13 @@ import UIKit -final class LoginViewController: UIViewController { +final class LoginViewController: KNViewController { // MARK: - Outlets @IBOutlet private weak var tableView: UITableView! // MARK: - Properties private weak var delegate: LoginViewControllerDelegate? + var coordinator: AuthenticationCoordinator? private var modelDataSource: LoginDataSource? private var modelController: LoginModelController? @@ -29,6 +30,7 @@ final class LoginViewController: UIViewController { // MARK: - UI extension LoginViewController { private func setupUI() { + isBackButtonHidden = true setupTableView() } @@ -39,12 +41,13 @@ extension LoginViewController { modelDataSource?.outputProtocol = modelController modelController?.inputProtocol = modelDataSource + modelController?.coordinator = self.coordinator + tableView.delegate = modelDataSource tableView.dataSource = modelDataSource tableView.keyboardDismissMode = .interactive - tableView.set(cells: [FieldTableViewCell.self]) - tableView.isScrollEnabled = false + tableView.set(cells: [FieldTableViewCell.self, ButtonTableViewCell.self]) tableView.reloadData() } diff --git a/TraccarClient/LoginViewController/Model/AuthenticationCoordinator.swift b/TraccarClient/LoginViewController/Model/AuthenticationCoordinator.swift new file mode 100644 index 0000000000000000000000000000000000000000..df34724dd8c96f5d432f4b92ee9f9949c60405dc --- /dev/null +++ b/TraccarClient/LoginViewController/Model/AuthenticationCoordinator.swift @@ -0,0 +1,75 @@ +// +// AuthenticationCoordinator.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +import UIKit + +final class AuthenticationCoordinator: Coordinatable { + // MARK: - Properties + var navigationController: UINavigationController + var step: AuthenticationStep? + + // MARK: - Init + init(navigationController: UINavigationController) { + self.navigationController = navigationController + } + + // MARK: - Next/Previous Step + func next() { + guard let step = step else { return } + + switch step { + case .signin: + signin() + } + } + + // MARK: - Start + func start() { + guard let step = step else { return } + + if step != .signin { + self.signin() + } else { + self.continue() + } + } + + func finish() { + navigationController.popToRootViewController(animated: true) + } + + func previous() { + navigationController.popViewController(animated: true) + } + + func cancel() { + navigationController.popToRootViewController(animated: true) + } + + func `continue`() { + guard let step = step else { return } + + switch step { + case .signin: + let signin = LoginViewController() + signin.coordinator = self + + navigationController.push(viewControllers: [signin], animated: true) + + } + } + + // MARK: - signin + private func signin() { + let signinViewController = LoginViewController() + signinViewController.coordinator = self + navigationController.pushViewController(signinViewController, animated: true) + } +} + diff --git a/TraccarClient/LoginViewController/Model/AuthenticationStep.swift b/TraccarClient/LoginViewController/Model/AuthenticationStep.swift new file mode 100644 index 0000000000000000000000000000000000000000..d98cd7b69d146f907d87e04a872c704f82a3afc7 --- /dev/null +++ b/TraccarClient/LoginViewController/Model/AuthenticationStep.swift @@ -0,0 +1,13 @@ +// +// AuthenticationStep.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation + +enum AuthenticationStep: String { + case signin +} diff --git a/TraccarClient/LoginViewController/Model/LoginDataProvider.swift b/TraccarClient/LoginViewController/Model/LoginDataProvider.swift index b963f64854c4608b0f34196b5701fb81fda29a97..f074def9715fc3086a39c2973476943d475596f6 100644 --- a/TraccarClient/LoginViewController/Model/LoginDataProvider.swift +++ b/TraccarClient/LoginViewController/Model/LoginDataProvider.swift @@ -15,23 +15,42 @@ final class LoginDataProvider { // MARK: - Methods func signinData() -> [KNField] { + let domain = KNField { field in + field.field = .domain + field.isRequired = true + field.type = .text + } + let email = KNField { field in field.field = .username - field.title = "Username" field.isRequired = true field.type = .text } let password = KNField { field in field.field = .password - field.title = "Password" field.isRequired = true field.type = .text } + + let space = KNField { field in + field.type = .empty + field.height = 30 + } + + let loginButton = KNField { field in + field.type = .button + field.title = "Login" + field.action = .tap + field.height = 51 + } fields = [ + domain, email, password, + space, + loginButton ] return fields } diff --git a/TraccarClient/LoginViewController/Model/LoginDataSource.swift b/TraccarClient/LoginViewController/Model/LoginDataSource.swift index e043b12cc18dc16af8d97b7097682dda3bd5de2f..d82dbbd59cc6d45ab7b9ab94ed72a5f2db19fe5a 100644 --- a/TraccarClient/LoginViewController/Model/LoginDataSource.swift +++ b/TraccarClient/LoginViewController/Model/LoginDataSource.swift @@ -37,6 +37,12 @@ extension LoginDataSource: UITableViewDelegate, UITableViewDataSource { cell.set(field: field, indexPath: indexPath) return cell } + if field.type == .button { + let cell = tableView.dequeue(withCell: ButtonTableViewCell.self) + cell.delegate = self + cell.set(field: field, indexPath: indexPath) + return cell + } return UITableViewCell() } @@ -63,3 +69,9 @@ extension LoginDataSource: KNTextFieldViewDelegate { } } +// MARK: - Button Delegate +extension LoginDataSource: ButtonTableViewCellDelegate { + func cellButtonTapped(action: KNButtonViewAction) { + outputProtocol?.cellButtonTapped(action: action) + } +} diff --git a/TraccarClient/LoginViewController/Model/LoginModelController.swift b/TraccarClient/LoginViewController/Model/LoginModelController.swift index 91f99e6395476d50715cc865fc2e4bc2028a4607..e63bc8226dbaa75e64ca5d3d2c12fcd70b5a24ff 100644 --- a/TraccarClient/LoginViewController/Model/LoginModelController.swift +++ b/TraccarClient/LoginViewController/Model/LoginModelController.swift @@ -8,14 +8,16 @@ import Foundation import UIKit +import Alamofire +import CoreData final class LoginModelController { // MARK: - Properties var fields: [KNField] = [] - var viewController: UIViewController + var viewController: KNViewController private weak var delegate: LoginViewControllerDelegate? weak var window: UIWindow? - + weak var coordinator: AuthenticationCoordinator? weak var inputProtocol: LoginInputProtocol? { didSet { setupData() @@ -23,7 +25,7 @@ final class LoginModelController { } // MARK: - Init - init(delegate: LoginViewControllerDelegate?, viewController: UIViewController) { + init(delegate: LoginViewControllerDelegate?, viewController: KNViewController) { self.delegate = delegate self.viewController = viewController } @@ -34,13 +36,114 @@ final class LoginModelController { inputProtocol?.setFields(fields: fields) } + + private func setupTracking() { + UIDevice.current.isBatteryMonitoringEnabled = true + + let userDefaults = UserDefaults.standard + if userDefaults.string(forKey: "device_id_preference") == nil { + let identifier = "\(Int.random(in: 100000..<1000000))" + userDefaults.setValue(identifier, forKey: "device_id_preference") + } + + AppDelegate.instance.registerDefaultsFromSettingsBundle() + + AppDelegate.instance.migrateLegacyDefaults() + + let modelUrl = Bundle.main.url(forResource: "TraccarClient", withExtension: "momd") + AppDelegate.instance.managedObjectModel = NSManagedObjectModel(contentsOf: modelUrl!) + + AppDelegate.instance.persistentStoreCoordinator = NSPersistentStoreCoordinator(managedObjectModel: AppDelegate.instance.managedObjectModel!) + let storeUrl = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).last?.appendingPathComponent("TraccarClient.sqlite") + let options = [NSMigratePersistentStoresAutomaticallyOption: true, NSInferMappingModelAutomaticallyOption: true] + try! AppDelegate.instance.persistentStoreCoordinator?.addPersistentStore(ofType: NSSQLiteStoreType, configurationName: nil, at: storeUrl, options: options) + + AppDelegate.instance.managedObjectContext = NSManagedObjectContext.init(concurrencyType: .mainQueueConcurrencyType) + AppDelegate.instance.managedObjectContext?.persistentStoreCoordinator = AppDelegate.instance.persistentStoreCoordinator + } } +// MARK: - Request +extension LoginModelController { + func signin() { + var params: PARAMS = [:] + + for item in fields { + if item.type == .text { + params.add(item.field.rawValue, item.value.value) + } + } + + let spinner = viewController.spin() + api.signin(params: params) { [weak self] result in + spinner.stop() + + guard let strongSelf = self else {return} + + if result.success { +// strongSelf.setupTracking() + let storyboard = UIStoryboard(name: "MainStoryboard", bundle: nil) + if let mainViewController = storyboard.instantiateViewController(withIdentifier: "MainViewController") as? MainViewController { + strongSelf.viewController.navigationController?.pushViewController(mainViewController, animated: true) + } + + + } else { + if result.statusCode == 401 { + do { + let value = try JSONSerialization.jsonObject(with: result.dataResponse?.data ?? Data()) + if let dict = value as? JSON { + if let display_errors = dict["display_errors"] as? [String] { + var displayError: String = "" + displayError = display_errors.joined(separator: "\n") + ok(displayError, viewController: self?.viewController) + } + } + + } catch { + print("Error while decoding response: \(String(data: result.dataResponse?.data ?? Data(), encoding: .utf8) ?? "")") + } + } else if result.statusCode == 422 { + var fieldErrors: JSON = [:] + var errors: String = "" + do{ + let value = try JSONSerialization.jsonObject(with: result.dataResponse?.data ?? Data()) + if let dict = value as? JSON { + if let field_errors = dict["errors"] as? JSON { + fieldErrors = field_errors + strongSelf.fields.forEach { field in + if let value = fieldErrors[field.field] as? [String] { + value.forEach { string in + errors = value.joined(separator: "\n") + field.error.value = errors + } + } + } + } + } + }catch { + print("Error while decoding response: \(String(data: result.dataResponse?.data ?? Data(), encoding: .utf8) ?? "")") + } + } + + else { + ok(result) + } + } + } + } +} + + // MARK: - Output Protocol extension LoginModelController: LoginOutputProtocol { + func cellButtonTapped(action: KNButtonViewAction) { + signin() + } func reloadUI() { delegate?.updateUI() } } + diff --git a/TraccarClient/LoginViewController/Model/LoginViewControllerProtocol.swift b/TraccarClient/LoginViewController/Model/LoginViewControllerProtocol.swift index d88270b1a235436092c2e53abc5dbc9b3accd1a5..f1015376bfd0fa9e2ed0f2583a3b0df44aae1531 100644 --- a/TraccarClient/LoginViewController/Model/LoginViewControllerProtocol.swift +++ b/TraccarClient/LoginViewController/Model/LoginViewControllerProtocol.swift @@ -20,4 +20,5 @@ protocol LoginInputProtocol: AnyObject { protocol LoginOutputProtocol: AnyObject { func reloadUI() + func cellButtonTapped(action: KNButtonViewAction) } diff --git a/TraccarClient/LoginViewController/View/LoginViewController.xib b/TraccarClient/LoginViewController/View/LoginViewController.xib index df509156c5949ee86121c7eed68324f7089da194..af884bc41a2f44d3b13d1d22a249ff19d4867635 100644 --- a/TraccarClient/LoginViewController/View/LoginViewController.xib +++ b/TraccarClient/LoginViewController/View/LoginViewController.xib @@ -24,7 +24,7 @@ </connections> </placeholder> <placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/> - <view clearsContextBeforeDrawing="NO" contentMode="scaleToFill" id="i5M-Pr-FkT"> + <view clearsContextBeforeDrawing="NO" contentMode="scaleToFill" restorationIdentifier="LoginViewController" id="i5M-Pr-FkT"> <rect key="frame" x="0.0" y="0.0" width="393" height="852"/> <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> <subviews> diff --git a/TraccarClient/Network+Endpoint.swift b/TraccarClient/Network+Endpoint.swift index 740eb764809b58dc8ad5a63e2c2437efdfac91d4..1a5cb621f0e05929e62e865e7852685e4203280a 100644 --- a/TraccarClient/Network+Endpoint.swift +++ b/TraccarClient/Network+Endpoint.swift @@ -13,191 +13,23 @@ enum Endpoint: String, CaseIterable { static var baseURL: String { switch app.environment { case .development, .sandbox: - return "\(serverURL)/api/v1/" + return "\(serverURL)/api/" case .production: - return "\(serverURL)/api/v1/" + return "\(serverURL)/api/" } } static var serverURL: String { switch app.environment { case .development, .sandbox: - return "https://traccaravenue.mykuwaitnet.net" + return "https://dev-nc-teams-api.nmo.ai" case .production: - return "https://www.traccaravenue.com" + return "https://dev-nc-teams-api.nmo.ai" } } case all - case signin = "account/login/" - case refreshToken = "account/token/refresh/" - case generateGuestToken = "generate-guest-token/" - case forgotPassword = "account/login/forgot-password/" - - case receiveCode = "account/login/forgot-password/receive-code/" - case verifyCode = "account/login/forgot-password/verify-code/" - case resetPassword = "account/login/forgot-password/reset-password/" - - case signup = "account/sign-up/user/" - case companySignup = "account/sign-up/company/" - - case signupPassword = "account/sign-up/user/password/" - case companySignupPassword = "account/sign-up/company/password/" - - case signupVerifyCode = "account/sign-up/user/verify/" - case companySignupVerifyCode = "account/sign-up/company/verify/" - case updateAccountType = "account/sign-up/set_account_type/" - case resendCode = "account/sign-up/resend-code/" - case contactInfo = "settings/contact-info/" - case changePassword = "settings/change-password/" - - case language = "account/language/" - case currency = "settings/currency/" - case about = "settings/about/" - case terms = "settings/terms-and-conditions/" - case privacy = "settings/privacy-policy/" - case settingsOptions = "settings/my-settings/" - case notificationsSettings = "notification/settings/" - case updateBadge = "account/me/badges_count/" - case updateBadgeGuest = "guest/" - - // ADDRESS - case addresses = "account/address/" - case deleteAddress = "account/address/%@/" - case editAddress = "account/address/%@/update_address/" - case searchAddress = "account/address/autocomplete_search/" - case getLocation = "account/address/place/%@/" - case set_location = "account/address/set-location/" - - // shippingWarehouse - case shippingWarehouse = "account/warehouse_address/my_warehouse/" - case editshippingWarehouse = "account/warehouse_address/%@/update_address/" - case searchshippingWarehouse = "account/warehouse_address/autocomplete_search/" - case getLocationShippingWarehouse = "account/warehouse_address/place/%@/" - case setlocationShippingWarehouse = "account/warehouse_address/set-location/" - - case bankAccount = "account/bank-account/" - case contactus = "settings/contact/" - - case documents = "account/documents/" - - // PRODUCT BUILDER - case product_builder = "product-builder/" - case product_builder_id = "product-builder/%@/" - case product_builder_new_id = "product-builder/new/%@/" - case product_builder_new = "product-builder/new/" - case product_builder_continue = "product-builder/continue/" - case product_builder_step_information = "product-builder/%@/information/" - case product_builder_step_information_media = "product-builder/%@/information/media/" - case product_builder_step_category = "product-builder/%@/category/" - case product_builder_step_attributes = "product-builder/%@/attributes/" - case product_builder_step_stock = "product-builder/%@/stock/" - case product_builder_step_set_for_all_variant = "product-builder/%@/set_for_all_variant/" - - case product_builder_generate_variants = "product-builder/%@/generate_variants/" - case product_builder_variant_detail = "product-builder/%@/variant_detail/%@/" - case product_builder_step_variants = "product-builder/%@/variants/" - case product_builder_step_variants_new = "product-builder/%@/variants/new/" - case product_builder_step_variants_id = "product-builder/%@/variants/%@/" - case product_builder_step_shipping = "product-builder/%@/shipping/" - case product_builder_step_shipping_add_location = "product-builder/%@/shipping/location/new/" - case product_builder_step_shipping_location_new = "%@/shipping/location/new/" - case product_builder_step_returns = "product-builder/%@/returns/" - case product_builder_cancel = "product-builder/cancel/" - case product_builder_single_variant = "product-builder/%@/add_single_variant/" - - // Media Upload - case upload_Media = "media-upload?field-name=web_media" - - // Product - case product = "product/" - case productLike = "product/%@/like/" - case productComments = "product/%@/comments/" - case deleteComment = "product/%@/comments/%@/" - case addToCart = "basket/add-product/" - case buyNow = "basket/buy_now/" - case productDetails = "product/%@/" - case productFilterOptions = "filter_options/" - - //Profile - case currentProfile = "account/me/" - case profile = "account/public/user/%@/" - case profileProduct = "product/compact_list/?partner_id=" - case recentlyViewedProduct = "product/my_recently_view_products/" - case vacationMode = "account/vacation_mode/" - case profileData = "account/update/" - case uploadProfilePicture = "media-upload?field-name=profile_image/" - case fetchFollowers = "account/followers/" - case fetchPublicFollowers = "account/public/user/%@/followers/" - case fetchFollowing = "account/following/" - case fetchPublicFollowing = "account/public/user/%@/following/" - case followUser = "account/follow/" - - case favorite = "favourites/" - case delete_favorite = "favourites/remove_lines/" - - case search = "search/" - case variant = "product/%@/variant/" - case categories = "category/" - case logout = "account/logout/" - - case activities = "account/me/activities/" - - // Sales Orders - case salesOrders = "trader_dashboard/received_orders/" - case salesOrderDetails = "trader_dashboard/received_orders/%@/" - case acceptOrder = "trader_dashboard/received_orders/%@/accept/" - case rejectOrder = "trader_dashboard/received_orders/%@/reject_order/" - case shippedOrder = "trader_dashboard/received_orders/%@/mark_shipped/" - case updateOrderStatus = "trader_dashboard/received_orders/%@/line_update/%@/" - - // Purchases Orders - case purchasesOrders = "orders/" - case purchasesOrderDetails = "orders/%@/" - - case markAllRead = "account/me/activities/mark-all-read/" -// case markActivityRead = "account/activities/mark-activity-read/" - case markActivityRead = "account/activity/%@/mark-read" - - case cart = "basket/" - case updateCart = "baskets/%@/lines/%@/" - - case checkout = "checkout/" - case checkoutDetails = "checkout/%@/" - - case notifications = "notification/my_notification/" - case markAllReadNotification = "notification/notification-mark-read/" - case markActivityReadNotification = "notification/notification-mark-read/?id=%@" - case markPushNotificationRead = "notification/notification-mark-read/?notification_group_id=%@" - case registerToken = "notification/device/apns/" - - // Checkout - case checkout_payment_methods = "checkout/%@/payment/" - case checkout_address = "checkout/%@/address/" - case checkout_shipping_methods = "checkout/%@/shipping_method/" - case checkout_add_coupon = "checkout/%@/add_coupon/" - case checkout_remove_coupon = "checkout/%@/remove_coupon/" - case retryPayment = "retry-payment" - - // Reports - case sales_summary = "summary_orders/" - case best_selling = "best_selling/" - case report_product = "report_product/" - - // Social login - case google = "login/google/" - case facebook = "login/facebook/" - case apple = "login/apple/" - - case delete_account = "account/me/delete_account/" - case static_pages = "settings/cms/pages/" - case static_page = "settings/cms/pages/%@/" - - case public_account_contact_details = "account/partner_contact_info/" - - // My Products - case myProducts = "trader_dashboard/my_products/" - case deleteProduct = "trader_dashboard/my_products/%@/" + case signin = "fleet-login" // /////////////// // get endpoint with resource diff --git a/TraccarClient/Network+Error.swift b/TraccarClient/Network+Error.swift index 3829600d418dc82a8ad31833b1868b00aafa900a..72823893f6e61f07cf7123471b345bb4af6fa2dc 100644 --- a/TraccarClient/Network+Error.swift +++ b/TraccarClient/Network+Error.swift @@ -8,7 +8,6 @@ import Foundation import UIKit import Alamofire - final class NetworkError: NSObject, Error { enum Typee { @@ -27,7 +26,7 @@ final class NetworkError: NSObject, Error { // MARK: - Properties private(set) var message: String = "" - private(set) var type: NetworkError.Typee = .none + var type: NetworkError.Typee = .none var error: AFError? diff --git a/TraccarClient/Network+Result.swift b/TraccarClient/Network+Result.swift index faf184c1aa2f70eb9dbea85c0784b18ea6e8688f..c2b95c4a395d589194eda1b6598a21cfd12d7a62 100644 --- a/TraccarClient/Network+Result.swift +++ b/TraccarClient/Network+Result.swift @@ -73,7 +73,7 @@ final class NetworkResultModel: KNObject { var appError: AppError? var errorType: ErrorType = .none - private var dataResponse: AFDataResponse<Data>? + var dataResponse: AFDataResponse<Data>? var success: Bool = false var data: Any? @@ -145,7 +145,7 @@ final class NetworkResultModel: KNObject { init(error: NetworkError) { self.error = error - displayError = error.message + displayError = "inavlid Credentials!" log() log("=========================") @@ -288,7 +288,7 @@ extension NetworkResultModel { if !appErrors.isEmpty { return } } - if let field_errors = dict["field_errors"] as? JSON { + if let field_errors = dict["errors"] as? JSON { fieldErrors = field_errors displayError = "" errorType = .field @@ -307,6 +307,7 @@ extension NetworkResultModel { success = false self.error = NetworkError(request: error) } + } func error(for field: JSONField) -> String { diff --git a/TraccarClient/NetworkRequest.swift b/TraccarClient/NetworkRequest.swift index a27b489b5f9ae0dcfc831c31654b059ef35239fc..3468138309c1df68cf4d1561e5f17bfbaecb122b 100644 --- a/TraccarClient/NetworkRequest.swift +++ b/TraccarClient/NetworkRequest.swift @@ -31,10 +31,9 @@ final class NetworkRequest: NSObject { var headers : HTTPHeaders { var _headers: HTTPHeaders = [:] _headers["Content-Type"] = "application/json" - -// _headers["Accept-Language"] = lang == .en ? "en" : "ar" _headers["Accept-Language"] = "en" _headers["X-App-Client"] = "ios" + _headers["x-Company"] = "dev-nextcloud.nmo.ai" return _headers } @@ -101,7 +100,6 @@ final class NetworkRequest: NSObject { completion(NetworkResultModel(dataResponse: response)) } } - func request(endpoint: URLConvertible, method: HTTPMethod, params: Parameters? = nil, _ completion: @escaping ResultClosure) { if !isConnected { diff --git a/TraccarClient/SignInModel.swift b/TraccarClient/SignInModel.swift new file mode 100644 index 0000000000000000000000000000000000000000..e074497c99d399f7f44153ef90b3fc1012b7e6af --- /dev/null +++ b/TraccarClient/SignInModel.swift @@ -0,0 +1,62 @@ +// +// SignInModel.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +final class SignInModel: KNObject { + + // MARK: - Properties + var token: String = "" + var user: UserModel = UserModel([:]) + var device: DeviceModel = DeviceModel([:]) + + // MARK: - Init + override init() { + super.init() + } + + init(_ dict: JSON, saveUser: Bool = false) { + super.init() + token = dict["token"] as? String ?? "" + + if let user = dict["user"] as? JSON { + self.user = UserModel(user, saveUser: true) + } + + if let device = dict["device"] as? JSON { + self.device = DeviceModel(device, saveUser: true) + } + + if saveUser { + saveToUserDefaults() + } + } + //MARK: - Methods + private func saveToUserDefaults() { + let ud = UserDefaults.standard + ud.set(token, forKey: Keys.User.access_token) + } + + class func loadFromUserDefaults() -> SignInModel { + let ud = UserDefaults.standard + var params: PARAMS = [:] + + if let token = ud.value(forKey: Keys.User.access_token) as? Int { + params.add(key: .access_token, token) + } + + return SignInModel(params) + } + + class func clearUserFromUserDefaults() { + let ud = UserDefaults.standard + + ud.removeObject(forKey: Keys.User.access_token) +// ud.removeObject(forKey: Keys.User.user) +// ud.removeObject(forKey: Keys.User.device) + } +} diff --git a/TraccarClient/TraccarClient-Info.plist b/TraccarClient/TraccarClient-Info.plist index dad50640b1bb1c0a127d5b7883883ef43be07559..9cb61f399dc13d0f48af83bfdb6e33f4a001f5c2 100644 --- a/TraccarClient/TraccarClient-Info.plist +++ b/TraccarClient/TraccarClient-Info.plist @@ -22,6 +22,8 @@ <string>????</string> <key>CFBundleVersion</key> <string>$(CURRENT_PROJECT_VERSION)</string> + <key>UILaunchStoryboardName</key> + <string>LaunchScreen</string> <key>LSRequiresIPhoneOS</key> <true/> <key>NSAppTransportSecurity</key> @@ -81,8 +83,6 @@ <array> <string>location</string> </array> - <key>UILaunchStoryboardName</key> - <string>LaunchScreen</string> <key>UIMainStoryboardFile</key> <string>MainStoryboard</string> <key>UIMainStoryboardFile~ipad</key> diff --git a/TraccarClient/UIButton.swift b/TraccarClient/UIButton.swift new file mode 100644 index 0000000000000000000000000000000000000000..71dc420973315c8cdf48f7b8fee6e0b92c8a4618 --- /dev/null +++ b/TraccarClient/UIButton.swift @@ -0,0 +1,64 @@ +// +// UIButton.swift +// TraccarClient +// +// Created by George Makhoul on 28/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +extension UIButton { + + // MARK: - Set Title + func set(title: String?) { + setTitle(title, for: .normal) + + contentVerticalAlignment = .center + contentHorizontalAlignment = .center + } + + func set(localized: String?) { + guard let title = localized else { return } + + setTitle(title, for: .normal) + + contentVerticalAlignment = .center + contentHorizontalAlignment = .center + } + + func set(color: UIColor?) { + setTitleColor(color, for: .normal) + } + + func set(font: UIFont?) { + titleLabel?.font = font + } + + func set(title: String?, color: UIColor?, font: UIFont?) { + set(title: title) + set(color: color) + set(font: font) + } + + func set(localized: String?, color: UIColor?, font: UIFont?) { + set(localized: localized) + set(color: color) + set(font: font) + } + + // MARK: - Set Icon + func set(icon: UIImage = UIImage()) { + setImage(icon.withRenderingMode(.alwaysOriginal), for: .normal) + + contentMode = .center + clipsToBounds = false + + contentVerticalAlignment = .center + contentHorizontalAlignment = .center + } + + func set(colorBackground: UIColor){ + backgroundColor = colorBackground + } +} diff --git a/TraccarClient/UIImage.swift b/TraccarClient/UIImage.swift new file mode 100644 index 0000000000000000000000000000000000000000..d1b809fb08ff2816e38eeffe26bd726be90037d7 --- /dev/null +++ b/TraccarClient/UIImage.swift @@ -0,0 +1,27 @@ +// +// UIImage.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +extension UIImage { + + // MARK: - Field Icons + static let field_show_password = img("field_show_password") + static let field_hide_password = img("field_hide_password") + +} +func img(_ name: String) -> UIImage { + if let img = UIImage(named: name) { + return img + } else { + return UIImage() + } +} + + + diff --git a/TraccarClient/UIImageView.swift b/TraccarClient/UIImageView.swift new file mode 100644 index 0000000000000000000000000000000000000000..0646de874bfaac6eeb90e40276ca082c3c4cc403 --- /dev/null +++ b/TraccarClient/UIImageView.swift @@ -0,0 +1,37 @@ +// +// UIImageView.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +extension UIImageView { + + // MARK: - Set Icon + func set(icon: UIImage) { + image = icon + contentMode = .center + clipsToBounds = false + } + + func set(aspectFill: UIImage) { + image = aspectFill + contentMode = .scaleAspectFill + clipsToBounds = false + } + + func set(toFit: UIImage) { + image = toFit + contentMode = .scaleAspectFit + clipsToBounds = false + } + + func set(toFill: UIImage) { + image = toFill + contentMode = .scaleToFill + clipsToBounds = true + } +} diff --git a/TraccarClient/UINavigationController.swift b/TraccarClient/UINavigationController.swift new file mode 100644 index 0000000000000000000000000000000000000000..ebb88cbec085f3fe31c92f2e648faa61be14c12c --- /dev/null +++ b/TraccarClient/UINavigationController.swift @@ -0,0 +1,20 @@ +// +// UINavigationController.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +extension UINavigationController { + + // MARK: - Push Multiple ViewControllers + func push(viewControllers: [UIViewController], animated: Bool) { + var stack = self.viewControllers + stack.append(contentsOf: viewControllers) + + setViewControllers(stack, animated: animated) + } +} diff --git a/TraccarClient/UITextField.swift b/TraccarClient/UITextField.swift new file mode 100644 index 0000000000000000000000000000000000000000..b018227a10d717744f238d7f048aa71a67282945 --- /dev/null +++ b/TraccarClient/UITextField.swift @@ -0,0 +1,29 @@ +// +// UITextField.swift +// TraccarClient +// +// Created by George Makhoul on 31/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +import UIKit + +extension UITextField { + func setForEnglighs(localizedPlaceholder: String, color: UIColor, font: UIFont) { + let paragraph = NSMutableParagraphStyle() + paragraph.alignment = .left + + let attributes: [NSAttributedString.Key : Any] = [ + .foregroundColor: color, + .font: font, + .paragraphStyle: paragraph + ] + + let range = NSString(string: localizedPlaceholder).range(of: localizedPlaceholder) + let attributedString = NSMutableAttributedString(string: localizedPlaceholder) + attributedString.addAttributes(attributes, range: range) + + attributedPlaceholder = attributedString + } +} diff --git a/TraccarClient/UITextView.swift b/TraccarClient/UITextView.swift new file mode 100644 index 0000000000000000000000000000000000000000..42736f9aa9a995c3f6eeb5b8ce2c099db9532394 --- /dev/null +++ b/TraccarClient/UITextView.swift @@ -0,0 +1,30 @@ +// +// UITextView.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation +import UIKit +extension UITextView { + func height() -> CGFloat { + if text.isEmpty { + return 0 + } + + let textView = UITextView(frame: CGRect(x: 0, y: 0, width: frame.width, height: .greatestFiniteMagnitude)) + textView.isHidden = true + textView.isEditable = false + textView.isSelectable = false + textView.isScrollEnabled = false + textView.textContainerInset = .zero + textView.textContainer.lineFragmentPadding = 0 + textView.text = text + textView.font = font + textView.sizeToFit() + + return textView.contentSize.height + } +} diff --git a/TraccarClient/UIView.swift b/TraccarClient/UIView.swift index 721d0e85151e284d5f0fc3a9788197014e312cf1..9175e37e0c7caa00286a3d25bad0c772ad4c73f0 100644 --- a/TraccarClient/UIView.swift +++ b/TraccarClient/UIView.swift @@ -38,7 +38,7 @@ extension UIView { func addCorners(_ radius: CGFloat) { layer.cornerRadius = radius - clipsToBounds = true + clipsToBounds = false } func addCorners(corners: CACornerMask, radius: CGFloat) { diff --git a/TraccarClient/UIWindow.swift b/TraccarClient/UIWindow.swift new file mode 100644 index 0000000000000000000000000000000000000000..c8e2f96c9847f55e6994468121e0aba6523d2aad --- /dev/null +++ b/TraccarClient/UIWindow.swift @@ -0,0 +1,33 @@ +// +// UIWindow.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import UIKit + +func topViewController() -> UIViewController? { + UIApplication.shared.windows.filter { $0.isKeyWindow }.first?.visibleViewController +} + +extension UIWindow { + var visibleViewController: UIViewController? { + return UIWindow.visibleViewController(viewController: self.rootViewController) + } + + static func visibleViewController(viewController: UIViewController?) -> UIViewController? { + if let navigation = viewController as? UINavigationController { + return UIWindow.visibleViewController(viewController: navigation.visibleViewController) + } else if let tabBar = viewController as? UITabBarController { + return UIWindow.visibleViewController(viewController: tabBar.selectedViewController) + } else { + if let presented = viewController?.presentedViewController { + return UIWindow.visibleViewController(viewController: presented) + } else { + return viewController + } + } + } +} diff --git a/TraccarClient/UserModel.swift b/TraccarClient/UserModel.swift new file mode 100644 index 0000000000000000000000000000000000000000..13ff2f940e03fb456e9a11cb578384c512584373 --- /dev/null +++ b/TraccarClient/UserModel.swift @@ -0,0 +1,102 @@ +// +// UserModel.swift +// TraccarClient +// +// Created by George Makhoul on 29/10/2023. +// Copyright © 2023 Traccar. All rights reserved. +// + +import Foundation + +final class UserModel: KNObject { + + // MARK: - Properties + var id: Int = 0 + var name: String = "" + var first_name: String = "" + var last_name: String = "" + var email: String = "" + var avatar: String = "" +// var cover: String? + var designation_id: Int = 0 + + // MARK: - Init + override init() { + super.init() + } + + init(_ dict: JSON, saveUser: Bool = false) { + super.init() + id = dict["id"] as? Int ?? 0 + name = dict["name"] as? String ?? "" + first_name = dict["first_name"] as? String ?? "" + last_name = dict["last_name"] as? String ?? "" + email = dict["email"] as? String ?? "" + avatar = dict["avatar"] as? String ?? "" +// cover = dict["cover"] as? String ?? "" + designation_id = dict["designation_id"] as? Int ?? 0 + + if saveUser { + saveToUserDefaults() + } + } + + //MARK: - Methods + private func saveToUserDefaults() { + let ud = UserDefaults.standard + + ud.set(id, forKey: Keys.User.id) + ud.set(name, forKey: Keys.User.name) + ud.set(first_name, forKey: Keys.User.firstName) + ud.set(last_name, forKey: Keys.User.lastName) + ud.set(email, forKey: Keys.User.email) + ud.set(avatar, forKey: Keys.User.avatar) +// ud.set(cover, forKey: Keys.User.cover) + ud.set(designation_id, forKey: Keys.User.designation_id) + } + + class func loadFromUserDefaults() -> UserModel { + let ud = UserDefaults.standard + var params: PARAMS = [:] + + if let id = ud.value(forKey: Keys.User.id) as? Int { + params.add(key: .id, id) + } + if let full_name = ud.value(forKey: Keys.User.name) as? String { + params.add(key: .name, full_name) + } + if let first_name = ud.value(forKey: Keys.User.firstName) as? String { + params.add(key: .fName, first_name) + } + if let last_name = ud.value(forKey: Keys.User.lastName) as? String { + params.add(key: .lName, last_name) + } + if let email = ud.value(forKey: Keys.User.email) as? String { + params.add(key: .email, email) + } + if let avatar = ud.value(forKey: Keys.User.avatar) as? String { + params.add(key: .avatar, avatar) + } +// if let cover = ud.value(forKey: Keys.User.cover) as? String { +// params.add(key: .cover, cover) +// } + if let designation_id = ud.value(forKey: Keys.User.designation_id) as? String { + params.add(key: .designation_id, designation_id) + } + + return UserModel(params) + } + + class func clearUserFromUserDefaults() { + let ud = UserDefaults.standard + + ud.removeObject(forKey: Keys.User.id) + ud.removeObject(forKey: Keys.User.name) + ud.removeObject(forKey: Keys.User.firstName) + ud.removeObject(forKey: Keys.User.lastName) + ud.removeObject(forKey: Keys.User.email) + ud.removeObject(forKey: Keys.User.avatar) +// ud.removeObject(forKey: Keys.User.cover) + ud.removeObject(forKey: Keys.User.designation_id) + } +} diff --git a/TraccarClient/splash.json b/TraccarClient/splash.json new file mode 100644 index 0000000000000000000000000000000000000000..ab9b820586508ae0f2c604f0d21663ebf9afb481 --- /dev/null +++ b/TraccarClient/splash.json @@ -0,0 +1 @@ +{"assets":[{"id":"eq47VlDpyQz2I25qYVdNB","layers":[{"ddd":0,"ind":2,"ty":2,"nm":"","ln":"image_2PugzZnMltvierFT7Gqx2","sr":1,"ks":{"a":{"a":0,"k":[320,320]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[4.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[8.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[12.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[15.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[19.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[23.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[26.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[29.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[33.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[36.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[39.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[42.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[45.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[47.8],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[50.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[53.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[55.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[57.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[60.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[62.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[64.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[66.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[68.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[70.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[72.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[74.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[75.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[77.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[78.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[80.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[81.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[82.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[84.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[85.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[86.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[87.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[88.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[89.54],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[90.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[91.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[92.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[92.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[93.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[94.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[94.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[95.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[95.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[96.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[96.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[97.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[97.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[97.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[98.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[98.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[98.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[98.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[99.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[99.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[99.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[99.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[99.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[99.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[99.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[99.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[99.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[99.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[99.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[99.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[99.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":1,"k":[{"t":0,"s":[25565.25,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[25565.25,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[25560.87,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[25553.61,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[25543.4,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[25530.19,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[25514.21,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[25496.08,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[25476.68,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[25456.96,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[25437.7,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[25419.45,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[25402.48,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25386.9,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[25372.7,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[25359.83,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[25348.18,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[25337.68,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[25328.21,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[25319.69,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[25312.04,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[25305.18,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[25299.05,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[25293.59,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[25288.74,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[25284.45,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[25280.69,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[25277.42,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[25274.6,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[25272.2,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[25270.2,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[25268.57,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[25267.29,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[25266.33,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[25265.69,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[25265.33,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[25265.25,25527.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":1,"k":[{"t":0,"s":[300,300],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[291.77,291.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[283.76,283.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[275.98,275.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[268.42,268.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[261.09,261.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[253.97,253.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[247.06,247.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[240.36,240.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[233.87,233.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[227.58,227.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[221.49,221.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[215.6,215.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[209.9,209.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[204.4,204.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[199.08,199.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[193.94,193.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[188.98,188.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[184.21,184.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[179.6,179.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[175.17,175.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[170.9,170.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[166.8,166.8],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[162.86,162.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[159.08,159.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[155.45,155.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[151.98,151.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[148.65,148.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[145.47,145.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[142.43,142.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[139.53,139.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[136.76,136.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[134.13,134.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[131.62,131.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[129.24,129.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[126.99,126.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[124.85,124.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[122.83,122.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[120.92,120.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[119.12,119.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[117.43,117.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[115.84,115.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[114.35,114.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[112.95,112.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[111.65,111.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[110.44,110.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[109.32,109.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[108.28,108.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[107.32,107.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[106.44,106.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[105.63,105.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[104.89,104.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[104.22,104.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[103.61,103.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[103.07,103.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[102.58,102.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[102.15,102.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[101.77,101.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[101.43,101.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[101.15,101.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[100.9,100.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[100.69,100.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[100.52,100.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[100.37,100.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[100.26,100.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[100.17,100.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[100.11,100.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[100.06,100.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[100.03,100.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[100.01,100.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[100,100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[100,100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[99.94,99.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[99.7,99.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[99.36,99.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[98.92,98.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[98.38,98.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[97.73,97.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[96.98,96.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[96.15,96.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[95.25,95.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[94.32,94.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[93.36,93.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[92.42,92.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[91.49,91.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[90.6,90.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[89.75,89.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[88.95,88.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[88.2,88.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[87.5,87.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[86.85,86.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[86.24,86.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[85.67,85.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[85.15,85.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[84.66,84.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[84.21,84.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[83.79,83.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[83.4,83.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[83.04,83.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[82.71,82.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[82.41,82.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[82.13,82.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[81.87,81.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[81.63,81.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[81.42,81.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[81.22,81.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[81.04,81.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[80.88,80.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[80.73,80.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[80.6,80.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[80.48,80.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[80.38,80.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[80.29,80.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[80.22,80.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[80.15,80.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[80.1,80.1],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[80.06,80.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[80.03,80.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[80.01,80.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":121,"s":[80,80],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"yAtkPcXu7hZwkn0ikKWVM"},{"ddd":0,"ind":3,"ty":2,"nm":"","ln":"image_U-9OiQhh1bWe7sQSghUNG","sr":1,"ks":{"a":{"a":0,"k":[429,429]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[1.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[10.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[19.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[28.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[36.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[43.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[50.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[56.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[62.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[68.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[73.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[78.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[82.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[86.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[89.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[93.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[96.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[98.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[100.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[102.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[104.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[106],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[107.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[108.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[108.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[109.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[109.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[109.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[109.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[109.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[109.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[109.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[108.8],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[108.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[107.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[106.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[106.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[105.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[104.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[104.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[103.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[102.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[101.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[101.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[100.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[100.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[100.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[100.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":0,"k":[25615.75,25600]},"r":{"a":0,"k":0},"s":{"a":1,"k":[{"t":0,"s":[25,25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[25,25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[26.05,26.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[33.15,33.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[39.84,39.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[46.13,46.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[52.04,52.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[57.58,57.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[62.75,62.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[67.58,67.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[72.06,72.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[76.22,76.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[80.06,80.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[83.59,83.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[86.83,86.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[89.78,89.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[92.46,92.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[94.88,94.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[97.05,97.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[98.97,98.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[100.67,100.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[102.15,102.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[103.43,103.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[104.5,104.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[105.4,105.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[106.12,106.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[106.67,106.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[107.08,107.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[107.34,107.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[107.48,107.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[107.49,107.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[107.4,107.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[107.22,107.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[106.94,106.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[106.6,106.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[106.19,106.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[105.73,105.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[105.23,105.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[104.69,104.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[104.14,104.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[103.58,103.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[103.03,103.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[102.49,102.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[101.97,101.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[101.49,101.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[101.06,101.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[100.69,100.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[100.39,100.39],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[100.16,100.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[100.03,100.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[100,100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"WTjD4nHYDSfJwFGqR_Ldr"}]},{"id":"IU7JjaSqYDIJbCJ9qv1lA","layers":[{"ddd":0,"ind":7,"ty":2,"nm":"","ln":"image_GOce4bErhARwDJd5WumJi","sr":1,"ks":{"a":{"a":0,"k":[151.5,151.5]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[1.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[5.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[9.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[13.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[16.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[20.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[24.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[28.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[32.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[35.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[39.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[43.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[47.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[51.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[54.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[58.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[62.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[66.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[70.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[73.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[77.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[81.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[85.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[89.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[92.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[96.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[95.56],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[90.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[86.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[81.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[72.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[67.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[63.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[58.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[53.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[49.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[44.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[39.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[35.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[30.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[25.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[21.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[16.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[12.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[7.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[2.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":0,"k":[26025.5,25623.19]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"2epu9UY0Hu5Ho2aAok-K2"},{"ddd":0,"ind":8,"ty":2,"nm":"","ln":"image_py2gmrqsekm08OWkPjitJ","sr":1,"ks":{"a":{"a":0,"k":[156,156]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[4.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[9.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[13.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[18.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[27.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[32.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[36.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[41.56],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[46.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[50.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[55.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[60.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[64.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[69.39],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[74.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[78.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[83.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[87.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[92.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[97.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":132,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":133,"s":[99.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":134,"s":[94.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":135,"s":[90.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":136,"s":[85.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":137,"s":[80.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":138,"s":[75.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":139,"s":[71.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":140,"s":[66.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":141,"s":[61.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":142,"s":[56.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":143,"s":[51.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":144,"s":[47.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":145,"s":[42.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":146,"s":[37.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":147,"s":[32.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":148,"s":[28.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":149,"s":[23.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":150,"s":[18.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":151,"s":[13.8],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":152,"s":[9.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":153,"s":[4.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":154,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":0,"k":[25179,25581.31]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"bXtGDwyFMuRTAoYrvsI_d"}]},{"id":"lDpiea0kASJKIIzYJ9MMs","layers":[{"ddd":0,"ind":12,"ty":2,"nm":"","ln":"image_ULRKdJRPct82aZjdmwZiI","sr":1,"ks":{"a":{"a":0,"k":[375,315]},"o":{"a":0,"k":100},"p":{"a":0,"k":[25600,25600]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"1tGyzCsdWDNksloW4XUOw"},{"ddd":0,"ind":13,"ty":2,"nm":"","ln":"image_qO27Tj5H8heHvFrf0KDg5","sr":1,"ks":{"a":{"a":0,"k":[270.5,263]},"o":{"a":0,"k":100},"p":{"a":0,"k":[25537.5,25625]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"LZ5G9BVUbaUPxXTh_pOcx"}]},{"id":"ivq2Zd7J6UgoNLHQL-ghu","layers":[{"ddd":0,"ind":10,"ty":2,"nm":"","ln":"image_3wXMXrzmVQ0EBKgPNrhho","sr":1,"ks":{"a":{"a":0,"k":[376.5,334]},"o":{"a":0,"k":100},"p":{"a":0,"k":[25600,25600]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"wsBWrCmdLeclMc_KOKX27"},{"ddd":0,"ind":11,"ty":0,"nm":"","ln":"precomp_QFe1KSONbkcDWsTMdnSMm11","sr":1,"ks":{"a":{"a":0,"k":[25600,25600]},"o":{"a":0,"k":100},"p":{"a":1,"k":[{"t":0,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[25599.5,25602.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[25599.5,25604.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[25599.5,25604.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[25599.5,25604.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[25599.5,25601.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[25599.5,25600.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[25599.5,25600.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[25599.5,25601.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[25599.5,25604.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[25599.5,25604.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[25599.5,25604.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[25599.5,25602.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[25599.5,25600.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[25599.5,25600.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[25599.5,25601.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[25599.5,25603.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[25599.5,25604.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[25599.5,25604.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[25599.5,25602.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[25599.5,25600.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[25599.5,25600.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[25599.5,25600.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[25599.5,25603.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[25599.5,25604.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[25599.5,25605],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[25599.5,25603.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[25599.5,25600.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[25599.5,25600.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[25599.5,25600.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[25599.5,25603.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[25599.5,25604.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[25599.5,25604.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[25599.5,25603.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[25599.5,25601.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[25599.5,25600.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[25599.5,25600.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[25599.5,25602.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[25599.5,25604.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[25599.5,25604.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[25599.5,25604.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[25599.5,25601.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[25599.5,25600.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[25599.5,25602.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[25599.5,25604.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[25599.5,25604.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[25599.5,25604.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[25599.5,25601.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[25599.5,25600.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[25599.5,25600.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[25599.5,25601.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[25599.5,25603.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[25599.5,25604.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[25599.5,25604.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[25599.5,25602.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[25599.5,25600.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[25599.5,25600.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[25599.5,25600.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[25599.5,25603.56],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[25599.5,25604.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[25599.5,25605],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[25599.5,25602.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[25599.5,25600.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[25599.5,25600.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[25599.5,25600.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[25599.5,25603.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[25599.5,25604.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[25599.5,25604.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[25599.5,25603.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[25599.5,25601.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[25599.5,25600.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[25599.5,25600.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[25599.5,25602.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[25599.5,25604.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[25599.5,25604.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[25599.5,25604.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[25599.5,25601.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[25599.5,25600.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[25599.5,25602.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[25599.5,25604.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[25599.5,25604.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[25599.5,25604.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[25599.5,25601.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25599.5,25600.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[25599.5,25600.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[25599.5,25601.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[25599.5,25603.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[25599.5,25604.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[25599.5,25604.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[25599.5,25602.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[25599.5,25600.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[25599.5,25600.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[25599.5,25601.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[25599.5,25603.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[25599.5,25604.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[25599.5,25605],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[25599.5,25602.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[25599.5,25600.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[25599.5,25600.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[25599.5,25600.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[25599.5,25603.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[25599.5,25604.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[25599.5,25604.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[25599.5,25603.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[25599.5,25600.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[25599.5,25600.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[25599.5,25600.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[25599.5,25602.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[25599.5,25604.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[25599.5,25604.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[25599.5,25603.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[25599.5,25601.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[25599.5,25600.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[25599.5,25600.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[25599.5,25602.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[25599.5,25604.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[25599.5,25604.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[25599.5,25604.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[25599.5,25601.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":121,"s":[25599.5,25600.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":122,"s":[25599.5,25600.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":123,"s":[25599.5,25601.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":124,"s":[25599.5,25604.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":125,"s":[25599.5,25604.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":126,"s":[25599.5,25604.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":127,"s":[25599.5,25601.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":128,"s":[25599.5,25600.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":129,"s":[25599.5,25600.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":130,"s":[25599.5,25601.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":131,"s":[25599.5,25603.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":132,"s":[25599.5,25604.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":133,"s":[25599.5,25604.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":134,"s":[25599.5,25602.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":135,"s":[25599.5,25600.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":136,"s":[25599.5,25600.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":137,"s":[25599.5,25600.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":138,"s":[25599.5,25603.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":139,"s":[25599.5,25604.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":140,"s":[25599.5,25605],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":141,"s":[25599.5,25603.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":142,"s":[25599.5,25600.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":143,"s":[25599.5,25600.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":144,"s":[25599.5,25600.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":145,"s":[25599.5,25603.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":146,"s":[25599.5,25604.54],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":147,"s":[25599.5,25604.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":148,"s":[25599.5,25603.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":149,"s":[25599.5,25601.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":150,"s":[25599.5,25600.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":151,"s":[25599.5,25600.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":152,"s":[25599.5,25602.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":153,"s":[25599.5,25604.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":154,"s":[25599.5,25604.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":155,"s":[25599.5,25604.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":156,"s":[25599.5,25601.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":157,"s":[25599.5,25600.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":158,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":159,"s":[25599.5,25602.1],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":160,"s":[25599.5,25604.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":161,"s":[25599.5,25604.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":162,"s":[25599.5,25604.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":163,"s":[25599.5,25601.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":164,"s":[25599.5,25600.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":165,"s":[25599.5,25600.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":166,"s":[25599.5,25601.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":167,"s":[25599.5,25603.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":168,"s":[25599.5,25604.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":169,"s":[25599.5,25604.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":170,"s":[25599.5,25602.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":171,"s":[25599.5,25600.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":172,"s":[25599.5,25600.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":173,"s":[25599.5,25600.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":174,"s":[25599.5,25603.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":175,"s":[25599.5,25604.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":176,"s":[25599.5,25605],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":177,"s":[25599.5,25602.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":178,"s":[25599.5,25600.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":179,"s":[25599.5,25600.1],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":180,"s":[25599.5,25600.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":181,"s":[25599.5,25603.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":182,"s":[25599.5,25604.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":183,"s":[25599.5,25604.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":184,"s":[25599.5,25603.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":185,"s":[25599.5,25601.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":186,"s":[25599.5,25600.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":187,"s":[25599.5,25600.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":188,"s":[25599.5,25602.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":189,"s":[25599.5,25604.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":190,"s":[25599.5,25604.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":191,"s":[25599.5,25603.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":192,"s":[25599.5,25601.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":193,"s":[25599.5,25600.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":194,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":195,"s":[25599.5,25602.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":196,"s":[25599.5,25604.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":197,"s":[25599.5,25604.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":198,"s":[25599.5,25604.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":199,"s":[25599.5,25601.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":200,"s":[25599.5,25600.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":201,"s":[25599.5,25600.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":202,"s":[25599.5,25601.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":203,"s":[25599.5,25604.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":204,"s":[25599.5,25604.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":205,"s":[25599.5,25604.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":206,"s":[25599.5,25602.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":207,"s":[25599.5,25600.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":208,"s":[25599.5,25600.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":209,"s":[25599.5,25601.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":210,"s":[25599.5,25603.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":211,"s":[25599.5,25604.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":212,"s":[25599.5,25604.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":213,"s":[25599.5,25602.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":214,"s":[25599.5,25600.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":215,"s":[25599.5,25600.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":216,"s":[25599.5,25600.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":217,"s":[25599.5,25603.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":218,"s":[25599.5,25604.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":219,"s":[25599.5,25604.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":220,"s":[25599.5,25603.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":221,"s":[25599.5,25600.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":222,"s":[25599.5,25600.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":223,"s":[25599.5,25600.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":224,"s":[25599.5,25603],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":225,"s":[25599.5,25604.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":226,"s":[25599.5,25604.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":227,"s":[25599.5,25603.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":228,"s":[25599.5,25601.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":229,"s":[25599.5,25600.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":230,"s":[25599.5,25600.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":231,"s":[25599.5,25602.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":232,"s":[25599.5,25604.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":233,"s":[25599.5,25604.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":234,"s":[25599.5,25604.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":235,"s":[25599.5,25601.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":236,"s":[25599.5,25600.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":237,"s":[25599.5,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":238,"s":[25599.5,25601.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":239,"s":[25599.5,25604.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"w":51200,"h":51200,"ip":0,"op":240,"st":0,"bm":0,"refId":"lDpiea0kASJKIIzYJ9MMs"}]},{"id":"Eujr1mqLiS5h0eJtEFwbw","layers":[{"ddd":0,"ind":5,"ty":2,"nm":"","ln":"image_xT778vcdTVeU3lzZqDvGs","sr":1,"ks":{"a":{"a":0,"k":[707.5,581.5]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[0.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[2.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[4.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[5.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[7.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[9.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[10.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[12.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[14.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[15.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[17.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[19.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[20.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[22.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[24.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[27.62],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[29.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[30.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[32.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[34.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[35.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[37.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[39.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[40.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[42.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[44.32],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[45.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[47.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[49.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[52.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[54.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[56.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[57.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[59.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[61.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[62.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[64.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[66.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[67.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[69.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[71.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[72.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[74.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[76.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[77.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[79.39],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[81.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[82.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[84.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":121,"s":[86.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":122,"s":[87.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":123,"s":[89.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":124,"s":[91.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":125,"s":[92.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":126,"s":[94.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":127,"s":[96.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":128,"s":[97.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":129,"s":[99.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":130,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":1,"k":[{"t":0,"s":[25347,25450],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[25347,25450],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[25347,25456.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[25347,25467.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[25347,25478.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[25347,25488.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[25347,25498.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[25347,25507.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[25347,25516.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[25347,25524.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[25347,25532.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[25347,25539.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[25347,25546.58],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[25347,25553.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[25347,25559.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[25347,25565.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[25347,25570.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25347,25575.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[25347,25580.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[25347,25584.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[25347,25588.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[25347,25591.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[25347,25595.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[25347,25598.26],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[25347,25600.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[25347,25603.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[25347,25605.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[25347,25607.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[25347,25609.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[25347,25610.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[25347,25611.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[25347,25612.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[25347,25613.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[25347,25614.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[25347,25614.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[25347,25614.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[25347,25615],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[25347,25614.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[25347,25614.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[25347,25614.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[25347,25614.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[25347,25613.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[25347,25612.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[25347,25612.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[25347,25611.56],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[25347,25610.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[25347,25609.94],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[25347,25609.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[25347,25608.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[25347,25607.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[25347,25606.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[25347,25605.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[25347,25604.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":121,"s":[25347,25603.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":122,"s":[25347,25603.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":123,"s":[25347,25602.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":124,"s":[25347,25601.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":125,"s":[25347,25601.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":126,"s":[25347,25600.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":127,"s":[25347,25600.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":128,"s":[25347,25600.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":129,"s":[25347,25600.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":130,"s":[25347,25600],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"fSOdzhxUijRNBBDcTg8SR"},{"ddd":0,"ind":6,"ty":0,"nm":"","ln":"precomp_lB8mwcTp2VNuH5GgktvfT6","sr":1,"ks":{"a":{"a":0,"k":[25600,25600]},"o":{"a":0,"k":100},"p":{"a":1,"k":[{"t":0,"s":[24560.5,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[24575.86,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[24591.23,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[24606.59,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[24621.96,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[24637.32,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[24652.68,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[24668.05,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[24683.41,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[24698.78,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[24714.14,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[24729.5,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[24744.87,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[24760.23,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[24775.6,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[24790.96,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[24806.32,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[24821.69,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[24837.05,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[24852.42,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[24867.78,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[24883.14,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[24898.51,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[24913.87,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[24929.24,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[24944.6,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[24959.96,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[24975.33,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[24990.69,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[25006.06,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[25021.42,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[25036.78,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[25052.15,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[25067.51,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[25082.88,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[25098.24,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[25113.6,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[25128.97,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[25144.33,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[25159.7,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[25175.06,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[25190.42,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[25205.79,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[25221.15,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[25236.52,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[25251.88,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[25267.24,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[25282.61,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[25297.97,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[25313.34,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[25328.7,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[25344.06,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[25359.43,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[25374.79,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[25390.16,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[25405.52,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[25420.88,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[25436.25,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[25451.61,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[25466.98,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[25482.34,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[25497.7,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[25513.07,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[25528.43,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[25543.8,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[25559.16,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[25574.52,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[25589.89,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[25605.25,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[25620.62,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[25635.98,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[25651.34,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[25666.71,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[25682.07,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[25697.44,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[25712.8,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[25728.16,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[25743.53,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[25758.89,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[25774.26,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[25789.62,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[25804.98,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[25820.35,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[25835.71,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[25851.08,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25866.44,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[25881.8,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[25897.17,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[25912.53,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[25927.9,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[25943.26,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[25958.62,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[25973.99,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[25989.35,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[26004.72,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[26020.08,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[26035.44,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[26050.81,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[26066.17,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[26081.54,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[26096.9,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[26112.26,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[26127.63,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[26142.99,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[26158.36,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[26173.72,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[26189.08,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[26204.45,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[26219.81,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[26235.18,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[26250.54,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[26265.9,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[26281.27,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[26296.63,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[26312,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[26327.36,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[26342.72,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[26358.09,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[26373.45,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[26388.82,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":120,"s":[26404.18,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":121,"s":[26419.54,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":122,"s":[26434.91,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":123,"s":[26450.27,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":124,"s":[26465.64,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":125,"s":[26481,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":126,"s":[26496.36,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":127,"s":[26511.73,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":128,"s":[26527.09,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":129,"s":[26542.46,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":130,"s":[26557.82,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":131,"s":[26573.18,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":132,"s":[26588.55,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":133,"s":[26603.91,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":134,"s":[26619.28,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":135,"s":[26634.64,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":136,"s":[26650,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":137,"s":[26665.37,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":138,"s":[26680.73,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":139,"s":[26696.1,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":140,"s":[26711.46,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":141,"s":[26726.82,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":142,"s":[26742.19,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":143,"s":[26757.55,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":144,"s":[26772.92,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":145,"s":[26788.28,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":146,"s":[26803.64,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":147,"s":[26819.01,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":148,"s":[26834.37,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":149,"s":[26849.74,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":150,"s":[26865.1,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":151,"s":[26880.46,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":152,"s":[26895.83,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":153,"s":[26911.19,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":154,"s":[26926.56,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":155,"s":[26941.92,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":156,"s":[26957.28,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":157,"s":[26972.65,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":158,"s":[26988.01,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":159,"s":[27003.38,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":160,"s":[27018.74,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":161,"s":[27034.1,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":162,"s":[27049.47,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":163,"s":[27064.83,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":164,"s":[27080.2,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":165,"s":[27095.56,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":166,"s":[27110.92,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":167,"s":[27126.29,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":168,"s":[27141.65,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":169,"s":[27157.02,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":170,"s":[27172.38,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":171,"s":[27187.74,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":172,"s":[27203.11,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":173,"s":[27218.47,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":174,"s":[27233.84,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":175,"s":[27249.2,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":176,"s":[27264.56,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":177,"s":[27279.93,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":178,"s":[27295.29,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":179,"s":[27310.66,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":180,"s":[27320.5,25501.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"w":51200,"h":51200,"ip":0,"op":240,"st":0,"bm":0,"refId":"IU7JjaSqYDIJbCJ9qv1lA"},{"ddd":0,"ind":9,"ty":0,"nm":"","ln":"precomp_9hCsiezp0I8KYEVWNmu979","sr":1,"ks":{"a":{"a":0,"k":[25600,25600]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[1.11],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[2.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[3.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[4.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[5.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[6.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[7.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[8.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[10.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[11.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[12.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[13.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[14.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[15.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[16.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[17.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[18.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[20.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[21.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[22.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[23.38],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[24.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[25.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[26.72],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[27.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[28.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[30.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[31.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[32.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[33.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[34.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[35.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[36.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[37.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[38.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[40.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[41.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[42.31],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[43.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[44.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[45.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[46.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[47.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[48.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[50.1],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[51.21],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[52.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[53.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[54.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[55.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[56.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[57.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[59.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[60.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[61.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[62.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[63.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[64.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[65.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[66.8],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[67.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[69.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[70.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[71.25],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[72.37],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[73.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[74.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[75.71],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[76.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[77.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[79.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[80.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[81.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[82.39],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[83.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[84.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[85.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[86.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[87.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[89.07],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[90.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[91.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[92.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[93.52],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[94.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[95.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[96.86],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[97.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[99.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":1,"k":[{"t":0,"s":[26154,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[26142.11,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[26130.48,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[26119.12,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[26108.01,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[26097.16,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[26086.57,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[26076.22,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[26066.12,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[26056.27,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[26046.65,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[26037.27,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[26028.13,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[26019.22,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[26010.54,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[26002.08,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[25993.85,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[25985.84,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[25978.04,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[25970.46,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[25963.09,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[25955.93,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[25948.97,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[25942.22,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[25935.66,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[25929.3,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[25923.14,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[25917.16,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[25911.37,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[25905.77,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[25900.35,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[25895.1,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[25890.03,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[25885.14,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[25880.41,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[25875.85,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[25871.45,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[25867.21,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[25863.13,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[25859.21,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[25855.43,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[25851.81,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[25848.33,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[25844.99,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[25841.79,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[25838.73,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[25835.8,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[25833.01,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[25830.34,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[25827.79,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[25825.37,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[25823.06,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[25820.88,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[25818.8,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[25816.83,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[25814.97,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[25813.22,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[25811.56,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[25810.01,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[25808.54,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[25807.17,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[25805.89,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[25804.7,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[25803.58,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[25802.55,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[25801.6,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[25800.71,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[25799.9,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[25799.16,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[25798.48,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[25797.87,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[25797.31,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[25796.81,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[25796.36,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[25795.97,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[25795.62,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[25795.31,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[25795.05,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[25794.82,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[25794.63,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[25794.47,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[25794.34,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[25794.24,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[25794.16,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[25794.1,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[25794.06,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[25794.03,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[25794.01,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[25794,25838],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"w":51200,"h":51200,"ip":0,"op":240,"st":0,"bm":0,"refId":"ivq2Zd7J6UgoNLHQL-ghu"},{"ddd":0,"ind":14,"ty":2,"nm":"","ln":"image_3juWPe745iiVgpjOHNq3-","sr":1,"ks":{"a":{"a":0,"k":[593,438]},"o":{"a":0,"k":100},"p":{"a":0,"k":[25967.5,25700]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"mKrpXSqsifgtBYGu5ryXm"}]},{"h":640,"id":"yAtkPcXu7hZwkn0ikKWVM","p":"","u":"","w":640,"e":1},{"h":858,"id":"WTjD4nHYDSfJwFGqR_Ldr","p":"","u":"","w":858,"e":1},{"h":1163,"id":"fSOdzhxUijRNBBDcTg8SR","p":"","u":"","w":1415,"e":1},{"h":303,"id":"2epu9UY0Hu5Ho2aAok-K2","p":"","u":"","w":303,"e":1},{"h":312,"id":"bXtGDwyFMuRTAoYrvsI_d","p":"","u":"","w":312,"e":1},{"h":668,"id":"wsBWrCmdLeclMc_KOKX27","p":"","u":"","w":753,"e":1},{"h":630,"id":"1tGyzCsdWDNksloW4XUOw","p":"","u":"","w":750,"e":1},{"h":526,"id":"LZ5G9BVUbaUPxXTh_pOcx","p":"","u":"","w":541,"e":1},{"h":876,"id":"mKrpXSqsifgtBYGu5ryXm","p":"","u":"","w":1186,"e":1},{"h":1436,"id":"nvr6_T8yXsKqnO-rkFfnz","p":"","u":"","w":1500,"e":1}],"ddd":0,"fr":60,"h":2560,"ip":0,"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"","ln":"precomp_i5TlynKsJlxk7M5St-ZP41","sr":1,"ks":{"a":{"a":0,"k":[25600,25600]},"o":{"a":0,"k":100},"p":{"a":0,"k":[834.75,642.65]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"w":51200,"h":51200,"ip":0,"op":240,"st":0,"bm":0,"refId":"eq47VlDpyQz2I25qYVdNB"},{"ddd":0,"ind":4,"ty":0,"nm":"","ln":"precomp_4GM9A_HSFvJTSKUFg6bRT4","sr":1,"ks":{"a":{"a":0,"k":[25600,25600]},"o":{"a":0,"k":100},"p":{"a":0,"k":[462.5,1264]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"w":51200,"h":51200,"ip":0,"op":240,"st":0,"bm":0,"refId":"Eujr1mqLiS5h0eJtEFwbw"},{"ddd":0,"ind":15,"ty":2,"nm":"","ln":"image_kdgzSQBgF2XNQvZMkgsMU","sr":1,"ks":{"a":{"a":0,"k":[750,718]},"o":{"a":1,"k":[{"t":0,"s":[0],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[4.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[9.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[14.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[18.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[23.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[27.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[31.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[34.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[38.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[42.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[45.61],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[48.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[52.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[55.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[57.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[60.65],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[63.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[65.79],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[68.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[70.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[72.63],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[74.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[76.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[78.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[80.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[81.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[83.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[84.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[86.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[87.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[88.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[89.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[90.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[91.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[92.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[93.66],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[94.42],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[95.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[95.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[96.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[96.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[97.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[97.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[98.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[98.47],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[98.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[99.22],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[99.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[99.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[99.67],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[99.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[99.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[99.91],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[99.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[99.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[99.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"p":{"a":1,"k":[{"t":0,"s":[366,2481],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":1,"s":[366,2473.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":2,"s":[366,2466.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":3,"s":[366,2458.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":4,"s":[366,2452.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":5,"s":[366,2445.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":6,"s":[366,2439.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":7,"s":[366,2433.06],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":8,"s":[366,2427.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":9,"s":[366,2421.48],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":10,"s":[366,2416.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":11,"s":[366,2410.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":12,"s":[366,2405.73],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":13,"s":[366,2400.9],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":14,"s":[366,2396.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":15,"s":[366,2391.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":16,"s":[366,2387.6],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":17,"s":[366,2383.55],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":18,"s":[366,2379.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":19,"s":[366,2376],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":20,"s":[366,2372.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":21,"s":[366,2369.16],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":22,"s":[366,2365.99],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":23,"s":[366,2362.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":24,"s":[366,2360.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":25,"s":[366,2357.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":26,"s":[366,2354.89],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":27,"s":[366,2352.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":28,"s":[366,2350.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":29,"s":[366,2348.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":30,"s":[366,2346.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":31,"s":[366,2344.28],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":32,"s":[366,2342.54],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":33,"s":[366,2340.93],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":34,"s":[366,2339.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":35,"s":[366,2338.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":36,"s":[366,2336.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":37,"s":[366,2335.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":38,"s":[366,2334.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":39,"s":[366,2333.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":40,"s":[366,2332.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[366,2331.83],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[366,2331.1],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[366,2330.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[366,2329.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[366,2329.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[366,2328.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[366,2328.53],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[366,2328.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[366,2327.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[366,2327.69],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[366,2327.5],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[366,2327.35],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[366,2327.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[366,2327.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[366,2327.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[366,2327.04],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[366,2327.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[366,2327],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"r":{"a":0,"k":0},"s":{"a":1,"k":[{"t":0,"s":[140,140],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":41,"s":[140,140],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":42,"s":[139.98,139.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":43,"s":[139.74,139.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":44,"s":[139.43,139.43],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":45,"s":[139.05,139.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":46,"s":[138.59,138.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":47,"s":[138.05,138.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":48,"s":[137.44,137.44],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":49,"s":[136.74,136.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":50,"s":[135.97,135.97],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":51,"s":[135.12,135.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":52,"s":[134.19,134.19],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":53,"s":[133.2,133.2],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":54,"s":[132.15,132.15],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":55,"s":[131.05,131.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":56,"s":[129.92,129.92],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":57,"s":[128.76,128.76],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":58,"s":[127.59,127.59],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":59,"s":[126.41,126.41],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":60,"s":[125.24,125.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":61,"s":[124.09,124.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":62,"s":[122.96,122.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":63,"s":[121.85,121.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":64,"s":[120.78,120.78],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":65,"s":[119.74,119.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":66,"s":[118.74,118.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":67,"s":[117.77,117.77],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":68,"s":[116.84,116.84],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":69,"s":[115.95,115.95],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":70,"s":[115.09,115.09],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":71,"s":[114.27,114.27],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":72,"s":[113.49,113.49],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":73,"s":[112.74,112.74],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":74,"s":[112.02,112.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":75,"s":[111.33,111.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":76,"s":[110.68,110.68],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":77,"s":[110.05,110.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":78,"s":[109.46,109.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":79,"s":[108.88,108.88],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":80,"s":[108.34,108.34],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":81,"s":[107.82,107.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":82,"s":[107.33,107.33],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":83,"s":[106.85,106.85],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":84,"s":[106.4,106.4],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":85,"s":[105.98,105.98],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":86,"s":[105.57,105.57],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":87,"s":[105.18,105.18],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":88,"s":[104.81,104.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":89,"s":[104.46,104.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":90,"s":[104.13,104.13],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":91,"s":[103.81,103.81],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":92,"s":[103.51,103.51],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":93,"s":[103.23,103.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":94,"s":[102.96,102.96],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":95,"s":[102.7,102.7],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":96,"s":[102.46,102.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":97,"s":[102.24,102.24],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":98,"s":[102.02,102.02],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":99,"s":[101.82,101.82],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":100,"s":[101.64,101.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":101,"s":[101.46,101.46],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":102,"s":[101.3,101.3],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":103,"s":[101.14,101.14],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":104,"s":[101,101],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":105,"s":[100.87,100.87],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":106,"s":[100.75,100.75],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":107,"s":[100.64,100.64],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":108,"s":[100.54,100.54],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":109,"s":[100.45,100.45],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":110,"s":[100.36,100.36],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":111,"s":[100.29,100.29],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":112,"s":[100.23,100.23],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":113,"s":[100.17,100.17],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":114,"s":[100.12,100.12],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":115,"s":[100.08,100.08],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":116,"s":[100.05,100.05],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":117,"s":[100.03,100.03],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":118,"s":[100.01,100.01],"i":{"x":0,"y":0},"o":{"x":1,"y":1}},{"t":119,"s":[100,100],"i":{"x":0,"y":0},"o":{"x":1,"y":1}}]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"refId":"nvr6_T8yXsKqnO-rkFfnz"},{"ddd":0,"ind":16,"ty":4,"nm":"","ln":"IrahbXW8iUfEJLyz50gLr","sr":1,"ks":{"a":{"a":0,"k":[0,0]},"o":{"a":0,"k":100},"p":{"a":0,"k":[800,1280]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}},"ao":0,"ip":0,"op":240,"st":0,"bm":0,"shapes":[{"ty":"gr","hd":false,"bm":0,"it":[{"ty":"rc","hd":false,"p":{"a":0,"k":[0,0]},"r":{"a":0,"k":0},"s":{"a":0,"k":[1600,2560]}},{"ty":"fl","hd":false,"bm":0,"c":{"a":0,"k":[1,1,1]},"r":1,"o":{"a":0,"k":100}},{"ty":"tr","nm":"Transform","a":{"a":0,"k":[0,0]},"o":{"a":0,"k":100},"p":{"a":0,"k":[0,0]},"r":{"a":0,"k":0},"s":{"a":0,"k":[100,100]},"sk":{"a":0,"k":0},"sa":{"a":0,"k":0}}],"np":0}]}],"meta":{"g":"https://jitter.video"},"nm":"New-file","op":240,"v":"5.7.4","w":1600} \ No newline at end of file