获取流程与付费墙 - iOS

What getFlow retrieves
Flows BETA Built in Flow builder — renders natively on device, no WebView
Paywall Builder paywalls All existing Paywall Builder content

设计好流程或付费墙编辑工具付费墙之后,你可以在移动应用中展示它。第一步是获取与版位关联的流程或付费墙及其视图配置,具体方法如下所示。

想看看 Adapty SDK 是如何集成到移动应用中的真实示例?查看我们的示例应用,其中展示了完整的集成流程,包括展示付费墙、发起购买以及其他基础功能。

开始之前
  1. 在 Adapty 看板中创建产品
  2. 在 Adapty 看板中创建付费墙并将产品加入其中
  3. 在 Adapty 看板中创建版位并将付费墙加入其中
  4. 在移动应用中安装 Adapty SDK

获取流程/付费墙

如果你已通过 Flow Builder 或付费墙编辑工具设计了流程或付费墙,无需在移动端代码中手动编写渲染逻辑来向用户展示它。这类流程或付费墙本身已包含展示内容和展示方式。不过,你仍需通过版位获取其 ID 和视图配置,然后在移动端将其呈现出来。 为了确保最佳性能,请尽早获取流程或付费墙及其视图配置,以便在向用户展示之前有足够的时间下载图片。

使用 getFlow 方法获取流程或付费墙:

参数:

参数是否必填描述
placementId必填目标版位的标识符。即你在 Adapty 看板创建版位时指定的值。
fetchPolicy默认值:.reloadRevalidatingCacheData

默认情况下,SDK 会尝试从服务器加载数据,失败时返回缓存数据。我们推荐使用此方式,因为它能确保用户始终获取最新数据。

不过,如果你认为用户的网络连接不稳定,可以考虑使用 .returnCacheDataElseLoad——在缓存存在时优先返回缓存数据。这样用户可能无法获取最新数据,但加载速度会更快,不受网络质量影响。缓存会定期更新,因此在会话期间使用缓存以减少网络请求是安全的。

请注意,缓存在应用重启后依然保留,仅在应用重新安装或手动清理时才会清除。

Adapty SDK 通过两层机制在本地存储付费墙:上述定期更新的缓存,以及备用付费墙。我们还使用 CDN 加快付费墙的加载速度,并设有独立的备用服务器,以防 CDN 不可用。该系统旨在确保你始终获取最新版本的付费墙,同时在网络条件有限的情况下也能保证可靠性。

loadTimeout默认值:5 秒

该值限制此方法的超时时间。若达到超时时间,将返回缓存数据或本地备用数据。

请注意,在极少数情况下,此方法的实际超时时间可能略晚于 loadTimeout 中指定的值,因为该操作在底层可能包含多个不同的请求。

响应参数:
参数描述
Flow一个 AdaptyFlow 对象,包含版位、标识符(idvariationId)、名称、远程配置,以及 hasViewConfiguration 标志(用于指示该流程是否包含视图配置)。如需预加载产品、自定义 UI 或以编程方式进行检查,请调用 getPaywallProducts(flow:)

获取视图配置

获取流程或付费墙后,通过 flow.hasViewConfiguration 检查其是否包含视图配置。该标志用于区分版位在 Adapty 看板中的设计方式:

  • true — 该版位使用 Flow Builder(流程)或 Paywall Builder(付费墙)设计,Adapty 会自动为你渲染 UI。请继续执行以下步骤来获取视图配置,并展示流程或付费墙
  • false — 该版位是不含 Builder UI 的自定义付费墙。 使用 getFlowConfiguration 方法加载视图配置。

guard flow.hasViewConfiguration else {
    // handle as remote config paywall
    return
}

let flowConfiguration = try await AdaptyUI.getFlowConfiguration(forFlow: flow)

参数:

参数必填性描述
forFlow必填通过 Adapty.getFlow 获取的 AdaptyFlow 对象。
locale

可选

默认值:nil

付费墙本地化的标识符。格式为语言代码,可包含一个或两个以 - 分隔的子标签(如 enpt-br)。详见本地化与语言代码
loadTimeout默认值:5 秒该参数限制此方法的超时时间。超时后将返回缓存数据或本地备用数据。请注意,在极少数情况下,由于该方法底层可能包含多个请求,实际超时时间可能略晚于 loadTimeout 中设定的值。
products可选提供 AdaptyPaywallProduct 对象数组,以优化产品在屏幕上的显示时机。若传入 nil,AdaptyUI 将自动获取所需产品。
systemRequestsHandler可选符合 AdaptySystemRequestsHandler 协议的对象,用于处理流程操作触发的系统权限请求和评价请求。仅当流程中包含此类操作时才需要提供。
assetsResolver可选类型为 [String: AdaptyCustomAsset] 的字典,用于覆盖流程/付费墙中的图片和视频资源。详见自定义资源
timerResolver可选符合 AdaptyTimerResolver 协议的对象,用于为开发者自定义计时器提供结束时间。详见设置开发者自定义计时器
加载完成后,展示流程/付费墙

为默认目标受众获取流程或付费墙以加快加载速度

通常情况下,流程和付费墙的获取速度几乎是即时的,无需担心性能问题。但如果你的版位和目标受众数量较多,而用户的网络状况又较差,获取流程或付费墙的时间可能会比预期更长。在这种情况下,你可能希望展示一个默认的流程或付费墙,以保证用户体验的流畅性,而不是让用户看到空白页面。 为了解决这个问题,你可以使用 getFlowForDefaultAudience 方法,该方法会获取指定版位中 All Users 目标受众的流程或付费墙。但请务必了解,推荐的做法是使用 getFlow 方法获取流程或付费墙,详情请参阅上方的获取付费墙信息部分。

为什么我们推荐使用 getFlow

getFlowForDefaultAudience 方法存在以下几个明显缺点:

  • 潜在的向后兼容性问题:如果你需要针对不同的应用版本(当前版本和未来版本)展示不同的付费墙,可能会遇到挑战。你要么设计兼容当前(旧版)版本的付费墙,要么接受当前(旧版)用户可能会遇到付费墙无法渲染的问题。
  • 失去精准定向:所有用户都将看到为 All Users 目标受众设计的同一个付费墙,这意味着你将失去个性化定向能力(包括基于国家、营销归因或自定义用户属性的定向)。 如果你愿意接受这些限制,以换取更快的流程或付费墙加载速度,请按如下方式使用 getFlowForDefaultAudience 方法。否则,请继续使用上文介绍的 getFlow
Adapty.getFlowForDefaultAudience(placementId: "YOUR_PLACEMENT_ID") { result in
    switch result {
        case let .success(flow):
            // the requested flow
        case let .failure(error):
            // handle the error
    }
}
参数是否必填描述
placementId必填版位的标识符,即你在 Adapty 看板中创建版位时指定的值。
fetchPolicy默认值:.reloadRevalidatingCacheData

默认情况下,SDK 会尝试从服务器加载数据,若加载失败则返回缓存数据。我们推荐使用此方式,因为它能确保用户始终获取最新数据。

不过,如果你认为用户的网络环境不稳定,可以考虑使用 .returnCacheDataElseLoad——有缓存时直接返回缓存数据。这样用户获取的数据可能不是最新的,但无论网络状况如何,加载速度都会更快。缓存会定期更新,因此在会话期间使用缓存来减少网络请求是安全的。

请注意,重启应用后缓存依然保留,只有卸载重装或手动清除时才会被清空。

自定义资源

要自定义付费墙/流程中的图片和视频,请实现自定义资源。

主图和视频有预定义的 ID:hero_imagehero_video。在自定义资源包中,你通过这些 ID 来定位对应元素并自定义其行为。

对于其他图片和视频,你需要在 Adapty 看板中设置自定义 ID

例如,你可以:

  • 向部分用户展示不同的图片或视频。
  • 在远程主图加载时,先展示本地预览图。
  • 在视频播放前展示预览图。
  • 提供视频的像素分辨率,以便播放器在视频加载前预留布局空间(宽高比 = width / height)。传入 nil 可跳过此项。

下面是一个通过简单字典提供自定义资源的示例:

let customAssets: [String: AdaptyCustomAsset] = [
    // Show a local image using a custom ID
    "custom_image": .image(
        .uiImage(value: UIImage(named: "image_name")!)
    ),

    // Show a local preview image while a remote main image is loading
    "hero_image": .image(
        .remote(
            url: URL(string: "https://example.com/image.jpg")!,
            preview: UIImage(named: "preview_image")
        )
    ),

    // Show a local video with a preview image and a known resolution
    "hero_video": .video(
        .file(
            url: Bundle.main.url(forResource: "custom_video", withExtension: "mp4")!,
            preview: .uiImage(value: UIImage(named: "video_preview")!),
            resolution: CGSize(width: 1080, height: 1920)
        )
    ),
]

let flowConfig = try await AdaptyUI.getFlowConfiguration(
    forFlow: flow,
    assetsResolver: customAssets
)

如果找不到某个资源,付费墙/流程将回退到其默认外观。

设置开发者定义的计时器

要在移动应用中使用自定义计时器,请创建一个遵循 AdaptyTimerResolver 协议的对象。该对象定义每个自定义计时器的渲染方式。如果您愿意,也可以直接使用 [String: Date] 字典,因为它已经符合该协议。以下是一个示例:

@MainActor
struct AdaptyTimerResolverImpl: AdaptyTimerResolver {
    func timerEndAtDate(for timerId: String) -> Date {
        switch timerId {
        case "CUSTOM_TIMER_6H":
            Date(timeIntervalSinceNow: 3600.0 * 6.0) // 6 hours
        case "CUSTOM_TIMER_NY":
            Calendar.current.date(from: DateComponents(year: 2025, month: 1, day: 1)) ?? Date(timeIntervalSinceNow: 3600.0)
        default:
            Date(timeIntervalSinceNow: 3600.0) // 1 hour
        }
    }
}

在此示例中,CUSTOM_TIMER_NYCUSTOM_TIMER_6H 是你在 Adapty 看板中设置的开发者自定义计时器的 Timer IDtimerResolver 确保你的应用动态地为每个计时器更新正确的值。例如:

  • CUSTOM_TIMER_NY:距离计时器结束(例如元旦)的剩余时间。
  • CUSTOM_TIMER_6H:从用户打开付费墙时开始计算的 6 小时倒计时的剩余时间。

Adapty 看板的付费墙编辑工具中完成付费墙的视觉设计之后,你可以在移动应用中展示它。第一步是获取与版位关联的付费墙及其视图配置,具体步骤如下。

请注意,本文介绍的是通过付费墙编辑工具自定义的付费墙。如果你是手动实现付费墙,请参阅获取远程配置付费墙的付费墙和产品

在您开始在移动应用中展示付费墙之前,请确保已完成以下步骤:

在您开始在移动应用中展示付费墙之前
  1. 在 Adapty 看板中创建产品
  2. 在 Adapty 看板中创建付费墙并将产品添加到其中
  3. 在 Adapty 看板中创建版位并将付费墙添加到其中
  4. 在移动应用中安装 Adapty SDK

获取使用付费墙编辑工具设计的付费墙

如果您已使用付费墙编辑工具设计了付费墙,则无需在移动应用代码中手动处理渲染逻辑来向用户展示它。此类付费墙同时包含应展示的内容及展示方式。尽管如此,您仍需通过版位获取其 ID、获取视图配置,然后在移动应用中进行展示。

为确保最佳性能,务必尽早获取付费墙及其视图配置,以便在向用户展示之前有足够的时间下载图片。

使用 getPaywall 方法获取付费墙:

参数:

参数是否必填描述
placementId必填目标版位的标识符。这是您在 Adapty 控制台中创建版位时指定的值。
locale

可选

默认值:en

付费墙本地化的标识符。该参数应为由一个或两个子标签通过减号(-)分隔的语言代码。第一个子标签表示语言,第二个表示地区。

示例:en 表示英语,pt-br 表示巴西葡萄牙语。

有关语言代码及推荐使用方式的更多信息,请参阅本地化与语言代码

fetchPolicy默认值:.reloadRevalidatingCacheData

默认情况下,SDK 会尝试从服务器加载数据,若加载失败则返回缓存数据。我们推荐此方式,因为它能确保用户始终获得最新数据。

但是,如果您认为用户的网络连接不稳定,可以考虑使用 .returnCacheDataElseLoad,在缓存存在时返回缓存数据。这样用户可能无法获得最新数据,但无论网络状况如何,都能享受更快的加载速度。缓存会定期更新,因此在会话期间使用缓存以避免网络请求是安全的。

请注意,缓存在应用重启后仍然有效,仅在应用卸载重装或手动清除时才会被清空。

Adapty SDK 通过两层方式在本地存储付费墙:上述定期更新的缓存和备用付费墙。我们还使用 CDN 加速付费墙获取,并在 CDN 不可用时提供独立的备用服务器。该系统旨在确保您始终获取最新版本的付费墙,同时在网络条件较差时也能保证可靠性。

loadTimeout默认值:5 秒

此值限制该方法的超时时间。超时后将返回缓存数据或本地备用数据。

请注意,在极少数情况下,此方法的实际超时时间可能略长于 loadTimeout 中指定的时间,因为该操作在底层可能包含多个不同请求。

响应参数:

参数描述
Paywall一个 AdaptyPaywall 对象,包含产品 ID 列表、付费墙标识符、远程配置及其他若干属性。

获取使用付费墙编辑工具设计的付费墙视图配置

请确保在付费墙编辑工具中开启了 Show on device 开关。如果未开启该选项,将无法获取视图配置。

获取付费墙后,检查其是否包含视图配置——视图配置的存在表明该付费墙是通过付费墙编辑工具创建的。这将指导你如何展示付费墙。如果存在视图配置,则将其视为付费墙编辑工具付费墙;否则,将其作为远程配置付费墙处理。 使用 getPaywallConfiguration 方法加载视图配置。


guard paywall.hasViewConfiguration else {
    //  use your custom logic
    return
}

do {
    let paywallConfiguration = try await AdaptyUI.getPaywallConfiguration(
            forPaywall: paywall,
            products: products
    )
    // use loaded configuration
} catch {
    // handle the error
}

参数:

参数是否必填描述
paywall必填用于获取目标付费墙控制器的 AdaptyPaywall 对象。
loadTimeout默认:5 秒该值限制此方法的超时时间。若超时,将返回缓存数据或本地备用内容。请注意,由于该方法底层可能包含多个请求,在极少数情况下实际超时时间可能略晚于 loadTimeout 中指定的值。
products可选传入 AdaptyPaywallProduct 对象数组,以优化产品在屏幕上的展示时机。若传入 nil,AdaptyUI 将自动获取所需产品。

如果您使用多种语言,请了解如何添加付费墙编辑工具本地化以及如何正确使用语言代码,详见此处

加载完成后,展示付费墙

获取默认目标受众的付费墙以加速获取

通常情况下,付费墙几乎可以立即获取,因此无需担心加速此过程。但是,当您拥有大量目标受众和付费墙,且用户网络连接较弱时,获取付费墙可能需要较长时间。在这种情况下,您可能希望展示默认付费墙以确保流畅的用户体验,而不是不显示任何付费墙。

要解决这个问题,你可以使用 getPaywallForDefaultAudience 方法,该方法会获取指定版位中针对 All Users 目标受众的付费墙。但请务必注意,推荐的方式仍然是使用 getPaywall 方法获取付费墙,详见上方的获取付费墙信息章节。

为什么我们推荐使用 getPaywall

getPaywallForDefaultAudience 方法存在以下几个明显缺点:

  • 潜在的向后兼容性问题:如果您需要为不同版本的应用(当前版本和未来版本)展示不同的付费墙,可能会面临挑战。您要么必须设计支持当前(旧版)版本的付费墙,要么接受使用当前(旧版)版本的用户可能遇到付费墙无法渲染的问题。
  • 失去精准定向:所有用户都将看到为 All Users 目标受众设计的同一付费墙,这意味着您将失去个性化定向能力(包括基于国家、营销归因或自定义属性的定向)。

如果您愿意接受这些缺点以换取更快的付费墙获取速度,请按如下方式使用 getPaywallForDefaultAudience 方法。否则,请坚持使用上文描述的 getPaywall

Adapty.getPaywallForDefaultAudience(placementId: "YOUR_PLACEMENT_ID", locale: "en") { result in
    switch result {
        case let .success(paywall):
            // the requested paywall
        case let .failure(error):
            // handle the error
    }
}

getPaywallForDefaultAudience 方法从 iOS SDK 2.11.2 版本起可用。

参数是否必填描述
placementId必填版位的标识符。这是您在 Adapty 控制台中创建版位时指定的值。
locale

可选

默认值:en

付费墙本地化的标识符。该参数应为由一个或多个子标签通过减号(-)分隔的语言代码。第一个子标签表示语言,第二个表示地区。

示例:en 表示英语,pt-br 表示巴西葡萄牙语。

有关语言代码及推荐使用方式的更多信息,请参阅本地化与语言代码

fetchPolicy默认值:.reloadRevalidatingCacheData

默认情况下,SDK 会尝试从服务器加载数据,若加载失败则返回缓存数据。我们推荐此方式,因为它能确保用户始终获得最新数据。

但是,如果您认为用户的网络连接不稳定,可以考虑使用 .returnCacheDataElseLoad,在缓存存在时返回缓存数据。这样用户可能无法获得最新数据,但无论网络状况如何,都能享受更快的加载速度。缓存会定期更新,因此在会话期间使用缓存以避免网络请求是安全的。

请注意,缓存在应用重启后仍然有效,仅在应用卸载重装或手动清除时才会被清空。

自定义资源

要在付费墙中自定义图片和视频,请实现自定义资源。

主图和视频有预定义的 ID:hero_imagehero_video。在自定义资源包中,你通过这些 ID 来定位对应元素并自定义其行为。

对于其他图片和视频,你需要在 Adapty 看板中设置自定义 ID

例如,你可以:

  • 为部分用户显示不同的图片或视频。
  • 在远程主图加载时,先显示本地预览图。
  • 在视频播放前,先显示预览图。

要使用此功能,请将 Adapty iOS SDK 更新至 3.7.0 或更高版本。

以下是通过简单字典提供自定义资源的示例:

let customAssets: [String: AdaptyCustomAsset] = [
    // 使用自定义 ID 显示本地图片
    "custom_image": .image(
        .uiImage(value: UIImage(named: "image_name")!)
    ),

    // 远程主图加载时显示本地预览图
    "hero_image": .image(
        .remote(
            url: URL(string: "https://example.com/image.jpg")!,
            preview: UIImage(named: "preview_image")
        )
    ),

    // 显示带预览图的本地视频
    "hero_video": .video(
        .file(
            url: Bundle.main.url(forResource: "custom_video", withExtension: "mp4")!,
            preview: .uiImage(value: UIImage(named: "video_preview")!)
        )
    ),
]

let paywallConfig = try await AdaptyUI.getPaywallConfiguration(
    forPaywall: paywall,
    assetsResolver: customAssets
)

如果找不到某个资源,付费墙将回退到其默认外观。

设置开发者自定义计时器

要在移动应用中使用自定义计时器,需创建一个遵循 AdaptyTimerResolver 协议的对象。该对象定义了每个自定义计时器的渲染方式。如果你更喜欢,也可以直接使用 [String: Date] 字典,因为它已经符合该协议。以下是示例:

@MainActor
struct AdaptyTimerResolverImpl: AdaptyTimerResolver {
    func timerEndAtDate(for timerId: String) -> Date {
        switch timerId {
        case "CUSTOM_TIMER_6H":
            Date(timeIntervalSinceNow: 3600.0 * 6.0) // 6 hours
        case "CUSTOM_TIMER_NY":
            Calendar.current.date(from: DateComponents(year: 2025, month: 1, day: 1)) ?? Date(timeIntervalSinceNow: 3600.0)
        default:
            Date(timeIntervalSinceNow: 3600.0) // 1 hour
        }
    }
}

在此示例中,CUSTOM_TIMER_NYCUSTOM_TIMER_6H 是你在 Adapty 看板中设置的开发者自定义计时器的 Timer IDtimerResolver 确保你的应用动态地为每个计时器更新正确的值。例如:

  • CUSTOM_TIMER_NY:距离计时器结束时间(如元旦)的剩余时间。
  • CUSTOM_TIMER_6H:用户打开付费墙后开始的 6 小时倒计时的剩余时间。