├── .gitignore ├── CombineSample.xcodeproj ├── project.pbxproj ├── project.xcworkspace │ ├── contents.xcworkspacedata │ └── xcshareddata │ │ └── IDEWorkspaceChecks.plist └── xcuserdata │ └── akifumi.fukaya.xcuserdatad │ └── xcschemes │ └── xcschememanagement.plist ├── CombineSample ├── AppDelegate.swift ├── Assets.xcassets │ ├── AppIcon.appiconset │ │ └── Contents.json │ └── Contents.json ├── Base.lproj │ └── LaunchScreen.storyboard ├── ContentView.swift ├── Info.plist ├── Preview Content │ └── Preview Assets.xcassets │ │ └── Contents.json └── SceneDelegate.swift ├── CombineSampleTests ├── CombineSampleTests.swift └── Info.plist ├── CombineSampleUITests ├── CombineSampleUITests.swift └── Info.plist └── README.md /.gitignore: -------------------------------------------------------------------------------- 1 | # Xcode 2 | # 3 | # gitignore contributors: remember to update Global/Xcode.gitignore, Objective-C.gitignore & Swift.gitignore 4 | 5 | ## Build generated 6 | build/ 7 | DerivedData/ 8 | 9 | ## Various settings 10 | *.pbxuser 11 | !default.pbxuser 12 | *.mode1v3 13 | !default.mode1v3 14 | *.mode2v3 15 | !default.mode2v3 16 | *.perspectivev3 17 | !default.perspectivev3 18 | xcuserdata/ 19 | 20 | ## Other 21 | *.moved-aside 22 | *.xccheckout 23 | *.xcscmblueprint 24 | 25 | ## Obj-C/Swift specific 26 | *.hmap 27 | *.ipa 28 | *.dSYM.zip 29 | *.dSYM 30 | 31 | ## Playgrounds 32 | timeline.xctimeline 33 | playground.xcworkspace 34 | 35 | # Swift Package Manager 36 | # 37 | # Add this line if you want to avoid checking in source code from Swift Package Manager dependencies. 38 | # Packages/ 39 | # Package.pins 40 | # Package.resolved 41 | .build/ 42 | 43 | # CocoaPods 44 | # 45 | # We recommend against adding the Pods directory to your .gitignore. However 46 | # you should judge for yourself, the pros and cons are mentioned at: 47 | # https://guides.cocoapods.org/using/using-cocoapods.html#should-i-check-the-pods-directory-into-source-control 48 | # 49 | # Pods/ 50 | 51 | # Carthage 52 | # 53 | # Add this line if you want to avoid checking in source code from Carthage dependencies. 54 | # Carthage/Checkouts 55 | 56 | Carthage/Build 57 | 58 | # fastlane 59 | # 60 | # It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the 61 | # screenshots whenever they are needed. 62 | # For more information about the recommended setup visit: 63 | # https://docs.fastlane.tools/best-practices/source-control/#source-control 64 | 65 | fastlane/report.xml 66 | fastlane/Preview.html 67 | fastlane/screenshots/**/*.png 68 | fastlane/test_output 69 | -------------------------------------------------------------------------------- /CombineSample.xcodeproj/project.pbxproj: -------------------------------------------------------------------------------- 1 | // !$*UTF8*$! 2 | { 3 | archiveVersion = 1; 4 | classes = { 5 | }; 6 | objectVersion = 50; 7 | objects = { 8 | 9 | /* Begin PBXBuildFile section */ 10 | 6EA9C8E922B1E7E20060EF1C /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EA9C8E822B1E7E20060EF1C /* AppDelegate.swift */; }; 11 | 6EA9C8EB22B1E7E20060EF1C /* SceneDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EA9C8EA22B1E7E20060EF1C /* SceneDelegate.swift */; }; 12 | 6EA9C8ED22B1E7E20060EF1C /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EA9C8EC22B1E7E20060EF1C /* ContentView.swift */; }; 13 | 6EA9C8EF22B1E7E30060EF1C /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 6EA9C8EE22B1E7E30060EF1C /* Assets.xcassets */; }; 14 | 6EA9C8F222B1E7E30060EF1C /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 6EA9C8F122B1E7E30060EF1C /* Preview Assets.xcassets */; }; 15 | 6EA9C8F522B1E7E30060EF1C /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 6EA9C8F322B1E7E30060EF1C /* LaunchScreen.storyboard */; }; 16 | 6EA9C90022B1E7E30060EF1C /* CombineSampleTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EA9C8FF22B1E7E30060EF1C /* CombineSampleTests.swift */; }; 17 | 6EA9C90B22B1E7E30060EF1C /* CombineSampleUITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6EA9C90A22B1E7E30060EF1C /* CombineSampleUITests.swift */; }; 18 | /* End PBXBuildFile section */ 19 | 20 | /* Begin PBXContainerItemProxy section */ 21 | 6EA9C8FC22B1E7E30060EF1C /* PBXContainerItemProxy */ = { 22 | isa = PBXContainerItemProxy; 23 | containerPortal = 6EA9C8DD22B1E7E20060EF1C /* Project object */; 24 | proxyType = 1; 25 | remoteGlobalIDString = 6EA9C8E422B1E7E20060EF1C; 26 | remoteInfo = CombineSample; 27 | }; 28 | 6EA9C90722B1E7E30060EF1C /* PBXContainerItemProxy */ = { 29 | isa = PBXContainerItemProxy; 30 | containerPortal = 6EA9C8DD22B1E7E20060EF1C /* Project object */; 31 | proxyType = 1; 32 | remoteGlobalIDString = 6EA9C8E422B1E7E20060EF1C; 33 | remoteInfo = CombineSample; 34 | }; 35 | /* End PBXContainerItemProxy section */ 36 | 37 | /* Begin PBXFileReference section */ 38 | 6EA9C8E522B1E7E20060EF1C /* CombineSample.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = CombineSample.app; sourceTree = BUILT_PRODUCTS_DIR; }; 39 | 6EA9C8E822B1E7E20060EF1C /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; 40 | 6EA9C8EA22B1E7E20060EF1C /* SceneDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SceneDelegate.swift; sourceTree = ""; }; 41 | 6EA9C8EC22B1E7E20060EF1C /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = ""; }; 42 | 6EA9C8EE22B1E7E30060EF1C /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; 43 | 6EA9C8F122B1E7E30060EF1C /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = ""; }; 44 | 6EA9C8F422B1E7E30060EF1C /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; 45 | 6EA9C8F622B1E7E30060EF1C /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 46 | 6EA9C8FB22B1E7E30060EF1C /* CombineSampleTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = CombineSampleTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 47 | 6EA9C8FF22B1E7E30060EF1C /* CombineSampleTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CombineSampleTests.swift; sourceTree = ""; }; 48 | 6EA9C90122B1E7E30060EF1C /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 49 | 6EA9C90622B1E7E30060EF1C /* CombineSampleUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = CombineSampleUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 50 | 6EA9C90A22B1E7E30060EF1C /* CombineSampleUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CombineSampleUITests.swift; sourceTree = ""; }; 51 | 6EA9C90C22B1E7E30060EF1C /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 52 | /* End PBXFileReference section */ 53 | 54 | /* Begin PBXFrameworksBuildPhase section */ 55 | 6EA9C8E222B1E7E20060EF1C /* Frameworks */ = { 56 | isa = PBXFrameworksBuildPhase; 57 | buildActionMask = 2147483647; 58 | files = ( 59 | ); 60 | runOnlyForDeploymentPostprocessing = 0; 61 | }; 62 | 6EA9C8F822B1E7E30060EF1C /* Frameworks */ = { 63 | isa = PBXFrameworksBuildPhase; 64 | buildActionMask = 2147483647; 65 | files = ( 66 | ); 67 | runOnlyForDeploymentPostprocessing = 0; 68 | }; 69 | 6EA9C90322B1E7E30060EF1C /* Frameworks */ = { 70 | isa = PBXFrameworksBuildPhase; 71 | buildActionMask = 2147483647; 72 | files = ( 73 | ); 74 | runOnlyForDeploymentPostprocessing = 0; 75 | }; 76 | /* End PBXFrameworksBuildPhase section */ 77 | 78 | /* Begin PBXGroup section */ 79 | 6EA9C8DC22B1E7E20060EF1C = { 80 | isa = PBXGroup; 81 | children = ( 82 | 6EA9C8E722B1E7E20060EF1C /* CombineSample */, 83 | 6EA9C8FE22B1E7E30060EF1C /* CombineSampleTests */, 84 | 6EA9C90922B1E7E30060EF1C /* CombineSampleUITests */, 85 | 6EA9C8E622B1E7E20060EF1C /* Products */, 86 | ); 87 | sourceTree = ""; 88 | }; 89 | 6EA9C8E622B1E7E20060EF1C /* Products */ = { 90 | isa = PBXGroup; 91 | children = ( 92 | 6EA9C8E522B1E7E20060EF1C /* CombineSample.app */, 93 | 6EA9C8FB22B1E7E30060EF1C /* CombineSampleTests.xctest */, 94 | 6EA9C90622B1E7E30060EF1C /* CombineSampleUITests.xctest */, 95 | ); 96 | name = Products; 97 | sourceTree = ""; 98 | }; 99 | 6EA9C8E722B1E7E20060EF1C /* CombineSample */ = { 100 | isa = PBXGroup; 101 | children = ( 102 | 6EA9C8E822B1E7E20060EF1C /* AppDelegate.swift */, 103 | 6EA9C8EA22B1E7E20060EF1C /* SceneDelegate.swift */, 104 | 6EA9C8EC22B1E7E20060EF1C /* ContentView.swift */, 105 | 6EA9C8EE22B1E7E30060EF1C /* Assets.xcassets */, 106 | 6EA9C8F322B1E7E30060EF1C /* LaunchScreen.storyboard */, 107 | 6EA9C8F622B1E7E30060EF1C /* Info.plist */, 108 | 6EA9C8F022B1E7E30060EF1C /* Preview Content */, 109 | ); 110 | path = CombineSample; 111 | sourceTree = ""; 112 | }; 113 | 6EA9C8F022B1E7E30060EF1C /* Preview Content */ = { 114 | isa = PBXGroup; 115 | children = ( 116 | 6EA9C8F122B1E7E30060EF1C /* Preview Assets.xcassets */, 117 | ); 118 | path = "Preview Content"; 119 | sourceTree = ""; 120 | }; 121 | 6EA9C8FE22B1E7E30060EF1C /* CombineSampleTests */ = { 122 | isa = PBXGroup; 123 | children = ( 124 | 6EA9C8FF22B1E7E30060EF1C /* CombineSampleTests.swift */, 125 | 6EA9C90122B1E7E30060EF1C /* Info.plist */, 126 | ); 127 | path = CombineSampleTests; 128 | sourceTree = ""; 129 | }; 130 | 6EA9C90922B1E7E30060EF1C /* CombineSampleUITests */ = { 131 | isa = PBXGroup; 132 | children = ( 133 | 6EA9C90A22B1E7E30060EF1C /* CombineSampleUITests.swift */, 134 | 6EA9C90C22B1E7E30060EF1C /* Info.plist */, 135 | ); 136 | path = CombineSampleUITests; 137 | sourceTree = ""; 138 | }; 139 | /* End PBXGroup section */ 140 | 141 | /* Begin PBXNativeTarget section */ 142 | 6EA9C8E422B1E7E20060EF1C /* CombineSample */ = { 143 | isa = PBXNativeTarget; 144 | buildConfigurationList = 6EA9C90F22B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSample" */; 145 | buildPhases = ( 146 | 6EA9C8E122B1E7E20060EF1C /* Sources */, 147 | 6EA9C8E222B1E7E20060EF1C /* Frameworks */, 148 | 6EA9C8E322B1E7E20060EF1C /* Resources */, 149 | ); 150 | buildRules = ( 151 | ); 152 | dependencies = ( 153 | ); 154 | name = CombineSample; 155 | productName = CombineSample; 156 | productReference = 6EA9C8E522B1E7E20060EF1C /* CombineSample.app */; 157 | productType = "com.apple.product-type.application"; 158 | }; 159 | 6EA9C8FA22B1E7E30060EF1C /* CombineSampleTests */ = { 160 | isa = PBXNativeTarget; 161 | buildConfigurationList = 6EA9C91222B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSampleTests" */; 162 | buildPhases = ( 163 | 6EA9C8F722B1E7E30060EF1C /* Sources */, 164 | 6EA9C8F822B1E7E30060EF1C /* Frameworks */, 165 | 6EA9C8F922B1E7E30060EF1C /* Resources */, 166 | ); 167 | buildRules = ( 168 | ); 169 | dependencies = ( 170 | 6EA9C8FD22B1E7E30060EF1C /* PBXTargetDependency */, 171 | ); 172 | name = CombineSampleTests; 173 | productName = CombineSampleTests; 174 | productReference = 6EA9C8FB22B1E7E30060EF1C /* CombineSampleTests.xctest */; 175 | productType = "com.apple.product-type.bundle.unit-test"; 176 | }; 177 | 6EA9C90522B1E7E30060EF1C /* CombineSampleUITests */ = { 178 | isa = PBXNativeTarget; 179 | buildConfigurationList = 6EA9C91522B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSampleUITests" */; 180 | buildPhases = ( 181 | 6EA9C90222B1E7E30060EF1C /* Sources */, 182 | 6EA9C90322B1E7E30060EF1C /* Frameworks */, 183 | 6EA9C90422B1E7E30060EF1C /* Resources */, 184 | ); 185 | buildRules = ( 186 | ); 187 | dependencies = ( 188 | 6EA9C90822B1E7E30060EF1C /* PBXTargetDependency */, 189 | ); 190 | name = CombineSampleUITests; 191 | productName = CombineSampleUITests; 192 | productReference = 6EA9C90622B1E7E30060EF1C /* CombineSampleUITests.xctest */; 193 | productType = "com.apple.product-type.bundle.ui-testing"; 194 | }; 195 | /* End PBXNativeTarget section */ 196 | 197 | /* Begin PBXProject section */ 198 | 6EA9C8DD22B1E7E20060EF1C /* Project object */ = { 199 | isa = PBXProject; 200 | attributes = { 201 | LastSwiftUpdateCheck = 1100; 202 | LastUpgradeCheck = 1100; 203 | ORGANIZATIONNAME = "Akifumi Fukaya"; 204 | TargetAttributes = { 205 | 6EA9C8E422B1E7E20060EF1C = { 206 | CreatedOnToolsVersion = 11.0; 207 | }; 208 | 6EA9C8FA22B1E7E30060EF1C = { 209 | CreatedOnToolsVersion = 11.0; 210 | TestTargetID = 6EA9C8E422B1E7E20060EF1C; 211 | }; 212 | 6EA9C90522B1E7E30060EF1C = { 213 | CreatedOnToolsVersion = 11.0; 214 | TestTargetID = 6EA9C8E422B1E7E20060EF1C; 215 | }; 216 | }; 217 | }; 218 | buildConfigurationList = 6EA9C8E022B1E7E20060EF1C /* Build configuration list for PBXProject "CombineSample" */; 219 | compatibilityVersion = "Xcode 9.3"; 220 | developmentRegion = en; 221 | hasScannedForEncodings = 0; 222 | knownRegions = ( 223 | en, 224 | Base, 225 | ); 226 | mainGroup = 6EA9C8DC22B1E7E20060EF1C; 227 | productRefGroup = 6EA9C8E622B1E7E20060EF1C /* Products */; 228 | projectDirPath = ""; 229 | projectRoot = ""; 230 | targets = ( 231 | 6EA9C8E422B1E7E20060EF1C /* CombineSample */, 232 | 6EA9C8FA22B1E7E30060EF1C /* CombineSampleTests */, 233 | 6EA9C90522B1E7E30060EF1C /* CombineSampleUITests */, 234 | ); 235 | }; 236 | /* End PBXProject section */ 237 | 238 | /* Begin PBXResourcesBuildPhase section */ 239 | 6EA9C8E322B1E7E20060EF1C /* Resources */ = { 240 | isa = PBXResourcesBuildPhase; 241 | buildActionMask = 2147483647; 242 | files = ( 243 | 6EA9C8F522B1E7E30060EF1C /* LaunchScreen.storyboard in Resources */, 244 | 6EA9C8F222B1E7E30060EF1C /* Preview Assets.xcassets in Resources */, 245 | 6EA9C8EF22B1E7E30060EF1C /* Assets.xcassets in Resources */, 246 | ); 247 | runOnlyForDeploymentPostprocessing = 0; 248 | }; 249 | 6EA9C8F922B1E7E30060EF1C /* Resources */ = { 250 | isa = PBXResourcesBuildPhase; 251 | buildActionMask = 2147483647; 252 | files = ( 253 | ); 254 | runOnlyForDeploymentPostprocessing = 0; 255 | }; 256 | 6EA9C90422B1E7E30060EF1C /* Resources */ = { 257 | isa = PBXResourcesBuildPhase; 258 | buildActionMask = 2147483647; 259 | files = ( 260 | ); 261 | runOnlyForDeploymentPostprocessing = 0; 262 | }; 263 | /* End PBXResourcesBuildPhase section */ 264 | 265 | /* Begin PBXSourcesBuildPhase section */ 266 | 6EA9C8E122B1E7E20060EF1C /* Sources */ = { 267 | isa = PBXSourcesBuildPhase; 268 | buildActionMask = 2147483647; 269 | files = ( 270 | 6EA9C8E922B1E7E20060EF1C /* AppDelegate.swift in Sources */, 271 | 6EA9C8EB22B1E7E20060EF1C /* SceneDelegate.swift in Sources */, 272 | 6EA9C8ED22B1E7E20060EF1C /* ContentView.swift in Sources */, 273 | ); 274 | runOnlyForDeploymentPostprocessing = 0; 275 | }; 276 | 6EA9C8F722B1E7E30060EF1C /* Sources */ = { 277 | isa = PBXSourcesBuildPhase; 278 | buildActionMask = 2147483647; 279 | files = ( 280 | 6EA9C90022B1E7E30060EF1C /* CombineSampleTests.swift in Sources */, 281 | ); 282 | runOnlyForDeploymentPostprocessing = 0; 283 | }; 284 | 6EA9C90222B1E7E30060EF1C /* Sources */ = { 285 | isa = PBXSourcesBuildPhase; 286 | buildActionMask = 2147483647; 287 | files = ( 288 | 6EA9C90B22B1E7E30060EF1C /* CombineSampleUITests.swift in Sources */, 289 | ); 290 | runOnlyForDeploymentPostprocessing = 0; 291 | }; 292 | /* End PBXSourcesBuildPhase section */ 293 | 294 | /* Begin PBXTargetDependency section */ 295 | 6EA9C8FD22B1E7E30060EF1C /* PBXTargetDependency */ = { 296 | isa = PBXTargetDependency; 297 | target = 6EA9C8E422B1E7E20060EF1C /* CombineSample */; 298 | targetProxy = 6EA9C8FC22B1E7E30060EF1C /* PBXContainerItemProxy */; 299 | }; 300 | 6EA9C90822B1E7E30060EF1C /* PBXTargetDependency */ = { 301 | isa = PBXTargetDependency; 302 | target = 6EA9C8E422B1E7E20060EF1C /* CombineSample */; 303 | targetProxy = 6EA9C90722B1E7E30060EF1C /* PBXContainerItemProxy */; 304 | }; 305 | /* End PBXTargetDependency section */ 306 | 307 | /* Begin PBXVariantGroup section */ 308 | 6EA9C8F322B1E7E30060EF1C /* LaunchScreen.storyboard */ = { 309 | isa = PBXVariantGroup; 310 | children = ( 311 | 6EA9C8F422B1E7E30060EF1C /* Base */, 312 | ); 313 | name = LaunchScreen.storyboard; 314 | sourceTree = ""; 315 | }; 316 | /* End PBXVariantGroup section */ 317 | 318 | /* Begin XCBuildConfiguration section */ 319 | 6EA9C90D22B1E7E30060EF1C /* Debug */ = { 320 | isa = XCBuildConfiguration; 321 | buildSettings = { 322 | ALWAYS_SEARCH_USER_PATHS = NO; 323 | CLANG_ANALYZER_NONNULL = YES; 324 | CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; 325 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; 326 | CLANG_CXX_LIBRARY = "libc++"; 327 | CLANG_ENABLE_MODULES = YES; 328 | CLANG_ENABLE_OBJC_ARC = YES; 329 | CLANG_ENABLE_OBJC_WEAK = YES; 330 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 331 | CLANG_WARN_BOOL_CONVERSION = YES; 332 | CLANG_WARN_COMMA = YES; 333 | CLANG_WARN_CONSTANT_CONVERSION = YES; 334 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 335 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 336 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 337 | CLANG_WARN_EMPTY_BODY = YES; 338 | CLANG_WARN_ENUM_CONVERSION = YES; 339 | CLANG_WARN_INFINITE_RECURSION = YES; 340 | CLANG_WARN_INT_CONVERSION = YES; 341 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 342 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 343 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 344 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 345 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 346 | CLANG_WARN_STRICT_PROTOTYPES = YES; 347 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 348 | CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; 349 | CLANG_WARN_UNREACHABLE_CODE = YES; 350 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 351 | COPY_PHASE_STRIP = NO; 352 | DEBUG_INFORMATION_FORMAT = dwarf; 353 | ENABLE_STRICT_OBJC_MSGSEND = YES; 354 | ENABLE_TESTABILITY = YES; 355 | GCC_C_LANGUAGE_STANDARD = gnu11; 356 | GCC_DYNAMIC_NO_PIC = NO; 357 | GCC_NO_COMMON_BLOCKS = YES; 358 | GCC_OPTIMIZATION_LEVEL = 0; 359 | GCC_PREPROCESSOR_DEFINITIONS = ( 360 | "DEBUG=1", 361 | "$(inherited)", 362 | ); 363 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 364 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 365 | GCC_WARN_UNDECLARED_SELECTOR = YES; 366 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 367 | GCC_WARN_UNUSED_FUNCTION = YES; 368 | GCC_WARN_UNUSED_VARIABLE = YES; 369 | IPHONEOS_DEPLOYMENT_TARGET = 13.0; 370 | MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; 371 | MTL_FAST_MATH = YES; 372 | ONLY_ACTIVE_ARCH = YES; 373 | SDKROOT = iphoneos; 374 | SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; 375 | SWIFT_OPTIMIZATION_LEVEL = "-Onone"; 376 | }; 377 | name = Debug; 378 | }; 379 | 6EA9C90E22B1E7E30060EF1C /* Release */ = { 380 | isa = XCBuildConfiguration; 381 | buildSettings = { 382 | ALWAYS_SEARCH_USER_PATHS = NO; 383 | CLANG_ANALYZER_NONNULL = YES; 384 | CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; 385 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; 386 | CLANG_CXX_LIBRARY = "libc++"; 387 | CLANG_ENABLE_MODULES = YES; 388 | CLANG_ENABLE_OBJC_ARC = YES; 389 | CLANG_ENABLE_OBJC_WEAK = YES; 390 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 391 | CLANG_WARN_BOOL_CONVERSION = YES; 392 | CLANG_WARN_COMMA = YES; 393 | CLANG_WARN_CONSTANT_CONVERSION = YES; 394 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 395 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 396 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 397 | CLANG_WARN_EMPTY_BODY = YES; 398 | CLANG_WARN_ENUM_CONVERSION = YES; 399 | CLANG_WARN_INFINITE_RECURSION = YES; 400 | CLANG_WARN_INT_CONVERSION = YES; 401 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 402 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 403 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 404 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 405 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 406 | CLANG_WARN_STRICT_PROTOTYPES = YES; 407 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 408 | CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; 409 | CLANG_WARN_UNREACHABLE_CODE = YES; 410 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 411 | COPY_PHASE_STRIP = NO; 412 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 413 | ENABLE_NS_ASSERTIONS = NO; 414 | ENABLE_STRICT_OBJC_MSGSEND = YES; 415 | GCC_C_LANGUAGE_STANDARD = gnu11; 416 | GCC_NO_COMMON_BLOCKS = YES; 417 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 418 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 419 | GCC_WARN_UNDECLARED_SELECTOR = YES; 420 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 421 | GCC_WARN_UNUSED_FUNCTION = YES; 422 | GCC_WARN_UNUSED_VARIABLE = YES; 423 | IPHONEOS_DEPLOYMENT_TARGET = 13.0; 424 | MTL_ENABLE_DEBUG_INFO = NO; 425 | MTL_FAST_MATH = YES; 426 | SDKROOT = iphoneos; 427 | SWIFT_COMPILATION_MODE = wholemodule; 428 | SWIFT_OPTIMIZATION_LEVEL = "-O"; 429 | VALIDATE_PRODUCT = YES; 430 | }; 431 | name = Release; 432 | }; 433 | 6EA9C91022B1E7E30060EF1C /* Debug */ = { 434 | isa = XCBuildConfiguration; 435 | buildSettings = { 436 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 437 | CODE_SIGN_STYLE = Automatic; 438 | DEVELOPMENT_ASSET_PATHS = "CombineSample/Preview\\ Content"; 439 | DEVELOPMENT_TEAM = 3JS5D26M66; 440 | ENABLE_PREVIEWS = YES; 441 | INFOPLIST_FILE = CombineSample/Info.plist; 442 | LD_RUNPATH_SEARCH_PATHS = ( 443 | "$(inherited)", 444 | "@executable_path/Frameworks", 445 | ); 446 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSample; 447 | PRODUCT_NAME = "$(TARGET_NAME)"; 448 | SWIFT_VERSION = 5.0; 449 | TARGETED_DEVICE_FAMILY = "1,2"; 450 | }; 451 | name = Debug; 452 | }; 453 | 6EA9C91122B1E7E30060EF1C /* Release */ = { 454 | isa = XCBuildConfiguration; 455 | buildSettings = { 456 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 457 | CODE_SIGN_STYLE = Automatic; 458 | DEVELOPMENT_ASSET_PATHS = "CombineSample/Preview\\ Content"; 459 | DEVELOPMENT_TEAM = 3JS5D26M66; 460 | ENABLE_PREVIEWS = YES; 461 | INFOPLIST_FILE = CombineSample/Info.plist; 462 | LD_RUNPATH_SEARCH_PATHS = ( 463 | "$(inherited)", 464 | "@executable_path/Frameworks", 465 | ); 466 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSample; 467 | PRODUCT_NAME = "$(TARGET_NAME)"; 468 | SWIFT_VERSION = 5.0; 469 | TARGETED_DEVICE_FAMILY = "1,2"; 470 | }; 471 | name = Release; 472 | }; 473 | 6EA9C91322B1E7E30060EF1C /* Debug */ = { 474 | isa = XCBuildConfiguration; 475 | buildSettings = { 476 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; 477 | BUNDLE_LOADER = "$(TEST_HOST)"; 478 | CODE_SIGN_STYLE = Automatic; 479 | DEVELOPMENT_TEAM = 3JS5D26M66; 480 | INFOPLIST_FILE = CombineSampleTests/Info.plist; 481 | LD_RUNPATH_SEARCH_PATHS = ( 482 | "$(inherited)", 483 | "@executable_path/Frameworks", 484 | "@loader_path/Frameworks", 485 | ); 486 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSampleTests; 487 | PRODUCT_NAME = "$(TARGET_NAME)"; 488 | SWIFT_VERSION = 5.0; 489 | TARGETED_DEVICE_FAMILY = "1,2"; 490 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/CombineSample.app/CombineSample"; 491 | }; 492 | name = Debug; 493 | }; 494 | 6EA9C91422B1E7E30060EF1C /* Release */ = { 495 | isa = XCBuildConfiguration; 496 | buildSettings = { 497 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; 498 | BUNDLE_LOADER = "$(TEST_HOST)"; 499 | CODE_SIGN_STYLE = Automatic; 500 | DEVELOPMENT_TEAM = 3JS5D26M66; 501 | INFOPLIST_FILE = CombineSampleTests/Info.plist; 502 | LD_RUNPATH_SEARCH_PATHS = ( 503 | "$(inherited)", 504 | "@executable_path/Frameworks", 505 | "@loader_path/Frameworks", 506 | ); 507 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSampleTests; 508 | PRODUCT_NAME = "$(TARGET_NAME)"; 509 | SWIFT_VERSION = 5.0; 510 | TARGETED_DEVICE_FAMILY = "1,2"; 511 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/CombineSample.app/CombineSample"; 512 | }; 513 | name = Release; 514 | }; 515 | 6EA9C91622B1E7E30060EF1C /* Debug */ = { 516 | isa = XCBuildConfiguration; 517 | buildSettings = { 518 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; 519 | CODE_SIGN_STYLE = Automatic; 520 | DEVELOPMENT_TEAM = 3JS5D26M66; 521 | INFOPLIST_FILE = CombineSampleUITests/Info.plist; 522 | LD_RUNPATH_SEARCH_PATHS = ( 523 | "$(inherited)", 524 | "@executable_path/Frameworks", 525 | "@loader_path/Frameworks", 526 | ); 527 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSampleUITests; 528 | PRODUCT_NAME = "$(TARGET_NAME)"; 529 | SWIFT_VERSION = 5.0; 530 | TARGETED_DEVICE_FAMILY = "1,2"; 531 | TEST_TARGET_NAME = CombineSample; 532 | }; 533 | name = Debug; 534 | }; 535 | 6EA9C91722B1E7E30060EF1C /* Release */ = { 536 | isa = XCBuildConfiguration; 537 | buildSettings = { 538 | ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; 539 | CODE_SIGN_STYLE = Automatic; 540 | DEVELOPMENT_TEAM = 3JS5D26M66; 541 | INFOPLIST_FILE = CombineSampleUITests/Info.plist; 542 | LD_RUNPATH_SEARCH_PATHS = ( 543 | "$(inherited)", 544 | "@executable_path/Frameworks", 545 | "@loader_path/Frameworks", 546 | ); 547 | PRODUCT_BUNDLE_IDENTIFIER = net.akifumifukaya.CombineSampleUITests; 548 | PRODUCT_NAME = "$(TARGET_NAME)"; 549 | SWIFT_VERSION = 5.0; 550 | TARGETED_DEVICE_FAMILY = "1,2"; 551 | TEST_TARGET_NAME = CombineSample; 552 | }; 553 | name = Release; 554 | }; 555 | /* End XCBuildConfiguration section */ 556 | 557 | /* Begin XCConfigurationList section */ 558 | 6EA9C8E022B1E7E20060EF1C /* Build configuration list for PBXProject "CombineSample" */ = { 559 | isa = XCConfigurationList; 560 | buildConfigurations = ( 561 | 6EA9C90D22B1E7E30060EF1C /* Debug */, 562 | 6EA9C90E22B1E7E30060EF1C /* Release */, 563 | ); 564 | defaultConfigurationIsVisible = 0; 565 | defaultConfigurationName = Release; 566 | }; 567 | 6EA9C90F22B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSample" */ = { 568 | isa = XCConfigurationList; 569 | buildConfigurations = ( 570 | 6EA9C91022B1E7E30060EF1C /* Debug */, 571 | 6EA9C91122B1E7E30060EF1C /* Release */, 572 | ); 573 | defaultConfigurationIsVisible = 0; 574 | defaultConfigurationName = Release; 575 | }; 576 | 6EA9C91222B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSampleTests" */ = { 577 | isa = XCConfigurationList; 578 | buildConfigurations = ( 579 | 6EA9C91322B1E7E30060EF1C /* Debug */, 580 | 6EA9C91422B1E7E30060EF1C /* Release */, 581 | ); 582 | defaultConfigurationIsVisible = 0; 583 | defaultConfigurationName = Release; 584 | }; 585 | 6EA9C91522B1E7E30060EF1C /* Build configuration list for PBXNativeTarget "CombineSampleUITests" */ = { 586 | isa = XCConfigurationList; 587 | buildConfigurations = ( 588 | 6EA9C91622B1E7E30060EF1C /* Debug */, 589 | 6EA9C91722B1E7E30060EF1C /* Release */, 590 | ); 591 | defaultConfigurationIsVisible = 0; 592 | defaultConfigurationName = Release; 593 | }; 594 | /* End XCConfigurationList section */ 595 | }; 596 | rootObject = 6EA9C8DD22B1E7E20060EF1C /* Project object */; 597 | } 598 | -------------------------------------------------------------------------------- /CombineSample.xcodeproj/project.xcworkspace/contents.xcworkspacedata: -------------------------------------------------------------------------------- 1 | 2 | 4 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /CombineSample.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | IDEDidComputeMac32BitWarning 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /CombineSample.xcodeproj/xcuserdata/akifumi.fukaya.xcuserdatad/xcschemes/xcschememanagement.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | SchemeUserState 6 | 7 | CombineSample.xcscheme_^#shared#^_ 8 | 9 | orderHint 10 | 0 11 | 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /CombineSample/AppDelegate.swift: -------------------------------------------------------------------------------- 1 | // 2 | // AppDelegate.swift 3 | // CombineSample 4 | // 5 | // Created by akifumi.fukaya on 2019/06/13. 6 | // Copyright © 2019 Akifumi Fukaya. All rights reserved. 7 | // 8 | 9 | import UIKit 10 | 11 | @UIApplicationMain 12 | class AppDelegate: UIResponder, UIApplicationDelegate { 13 | 14 | 15 | 16 | func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool { 17 | // Override point for customization after application launch. 18 | return true 19 | } 20 | 21 | func applicationWillTerminate(_ application: UIApplication) { 22 | // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:. 23 | } 24 | 25 | // MARK: UISceneSession Lifecycle 26 | 27 | func application(_ application: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options: UIScene.ConnectionOptions) -> UISceneConfiguration { 28 | // Called when a new scene session is being created. 29 | // Use this method to select a configuration to create the new scene with. 30 | return UISceneConfiguration(name: "Default Configuration", sessionRole: connectingSceneSession.role) 31 | } 32 | 33 | func application(_ application: UIApplication, didDiscardSceneSessions sceneSessions: Set) { 34 | // Called when the user discards a scene session. 35 | // If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions. 36 | // Use this method to release any resources that were specific to the discarded scenes, as they will not return. 37 | } 38 | 39 | 40 | } 41 | 42 | -------------------------------------------------------------------------------- /CombineSample/Assets.xcassets/AppIcon.appiconset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images" : [ 3 | { 4 | "idiom" : "iphone", 5 | "size" : "20x20", 6 | "scale" : "2x" 7 | }, 8 | { 9 | "idiom" : "iphone", 10 | "size" : "20x20", 11 | "scale" : "3x" 12 | }, 13 | { 14 | "idiom" : "iphone", 15 | "size" : "29x29", 16 | "scale" : "2x" 17 | }, 18 | { 19 | "idiom" : "iphone", 20 | "size" : "29x29", 21 | "scale" : "3x" 22 | }, 23 | { 24 | "idiom" : "iphone", 25 | "size" : "40x40", 26 | "scale" : "2x" 27 | }, 28 | { 29 | "idiom" : "iphone", 30 | "size" : "40x40", 31 | "scale" : "3x" 32 | }, 33 | { 34 | "idiom" : "iphone", 35 | "size" : "60x60", 36 | "scale" : "2x" 37 | }, 38 | { 39 | "idiom" : "iphone", 40 | "size" : "60x60", 41 | "scale" : "3x" 42 | }, 43 | { 44 | "idiom" : "ipad", 45 | "size" : "20x20", 46 | "scale" : "1x" 47 | }, 48 | { 49 | "idiom" : "ipad", 50 | "size" : "20x20", 51 | "scale" : "2x" 52 | }, 53 | { 54 | "idiom" : "ipad", 55 | "size" : "29x29", 56 | "scale" : "1x" 57 | }, 58 | { 59 | "idiom" : "ipad", 60 | "size" : "29x29", 61 | "scale" : "2x" 62 | }, 63 | { 64 | "idiom" : "ipad", 65 | "size" : "40x40", 66 | "scale" : "1x" 67 | }, 68 | { 69 | "idiom" : "ipad", 70 | "size" : "40x40", 71 | "scale" : "2x" 72 | }, 73 | { 74 | "idiom" : "ipad", 75 | "size" : "76x76", 76 | "scale" : "1x" 77 | }, 78 | { 79 | "idiom" : "ipad", 80 | "size" : "76x76", 81 | "scale" : "2x" 82 | }, 83 | { 84 | "idiom" : "ipad", 85 | "size" : "83.5x83.5", 86 | "scale" : "2x" 87 | }, 88 | { 89 | "idiom" : "ios-marketing", 90 | "size" : "1024x1024", 91 | "scale" : "1x" 92 | } 93 | ], 94 | "info" : { 95 | "version" : 1, 96 | "author" : "xcode" 97 | } 98 | } -------------------------------------------------------------------------------- /CombineSample/Assets.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "version" : 1, 4 | "author" : "xcode" 5 | } 6 | } -------------------------------------------------------------------------------- /CombineSample/Base.lproj/LaunchScreen.storyboard: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | -------------------------------------------------------------------------------- /CombineSample/ContentView.swift: -------------------------------------------------------------------------------- 1 | // 2 | // ContentView.swift 3 | // CombineSample 4 | // 5 | // Created by akifumi.fukaya on 2019/06/13. 6 | // Copyright © 2019 Akifumi Fukaya. All rights reserved. 7 | // 8 | 9 | import SwiftUI 10 | import Combine 11 | 12 | final class ContentViewModel : ObservableObject, Identifiable { 13 | @Published 14 | var username: String = "" 15 | struct StatusText { 16 | let content: String 17 | let color: Color 18 | } 19 | @Published 20 | var status: StatusText = StatusText(content: "NG", color: .red) 21 | private var validatedUsername: AnyPublisher { 22 | return $username 23 | .debounce(for: 0.1, scheduler: RunLoop.main) 24 | .removeDuplicates() 25 | .flatMap { (username) -> AnyPublisher in 26 | Future { (promise) in 27 | // FIXME: API request 28 | if 1...10 ~= username.count { 29 | promise(.success(username)) 30 | } else { 31 | promise(.success(nil)) 32 | } 33 | } 34 | .eraseToAnyPublisher() 35 | } 36 | .eraseToAnyPublisher() 37 | } 38 | 39 | private var cancellables: [AnyCancellable] = [] 40 | 41 | private(set) lazy var onAppear: () -> Void = { [weak self] in 42 | guard let self = self else { return } 43 | self.validatedUsername 44 | .sink(receiveValue: { [weak self] (value) in 45 | if let value = value { 46 | self?.username = value 47 | } else { 48 | print("validatedUsername.receiveValue: Invalid username") 49 | } 50 | }) 51 | .store(in: &self.cancellables) 52 | 53 | // Update StatusText 54 | self.validatedUsername 55 | .map { (value) -> StatusText in 56 | if let _ = value { 57 | return StatusText(content: "OK", color: .green) 58 | } else { 59 | return StatusText(content: "NG", color: .red) 60 | } 61 | } 62 | .sink(receiveValue: { [weak self] (value) in 63 | self?.status = value 64 | }) 65 | .store(in: &self.cancellables) 66 | } 67 | 68 | private(set) lazy var onDisappear: () -> Void = { [weak self] in 69 | guard let self = self else { return } 70 | self.cancellables.forEach { $0.cancel() } 71 | self.cancellables = [] 72 | } 73 | } 74 | 75 | struct ContentView : View { 76 | @ObservedObject var viewModel: ContentViewModel 77 | 78 | var body: some View { 79 | VStack { 80 | HStack { 81 | Text($viewModel.status.wrappedValue.content) 82 | .foregroundColor($viewModel.status.wrappedValue.color) 83 | Spacer() 84 | } 85 | TextField("Placeholder", text: $viewModel.username, onEditingChanged: { (changed) in 86 | print("onEditingChanged: \(changed)") 87 | }, onCommit: { 88 | print("onCommit") 89 | }) 90 | } 91 | .padding(.horizontal) 92 | .onAppear(perform: viewModel.onAppear) 93 | .onDisappear(perform: viewModel.onDisappear) 94 | } 95 | } 96 | 97 | #if DEBUG 98 | struct ContentView_Previews : PreviewProvider { 99 | static var previews: some View { 100 | ContentView(viewModel: ContentViewModel()) 101 | } 102 | } 103 | #endif 104 | -------------------------------------------------------------------------------- /CombineSample/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | $(DEVELOPMENT_LANGUAGE) 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | $(PRODUCT_BUNDLE_PACKAGE_TYPE) 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleVersion 20 | 1 21 | LSRequiresIPhoneOS 22 | 23 | UIApplicationSceneManifest 24 | 25 | UIApplicationSupportsMultipleScenes 26 | 27 | UISceneConfigurations 28 | 29 | UIWindowSceneSessionRoleApplication 30 | 31 | 32 | UILaunchStoryboardName 33 | LaunchScreen 34 | UISceneConfigurationName 35 | Default Configuration 36 | UISceneDelegateClassName 37 | $(PRODUCT_MODULE_NAME).SceneDelegate 38 | 39 | 40 | 41 | 42 | UILaunchStoryboardName 43 | LaunchScreen 44 | UIRequiredDeviceCapabilities 45 | 46 | armv7 47 | 48 | UISupportedInterfaceOrientations 49 | 50 | UIInterfaceOrientationPortrait 51 | UIInterfaceOrientationLandscapeLeft 52 | UIInterfaceOrientationLandscapeRight 53 | 54 | UISupportedInterfaceOrientations~ipad 55 | 56 | UIInterfaceOrientationPortrait 57 | UIInterfaceOrientationPortraitUpsideDown 58 | UIInterfaceOrientationLandscapeLeft 59 | UIInterfaceOrientationLandscapeRight 60 | 61 | 62 | 63 | -------------------------------------------------------------------------------- /CombineSample/Preview Content/Preview Assets.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "version" : 1, 4 | "author" : "xcode" 5 | } 6 | } -------------------------------------------------------------------------------- /CombineSample/SceneDelegate.swift: -------------------------------------------------------------------------------- 1 | // 2 | // SceneDelegate.swift 3 | // CombineSample 4 | // 5 | // Created by akifumi.fukaya on 2019/06/13. 6 | // Copyright © 2019 Akifumi Fukaya. All rights reserved. 7 | // 8 | 9 | import UIKit 10 | import SwiftUI 11 | 12 | class SceneDelegate: UIResponder, UIWindowSceneDelegate { 13 | 14 | var window: UIWindow? 15 | 16 | 17 | func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) { 18 | // Use this method to optionally configure and attach the UIWindow `window` to the provided UIWindowScene `scene`. 19 | // If using a storyboard, the `window` property will automatically be initialized and attached to the scene. 20 | // This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead). 21 | 22 | if let windowScene = scene as? UIWindowScene { 23 | let rootView = ContentView(viewModel: ContentViewModel()) 24 | let window = UIWindow(windowScene: windowScene) 25 | window.rootViewController = UIHostingController(rootView: rootView) 26 | self.window = window 27 | window.makeKeyAndVisible() 28 | } 29 | } 30 | 31 | func sceneDidDisconnect(_ scene: UIScene) { 32 | // Called as the scene is being released by the system. 33 | // This occurs shortly after the scene enters the background, or when its session is discarded. 34 | // Release any resources associated with this scene that can be re-created the next time the scene connects. 35 | // The scene may re-connect later, as its session was not neccessarily discarded (see `application:didDiscardSceneSessions` instead). 36 | } 37 | 38 | func sceneDidBecomeActive(_ scene: UIScene) { 39 | // Called when the scene has moved from an inactive state to an active state. 40 | // Use this method to restart any tasks that were paused (or not yet started) when the scene was inactive. 41 | } 42 | 43 | func sceneWillResignActive(_ scene: UIScene) { 44 | // Called when the scene will move from an active state to an inactive state. 45 | // This may occur due to temporary interruptions (ex. an incoming phone call). 46 | } 47 | 48 | func sceneWillEnterForeground(_ scene: UIScene) { 49 | // Called as the scene transitions from the background to the foreground. 50 | // Use this method to undo the changes made on entering the background. 51 | } 52 | 53 | func sceneDidEnterBackground(_ scene: UIScene) { 54 | // Called as the scene transitions from the foreground to the background. 55 | // Use this method to save data, release shared resources, and store enough scene-specific state information 56 | // to restore the scene back to its current state. 57 | } 58 | 59 | 60 | } 61 | 62 | -------------------------------------------------------------------------------- /CombineSampleTests/CombineSampleTests.swift: -------------------------------------------------------------------------------- 1 | // 2 | // CombineSampleTests.swift 3 | // CombineSampleTests 4 | // 5 | // Created by akifumi.fukaya on 2019/06/13. 6 | // Copyright © 2019 Akifumi Fukaya. All rights reserved. 7 | // 8 | 9 | import XCTest 10 | @testable import CombineSample 11 | 12 | class CombineSampleTests: XCTestCase { 13 | 14 | override func setUp() { 15 | // Put setup code here. This method is called before the invocation of each test method in the class. 16 | } 17 | 18 | override func tearDown() { 19 | // Put teardown code here. This method is called after the invocation of each test method in the class. 20 | } 21 | 22 | func testExample() { 23 | // This is an example of a functional test case. 24 | // Use XCTAssert and related functions to verify your tests produce the correct results. 25 | } 26 | 27 | func testPerformanceExample() { 28 | // This is an example of a performance test case. 29 | self.measure { 30 | // Put the code you want to measure the time of here. 31 | } 32 | } 33 | 34 | } 35 | -------------------------------------------------------------------------------- /CombineSampleTests/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | $(DEVELOPMENT_LANGUAGE) 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | $(PRODUCT_BUNDLE_PACKAGE_TYPE) 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleVersion 20 | 1 21 | 22 | 23 | -------------------------------------------------------------------------------- /CombineSampleUITests/CombineSampleUITests.swift: -------------------------------------------------------------------------------- 1 | // 2 | // CombineSampleUITests.swift 3 | // CombineSampleUITests 4 | // 5 | // Created by akifumi.fukaya on 2019/06/13. 6 | // Copyright © 2019 Akifumi Fukaya. All rights reserved. 7 | // 8 | 9 | import XCTest 10 | 11 | class CombineSampleUITests: XCTestCase { 12 | 13 | override func setUp() { 14 | // Put setup code here. This method is called before the invocation of each test method in the class. 15 | 16 | // In UI tests it is usually best to stop immediately when a failure occurs. 17 | continueAfterFailure = false 18 | 19 | // UI tests must launch the application that they test. Doing this in setup will make sure it happens for each test method. 20 | XCUIApplication().launch() 21 | 22 | // In UI tests it’s important to set the initial state - such as interface orientation - required for your tests before they run. The setUp method is a good place to do this. 23 | } 24 | 25 | override func tearDown() { 26 | // Put teardown code here. This method is called after the invocation of each test method in the class. 27 | } 28 | 29 | func testExample() { 30 | // Use recording to get started writing UI tests. 31 | // Use XCTAssert and related functions to verify your tests produce the correct results. 32 | } 33 | 34 | } 35 | -------------------------------------------------------------------------------- /CombineSampleUITests/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | $(DEVELOPMENT_LANGUAGE) 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | $(PRODUCT_BUNDLE_PACKAGE_TYPE) 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleVersion 20 | 1 21 | 22 | 23 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # MVVM with Combine in SwiftUI 2 | MVVM with Combine in SwiftUI 3 | 4 | ## Requirements 5 | - Xcode Version 11.0 (11A420a) 6 | 7 | --------------------------------------------------------------------------------