diff --git a/AsyncSwift.xcodeproj/project.pbxproj b/AsyncSwift.xcodeproj/project.pbxproj index 33a61d9..43d925b 100644 --- a/AsyncSwift.xcodeproj/project.pbxproj +++ b/AsyncSwift.xcodeproj/project.pbxproj @@ -19,7 +19,6 @@ C63D4450291BDD2B005D5AE6 /* String+.swift in Sources */ = {isa = PBXBuildFile; fileRef = C63D444F291BDD2B005D5AE6 /* String+.swift */; }; C66C68D328D1B00A0091F960 /* EventModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66C68D228D1B00A0091F960 /* EventModel.swift */; }; C66C68D528D1B0130091F960 /* SessionModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66C68D428D1B0130091F960 /* SessionModel.swift */; }; - C66DAD5028CF478700195DEB /* SessionView+Observed.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66DAD4F28CF478700195DEB /* SessionView+Observed.swift */; }; C66E3D95290BA48500097BEA /* ProfileRegisterView.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66E3D94290BA48500097BEA /* ProfileRegisterView.swift */; }; C66E3D97290BA4FC00097BEA /* ProfileRegisterViewObserved.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66E3D96290BA4FC00097BEA /* ProfileRegisterViewObserved.swift */; }; C66E3D99290BB9C100097BEA /* TextEditor+.swift in Sources */ = {isa = PBXBuildFile; fileRef = C66E3D98290BB9C100097BEA /* TextEditor+.swift */; }; @@ -56,8 +55,42 @@ E9171F0028D15426002FAF52 /* TicketingView+Observed.swift in Sources */ = {isa = PBXBuildFile; fileRef = E9171EFF28D15426002FAF52 /* TicketingView+Observed.swift */; }; E94F92C728D2505100D9E759 /* Ticketing.swift in Sources */ = {isa = PBXBuildFile; fileRef = E94F92C628D2505100D9E759 /* Ticketing.swift */; }; E9E2A4D828CEC5680016AEFF /* WebView.swift in Sources */ = {isa = PBXBuildFile; fileRef = E9E2A4D728CEC5680016AEFF /* WebView.swift */; }; + FBCCC4F32ACAB5BD00FFF26B /* SDWebImageSwiftUI in Frameworks */ = {isa = PBXBuildFile; productRef = FBCCC4F22ACAB5BD00FFF26B /* SDWebImageSwiftUI */; }; + FBF437072AC9562B00B22B05 /* WidgetKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = FBF437062AC9562B00B22B05 /* WidgetKit.framework */; }; + FBF437092AC9562B00B22B05 /* SwiftUI.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = FBF437082AC9562B00B22B05 /* SwiftUI.framework */; }; + FBF4370C2AC9562B00B22B05 /* AsyncSwiftWidgetBundle.swift in Sources */ = {isa = PBXBuildFile; fileRef = FBF4370B2AC9562B00B22B05 /* AsyncSwiftWidgetBundle.swift */; }; + FBF4370E2AC9562B00B22B05 /* AsyncSwiftWidget.swift in Sources */ = {isa = PBXBuildFile; fileRef = FBF4370D2AC9562B00B22B05 /* AsyncSwiftWidget.swift */; }; + FBF437102AC9562B00B22B05 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = FBF4370F2AC9562B00B22B05 /* Assets.xcassets */; }; + FBF437142AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = FBF437052AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; + FBF4371B2AC9588800B22B05 /* AsyncSwiftWidgetEntryView.swift in Sources */ = {isa = PBXBuildFile; fileRef = FBF437192AC9586700B22B05 /* AsyncSwiftWidgetEntryView.swift */; }; + FBF4371F2AC95C3200B22B05 /* SVGKit in Frameworks */ = {isa = PBXBuildFile; productRef = FBF4371E2AC95C3200B22B05 /* SVGKit */; }; + FBF437212AC95C5800B22B05 /* SVGKit in Frameworks */ = {isa = PBXBuildFile; productRef = FBF437202AC95C5800B22B05 /* SVGKit */; }; /* End PBXBuildFile section */ +/* Begin PBXContainerItemProxy section */ + FBF437122AC9562B00B22B05 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = C68DE92A28C7685800CA4CC8 /* Project object */; + proxyType = 1; + remoteGlobalIDString = FBF437042AC9562B00B22B05; + remoteInfo = AsyncSwiftWidgetExtension; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + FBF437152AC9562C00B22B05 /* Embed Foundation Extensions */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 13; + files = ( + FBF437142AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex in Embed Foundation Extensions */, + ); + name = "Embed Foundation Extensions"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + /* Begin PBXFileReference section */ B289943228CA69FF002B9F67 /* StampView+Observed.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "StampView+Observed.swift"; sourceTree = ""; }; B2FC6F6228D309FF00D2ACBF /* Stamp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Stamp.swift; sourceTree = ""; }; @@ -71,7 +104,6 @@ C63D444F291BDD2B005D5AE6 /* String+.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "String+.swift"; sourceTree = ""; }; C66C68D228D1B00A0091F960 /* EventModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EventModel.swift; sourceTree = ""; }; C66C68D428D1B0130091F960 /* SessionModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SessionModel.swift; sourceTree = ""; }; - C66DAD4F28CF478700195DEB /* SessionView+Observed.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SessionView+Observed.swift"; sourceTree = ""; }; C66E3D94290BA48500097BEA /* ProfileRegisterView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ProfileRegisterView.swift; sourceTree = ""; }; C66E3D96290BA4FC00097BEA /* ProfileRegisterViewObserved.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ProfileRegisterViewObserved.swift; sourceTree = ""; }; C66E3D98290BB9C100097BEA /* TextEditor+.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "TextEditor+.swift"; sourceTree = ""; }; @@ -106,6 +138,15 @@ E9171EFF28D15426002FAF52 /* TicketingView+Observed.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "TicketingView+Observed.swift"; sourceTree = ""; }; E94F92C628D2505100D9E759 /* Ticketing.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Ticketing.swift; sourceTree = ""; }; E9E2A4D728CEC5680016AEFF /* WebView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WebView.swift; sourceTree = ""; }; + FBF437052AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = AsyncSwiftWidgetExtension.appex; sourceTree = BUILT_PRODUCTS_DIR; }; + FBF437062AC9562B00B22B05 /* WidgetKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = WidgetKit.framework; path = System/Library/Frameworks/WidgetKit.framework; sourceTree = SDKROOT; }; + FBF437082AC9562B00B22B05 /* SwiftUI.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SwiftUI.framework; path = System/Library/Frameworks/SwiftUI.framework; sourceTree = SDKROOT; }; + FBF4370B2AC9562B00B22B05 /* AsyncSwiftWidgetBundle.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AsyncSwiftWidgetBundle.swift; sourceTree = ""; }; + FBF4370D2AC9562B00B22B05 /* AsyncSwiftWidget.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AsyncSwiftWidget.swift; sourceTree = ""; }; + FBF4370F2AC9562B00B22B05 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; + FBF437112AC9562B00B22B05 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; + FBF437192AC9586700B22B05 /* AsyncSwiftWidgetEntryView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AsyncSwiftWidgetEntryView.swift; sourceTree = ""; }; + FBF4371C2AC959A000B22B05 /* AsyncSwiftWidgetExtension.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = AsyncSwiftWidgetExtension.entitlements; sourceTree = ""; }; /* End PBXFileReference section */ /* Begin PBXFrameworksBuildPhase section */ @@ -113,6 +154,8 @@ isa = PBXFrameworksBuildPhase; buildActionMask = 2147483647; files = ( + FBCCC4F32ACAB5BD00FFF26B /* SDWebImageSwiftUI in Frameworks */, + FBF4371F2AC95C3200B22B05 /* SVGKit in Frameworks */, C69C13B32913B08F00D9B47F /* FirebaseDatabase in Frameworks */, C69C13B52913B09500D9B47F /* FirebaseFirestore in Frameworks */, C68DE94728C76BC500CA4CC8 /* FirebaseMessaging in Frameworks */, @@ -121,13 +164,25 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + FBF437022AC9562B00B22B05 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + FBF437212AC95C5800B22B05 /* SVGKit in Frameworks */, + FBF437092AC9562B00B22B05 /* SwiftUI.framework in Frameworks */, + FBF437072AC9562B00B22B05 /* WidgetKit.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXFrameworksBuildPhase section */ /* Begin PBXGroup section */ C68DE92928C7685800CA4CC8 = { isa = PBXGroup; children = ( + FBF4371C2AC959A000B22B05 /* AsyncSwiftWidgetExtension.entitlements */, C68DE93428C7685800CA4CC8 /* AsyncSwift */, + FBF4370A2AC9562B00B22B05 /* AsyncSwiftWidget */, C68DE93328C7685800CA4CC8 /* Products */, C69C13B12913B08F00D9B47F /* Frameworks */, ); @@ -137,6 +192,7 @@ isa = PBXGroup; children = ( C68DE93228C7685800CA4CC8 /* AsyncSwift.app */, + FBF437052AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex */, ); name = Products; sourceTree = ""; @@ -206,6 +262,8 @@ C69C13B12913B08F00D9B47F /* Frameworks */ = { isa = PBXGroup; children = ( + FBF437062AC9562B00B22B05 /* WidgetKit.framework */, + FBF437082AC9562B00B22B05 /* SwiftUI.framework */, ); name = Frameworks; sourceTree = ""; @@ -257,7 +315,6 @@ C63D444D291BDD09005D5AE6 /* MainTabView+Observed.swift */, C6F7798A28C9CBC60036773B /* EventView+Observed.swift */, C63A865E28CA70ED0064C417 /* EventDetailView+Observed.swift */, - C66DAD4F28CF478700195DEB /* SessionView+Observed.swift */, B289943228CA69FF002B9F67 /* StampView+Observed.swift */, E9171EFF28D15426002FAF52 /* TicketingView+Observed.swift */, C69C13BD2913EC3200D9B47F /* ProfileView */, @@ -265,6 +322,18 @@ path = Observed; sourceTree = ""; }; + FBF4370A2AC9562B00B22B05 /* AsyncSwiftWidget */ = { + isa = PBXGroup; + children = ( + FBF4370B2AC9562B00B22B05 /* AsyncSwiftWidgetBundle.swift */, + FBF4370D2AC9562B00B22B05 /* AsyncSwiftWidget.swift */, + FBF437192AC9586700B22B05 /* AsyncSwiftWidgetEntryView.swift */, + FBF4370F2AC9562B00B22B05 /* Assets.xcassets */, + FBF437112AC9562B00B22B05 /* Info.plist */, + ); + path = AsyncSwiftWidget; + sourceTree = ""; + }; /* End PBXGroup section */ /* Begin PBXNativeTarget section */ @@ -275,10 +344,12 @@ C68DE92E28C7685800CA4CC8 /* Sources */, C68DE92F28C7685800CA4CC8 /* Frameworks */, C68DE93028C7685800CA4CC8 /* Resources */, + FBF437152AC9562C00B22B05 /* Embed Foundation Extensions */, ); buildRules = ( ); dependencies = ( + FBF437132AC9562B00B22B05 /* PBXTargetDependency */, ); name = AsyncSwift; packageProductDependencies = ( @@ -287,11 +358,33 @@ C69C13B22913B08F00D9B47F /* FirebaseDatabase */, C69C13B42913B09500D9B47F /* FirebaseFirestore */, C69C13C5291425F200D9B47F /* CodeScanner */, + FBF4371E2AC95C3200B22B05 /* SVGKit */, + FBCCC4F22ACAB5BD00FFF26B /* SDWebImageSwiftUI */, ); productName = AsyncSwift; productReference = C68DE93228C7685800CA4CC8 /* AsyncSwift.app */; productType = "com.apple.product-type.application"; }; + FBF437042AC9562B00B22B05 /* AsyncSwiftWidgetExtension */ = { + isa = PBXNativeTarget; + buildConfigurationList = FBF437182AC9562C00B22B05 /* Build configuration list for PBXNativeTarget "AsyncSwiftWidgetExtension" */; + buildPhases = ( + FBF437012AC9562B00B22B05 /* Sources */, + FBF437022AC9562B00B22B05 /* Frameworks */, + FBF437032AC9562B00B22B05 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + ); + name = AsyncSwiftWidgetExtension; + packageProductDependencies = ( + FBF437202AC95C5800B22B05 /* SVGKit */, + ); + productName = AsyncSwiftWidgetExtension; + productReference = FBF437052AC9562B00B22B05 /* AsyncSwiftWidgetExtension.appex */; + productType = "com.apple.product-type.app-extension"; + }; /* End PBXNativeTarget section */ /* Begin PBXProject section */ @@ -299,12 +392,15 @@ isa = PBXProject; attributes = { BuildIndependentTargetsInParallel = 1; - LastSwiftUpdateCheck = 1340; + LastSwiftUpdateCheck = 1430; LastUpgradeCheck = 1340; TargetAttributes = { C68DE93128C7685800CA4CC8 = { CreatedOnToolsVersion = 13.4.1; }; + FBF437042AC9562B00B22B05 = { + CreatedOnToolsVersion = 14.3.1; + }; }; }; buildConfigurationList = C68DE92D28C7685800CA4CC8 /* Build configuration list for PBXProject "AsyncSwift" */; @@ -319,12 +415,15 @@ packageReferences = ( C68DE94328C76BC500CA4CC8 /* XCRemoteSwiftPackageReference "firebase-ios-sdk" */, C69C13C4291425F200D9B47F /* XCRemoteSwiftPackageReference "CodeScanner" */, + FBF4371D2AC95C3200B22B05 /* XCRemoteSwiftPackageReference "SVGKit" */, + FBCCC4F12ACAB5BD00FFF26B /* XCRemoteSwiftPackageReference "SDWebImageSwiftUI" */, ); productRefGroup = C68DE93328C7685800CA4CC8 /* Products */; projectDirPath = ""; projectRoot = ""; targets = ( C68DE93128C7685800CA4CC8 /* AsyncSwift */, + FBF437042AC9562B00B22B05 /* AsyncSwiftWidgetExtension */, ); }; /* End PBXProject section */ @@ -340,6 +439,14 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + FBF437032AC9562B00B22B05 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + FBF437102AC9562B00B22B05 /* Assets.xcassets in Resources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXResourcesBuildPhase section */ /* Begin PBXSourcesBuildPhase section */ @@ -354,7 +461,6 @@ C63D4450291BDD2B005D5AE6 /* String+.swift in Sources */, C63D444C291BDCDC005D5AE6 /* KeyChainManager.swift in Sources */, C631EBC3291537E300A54143 /* SafariView.swift in Sources */, - C66DAD5028CF478700195DEB /* SessionView+Observed.swift in Sources */, C69C13A22912868F00D9B47F /* ProfileFriendDetailView.swift in Sources */, C6E744A028CA557100B7B2BD /* Color+.swift in Sources */, C68DE95128C77DDA00CA4CC8 /* TicketingView.swift in Sources */, @@ -391,8 +497,26 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + FBF437012AC9562B00B22B05 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + FBF4370E2AC9562B00B22B05 /* AsyncSwiftWidget.swift in Sources */, + FBF4370C2AC9562B00B22B05 /* AsyncSwiftWidgetBundle.swift in Sources */, + FBF4371B2AC9588800B22B05 /* AsyncSwiftWidgetEntryView.swift in Sources */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXSourcesBuildPhase section */ +/* Begin PBXTargetDependency section */ + FBF437132AC9562B00B22B05 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = FBF437042AC9562B00B22B05 /* AsyncSwiftWidgetExtension */; + targetProxy = FBF437122AC9562B00B22B05 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + /* Begin XCBuildConfiguration section */ C68DE93E28C7685900CA4CC8 /* Debug */ = { isa = XCBuildConfiguration; @@ -511,6 +635,7 @@ C68DE94128C7685900CA4CC8 /* Debug */ = { isa = XCBuildConfiguration; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; CODE_SIGN_ENTITLEMENTS = AsyncSwift/AsyncSwift.entitlements; @@ -537,7 +662,7 @@ "$(inherited)", "@executable_path/Frameworks", ); - MARKETING_VERSION = 2.0.2; + MARKETING_VERSION = 2.0.3; PRODUCT_BUNDLE_IDENTIFIER = com.kim.AsyncSwift; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -553,6 +678,7 @@ C68DE94228C7685900CA4CC8 /* Release */ = { isa = XCBuildConfiguration; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; CODE_SIGN_ENTITLEMENTS = AsyncSwift/AsyncSwift.entitlements; @@ -579,7 +705,7 @@ "$(inherited)", "@executable_path/Frameworks", ); - MARKETING_VERSION = 2.0.2; + MARKETING_VERSION = 2.0.3; PRODUCT_BUNDLE_IDENTIFIER = com.kim.AsyncSwift; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -592,6 +718,68 @@ }; name = Release; }; + FBF437162AC9562C00B22B05 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = WidgetBackground; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CODE_SIGN_ENTITLEMENTS = AsyncSwiftWidgetExtension.entitlements; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 76AJ433CP5; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = AsyncSwiftWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = AsyncSwiftWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 15.0; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.kim.AsyncSwift.AsyncSwiftWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Debug; + }; + FBF437172AC9562C00B22B05 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = WidgetBackground; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CODE_SIGN_ENTITLEMENTS = AsyncSwiftWidgetExtension.entitlements; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 76AJ433CP5; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = AsyncSwiftWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = AsyncSwiftWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 15.0; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + MARKETING_VERSION = 1.0; + PRODUCT_BUNDLE_IDENTIFIER = com.kim.AsyncSwift.AsyncSwiftWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = 1; + }; + name = Release; + }; /* End XCBuildConfiguration section */ /* Begin XCConfigurationList section */ @@ -613,6 +801,15 @@ defaultConfigurationIsVisible = 0; defaultConfigurationName = Release; }; + FBF437182AC9562C00B22B05 /* Build configuration list for PBXNativeTarget "AsyncSwiftWidgetExtension" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + FBF437162AC9562C00B22B05 /* Debug */, + FBF437172AC9562C00B22B05 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; /* End XCConfigurationList section */ /* Begin XCRemoteSwiftPackageReference section */ @@ -632,6 +829,22 @@ minimumVersion = 2.0.0; }; }; + FBCCC4F12ACAB5BD00FFF26B /* XCRemoteSwiftPackageReference "SDWebImageSwiftUI" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/SDWebImage/SDWebImageSwiftUI.git"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 2.0.0; + }; + }; + FBF4371D2AC95C3200B22B05 /* XCRemoteSwiftPackageReference "SVGKit" */ = { + isa = XCRemoteSwiftPackageReference; + repositoryURL = "https://github.com/SVGKit/SVGKit.git"; + requirement = { + kind = upToNextMajorVersion; + minimumVersion = 3.0.0; + }; + }; /* End XCRemoteSwiftPackageReference section */ /* Begin XCSwiftPackageProductDependency section */ @@ -660,6 +873,21 @@ package = C69C13C4291425F200D9B47F /* XCRemoteSwiftPackageReference "CodeScanner" */; productName = CodeScanner; }; + FBCCC4F22ACAB5BD00FFF26B /* SDWebImageSwiftUI */ = { + isa = XCSwiftPackageProductDependency; + package = FBCCC4F12ACAB5BD00FFF26B /* XCRemoteSwiftPackageReference "SDWebImageSwiftUI" */; + productName = SDWebImageSwiftUI; + }; + FBF4371E2AC95C3200B22B05 /* SVGKit */ = { + isa = XCSwiftPackageProductDependency; + package = FBF4371D2AC95C3200B22B05 /* XCRemoteSwiftPackageReference "SVGKit" */; + productName = SVGKit; + }; + FBF437202AC95C5800B22B05 /* SVGKit */ = { + isa = XCSwiftPackageProductDependency; + package = FBF4371D2AC95C3200B22B05 /* XCRemoteSwiftPackageReference "SVGKit" */; + productName = SVGKit; + }; /* End XCSwiftPackageProductDependency section */ }; rootObject = C68DE92A28C7685800CA4CC8 /* Project object */; diff --git a/AsyncSwift.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/AsyncSwift.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index 21366cd..1ca5192 100644 --- a/AsyncSwift.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/AsyncSwift.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -18,6 +18,15 @@ "version" : "0.9.1" } }, + { + "identity" : "cocoalumberjack", + "kind" : "remoteSourceControl", + "location" : "https://github.com/CocoaLumberjack/CocoaLumberjack.git", + "state" : { + "revision" : "67ec5818a757aba4d7c534e21a905d878d128dbf", + "version" : "3.8.1" + } + }, { "identity" : "codescanner", "kind" : "remoteSourceControl", @@ -108,6 +117,42 @@ "version" : "2.1.1" } }, + { + "identity" : "sdwebimage", + "kind" : "remoteSourceControl", + "location" : "https://github.com/SDWebImage/SDWebImage.git", + "state" : { + "revision" : "936f1c7067728d16c362ba4fb93c17df78b5fd79", + "version" : "5.18.2" + } + }, + { + "identity" : "sdwebimageswiftui", + "kind" : "remoteSourceControl", + "location" : "https://github.com/SDWebImage/SDWebImageSwiftUI.git", + "state" : { + "revision" : "e837c37d45449fbd3b4745c10c5b5274e73edead", + "version" : "2.2.3" + } + }, + { + "identity" : "svgkit", + "kind" : "remoteSourceControl", + "location" : "https://github.com/SVGKit/SVGKit.git", + "state" : { + "revision" : "58152b9f7c85eab239160b36ffdfd364aa43d666", + "version" : "3.0.0" + } + }, + { + "identity" : "swift-log", + "kind" : "remoteSourceControl", + "location" : "https://github.com/apple/swift-log.git", + "state" : { + "revision" : "532d8b529501fb73a2455b179e0bbb6d49b652ed", + "version" : "1.5.3" + } + }, { "identity" : "swift-protobuf", "kind" : "remoteSourceControl", diff --git a/AsyncSwift/AsyncSwift.entitlements b/AsyncSwift/AsyncSwift.entitlements index 3575a32..9f562cf 100644 --- a/AsyncSwift/AsyncSwift.entitlements +++ b/AsyncSwift/AsyncSwift.entitlements @@ -8,5 +8,9 @@ applinks:asyncswift.info + com.apple.security.application-groups + + group.com.kim.AsyncSwift + diff --git a/AsyncSwift/Observed/EventDetailView+Observed.swift b/AsyncSwift/Observed/EventDetailView+Observed.swift index d916f60..c120520 100644 --- a/AsyncSwift/Observed/EventDetailView+Observed.swift +++ b/AsyncSwift/Observed/EventDetailView+Observed.swift @@ -9,28 +9,27 @@ import EventKit import SwiftUI extension EventDetailView { + final class Observed: ObservableObject { init(event: Event) { self.event = event } - @Published var event: Event + let event: Event @Published var isShowingSheet = false @Published var isShowingAddEventConfirmationAlert = false @Published var isShowingAddEventSuccessAlert = false @Published var isShowingAddEventFailureAlert = false func additionConfirmed() { - addEventOnCalendar { isSuccess in - DispatchQueue.main.async { [weak self] in - if let self = self { - switch isSuccess { - case true: - self.isShowingAddEventSuccessAlert = true - case false: - self.isShowingAddEventFailureAlert = true - } + addEventOnCalendar { [weak self] isSuccess in + DispatchQueue.main.async { + switch isSuccess { + case true: + self?.isShowingAddEventSuccessAlert = true + case false: + self?.isShowingAddEventFailureAlert = true } } } @@ -39,11 +38,8 @@ extension EventDetailView { func addEventOnCalendar(completion: @escaping ((Bool) -> Void) ) { let eventStore = EKEventStore() - eventStore.requestAccess(to: .event) { (granted, error) in - if let error = error { - print("failed to save event with error : \(error) or access not granted") - return - } + eventStore.requestAccess(to: .event) { [weak self] (granted, error) in + guard let self, error == nil else { return } let event = EKEvent(eventStore: eventStore) let formatter = DateFormatter.calendarFormatter event.title = self.event.title diff --git a/AsyncSwift/Observed/EventView+Observed.swift b/AsyncSwift/Observed/EventView+Observed.swift index 21f00e9..9e0a2ae 100644 --- a/AsyncSwift/Observed/EventView+Observed.swift +++ b/AsyncSwift/Observed/EventView+Observed.swift @@ -6,6 +6,7 @@ // import SwiftUI +import Combine final class EventViewObserved: ObservableObject { @@ -13,35 +14,23 @@ final class EventViewObserved: ObservableObject { @Published var eventStatus: EventStatus = .upcoming @Published var isLoading = true let onLoadingCells = Array(repeating: [0], count: 6) - - init() { - self.fetchJson { - self.calculateEventStatus() - self.isLoading = false - } - } - - func fetchJson(completion: @escaping () -> Void) { - guard let url = URL(string: "https://async-swift.github.io/jsonstorage/asyncswift.json") else { return } - let request = URLRequest(url: url) - let dataTask = URLSession.shared.dataTask(with: request) { data, response, _ in - guard - let response = response as? HTTPURLResponse, - response.statusCode == 200, - let data = data - else { return } - DispatchQueue.main.async { [weak self] in - guard let self = self else { return } - do { - let decodedData = try JSONDecoder().decode(Event.self, from: data) - self.event = decodedData - completion() - } catch let error { - print("❌ \(error.localizedDescription)") - } - } - } - dataTask.resume() + var cancellable = Set() + + func getEventData() { + let urlString = "https://async-swift.github.io/jsonstorage/asyncswift.json" + let url = URL(string: urlString)! + URLSession.shared.dataTaskPublisher(for: url) + .map(\.data) + .decode(type: Event.self, decoder: JSONDecoder()) + .receive(on: RunLoop.main) + .sink { _ in + + } receiveValue: { [weak self] event in + self?.event = event + self?.calculateEventStatus() + self?.isLoading = false + } + .store(in: &cancellable) } func calculateEventStatus() { diff --git a/AsyncSwift/Observed/SessionView+Observed.swift b/AsyncSwift/Observed/SessionView+Observed.swift deleted file mode 100644 index efc63e7..0000000 --- a/AsyncSwift/Observed/SessionView+Observed.swift +++ /dev/null @@ -1,20 +0,0 @@ -// -// SessionView+Observed.swift -// AsyncSwift -// -// Created by Kim Insub on 2022/09/12. -// - -import SwiftUI - -extension SessionView { - final class Observed: ObservableObject { - - init(session: Session) { - self.session = session - } - - @Published var session: Session - let speakerImageSize: CGFloat = 80 - } -} diff --git a/AsyncSwift/Observed/StampView+Observed.swift b/AsyncSwift/Observed/StampView+Observed.swift index aa7229a..d35bec6 100644 --- a/AsyncSwift/Observed/StampView+Observed.swift +++ b/AsyncSwift/Observed/StampView+Observed.swift @@ -6,9 +6,10 @@ // import SwiftUI +import Combine extension StampView { - @MainActor final class Observed: ObservableObject { + final class Observed: ObservableObject { @Published var cards: [Card] = [] @Published var events = [String]() @Published var currentIndex = 0 @@ -16,6 +17,7 @@ extension StampView { private let keyChainManager = KeyChainManager() private let cardInterval: CGFloat = (UIScreen.main.bounds.width - 32) * 56 / 358 private let cardSize: CGFloat = UIScreen.main.bounds.width - 32 + private var cancenllable = Set() init() { fetchStampsImages() @@ -30,38 +32,41 @@ extension StampView { /// Storage에 저장되어 있는 Stamp Image를 가져오는 함수이다. /// - - private func fetchStampsImages(){ + private func fetchStampsImages() { + let events = getEvents() - - guard !events.isEmpty else { - isLoading = false - return - } - + guard !events.isEmpty else { return isLoading = false } + events.enumerated().forEach { [weak self] in guard let self else { return } let event = $0.element let index = $0.offset - Task { @MainActor () -> Void in - guard let cardImageURL = URL(string: "https://raw.githubusercontent.com/Async-Swift/jsonstorage/main/Images/Stamp/" + event + "/stamp.png") - else { return } - - let cardImageRequest = URLRequest(url: cardImageURL) - let (cardImageData, cardImageResponse) = try await URLSession.shared.data(for: cardImageRequest) - guard let httpsResponse = cardImageResponse as? HTTPURLResponse, httpsResponse.statusCode == 200 else { return } - - guard let cardUIImage = UIImage(data: cardImageData) else { return } - let card = Card( - originalPosition: self.cardInterval * CGFloat(index), - image: Image(uiImage: cardUIImage), - event: event - ) - self.cards.append(card) - if index == events.count - 1 { - self.isLoading = false + let urlString = "https://raw.githubusercontent.com/Async-Swift/jsonstorage/main/Images/Stamp/" + event + "/stamp.png" + let url = URL(string: urlString)! + + URLSession.shared.dataTaskPublisher(for: url) + .map(\.data) + .tryMap { + guard let image = UIImage(data: $0) else { + throw URLError(.badURL) + } + return Card( + originalPosition: self.cardInterval * CGFloat(index), + image: Image(uiImage: image), + event: event + ) } - } + .receive(on: RunLoop.main) + .sink(receiveCompletion: { _ in + + }, receiveValue: { [weak self] card in + self?.cards.append(card) + if index == events.count - 1 { + self?.isLoading = false + } + }) + .store(in: &cancenllable) } } diff --git a/AsyncSwift/Observed/TicketingView+Observed.swift b/AsyncSwift/Observed/TicketingView+Observed.swift index 3f15a1d..4de6fc8 100644 --- a/AsyncSwift/Observed/TicketingView+Observed.swift +++ b/AsyncSwift/Observed/TicketingView+Observed.swift @@ -11,8 +11,8 @@ import Foundation extension TicketingView { final class Observed: ObservableObject { @Published var ticketing: Ticketing? - @Published var isActivatedWebViewNavigationLink = false + var cancellable = Set() var hasAvailableTicket: Bool { let currentDate = Date() @@ -22,32 +22,20 @@ extension TicketingView { var isTicketingLinkDisabled: Bool { ticketing?.currentTicket?.ticketingURL == nil && !hasAvailableTicket } - - func onAppear() { - guard - let url = URL(string: "https://raw.githubusercontent.com/Async-Swift/jsonstorage/main/ticketing.json") - else { return } - - - let request = URLRequest(url: url) - let dataTask = URLSession.shared.dataTask(with: request) { data, response, _ in - guard - let response = response as? HTTPURLResponse, - response.statusCode == 200, - let data = data - else { return } - - DispatchQueue.main.async { [weak self] in - do { - let ticketing = try JSONDecoder().decode(Ticketing.self, from: data) - self?.ticketing = ticketing - } catch { - self?.ticketing = nil - } - } - } - - dataTask.resume() + + func getTicketingData() { + let urlString = "https://raw.githubusercontent.com/Async-Swift/jsonstorage/main/ticketing.json" + let url = URL(string: urlString)! + URLSession.shared.dataTaskPublisher(for: url) + .map(\.data) + .decode(type: Ticketing.self, decoder: JSONDecoder()) + .receive(on: RunLoop.main) + .sink { _ in + + } receiveValue: { [weak self] event in + self?.ticketing = event + } + .store(in: &cancellable) } func didTappedTicketingButton() { diff --git a/AsyncSwift/Views/EventView.swift b/AsyncSwift/Views/EventView.swift index 97af5fc..363161f 100644 --- a/AsyncSwift/Views/EventView.swift +++ b/AsyncSwift/Views/EventView.swift @@ -33,6 +33,7 @@ struct EventView: View { } } .navigationTitle(Tab.event.title) + .onAppear { observed.getEventData() } } } } diff --git a/AsyncSwift/Views/SessionView.swift b/AsyncSwift/Views/SessionView.swift index 456fdc0..960ac68 100644 --- a/AsyncSwift/Views/SessionView.swift +++ b/AsyncSwift/Views/SessionView.swift @@ -6,15 +6,13 @@ // import SwiftUI +import SDWebImageSwiftUI struct SessionView: View { - @ObservedObject var observed: Observed - - init(session: Session) { - observed = Observed(session: session) - } - + let session: Session + let speakerImageSize: CGFloat = 80 + var body: some View { ZStack { ScrollView { @@ -37,12 +35,12 @@ private extension SessionView { var sessionDetail: some View { HStack(spacing: 0) { VStack(alignment: .leading, spacing: 0) { - Text(observed.session.title) + Text(session.title) .font(.title3) .fontWeight(.semibold) .padding(.vertical, 24) VStack(alignment: .leading, spacing: 8) { - ForEach(observed.session.description, id: \.self) { paragraph in + ForEach(session.description, id: \.self) { paragraph in Text(paragraph.content) } } @@ -57,31 +55,28 @@ private extension SessionView { HStack(spacing: 0) { VStack(alignment: .leading, spacing: 4) { - AsyncImage(url: URL(string: observed.session.speaker.imageURL), transaction: Transaction(animation: .default)) { phase in - if let image = phase.image { - image - .resizable() - } else if phase.error != nil { - Image(systemName: "person.crop.circle.fill") - .resizable() - .opacity(0.04) - } else { + + WebImage(url: URL(string: session.speaker.imageURL)) + .resizable() + .placeholder { Image(systemName: "person.crop.circle.fill") .resizable() + .frame(width: speakerImageSize, height: speakerImageSize) .opacity(0.04) } - } - .aspectRatio(contentMode: .fit) - .frame(width: observed.speakerImageSize, height: observed.speakerImageSize) - .clipShape(Circle()) - .padding(.vertical, 24) + .transition(.fade) + .scaledToFit() + .frame(width: speakerImageSize, height: speakerImageSize) + .clipShape(Circle()) + .padding(.vertical, 24) + VStack(alignment: .leading, spacing: 2) { - Text("\(observed.session.speaker.name) 님") + Text("\(session.speaker.name) 님") .font(.headline) - Text(observed.session.speaker.role) + Text(session.speaker.role) .font(.caption2) } - Text(observed.session.speaker.description) + Text(session.speaker.description) .font(.footnote) } .padding(.horizontal, 32) diff --git a/AsyncSwift/Views/TicketingView.swift b/AsyncSwift/Views/TicketingView.swift index 3ed232e..a041ebf 100644 --- a/AsyncSwift/Views/TicketingView.swift +++ b/AsyncSwift/Views/TicketingView.swift @@ -6,11 +6,13 @@ // import SwiftUI +import SDWebImageSwiftUI struct TicketingView: View { @StateObject private var observed = Observed() var body: some View { + NavigationView { ScrollView { VStack(spacing: 30) { @@ -37,9 +39,8 @@ struct TicketingView: View { } .navigationTitle("Ticketing") - }.onAppear { - observed.onAppear() } + .onAppear { observed.getTicketingData() } } } @@ -72,28 +73,21 @@ private extension TicketingView { .animation(.linear(duration: 3.0), value: 1.0) } - var ticketingView: some View { - NavigationLink( - isActive: $observed.isActivatedWebViewNavigationLink - ) { - if let upcomingEventURL = observed.ticketing?.currentTicket?.ticketingURL { - WebView(url: upcomingEventURL) + @ViewBuilder var ticketingView: some View { + + if let url = URL(string: observed.ticketing?.currentTicket?.ticketingURL ?? "") { + Link(destination: url) { + WebImage(url: URL(string: observed.ticketing?.currentTicket?.ticketingImageURL ?? "")) + .resizable() + .placeholder { + skeletonView + .aspectRatio(0.85, contentMode: .fill) + } + .scaledToFill() + .transition(.opacity.animation(.easeOut)) } - } label: { - AsyncImage( - url: URL(string: observed.ticketing?.currentTicket?.ticketingImageURL ?? ""), - content: { image in - image - .resizable() - .aspectRatio(contentMode: .fill) - .transition(AnyTransition.opacity.animation(.easeInOut)) - }, - placeholder: { - skeletonView - .aspectRatio(0.85, contentMode: .fill) - } - ) - }.disabled(observed.isTicketingLinkDisabled) + .disabled(observed.isTicketingLinkDisabled) + } } var emptyTicketingView: some View { diff --git a/AsyncSwiftWidget/Assets.xcassets/AccentColor.colorset/Contents.json b/AsyncSwiftWidget/Assets.xcassets/AccentColor.colorset/Contents.json new file mode 100644 index 0000000..eb87897 --- /dev/null +++ b/AsyncSwiftWidget/Assets.xcassets/AccentColor.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/AsyncSwiftWidget/Assets.xcassets/AppIcon.appiconset/Contents.json b/AsyncSwiftWidget/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 0000000..13613e3 --- /dev/null +++ b/AsyncSwiftWidget/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,13 @@ +{ + "images" : [ + { + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/AsyncSwiftWidget/Assets.xcassets/Contents.json b/AsyncSwiftWidget/Assets.xcassets/Contents.json new file mode 100644 index 0000000..73c0059 --- /dev/null +++ b/AsyncSwiftWidget/Assets.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/AsyncSwiftWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json b/AsyncSwiftWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json new file mode 100644 index 0000000..eb87897 --- /dev/null +++ b/AsyncSwiftWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/AsyncSwiftWidget/AsyncSwiftWidget.swift b/AsyncSwiftWidget/AsyncSwiftWidget.swift new file mode 100644 index 0000000..f164114 --- /dev/null +++ b/AsyncSwiftWidget/AsyncSwiftWidget.swift @@ -0,0 +1,60 @@ +// +// AsyncSwiftWidget.swift +// AsyncSwiftWidget +// +// Created by 김인섭 on 2023/10/01. +// + +import WidgetKit +import SwiftUI + +struct Provider: TimelineProvider { + func placeholder(in context: Context) -> SimpleEntry { + SimpleEntry(date: Date(), imageData: getRemoteImage()) + } + + func getSnapshot(in context: Context, completion: @escaping (SimpleEntry) -> ()) { + let entry = SimpleEntry(date: Date(), imageData: getRemoteImage()) + completion(entry) + } + + func getTimeline(in context: Context, completion: @escaping (Timeline) -> ()) { + + let entry = SimpleEntry(date: Date(), imageData: getRemoteImage()) + let nextUpdate = Calendar.current.date(byAdding: .minute, value: 15, to: Date()) + let timeline = Timeline(entries: [entry], policy: .after(nextUpdate!)) + completion(timeline) + } + + func getRemoteImage() -> Data? { + let urlString = "https://raw.githubusercontent.com/Async-Swift/jsonstorage/a94eb982e9b9db90543de1d574a6dc5f0c637f5b/Images/widget-large.svg" + return try? Data(contentsOf: URL(string: urlString)!) + } +} + +struct SimpleEntry: TimelineEntry { + var date: Date + let imageData: Data? +} + +struct AsyncSwiftWidget: Widget { + let kind: String = "AsyncSwiftWidget" + + var body: some WidgetConfiguration { + StaticConfiguration(kind: kind, provider: Provider()) { entry in + AsyncSwiftWidgetEntryView(entry: entry) + } + .configurationDisplayName("AsyncSwift") + .description("행사 정보를 확인하세요.") + .supportedFamilies( + [.systemLarge] + ) + } +} + +struct AsyncSwiftWidget_Previews: PreviewProvider { + static var previews: some View { + AsyncSwiftWidgetEntryView(entry: SimpleEntry(date: Date(), imageData: nil)) + .previewContext(WidgetPreviewContext(family: .systemSmall)) + } +} diff --git a/AsyncSwiftWidget/AsyncSwiftWidgetBundle.swift b/AsyncSwiftWidget/AsyncSwiftWidgetBundle.swift new file mode 100644 index 0000000..5b8cf8d --- /dev/null +++ b/AsyncSwiftWidget/AsyncSwiftWidgetBundle.swift @@ -0,0 +1,16 @@ +// +// AsyncSwiftWidgetBundle.swift +// AsyncSwiftWidget +// +// Created by 김인섭 on 2023/10/01. +// + +import WidgetKit +import SwiftUI + +@main +struct AsyncSwiftWidgetBundle: WidgetBundle { + var body: some Widget { + AsyncSwiftWidget() + } +} diff --git a/AsyncSwiftWidget/AsyncSwiftWidgetEntryView.swift b/AsyncSwiftWidget/AsyncSwiftWidgetEntryView.swift new file mode 100644 index 0000000..b98cb7d --- /dev/null +++ b/AsyncSwiftWidget/AsyncSwiftWidgetEntryView.swift @@ -0,0 +1,25 @@ +// +// AsyncSwiftWidgetEntryView.swift +// AsyncSwift +// +// Created by 김인섭 on 2023/10/01. +// + +import SwiftUI +import WidgetKit +import SVGKit + +struct AsyncSwiftWidgetEntryView : View { + var entry: Provider.Entry + + var body: some View { + if let imageData = entry.imageData, let image = SVGKImage(data: imageData) { + Image(uiImage: image.uiImage) + .resizable() + .scaledToFill() + .offset(y: 10) + } else { + Text("다음 행사때 만나요. 🤗") + } + } +} diff --git a/AsyncSwiftWidget/Info.plist b/AsyncSwiftWidget/Info.plist new file mode 100644 index 0000000..0f118fb --- /dev/null +++ b/AsyncSwiftWidget/Info.plist @@ -0,0 +1,11 @@ + + + + + NSExtension + + NSExtensionPointIdentifier + com.apple.widgetkit-extension + + + diff --git a/AsyncSwiftWidgetExtension.entitlements b/AsyncSwiftWidgetExtension.entitlements new file mode 100644 index 0000000..04d5f4c --- /dev/null +++ b/AsyncSwiftWidgetExtension.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.com.kim.AsyncSwift + + +