- 
Flexible release history: Unlike most What's New Screen libraries, Recap can display multiple releases, showcasing your app's entire feature history. 
- 
No code, no problems: Powered by a simple Markdown file, stored locally or loaded remotely. 
- 
Built-in customization: Presents a beautiful, standardized interface out of the box, while offering full customization to match any app's design language. 
- 
Semantic versioning support: Simple logic to help decide when to display your What's New Screen. 
Setting up a screen to display your app's releases is simple:
- Create a markdown file with your releases (we'll cover the format later).
- Add two lines of code.
// Initialize releases from a markdown file in your app's bundle
extension [Release] {
    static var appReleases: [Release] {
        ReleasesParser(fileName: "Releases").releases
    }
}
// Create and display the RecapScreen
RecapScreen(releases: .appReleases)That's all you need to get started!
Your releases are defined in a simple markdown file, with a very simple spec:
# Version Number: [String] (Supports arbitrary number formats such as 1, 1.0, or 1.0.0)
## Release Title: [String]
### Semantic Version Change Type: [Major | Minor | Patch]
- title: [String] (Feature Title)
- description: [String] (Feature Description)
- symbol: [String] (SF Symbol Identifier)
- color: [String] (Hex Color Code or System Color Name)Note
For a full list of supported System Color Names, see Sources/Recap/Internal/Color+SystemNames.swift.
# 1.1
## New Features ❤️
### Minor
- title: Super Cool Feature #1
- description: You won't believe how this works
- symbol: sparkles
- color: cyan
- title: Super Cool Feature #2
- description: You will believe how this one works though
- symbol: text.magnifyingglass
- color: blue
- title: Super Cool Feature #3
- description: This feature is for the nerds in the house
- symbol: apple.terminal.on.rectangle
- color: orange
# 1.0.1
## Bug Fixes! 🐛
### Patch
- title: Welcome to some bug fixes!
- description: My bad y'all…
- symbol: ladybug.circle
- color: #FFC933
# 1.0
## Introducing Recap 🥳
### Major
- title: Welcome to Recap
- description: It's live! 🥰
- symbol: party.popper
- color: #F72585With just a few lines of markdown we've built a Releases screen that has three pages, with a consistent look and feel.
Recap provides various modifiers to tailor the Releases screen to your app's visual identity:
// Available customization modifiers
func recapScreenStartIndex(_ startIndex: RecapScreenStartIndex) -> some View
func recapScreenTitleStyle(_ style: some ShapeStyle) -> some View
func recapScreenDismissButtonStyle(_ style: some ShapeStyle) -> some View
func recapScreenDismissButtonStyle(_ backgroundStyle: some ShapeStyle, _ foregroundStyle: some ShapeStyle) -> some View
func recapScreenDismissButtonTitle(_ title: LocalizedStringResource) -> some View
func recapScreenIconFillMode(_ style: IconFillMode) -> some View
func recapScreenIconAlignment(_ alignment: VerticalAlignment) -> some View
func recapScreenPageIndicatorColors(selected: Color, deselected: Color) -> some View
func recapScreenBackground(_ style: AnyShapeStyle?) -> some View
func recapScreenBackground(_ color: Color) -> some View
func recapScreenPadding(_ insets: EdgeInsets) -> some View
func recapScreenHeaderSpacing(_ spacing: CGFloat) -> some View
func recapScreenItemSpacing(_ spacing: CGFloat) -> some View
func recapScreenDismissAction(_ dismissAction: (() -> Void)?) -> some ViewExample usage:
RecapScreen(releases: .appReleases)
    .recapScreenTitleStyle(LinearGradient(colors: [.purple, .pink, .orange, .yellow], startPoint: .topLeading, endPoint: .bottomTrailing))
    .recapScreenDismissButtonStyle(Color.pink, Color.white)
    .recapScreenIconFillMode(.gradient)
    .recapScreenPageIndicatorColors(
        selected: Color.pink,
        deselected: Color.gray
    )RecapScreen also supports leading and trailing views:
RecapScreen(
    releases: .plinkyReleases
    leadingView: {
        UpcomingRoadmapView()
    }, trailingView: {
        SupportView()
    }
)
.recapScreenStartIndex(.release())
// Alternatively: .recapScreenStartIndex(.leadingView) or .recapScreenStartIndex(.trailingView)In my app Plinky, I use the leading view to display the app's upcoming roadmap ahead of the most recent features, and the trailing view displays a support screen for people to reach out to me after browsing the feature list.
RecapDisplayPolicy and RecapDisplayPolicy.Trigger provide a handy way to define when your Recap screen should display. It encapsulates the most common strategies people choose for displaying a What's New screen, for example "when there's a new version that has release notes" or "if there are any release notes since the last version the user launched". It achieves this with a composable fluent syntax, and the rest is handled for you.
Below is a simplified example of how you can present a Recap screen. In this case we will display our screen when there are release notes for any version since the last launch, and the update is notable (major/minor).
func presentRecapScreen() {
    let currentVersion = (Bundle.main.object(forInfoDictionaryKey: "CFBundleShortVersionString") as? String) ?? "0"
    let previousVersion = UserDefaults.standard.string(forKey: "previouslyLaunchedVersion")
    // Versions of your app that have release notes
    let releaseVersions: [SemanticVersion] = [Release].appReleases.map(\.version.semanticVersion)
	// Create a RecapDisplayPolicy based on the app's current version, the previously launched version, and your release notes.
    let policy = RecapDisplayPolicy(
        currentVersion: SemanticVersion(version: currentVersion),
        previousVersion: previousVersion.map(SemanticVersion.init(version:)),
        releases: releaseVersions
    )
	// A Trigger that shows the release notes for notable versions 
	// for a user who hasn't seen opened the app since the last version with release notes. 
    let trigger = RecapDisplayPolicy.Trigger
        .updateWindow(.sincePrevious)
        .notability(.notableOnly)
	// An alternative example trigger that always shows release notes for any new version.
    // let trigger = RecapDisplayPolicy.Trigger
    //     .updateWindow(.current)
    //     .notability(.any)
    //     .ignoringReleaseNotesRequirement()
    if policy.shouldTrigger(using: trigger) {
		// Present a RecapScreen in your app in a contextually relevant manner. 
		let recapScreen = RecapScreen(releases: .appReleases)
        self.router.present(recapScreen)
    }
    // Persist version state for next launch
    UserDefaults.standard.set(currentVersion, forKey: "previouslyLaunchedVersion")
}Recap includes a SemanticVersion type to represent and compare versions using the standard major.minor.patch scheme. It powers the display policy described below and can be used directly anywhere version comparisons are needed.
Examples:
- SemanticVersion(version: "1.2.3")
- Compare with ==,<,>to implement custom logic
Try Recap with the demo project to see if it fits your app's needs. 📱
- iOS 17.0+
- Xcode 14+
The Swift Package Manager is a tool for automating the distribution of Swift code and is integrated into the Swift build system.
Once you have your Swift package set up, adding Recap as a dependency is as easy as adding it to the dependencies value of your Package.swift.
dependencies: [
    .package(url: "https://github.com/mergesort/Recap.git", .upToNextMajor(from: "1.0.0"))
]If you prefer not to use SPM, you can integrate Recap into your project manually by copying the files in.
Hi, I'm Joe everywhere on the web, but especially on Threads.
See the license for more information about how you can use Recap.
Recap is a labor of love to help developers build better apps, making it easier for you to unlock your creativity and make something amazing for your yourself and your users. If you find Recap valuable I would really appreciate it if you'd consider helping sponsor my open source work, so I can continue to work on projects like Recap to help developers like yourself.

