Skip to content

Commit

Permalink
[Connect] Add support for Dashboard app (#4034)
Browse files Browse the repository at this point in the history
## Summary
<!-- Simple summary of what was changed. -->

Adds Dashboard app support for the StripeConnect SDK.

-  **Adds Dashboard app setting override keys**
If the API client's key is a `uk_` key, then `apiKeyOverride`,
`merchantIdOverride`, `platformIdOverride`, and `livemodeOverride` URL
parameters are configured and sent to the connect webview, enabling it
for use with direct accounts.

- **Fixes the `StripeConnect` package to compile as an SPM package**
Removes `StripeConnectBundleLocator` and localization files. This file
wasn't compiling because all the files in the package's `Resource`
folder were empty so `Resource.bundle` wasn't being generated. Since we
don't have localized strings currently, we can delete these helpers for
now.

Annotates all public types with `@available(iOS 15, *)`. The
Package.swift file has a min supported iOS version of 13 but
StripeConnect uses types only available on 15+.

- **Adds the payment-details component for use in the Dashboard**
Because payment-details will not be included in the beta release, these
interfaces are marked with `@_spi(DashboardOnly)` to only make them
available to Dashboard.

- **Enables dynamic font scaling**
Automatically scales appearance font sizes when the component's size
class changes and updates the doc strings appropriately.

## Motivation
<!-- Why are you making this change? If it's for fixing a bug, if
possible, please include a code snippet or example project that
demonstrates the issue. -->

https://jira.corp.stripe.com/browse/MXMOBILE-2502

## Testing
<!-- How was the code tested? Be as specific as possible. -->
Unit tests:
- AppearanceTests.testFontSizesChangeBasedOnTraitCollection
- ConnectJSURLParamsTests
- PaymentDetailsViewControllersTests

Manually testing dynamic font sizing and localization:


https://github.com/user-attachments/assets/6f492031-0ca3-4cac-bc09-226f7e9714b1
  • Loading branch information
mludowise-stripe authored Sep 24, 2024
1 parent 8faa32b commit 75bdab9
Show file tree
Hide file tree
Showing 74 changed files with 532 additions and 845 deletions.
52 changes: 52 additions & 0 deletions Example/StripeConnectExample/StripeConnectExample/Info.plist
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,58 @@
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>CFBundleLocalizations</key>
<array>
<string>bg-BG</string>
<string>zh-Hans</string>
<string>zh-Hant-HK</string>
<string>zh-Hant-TW</string>
<string>hr-HR</string>
<string>cs-CZ</string>
<string>da-DK</string>
<string>nl-NL</string>
<string>en-AU</string>
<string>en-IN</string>
<string>en-IE</string>
<string>en-NZ</string>
<string>en-SG</string>
<string>en-GB</string>
<string>en-US</string>
<string>et-EE</string>
<string>fil-PH</string>
<string>fi-FI</string>
<string>fr-CA</string>
<string>fr-FR</string>
<string>de-DE</string>
<string>el-GR</string>
<string>hu-HU</string>
<string>id-ID</string>
<string>it-IT</string>
<string>ja-JP</string>
<string>ko-KR</string>
<string>lv-LV</string>
<string>lt-LT</string>
<string>ms-MY</string>
<string>mt-MT</string>
<string>nb-NO</string>
<string>pl-PL</string>
<string>pt-BR</string>
<string>pt-PT</string>
<string>ro-RO</string>
<string>sk-SK</string>
<string>sl-SI</string>
<string>es-AR</string>
<string>es-BR</string>
<string>es-419</string>
<string>es-MX</string>
<string>es-ES</string>
<string>sv-SE</string>
<string>th-TH</string>
<string>tr-TR</string>
<string>vi-VN</string>
</array>
<key>CFBundleDevelopmentRegion</key>
<string>$(DEVELOPMENT_LANGUAGE)</string>
<key>UIAppFonts</key>
<array>
<string>Handjet-Regular.ttf</string>
Expand Down
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
Empty file.
681 changes: 23 additions & 658 deletions StripeConnect/StripeConnect.xcodeproj/project.pbxproj

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,12 @@

import Foundation

enum ComponentType: String {
/// The name of the embedded component tag in JS ([docs](https://docs.stripe.com/connect/supported-embedded-components))
enum ComponentType: String, Encodable {
/// Displays the balance summary, the payout schedule, and a list of payouts for the connected account
case payouts
/// The onboarding flow for the account.
case onboarding = "account-onboarding"
/// Show details of a given payment and allow users to manage disputes and perform refunds.
case paymentDetails = "payment-details"
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
//
// PaymentDetailsViewController.swift
// StripeConnect
//
// Created by Mel Ludowise on 8/30/24.
//

import UIKit

/**
Show details of a given payment and allow users to manage disputes and perform refunds.
*/
@_spi(DashboardOnly)
@available(iOS 15, *)
public class PaymentDetailsViewController: UIViewController {
let webView: ConnectComponentWebView

public weak var delegate: PaymentDetailsViewControllerDelegate?

init(componentManager: EmbeddedComponentManager) {
webView = ConnectComponentWebView(
componentManager: componentManager,
componentType: .paymentDetails
)
super.init(nibName: nil, bundle: nil)
webView.addMessageHandler(OnLoadErrorMessageHandler { [weak self] value in
guard let self else { return }
self.delegate?.paymentDetailsLoadDidFail(self, withError: value.error.connectEmbedError)
})
webView.presentPopup = { [weak self] vc in
self?.present(vc, animated: true)
}
}

required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

public override func loadView() {
view = webView
}

public func setPayment(id: String) {
webView.sendMessage(CallSetterWithSerializableValueSender(payload: .init(
setter: "setPayment",
value: id
)))
}
}

/// Delegate of an `PaymentDetailsViewController`
@available(iOS 15, *)
@_spi(DashboardOnly)
public protocol PaymentDetailsViewControllerDelegate: AnyObject {

/**
Triggered when an error occurs loading the payment details component
- Parameters:
- paymentDetails: The payment details component that errored when loading
- error: The error that occurred when loading the component
*/
func paymentDetailsLoadDidFail(_ paymentDetails: PaymentDetailsViewController,
withError error: Error)

}

@available(iOS 15, *)
public extension PaymentDetailsViewControllerDelegate {
// Default implementation to make optional
func paymentDetailsLoadDidFail(_ paymentDetails: PaymentDetailsViewController,
withError error: Error) { }
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,10 @@ import UIKit
The balance summary, the payout schedule, and a list of payouts for the connected account. It can also allow the user to perform instant or manual payouts.
*/
@_spi(PrivateBetaConnect)
@available(iOS 15, *)
public class PayoutsViewController: UIViewController {
let webView: ConnectComponentWebView

public weak var delegate: PayoutsViewControllerDelegate?

init(componentManager: EmbeddedComponentManager,
Expand Down Expand Up @@ -44,6 +45,7 @@ public class PayoutsViewController: UIViewController {

/// Delegate of an `PayoutsViewController`
@_spi(PrivateBetaConnect)
@available(iOS 15, *)
public protocol PayoutsViewControllerDelegate: AnyObject {

/**
Expand All @@ -57,9 +59,9 @@ public protocol PayoutsViewControllerDelegate: AnyObject {

}

@available(iOS 15, *)
public extension PayoutsViewControllerDelegate {
// Default implementation to make optional
func payoutsLoadDidFail(_ payouts: PayoutsViewController,
withError error: Error) { }
}

15 changes: 8 additions & 7 deletions StripeConnect/StripeConnect/Source/CustomFontSource.swift
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,9 @@
import UIKit

@_spi(PrivateBetaConnect)
@available(iOS 15, *)
extension EmbeddedComponentManager {

/// Use a `CustomFontSource` pass custom fonts embedded in your app's binary when initializing a
/// `EmbeddedComponentManager`.
/// - Seealso: https://docs.stripe.com/connect/get-started-connect-embedded-components#customize-the-look-of-connect-embedded-components
Expand All @@ -19,7 +20,7 @@ extension EmbeddedComponentManager {
let style: String?
let weight: String?
let src: FontSource

/**
Initializes a CustomFontSource from a base font and its original file URL
- Parameters:
Expand All @@ -37,15 +38,15 @@ extension EmbeddedComponentManager {
}

let fontData = try Data(contentsOf: fileUrl)
self.src = .init(fileType: fileUrl.pathExtension, encoding:fontData.base64EncodedString())
self.src = .init(fileType: fileUrl.pathExtension, encoding: fontData.base64EncodedString())
family = font.familyName
style = font.isItalic ? "italic" : nil
self.weight = font.weight.cssValue
}

enum FontLoadError: Error, CustomDebugStringConvertible {
case notFileURL

var debugDescription: String {
switch self {
case .notFileURL:
Expand All @@ -54,11 +55,11 @@ extension EmbeddedComponentManager {
}
}
}

struct FontSource: Encodable, Equatable {
let fileType: String
let encoding: String

// TODO: CAUI-2844 Move to encoding FontSource directly instead of using a string.
var stringValue: String {
"url(data:font/\(fileType);charset=utf-8;base64,\(encoding))"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import UIKit

@_spi(PrivateBetaConnect)
@available(iOS 15, *)
extension EmbeddedComponentManager {
/// Describes the appearance of embedded components.
/// - seealso: https://docs.stripe.com/connect/embedded-appearance-options
Expand All @@ -22,7 +23,7 @@ extension EmbeddedComponentManager {
case lowercase
/// Displays the text with the first character capitalized.
case capitalize

// Since the public API does not call for TextTransform to be a string
// we manually create a raw value here.
var rawValue: String {
Expand All @@ -37,7 +38,7 @@ extension EmbeddedComponentManager {
return "capitalize"
}
}

init?(rawValue: String) {
switch rawValue {
case "none":
Expand All @@ -53,13 +54,14 @@ extension EmbeddedComponentManager {
}
}
}

/// Describes the typography attributes used in embedded components
public struct Typography {
/// Describes the font attributes used for a
/// typography style in embedded components.
public struct Style {
/// The font size for this typography style.
/// The unscaled font size for this typography style.
/// The displayed fonts are automatically scaled when the component's size category is updated.
public var fontSize: CGFloat?
/// The font weight for this typography style.
public var weight: UIFont.Weight?
Expand All @@ -68,9 +70,9 @@ extension EmbeddedComponentManager {

/// Creates a `EmbeddedComponentManager.Appearance.Typography.Stylye` with default values
public init() {}

}

/// Determines the font family value used throughout embedded components.
/// Only the family is used from the specified font. The size and weight can be
/// configured from `fontSizeBase` or `fontSize` and `fontWeight`
Expand All @@ -80,9 +82,10 @@ extension EmbeddedComponentManager {
/// `CustomFontSource` when initializing the `EmbeddedComponentManager` before
/// referencing them in the appearance's `typography.font` property.
public var font: UIFont?
/// The baseline font size set on the embedded component root.
/// This scales the value of other font size variables.
public var fontSizeBase: CGFloat?
/// The unscaled baseline font size set on the embedded component root.
/// This scales the value of other font size variables and is automatically scaled
/// when the component's size category is updated.
public var fontSizeBase: CGFloat? = 16
/// Describes the font size and weight for the medium body typography.
/// The `textTransform` property is ignored.
public var bodyMd: Style = .init()
Expand All @@ -103,11 +106,11 @@ extension EmbeddedComponentManager {
public var labelMd: Style = .init()
/// Describes the font size and weight for the small label typography.
public var labelSm: Style = .init()

/// Creates a `EmbeddedComponentManager.Appearance.Typography` with default values
public init() { }
}

/// Describes the colors used in embedded components.
/// - Note: If UIColors using dynamicProviders are specified, the appearance will automatically
/// update when the component's UITraitCollection is updated (e.g. dark mode)
Expand Down Expand Up @@ -147,11 +150,11 @@ extension EmbeddedComponentManager {
/// The color used for to fill in form items like checkboxes,
/// radio buttons and switches. The alpha component is ignored.
public var formAccent: UIColor?

/// Creates a `EmbeddedComponentManager.Appearance.Colors` with default values
public init() {}
}

/// Describes the appearance of a button type used in embedded components
public struct Button {
/// The color used as a background for this button type.
Expand All @@ -163,11 +166,11 @@ extension EmbeddedComponentManager {
/// The text color used for this button type.
/// The alpha component is ignored.
public var colorText: UIColor?

/// Creates a `EmbeddedComponentManager.Appearance.Button` with default values
public init() { }
}

/// Describes the appearance of a badge type usied in embedded components.
public struct Badge {
/// The background color for this badge type.
Expand All @@ -177,11 +180,11 @@ extension EmbeddedComponentManager {
public var colorBorder: UIColor?
/// The text color for this badge type. The alpha component is ignored.
public var colorText: UIColor?

/// Creates a `EmbeddedComponentManager.Appearance.Badge` with default values
public init() {}
}

/// Describes the corner radius used in embedded components.
public struct CornerRadius {
/// The general border radius used in embedded components.
Expand All @@ -195,14 +198,14 @@ extension EmbeddedComponentManager {
public var badge: CGFloat?
/// The corner radius used for overlays.
public var overlay: CGFloat?

/// Creates a `EmbeddedComponentManager.Appearance.CornerRadius` with default values
public init() {}
}

/// The default appearance
public static let `default`: Appearance = .init()

/// Describes the appearance of typography used in embedded components.
public var typography: Typography = .init()
/// Describes the colors used in embedded components.
Expand All @@ -229,9 +232,8 @@ extension EmbeddedComponentManager {
public var badgeDanger: Badge = .init()
/// Describes the corner radius used in embedded components.
public var cornerRadius: CornerRadius = .init()

/// Creates a `EmbeddedComponentManager.Appearance` with default values
public init() {}
}
}

Loading

0 comments on commit 75bdab9

Please sign in to comment.