@devo-bmad-custom/agent-orchestration 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/installer.js +33 -0
- package/package.json +1 -1
- package/src/.agents/skills/audit-website/README.md +20 -20
- package/src/.agents/skills/audit-website/SKILL.md +470 -470
- package/src/.agents/skills/audit-website/agents/openai.yaml +6 -6
- package/src/.agents/skills/audit-website/assets/icon-small.svg +41 -41
- package/src/.agents/skills/audit-website/references/OUTPUT-FORMAT.md +250 -250
- package/src/.agents/skills/clean-code-standards/SKILL.md +104 -104
- package/src/.agents/skills/excalidraw-dark-standard/SKILL.md +281 -281
- package/src/.agents/skills/frontend-responsive-design-standards/SKILL.md +434 -434
- package/src/.agents/skills/java-fundamentals/SKILL.md +116 -116
- package/src/.agents/skills/java-performance/SKILL.md +119 -119
- package/src/.agents/skills/next-best-practices/SKILL.md +153 -153
- package/src/.agents/skills/next-best-practices/async-patterns.md +87 -87
- package/src/.agents/skills/next-best-practices/bundling.md +180 -180
- package/src/.agents/skills/next-best-practices/data-patterns.md +297 -297
- package/src/.agents/skills/next-best-practices/debug-tricks.md +105 -105
- package/src/.agents/skills/next-best-practices/directives.md +73 -73
- package/src/.agents/skills/next-best-practices/error-handling.md +227 -227
- package/src/.agents/skills/next-best-practices/file-conventions.md +140 -140
- package/src/.agents/skills/next-best-practices/font.md +245 -245
- package/src/.agents/skills/next-best-practices/functions.md +108 -108
- package/src/.agents/skills/next-best-practices/hydration-error.md +91 -91
- package/src/.agents/skills/next-best-practices/image.md +173 -173
- package/src/.agents/skills/next-best-practices/metadata.md +301 -301
- package/src/.agents/skills/next-best-practices/parallel-routes.md +287 -287
- package/src/.agents/skills/next-best-practices/route-handlers.md +146 -146
- package/src/.agents/skills/next-best-practices/rsc-boundaries.md +159 -159
- package/src/.agents/skills/next-best-practices/runtime-selection.md +39 -39
- package/src/.agents/skills/next-best-practices/scripts.md +141 -141
- package/src/.agents/skills/next-best-practices/self-hosting.md +371 -371
- package/src/.agents/skills/next-best-practices/suspense-boundaries.md +67 -67
- package/src/.agents/skills/nextjs-app-router-patterns/SKILL.md +537 -537
- package/src/.agents/skills/postgresql-optimization/SKILL.md +404 -404
- package/src/.agents/skills/python-backend/SKILL.md +153 -153
- package/src/.agents/skills/python-fundamentals/SKILL.md +234 -234
- package/src/.agents/skills/python-performance/SKILL.md +404 -404
- package/src/.agents/skills/react-expert/SKILL.md +335 -335
- package/src/.agents/skills/redis-best-practices/SKILL.md +438 -438
- package/src/.agents/skills/security-best-practices/SKILL.md +288 -288
- package/src/.agents/skills/security-review/LICENSE +22 -22
- package/src/.agents/skills/security-review/SKILL.md +312 -312
- package/src/.agents/skills/security-review/infrastructure/docker.md +432 -432
- package/src/.agents/skills/security-review/languages/javascript.md +388 -388
- package/src/.agents/skills/security-review/languages/python.md +363 -363
- package/src/.agents/skills/security-review/references/api-security.md +519 -519
- package/src/.agents/skills/security-review/references/authentication.md +353 -353
- package/src/.agents/skills/security-review/references/authorization.md +372 -372
- package/src/.agents/skills/security-review/references/business-logic.md +443 -443
- package/src/.agents/skills/security-review/references/cryptography.md +329 -329
- package/src/.agents/skills/security-review/references/csrf.md +398 -398
- package/src/.agents/skills/security-review/references/data-protection.md +378 -378
- package/src/.agents/skills/security-review/references/deserialization.md +410 -410
- package/src/.agents/skills/security-review/references/error-handling.md +436 -436
- package/src/.agents/skills/security-review/references/file-security.md +457 -457
- package/src/.agents/skills/security-review/references/injection.md +259 -259
- package/src/.agents/skills/security-review/references/logging.md +433 -433
- package/src/.agents/skills/security-review/references/misconfiguration.md +435 -435
- package/src/.agents/skills/security-review/references/modern-threats.md +475 -475
- package/src/.agents/skills/security-review/references/ssrf.md +415 -415
- package/src/.agents/skills/security-review/references/supply-chain.md +405 -405
- package/src/.agents/skills/security-review/references/xss.md +336 -336
- package/src/.agents/skills/subagent-driven-development/SKILL.md +275 -275
- package/src/.agents/skills/subagent-driven-development/code-quality-reviewer-prompt.md +26 -26
- package/src/.agents/skills/subagent-driven-development/implementer-prompt.md +113 -113
- package/src/.agents/skills/subagent-driven-development/spec-reviewer-prompt.md +61 -61
- package/src/.agents/skills/systematic-debugging/CREATION-LOG.md +119 -119
- package/src/.agents/skills/systematic-debugging/SKILL.md +296 -296
- package/src/.agents/skills/systematic-debugging/condition-based-waiting-example.ts +158 -158
- package/src/.agents/skills/systematic-debugging/condition-based-waiting.md +115 -115
- package/src/.agents/skills/systematic-debugging/defense-in-depth.md +122 -122
- package/src/.agents/skills/systematic-debugging/root-cause-tracing.md +169 -169
- package/src/.agents/skills/systematic-debugging/test-academic.md +14 -14
- package/src/.agents/skills/systematic-debugging/test-pressure-1.md +58 -58
- package/src/.agents/skills/systematic-debugging/test-pressure-2.md +68 -68
- package/src/.agents/skills/systematic-debugging/test-pressure-3.md +69 -69
- package/src/.agents/skills/typescript-best-practices/SKILL.md +373 -373
- package/src/.agents/skills/ui-ux-pro-custom/SKILL.md +348 -348
- package/src/.agents/skills/ui-ux-pro-custom/data/charts.csv +26 -26
- package/src/.agents/skills/ui-ux-pro-custom/data/colors.csv +97 -97
- package/src/.agents/skills/ui-ux-pro-custom/data/icons.csv +101 -101
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/SKILL.md +106 -106
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/accessibility.md +475 -475
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/animation.md +466 -466
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/composition-locals.md +231 -231
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/deprecated-patterns.md +323 -323
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/lists-scrolling.md +400 -400
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/modifiers.md +331 -331
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/navigation.md +416 -416
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/performance.md +446 -446
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/side-effects.md +516 -516
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/foundation-source.md +13327 -13327
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/material3-source.md +19097 -19097
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/navigation-source.md +2947 -2947
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/runtime-source.md +11316 -11316
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/ui-source.md +7896 -7896
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/state-management.md +377 -377
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/styles-experimental.md +470 -470
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/theming-material3.md +349 -349
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/view-composition.md +595 -595
- package/src/.agents/skills/ui-ux-pro-custom/data/landing.csv +31 -31
- package/src/.agents/skills/ui-ux-pro-custom/data/mobile-ui-layout.md +654 -654
- package/src/.agents/skills/ui-ux-pro-custom/data/products.csv +96 -96
- package/src/.agents/skills/ui-ux-pro-custom/data/react-performance.csv +45 -45
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/astro.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/flutter.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/html-tailwind.csv +56 -56
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/jetpack-compose.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nextjs.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nuxt-ui.csv +51 -51
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nuxtjs.csv +59 -59
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/react-native.csv +56 -56
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/react.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/shadcn.csv +61 -61
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/svelte.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/swiftui.csv +51 -51
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/vue.csv +50 -50
- package/src/.agents/skills/ui-ux-pro-custom/data/styles.csv +68 -68
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/alarmkit/SKILL.md +438 -438
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/alarmkit/references/alarmkit-patterns.md +584 -584
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-clips/SKILL.md +436 -436
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-intents/SKILL.md +489 -489
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-intents/references/appintents-advanced.md +1076 -1076
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/SKILL.md +340 -340
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/references/privacy-manifest.md +90 -90
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/references/review-checklists.md +106 -106
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/coreml-conversion.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/coreml-optimization.md +344 -344
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/foundation-models.md +508 -508
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/mlx-swift.md +285 -285
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/authentication/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/authentication/references/keychain-biometric.md +211 -211
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/background-processing/SKILL.md +499 -499
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/background-processing/references/background-task-patterns.md +390 -390
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/callkit-voip/SKILL.md +461 -461
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/callkit-voip/references/callkit-patterns.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/cloudkit-sync/SKILL.md +492 -492
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/cloudkit-sync/references/cloudkit-patterns.md +461 -461
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/codable-patterns/SKILL.md +467 -467
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/contacts-framework/SKILL.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/contacts-framework/references/contacts-patterns.md +409 -409
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-bluetooth/SKILL.md +491 -491
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-bluetooth/references/ble-patterns.md +435 -435
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-motion/SKILL.md +388 -388
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-motion/references/motion-patterns.md +405 -405
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-nfc/SKILL.md +495 -495
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-nfc/references/nfc-patterns.md +420 -420
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/coreml/SKILL.md +459 -459
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/coreml/references/coreml-swift-integration.md +765 -765
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/SKILL.md +422 -422
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/references/instruments-guide.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/references/lldb-patterns.md +298 -298
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/device-integrity/SKILL.md +477 -477
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/energykit/SKILL.md +460 -460
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/energykit/references/energykit-patterns.md +541 -541
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/eventkit-calendar/SKILL.md +483 -483
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/eventkit-calendar/references/eventkit-patterns.md +326 -326
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/healthkit/SKILL.md +498 -498
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/healthkit/references/healthkit-patterns.md +602 -602
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/homekit-matter/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/homekit-matter/references/matter-commissioning.md +455 -455
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-accessibility/SKILL.md +301 -301
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-accessibility/references/a11y-patterns.md +140 -140
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/SKILL.md +418 -418
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/references/formatstyle-locale.md +627 -627
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/references/string-catalogs.md +462 -462
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/SKILL.md +441 -441
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/background-websocket.md +862 -862
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/lightweight-clients.md +93 -93
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/network-framework.md +563 -563
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/urlsession-patterns.md +1116 -1116
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/app-review-guidelines.md +174 -174
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/cryptokit-advanced.md +296 -296
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/file-storage-patterns.md +354 -354
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/privacy-manifest.md +117 -117
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/live-activities/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/live-activities/references/live-activity-patterns.md +868 -868
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/SKILL.md +485 -485
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/references/corelocation-patterns.md +730 -730
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/references/mapkit-patterns.md +748 -748
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/metrickit-diagnostics/SKILL.md +479 -479
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/musickit-audio/SKILL.md +395 -395
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/musickit-audio/references/musickit-patterns.md +363 -363
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/natural-language/SKILL.md +412 -412
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/natural-language/references/translation-patterns.md +311 -311
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/passkit-wallet/SKILL.md +398 -398
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/passkit-wallet/references/wallet-passes.md +254 -254
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/SKILL.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/references/paperkit-integration.md +376 -376
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/references/pencilkit-patterns.md +302 -302
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/permissionkit/SKILL.md +446 -446
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/permissionkit/references/permissionkit-patterns.md +435 -435
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/av-playback.md +701 -701
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/camera-capture.md +774 -774
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/image-loading-caching.md +869 -869
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/photospicker-patterns.md +597 -597
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/references/notification-patterns.md +677 -677
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/references/rich-notifications.md +745 -745
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/realitykit-ar/SKILL.md +479 -479
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/realitykit-ar/references/realitykit-patterns.md +480 -480
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/shareplay-activities/SKILL.md +483 -483
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/shareplay-activities/references/shareplay-patterns.md +544 -544
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/speech-recognition/SKILL.md +485 -485
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/SKILL.md +478 -478
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/references/app-review-guidelines.md +58 -58
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/references/storekit-advanced.md +755 -755
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-charts/SKILL.md +487 -487
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-charts/references/charts-patterns.md +895 -895
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/SKILL.md +408 -408
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/approachable-concurrency.md +80 -80
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/swift-6-2-concurrency.md +233 -233
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/swiftui-concurrency.md +187 -187
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/synchronization-primitives.md +341 -341
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-language/SKILL.md +498 -498
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-language/references/swift-patterns-extended.md +505 -505
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-testing/SKILL.md +467 -467
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-testing/references/testing-patterns.md +504 -504
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/SKILL.md +334 -334
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/core-data-coexistence.md +504 -504
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/swiftdata-advanced.md +975 -975
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/swiftdata-queries.md +675 -675
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/SKILL.md +481 -481
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/references/animation-advanced.md +804 -804
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/references/core-animation-bridge.md +553 -553
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-gestures/SKILL.md +450 -450
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-gestures/references/gesture-patterns.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/SKILL.md +336 -336
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/form.md +97 -97
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/grids.md +69 -69
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/list.md +99 -99
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/scrollview.md +147 -147
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-liquid-glass/SKILL.md +325 -325
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-liquid-glass/references/liquid-glass.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/SKILL.md +262 -262
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/deeplinks.md +207 -207
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/navigationstack.md +177 -177
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/sheets.md +169 -169
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/tabview.md +178 -178
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/SKILL.md +381 -381
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/architecture-patterns.md +486 -486
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/deprecated-migration.md +1097 -1097
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/design-polish.md +780 -780
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/platform-and-sharing.md +696 -696
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/SKILL.md +491 -491
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/demystify-swiftui-performance-wwdc23.md +46 -46
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/optimizing-swiftui-performance-instruments.md +29 -29
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/understanding-hangs-in-your-app.md +33 -33
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/understanding-improving-swiftui-performance.md +52 -52
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/SKILL.md +428 -428
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/references/hosting-migration.md +534 -534
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/references/representable-recipes.md +1133 -1133
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/tipkit/SKILL.md +494 -494
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/tipkit/references/tipkit-patterns.md +782 -782
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/SKILL.md +475 -475
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/references/vision-requests.md +736 -736
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/references/visionkit-scanner.md +738 -738
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/weatherkit/SKILL.md +410 -410
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/weatherkit/references/weatherkit-patterns.md +567 -567
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/widgetkit/SKILL.md +497 -497
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/widgetkit/references/widgetkit-advanced.md +871 -871
- package/src/.agents/skills/ui-ux-pro-custom/data/typography.csv +57 -57
- package/src/.agents/skills/ui-ux-pro-custom/data/ui-reasoning.csv +101 -101
- package/src/.agents/skills/ui-ux-pro-custom/data/ux-guidelines.csv +99 -99
- package/src/.agents/skills/ui-ux-pro-custom/data/web-interface.csv +31 -31
- package/src/.agents/skills/ui-ux-pro-custom/scripts/core.py +253 -253
- package/src/.agents/skills/ui-ux-pro-custom/scripts/design_system.py +1067 -1067
- package/src/.agents/skills/ui-ux-pro-custom/scripts/search.py +114 -114
- package/src/.agents/skills/ux-audit/SKILL.md +150 -150
- package/src/.agents/skills/websocket-engineer/SKILL.md +168 -168
- package/src/.agents/skills/websocket-engineer/references/alternatives.md +391 -391
- package/src/.agents/skills/websocket-engineer/references/patterns.md +400 -400
- package/src/.agents/skills/websocket-engineer/references/protocol.md +195 -195
- package/src/.agents/skills/websocket-engineer/references/scaling.md +333 -333
- package/src/.agents/skills/websocket-engineer/references/security.md +474 -474
- package/src/.agents/skills/writing-skills/SKILL.md +655 -655
- package/src/.agents/skills/writing-skills/anthropic-best-practices.md +1150 -1150
- package/src/.agents/skills/writing-skills/examples/CLAUDE_MD_TESTING.md +189 -189
- package/src/.agents/skills/writing-skills/graphviz-conventions.dot +171 -171
- package/src/.agents/skills/writing-skills/persuasion-principles.md +187 -187
- package/src/.agents/skills/writing-skills/render-graphs.js +168 -168
- package/src/.agents/skills/writing-skills/testing-skills-with-subagents.md +384 -384
- package/src/.claude/commands/master-orchestrator.md +15 -0
- package/src/_memory/config.yaml +11 -11
- package/src/_memory/master-orchestrator-sidecar/instructions.md +97 -47
- package/src/_memory/skills/nimbalyst-tracking/SKILL.md +103 -103
- package/src/_memory/skills/writing-skills/SKILL.md +655 -655
- package/src/bmb/agents/agent-builder.md +59 -59
- package/src/bmb/agents/module-builder.md +60 -60
- package/src/bmb/agents/workflow-builder.md +61 -61
- package/src/bmb/config.yaml +12 -12
- package/src/bmb/module-help.csv +13 -13
- package/src/bmb/workflows/agent/data/agent-architecture.md +258 -258
- package/src/bmb/workflows/agent/data/agent-compilation.md +185 -185
- package/src/bmb/workflows/agent/data/agent-menu-patterns.md +189 -189
- package/src/bmb/workflows/agent/data/agent-metadata.md +133 -133
- package/src/bmb/workflows/agent/data/agent-validation.md +111 -111
- package/src/bmb/workflows/agent/data/brainstorm-context.md +96 -96
- package/src/bmb/workflows/agent/data/communication-presets.csv +61 -61
- package/src/bmb/workflows/agent/data/critical-actions.md +75 -75
- package/src/bmb/workflows/agent/data/persona-properties.md +252 -252
- package/src/bmb/workflows/agent/data/principles-crafting.md +142 -142
- package/src/bmb/workflows/agent/data/reference/module-examples/architect.md +68 -68
- package/src/bmb/workflows/agent/data/reference/with-sidecar/journal-keeper/journal-keeper-sidecar/entries/yy-mm-dd-entry-template.md +16 -16
- package/src/bmb/workflows/agent/data/understanding-agent-types.md +126 -126
- package/src/bmb/workflows/agent/steps-c/step-01-brainstorm.md +129 -129
- package/src/bmb/workflows/agent/steps-c/step-02-discovery.md +170 -170
- package/src/bmb/workflows/agent/steps-c/step-03-sidecar-metadata.md +309 -309
- package/src/bmb/workflows/agent/steps-c/step-04-persona.md +213 -213
- package/src/bmb/workflows/agent/steps-c/step-05-commands-menu.md +179 -179
- package/src/bmb/workflows/agent/steps-c/step-06-activation.md +278 -278
- package/src/bmb/workflows/agent/steps-c/step-07-build-agent.md +316 -316
- package/src/bmb/workflows/agent/steps-c/step-08-celebrate.md +247 -247
- package/src/bmb/workflows/agent/steps-e/e-01-load-existing.md +221 -221
- package/src/bmb/workflows/agent/steps-e/e-02-discover-edits.md +195 -195
- package/src/bmb/workflows/agent/steps-e/e-04-sidecar-metadata.md +126 -126
- package/src/bmb/workflows/agent/steps-e/e-05-persona.md +135 -135
- package/src/bmb/workflows/agent/steps-e/e-06-commands-menu.md +123 -123
- package/src/bmb/workflows/agent/steps-e/e-07-activation.md +124 -124
- package/src/bmb/workflows/agent/steps-e/e-08-edit-agent.md +197 -197
- package/src/bmb/workflows/agent/steps-e/e-09-celebrate.md +155 -155
- package/src/bmb/workflows/agent/steps-v/v-01-load-review.md +137 -137
- package/src/bmb/workflows/agent/steps-v/v-02a-validate-metadata.md +116 -116
- package/src/bmb/workflows/agent/steps-v/v-02b-validate-persona.md +124 -124
- package/src/bmb/workflows/agent/steps-v/v-02c-validate-menu.md +127 -127
- package/src/bmb/workflows/agent/steps-v/v-02d-validate-structure.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-02e-validate-sidecar.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-03-summary.md +104 -104
- package/src/bmb/workflows/agent/templates/agent-plan.template.md +5 -5
- package/src/bmb/workflows/agent/templates/agent-template.md +89 -89
- package/src/bmb/workflows/agent/workflow-create-agent.md +72 -72
- package/src/bmb/workflows/agent/workflow-edit-agent.md +75 -75
- package/src/bmb/workflows/agent/workflow-validate-agent.md +73 -73
- package/src/bmb/workflows/module/data/agent-architecture.md +179 -179
- package/src/bmb/workflows/module/data/agent-spec-template.md +79 -79
- package/src/bmb/workflows/module/data/module-standards.md +263 -263
- package/src/bmb/workflows/module/data/module-yaml-conventions.md +392 -392
- package/src/bmb/workflows/module/module-help-generate.md +254 -254
- package/src/bmb/workflows/module/steps-b/step-01-welcome.md +148 -148
- package/src/bmb/workflows/module/steps-b/step-02-spark.md +141 -141
- package/src/bmb/workflows/module/steps-b/step-03-module-type.md +149 -149
- package/src/bmb/workflows/module/steps-b/step-04-vision.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-05-identity.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-06-users.md +86 -86
- package/src/bmb/workflows/module/steps-b/step-07-value.md +76 -76
- package/src/bmb/workflows/module/steps-b/step-08-agents.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-09-workflows.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-10-tools.md +91 -91
- package/src/bmb/workflows/module/steps-b/step-11-scenarios.md +84 -84
- package/src/bmb/workflows/module/steps-b/step-12-creative.md +95 -95
- package/src/bmb/workflows/module/steps-b/step-13-review.md +105 -105
- package/src/bmb/workflows/module/steps-b/step-14-finalize.md +117 -117
- package/src/bmb/workflows/module/steps-c/step-01-load-brief.md +179 -179
- package/src/bmb/workflows/module/steps-c/step-01b-continue.md +82 -82
- package/src/bmb/workflows/module/steps-c/step-02-structure.md +105 -105
- package/src/bmb/workflows/module/steps-c/step-03-config.md +119 -119
- package/src/bmb/workflows/module/steps-c/step-04-agents.md +168 -168
- package/src/bmb/workflows/module/steps-c/step-05-workflows.md +184 -184
- package/src/bmb/workflows/module/steps-c/step-06-docs.md +401 -401
- package/src/bmb/workflows/module/steps-c/step-07-complete.md +152 -152
- package/src/bmb/workflows/module/steps-e/step-01-load-target.md +81 -81
- package/src/bmb/workflows/module/steps-e/step-02-select-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-03-apply-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-04-review.md +80 -80
- package/src/bmb/workflows/module/steps-e/step-05-confirm.md +75 -75
- package/src/bmb/workflows/module/steps-v/step-01-load-target.md +96 -96
- package/src/bmb/workflows/module/steps-v/step-02-file-structure.md +93 -93
- package/src/bmb/workflows/module/steps-v/step-03-module-yaml.md +99 -99
- package/src/bmb/workflows/module/steps-v/step-04-agent-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-05-workflow-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-06-documentation.md +143 -143
- package/src/bmb/workflows/module/steps-v/step-07-installation.md +102 -102
- package/src/bmb/workflows/module/steps-v/step-08-report.md +197 -197
- package/src/bmb/workflows/module/templates/brief-template.md +154 -154
- package/src/bmb/workflows/module/templates/workflow-spec-template.md +96 -96
- package/src/bmb/workflows/module/workflow-create-module-brief.md +71 -71
- package/src/bmb/workflows/module/workflow-create-module.md +86 -86
- package/src/bmb/workflows/module/workflow-edit-module.md +66 -66
- package/src/bmb/workflows/module/workflow-validate-module.md +66 -66
- package/src/bmb/workflows/workflow/data/architecture.md +150 -150
- package/src/bmb/workflows/workflow/data/common-workflow-tools.csv +19 -19
- package/src/bmb/workflows/workflow/data/csv-data-file-standards.md +53 -53
- package/src/bmb/workflows/workflow/data/frontmatter-standards.md +184 -184
- package/src/bmb/workflows/workflow/data/input-discovery-standards.md +191 -191
- package/src/bmb/workflows/workflow/data/intent-vs-prescriptive-spectrum.md +44 -44
- package/src/bmb/workflows/workflow/data/menu-handling-standards.md +133 -133
- package/src/bmb/workflows/workflow/data/output-format-standards.md +135 -135
- package/src/bmb/workflows/workflow/data/step-file-rules.md +235 -235
- package/src/bmb/workflows/workflow/data/step-type-patterns.md +257 -257
- package/src/bmb/workflows/workflow/data/subprocess-optimization-patterns.md +188 -188
- package/src/bmb/workflows/workflow/data/trimodal-workflow-structure.md +164 -164
- package/src/bmb/workflows/workflow/data/workflow-chaining-standards.md +222 -222
- package/src/bmb/workflows/workflow/data/workflow-examples.md +232 -232
- package/src/bmb/workflows/workflow/data/workflow-type-criteria.md +134 -134
- package/src/bmb/workflows/workflow/steps-c/step-00-conversion.md +263 -263
- package/src/bmb/workflows/workflow/steps-c/step-01-discovery.md +194 -194
- package/src/bmb/workflows/workflow/steps-c/step-01b-continuation.md +3 -3
- package/src/bmb/workflows/workflow/steps-c/step-02-classification.md +270 -270
- package/src/bmb/workflows/workflow/steps-c/step-03-requirements.md +283 -283
- package/src/bmb/workflows/workflow/steps-c/step-04-tools.md +282 -282
- package/src/bmb/workflows/workflow/steps-c/step-05-plan-review.md +243 -243
- package/src/bmb/workflows/workflow/steps-c/step-06-design.md +330 -330
- package/src/bmb/workflows/workflow/steps-c/step-07-foundation.md +239 -239
- package/src/bmb/workflows/workflow/steps-c/step-08-build-step-01.md +379 -379
- package/src/bmb/workflows/workflow/steps-c/step-09-build-next-step.md +350 -350
- package/src/bmb/workflows/workflow/steps-c/step-10-confirmation.md +322 -322
- package/src/bmb/workflows/workflow/steps-c/step-11-completion.md +191 -191
- package/src/bmb/workflows/workflow/steps-e/step-e-01-assess-workflow.md +237 -237
- package/src/bmb/workflows/workflow/steps-e/step-e-02-discover-edits.md +251 -251
- package/src/bmb/workflows/workflow/steps-e/step-e-03-fix-validation.md +254 -254
- package/src/bmb/workflows/workflow/steps-e/step-e-04-direct-edit.md +277 -277
- package/src/bmb/workflows/workflow/steps-e/step-e-05-apply-edit.md +154 -154
- package/src/bmb/workflows/workflow/steps-e/step-e-06-validate-after.md +190 -190
- package/src/bmb/workflows/workflow/steps-e/step-e-07-complete.md +206 -206
- package/src/bmb/workflows/workflow/steps-v/step-01-validate-max-mode.md +109 -109
- package/src/bmb/workflows/workflow/steps-v/step-01-validate.md +221 -221
- package/src/bmb/workflows/workflow/steps-v/step-01b-structure.md +152 -152
- package/src/bmb/workflows/workflow/steps-v/step-02-frontmatter-validation.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-02b-path-violations.md +265 -265
- package/src/bmb/workflows/workflow/steps-v/step-03-menu-validation.md +164 -164
- package/src/bmb/workflows/workflow/steps-v/step-04-step-type-validation.md +211 -211
- package/src/bmb/workflows/workflow/steps-v/step-05-output-format-validation.md +200 -200
- package/src/bmb/workflows/workflow/steps-v/step-06-validation-design-check.md +195 -195
- package/src/bmb/workflows/workflow/steps-v/step-07-instruction-style-check.md +209 -209
- package/src/bmb/workflows/workflow/steps-v/step-08-collaborative-experience-check.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-08b-subprocess-optimization.md +179 -179
- package/src/bmb/workflows/workflow/steps-v/step-09-cohesive-review.md +186 -186
- package/src/bmb/workflows/workflow/steps-v/step-10-report-complete.md +154 -154
- package/src/bmb/workflows/workflow/steps-v/step-11-plan-validation.md +237 -237
- package/src/bmb/workflows/workflow/templates/minimal-output-template.md +11 -11
- package/src/bmb/workflows/workflow/templates/step-01-init-continuable-template.md +241 -241
- package/src/bmb/workflows/workflow/templates/step-1b-template.md +224 -224
- package/src/bmb/workflows/workflow/templates/step-template.md +294 -294
- package/src/bmb/workflows/workflow/templates/workflow-template.md +102 -102
- package/src/bmb/workflows/workflow/workflow-create-workflow.md +79 -79
- package/src/bmb/workflows/workflow/workflow-edit-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-rework-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-validate-max-parallel-workflow.md +66 -66
- package/src/bmb/workflows/workflow/workflow-validate-workflow.md +65 -65
- package/src/bmm/agents/analyst.md +104 -104
- package/src/bmm/agents/dev.md +100 -100
- package/src/bmm/agents/qa.md +100 -90
- package/src/bmm/agents/tech-writer/tech-writer.md +94 -94
- package/src/bmm/module-help.csv +31 -31
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-01-init.md +115 -115
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-01b-continue.md +107 -107
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-02-vision.md +141 -141
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-03-users.md +144 -144
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-04-metrics.md +147 -147
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-05-scope.md +161 -161
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-06-complete.md +99 -99
- package/src/bmm/workflows/1-analysis/create-product-brief/workflow.md +57 -57
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-01-init.md +87 -87
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-02-domain-analysis.md +156 -156
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-03-competitive-landscape.md +165 -165
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-04-regulatory-focus.md +140 -140
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-05-technical-trends.md +152 -152
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-06-research-synthesis.md +345 -345
- package/src/bmm/workflows/1-analysis/research/market-steps/step-01-init.md +92 -92
- package/src/bmm/workflows/1-analysis/research/market-steps/step-02-customer-behavior.md +164 -164
- package/src/bmm/workflows/1-analysis/research/market-steps/step-03-customer-pain-points.md +174 -174
- package/src/bmm/workflows/1-analysis/research/market-steps/step-04-customer-decisions.md +184 -184
- package/src/bmm/workflows/1-analysis/research/market-steps/step-05-competitive-analysis.md +105 -105
- package/src/bmm/workflows/1-analysis/research/market-steps/step-06-research-completion.md +360 -360
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-01-init.md +87 -87
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-02-technical-overview.md +165 -165
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-03-integration-patterns.md +174 -174
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-04-architectural-patterns.md +141 -141
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-05-implementation-research.md +159 -159
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-06-research-synthesis.md +387 -387
- package/src/bmm/workflows/1-analysis/research/workflow-domain-research.md +54 -54
- package/src/bmm/workflows/1-analysis/research/workflow-market-research.md +54 -54
- package/src/bmm/workflows/1-analysis/research/workflow-technical-research.md +54 -54
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-01b-continue.md +100 -100
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02-discovery.md +160 -160
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02b-vision.md +88 -88
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02c-executive-summary.md +99 -99
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-03-success.md +169 -169
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-04-journeys.md +156 -156
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-05-domain.md +136 -136
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-06-innovation.md +176 -176
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-07-project-type.md +184 -184
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-08-scoping.md +174 -174
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-09-functional.md +175 -175
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-10-nonfunctional.md +189 -189
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-11-polish.md +162 -162
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-12-complete.md +79 -79
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-01-discovery.md +183 -183
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-01b-legacy-conversion.md +149 -149
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-02-review.md +187 -187
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-03-edit.md +192 -192
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-04-complete.md +108 -108
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-01-discovery.md +166 -166
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-02-format-detection.md +131 -131
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-02b-parity-check.md +150 -150
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-03-density-validation.md +118 -118
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-04-brief-coverage-validation.md +155 -155
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-05-measurability-validation.md +170 -170
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-06-traceability-validation.md +158 -158
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-07-implementation-leakage-validation.md +147 -147
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-08-domain-compliance-validation.md +182 -182
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-09-project-type-validation.md +202 -202
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-10-smart-validation.md +148 -148
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-11-holistic-quality-validation.md +201 -201
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-12-completeness-validation.md +179 -179
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-13-report-complete.md +164 -164
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-create-prd.md +65 -65
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-edit-prd.md +65 -65
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-validate-prd.md +63 -63
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-01b-continue.md +63 -63
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-02-discovery.md +106 -106
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-03-core-experience.md +111 -111
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-04-emotional-response.md +115 -115
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-05-inspiration.md +127 -127
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-06-design-system.md +167 -167
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-07-defining-experience.md +143 -143
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-08-visual-foundation.md +118 -118
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-09-design-directions.md +154 -154
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-10-user-journeys.md +136 -136
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-11-component-strategy.md +165 -165
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-12-ux-patterns.md +135 -135
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-13-responsive-accessibility.md +192 -192
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-14-complete.md +101 -101
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/workflow.md +45 -45
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-01-document-discovery.md +185 -185
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-02-prd-analysis.md +129 -129
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-03-epic-coverage-validation.md +130 -130
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-04-ux-alignment.md +93 -93
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-05-epic-quality-review.md +196 -196
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-06-final-assessment.md +129 -129
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/workflow.md +54 -54
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-01b-continue.md +82 -82
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-02-context.md +106 -106
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-03-starter.md +138 -138
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-04-decisions.md +129 -129
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-05-patterns.md +166 -166
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-06-structure.md +186 -186
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-07-validation.md +163 -163
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-08-complete.md +38 -38
- package/src/bmm/workflows/3-solutioning/create-architecture/workflow.md +49 -49
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-02-design-epics.md +124 -124
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-03-create-stories.md +122 -122
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-04-final-validation.md +84 -84
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/workflow.md +58 -58
- package/src/bmm/workflows/4-implementation/code-review/workflow.yaml +43 -43
- package/src/bmm/workflows/4-implementation/correct-course/workflow.yaml +53 -53
- package/src/bmm/workflows/4-implementation/create-story/checklist.md +159 -159
- package/src/bmm/workflows/4-implementation/create-story/template.md +79 -79
- package/src/bmm/workflows/4-implementation/create-story/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/dev-story/workflow.yaml +20 -20
- package/src/bmm/workflows/4-implementation/retrospective/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/sprint-planning/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/sprint-status/workflow.yaml +25 -25
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-01-mode-detection.md +158 -158
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-02-context-gathering.md +122 -122
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-03-execute.md +93 -93
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-04-self-check.md +93 -93
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-05-adversarial-review.md +87 -87
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-06-resolve-findings.md +146 -146
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/workflow.md +50 -50
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-02-investigate.md +152 -152
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-03-generate.md +123 -123
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-04-review.md +201 -201
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/workflow.md +79 -79
- package/src/bmm/workflows/document-project/workflow.yaml +22 -22
- package/src/bmm/workflows/generate-project-context/steps/step-01-discover.md +184 -184
- package/src/bmm/workflows/generate-project-context/steps/step-02-generate.md +322 -322
- package/src/bmm/workflows/generate-project-context/steps/step-03-complete.md +235 -235
- package/src/bmm/workflows/generate-project-context/workflow.md +49 -49
- package/src/bmm/workflows/qa/automate/workflow.yaml +233 -233
- package/src/bmm/workflows/qa-generate-e2e-tests/workflow.yaml +42 -42
- package/src/core/config.yaml +9 -9
- package/src/core/module-help.csv +10 -10
- package/src/core/scripts/generate-loop-report.py +72 -72
- package/src/core/tasks/editorial-review-prose.xml +101 -101
- package/src/core/tasks/editorial-review-structure.xml +207 -207
- package/src/core/tasks/help.md +86 -86
- package/src/core/tasks/index-docs.xml +64 -64
- package/src/core/tasks/review-adversarial-general.xml +66 -66
- package/src/core/tasks/review-adversarial-loop.xml +46 -46
- package/src/core/tasks/review-edge-case-hunter.xml +63 -63
- package/src/core/tasks/review-party-loop.xml +46 -46
- package/src/core/tasks/shard-doc.xml +107 -107
- package/src/core/tasks/workflow.xml +235 -235
- package/src/core/templates/review-loop-report.html +88 -88
- package/src/core/templates/review-loop-report.md +5 -5
- package/src/core/workflows/advanced-elicitation/workflow.xml +117 -117
- package/src/core/workflows/brainstorming/steps/step-01-session-setup.md +212 -212
- package/src/core/workflows/brainstorming/steps/step-01b-continue.md +122 -122
- package/src/core/workflows/brainstorming/steps/step-02a-user-selected.md +225 -225
- package/src/core/workflows/brainstorming/steps/step-02b-ai-recommended.md +237 -237
- package/src/core/workflows/brainstorming/steps/step-02c-random-selection.md +209 -209
- package/src/core/workflows/brainstorming/steps/step-02d-progressive-flow.md +264 -264
- package/src/core/workflows/brainstorming/steps/step-02e-deep-dive.md +68 -68
- package/src/core/workflows/brainstorming/steps/step-03-technique-execution.md +403 -403
- package/src/core/workflows/brainstorming/steps/step-04-idea-organization.md +303 -303
- package/src/core/workflows/brainstorming/workflow.md +60 -60
- package/src/core/workflows/extract-trackers/workflow.md +45 -45
- package/src/core/workflows/party-mode/steps/step-01-agent-loading.md +142 -142
- package/src/core/workflows/party-mode/workflow.md +194 -194
- package/src/docs/dev/tmux/actions_popup.py +291 -291
- package/src/docs/dev/tmux/tmux-setup.md +62 -1
|
@@ -1,597 +1,597 @@
|
|
|
1
|
-
# PhotosPicker Patterns
|
|
2
|
-
|
|
3
|
-
Complete recipes for photo and video selection, loading, saving, and processing. All patterns target iOS 16+ with SwiftUI and async/await.
|
|
4
|
-
|
|
5
|
-
---
|
|
6
|
-
|
|
7
|
-
## Contents
|
|
8
|
-
|
|
9
|
-
- [1. Single Photo Selection with Loading State](#1-single-photo-selection-with-loading-state)
|
|
10
|
-
- [2. Multi-Selection with Progress Tracking](#2-multi-selection-with-progress-tracking)
|
|
11
|
-
- [3. Loading Videos from PhotosPicker](#3-loading-videos-from-photospicker)
|
|
12
|
-
- [4. Loading Live Photos](#4-loading-live-photos)
|
|
13
|
-
- [5. PHPickerViewController Wrapping (UIKit Interop)](#5-phpickerviewcontroller-wrapping-uikit-interop)
|
|
14
|
-
- [6. Saving Images to Photo Library](#6-saving-images-to-photo-library)
|
|
15
|
-
- [7. Thumbnail Generation with ImageIO Downsampling](#7-thumbnail-generation-with-imageio-downsampling)
|
|
16
|
-
- [8. HEIC/HEIF Handling](#8-heicheif-handling)
|
|
17
|
-
- [9. Custom PhotosPicker Appearance](#9-custom-photospicker-appearance)
|
|
18
|
-
- [10. Image Cropping Pattern](#10-image-cropping-pattern)
|
|
19
|
-
|
|
20
|
-
## 1. Single Photo Selection with Loading State
|
|
21
|
-
|
|
22
|
-
Handle the full lifecycle: idle, loading, loaded, and error.
|
|
23
|
-
|
|
24
|
-
```swift
|
|
25
|
-
import SwiftUI
|
|
26
|
-
import PhotosUI
|
|
27
|
-
|
|
28
|
-
struct ProfilePhotoPicker: View {
|
|
29
|
-
@State private var selectedItem: PhotosPickerItem?
|
|
30
|
-
@State private var loadState: LoadState = .idle
|
|
31
|
-
|
|
32
|
-
enum LoadState {
|
|
33
|
-
case idle, loading, loaded(Image), error(String)
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
var body: some View {
|
|
37
|
-
VStack {
|
|
38
|
-
switch loadState {
|
|
39
|
-
case .idle:
|
|
40
|
-
Image(systemName: "person.crop.circle.fill")
|
|
41
|
-
.resizable()
|
|
42
|
-
.frame(width: 120, height: 120)
|
|
43
|
-
.foregroundStyle(.secondary)
|
|
44
|
-
case .loading:
|
|
45
|
-
ProgressView()
|
|
46
|
-
.frame(width: 120, height: 120)
|
|
47
|
-
case .loaded(let image):
|
|
48
|
-
image
|
|
49
|
-
.resizable()
|
|
50
|
-
.scaledToFill()
|
|
51
|
-
.frame(width: 120, height: 120)
|
|
52
|
-
.clipShape(Circle())
|
|
53
|
-
case .error(let message):
|
|
54
|
-
ContentUnavailableView(message, systemImage: "exclamationmark.triangle")
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
PhotosPicker("Choose Photo", selection: $selectedItem, matching: .images)
|
|
58
|
-
.buttonStyle(.borderedProminent)
|
|
59
|
-
}
|
|
60
|
-
.onChange(of: selectedItem) { _, newItem in
|
|
61
|
-
guard let newItem else {
|
|
62
|
-
loadState = .idle
|
|
63
|
-
return
|
|
64
|
-
}
|
|
65
|
-
loadState = .loading
|
|
66
|
-
Task {
|
|
67
|
-
do {
|
|
68
|
-
if let data = try await newItem.loadTransferable(type: Data.self),
|
|
69
|
-
let uiImage = UIImage(data: data) {
|
|
70
|
-
loadState = .loaded(Image(uiImage: uiImage))
|
|
71
|
-
} else {
|
|
72
|
-
loadState = .error("Unable to load image")
|
|
73
|
-
}
|
|
74
|
-
} catch {
|
|
75
|
-
loadState = .error(error.localizedDescription)
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
```
|
|
82
|
-
|
|
83
|
-
---
|
|
84
|
-
|
|
85
|
-
## 2. Multi-Selection with Progress Tracking
|
|
86
|
-
|
|
87
|
-
Load multiple images sequentially and report progress back to the UI.
|
|
88
|
-
|
|
89
|
-
```swift
|
|
90
|
-
import SwiftUI
|
|
91
|
-
import PhotosUI
|
|
92
|
-
|
|
93
|
-
struct MultiPhotoLoader: View {
|
|
94
|
-
@State private var selectedItems: [PhotosPickerItem] = []
|
|
95
|
-
@State private var loadedImages: [UIImage] = []
|
|
96
|
-
@State private var loadProgress: Double = 0
|
|
97
|
-
@State private var isLoading = false
|
|
98
|
-
|
|
99
|
-
var body: some View {
|
|
100
|
-
VStack {
|
|
101
|
-
if isLoading {
|
|
102
|
-
ProgressView(value: loadProgress)
|
|
103
|
-
.padding()
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
ScrollView(.horizontal) {
|
|
107
|
-
LazyHStack(spacing: 8) {
|
|
108
|
-
ForEach(loadedImages.indices, id: \.self) { index in
|
|
109
|
-
Image(uiImage: loadedImages[index])
|
|
110
|
-
.resizable()
|
|
111
|
-
.scaledToFill()
|
|
112
|
-
.frame(width: 100, height: 100)
|
|
113
|
-
.clipShape(RoundedRectangle(cornerRadius: 8))
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
.padding(.horizontal)
|
|
117
|
-
}
|
|
118
|
-
.frame(height: loadedImages.isEmpty ? 0 : 116)
|
|
119
|
-
|
|
120
|
-
PhotosPicker(
|
|
121
|
-
"Select Photos",
|
|
122
|
-
selection: $selectedItems,
|
|
123
|
-
maxSelectionCount: 10,
|
|
124
|
-
matching: .images
|
|
125
|
-
)
|
|
126
|
-
}
|
|
127
|
-
.onChange(of: selectedItems) { _, newItems in
|
|
128
|
-
Task { await loadImages(from: newItems) }
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
private func loadImages(from items: [PhotosPickerItem]) async {
|
|
133
|
-
isLoading = true
|
|
134
|
-
loadProgress = 0
|
|
135
|
-
var images: [UIImage] = []
|
|
136
|
-
|
|
137
|
-
for (index, item) in items.enumerated() {
|
|
138
|
-
if let data = try? await item.loadTransferable(type: Data.self),
|
|
139
|
-
let uiImage = UIImage(data: data) {
|
|
140
|
-
images.append(uiImage)
|
|
141
|
-
}
|
|
142
|
-
loadProgress = Double(index + 1) / Double(items.count)
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
loadedImages = images
|
|
146
|
-
isLoading = false
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
```
|
|
150
|
-
|
|
151
|
-
Load sequentially rather than concurrently to control memory usage. Each full-resolution image can be large; loading ten simultaneously risks memory termination.
|
|
152
|
-
|
|
153
|
-
---
|
|
154
|
-
|
|
155
|
-
## 3. Loading Videos from PhotosPicker
|
|
156
|
-
|
|
157
|
-
Use a `Transferable` wrapper that writes video data to a temporary file, since videos are too large to hold in memory as `Data`.
|
|
158
|
-
|
|
159
|
-
```swift
|
|
160
|
-
import SwiftUI
|
|
161
|
-
import PhotosUI
|
|
162
|
-
import AVKit
|
|
163
|
-
|
|
164
|
-
struct PickedMovie: Transferable {
|
|
165
|
-
let url: URL
|
|
166
|
-
|
|
167
|
-
static var transferRepresentation: some TransferRepresentation {
|
|
168
|
-
FileRepresentation(contentType: .movie) { movie in
|
|
169
|
-
SentTransferredFile(movie.url)
|
|
170
|
-
} importing: { received in
|
|
171
|
-
// Copy to a temporary location the app controls
|
|
172
|
-
let tempURL = FileManager.default.temporaryDirectory
|
|
173
|
-
.appendingPathComponent(UUID().uuidString)
|
|
174
|
-
.appendingPathExtension("mov")
|
|
175
|
-
try FileManager.default.copyItem(at: received.file, to: tempURL)
|
|
176
|
-
return PickedMovie(url: tempURL)
|
|
177
|
-
}
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
struct VideoPickerView: View {
|
|
182
|
-
@State private var selectedItem: PhotosPickerItem?
|
|
183
|
-
@State private var player: AVPlayer?
|
|
184
|
-
|
|
185
|
-
var body: some View {
|
|
186
|
-
VStack {
|
|
187
|
-
if let player {
|
|
188
|
-
VideoPlayer(player: player)
|
|
189
|
-
.frame(height: 300)
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
PhotosPicker("Select Video", selection: $selectedItem, matching: .videos)
|
|
193
|
-
}
|
|
194
|
-
.onChange(of: selectedItem) { _, newItem in
|
|
195
|
-
Task {
|
|
196
|
-
if let movie = try? await newItem?.loadTransferable(type: PickedMovie.self) {
|
|
197
|
-
player = AVPlayer(url: movie.url)
|
|
198
|
-
}
|
|
199
|
-
}
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
```
|
|
204
|
-
|
|
205
|
-
Always copy the received file to a temporary directory you control. The system may delete the original transfer file at any time.
|
|
206
|
-
|
|
207
|
-
---
|
|
208
|
-
|
|
209
|
-
## 4. Loading Live Photos
|
|
210
|
-
|
|
211
|
-
```swift
|
|
212
|
-
import SwiftUI
|
|
213
|
-
import PhotosUI
|
|
214
|
-
|
|
215
|
-
@available(iOS 17.0, *)
|
|
216
|
-
struct LivePhotoPickerView: View {
|
|
217
|
-
@State private var selectedItem: PhotosPickerItem?
|
|
218
|
-
@State private var livePhoto: PHLivePhoto?
|
|
219
|
-
|
|
220
|
-
var body: some View {
|
|
221
|
-
VStack {
|
|
222
|
-
if let livePhoto {
|
|
223
|
-
LivePhotoView(livePhoto: livePhoto)
|
|
224
|
-
.frame(height: 300)
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
PhotosPicker(
|
|
228
|
-
"Select Live Photo",
|
|
229
|
-
selection: $selectedItem,
|
|
230
|
-
matching: .livePhotos
|
|
231
|
-
)
|
|
232
|
-
}
|
|
233
|
-
.onChange(of: selectedItem) { _, newItem in
|
|
234
|
-
Task {
|
|
235
|
-
livePhoto = try? await newItem?.loadTransferable(type: PHLivePhoto.self)
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
```
|
|
241
|
-
|
|
242
|
-
`PHLivePhoto` conforms to `Transferable` on iOS 17+. On iOS 16, load the Live Photo components manually using `PHAsset`.
|
|
243
|
-
|
|
244
|
-
---
|
|
245
|
-
|
|
246
|
-
## 5. PHPickerViewController Wrapping (UIKit Interop)
|
|
247
|
-
|
|
248
|
-
Use `PHPickerViewController` when you need UIKit-level control or are integrating into an existing UIKit codebase. Prefer `PhotosPicker` for pure SwiftUI apps.
|
|
249
|
-
|
|
250
|
-
```swift
|
|
251
|
-
import SwiftUI
|
|
252
|
-
import PhotosUI
|
|
253
|
-
|
|
254
|
-
struct PHPickerWrapper: UIViewControllerRepresentable {
|
|
255
|
-
@Binding var selectedImages: [UIImage]
|
|
256
|
-
var selectionLimit: Int = 0
|
|
257
|
-
@Environment(\.dismiss) private var dismiss
|
|
258
|
-
|
|
259
|
-
func makeCoordinator() -> Coordinator { Coordinator(self) }
|
|
260
|
-
|
|
261
|
-
func makeUIViewController(context: Context) -> PHPickerViewController {
|
|
262
|
-
var config = PHPickerConfiguration(photoLibrary: .shared())
|
|
263
|
-
config.filter = .images
|
|
264
|
-
config.selectionLimit = selectionLimit
|
|
265
|
-
config.preferredAssetRepresentationMode = .current
|
|
266
|
-
|
|
267
|
-
let picker = PHPickerViewController(configuration: config)
|
|
268
|
-
picker.delegate = context.coordinator
|
|
269
|
-
return picker
|
|
270
|
-
}
|
|
271
|
-
|
|
272
|
-
func updateUIViewController(_ uiViewController: PHPickerViewController, context: Context) {
|
|
273
|
-
// Configuration is immutable after creation
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
final class Coordinator: NSObject, PHPickerViewControllerDelegate {
|
|
277
|
-
let parent: PHPickerWrapper
|
|
278
|
-
|
|
279
|
-
init(_ parent: PHPickerWrapper) { self.parent = parent }
|
|
280
|
-
|
|
281
|
-
func picker(
|
|
282
|
-
_ picker: PHPickerViewController,
|
|
283
|
-
didFinishPicking results: [PHPickerResult]
|
|
284
|
-
) {
|
|
285
|
-
parent.dismiss()
|
|
286
|
-
guard !results.isEmpty else { return }
|
|
287
|
-
|
|
288
|
-
Task { @MainActor in
|
|
289
|
-
var images: [UIImage] = []
|
|
290
|
-
for result in results {
|
|
291
|
-
if let image = await loadImage(from: result.itemProvider) {
|
|
292
|
-
images.append(image)
|
|
293
|
-
}
|
|
294
|
-
}
|
|
295
|
-
parent.selectedImages = images
|
|
296
|
-
}
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
private func loadImage(from provider: NSItemProvider) async -> UIImage? {
|
|
300
|
-
await withCheckedContinuation { continuation in
|
|
301
|
-
if provider.canLoadObject(ofClass: UIImage.self) {
|
|
302
|
-
provider.loadObject(ofClass: UIImage.self) { image, _ in
|
|
303
|
-
continuation.resume(returning: image as? UIImage)
|
|
304
|
-
}
|
|
305
|
-
} else {
|
|
306
|
-
continuation.resume(returning: nil)
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
}
|
|
310
|
-
}
|
|
311
|
-
}
|
|
312
|
-
```
|
|
313
|
-
|
|
314
|
-
Always call `parent.dismiss()` in `picker(_:didFinishPicking:)` -- this delegate method fires for both selection and cancellation (with empty results).
|
|
315
|
-
|
|
316
|
-
---
|
|
317
|
-
|
|
318
|
-
## 6. Saving Images to Photo Library
|
|
319
|
-
|
|
320
|
-
Saving requires `NSPhotoLibraryAddUsageDescription` in Info.plist. Use `PHPhotoLibrary` for saving with metadata, or `UIImageWriteToSavedPhotosAlbum` for simple saves.
|
|
321
|
-
|
|
322
|
-
### Using PHPhotoLibrary (Preferred)
|
|
323
|
-
|
|
324
|
-
```swift
|
|
325
|
-
import Photos
|
|
326
|
-
|
|
327
|
-
func saveImageToLibrary(_ image: UIImage) async throws {
|
|
328
|
-
try await PHPhotoLibrary.shared().performChanges {
|
|
329
|
-
let request = PHAssetCreationRequest.forAsset()
|
|
330
|
-
request.addResource(with: .photo, data: image.jpegData(compressionQuality: 0.9)!, options: nil)
|
|
331
|
-
}
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
func saveImageDataToLibrary(_ data: Data) async throws {
|
|
335
|
-
try await PHPhotoLibrary.shared().performChanges {
|
|
336
|
-
let request = PHAssetCreationRequest.forAsset()
|
|
337
|
-
let options = PHAssetResourceCreationOptions()
|
|
338
|
-
request.addResource(with: .photo, data: data, options: options)
|
|
339
|
-
}
|
|
340
|
-
}
|
|
341
|
-
```
|
|
342
|
-
|
|
343
|
-
### Saving to a Specific Album
|
|
344
|
-
|
|
345
|
-
```swift
|
|
346
|
-
import Photos
|
|
347
|
-
|
|
348
|
-
func saveToAlbum(image: UIImage, albumName: String) async throws {
|
|
349
|
-
let album = try await findOrCreateAlbum(named: albumName)
|
|
350
|
-
|
|
351
|
-
try await PHPhotoLibrary.shared().performChanges {
|
|
352
|
-
let assetRequest = PHAssetCreationRequest.forAsset()
|
|
353
|
-
assetRequest.addResource(
|
|
354
|
-
with: .photo,
|
|
355
|
-
data: image.jpegData(compressionQuality: 0.9)!,
|
|
356
|
-
options: nil
|
|
357
|
-
)
|
|
358
|
-
|
|
359
|
-
guard let placeholder = assetRequest.placeholderForCreatedAsset,
|
|
360
|
-
let albumChangeRequest = PHAssetCollectionChangeRequest(for: album) else { return }
|
|
361
|
-
albumChangeRequest.addAssets([placeholder] as NSArray)
|
|
362
|
-
}
|
|
363
|
-
}
|
|
364
|
-
|
|
365
|
-
private func findOrCreateAlbum(named name: String) async throws -> PHAssetCollection {
|
|
366
|
-
let fetchOptions = PHFetchOptions()
|
|
367
|
-
fetchOptions.predicate = NSPredicate(format: "title = %@", name)
|
|
368
|
-
let result = PHAssetCollection.fetchAssetCollections(with: .album, subtype: .any, options: fetchOptions)
|
|
369
|
-
|
|
370
|
-
if let existing = result.firstObject {
|
|
371
|
-
return existing
|
|
372
|
-
}
|
|
373
|
-
|
|
374
|
-
// Create the album
|
|
375
|
-
var placeholder: PHObjectPlaceholder?
|
|
376
|
-
try await PHPhotoLibrary.shared().performChanges {
|
|
377
|
-
let request = PHAssetCollectionChangeRequest.creationRequestForAssetCollection(withTitle: name)
|
|
378
|
-
placeholder = request.placeholderForCreatedAssetCollection
|
|
379
|
-
}
|
|
380
|
-
|
|
381
|
-
guard let placeholder,
|
|
382
|
-
let collection = PHAssetCollection.fetchAssetCollections(
|
|
383
|
-
withLocalIdentifiers: [placeholder.localIdentifier],
|
|
384
|
-
options: nil
|
|
385
|
-
).firstObject else {
|
|
386
|
-
throw PhotoLibraryError.albumCreationFailed
|
|
387
|
-
}
|
|
388
|
-
|
|
389
|
-
return collection
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
enum PhotoLibraryError: Error {
|
|
393
|
-
case albumCreationFailed
|
|
394
|
-
}
|
|
395
|
-
```
|
|
396
|
-
|
|
397
|
-
---
|
|
398
|
-
|
|
399
|
-
## 7. Thumbnail Generation with ImageIO Downsampling
|
|
400
|
-
|
|
401
|
-
Generate thumbnails efficiently without decoding the full image into memory. This is critical for grids and lists displaying many photos.
|
|
402
|
-
|
|
403
|
-
```swift
|
|
404
|
-
import ImageIO
|
|
405
|
-
import UIKit
|
|
406
|
-
|
|
407
|
-
/// Downsample image data to a target display size.
|
|
408
|
-
/// Use this instead of UIImage(data:) followed by resizing, which decodes the full image.
|
|
409
|
-
func downsample(
|
|
410
|
-
data: Data,
|
|
411
|
-
to pointSize: CGSize,
|
|
412
|
-
scale: CGFloat = UITraitCollection.current.displayScale
|
|
413
|
-
) -> UIImage? {
|
|
414
|
-
let maxDimension = max(pointSize.width, pointSize.height) * scale
|
|
415
|
-
|
|
416
|
-
let options: [CFString: Any] = [
|
|
417
|
-
kCGImageSourceCreateThumbnailFromImageAlways: true,
|
|
418
|
-
kCGImageSourceShouldCacheImmediately: true,
|
|
419
|
-
kCGImageSourceCreateThumbnailWithTransform: true,
|
|
420
|
-
kCGImageSourceThumbnailMaxPixelSize: maxDimension
|
|
421
|
-
]
|
|
422
|
-
|
|
423
|
-
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
|
|
424
|
-
let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, options as CFDictionary) else {
|
|
425
|
-
return nil
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
return UIImage(cgImage: cgImage)
|
|
429
|
-
}
|
|
430
|
-
|
|
431
|
-
/// Downsample from a file URL (avoids loading full data into memory).
|
|
432
|
-
func downsample(
|
|
433
|
-
url: URL,
|
|
434
|
-
to pointSize: CGSize,
|
|
435
|
-
scale: CGFloat = UITraitCollection.current.displayScale
|
|
436
|
-
) -> UIImage? {
|
|
437
|
-
let maxDimension = max(pointSize.width, pointSize.height) * scale
|
|
438
|
-
|
|
439
|
-
let sourceOptions: [CFString: Any] = [kCGImageSourceShouldCache: false]
|
|
440
|
-
guard let source = CGImageSourceCreateWithURL(url as CFURL, sourceOptions as CFDictionary) else {
|
|
441
|
-
return nil
|
|
442
|
-
}
|
|
443
|
-
|
|
444
|
-
let downsampleOptions: [CFString: Any] = [
|
|
445
|
-
kCGImageSourceCreateThumbnailFromImageAlways: true,
|
|
446
|
-
kCGImageSourceShouldCacheImmediately: true,
|
|
447
|
-
kCGImageSourceCreateThumbnailWithTransform: true,
|
|
448
|
-
kCGImageSourceThumbnailMaxPixelSize: maxDimension
|
|
449
|
-
]
|
|
450
|
-
|
|
451
|
-
guard let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, downsampleOptions as CFDictionary) else {
|
|
452
|
-
return nil
|
|
453
|
-
}
|
|
454
|
-
|
|
455
|
-
return UIImage(cgImage: cgImage)
|
|
456
|
-
}
|
|
457
|
-
```
|
|
458
|
-
|
|
459
|
-
### Usage in a Photo Grid
|
|
460
|
-
|
|
461
|
-
```swift
|
|
462
|
-
struct PhotoGridItem: View {
|
|
463
|
-
let imageData: Data
|
|
464
|
-
let thumbnailSize = CGSize(width: 100, height: 100)
|
|
465
|
-
|
|
466
|
-
@State private var thumbnail: UIImage?
|
|
467
|
-
|
|
468
|
-
var body: some View {
|
|
469
|
-
Group {
|
|
470
|
-
if let thumbnail {
|
|
471
|
-
Image(uiImage: thumbnail)
|
|
472
|
-
.resizable()
|
|
473
|
-
.scaledToFill()
|
|
474
|
-
} else {
|
|
475
|
-
Rectangle()
|
|
476
|
-
.fill(.quaternary)
|
|
477
|
-
.overlay(ProgressView())
|
|
478
|
-
}
|
|
479
|
-
}
|
|
480
|
-
.frame(width: 100, height: 100)
|
|
481
|
-
.clipShape(RoundedRectangle(cornerRadius: 6))
|
|
482
|
-
.task {
|
|
483
|
-
thumbnail = downsample(data: imageData, to: thumbnailSize)
|
|
484
|
-
}
|
|
485
|
-
}
|
|
486
|
-
}
|
|
487
|
-
```
|
|
488
|
-
|
|
489
|
-
---
|
|
490
|
-
|
|
491
|
-
## 8. HEIC/HEIF Handling
|
|
492
|
-
|
|
493
|
-
Modern iPhones capture photos in HEIC format by default. Handle both HEIC and JPEG transparently.
|
|
494
|
-
|
|
495
|
-
### Checking Image Format
|
|
496
|
-
|
|
497
|
-
```swift
|
|
498
|
-
import UniformTypeIdentifiers
|
|
499
|
-
|
|
500
|
-
func imageContentType(data: Data) -> UTType? {
|
|
501
|
-
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
|
|
502
|
-
let uti = CGImageSourceGetType(source) as? String else {
|
|
503
|
-
return nil
|
|
504
|
-
}
|
|
505
|
-
return UTType(uti)
|
|
506
|
-
}
|
|
507
|
-
```
|
|
508
|
-
|
|
509
|
-
### Converting HEIC to JPEG
|
|
510
|
-
|
|
511
|
-
```swift
|
|
512
|
-
import UIKit
|
|
513
|
-
|
|
514
|
-
func convertToJPEG(heicData: Data, compressionQuality: CGFloat = 0.9) -> Data? {
|
|
515
|
-
guard let image = UIImage(data: heicData) else { return nil }
|
|
516
|
-
return image.jpegData(compressionQuality: compressionQuality)
|
|
517
|
-
}
|
|
518
|
-
```
|
|
519
|
-
|
|
520
|
-
### Preserving HEIC When Possible
|
|
521
|
-
|
|
522
|
-
When saving or uploading, prefer keeping the original HEIC format to preserve quality and reduce file size. Convert to JPEG only when the destination requires it (e.g., a server that rejects HEIC).
|
|
523
|
-
|
|
524
|
-
```swift
|
|
525
|
-
import PhotosUI
|
|
526
|
-
|
|
527
|
-
// Request the current representation to avoid transcoding
|
|
528
|
-
var config = PHPickerConfiguration(photoLibrary: .shared())
|
|
529
|
-
config.preferredAssetRepresentationMode = .current // Keeps HEIC as-is
|
|
530
|
-
// vs .compatible which transcodes to JPEG
|
|
531
|
-
```
|
|
532
|
-
|
|
533
|
-
---
|
|
534
|
-
|
|
535
|
-
## 9. Custom PhotosPicker Appearance
|
|
536
|
-
|
|
537
|
-
Customize the picker trigger with any SwiftUI view using the label closure:
|
|
538
|
-
|
|
539
|
-
```swift
|
|
540
|
-
PhotosPicker(selection: $selectedItems, maxSelectionCount: 3, matching: .images) {
|
|
541
|
-
Label("Add Photos", systemImage: "photo.on.rectangle.angled")
|
|
542
|
-
.font(.headline)
|
|
543
|
-
.padding()
|
|
544
|
-
.background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 12))
|
|
545
|
-
}
|
|
546
|
-
```
|
|
547
|
-
|
|
548
|
-
### Inline PhotosPicker (iOS 17+)
|
|
549
|
-
|
|
550
|
-
Display the picker inline rather than as a sheet:
|
|
551
|
-
|
|
552
|
-
```swift
|
|
553
|
-
@available(iOS 17.0, *)
|
|
554
|
-
PhotosPicker(
|
|
555
|
-
selection: $selectedItems,
|
|
556
|
-
maxSelectionCount: 5,
|
|
557
|
-
matching: .images
|
|
558
|
-
)
|
|
559
|
-
.photosPickerStyle(.inline)
|
|
560
|
-
.frame(height: 300)
|
|
561
|
-
```
|
|
562
|
-
|
|
563
|
-
### Photos Picker Access Behavior (iOS 17+)
|
|
564
|
-
|
|
565
|
-
Control how the picker interacts with limited library access:
|
|
566
|
-
|
|
567
|
-
```swift
|
|
568
|
-
@available(iOS 17.0, *)
|
|
569
|
-
PhotosPicker(selection: $selectedItems, matching: .images)
|
|
570
|
-
.photosPickerAccessBehavior(.limited) // Only show user-approved photos
|
|
571
|
-
// .automatic (default) -- system decides
|
|
572
|
-
// .limited -- only previously approved photos
|
|
573
|
-
```
|
|
574
|
-
|
|
575
|
-
---
|
|
576
|
-
|
|
577
|
-
## 10. Image Cropping Pattern
|
|
578
|
-
|
|
579
|
-
A basic square crop using Core Graphics after selecting a photo:
|
|
580
|
-
|
|
581
|
-
```swift
|
|
582
|
-
import UIKit
|
|
583
|
-
|
|
584
|
-
func cropToSquare(_ image: UIImage) -> UIImage? {
|
|
585
|
-
guard let cgImage = image.cgImage else { return nil }
|
|
586
|
-
|
|
587
|
-
let side = min(cgImage.width, cgImage.height)
|
|
588
|
-
let x = (cgImage.width - side) / 2
|
|
589
|
-
let y = (cgImage.height - side) / 2
|
|
590
|
-
let cropRect = CGRect(x: x, y: y, width: side, height: side)
|
|
591
|
-
|
|
592
|
-
guard let cropped = cgImage.cropping(to: cropRect) else { return nil }
|
|
593
|
-
return UIImage(cgImage: cropped, scale: image.scale, orientation: image.imageOrientation)
|
|
594
|
-
}
|
|
595
|
-
```
|
|
596
|
-
|
|
597
|
-
For interactive cropping, consider wrapping a third-party crop view or building a gesture-based crop overlay in SwiftUI. The system does not provide a built-in crop controller.
|
|
1
|
+
# PhotosPicker Patterns
|
|
2
|
+
|
|
3
|
+
Complete recipes for photo and video selection, loading, saving, and processing. All patterns target iOS 16+ with SwiftUI and async/await.
|
|
4
|
+
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
## Contents
|
|
8
|
+
|
|
9
|
+
- [1. Single Photo Selection with Loading State](#1-single-photo-selection-with-loading-state)
|
|
10
|
+
- [2. Multi-Selection with Progress Tracking](#2-multi-selection-with-progress-tracking)
|
|
11
|
+
- [3. Loading Videos from PhotosPicker](#3-loading-videos-from-photospicker)
|
|
12
|
+
- [4. Loading Live Photos](#4-loading-live-photos)
|
|
13
|
+
- [5. PHPickerViewController Wrapping (UIKit Interop)](#5-phpickerviewcontroller-wrapping-uikit-interop)
|
|
14
|
+
- [6. Saving Images to Photo Library](#6-saving-images-to-photo-library)
|
|
15
|
+
- [7. Thumbnail Generation with ImageIO Downsampling](#7-thumbnail-generation-with-imageio-downsampling)
|
|
16
|
+
- [8. HEIC/HEIF Handling](#8-heicheif-handling)
|
|
17
|
+
- [9. Custom PhotosPicker Appearance](#9-custom-photospicker-appearance)
|
|
18
|
+
- [10. Image Cropping Pattern](#10-image-cropping-pattern)
|
|
19
|
+
|
|
20
|
+
## 1. Single Photo Selection with Loading State
|
|
21
|
+
|
|
22
|
+
Handle the full lifecycle: idle, loading, loaded, and error.
|
|
23
|
+
|
|
24
|
+
```swift
|
|
25
|
+
import SwiftUI
|
|
26
|
+
import PhotosUI
|
|
27
|
+
|
|
28
|
+
struct ProfilePhotoPicker: View {
|
|
29
|
+
@State private var selectedItem: PhotosPickerItem?
|
|
30
|
+
@State private var loadState: LoadState = .idle
|
|
31
|
+
|
|
32
|
+
enum LoadState {
|
|
33
|
+
case idle, loading, loaded(Image), error(String)
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
var body: some View {
|
|
37
|
+
VStack {
|
|
38
|
+
switch loadState {
|
|
39
|
+
case .idle:
|
|
40
|
+
Image(systemName: "person.crop.circle.fill")
|
|
41
|
+
.resizable()
|
|
42
|
+
.frame(width: 120, height: 120)
|
|
43
|
+
.foregroundStyle(.secondary)
|
|
44
|
+
case .loading:
|
|
45
|
+
ProgressView()
|
|
46
|
+
.frame(width: 120, height: 120)
|
|
47
|
+
case .loaded(let image):
|
|
48
|
+
image
|
|
49
|
+
.resizable()
|
|
50
|
+
.scaledToFill()
|
|
51
|
+
.frame(width: 120, height: 120)
|
|
52
|
+
.clipShape(Circle())
|
|
53
|
+
case .error(let message):
|
|
54
|
+
ContentUnavailableView(message, systemImage: "exclamationmark.triangle")
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
PhotosPicker("Choose Photo", selection: $selectedItem, matching: .images)
|
|
58
|
+
.buttonStyle(.borderedProminent)
|
|
59
|
+
}
|
|
60
|
+
.onChange(of: selectedItem) { _, newItem in
|
|
61
|
+
guard let newItem else {
|
|
62
|
+
loadState = .idle
|
|
63
|
+
return
|
|
64
|
+
}
|
|
65
|
+
loadState = .loading
|
|
66
|
+
Task {
|
|
67
|
+
do {
|
|
68
|
+
if let data = try await newItem.loadTransferable(type: Data.self),
|
|
69
|
+
let uiImage = UIImage(data: data) {
|
|
70
|
+
loadState = .loaded(Image(uiImage: uiImage))
|
|
71
|
+
} else {
|
|
72
|
+
loadState = .error("Unable to load image")
|
|
73
|
+
}
|
|
74
|
+
} catch {
|
|
75
|
+
loadState = .error(error.localizedDescription)
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
---
|
|
84
|
+
|
|
85
|
+
## 2. Multi-Selection with Progress Tracking
|
|
86
|
+
|
|
87
|
+
Load multiple images sequentially and report progress back to the UI.
|
|
88
|
+
|
|
89
|
+
```swift
|
|
90
|
+
import SwiftUI
|
|
91
|
+
import PhotosUI
|
|
92
|
+
|
|
93
|
+
struct MultiPhotoLoader: View {
|
|
94
|
+
@State private var selectedItems: [PhotosPickerItem] = []
|
|
95
|
+
@State private var loadedImages: [UIImage] = []
|
|
96
|
+
@State private var loadProgress: Double = 0
|
|
97
|
+
@State private var isLoading = false
|
|
98
|
+
|
|
99
|
+
var body: some View {
|
|
100
|
+
VStack {
|
|
101
|
+
if isLoading {
|
|
102
|
+
ProgressView(value: loadProgress)
|
|
103
|
+
.padding()
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
ScrollView(.horizontal) {
|
|
107
|
+
LazyHStack(spacing: 8) {
|
|
108
|
+
ForEach(loadedImages.indices, id: \.self) { index in
|
|
109
|
+
Image(uiImage: loadedImages[index])
|
|
110
|
+
.resizable()
|
|
111
|
+
.scaledToFill()
|
|
112
|
+
.frame(width: 100, height: 100)
|
|
113
|
+
.clipShape(RoundedRectangle(cornerRadius: 8))
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
.padding(.horizontal)
|
|
117
|
+
}
|
|
118
|
+
.frame(height: loadedImages.isEmpty ? 0 : 116)
|
|
119
|
+
|
|
120
|
+
PhotosPicker(
|
|
121
|
+
"Select Photos",
|
|
122
|
+
selection: $selectedItems,
|
|
123
|
+
maxSelectionCount: 10,
|
|
124
|
+
matching: .images
|
|
125
|
+
)
|
|
126
|
+
}
|
|
127
|
+
.onChange(of: selectedItems) { _, newItems in
|
|
128
|
+
Task { await loadImages(from: newItems) }
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
private func loadImages(from items: [PhotosPickerItem]) async {
|
|
133
|
+
isLoading = true
|
|
134
|
+
loadProgress = 0
|
|
135
|
+
var images: [UIImage] = []
|
|
136
|
+
|
|
137
|
+
for (index, item) in items.enumerated() {
|
|
138
|
+
if let data = try? await item.loadTransferable(type: Data.self),
|
|
139
|
+
let uiImage = UIImage(data: data) {
|
|
140
|
+
images.append(uiImage)
|
|
141
|
+
}
|
|
142
|
+
loadProgress = Double(index + 1) / Double(items.count)
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
loadedImages = images
|
|
146
|
+
isLoading = false
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
```
|
|
150
|
+
|
|
151
|
+
Load sequentially rather than concurrently to control memory usage. Each full-resolution image can be large; loading ten simultaneously risks memory termination.
|
|
152
|
+
|
|
153
|
+
---
|
|
154
|
+
|
|
155
|
+
## 3. Loading Videos from PhotosPicker
|
|
156
|
+
|
|
157
|
+
Use a `Transferable` wrapper that writes video data to a temporary file, since videos are too large to hold in memory as `Data`.
|
|
158
|
+
|
|
159
|
+
```swift
|
|
160
|
+
import SwiftUI
|
|
161
|
+
import PhotosUI
|
|
162
|
+
import AVKit
|
|
163
|
+
|
|
164
|
+
struct PickedMovie: Transferable {
|
|
165
|
+
let url: URL
|
|
166
|
+
|
|
167
|
+
static var transferRepresentation: some TransferRepresentation {
|
|
168
|
+
FileRepresentation(contentType: .movie) { movie in
|
|
169
|
+
SentTransferredFile(movie.url)
|
|
170
|
+
} importing: { received in
|
|
171
|
+
// Copy to a temporary location the app controls
|
|
172
|
+
let tempURL = FileManager.default.temporaryDirectory
|
|
173
|
+
.appendingPathComponent(UUID().uuidString)
|
|
174
|
+
.appendingPathExtension("mov")
|
|
175
|
+
try FileManager.default.copyItem(at: received.file, to: tempURL)
|
|
176
|
+
return PickedMovie(url: tempURL)
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
struct VideoPickerView: View {
|
|
182
|
+
@State private var selectedItem: PhotosPickerItem?
|
|
183
|
+
@State private var player: AVPlayer?
|
|
184
|
+
|
|
185
|
+
var body: some View {
|
|
186
|
+
VStack {
|
|
187
|
+
if let player {
|
|
188
|
+
VideoPlayer(player: player)
|
|
189
|
+
.frame(height: 300)
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
PhotosPicker("Select Video", selection: $selectedItem, matching: .videos)
|
|
193
|
+
}
|
|
194
|
+
.onChange(of: selectedItem) { _, newItem in
|
|
195
|
+
Task {
|
|
196
|
+
if let movie = try? await newItem?.loadTransferable(type: PickedMovie.self) {
|
|
197
|
+
player = AVPlayer(url: movie.url)
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
Always copy the received file to a temporary directory you control. The system may delete the original transfer file at any time.
|
|
206
|
+
|
|
207
|
+
---
|
|
208
|
+
|
|
209
|
+
## 4. Loading Live Photos
|
|
210
|
+
|
|
211
|
+
```swift
|
|
212
|
+
import SwiftUI
|
|
213
|
+
import PhotosUI
|
|
214
|
+
|
|
215
|
+
@available(iOS 17.0, *)
|
|
216
|
+
struct LivePhotoPickerView: View {
|
|
217
|
+
@State private var selectedItem: PhotosPickerItem?
|
|
218
|
+
@State private var livePhoto: PHLivePhoto?
|
|
219
|
+
|
|
220
|
+
var body: some View {
|
|
221
|
+
VStack {
|
|
222
|
+
if let livePhoto {
|
|
223
|
+
LivePhotoView(livePhoto: livePhoto)
|
|
224
|
+
.frame(height: 300)
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
PhotosPicker(
|
|
228
|
+
"Select Live Photo",
|
|
229
|
+
selection: $selectedItem,
|
|
230
|
+
matching: .livePhotos
|
|
231
|
+
)
|
|
232
|
+
}
|
|
233
|
+
.onChange(of: selectedItem) { _, newItem in
|
|
234
|
+
Task {
|
|
235
|
+
livePhoto = try? await newItem?.loadTransferable(type: PHLivePhoto.self)
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
```
|
|
241
|
+
|
|
242
|
+
`PHLivePhoto` conforms to `Transferable` on iOS 17+. On iOS 16, load the Live Photo components manually using `PHAsset`.
|
|
243
|
+
|
|
244
|
+
---
|
|
245
|
+
|
|
246
|
+
## 5. PHPickerViewController Wrapping (UIKit Interop)
|
|
247
|
+
|
|
248
|
+
Use `PHPickerViewController` when you need UIKit-level control or are integrating into an existing UIKit codebase. Prefer `PhotosPicker` for pure SwiftUI apps.
|
|
249
|
+
|
|
250
|
+
```swift
|
|
251
|
+
import SwiftUI
|
|
252
|
+
import PhotosUI
|
|
253
|
+
|
|
254
|
+
struct PHPickerWrapper: UIViewControllerRepresentable {
|
|
255
|
+
@Binding var selectedImages: [UIImage]
|
|
256
|
+
var selectionLimit: Int = 0
|
|
257
|
+
@Environment(\.dismiss) private var dismiss
|
|
258
|
+
|
|
259
|
+
func makeCoordinator() -> Coordinator { Coordinator(self) }
|
|
260
|
+
|
|
261
|
+
func makeUIViewController(context: Context) -> PHPickerViewController {
|
|
262
|
+
var config = PHPickerConfiguration(photoLibrary: .shared())
|
|
263
|
+
config.filter = .images
|
|
264
|
+
config.selectionLimit = selectionLimit
|
|
265
|
+
config.preferredAssetRepresentationMode = .current
|
|
266
|
+
|
|
267
|
+
let picker = PHPickerViewController(configuration: config)
|
|
268
|
+
picker.delegate = context.coordinator
|
|
269
|
+
return picker
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
func updateUIViewController(_ uiViewController: PHPickerViewController, context: Context) {
|
|
273
|
+
// Configuration is immutable after creation
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
final class Coordinator: NSObject, PHPickerViewControllerDelegate {
|
|
277
|
+
let parent: PHPickerWrapper
|
|
278
|
+
|
|
279
|
+
init(_ parent: PHPickerWrapper) { self.parent = parent }
|
|
280
|
+
|
|
281
|
+
func picker(
|
|
282
|
+
_ picker: PHPickerViewController,
|
|
283
|
+
didFinishPicking results: [PHPickerResult]
|
|
284
|
+
) {
|
|
285
|
+
parent.dismiss()
|
|
286
|
+
guard !results.isEmpty else { return }
|
|
287
|
+
|
|
288
|
+
Task { @MainActor in
|
|
289
|
+
var images: [UIImage] = []
|
|
290
|
+
for result in results {
|
|
291
|
+
if let image = await loadImage(from: result.itemProvider) {
|
|
292
|
+
images.append(image)
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
parent.selectedImages = images
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
private func loadImage(from provider: NSItemProvider) async -> UIImage? {
|
|
300
|
+
await withCheckedContinuation { continuation in
|
|
301
|
+
if provider.canLoadObject(ofClass: UIImage.self) {
|
|
302
|
+
provider.loadObject(ofClass: UIImage.self) { image, _ in
|
|
303
|
+
continuation.resume(returning: image as? UIImage)
|
|
304
|
+
}
|
|
305
|
+
} else {
|
|
306
|
+
continuation.resume(returning: nil)
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
```
|
|
313
|
+
|
|
314
|
+
Always call `parent.dismiss()` in `picker(_:didFinishPicking:)` -- this delegate method fires for both selection and cancellation (with empty results).
|
|
315
|
+
|
|
316
|
+
---
|
|
317
|
+
|
|
318
|
+
## 6. Saving Images to Photo Library
|
|
319
|
+
|
|
320
|
+
Saving requires `NSPhotoLibraryAddUsageDescription` in Info.plist. Use `PHPhotoLibrary` for saving with metadata, or `UIImageWriteToSavedPhotosAlbum` for simple saves.
|
|
321
|
+
|
|
322
|
+
### Using PHPhotoLibrary (Preferred)
|
|
323
|
+
|
|
324
|
+
```swift
|
|
325
|
+
import Photos
|
|
326
|
+
|
|
327
|
+
func saveImageToLibrary(_ image: UIImage) async throws {
|
|
328
|
+
try await PHPhotoLibrary.shared().performChanges {
|
|
329
|
+
let request = PHAssetCreationRequest.forAsset()
|
|
330
|
+
request.addResource(with: .photo, data: image.jpegData(compressionQuality: 0.9)!, options: nil)
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
func saveImageDataToLibrary(_ data: Data) async throws {
|
|
335
|
+
try await PHPhotoLibrary.shared().performChanges {
|
|
336
|
+
let request = PHAssetCreationRequest.forAsset()
|
|
337
|
+
let options = PHAssetResourceCreationOptions()
|
|
338
|
+
request.addResource(with: .photo, data: data, options: options)
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
```
|
|
342
|
+
|
|
343
|
+
### Saving to a Specific Album
|
|
344
|
+
|
|
345
|
+
```swift
|
|
346
|
+
import Photos
|
|
347
|
+
|
|
348
|
+
func saveToAlbum(image: UIImage, albumName: String) async throws {
|
|
349
|
+
let album = try await findOrCreateAlbum(named: albumName)
|
|
350
|
+
|
|
351
|
+
try await PHPhotoLibrary.shared().performChanges {
|
|
352
|
+
let assetRequest = PHAssetCreationRequest.forAsset()
|
|
353
|
+
assetRequest.addResource(
|
|
354
|
+
with: .photo,
|
|
355
|
+
data: image.jpegData(compressionQuality: 0.9)!,
|
|
356
|
+
options: nil
|
|
357
|
+
)
|
|
358
|
+
|
|
359
|
+
guard let placeholder = assetRequest.placeholderForCreatedAsset,
|
|
360
|
+
let albumChangeRequest = PHAssetCollectionChangeRequest(for: album) else { return }
|
|
361
|
+
albumChangeRequest.addAssets([placeholder] as NSArray)
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
private func findOrCreateAlbum(named name: String) async throws -> PHAssetCollection {
|
|
366
|
+
let fetchOptions = PHFetchOptions()
|
|
367
|
+
fetchOptions.predicate = NSPredicate(format: "title = %@", name)
|
|
368
|
+
let result = PHAssetCollection.fetchAssetCollections(with: .album, subtype: .any, options: fetchOptions)
|
|
369
|
+
|
|
370
|
+
if let existing = result.firstObject {
|
|
371
|
+
return existing
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
// Create the album
|
|
375
|
+
var placeholder: PHObjectPlaceholder?
|
|
376
|
+
try await PHPhotoLibrary.shared().performChanges {
|
|
377
|
+
let request = PHAssetCollectionChangeRequest.creationRequestForAssetCollection(withTitle: name)
|
|
378
|
+
placeholder = request.placeholderForCreatedAssetCollection
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
guard let placeholder,
|
|
382
|
+
let collection = PHAssetCollection.fetchAssetCollections(
|
|
383
|
+
withLocalIdentifiers: [placeholder.localIdentifier],
|
|
384
|
+
options: nil
|
|
385
|
+
).firstObject else {
|
|
386
|
+
throw PhotoLibraryError.albumCreationFailed
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
return collection
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
enum PhotoLibraryError: Error {
|
|
393
|
+
case albumCreationFailed
|
|
394
|
+
}
|
|
395
|
+
```
|
|
396
|
+
|
|
397
|
+
---
|
|
398
|
+
|
|
399
|
+
## 7. Thumbnail Generation with ImageIO Downsampling
|
|
400
|
+
|
|
401
|
+
Generate thumbnails efficiently without decoding the full image into memory. This is critical for grids and lists displaying many photos.
|
|
402
|
+
|
|
403
|
+
```swift
|
|
404
|
+
import ImageIO
|
|
405
|
+
import UIKit
|
|
406
|
+
|
|
407
|
+
/// Downsample image data to a target display size.
|
|
408
|
+
/// Use this instead of UIImage(data:) followed by resizing, which decodes the full image.
|
|
409
|
+
func downsample(
|
|
410
|
+
data: Data,
|
|
411
|
+
to pointSize: CGSize,
|
|
412
|
+
scale: CGFloat = UITraitCollection.current.displayScale
|
|
413
|
+
) -> UIImage? {
|
|
414
|
+
let maxDimension = max(pointSize.width, pointSize.height) * scale
|
|
415
|
+
|
|
416
|
+
let options: [CFString: Any] = [
|
|
417
|
+
kCGImageSourceCreateThumbnailFromImageAlways: true,
|
|
418
|
+
kCGImageSourceShouldCacheImmediately: true,
|
|
419
|
+
kCGImageSourceCreateThumbnailWithTransform: true,
|
|
420
|
+
kCGImageSourceThumbnailMaxPixelSize: maxDimension
|
|
421
|
+
]
|
|
422
|
+
|
|
423
|
+
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
|
|
424
|
+
let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, options as CFDictionary) else {
|
|
425
|
+
return nil
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
return UIImage(cgImage: cgImage)
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
/// Downsample from a file URL (avoids loading full data into memory).
|
|
432
|
+
func downsample(
|
|
433
|
+
url: URL,
|
|
434
|
+
to pointSize: CGSize,
|
|
435
|
+
scale: CGFloat = UITraitCollection.current.displayScale
|
|
436
|
+
) -> UIImage? {
|
|
437
|
+
let maxDimension = max(pointSize.width, pointSize.height) * scale
|
|
438
|
+
|
|
439
|
+
let sourceOptions: [CFString: Any] = [kCGImageSourceShouldCache: false]
|
|
440
|
+
guard let source = CGImageSourceCreateWithURL(url as CFURL, sourceOptions as CFDictionary) else {
|
|
441
|
+
return nil
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
let downsampleOptions: [CFString: Any] = [
|
|
445
|
+
kCGImageSourceCreateThumbnailFromImageAlways: true,
|
|
446
|
+
kCGImageSourceShouldCacheImmediately: true,
|
|
447
|
+
kCGImageSourceCreateThumbnailWithTransform: true,
|
|
448
|
+
kCGImageSourceThumbnailMaxPixelSize: maxDimension
|
|
449
|
+
]
|
|
450
|
+
|
|
451
|
+
guard let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, downsampleOptions as CFDictionary) else {
|
|
452
|
+
return nil
|
|
453
|
+
}
|
|
454
|
+
|
|
455
|
+
return UIImage(cgImage: cgImage)
|
|
456
|
+
}
|
|
457
|
+
```
|
|
458
|
+
|
|
459
|
+
### Usage in a Photo Grid
|
|
460
|
+
|
|
461
|
+
```swift
|
|
462
|
+
struct PhotoGridItem: View {
|
|
463
|
+
let imageData: Data
|
|
464
|
+
let thumbnailSize = CGSize(width: 100, height: 100)
|
|
465
|
+
|
|
466
|
+
@State private var thumbnail: UIImage?
|
|
467
|
+
|
|
468
|
+
var body: some View {
|
|
469
|
+
Group {
|
|
470
|
+
if let thumbnail {
|
|
471
|
+
Image(uiImage: thumbnail)
|
|
472
|
+
.resizable()
|
|
473
|
+
.scaledToFill()
|
|
474
|
+
} else {
|
|
475
|
+
Rectangle()
|
|
476
|
+
.fill(.quaternary)
|
|
477
|
+
.overlay(ProgressView())
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
.frame(width: 100, height: 100)
|
|
481
|
+
.clipShape(RoundedRectangle(cornerRadius: 6))
|
|
482
|
+
.task {
|
|
483
|
+
thumbnail = downsample(data: imageData, to: thumbnailSize)
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
```
|
|
488
|
+
|
|
489
|
+
---
|
|
490
|
+
|
|
491
|
+
## 8. HEIC/HEIF Handling
|
|
492
|
+
|
|
493
|
+
Modern iPhones capture photos in HEIC format by default. Handle both HEIC and JPEG transparently.
|
|
494
|
+
|
|
495
|
+
### Checking Image Format
|
|
496
|
+
|
|
497
|
+
```swift
|
|
498
|
+
import UniformTypeIdentifiers
|
|
499
|
+
|
|
500
|
+
func imageContentType(data: Data) -> UTType? {
|
|
501
|
+
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
|
|
502
|
+
let uti = CGImageSourceGetType(source) as? String else {
|
|
503
|
+
return nil
|
|
504
|
+
}
|
|
505
|
+
return UTType(uti)
|
|
506
|
+
}
|
|
507
|
+
```
|
|
508
|
+
|
|
509
|
+
### Converting HEIC to JPEG
|
|
510
|
+
|
|
511
|
+
```swift
|
|
512
|
+
import UIKit
|
|
513
|
+
|
|
514
|
+
func convertToJPEG(heicData: Data, compressionQuality: CGFloat = 0.9) -> Data? {
|
|
515
|
+
guard let image = UIImage(data: heicData) else { return nil }
|
|
516
|
+
return image.jpegData(compressionQuality: compressionQuality)
|
|
517
|
+
}
|
|
518
|
+
```
|
|
519
|
+
|
|
520
|
+
### Preserving HEIC When Possible
|
|
521
|
+
|
|
522
|
+
When saving or uploading, prefer keeping the original HEIC format to preserve quality and reduce file size. Convert to JPEG only when the destination requires it (e.g., a server that rejects HEIC).
|
|
523
|
+
|
|
524
|
+
```swift
|
|
525
|
+
import PhotosUI
|
|
526
|
+
|
|
527
|
+
// Request the current representation to avoid transcoding
|
|
528
|
+
var config = PHPickerConfiguration(photoLibrary: .shared())
|
|
529
|
+
config.preferredAssetRepresentationMode = .current // Keeps HEIC as-is
|
|
530
|
+
// vs .compatible which transcodes to JPEG
|
|
531
|
+
```
|
|
532
|
+
|
|
533
|
+
---
|
|
534
|
+
|
|
535
|
+
## 9. Custom PhotosPicker Appearance
|
|
536
|
+
|
|
537
|
+
Customize the picker trigger with any SwiftUI view using the label closure:
|
|
538
|
+
|
|
539
|
+
```swift
|
|
540
|
+
PhotosPicker(selection: $selectedItems, maxSelectionCount: 3, matching: .images) {
|
|
541
|
+
Label("Add Photos", systemImage: "photo.on.rectangle.angled")
|
|
542
|
+
.font(.headline)
|
|
543
|
+
.padding()
|
|
544
|
+
.background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 12))
|
|
545
|
+
}
|
|
546
|
+
```
|
|
547
|
+
|
|
548
|
+
### Inline PhotosPicker (iOS 17+)
|
|
549
|
+
|
|
550
|
+
Display the picker inline rather than as a sheet:
|
|
551
|
+
|
|
552
|
+
```swift
|
|
553
|
+
@available(iOS 17.0, *)
|
|
554
|
+
PhotosPicker(
|
|
555
|
+
selection: $selectedItems,
|
|
556
|
+
maxSelectionCount: 5,
|
|
557
|
+
matching: .images
|
|
558
|
+
)
|
|
559
|
+
.photosPickerStyle(.inline)
|
|
560
|
+
.frame(height: 300)
|
|
561
|
+
```
|
|
562
|
+
|
|
563
|
+
### Photos Picker Access Behavior (iOS 17+)
|
|
564
|
+
|
|
565
|
+
Control how the picker interacts with limited library access:
|
|
566
|
+
|
|
567
|
+
```swift
|
|
568
|
+
@available(iOS 17.0, *)
|
|
569
|
+
PhotosPicker(selection: $selectedItems, matching: .images)
|
|
570
|
+
.photosPickerAccessBehavior(.limited) // Only show user-approved photos
|
|
571
|
+
// .automatic (default) -- system decides
|
|
572
|
+
// .limited -- only previously approved photos
|
|
573
|
+
```
|
|
574
|
+
|
|
575
|
+
---
|
|
576
|
+
|
|
577
|
+
## 10. Image Cropping Pattern
|
|
578
|
+
|
|
579
|
+
A basic square crop using Core Graphics after selecting a photo:
|
|
580
|
+
|
|
581
|
+
```swift
|
|
582
|
+
import UIKit
|
|
583
|
+
|
|
584
|
+
func cropToSquare(_ image: UIImage) -> UIImage? {
|
|
585
|
+
guard let cgImage = image.cgImage else { return nil }
|
|
586
|
+
|
|
587
|
+
let side = min(cgImage.width, cgImage.height)
|
|
588
|
+
let x = (cgImage.width - side) / 2
|
|
589
|
+
let y = (cgImage.height - side) / 2
|
|
590
|
+
let cropRect = CGRect(x: x, y: y, width: side, height: side)
|
|
591
|
+
|
|
592
|
+
guard let cropped = cgImage.cropping(to: cropRect) else { return nil }
|
|
593
|
+
return UIImage(cgImage: cropped, scale: image.scale, orientation: image.imageOrientation)
|
|
594
|
+
}
|
|
595
|
+
```
|
|
596
|
+
|
|
597
|
+
For interactive cropping, consider wrapping a third-party crop view or building a gesture-based crop overlay in SwiftUI. The system does not provide a built-in crop controller.
|