Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added algorithm for pagination to PDF export, allowing to export consent forms with more than 1 page (#49) #52

Open
wants to merge 32 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 31 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
ec18eab
Introduced algorithm to split exported consent document across multip…
RealLast Jun 22, 2024
4ee81b1
Removed leftover continuation from older version of the code.
RealLast Jun 22, 2024
457b273
Resolved swiftlint issues.
RealLast Jun 26, 2024
42c2443
Removed unused variables, further cleaned up code.
RealLast Jun 27, 2024
ee2f2f4
Changed PDF export to use TPPDF for PDF generation, instead of creati…
RealLast Jul 11, 2024
23e5333
Changed personName -> signature.
RealLast Jul 11, 2024
a947d7d
Reverted changes.
RealLast Jul 11, 2024
5fbc2a7
Update iPad Testing Identifier.
RealLast Jul 12, 2024
8ec56a2
Update Tests and try Beta 4
PSchmiedmayer Aug 3, 2024
5238a7b
Update Tests
PSchmiedmayer Aug 3, 2024
a417fa9
Update GitHub Action
PSchmiedmayer Aug 3, 2024
f2e767c
Merging in changes from main.
RealLast Aug 14, 2024
a11df71
Separated PDF export functionality from ConsentDocument. Added type C…
RealLast Aug 14, 2024
2d793db
Added known good PDF files for iOS, macOS and visionOS, which is used…
RealLast Aug 14, 2024
c7cc518
Added known-good PDF documents to test against in testPDFExport.
RealLast Aug 14, 2024
0d287e5
Merge remote-tracking branch 'upstream/main' into PDFPagination
RealLast Aug 24, 2024
e6080f6
Merged in changes from #52.
RealLast Aug 24, 2024
d15f5e3
Resolved errors on macOS.
RealLast Aug 24, 2024
4e919db
Expanded unit test for PDF export to include documents with two pages.
RealLast Aug 24, 2024
c747606
Fixed swiftlint issues. Added license information.
RealLast Aug 24, 2024
d42fa7d
Added missing files.
RealLast Aug 24, 2024
eb1bcea
Added missing comments.
RealLast Aug 24, 2024
b716ada
Merge branch 'main' into PDFPagination
PSchmiedmayer Aug 30, 2024
bd5d975
Try LFS Support
PSchmiedmayer Aug 30, 2024
b813492
Update build-and-test.yml
PSchmiedmayer Aug 30, 2024
aa84929
Update build-and-test.yml
PSchmiedmayer Aug 30, 2024
1e0c539
Added new pdf documents for the tests with better spacing.
RealLast Sep 4, 2024
448e0e4
Resolved swiftlint issues.
RealLast Sep 4, 2024
af181aa
Removed leftover code.
RealLast Sep 4, 2024
2697855
Made PDF export throwing if PDF generation fails. A possible exceptio…
RealLast Sep 4, 2024
9194b05
Resolved swiftlint issue.
RealLast Sep 4, 2024
7abf6cc
Introduced ExportConfiguration.FontSettings to enable more precise co…
RealLast Oct 30, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 14 additions & 0 deletions .gitattributes
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
#
# This source file is part of the Stanford Spezi open-source project
#
# SPDX-FileCopyrightText: 2023 Stanford University and the project authors (see CONTRIBUTORS.md)
#
# SPDX-License-Identifier: MIT
#

Tests/SpeziOnboardingTests/Resources/known_good_pdf_one_page_mac_os.pdf filter=lfs diff=lfs merge=lfs -text
Tests/SpeziOnboardingTests/Resources/known_good_pdf_one_page_vision_os.pdf filter=lfs diff=lfs merge=lfs -text
Tests/SpeziOnboardingTests/Resources/known_good_pdf_two_pages_ios.pdf filter=lfs diff=lfs merge=lfs -text
Tests/SpeziOnboardingTests/Resources/known_good_pdf_two_pages_mac_os.pdf filter=lfs diff=lfs merge=lfs -text
Tests/SpeziOnboardingTests/Resources/known_good_pdf_two_pages_vision_os.pdf filter=lfs diff=lfs merge=lfs -text
Tests/SpeziOnboardingTests/Resources/known_good_pdf_one_page_ios.pdf filter=lfs diff=lfs merge=lfs -text
31 changes: 10 additions & 21 deletions .github/workflows/build-and-test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ jobs:
resultBundle: SpeziOnboarding-iOS-Release.xcresult
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
scheme: SpeziOnboarding
buildConfig: ${{ matrix.buildConfig }}
resultBundle: ${{ matrix.resultBundle }}
Expand All @@ -39,6 +40,7 @@ jobs:
uses: StanfordSpezi/.github/.github/workflows/xcodebuild-or-fastlane.yml@v2
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
scheme: SpeziOnboarding
xcodeversion: latest
swiftVersion: 6
Expand All @@ -58,6 +60,7 @@ jobs:
resultBundle: SpeziOnboarding-visionOS-Release.xcresult
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
scheme: SpeziOnboarding
destination: 'platform=visionOS Simulator,name=Apple Vision Pro'
buildConfig: ${{ matrix.buildConfig }}
Expand All @@ -77,6 +80,7 @@ jobs:
resultBundle: SpeziOnboarding-macOS-Release.xcresult
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
scheme: SpeziOnboarding
destination: 'platform=macOS,arch=arm64'
buildConfig: ${{ matrix.buildConfig }}
Expand All @@ -85,17 +89,9 @@ jobs:
buildandtestuitests_ios:
name: Build and Test UI Tests iOS
uses: StanfordSpezi/.github/.github/workflows/xcodebuild-or-fastlane.yml@v2
strategy:
matrix:
include:
- buildConfig: Debug
resultBundle: TestApp-iOS.xcresult
artifactname: TestApp-iOS.xcresult
- buildConfig: Release
resultBundle: TestApp-iOS-Release.xcresult
artifactname: TestApp-iOS-Release.xcresult
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
path: 'Tests/UITests'
scheme: TestApp
buildConfig: ${{ matrix.buildConfig }}
Expand All @@ -106,6 +102,7 @@ jobs:
uses: StanfordSpezi/.github/.github/workflows/xcodebuild-or-fastlane.yml@v2
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
path: Tests/UITests
scheme: TestApp
xcodeversion: latest
Expand All @@ -117,6 +114,7 @@ jobs:
uses: StanfordSpezi/.github/.github/workflows/xcodebuild-or-fastlane.yml@v2
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
path: 'Tests/UITests'
scheme: TestApp
destination: 'platform=iOS Simulator,name=iPad Pro 11-inch (M4)'
Expand All @@ -125,23 +123,14 @@ jobs:
buildandtestuitests_visionos:
name: Build and Test UI Tests visionOS
uses: StanfordSpezi/.github/.github/workflows/xcodebuild-or-fastlane.yml@v2
strategy:
matrix:
include:
- buildConfig: Debug
resultBundle: TestApp-visionOS.xcresult
artifactname: TestApp-visionOS.xcresult
- buildConfig: Release
resultBundle: TestApp-visionOS-Release.xcresult
artifactname: TestApp-visionOS-Release.xcresult
with:
runsonlabels: '["macOS", "self-hosted"]'
checkout_lfs: true
path: 'Tests/UITests'
scheme: TestApp
destination: 'platform=visionOS Simulator,name=Apple Vision Pro'
buildConfig: ${{ matrix.buildConfig }}
resultBundle: ${{ matrix.resultBundle }}
artifactname: ${{ matrix.artifactname }}
resultBundle: TestApp-visionOS.xcresult
artifactname: TestApp-visionOS.xcresult
uploadcoveragereport:
name: Upload Coverage Report
needs: [buildandtest_ios, buildandtest_visionos, buildandtest_macos, buildandtestuitests_ios, buildandtestuitests_ipad, buildandtestuitests_visionos]
Expand Down
9 changes: 9 additions & 0 deletions .reuse/dep5
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/

Files: Tests/SpeziOnboardingTests/Resources/*.pdf
Copyright: 2024 Stanford University and the project authors (see CONTRIBUTORS.md)
License: MIT

Files: Tests/SpeziOnboardingTests/Resources/*.md
Copyright: 2024 Stanford University and the project authors (see CONTRIBUTORS.md)
License: MIT
13 changes: 9 additions & 4 deletions Package.swift
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@

//
// This source file is part of the Stanford Spezi open-source project
//
//
// SPDX-FileCopyrightText: 2022 Stanford University and the project authors (see CONTRIBUTORS.md)
//
//
// SPDX-License-Identifier: MIT
//

Expand Down Expand Up @@ -33,7 +33,8 @@ let package = Package(
dependencies: [
.package(url: "https://github.com/StanfordSpezi/Spezi", from: "1.2.1"),
.package(url: "https://github.com/StanfordSpezi/SpeziViews", from: "1.3.1"),
.package(url: "https://github.com/apple/swift-collections.git", from: "1.1.0")
.package(url: "https://github.com/apple/swift-collections.git", from: "1.1.0"),
.package(url: "https://github.com/techprimate/TPPDF", from: "2.6.0")
] + swiftLintPackage(),
targets: [
.target(
Expand All @@ -42,7 +43,8 @@ let package = Package(
.product(name: "Spezi", package: "Spezi"),
.product(name: "SpeziViews", package: "SpeziViews"),
.product(name: "SpeziPersonalInfo", package: "SpeziViews"),
.product(name: "OrderedCollections", package: "swift-collections")
.product(name: "OrderedCollections", package: "swift-collections"),
.product(name: "TPPDF", package: "TPPDF")
],
swiftSettings: [
swiftConcurrency
Expand All @@ -54,6 +56,9 @@ let package = Package(
dependencies: [
.target(name: "SpeziOnboarding")
],
resources: [
.process("Resources/")
],
swiftSettings: [
swiftConcurrency
],
Expand Down
8 changes: 8 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,14 @@ For more information, please refer to the [API documentation](https://swiftpacka
The [Spezi Template Application](https://github.com/StanfordSpezi/SpeziTemplateApplication) provides a great starting point and example using the `SpeziOnboarding` module.


## Running Tests Locally
If you would like to clone this repo and run the unit and UI tests locally, please be aware that you need to have git lfs (large file storage) configured. The unit tests load some binary data (e.g., PDF files) at runtime, which are stored in the [test resources](Tests/SpeziOnboardingTests/Resources/) as git lfs tags. To install git lfs, please refer to the [official documentation](https://git-lfs.com/). Afterward set up git lfs for your user by executing the following command:
```sh
git lfs install
```
Now, you can clone the repo and the binary files should be checked out correctly.
RealLast marked this conversation as resolved.
Show resolved Hide resolved


## Contributing

Contributions to this project are welcome. Please make sure to read the [contribution guidelines](https://github.com/StanfordSpezi/.github/blob/main/CONTRIBUTING.md) and the [contributor covenant code of conduct](https://github.com/StanfordSpezi/.github/blob/main/CODE_OF_CONDUCT.md) first.
Expand Down
107 changes: 13 additions & 94 deletions Sources/SpeziOnboarding/ConsentView/ConsentDocument+Export.swift
RealLast marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
import PDFKit
import PencilKit
import SwiftUI

import TPPDF

/// Extension of `ConsentDocument` enabling the export of the signed consent page.
extension ConsentDocument {
Expand All @@ -18,7 +18,7 @@ extension ConsentDocument {
/// force the ink used in the `UIImage` of the `PKDrawing` to always be black by adjusting the signature ink according to the color scheme.
@MainActor private var blackInkSignatureImage: UIImage {
var updatedDrawing = PKDrawing()

for stroke in signature.strokes {
let blackStroke = PKStroke(
ink: PKInk(stroke.ink.inkType, color: colorScheme == .light ? .black : .white),
Expand All @@ -43,101 +43,20 @@ extension ConsentDocument {
}
#endif


/// Creates a representation of the consent form that is ready to be exported via the SwiftUI `ImageRenderer`.
///
/// - Parameters:
/// - markdown: The markdown consent content as an `AttributedString`.
///
/// - Returns: A SwiftUI `View` representation of the consent content and signature.
///
/// - Note: This function avoids the use of asynchronous operations.
/// Asynchronous tasks are incompatible with SwiftUI's `ImageRenderer`,
/// which expects all rendering processes to be synchronous.
@MainActor
private func exportBody(markdown: AttributedString) -> some View {
VStack {
if exportConfiguration.includingTimestamp {
HStack {
Spacer()

Text("EXPORTED_TAG", bundle: .module)
+ Text(verbatim: ": \(DateFormatter.localizedString(from: Date(), dateStyle: .medium, timeStyle: .short))")
}
.font(.caption)
.padding()
}

OnboardingTitleView(title: exportConfiguration.consentTitle)

Text(markdown)
.padding()

Spacer()

ZStack(alignment: .bottomLeading) {
SignatureViewBackground(name: name, backgroundColor: .clear)

#if !os(macOS)
Image(uiImage: blackInkSignatureImage)
#else
Text(signature)
.padding(.bottom, 32)
.padding(.leading, 46)
.font(.custom("Snell Roundhand", size: 24))
#endif
}
#if !os(macOS)
.frame(width: signatureSize.width, height: signatureSize.height)
#else
.padding(.horizontal, 100)
#endif
}
}

/// Exports the signed consent form as a `PDFDocument` via the SwiftUI `ImageRenderer`.
///
/// Exports the signed consent form as a `PDFKit.PDFDocument`.
/// The PDF generated by TPPDF and then converted to a TPDFKit.PDFDocument.
/// Renders the `PDFDocument` according to the specified ``ConsentDocument/ExportConfiguration``.
///
/// - Returns: The exported consent form in PDF format as a PDFKit `PDFDocument`
@MainActor
func export() async -> PDFDocument? {
let markdown = await asyncMarkdown()

let markdownString = (try? AttributedString(
markdown: markdown,
options: .init(interpretedSyntax: .inlineOnlyPreservingWhitespace)
)) ?? AttributedString(String(localized: "MARKDOWN_LOADING_ERROR", bundle: .module))

let renderer = ImageRenderer(content: exportBody(markdown: markdownString))
let paperSize = CGSize(
width: exportConfiguration.paperSize.dimensions.width,
height: exportConfiguration.paperSize.dimensions.height
)
renderer.proposedSize = .init(paperSize)

return await withCheckedContinuation { continuation in
renderer.render { _, context in
var box = CGRect(origin: .zero, size: paperSize)

/// Create in-memory `CGContext` that stores the PDF
guard let mutableData = CFDataCreateMutable(kCFAllocatorDefault, 0),
let consumer = CGDataConsumer(data: mutableData),
let pdf = CGContext(consumer: consumer, mediaBox: &box, nil) else {
continuation.resume(returning: nil)
return
}

pdf.beginPDFPage(nil)
pdf.translateBy(x: 0, y: 0)

context(pdf)

pdf.endPDFPage()
pdf.closePDF()

continuation.resume(returning: PDFDocument(data: mutableData as Data))
}
}
func export() async throws -> PDFKit.PDFDocument {
documentExport.signature = signature
documentExport.name = name
#if !os(macOS)
documentExport.signatureImage = blackInkSignatureImage
return try await documentExport.export()
#else
return try await documentExport.export()
#endif
}
}
30 changes: 22 additions & 8 deletions Sources/SpeziOnboarding/ConsentView/ConsentDocument.swift
Original file line number Diff line number Diff line change
Expand Up @@ -41,12 +41,12 @@ public struct ConsentDocument: View {
/// The maximum width such that the drawing canvas fits onto the PDF.
static let maxWidthDrawing: CGFloat = 550

let asyncMarkdown: () async -> Data
private let givenNameTitle: LocalizedStringResource
private let givenNamePlaceholder: LocalizedStringResource
private let familyNameTitle: LocalizedStringResource
private let familyNamePlaceholder: LocalizedStringResource
let exportConfiguration: ExportConfiguration

let documentExport: ConsentDocumentExport

@Environment(\.colorScheme) var colorScheme
@State var name = PersonNameComponents()
Expand Down Expand Up @@ -85,6 +85,7 @@ public struct ConsentDocument: View {
signature.removeAll()
#endif
}
documentExport.name = name
}

Divider()
Expand Down Expand Up @@ -129,12 +130,13 @@ public struct ConsentDocument: View {
} else {
viewState = .namesEntered
}
documentExport.signature = signature
}
}

public var body: some View {
VStack {
MarkdownView(asyncMarkdown: asyncMarkdown, state: $viewState.base)
MarkdownView(asyncMarkdown: documentExport.asyncMarkdown, state: $viewState.base)
Spacer()
Group {
nameView
Expand All @@ -152,11 +154,13 @@ public struct ConsentDocument: View {
.onChange(of: viewState) {
if case .export = viewState {
Task {
guard let exportedConsent = await export() else {
guard let exportedConsent = try? await export() else {
RealLast marked this conversation as resolved.
Show resolved Hide resolved
viewState = .base(.error(Error.memoryAllocationError))
return
}
viewState = .exported(document: exportedConsent)

documentExport.cachedPDF = exportedConsent
viewState = .exported(document: exportedConsent, export: documentExport)
}
} else if case .base(let baseViewState) = viewState,
case .idle = baseViewState {
Expand Down Expand Up @@ -194,22 +198,32 @@ public struct ConsentDocument: View {
/// - familyNameTitle: The localization to use for the family (last) name field.
/// - familyNamePlaceholder: The localization to use for the family name field placeholder.
/// - exportConfiguration: Defines the properties of the exported consent form via ``ConsentDocument/ExportConfiguration``.
/// - documentIdentifier: A unique identifier or "name" for the consent form, helpful for distinguishing consent forms when storing in the `Standard`.
public init(
markdown: @escaping () async -> Data,
viewState: Binding<ConsentViewState>,
givenNameTitle: LocalizedStringResource = LocalizationDefaults.givenNameTitle,
givenNamePlaceholder: LocalizedStringResource = LocalizationDefaults.givenNamePlaceholder,
familyNameTitle: LocalizedStringResource = LocalizationDefaults.familyNameTitle,
familyNamePlaceholder: LocalizedStringResource = LocalizationDefaults.familyNamePlaceholder,
exportConfiguration: ExportConfiguration = .init()
exportConfiguration: ExportConfiguration = .init(),
documentIdentifier: String = ConsentDocumentExport.Defaults.documentIdentifier
) {
self.asyncMarkdown = markdown
self._viewState = viewState
self.givenNameTitle = givenNameTitle
self.givenNamePlaceholder = givenNamePlaceholder
self.familyNameTitle = familyNameTitle
self.familyNamePlaceholder = familyNamePlaceholder
self.exportConfiguration = exportConfiguration

self.documentExport = ConsentDocumentExport(
markdown: markdown,
exportConfiguration: exportConfiguration,
documentIdentifier: documentIdentifier
)
// Set initial values for the name and signature.
// These will be updated once the name and signature change.
self.documentExport.name = name
self.documentExport.signature = signature
}
}

Expand Down
Loading
Loading