mirror of
https://github.com/pointfreeco/swift-composable-architecture.git
synced 2025-12-20 09:11:33 +01:00
* bring back view store performance
* wip
* Allow chaining of store bindings
* wip
* Localize ignoring bindings to text field resignation/dismissal
* wip
* fix DiagnosticsError message (#2597)
* store collection
* wip
* wip
* update migration guide
* Add `@Presents` macro for observable presentation
While it would be nice for the `@PresentationState` property wrapper to
"just work" with TCA's upcoming observable tools, sadly that does not
seem to be the case. Adding observation directly to
`@PresentationState`, as we have done with the beta so far, can break
existing projects due to the additional observation. This primarily
manifests itself in projects that present navigation stacks, where the
`@PresentationState` observation can cause the navigation hierarchy to
recompute and trigger SwiftUI bugs.
The best we've come up with so far is introducing a brand new macro that
automatically wraps a property with `@PresentationState` _and_
instruments it with observation.
We're open to other ideas, and we do have future plans to eliminate the
need for a property wrapper or macro at all, but till then this offers a
non-breaking upgrade path!
* fixes
* Observe child store changes
* wip
* wip
* wip
* Fix typo in MigratingTo1.6.md (#2608)
* Rename bindingViewStore argument to store in MigratingTo1.6.md (#2611)
* wip
* Revert "wip"
This reverts commit f221ed0e1a.
* Add `@Presents` macro for observable presentation (#2604)
* Add `@Presents` macro for observable presentation
While it would be nice for the `@PresentationState` property wrapper to
"just work" with TCA's upcoming observable tools, sadly that does not
seem to be the case. Adding observation directly to
`@PresentationState`, as we have done with the beta so far, can break
existing projects due to the additional observation. This primarily
manifests itself in projects that present navigation stacks, where the
`@PresentationState` observation can cause the navigation hierarchy to
recompute and trigger SwiftUI bugs.
The best we've come up with so far is introducing a brand new macro that
automatically wraps a property with `@PresentationState` _and_
instruments it with observation.
We're open to other ideas, and we do have future plans to eliminate the
need for a property wrapper or macro at all, but till then this offers a
non-breaking upgrade path!
* wip
* Fix perception bindings (#2609)
* Fix runtime warning when binding accesses perceptible state.
* Fix runtime warning in SwiftUI bindings.
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* wip
* wip
* fix
* wip
* wip
* wip
* Check observable state identity for presentation state.
* Add willSset/didSet to registrar types.
* clean up @Presents
* clean up
* fix
* Emit observation warnings in escaping contexts like `ForEach` and `sheet` (#2613)
* Fix perception warning in ForEach.
* fix
---------
Co-authored-by: Brandon Williams <mbrandonw@hey.com>
* Introduce @ViewAction(for:) macro. (#2612)
* Add back @ViewAction macro.
* wip
* wip
* wip
* wip
* wip
* clean up
* wip
* wip
* fix migration guide'
* ViewActionable
* wip
* rename
* wip
* wip
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* Introduce @BindableStore for bindings in pre-iOS 17 (#2610)
* Introduce @BindableStore.
* docs
* wip
* wip
* fixc
* wip
* wip
* wip
* wip
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* re-record intergration logs
* wip
* wip
* localize invalid stores to store collection
* Deprecate closure-based `store.scope` operations (#2618)
These uncached operations can be problematic, especially when working
with observation, which often depends on the stable identity of stores.
* document
* Update warning message
* Performance Improvement: Skip perception checks when calling reducers. (#2622)
* Skip perception checks when calling reducers.
* inline withoutPerceptionChecking() for RELEASE
Co-authored-by: Brandon Williams <135203+mbrandonw@users.noreply.github.com>
---------
Co-authored-by: Brandon Williams <135203+mbrandonw@users.noreply.github.com>
* Don't show perception warnings in action closures. (#2614)
* Don't show perception warnings in action closures.
* wip
* wip
* wip
* clean up
* wip
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* fix BindableStore + release
* Add docs
* Change associated type names of ViewActionSending (#2629)
* Fix some @ViewAction annoyances.
* wip
* wip
* wip
* wip
* wip
* wip
* fix
* wip
* fixed merge
* Add new view modifiers for observing alerts/dialogs (#2628)
* Add new view modifiers for observing alerts/dialogs
Instead of:
```swift
.alert(store: store.scope(state: \.$alert, action: \.alert))
```
You can now do:
```swift
.alert($store.scope(state: \.alert, action: \.alert))
```
This new modifier is powered by the same store binding scope operation
that can power `sheet(item:)`, etc., and is much lighter weight than the
previous view modifier, which spun up view stores and `WithViewStore`
views.
* wip
* wip
* wip
---------
Co-authored-by: Brandon Williams <mbrandonw@hey.com>
* Fix uncached warning when using Store.ifLet (#2625)
* Fix uncached warning when using Store.ifLet
* wip
* wip
* wip
* wip
* wip
* wip
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* Resolve packages
* Updated scopes
* wip
* wip
* updated binding docs
* adding docs
* clean up
* wip
* wip
* wip
* clean up
* clean up
* clean up
* wip;
* lots of fixes
* update more docs
* fix
* wip
* wip
* Remove ObservationRegistrarWrapper. (#2634)
* Remove ObservationRegistrarWrapper.
* Delete Sources/ComposableArchitecture/Internal/ObservationRegistrarWrapper.swift
---------
Co-authored-by: Stephen Celis <stephen@stephencelis.com>
* more docs
* update docs
* a few more tests
* fix
* wip
* wip
* wip
* Cache data in store collections (#2635)
* fix tutorial highlighting
* wip
* wip
* wip
* wip
* tests for observation of special domain types
* another test
* fix
* wip
* Implement memoization for perception checks (#2630)
* Implement memoization for isInSwiftUIBody
* tidy up
* Perception caching updates (#2649)
* Small updates to perception caching.
* wip
* debug
* some more macro tests
* syncups tutorial beginnings
* wip
* wip
* wip
* wip
* wip
* merge fixes
* wip
* update tests
* fix
* fix
* fix perception checking in store
* rename task local
* delete old test
* deprecate test using old apis
* fix test
* perception tests for store
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* Opt out of key path for Store.ifLet
* sync ups
* lots more sync up tutorial
* more sync ups tutorial
* wip
* wip
* wio
* wip
* wip
* wip
* updated references of 1.6 to 1.7
* wip
* no need to force unwrap here
* fixed crash in ForEach with bindings
* more sync ups tutorial
* more sync ups tutorial
* wip
* more sync ups
* wip
* wip
* Better support for observing copies of values (#2650)
* Explore using _modify
* wip
* wip
* wip
* wip
* wip
* wip
* more tests
* wip
* get another failing test for an edge case
* wip
* tests all passing
* flag for determining when new state was created
* wip
* clean up
* wip
* wip
* wip;
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* New test that currently fails.
* wip
* wip
* Update Sources/ComposableArchitectureMacros/PresentsMacro.swift
* wip
* remove redundant attached member attribute
* storage
* cleanup
* more benchmarks and tests
* wip
* wip
* wip
* wip
* update tests
* wip
* wip
---------
Co-authored-by: Brandon Williams <mbrandonw@hey.com>
* wip
* wip
* wip
* swift-format
* fix
* wip
* wip
* wip
* wip
* Perception
* wip
* wip
* clean up shared state
* fix shared state tests
* wip
* add alert test
* wip
* wip
* wip
* wip
* Use transaction in binding
* wip
* wip
* wip
* wip
* wip
* wip
* uikit
* keep references to controllers when presenting so that we can properly dismiss
* change order of features in shared state demo
* wip
* cleanup
* cleanup
* wip
* wip
* wip
* Fix perception checking for effect actions.
* wip
* wip
* wip
* Fix perception checking for effect actions.
* wip
* wip
* remove sync ups tutorial
* wip
* wip
* wip
* wip
* wip
* docs for observe function for uikit
* Add cancellation to observation'
* re-record integration test snapshots
* fixed some todos
* update test
* remove 5.9.2 checks
* wip
* improve docs
* update docs
* updates
* lots of fixes
* more docs
* remove unneeded file;
* wip
* wip
* wip
* update readme and getting started
* wip
* simplify
* migration stuff
* wip
* Update Models.swift
* wip
* wip
* wip
* Update Bindings.md
* wip
* wip
* wip
* wip
* fix
* wip
* wip
* wip
* wip
* wip
Co-authored-by: Kabir Oberai <oberai.kabir@gmail.com>
---------
Co-authored-by: Brandon Williams <mbrandonw@hey.com>
Co-authored-by: hmhv <admin@hmhv.info>
Co-authored-by: Jimmy Prime <jimmylevelup@gmail.com>
Co-authored-by: Michael Pohl <15653162+Mika5652@users.noreply.github.com>
Co-authored-by: Brandon Williams <135203+mbrandonw@users.noreply.github.com>
Co-authored-by: George Scott <gscott@gekkoto.com>
Co-authored-by: Kabir Oberai <oberai.kabir@gmail.com>
147 lines
4.0 KiB
Swift
147 lines
4.0 KiB
Swift
import Accessibility
|
|
import CustomDump
|
|
import InlineSnapshotTesting
|
|
import XCTest
|
|
|
|
@MainActor
|
|
class BaseIntegrationTests: XCTestCase {
|
|
var app: XCUIApplication!
|
|
var logs: XCUIElement!
|
|
private var _expectRuntimeWarnings: (file: StaticString, line: UInt)?
|
|
|
|
func expectRuntimeWarnings(file: StaticString = #file, line: UInt = #line) {
|
|
self._expectRuntimeWarnings = (file, line)
|
|
}
|
|
|
|
override func setUp() async throws {
|
|
//SnapshotTesting.isRecording = true
|
|
// self.continueAfterFailure = false
|
|
self.app = XCUIApplication()
|
|
self.app.launchEnvironment["UI_TEST"] = "true"
|
|
self.app.launch()
|
|
self.app.activate()
|
|
self.logs = self.app.staticTexts["composable-architecture.debug.logs"]
|
|
}
|
|
|
|
override func tearDown() {
|
|
super.tearDown()
|
|
if let (file, line) = self._expectRuntimeWarnings {
|
|
XCTAssert(
|
|
self.app.staticTexts["Runtime warning"].waitForExistence(timeout: 1),
|
|
"Expected runtime warning(s)",
|
|
file: file,
|
|
line: line
|
|
)
|
|
} else {
|
|
XCTAssertFalse(
|
|
self.app.staticTexts["Runtime warning"].exists,
|
|
"\(self.name) emitted an unexpected runtime warning"
|
|
)
|
|
}
|
|
SnapshotTesting.isRecording = false
|
|
}
|
|
|
|
func clearLogs() {
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) {
|
|
let alert = XCUIApplication(bundleIdentifier: "com.apple.springboard").alerts
|
|
let open = alert.buttons["Open"]
|
|
if alert.firstMatch.waitForExistence(timeout: 0.3),
|
|
open.waitForExistence(timeout: 0.3)
|
|
{
|
|
alert.buttons["Open"].tap()
|
|
}
|
|
}
|
|
XCUIDevice.shared.system.open(URL(string: "integration:///clear-logs")!)
|
|
}
|
|
|
|
func assertLogs(
|
|
_ logConfiguration: LogConfiguration = .unordered,
|
|
matches expectedLogs: (() -> String)? = nil,
|
|
file: StaticString = #file,
|
|
function: StaticString = #function,
|
|
line: UInt = #line,
|
|
column: UInt = #column
|
|
) {
|
|
defer { self.clearLogs() }
|
|
let logs: String
|
|
switch logConfiguration {
|
|
case .exact:
|
|
logs = self.logs.label
|
|
case .unordered:
|
|
logs = self.logs.label.split(separator: "\n").sorted().joined(separator: "\n")
|
|
}
|
|
assertInlineSnapshot(
|
|
of: logs,
|
|
as: ._lines,
|
|
matches: expectedLogs,
|
|
file: file,
|
|
function: function,
|
|
line: line,
|
|
column: column
|
|
)
|
|
}
|
|
}
|
|
|
|
enum LogConfiguration {
|
|
case exact
|
|
case unordered
|
|
}
|
|
|
|
extension Snapshotting where Value == String, Format == String {
|
|
fileprivate static let _lines = Snapshotting(
|
|
pathExtension: "txt",
|
|
diffing: Diffing(
|
|
toData: { Data($0.utf8) },
|
|
fromData: { String(decoding: $0, as: UTF8.self) }
|
|
) { old, new in
|
|
guard old != new else { return nil }
|
|
|
|
let newLines = new.split(separator: "\n", omittingEmptySubsequences: false)
|
|
|
|
let oldLines = old.split(separator: "\n", omittingEmptySubsequences: false)
|
|
let difference = newLines.difference(from: oldLines)
|
|
|
|
var result = ""
|
|
|
|
var insertions = [Int: Substring]()
|
|
var removals = [Int: Substring]()
|
|
|
|
for change in difference {
|
|
switch change {
|
|
case let .insert(offset, element, _):
|
|
insertions[offset] = element
|
|
case let .remove(offset, element, _):
|
|
removals[offset] = element
|
|
}
|
|
}
|
|
|
|
var oldLine = 0
|
|
var newLine = 0
|
|
|
|
while oldLine < oldLines.count || newLine < newLines.count {
|
|
if let removal = removals[oldLine] {
|
|
result += "\(oldPrefix) \(removal)\n"
|
|
oldLine += 1
|
|
} else if let insertion = insertions[newLine] {
|
|
result += "\(newPrefix) \(insertion)\n"
|
|
newLine += 1
|
|
} else {
|
|
result += "\(prefix) \(oldLines[oldLine])\n"
|
|
oldLine += 1
|
|
newLine += 1
|
|
}
|
|
}
|
|
|
|
let attachment = XCTAttachment(
|
|
data: Data(result.utf8),
|
|
uniformTypeIdentifier: "public.patch-file"
|
|
)
|
|
return (result, [attachment])
|
|
}
|
|
)
|
|
}
|
|
|
|
private let oldPrefix = "\u{2212}"
|
|
private let newPrefix = "+"
|
|
private let prefix = "\u{2007}"
|