Rendering App Store Screenshots with SwiftUI

When given an option between doing something with a GUI or by typing a set of commands or writing code, I’ll typically choose the latter. Tools like TeX and OpenSCAD really make it feel like I have full control over the engine that drives the tool, unrestricted by the precision of a mouse or my ability to navigate a menu. Plus you can version control these types of files much easier than a the proprietary binary formats usually associated with GUI-based tools.

So, as someone not very adept with typical design tools like Figma or Photoshop, I found myself looking for something more code-based when getting ready to publish my first iOS app, PassKeepr.

The solution took a little while for me to come to, but seems obvious in hindsight now: just render a SwiftUI view into an image. This solution allows you to use all of the typical SwifUI controls to perfectly arrange your assets, text, background, etc., plus you can pull in all the customization options for text, gradients, shadows, etc. that come with SwiftUI.

So, to serve this goal of rendering SwiftUI views as images, I created a little MacOS app. Here’s a quick tutorial on how I got started with it:

Creating a basic view for an App Store screenshot

This part is pretty simple, and it’s really up to you how you want to arrange and customize the view to your liking. Mine ended up looking like this:

Screenshot.swift
import SwiftUI

private let FRAME_WIDTH = 1320
private let FRAME_HEIGHT = 2868

struct ScreenshotView: View {
    var headerText: String
    var subtitleText: String
    var imageName: String // The name of an image added as an xcasset

    let maincolor: Color = .init(hex: 0x173C1C)
    let maincolortint2: Color = .init(hex: 0x456349)

    var body: some View {
        ZStack {
            LinearGradient(colors: [maincolortint2, maincolor], startPoint: .top, endPoint: .bottom)
                .ignoresSafeArea()
            VStack {
                Spacer()
                VStack {
                    Text(headerText)
                        .font(Font.system(size: 100))
                        .fontWeight(.bold)
                        .fontDesign(.rounded)
                        .foregroundStyle(Color.white)
                    Text(subtitleText)
                        .font(Font.system(size: 90))
                        .fontWeight(.bold)
                        .fontDesign(.rounded)
                        .foregroundStyle(Color.white)
                }
                Spacer()
                Image(imageName)
                    .resizable()
                    .scaledToFit()
                    .padding([.leading, .trailing, .bottom], 50)
            }
            .padding()
        }
        .frame(width: CGFloat(FRAME_WIDTH), height: CGFloat(FRAME_HEIGHT))
    }
}

Previewing the view in the Canvas

Since we are setting the frame to the exact dimensions of the image that we want to render, it will be previewed at those dimensions in the Canvas as well, which is much too large:

To fix this, the preview must be updated so both the view and the window that contains the view are scaled down:

private let PREVIEW_SCALE = 0.25

#Preview {
    ScreenshotView(headerText: "Manage your passes", subtitleText: "To edit later", imageName: "Screenshot1")
        .scaleEffect(PREVIEW_SCALE)
        .frame(width: Double(FRAME_WIDTH) * PREVIEW_SCALE, height: Double(FRAME_HEIGHT) * PREVIEW_SCALE)
}

Rendering a view to an image

Once the view is to your liking, you’ll need a way to save it as a .png. The following function uses SwiftUI’s ImageRenderer. The rendered cgImage can then be converted to a .png and saved wherever the user desires.

SaveAsPNG.swift
func saveViewAsPNG<V: View>(view: V, filename: String) {
        let renderer = ImageRenderer(content: view)

        if let image = renderer.cgImage {
            let bitmapRep = NSBitmapImageRep(cgImage: image)
            if let pngData = bitmapRep.representation(using: .png, properties: [:]) {
                let savePanel = NSSavePanel()
                savePanel.title = "Save PNG"
                savePanel.allowedContentTypes = [.png]
                savePanel.nameFieldStringValue = filename

                savePanel.begin { result in
                    if result == .OK, let saveURL = savePanel.url {
                        do {
                            try pngData.write(to: saveURL)
                            print("View saved as PNG at \(saveURL)")
                        } catch {
                            print("Failed to save PNG: \(error)")
                        }
                    }
                }
            } else {
                print("Failed to create PNG data.")
            }
        } else {
            print("Failed to render view as image.")
        }
    }

Wrapping it up into an app

From here, a basic main view with a save button to render each screenshot can be added, and it’s done! A simple little app to render SwiftUI views as images.

ContentView.swift
struct ContentView: View {
    var body: some View {
        Button("Save screenshot 1") {
            saveViewAsPNG(view: ScreenshotView(headerText: "Manage your passes", subtitleText: "To edit later", imageName: "Screenshot1"), filename: "Screenshot1")
        }
        .padding()

        Button("Save screenshot 2") {
            saveViewAsPNG(view: ScreenshotView(headerText: "Customize Passes", subtitleText: "To create unique cards", imageName: "Screenshot2"), filename: "Screenshot2")
        }
        .padding()

        Button("Save screenshot 3") {
            saveViewAsPNG(view: ScreenshotView(headerText: "Scan existing cards", subtitleText: "And save them digitally", imageName: "Screenshot3"), filename: "Screenshot3")
        }
        .padding()
    }
}

Github social preview

After the app screenshots were done, I tried to find some more places to apply this setup. I found the GitHub social media preview, which asks for images with specific dimensions for use when sharing your repo across various social media sites. I could have spent however long it took to fumble through designing and perfecting the layout for my App’s social media preview, or… I could add another view to the screenshots app! Here’s what I came up with:

GitHubSocialPreview.swift
import SwiftUI

private let FRAME_WIDTH = 1280
private let FRAME_HEIGHT = 640
private let PREVIEW_SCALE = 0.25

struct GithubSocialPreview: View {
    let backgroundcolor1: Color = .init(hex: 0x173C1C)
    let backgroundcolor2: Color = .init(hex: 0x102A13)
    let textColor: Color = .init(hex: 0x88BD8F)

    var body: some View {
        ZStack {
            LinearGradient(colors: [backgroundcolor1, backgroundcolor2], startPoint: .top, endPoint: .bottom)
                .ignoresSafeArea()
            HStack {
                Image("RoundedAppIcon")
                    .resizable()
                    .scaledToFit()
                    .padding(60)
                Spacer()
                Text("PassKeepr")
                    .font(.system(size: 200)) // Start with a very large font size
                    .lineLimit(1) // Ensure the text stays on one line
                    .minimumScaleFactor(0.1) // Allow scaling down to fit
                    .fontWeight(.bold)
                    .fontDesign(.rounded)
                    .foregroundStyle(textColor)
                    .shadow(radius: 5.0)
                    .padding(.trailing, 60)
            }
            .padding()
        }
        .frame(width: CGFloat(FRAME_WIDTH), height: CGFloat(FRAME_HEIGHT))
    }
}

#Preview {
    GithubSocialPreview()
        .scaleEffect(PREVIEW_SCALE)
        .frame(width: Double(FRAME_WIDTH) * PREVIEW_SCALE, height: Double(FRAME_HEIGHT) * PREVIEW_SCALE)
}

Which looks like:

Pretty nice for a couple-dozen lines of code, I think.