diff --git a/.gitignore b/.gitignore index e284f64..5d087d6 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,5 @@ .DS_Store -/.build +.build/ /Packages /*.xcodeproj xcuserdata/ diff --git a/Examples/HoodsApp/HoodsApp.xcodeproj/project.pbxproj b/Examples/HoodsApp/HoodsApp.xcodeproj/project.pbxproj index 14889ec..2b8a840 100644 --- a/Examples/HoodsApp/HoodsApp.xcodeproj/project.pbxproj +++ b/Examples/HoodsApp/HoodsApp.xcodeproj/project.pbxproj @@ -3,7 +3,7 @@ archiveVersion = 1; classes = { }; - objectVersion = 56; + objectVersion = 70; objects = { /* Begin PBXBuildFile section */ @@ -11,6 +11,8 @@ 0004E7C42B86A2F4008EBCF9 /* CopyTextDemoView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0004E7C32B86A2F4008EBCF9 /* CopyTextDemoView.swift */; }; 0004E7C62B86A313008EBCF9 /* CopyTextDemoTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0004E7C52B86A313008EBCF9 /* CopyTextDemoTests.swift */; }; 004F27742AE6DF3A00D118BA /* ComposableArchitecture in Frameworks */ = {isa = PBXBuildFile; productRef = 004F27732AE6DF3A00D118BA /* ComposableArchitecture */; }; + 0068F7132D1486A000295745 /* CounterFeatureTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0068F7122D1486A000295745 /* CounterFeatureTests.swift */; }; + 0068F71B2D1574A600295745 /* CounterTabFeatureTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0068F71A2D1574A600295745 /* CounterTabFeatureTests.swift */; }; 00BC0F6D2B632FE2000FC408 /* HoodsTestsTools in Frameworks */ = {isa = PBXBuildFile; productRef = 00BC0F6C2B632FE2000FC408 /* HoodsTestsTools */; }; 00D120752C88E997001CDA65 /* ImagePickerFeature.swift in Sources */ = {isa = PBXBuildFile; fileRef = 00D120742C88E997001CDA65 /* ImagePickerFeature.swift */; }; 00D120772C88E99C001CDA65 /* ImagePickerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 00D120762C88E99C001CDA65 /* ImagePickerView.swift */; }; @@ -43,6 +45,8 @@ 0004E7C12B86A2E8008EBCF9 /* CopyTextDemoFeature.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CopyTextDemoFeature.swift; sourceTree = ""; }; 0004E7C32B86A2F4008EBCF9 /* CopyTextDemoView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CopyTextDemoView.swift; sourceTree = ""; }; 0004E7C52B86A313008EBCF9 /* CopyTextDemoTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CopyTextDemoTests.swift; sourceTree = ""; }; + 0068F7122D1486A000295745 /* CounterFeatureTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CounterFeatureTests.swift; sourceTree = ""; }; + 0068F71A2D1574A600295745 /* CounterTabFeatureTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CounterTabFeatureTests.swift; sourceTree = ""; }; 00CC6CBA2A68425F0093300A /* DefaultTestPlan.xctestplan */ = {isa = PBXFileReference; lastKnownFileType = text; path = DefaultTestPlan.xctestplan; sourceTree = ""; }; 00D120742C88E997001CDA65 /* ImagePickerFeature.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ImagePickerFeature.swift; sourceTree = ""; }; 00D120762C88E99C001CDA65 /* ImagePickerView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ImagePickerView.swift; sourceTree = ""; }; @@ -61,6 +65,10 @@ 00F84B012A66DE0B00A7EC95 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; /* End PBXFileReference section */ +/* Begin PBXFileSystemSynchronizedRootGroup section */ + 0068F70F2D146A1800295745 /* TCATutorial */ = {isa = PBXFileSystemSynchronizedRootGroup; explicitFileTypes = {}; explicitFolders = (); path = TCATutorial; sourceTree = ""; }; +/* End PBXFileSystemSynchronizedRootGroup section */ + /* Begin PBXFrameworksBuildPhase section */ 00F6F5512A66CBE40088B530 /* Frameworks */ = { isa = PBXFrameworksBuildPhase; @@ -135,6 +143,7 @@ 00F6F5562A66CBE40088B530 /* HoodsApp */ = { isa = PBXGroup; children = ( + 0068F70F2D146A1800295745 /* TCATutorial */, 0004E7C02B86A2B7008EBCF9 /* CopyTextDemo */, 00D120732C88E987001CDA65 /* ImagePickerDemo */, 00DCF43E2B62CFBE00082C13 /* MailButtonDemo */, @@ -162,6 +171,8 @@ 00CC6CBA2A68425F0093300A /* DefaultTestPlan.xctestplan */, 00DCF4432B62CFF800082C13 /* MailButtonDemoTests.swift */, 0004E7C52B86A313008EBCF9 /* CopyTextDemoTests.swift */, + 0068F7122D1486A000295745 /* CounterFeatureTests.swift */, + 0068F71A2D1574A600295745 /* CounterTabFeatureTests.swift */, ); path = HoodsAppTests; sourceTree = ""; @@ -188,6 +199,9 @@ ); dependencies = ( ); + fileSystemSynchronizedGroups = ( + 0068F70F2D146A1800295745 /* TCATutorial */, + ); name = HoodsApp; packageProductDependencies = ( 004F27732AE6DF3A00D118BA /* ComposableArchitecture */, @@ -303,8 +317,10 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + 0068F7132D1486A000295745 /* CounterFeatureTests.swift in Sources */, 0004E7C62B86A313008EBCF9 /* CopyTextDemoTests.swift in Sources */, 00F6F5692A66CBE50088B530 /* HoodsAppTests.swift in Sources */, + 0068F71B2D1574A600295745 /* CounterTabFeatureTests.swift in Sources */, 00DCF4442B62CFF800082C13 /* MailButtonDemoTests.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; @@ -455,7 +471,7 @@ INFOPLIST_KEY_UILaunchScreen_Generation = YES; INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; - IPHONEOS_DEPLOYMENT_TARGET = 17.0; + IPHONEOS_DEPLOYMENT_TARGET = 18.0; LD_RUNPATH_SEARCH_PATHS = ( "$(inherited)", "@executable_path/Frameworks", @@ -489,7 +505,7 @@ INFOPLIST_KEY_UILaunchScreen_Generation = YES; INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight"; - IPHONEOS_DEPLOYMENT_TARGET = 17.0; + IPHONEOS_DEPLOYMENT_TARGET = 18.0; LD_RUNPATH_SEARCH_PATHS = ( "$(inherited)", "@executable_path/Frameworks", @@ -513,7 +529,7 @@ CURRENT_PROJECT_VERSION = 1; DEVELOPMENT_TEAM = N77R86QZUG; GENERATE_INFOPLIST_FILE = YES; - IPHONEOS_DEPLOYMENT_TARGET = 17.0; + IPHONEOS_DEPLOYMENT_TARGET = 18.0; MARKETING_VERSION = 1.0; PRODUCT_BUNDLE_IDENTIFIER = net.mickf.HoodsAppTests; PRODUCT_NAME = "$(TARGET_NAME)"; @@ -533,7 +549,7 @@ CURRENT_PROJECT_VERSION = 1; DEVELOPMENT_TEAM = N77R86QZUG; GENERATE_INFOPLIST_FILE = YES; - IPHONEOS_DEPLOYMENT_TARGET = 17.0; + IPHONEOS_DEPLOYMENT_TARGET = 18.0; MARKETING_VERSION = 1.0; PRODUCT_BUNDLE_IDENTIFIER = net.mickf.HoodsAppTests; PRODUCT_NAME = "$(TARGET_NAME)"; diff --git a/Examples/HoodsApp/HoodsApp/RootView.swift b/Examples/HoodsApp/HoodsApp/RootView.swift index 4496677..45d3c53 100644 --- a/Examples/HoodsApp/HoodsApp/RootView.swift +++ b/Examples/HoodsApp/HoodsApp/RootView.swift @@ -6,6 +6,11 @@ struct RootView: View { @State var currentPath: Path? @State private var preferredColumn = NavigationSplitViewColumn.detail + static let counterTabStore = Store(initialState: CounterTabFeature.State()) { + CounterTabFeature() + ._printChanges() + } + func navigate(to path: Path) { currentPath = path preferredColumn = .detail @@ -16,6 +21,7 @@ struct RootView: View { case mailer case copyText case imagePicker + case tcaCounter } var body: some View { @@ -25,6 +31,7 @@ struct RootView: View { Button("Mailer") { navigate(to: .mailer) } Button("CopyText") { navigate(to: .copyText) } Button("ImagePicker") { navigate(to: .imagePicker) } + Button("TCA Counter") { navigate(to: .tcaCounter) } } } detail: { switch currentPath { @@ -52,6 +59,8 @@ struct RootView: View { ImagePickerDemoFeature() } ) + case .tcaCounter: + CounterTabView(store: Self.counterTabStore) case nil: VStack { Text("🏘️ Welcome to the ’hoods!") diff --git a/Examples/HoodsApp/HoodsApp/TCATutorial/CounterFeature.swift b/Examples/HoodsApp/HoodsApp/TCATutorial/CounterFeature.swift new file mode 100644 index 0000000..1f89b2e --- /dev/null +++ b/Examples/HoodsApp/HoodsApp/TCATutorial/CounterFeature.swift @@ -0,0 +1,140 @@ +import ComposableArchitecture +import SwiftUI + +@Reducer +struct CounterFeature { + // An equatable state is required by the TestStore + @ObservableState + struct State: Equatable { + var count = 0 + var fact: String? + var isLoading = false + var isTimerRunning = false + } + + enum Action { + case decrementButtonTapped + case incrementButtonTapped + case factButtonTapped + case factResponse(String) + case timerTick + case toggleTimerButtonTapped + } + + enum CancelID { + case timer + } + + @Dependency(\.continuousClock) var clock + @Dependency(\.numberFact) var numberFact + + var body: some ReducerOf { + Reduce { state, action in + switch action { + case .decrementButtonTapped: + state.count -= 1 + state.fact = nil + return .none + + case .factButtonTapped: + state.fact = nil + state.isLoading = true + return .run { [count = state.count] send in + try await send(.factResponse(numberFact.fetch(count))) + } + + case let .factResponse(fact): + state.fact = fact + state.isLoading = false + return .none + + case .incrementButtonTapped: + state.count += 1 + state.fact = nil + return .none + + case .timerTick: + state.count += 1 + state.fact = nil + return .none + + case .toggleTimerButtonTapped: + state.isTimerRunning.toggle() + if state.isTimerRunning { + return .run { send in + for await _ in clock.timer(interval: .seconds(1)) { + await send(.timerTick) + } + } + .cancellable(id: CancelID.timer) + } else { + return .cancel(id: CancelID.timer) + } + } + } + } +} + +struct CounterView: View { + let store: StoreOf + + var body: some View { + VStack { + Text("\(store.count)") + .font(.largeTitle) + .padding() + .background(Color.black.opacity(0.1)) + .cornerRadius(10) + HStack { + Button("-") { + store.send(.decrementButtonTapped) + } + .font(.largeTitle) + .padding() + .background(Color.black.opacity(0.1)) + .cornerRadius(10) + + Button("+") { + store.send(.incrementButtonTapped) + } + .font(.largeTitle) + .padding() + .background(Color.black.opacity(0.1)) + .cornerRadius(10) + } + + Button(store.isTimerRunning ? "Stop timer" : "Start timer") { + store.send(.toggleTimerButtonTapped) + } + .font(.largeTitle) + .padding() + .background(Color.black.opacity(0.1)) + .cornerRadius(10) + + Button("Fact") { + store.send(.factButtonTapped) + } + .font(.largeTitle) + .padding() + .background(Color.black.opacity(0.1)) + .cornerRadius(10) + + if store.isLoading { + ProgressView() + } else if let fact = store.fact { + Text(fact) + .font(.largeTitle) + .multilineTextAlignment(.center) + .padding() + } + } + } +} + +#Preview { + CounterView( + store: Store(initialState: CounterFeature.State()) { + CounterFeature() + } + ) +} diff --git a/Examples/HoodsApp/HoodsApp/TCATutorial/CounterTabFeature.swift b/Examples/HoodsApp/HoodsApp/TCATutorial/CounterTabFeature.swift new file mode 100644 index 0000000..1fe5d2a --- /dev/null +++ b/Examples/HoodsApp/HoodsApp/TCATutorial/CounterTabFeature.swift @@ -0,0 +1,52 @@ +import ComposableArchitecture +import SwiftUI + +@Reducer +struct CounterTabFeature { + struct State: Equatable { + var tab1 = CounterFeature.State() + var tab2 = CounterFeature.State() + } + + enum Action { + case tab1(CounterFeature.Action) + case tab2(CounterFeature.Action) + } + + var body: some ReducerOf { + Scope(state: \.tab1, action: \.tab1) { + CounterFeature() + } + Scope(state: \.tab2, action: \.tab2) { + CounterFeature() + } + Reduce { _, _ in + // Core logic of the app feature + .none + } + } +} + +struct CounterTabView: View { + let store: StoreOf + + var body: some View { + TabView { + Tab("Counter 1", systemImage: "number.square") { + CounterView(store: store.scope(state: \.tab1, action: \.tab1)) + } + + Tab("Counter 2", systemImage: "number.circle.fill") { + CounterView(store: store.scope(state: \.tab2, action: \.tab2)) + } + } + } +} + +#Preview { + CounterTabView( + store: Store(initialState: CounterTabFeature.State()) { + CounterTabFeature() + } + ) +} diff --git a/Examples/HoodsApp/HoodsApp/TCATutorial/NumberFactClient.swift b/Examples/HoodsApp/HoodsApp/TCATutorial/NumberFactClient.swift new file mode 100644 index 0000000..6b99f2c --- /dev/null +++ b/Examples/HoodsApp/HoodsApp/TCATutorial/NumberFactClient.swift @@ -0,0 +1,23 @@ +import ComposableArchitecture +import Foundation + +struct NumberFactClient { + var fetch: (Int) async throws -> String +} + +extension NumberFactClient: DependencyKey { + static let liveValue = Self( + fetch: { number in + let (data, _) = try await URLSession.shared + .data(from: URL(string: "https://statium-monorepo.vercel.app/newsletter/latest?query=\(number)")!) + return String(decoding: data, as: UTF8.self) + } + ) +} + +extension DependencyValues { + var numberFact: NumberFactClient { + get { self[NumberFactClient.self] } + set { self[NumberFactClient.self] = newValue } + } +} diff --git a/Examples/HoodsApp/HoodsAppTests/CounterFeatureTests.swift b/Examples/HoodsApp/HoodsAppTests/CounterFeatureTests.swift new file mode 100644 index 0000000..b0526be --- /dev/null +++ b/Examples/HoodsApp/HoodsAppTests/CounterFeatureTests.swift @@ -0,0 +1,60 @@ +import ComposableArchitecture +import Testing + +@testable import HoodsApp + +@MainActor +struct CounterFeatureTests { + @Test + func basics() async { + let store = TestStore(initialState: CounterFeature.State()) { + CounterFeature() + } + + await store.send(.incrementButtonTapped) { + $0.count = 1 + } + await store.send(.decrementButtonTapped) { + $0.count = 0 + } + } + + @Test + func timer() async { + let clock = TestClock() + + let store = TestStore(initialState: CounterFeature.State()) { + CounterFeature() + } withDependencies: { + $0.continuousClock = clock + } + + await store.send(.toggleTimerButtonTapped) { + $0.isTimerRunning = true + } + await clock.advance(by: .seconds(1)) + await store.receive(\.timerTick) { + $0.count = 1 + } + await store.send(.toggleTimerButtonTapped) { + $0.isTimerRunning = false + } + } + + @Test + func numberFact() async { + let store = TestStore(initialState: CounterFeature.State()) { + CounterFeature() + } withDependencies: { + $0.numberFact.fetch = { "\($0) is a good number." } + } + + await store.send(.factButtonTapped) { + $0.isLoading = true + } + await store.receive(\.factResponse) { + $0.isLoading = false + $0.fact = "0 is a good number." + } + } +} diff --git a/Examples/HoodsApp/HoodsAppTests/CounterTabFeatureTests.swift b/Examples/HoodsApp/HoodsAppTests/CounterTabFeatureTests.swift new file mode 100644 index 0000000..c28611c --- /dev/null +++ b/Examples/HoodsApp/HoodsAppTests/CounterTabFeatureTests.swift @@ -0,0 +1,18 @@ +import ComposableArchitecture +import Testing + +@testable import HoodsApp + +@MainActor +struct CounterTabFeatureTests { + @Test + func incrementInFirstTab() async { + let store = TestStore(initialState: CounterTabFeature.State()) { + CounterTabFeature() + } + + await store.send(\.tab1.incrementButtonTapped) { + $0.tab1.count = 1 + } + } +} diff --git a/Hoods.xcworkspace/xcshareddata/swiftpm/Package.resolved b/Hoods.xcworkspace/xcshareddata/swiftpm/Package.resolved index 1ba32ec..9e41033 100644 --- a/Hoods.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/Hoods.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -42,7 +42,7 @@ "location" : "https://github.com/dirtyhenry/swift-blocks", "state" : { "branch" : "main", - "revision" : "281d5d4627e3d5b11b524cf4223f2fbc09b07910" + "revision" : "b084b311c84fbaf3f5818a59890c5f2567da2588" } }, { @@ -86,8 +86,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/pointfreeco/swift-composable-architecture", "state" : { - "revision" : "69247baf7be2fd6f5820192caef0082d01849cd0", - "version" : "1.16.1" + "revision" : "d602618c628e5123f66643437151079d3664970d", + "version" : "1.17.0" } }, { @@ -95,8 +95,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/pointfreeco/swift-concurrency-extras", "state" : { - "revision" : "163409ef7dae9d960b87f34b51587b6609a76c1f", - "version" : "1.3.0" + "revision" : "82a4ae7170d98d8538ec77238b7eb8e7199ef2e8", + "version" : "1.3.1" } }, { @@ -122,8 +122,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/pointfreeco/swift-dependencies", "state" : { - "revision" : "7d2eb4ad20efb2838269645410d26b64ca48d8aa", - "version" : "1.6.1" + "revision" : "5526c8a27675dc7b18d6fa643abfb64bcb200b77", + "version" : "1.6.2" } }, { @@ -162,6 +162,15 @@ "version" : "1.4.1" } }, + { + "identity" : "swift-sharing", + "kind" : "remoteSourceControl", + "location" : "https://github.com/pointfreeco/swift-sharing", + "state" : { + "revision" : "b68bf99b05cb974392f6ffa380351e9b7391e233", + "version" : "1.1.0" + } + }, { "identity" : "swift-snapshot-testing", "kind" : "remoteSourceControl",