@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,895 +1,895 @@
|
|
|
1
|
-
# Swift Charts Patterns Reference
|
|
2
|
-
|
|
3
|
-
Extended patterns, accessibility guidance, and theming for Swift Charts on
|
|
4
|
-
iOS 26+. Import `Charts` in every file that uses these APIs.
|
|
5
|
-
|
|
6
|
-
```swift
|
|
7
|
-
import SwiftUI
|
|
8
|
-
import Charts
|
|
9
|
-
```
|
|
10
|
-
|
|
11
|
-
---
|
|
12
|
-
|
|
13
|
-
## Contents
|
|
14
|
-
|
|
15
|
-
- [Data Modeling](#data-modeling)
|
|
16
|
-
- [Bar Chart Patterns](#bar-chart-patterns)
|
|
17
|
-
- [Line Chart Patterns](#line-chart-patterns)
|
|
18
|
-
- [Pie and Donut Chart Patterns (SectorMark, iOS 17+)](#pie-and-donut-chart-patterns-sectormark-ios-17)
|
|
19
|
-
- [Combined Chart Patterns](#combined-chart-patterns)
|
|
20
|
-
- [Chart Selection with Overlay Annotation](#chart-selection-with-overlay-annotation)
|
|
21
|
-
- [Scrollable Chart with Visible Domain](#scrollable-chart-with-visible-domain)
|
|
22
|
-
- [Function Plotting (LinePlot, iOS 18+)](#function-plotting-lineplot-ios-18)
|
|
23
|
-
- [Accessibility](#accessibility)
|
|
24
|
-
- [Dynamic Type and Color Considerations](#dynamic-type-and-color-considerations)
|
|
25
|
-
- [Performance: Vectorized Plots for Large Datasets](#performance-vectorized-plots-for-large-datasets)
|
|
26
|
-
- [Dark Mode and Theming](#dark-mode-and-theming)
|
|
27
|
-
- [Heat Map Pattern](#heat-map-pattern)
|
|
28
|
-
- [Stacking Methods](#stacking-methods)
|
|
29
|
-
- [MarkDimension Options](#markdimension-options)
|
|
30
|
-
- [Symbol Configuration](#symbol-configuration)
|
|
31
|
-
- [ChartProxy and Coordinate Conversion](#chartproxy-and-coordinate-conversion)
|
|
32
|
-
- [Quick Reference: Chart View Modifiers](#quick-reference-chart-view-modifiers)
|
|
33
|
-
- [Apple Documentation Links](#apple-documentation-links)
|
|
34
|
-
|
|
35
|
-
## Data Modeling
|
|
36
|
-
|
|
37
|
-
Use `@Observable` for chart data models. Pair with `@State` in views.
|
|
38
|
-
|
|
39
|
-
```swift
|
|
40
|
-
@Observable
|
|
41
|
-
class SalesModel {
|
|
42
|
-
var monthlySales: [MonthlySale] = []
|
|
43
|
-
|
|
44
|
-
func load() async {
|
|
45
|
-
monthlySales = await SalesService.fetchMonthlySales()
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
struct MonthlySale: Identifiable {
|
|
50
|
-
let id = UUID()
|
|
51
|
-
let month: Date
|
|
52
|
-
let revenue: Double
|
|
53
|
-
let category: String
|
|
54
|
-
}
|
|
55
|
-
```
|
|
56
|
-
|
|
57
|
-
```swift
|
|
58
|
-
struct SalesDashboard: View {
|
|
59
|
-
@State private var model = SalesModel()
|
|
60
|
-
|
|
61
|
-
var body: some View {
|
|
62
|
-
Chart(model.monthlySales) { item in
|
|
63
|
-
BarMark(
|
|
64
|
-
x: .value("Month", item.month, unit: .month),
|
|
65
|
-
y: .value("Revenue", item.revenue)
|
|
66
|
-
)
|
|
67
|
-
.foregroundStyle(by: .value("Category", item.category))
|
|
68
|
-
}
|
|
69
|
-
.task { await model.load() }
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
```
|
|
73
|
-
|
|
74
|
-
---
|
|
75
|
-
|
|
76
|
-
## Bar Chart Patterns
|
|
77
|
-
|
|
78
|
-
### Simple vertical bars
|
|
79
|
-
|
|
80
|
-
```swift
|
|
81
|
-
Chart(data) { item in
|
|
82
|
-
BarMark(
|
|
83
|
-
x: .value("Department", item.department),
|
|
84
|
-
y: .value("Revenue", item.revenue)
|
|
85
|
-
)
|
|
86
|
-
}
|
|
87
|
-
```
|
|
88
|
-
|
|
89
|
-
### Stacked bars (automatic)
|
|
90
|
-
|
|
91
|
-
When multiple bars share the same x value, they stack automatically:
|
|
92
|
-
|
|
93
|
-
```swift
|
|
94
|
-
Chart(data) { item in
|
|
95
|
-
BarMark(
|
|
96
|
-
x: .value("Quarter", item.quarter),
|
|
97
|
-
y: .value("Sales", item.sales)
|
|
98
|
-
)
|
|
99
|
-
.foregroundStyle(by: .value("Product", item.product))
|
|
100
|
-
}
|
|
101
|
-
```
|
|
102
|
-
|
|
103
|
-
### Grouped bars
|
|
104
|
-
|
|
105
|
-
Use `.position(by:)` to place bars side by side instead of stacking:
|
|
106
|
-
|
|
107
|
-
```swift
|
|
108
|
-
Chart(data) { item in
|
|
109
|
-
BarMark(
|
|
110
|
-
x: .value("Quarter", item.quarter),
|
|
111
|
-
y: .value("Sales", item.sales)
|
|
112
|
-
)
|
|
113
|
-
.foregroundStyle(by: .value("Product", item.product))
|
|
114
|
-
.position(by: .value("Product", item.product))
|
|
115
|
-
}
|
|
116
|
-
```
|
|
117
|
-
|
|
118
|
-
### Horizontal bars
|
|
119
|
-
|
|
120
|
-
Swap the x and y axes:
|
|
121
|
-
|
|
122
|
-
```swift
|
|
123
|
-
Chart(data) { item in
|
|
124
|
-
BarMark(
|
|
125
|
-
x: .value("Sales", item.sales),
|
|
126
|
-
y: .value("Region", item.region)
|
|
127
|
-
)
|
|
128
|
-
}
|
|
129
|
-
.chartYAxis {
|
|
130
|
-
AxisMarks { _ in
|
|
131
|
-
AxisValueLabel()
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
```
|
|
135
|
-
|
|
136
|
-
### Normalized stacked bars (100%)
|
|
137
|
-
|
|
138
|
-
```swift
|
|
139
|
-
Chart(data) { item in
|
|
140
|
-
BarMark(
|
|
141
|
-
x: .value("Quarter", item.quarter),
|
|
142
|
-
y: .value("Sales", item.sales),
|
|
143
|
-
stacking: .normalized
|
|
144
|
-
)
|
|
145
|
-
.foregroundStyle(by: .value("Product", item.product))
|
|
146
|
-
}
|
|
147
|
-
```
|
|
148
|
-
|
|
149
|
-
### Bar with annotation
|
|
150
|
-
|
|
151
|
-
```swift
|
|
152
|
-
Chart(data) { item in
|
|
153
|
-
BarMark(
|
|
154
|
-
x: .value("Month", item.month),
|
|
155
|
-
y: .value("Revenue", item.revenue)
|
|
156
|
-
)
|
|
157
|
-
.annotation(position: .top, alignment: .center, spacing: 4) {
|
|
158
|
-
Text(item.revenue, format: .currency(code: "USD").precision(.fractionLength(0)))
|
|
159
|
-
.font(.caption2)
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
```
|
|
163
|
-
|
|
164
|
-
### Gantt chart (interval bars)
|
|
165
|
-
|
|
166
|
-
```swift
|
|
167
|
-
Chart(tasks) { task in
|
|
168
|
-
BarMark(
|
|
169
|
-
xStart: .value("Start", task.startDate),
|
|
170
|
-
xEnd: .value("End", task.endDate),
|
|
171
|
-
y: .value("Task", task.name)
|
|
172
|
-
)
|
|
173
|
-
.foregroundStyle(by: .value("Status", task.status))
|
|
174
|
-
}
|
|
175
|
-
```
|
|
176
|
-
|
|
177
|
-
---
|
|
178
|
-
|
|
179
|
-
## Line Chart Patterns
|
|
180
|
-
|
|
181
|
-
### Single line with points
|
|
182
|
-
|
|
183
|
-
```swift
|
|
184
|
-
Chart(data) { item in
|
|
185
|
-
LineMark(
|
|
186
|
-
x: .value("Date", item.date),
|
|
187
|
-
y: .value("Price", item.price)
|
|
188
|
-
)
|
|
189
|
-
PointMark(
|
|
190
|
-
x: .value("Date", item.date),
|
|
191
|
-
y: .value("Price", item.price)
|
|
192
|
-
)
|
|
193
|
-
.symbolSize(30)
|
|
194
|
-
}
|
|
195
|
-
```
|
|
196
|
-
|
|
197
|
-
### Multi-series lines
|
|
198
|
-
|
|
199
|
-
```swift
|
|
200
|
-
Chart(temperatures) { item in
|
|
201
|
-
LineMark(
|
|
202
|
-
x: .value("Date", item.date),
|
|
203
|
-
y: .value("Temp", item.temperature)
|
|
204
|
-
)
|
|
205
|
-
.foregroundStyle(by: .value("City", item.city))
|
|
206
|
-
.symbol(by: .value("City", item.city))
|
|
207
|
-
}
|
|
208
|
-
```
|
|
209
|
-
|
|
210
|
-
### Line with area fill
|
|
211
|
-
|
|
212
|
-
```swift
|
|
213
|
-
Chart(data) { item in
|
|
214
|
-
AreaMark(
|
|
215
|
-
x: .value("Date", item.date),
|
|
216
|
-
y: .value("Value", item.value)
|
|
217
|
-
)
|
|
218
|
-
.foregroundStyle(
|
|
219
|
-
.linearGradient(
|
|
220
|
-
colors: [.blue.opacity(0.3), .blue.opacity(0.05)],
|
|
221
|
-
startPoint: .top,
|
|
222
|
-
endPoint: .bottom
|
|
223
|
-
)
|
|
224
|
-
)
|
|
225
|
-
LineMark(
|
|
226
|
-
x: .value("Date", item.date),
|
|
227
|
-
y: .value("Value", item.value)
|
|
228
|
-
)
|
|
229
|
-
.foregroundStyle(.blue)
|
|
230
|
-
}
|
|
231
|
-
```
|
|
232
|
-
|
|
233
|
-
### Interpolation methods
|
|
234
|
-
|
|
235
|
-
| Method | Use Case |
|
|
236
|
-
|---|---|
|
|
237
|
-
| `.linear` | Default; straight segments between points |
|
|
238
|
-
| `.monotone` | Smooth curve that preserves monotonicity |
|
|
239
|
-
| `.catmullRom` | Smooth general-purpose curve |
|
|
240
|
-
| `.cardinal` | Smooth with adjustable tension |
|
|
241
|
-
| `.stepStart` | Step function starting at data point |
|
|
242
|
-
| `.stepCenter` | Step function centered on data point |
|
|
243
|
-
| `.stepEnd` | Step function ending at data point |
|
|
244
|
-
|
|
245
|
-
```swift
|
|
246
|
-
LineMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
247
|
-
.interpolationMethod(.monotone)
|
|
248
|
-
```
|
|
249
|
-
|
|
250
|
-
### Sparkline (minimal inline chart)
|
|
251
|
-
|
|
252
|
-
```swift
|
|
253
|
-
Chart(recentData) { item in
|
|
254
|
-
LineMark(
|
|
255
|
-
x: .value("Time", item.time),
|
|
256
|
-
y: .value("Value", item.value)
|
|
257
|
-
)
|
|
258
|
-
.interpolationMethod(.catmullRom)
|
|
259
|
-
}
|
|
260
|
-
.chartXAxis(.hidden)
|
|
261
|
-
.chartYAxis(.hidden)
|
|
262
|
-
.chartLegend(.hidden)
|
|
263
|
-
.frame(width: 80, height: 30)
|
|
264
|
-
```
|
|
265
|
-
|
|
266
|
-
---
|
|
267
|
-
|
|
268
|
-
## Pie and Donut Chart Patterns (SectorMark, iOS 17+)
|
|
269
|
-
|
|
270
|
-
### Basic pie chart
|
|
271
|
-
|
|
272
|
-
```swift
|
|
273
|
-
Chart(products, id: \.name) { item in
|
|
274
|
-
SectorMark(angle: .value("Sales", item.sales))
|
|
275
|
-
.foregroundStyle(by: .value("Product", item.name))
|
|
276
|
-
}
|
|
277
|
-
```
|
|
278
|
-
|
|
279
|
-
### Donut chart with golden ratio inner radius
|
|
280
|
-
|
|
281
|
-
```swift
|
|
282
|
-
Chart(products, id: \.name) { item in
|
|
283
|
-
SectorMark(
|
|
284
|
-
angle: .value("Sales", item.sales),
|
|
285
|
-
innerRadius: .ratio(0.618),
|
|
286
|
-
outerRadius: .inset(10),
|
|
287
|
-
angularInset: 1
|
|
288
|
-
)
|
|
289
|
-
.cornerRadius(4)
|
|
290
|
-
.foregroundStyle(by: .value("Product", item.name))
|
|
291
|
-
}
|
|
292
|
-
```
|
|
293
|
-
|
|
294
|
-
### Donut chart with center label
|
|
295
|
-
|
|
296
|
-
```swift
|
|
297
|
-
Chart(products, id: \.name) { item in
|
|
298
|
-
SectorMark(
|
|
299
|
-
angle: .value("Sales", item.sales),
|
|
300
|
-
innerRadius: .ratio(0.618),
|
|
301
|
-
angularInset: 1
|
|
302
|
-
)
|
|
303
|
-
.cornerRadius(4)
|
|
304
|
-
.foregroundStyle(by: .value("Product", item.name))
|
|
305
|
-
}
|
|
306
|
-
.chartBackground { _ in
|
|
307
|
-
VStack {
|
|
308
|
-
Text("Total")
|
|
309
|
-
.font(.caption)
|
|
310
|
-
.foregroundStyle(.secondary)
|
|
311
|
-
Text("\(totalSales, format: .number)")
|
|
312
|
-
.font(.title2.bold())
|
|
313
|
-
}
|
|
314
|
-
}
|
|
315
|
-
```
|
|
316
|
-
|
|
317
|
-
### Angular selection on donut
|
|
318
|
-
|
|
319
|
-
```swift
|
|
320
|
-
@State private var selectedProduct: String?
|
|
321
|
-
|
|
322
|
-
Chart(products, id: \.name) { item in
|
|
323
|
-
SectorMark(
|
|
324
|
-
angle: .value("Sales", item.sales),
|
|
325
|
-
innerRadius: .ratio(0.618),
|
|
326
|
-
angularInset: 1
|
|
327
|
-
)
|
|
328
|
-
.cornerRadius(4)
|
|
329
|
-
.foregroundStyle(by: .value("Product", item.name))
|
|
330
|
-
.opacity(selectedProduct == nil || selectedProduct == item.name ? 1.0 : 0.4)
|
|
331
|
-
}
|
|
332
|
-
.chartAngleSelection(value: $selectedProduct)
|
|
333
|
-
```
|
|
334
|
-
|
|
335
|
-
### Grouping small slices
|
|
336
|
-
|
|
337
|
-
Limit pie/donut charts to 5-7 sectors. Group the rest into "Other":
|
|
338
|
-
|
|
339
|
-
```swift
|
|
340
|
-
func groupSmallSlices(_ data: [CategorySales], topN: Int = 5) -> [CategorySales] {
|
|
341
|
-
let sorted = data.sorted { $0.sales > $1.sales }
|
|
342
|
-
let top = Array(sorted.prefix(topN))
|
|
343
|
-
let otherTotal = sorted.dropFirst(topN).reduce(0) { $0 + $1.sales }
|
|
344
|
-
guard otherTotal > 0 else { return top }
|
|
345
|
-
return top + [CategorySales(name: "Other", sales: otherTotal)]
|
|
346
|
-
}
|
|
347
|
-
```
|
|
348
|
-
|
|
349
|
-
---
|
|
350
|
-
|
|
351
|
-
## Combined Chart Patterns
|
|
352
|
-
|
|
353
|
-
### Line + area (trend with fill)
|
|
354
|
-
|
|
355
|
-
```swift
|
|
356
|
-
Chart(data) { item in
|
|
357
|
-
AreaMark(
|
|
358
|
-
x: .value("Date", item.date),
|
|
359
|
-
yStart: .value("Min", item.low),
|
|
360
|
-
yEnd: .value("Max", item.high)
|
|
361
|
-
)
|
|
362
|
-
.foregroundStyle(.blue.opacity(0.15))
|
|
363
|
-
|
|
364
|
-
LineMark(
|
|
365
|
-
x: .value("Date", item.date),
|
|
366
|
-
y: .value("Average", item.average)
|
|
367
|
-
)
|
|
368
|
-
.foregroundStyle(.blue)
|
|
369
|
-
.lineStyle(StrokeStyle(lineWidth: 2))
|
|
370
|
-
}
|
|
371
|
-
```
|
|
372
|
-
|
|
373
|
-
### Bar + threshold rule
|
|
374
|
-
|
|
375
|
-
```swift
|
|
376
|
-
Chart {
|
|
377
|
-
ForEach(data) { item in
|
|
378
|
-
BarMark(
|
|
379
|
-
x: .value("Month", item.month),
|
|
380
|
-
y: .value("Revenue", item.revenue)
|
|
381
|
-
)
|
|
382
|
-
}
|
|
383
|
-
RuleMark(y: .value("Target", targetRevenue))
|
|
384
|
-
.foregroundStyle(.red)
|
|
385
|
-
.lineStyle(StrokeStyle(lineWidth: 1, dash: [5, 3]))
|
|
386
|
-
.annotation(position: .top, alignment: .leading) {
|
|
387
|
-
Text("Target: \(targetRevenue, format: .number)")
|
|
388
|
-
.font(.caption)
|
|
389
|
-
.foregroundStyle(.red)
|
|
390
|
-
}
|
|
391
|
-
}
|
|
392
|
-
```
|
|
393
|
-
|
|
394
|
-
### Scatter + trend line
|
|
395
|
-
|
|
396
|
-
```swift
|
|
397
|
-
Chart {
|
|
398
|
-
ForEach(data) { item in
|
|
399
|
-
PointMark(
|
|
400
|
-
x: .value("Experience", item.yearsExperience),
|
|
401
|
-
y: .value("Salary", item.salary)
|
|
402
|
-
)
|
|
403
|
-
.opacity(0.6)
|
|
404
|
-
}
|
|
405
|
-
LinePlot(x: "Experience", y: "Salary", domain: 0...20) { x in
|
|
406
|
-
baseSalary + x * salaryPerYear // linear trend
|
|
407
|
-
}
|
|
408
|
-
.foregroundStyle(.red)
|
|
409
|
-
.lineStyle(StrokeStyle(lineWidth: 1.5, dash: [4, 2]))
|
|
410
|
-
}
|
|
411
|
-
```
|
|
412
|
-
|
|
413
|
-
---
|
|
414
|
-
|
|
415
|
-
## Chart Selection with Overlay Annotation
|
|
416
|
-
|
|
417
|
-
Show a tooltip at the selected position using `chartOverlay`:
|
|
418
|
-
|
|
419
|
-
```swift
|
|
420
|
-
@State private var selectedDate: Date?
|
|
421
|
-
|
|
422
|
-
var body: some View {
|
|
423
|
-
Chart(data) { item in
|
|
424
|
-
LineMark(
|
|
425
|
-
x: .value("Date", item.date),
|
|
426
|
-
y: .value("Value", item.value)
|
|
427
|
-
)
|
|
428
|
-
if let selectedDate,
|
|
429
|
-
let match = data.first(where: { Calendar.current.isDate($0.date, inSameDayAs: selectedDate) }) {
|
|
430
|
-
RuleMark(x: .value("Selected", match.date))
|
|
431
|
-
.foregroundStyle(.secondary)
|
|
432
|
-
PointMark(
|
|
433
|
-
x: .value("Date", match.date),
|
|
434
|
-
y: .value("Value", match.value)
|
|
435
|
-
)
|
|
436
|
-
.symbolSize(60)
|
|
437
|
-
.annotation(position: .top) {
|
|
438
|
-
Text("\(match.value, format: .number)")
|
|
439
|
-
.font(.caption)
|
|
440
|
-
.padding(4)
|
|
441
|
-
.background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 4))
|
|
442
|
-
}
|
|
443
|
-
}
|
|
444
|
-
}
|
|
445
|
-
.chartXSelection(value: $selectedDate)
|
|
446
|
-
}
|
|
447
|
-
```
|
|
448
|
-
|
|
449
|
-
---
|
|
450
|
-
|
|
451
|
-
## Scrollable Chart with Visible Domain
|
|
452
|
-
|
|
453
|
-
```swift
|
|
454
|
-
@State private var scrollPosition: Date?
|
|
455
|
-
|
|
456
|
-
var body: some View {
|
|
457
|
-
Chart(dailySteps) { item in
|
|
458
|
-
BarMark(
|
|
459
|
-
x: .value("Date", item.date, unit: .day),
|
|
460
|
-
y: .value("Steps", item.steps)
|
|
461
|
-
)
|
|
462
|
-
}
|
|
463
|
-
.chartScrollableAxes(.horizontal)
|
|
464
|
-
.chartXVisibleDomain(length: 3600 * 24 * 7) // 7 days
|
|
465
|
-
.chartScrollPosition(x: $scrollPosition)
|
|
466
|
-
.chartScrollTargetBehavior(
|
|
467
|
-
.valueAligned(matching: DateComponents(hour: 0), majorAlignment: .page)
|
|
468
|
-
)
|
|
469
|
-
.chartXAxis {
|
|
470
|
-
AxisMarks(values: .stride(by: .day)) { value in
|
|
471
|
-
AxisGridLine()
|
|
472
|
-
AxisValueLabel(format: .dateTime.weekday(.abbreviated))
|
|
473
|
-
}
|
|
474
|
-
}
|
|
475
|
-
}
|
|
476
|
-
```
|
|
477
|
-
|
|
478
|
-
---
|
|
479
|
-
|
|
480
|
-
## Function Plotting (LinePlot, iOS 18+)
|
|
481
|
-
|
|
482
|
-
### Standard function y = f(x)
|
|
483
|
-
|
|
484
|
-
```swift
|
|
485
|
-
Chart {
|
|
486
|
-
LinePlot(x: "x", y: "y", domain: -2 * .pi ... 2 * .pi) { x in
|
|
487
|
-
sin(x)
|
|
488
|
-
}
|
|
489
|
-
.foregroundStyle(.blue)
|
|
490
|
-
}
|
|
491
|
-
.chartYScale(domain: -1.5...1.5)
|
|
492
|
-
```
|
|
493
|
-
|
|
494
|
-
### Parametric function (x, y) = f(t)
|
|
495
|
-
|
|
496
|
-
```swift
|
|
497
|
-
Chart {
|
|
498
|
-
LinePlot(x: "x", y: "y", t: "t", domain: 0 ... 2 * .pi) { t in
|
|
499
|
-
(x: cos(t), y: sin(t))
|
|
500
|
-
}
|
|
501
|
-
}
|
|
502
|
-
.chartXScale(domain: -1.5...1.5)
|
|
503
|
-
.chartYScale(domain: -1.5...1.5)
|
|
504
|
-
```
|
|
505
|
-
|
|
506
|
-
### Range area function
|
|
507
|
-
|
|
508
|
-
```swift
|
|
509
|
-
Chart {
|
|
510
|
-
AreaPlot(x: "x", yStart: "min", yEnd: "max", domain: 0...10) { x in
|
|
511
|
-
(yStart: sin(x) - 0.5, yEnd: sin(x) + 0.5)
|
|
512
|
-
}
|
|
513
|
-
.foregroundStyle(.blue.opacity(0.2))
|
|
514
|
-
}
|
|
515
|
-
```
|
|
516
|
-
|
|
517
|
-
---
|
|
518
|
-
|
|
519
|
-
## Accessibility
|
|
520
|
-
|
|
521
|
-
### Automatic VoiceOver support
|
|
522
|
-
|
|
523
|
-
Swift Charts provides automatic VoiceOver descriptions for chart elements. The
|
|
524
|
-
framework reads axis labels and values to visually impaired users without
|
|
525
|
-
additional code. Ensure `.value("Label", ...)` strings are descriptive.
|
|
526
|
-
|
|
527
|
-
### Custom accessibility labels
|
|
528
|
-
|
|
529
|
-
```swift
|
|
530
|
-
Chart(data) { item in
|
|
531
|
-
BarMark(
|
|
532
|
-
x: .value("Month", item.month),
|
|
533
|
-
y: .value("Sales", item.sales)
|
|
534
|
-
)
|
|
535
|
-
.accessibilityLabel("Sales for \(item.month)")
|
|
536
|
-
.accessibilityValue("\(item.sales) units sold")
|
|
537
|
-
}
|
|
538
|
-
```
|
|
539
|
-
|
|
540
|
-
### Accessibility on vectorized plots (KeyPath-based)
|
|
541
|
-
|
|
542
|
-
```swift
|
|
543
|
-
BarPlot(data, x: .value("Month", \.month), y: .value("Sales", \.sales))
|
|
544
|
-
.accessibilityLabel(\.accessibilityDescription)
|
|
545
|
-
.accessibilityValue(\.formattedSales)
|
|
546
|
-
```
|
|
547
|
-
|
|
548
|
-
### Audio graphs
|
|
549
|
-
|
|
550
|
-
The system automatically generates audio representations of chart data for
|
|
551
|
-
VoiceOver users. Use clear, consistent data labels to ensure audio graphs
|
|
552
|
-
convey meaningful patterns.
|
|
553
|
-
|
|
554
|
-
### Best practices
|
|
555
|
-
|
|
556
|
-
- Use descriptive strings in `.value("Label", ...)` -- these become VoiceOver labels.
|
|
557
|
-
- Add `.accessibilityLabel` and `.accessibilityValue` for context beyond raw numbers.
|
|
558
|
-
- Test with VoiceOver enabled: navigate the chart and verify each element is announced.
|
|
559
|
-
- Avoid `.accessibilityHidden(true)` on data-bearing marks.
|
|
560
|
-
|
|
561
|
-
---
|
|
562
|
-
|
|
563
|
-
## Dynamic Type and Color Considerations
|
|
564
|
-
|
|
565
|
-
### Dynamic Type
|
|
566
|
-
|
|
567
|
-
Charts automatically adjust axis label sizes with Dynamic Type. Avoid fixed
|
|
568
|
-
frame heights that clip labels at larger text sizes.
|
|
569
|
-
|
|
570
|
-
```swift
|
|
571
|
-
// WRONG -- clips at large text sizes
|
|
572
|
-
Chart(data) { ... }
|
|
573
|
-
.frame(height: 200)
|
|
574
|
-
|
|
575
|
-
// CORRECT -- adaptive height
|
|
576
|
-
Chart(data) { ... }
|
|
577
|
-
.frame(minHeight: 200)
|
|
578
|
-
.frame(maxHeight: 400)
|
|
579
|
-
```
|
|
580
|
-
|
|
581
|
-
Test charts at the "Accessibility Extra Extra Extra Large" text size to verify
|
|
582
|
-
axis labels, annotations, and legends remain readable.
|
|
583
|
-
|
|
584
|
-
### Color
|
|
585
|
-
|
|
586
|
-
- Avoid encoding meaning solely in color. Pair `.foregroundStyle(by:)` with
|
|
587
|
-
`.symbol(by:)` or `.lineStyle(by:)` for distinguishability.
|
|
588
|
-
- Use system colors that adapt to both light and dark modes.
|
|
589
|
-
- Test with color blindness simulations in the Accessibility Inspector.
|
|
590
|
-
|
|
591
|
-
```swift
|
|
592
|
-
LineMark(x: .value("Date", item.date), y: .value("Value", item.value))
|
|
593
|
-
.foregroundStyle(by: .value("Category", item.category))
|
|
594
|
-
.symbol(by: .value("Category", item.category))
|
|
595
|
-
.lineStyle(by: .value("Category", item.category))
|
|
596
|
-
```
|
|
597
|
-
|
|
598
|
-
---
|
|
599
|
-
|
|
600
|
-
## Performance: Vectorized Plots for Large Datasets
|
|
601
|
-
|
|
602
|
-
For datasets exceeding 1000 data points, use vectorized plot types instead of
|
|
603
|
-
individual marks. Vectorized plots accept entire collections and render
|
|
604
|
-
efficiently.
|
|
605
|
-
|
|
606
|
-
### When to use vectorized plots
|
|
607
|
-
|
|
608
|
-
| Data Points | Recommended Approach |
|
|
609
|
-
|---|---|
|
|
610
|
-
| < 100 | Individual marks (`BarMark`, `LineMark`, etc.) |
|
|
611
|
-
| 100 - 1000 | Either approach; profile if performance matters |
|
|
612
|
-
| > 1000 | Vectorized plots (`BarPlot`, `LinePlot`, etc.) |
|
|
613
|
-
|
|
614
|
-
### Data-driven vectorized plot
|
|
615
|
-
|
|
616
|
-
```swift
|
|
617
|
-
struct SensorReading: Identifiable {
|
|
618
|
-
let id: Int
|
|
619
|
-
let timestamp: Date
|
|
620
|
-
let temperature: Double
|
|
621
|
-
var color: Color { temperature > 30 ? .red : .blue }
|
|
622
|
-
var accessibilityDescription: Text {
|
|
623
|
-
Text("\(timestamp.formatted(.dateTime.hour().minute())): \(temperature, specifier: "%.1f") degrees")
|
|
624
|
-
}
|
|
625
|
-
}
|
|
626
|
-
|
|
627
|
-
Chart {
|
|
628
|
-
LinePlot(
|
|
629
|
-
readings,
|
|
630
|
-
x: .value("Time", \.timestamp),
|
|
631
|
-
y: .value("Temperature", \.temperature)
|
|
632
|
-
)
|
|
633
|
-
.foregroundStyle(.blue)
|
|
634
|
-
}
|
|
635
|
-
```
|
|
636
|
-
|
|
637
|
-
### KeyPath modifier ordering
|
|
638
|
-
|
|
639
|
-
Apply KeyPath-based modifiers before simple-value modifiers:
|
|
640
|
-
|
|
641
|
-
```swift
|
|
642
|
-
// WRONG
|
|
643
|
-
BarPlot(data, x: .value("X", \.x), y: .value("Y", \.y))
|
|
644
|
-
.opacity(0.8) // value modifier
|
|
645
|
-
.foregroundStyle(\.color) // KeyPath -- compiler error
|
|
646
|
-
|
|
647
|
-
// CORRECT
|
|
648
|
-
BarPlot(data, x: .value("X", \.x), y: .value("Y", \.y))
|
|
649
|
-
.foregroundStyle(\.color) // KeyPath first
|
|
650
|
-
.opacity(0.8) // value modifier second
|
|
651
|
-
```
|
|
652
|
-
|
|
653
|
-
### Available vectorized plot types
|
|
654
|
-
|
|
655
|
-
| Plot Type | Mark Equivalent | Available From |
|
|
656
|
-
|---|---|---|
|
|
657
|
-
| `BarPlot` | `BarMark` | iOS 18+ |
|
|
658
|
-
| `LinePlot` | `LineMark` | iOS 18+ |
|
|
659
|
-
| `PointPlot` | `PointMark` | iOS 18+ |
|
|
660
|
-
| `AreaPlot` | `AreaMark` | iOS 18+ |
|
|
661
|
-
| `RulePlot` | `RuleMark` | iOS 18+ |
|
|
662
|
-
| `RectanglePlot` | `RectangleMark` | iOS 18+ |
|
|
663
|
-
| `SectorPlot` | `SectorMark` | iOS 18+ |
|
|
664
|
-
|
|
665
|
-
---
|
|
666
|
-
|
|
667
|
-
## Dark Mode and Theming
|
|
668
|
-
|
|
669
|
-
### Automatic adaptation
|
|
670
|
-
|
|
671
|
-
Swift Charts inherits the current color scheme automatically. System colors
|
|
672
|
-
(`.blue`, `.orange`, `.green`) adapt to light and dark modes without extra code.
|
|
673
|
-
|
|
674
|
-
### Custom color palettes
|
|
675
|
-
|
|
676
|
-
Use `.chartForegroundStyleScale` to define a consistent palette:
|
|
677
|
-
|
|
678
|
-
```swift
|
|
679
|
-
Chart(data) { item in
|
|
680
|
-
BarMark(
|
|
681
|
-
x: .value("Category", item.category),
|
|
682
|
-
y: .value("Value", item.value)
|
|
683
|
-
)
|
|
684
|
-
.foregroundStyle(by: .value("Category", item.category))
|
|
685
|
-
}
|
|
686
|
-
.chartForegroundStyleScale([
|
|
687
|
-
"Electronics": .blue,
|
|
688
|
-
"Clothing": .purple,
|
|
689
|
-
"Food": .orange,
|
|
690
|
-
"Books": .green,
|
|
691
|
-
"Other": .gray
|
|
692
|
-
])
|
|
693
|
-
```
|
|
694
|
-
|
|
695
|
-
### Background and plot area styling
|
|
696
|
-
|
|
697
|
-
```swift
|
|
698
|
-
Chart(data) { ... }
|
|
699
|
-
.chartPlotStyle { plotArea in
|
|
700
|
-
plotArea
|
|
701
|
-
.background(.quaternary.opacity(0.3))
|
|
702
|
-
.border(.quaternary, width: 0.5)
|
|
703
|
-
}
|
|
704
|
-
```
|
|
705
|
-
|
|
706
|
-
### Axis styling
|
|
707
|
-
|
|
708
|
-
```swift
|
|
709
|
-
.chartXAxisStyle { axis in
|
|
710
|
-
axis.background(.blue.opacity(0.05))
|
|
711
|
-
}
|
|
712
|
-
```
|
|
713
|
-
|
|
714
|
-
### Testing dark mode
|
|
715
|
-
|
|
716
|
-
Always preview charts in both light and dark color schemes. In Xcode previews:
|
|
717
|
-
|
|
718
|
-
```swift
|
|
719
|
-
#Preview {
|
|
720
|
-
ChartView()
|
|
721
|
-
.preferredColorScheme(.dark)
|
|
722
|
-
}
|
|
723
|
-
```
|
|
724
|
-
|
|
725
|
-
Verify:
|
|
726
|
-
- Axis labels and grid lines are readable.
|
|
727
|
-
- Data colors maintain sufficient contrast.
|
|
728
|
-
- Annotations and legend text adapt properly.
|
|
729
|
-
|
|
730
|
-
---
|
|
731
|
-
|
|
732
|
-
## Heat Map Pattern
|
|
733
|
-
|
|
734
|
-
```swift
|
|
735
|
-
Chart(heatMapData) { item in
|
|
736
|
-
RectangleMark(
|
|
737
|
-
x: .value("Hour", item.hour),
|
|
738
|
-
y: .value("Day", item.day)
|
|
739
|
-
)
|
|
740
|
-
.foregroundStyle(by: .value("Count", item.count))
|
|
741
|
-
}
|
|
742
|
-
.chartForegroundStyleScale(range: Gradient(colors: [.blue, .yellow, .red]))
|
|
743
|
-
```
|
|
744
|
-
|
|
745
|
-
---
|
|
746
|
-
|
|
747
|
-
## Stacking Methods
|
|
748
|
-
|
|
749
|
-
| Method | Behavior |
|
|
750
|
-
|---|---|
|
|
751
|
-
| `.standard` | Default. Regions stack on top showing absolute values. |
|
|
752
|
-
| `.normalized` | Scales to 0-100% proportional view. |
|
|
753
|
-
| `.center` | Baseline centered (streamgraph). |
|
|
754
|
-
| `.unstacked` | Overlapping; no stacking. |
|
|
755
|
-
|
|
756
|
-
```swift
|
|
757
|
-
AreaMark(
|
|
758
|
-
x: .value("Date", item.date),
|
|
759
|
-
y: .value("Revenue", item.revenue),
|
|
760
|
-
stacking: .normalized
|
|
761
|
-
)
|
|
762
|
-
.foregroundStyle(by: .value("Category", item.category))
|
|
763
|
-
```
|
|
764
|
-
|
|
765
|
-
---
|
|
766
|
-
|
|
767
|
-
## MarkDimension Options
|
|
768
|
-
|
|
769
|
-
| Dimension | Description |
|
|
770
|
-
|---|---|
|
|
771
|
-
| `.automatic` | Framework decides |
|
|
772
|
-
| `.fixed(CGFloat)` | Exact pixel size |
|
|
773
|
-
| `.inset(CGFloat)` | Inset from available space |
|
|
774
|
-
| `.ratio(CGFloat)` | Proportion of available space (0...1) |
|
|
775
|
-
|
|
776
|
-
Use for `width`, `height` on `BarMark` and `innerRadius`, `outerRadius` on `SectorMark`.
|
|
777
|
-
|
|
778
|
-
---
|
|
779
|
-
|
|
780
|
-
## Symbol Configuration
|
|
781
|
-
|
|
782
|
-
### Built-in shapes
|
|
783
|
-
|
|
784
|
-
`circle`, `square`, `triangle`, `diamond`, `pentagon`, `plus`, `cross`, `asterisk`
|
|
785
|
-
|
|
786
|
-
```swift
|
|
787
|
-
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
788
|
-
.symbol(.diamond)
|
|
789
|
-
.symbolSize(80)
|
|
790
|
-
```
|
|
791
|
-
|
|
792
|
-
### Data-driven symbol encoding
|
|
793
|
-
|
|
794
|
-
```swift
|
|
795
|
-
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
796
|
-
.symbol(by: .value("Category", item.category))
|
|
797
|
-
```
|
|
798
|
-
|
|
799
|
-
### Custom symbol view
|
|
800
|
-
|
|
801
|
-
```swift
|
|
802
|
-
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
803
|
-
.symbol {
|
|
804
|
-
Image(systemName: "star.fill")
|
|
805
|
-
.font(.caption2)
|
|
806
|
-
}
|
|
807
|
-
```
|
|
808
|
-
|
|
809
|
-
---
|
|
810
|
-
|
|
811
|
-
## ChartProxy and Coordinate Conversion
|
|
812
|
-
|
|
813
|
-
Use `chartOverlay` or `chartBackground` to access `ChartProxy`:
|
|
814
|
-
|
|
815
|
-
```swift
|
|
816
|
-
.chartOverlay { proxy in
|
|
817
|
-
GeometryReader { geometry in
|
|
818
|
-
Rectangle()
|
|
819
|
-
.fill(.clear)
|
|
820
|
-
.contentShape(Rectangle())
|
|
821
|
-
.gesture(
|
|
822
|
-
DragGesture()
|
|
823
|
-
.onChanged { value in
|
|
824
|
-
let origin = geometry[proxy.plotAreaFrame].origin
|
|
825
|
-
let location = CGPoint(
|
|
826
|
-
x: value.location.x - origin.x,
|
|
827
|
-
y: value.location.y - origin.y
|
|
828
|
-
)
|
|
829
|
-
if let date: Date = proxy.value(atX: location.x) {
|
|
830
|
-
selectedDate = date
|
|
831
|
-
}
|
|
832
|
-
}
|
|
833
|
-
)
|
|
834
|
-
}
|
|
835
|
-
}
|
|
836
|
-
```
|
|
837
|
-
|
|
838
|
-
### Key ChartProxy methods
|
|
839
|
-
|
|
840
|
-
| Method | Purpose |
|
|
841
|
-
|---|---|
|
|
842
|
-
| `position(forX:)` | Data value to screen x-coordinate |
|
|
843
|
-
| `position(forY:)` | Data value to screen y-coordinate |
|
|
844
|
-
| `value(atX:as:)` | Screen x-coordinate to data value |
|
|
845
|
-
| `value(atY:as:)` | Screen y-coordinate to data value |
|
|
846
|
-
| `plotAreaSize` | Size of the plot area |
|
|
847
|
-
| `plotAreaFrame` | Anchor for the plot area frame |
|
|
848
|
-
|
|
849
|
-
---
|
|
850
|
-
|
|
851
|
-
## Quick Reference: Chart View Modifiers
|
|
852
|
-
|
|
853
|
-
### Axes
|
|
854
|
-
- `chartXAxis(_:)` / `chartXAxis(content:)`
|
|
855
|
-
- `chartYAxis(_:)` / `chartYAxis(content:)`
|
|
856
|
-
- `chartXAxisLabel(...)` / `chartYAxisLabel(...)`
|
|
857
|
-
- `chartXAxisStyle(content:)` / `chartYAxisStyle(content:)`
|
|
858
|
-
|
|
859
|
-
### Scales
|
|
860
|
-
- `chartXScale(domain:range:type:)` and variants
|
|
861
|
-
- `chartYScale(domain:range:type:)` and variants
|
|
862
|
-
- `chartForegroundStyleScale(_:)` -- custom color mapping
|
|
863
|
-
|
|
864
|
-
### Legend
|
|
865
|
-
- `chartLegend(_:)` -- visibility
|
|
866
|
-
- `chartLegend(position:alignment:spacing:)` -- positioning
|
|
867
|
-
- `chartLegend(position:alignment:spacing:content:)` -- custom content
|
|
868
|
-
|
|
869
|
-
### Selection (iOS 17+)
|
|
870
|
-
- `chartXSelection(value:)` / `chartXSelection(range:)`
|
|
871
|
-
- `chartYSelection(value:)` / `chartYSelection(range:)`
|
|
872
|
-
- `chartAngleSelection(value:)` -- for `SectorMark`
|
|
873
|
-
|
|
874
|
-
### Scrolling (iOS 17+)
|
|
875
|
-
- `chartScrollableAxes(_:)`
|
|
876
|
-
- `chartXVisibleDomain(length:)` / `chartYVisibleDomain(length:)`
|
|
877
|
-
- `chartScrollPosition(initialX:)` / `chartScrollPosition(x:)`
|
|
878
|
-
- `chartScrollTargetBehavior(_:)`
|
|
879
|
-
|
|
880
|
-
### Overlay and Background
|
|
881
|
-
- `chartOverlay(alignment:content:)` -- with `ChartProxy`
|
|
882
|
-
- `chartBackground(alignment:content:)` -- with `ChartProxy`
|
|
883
|
-
- `chartPlotStyle(content:)` -- plot area styling
|
|
884
|
-
|
|
885
|
-
---
|
|
886
|
-
|
|
887
|
-
## Apple Documentation Links
|
|
888
|
-
|
|
889
|
-
- [Swift Charts](https://sosumi.ai/documentation/Charts)
|
|
890
|
-
- [Creating a chart using Swift Charts](https://sosumi.ai/documentation/Charts/Creating-a-chart-using-Swift-Charts)
|
|
891
|
-
- [BarMark](https://sosumi.ai/documentation/Charts/BarMark)
|
|
892
|
-
- [LineMark](https://sosumi.ai/documentation/Charts/LineMark)
|
|
893
|
-
- [SectorMark](https://sosumi.ai/documentation/Charts/SectorMark)
|
|
894
|
-
- [LinePlot](https://sosumi.ai/documentation/Charts/LinePlot)
|
|
895
|
-
- [AxisMarks](https://sosumi.ai/documentation/Charts/AxisMarks)
|
|
1
|
+
# Swift Charts Patterns Reference
|
|
2
|
+
|
|
3
|
+
Extended patterns, accessibility guidance, and theming for Swift Charts on
|
|
4
|
+
iOS 26+. Import `Charts` in every file that uses these APIs.
|
|
5
|
+
|
|
6
|
+
```swift
|
|
7
|
+
import SwiftUI
|
|
8
|
+
import Charts
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
---
|
|
12
|
+
|
|
13
|
+
## Contents
|
|
14
|
+
|
|
15
|
+
- [Data Modeling](#data-modeling)
|
|
16
|
+
- [Bar Chart Patterns](#bar-chart-patterns)
|
|
17
|
+
- [Line Chart Patterns](#line-chart-patterns)
|
|
18
|
+
- [Pie and Donut Chart Patterns (SectorMark, iOS 17+)](#pie-and-donut-chart-patterns-sectormark-ios-17)
|
|
19
|
+
- [Combined Chart Patterns](#combined-chart-patterns)
|
|
20
|
+
- [Chart Selection with Overlay Annotation](#chart-selection-with-overlay-annotation)
|
|
21
|
+
- [Scrollable Chart with Visible Domain](#scrollable-chart-with-visible-domain)
|
|
22
|
+
- [Function Plotting (LinePlot, iOS 18+)](#function-plotting-lineplot-ios-18)
|
|
23
|
+
- [Accessibility](#accessibility)
|
|
24
|
+
- [Dynamic Type and Color Considerations](#dynamic-type-and-color-considerations)
|
|
25
|
+
- [Performance: Vectorized Plots for Large Datasets](#performance-vectorized-plots-for-large-datasets)
|
|
26
|
+
- [Dark Mode and Theming](#dark-mode-and-theming)
|
|
27
|
+
- [Heat Map Pattern](#heat-map-pattern)
|
|
28
|
+
- [Stacking Methods](#stacking-methods)
|
|
29
|
+
- [MarkDimension Options](#markdimension-options)
|
|
30
|
+
- [Symbol Configuration](#symbol-configuration)
|
|
31
|
+
- [ChartProxy and Coordinate Conversion](#chartproxy-and-coordinate-conversion)
|
|
32
|
+
- [Quick Reference: Chart View Modifiers](#quick-reference-chart-view-modifiers)
|
|
33
|
+
- [Apple Documentation Links](#apple-documentation-links)
|
|
34
|
+
|
|
35
|
+
## Data Modeling
|
|
36
|
+
|
|
37
|
+
Use `@Observable` for chart data models. Pair with `@State` in views.
|
|
38
|
+
|
|
39
|
+
```swift
|
|
40
|
+
@Observable
|
|
41
|
+
class SalesModel {
|
|
42
|
+
var monthlySales: [MonthlySale] = []
|
|
43
|
+
|
|
44
|
+
func load() async {
|
|
45
|
+
monthlySales = await SalesService.fetchMonthlySales()
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
struct MonthlySale: Identifiable {
|
|
50
|
+
let id = UUID()
|
|
51
|
+
let month: Date
|
|
52
|
+
let revenue: Double
|
|
53
|
+
let category: String
|
|
54
|
+
}
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
```swift
|
|
58
|
+
struct SalesDashboard: View {
|
|
59
|
+
@State private var model = SalesModel()
|
|
60
|
+
|
|
61
|
+
var body: some View {
|
|
62
|
+
Chart(model.monthlySales) { item in
|
|
63
|
+
BarMark(
|
|
64
|
+
x: .value("Month", item.month, unit: .month),
|
|
65
|
+
y: .value("Revenue", item.revenue)
|
|
66
|
+
)
|
|
67
|
+
.foregroundStyle(by: .value("Category", item.category))
|
|
68
|
+
}
|
|
69
|
+
.task { await model.load() }
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
```
|
|
73
|
+
|
|
74
|
+
---
|
|
75
|
+
|
|
76
|
+
## Bar Chart Patterns
|
|
77
|
+
|
|
78
|
+
### Simple vertical bars
|
|
79
|
+
|
|
80
|
+
```swift
|
|
81
|
+
Chart(data) { item in
|
|
82
|
+
BarMark(
|
|
83
|
+
x: .value("Department", item.department),
|
|
84
|
+
y: .value("Revenue", item.revenue)
|
|
85
|
+
)
|
|
86
|
+
}
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
### Stacked bars (automatic)
|
|
90
|
+
|
|
91
|
+
When multiple bars share the same x value, they stack automatically:
|
|
92
|
+
|
|
93
|
+
```swift
|
|
94
|
+
Chart(data) { item in
|
|
95
|
+
BarMark(
|
|
96
|
+
x: .value("Quarter", item.quarter),
|
|
97
|
+
y: .value("Sales", item.sales)
|
|
98
|
+
)
|
|
99
|
+
.foregroundStyle(by: .value("Product", item.product))
|
|
100
|
+
}
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
### Grouped bars
|
|
104
|
+
|
|
105
|
+
Use `.position(by:)` to place bars side by side instead of stacking:
|
|
106
|
+
|
|
107
|
+
```swift
|
|
108
|
+
Chart(data) { item in
|
|
109
|
+
BarMark(
|
|
110
|
+
x: .value("Quarter", item.quarter),
|
|
111
|
+
y: .value("Sales", item.sales)
|
|
112
|
+
)
|
|
113
|
+
.foregroundStyle(by: .value("Product", item.product))
|
|
114
|
+
.position(by: .value("Product", item.product))
|
|
115
|
+
}
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
### Horizontal bars
|
|
119
|
+
|
|
120
|
+
Swap the x and y axes:
|
|
121
|
+
|
|
122
|
+
```swift
|
|
123
|
+
Chart(data) { item in
|
|
124
|
+
BarMark(
|
|
125
|
+
x: .value("Sales", item.sales),
|
|
126
|
+
y: .value("Region", item.region)
|
|
127
|
+
)
|
|
128
|
+
}
|
|
129
|
+
.chartYAxis {
|
|
130
|
+
AxisMarks { _ in
|
|
131
|
+
AxisValueLabel()
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
```
|
|
135
|
+
|
|
136
|
+
### Normalized stacked bars (100%)
|
|
137
|
+
|
|
138
|
+
```swift
|
|
139
|
+
Chart(data) { item in
|
|
140
|
+
BarMark(
|
|
141
|
+
x: .value("Quarter", item.quarter),
|
|
142
|
+
y: .value("Sales", item.sales),
|
|
143
|
+
stacking: .normalized
|
|
144
|
+
)
|
|
145
|
+
.foregroundStyle(by: .value("Product", item.product))
|
|
146
|
+
}
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
### Bar with annotation
|
|
150
|
+
|
|
151
|
+
```swift
|
|
152
|
+
Chart(data) { item in
|
|
153
|
+
BarMark(
|
|
154
|
+
x: .value("Month", item.month),
|
|
155
|
+
y: .value("Revenue", item.revenue)
|
|
156
|
+
)
|
|
157
|
+
.annotation(position: .top, alignment: .center, spacing: 4) {
|
|
158
|
+
Text(item.revenue, format: .currency(code: "USD").precision(.fractionLength(0)))
|
|
159
|
+
.font(.caption2)
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
```
|
|
163
|
+
|
|
164
|
+
### Gantt chart (interval bars)
|
|
165
|
+
|
|
166
|
+
```swift
|
|
167
|
+
Chart(tasks) { task in
|
|
168
|
+
BarMark(
|
|
169
|
+
xStart: .value("Start", task.startDate),
|
|
170
|
+
xEnd: .value("End", task.endDate),
|
|
171
|
+
y: .value("Task", task.name)
|
|
172
|
+
)
|
|
173
|
+
.foregroundStyle(by: .value("Status", task.status))
|
|
174
|
+
}
|
|
175
|
+
```
|
|
176
|
+
|
|
177
|
+
---
|
|
178
|
+
|
|
179
|
+
## Line Chart Patterns
|
|
180
|
+
|
|
181
|
+
### Single line with points
|
|
182
|
+
|
|
183
|
+
```swift
|
|
184
|
+
Chart(data) { item in
|
|
185
|
+
LineMark(
|
|
186
|
+
x: .value("Date", item.date),
|
|
187
|
+
y: .value("Price", item.price)
|
|
188
|
+
)
|
|
189
|
+
PointMark(
|
|
190
|
+
x: .value("Date", item.date),
|
|
191
|
+
y: .value("Price", item.price)
|
|
192
|
+
)
|
|
193
|
+
.symbolSize(30)
|
|
194
|
+
}
|
|
195
|
+
```
|
|
196
|
+
|
|
197
|
+
### Multi-series lines
|
|
198
|
+
|
|
199
|
+
```swift
|
|
200
|
+
Chart(temperatures) { item in
|
|
201
|
+
LineMark(
|
|
202
|
+
x: .value("Date", item.date),
|
|
203
|
+
y: .value("Temp", item.temperature)
|
|
204
|
+
)
|
|
205
|
+
.foregroundStyle(by: .value("City", item.city))
|
|
206
|
+
.symbol(by: .value("City", item.city))
|
|
207
|
+
}
|
|
208
|
+
```
|
|
209
|
+
|
|
210
|
+
### Line with area fill
|
|
211
|
+
|
|
212
|
+
```swift
|
|
213
|
+
Chart(data) { item in
|
|
214
|
+
AreaMark(
|
|
215
|
+
x: .value("Date", item.date),
|
|
216
|
+
y: .value("Value", item.value)
|
|
217
|
+
)
|
|
218
|
+
.foregroundStyle(
|
|
219
|
+
.linearGradient(
|
|
220
|
+
colors: [.blue.opacity(0.3), .blue.opacity(0.05)],
|
|
221
|
+
startPoint: .top,
|
|
222
|
+
endPoint: .bottom
|
|
223
|
+
)
|
|
224
|
+
)
|
|
225
|
+
LineMark(
|
|
226
|
+
x: .value("Date", item.date),
|
|
227
|
+
y: .value("Value", item.value)
|
|
228
|
+
)
|
|
229
|
+
.foregroundStyle(.blue)
|
|
230
|
+
}
|
|
231
|
+
```
|
|
232
|
+
|
|
233
|
+
### Interpolation methods
|
|
234
|
+
|
|
235
|
+
| Method | Use Case |
|
|
236
|
+
|---|---|
|
|
237
|
+
| `.linear` | Default; straight segments between points |
|
|
238
|
+
| `.monotone` | Smooth curve that preserves monotonicity |
|
|
239
|
+
| `.catmullRom` | Smooth general-purpose curve |
|
|
240
|
+
| `.cardinal` | Smooth with adjustable tension |
|
|
241
|
+
| `.stepStart` | Step function starting at data point |
|
|
242
|
+
| `.stepCenter` | Step function centered on data point |
|
|
243
|
+
| `.stepEnd` | Step function ending at data point |
|
|
244
|
+
|
|
245
|
+
```swift
|
|
246
|
+
LineMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
247
|
+
.interpolationMethod(.monotone)
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
### Sparkline (minimal inline chart)
|
|
251
|
+
|
|
252
|
+
```swift
|
|
253
|
+
Chart(recentData) { item in
|
|
254
|
+
LineMark(
|
|
255
|
+
x: .value("Time", item.time),
|
|
256
|
+
y: .value("Value", item.value)
|
|
257
|
+
)
|
|
258
|
+
.interpolationMethod(.catmullRom)
|
|
259
|
+
}
|
|
260
|
+
.chartXAxis(.hidden)
|
|
261
|
+
.chartYAxis(.hidden)
|
|
262
|
+
.chartLegend(.hidden)
|
|
263
|
+
.frame(width: 80, height: 30)
|
|
264
|
+
```
|
|
265
|
+
|
|
266
|
+
---
|
|
267
|
+
|
|
268
|
+
## Pie and Donut Chart Patterns (SectorMark, iOS 17+)
|
|
269
|
+
|
|
270
|
+
### Basic pie chart
|
|
271
|
+
|
|
272
|
+
```swift
|
|
273
|
+
Chart(products, id: \.name) { item in
|
|
274
|
+
SectorMark(angle: .value("Sales", item.sales))
|
|
275
|
+
.foregroundStyle(by: .value("Product", item.name))
|
|
276
|
+
}
|
|
277
|
+
```
|
|
278
|
+
|
|
279
|
+
### Donut chart with golden ratio inner radius
|
|
280
|
+
|
|
281
|
+
```swift
|
|
282
|
+
Chart(products, id: \.name) { item in
|
|
283
|
+
SectorMark(
|
|
284
|
+
angle: .value("Sales", item.sales),
|
|
285
|
+
innerRadius: .ratio(0.618),
|
|
286
|
+
outerRadius: .inset(10),
|
|
287
|
+
angularInset: 1
|
|
288
|
+
)
|
|
289
|
+
.cornerRadius(4)
|
|
290
|
+
.foregroundStyle(by: .value("Product", item.name))
|
|
291
|
+
}
|
|
292
|
+
```
|
|
293
|
+
|
|
294
|
+
### Donut chart with center label
|
|
295
|
+
|
|
296
|
+
```swift
|
|
297
|
+
Chart(products, id: \.name) { item in
|
|
298
|
+
SectorMark(
|
|
299
|
+
angle: .value("Sales", item.sales),
|
|
300
|
+
innerRadius: .ratio(0.618),
|
|
301
|
+
angularInset: 1
|
|
302
|
+
)
|
|
303
|
+
.cornerRadius(4)
|
|
304
|
+
.foregroundStyle(by: .value("Product", item.name))
|
|
305
|
+
}
|
|
306
|
+
.chartBackground { _ in
|
|
307
|
+
VStack {
|
|
308
|
+
Text("Total")
|
|
309
|
+
.font(.caption)
|
|
310
|
+
.foregroundStyle(.secondary)
|
|
311
|
+
Text("\(totalSales, format: .number)")
|
|
312
|
+
.font(.title2.bold())
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
```
|
|
316
|
+
|
|
317
|
+
### Angular selection on donut
|
|
318
|
+
|
|
319
|
+
```swift
|
|
320
|
+
@State private var selectedProduct: String?
|
|
321
|
+
|
|
322
|
+
Chart(products, id: \.name) { item in
|
|
323
|
+
SectorMark(
|
|
324
|
+
angle: .value("Sales", item.sales),
|
|
325
|
+
innerRadius: .ratio(0.618),
|
|
326
|
+
angularInset: 1
|
|
327
|
+
)
|
|
328
|
+
.cornerRadius(4)
|
|
329
|
+
.foregroundStyle(by: .value("Product", item.name))
|
|
330
|
+
.opacity(selectedProduct == nil || selectedProduct == item.name ? 1.0 : 0.4)
|
|
331
|
+
}
|
|
332
|
+
.chartAngleSelection(value: $selectedProduct)
|
|
333
|
+
```
|
|
334
|
+
|
|
335
|
+
### Grouping small slices
|
|
336
|
+
|
|
337
|
+
Limit pie/donut charts to 5-7 sectors. Group the rest into "Other":
|
|
338
|
+
|
|
339
|
+
```swift
|
|
340
|
+
func groupSmallSlices(_ data: [CategorySales], topN: Int = 5) -> [CategorySales] {
|
|
341
|
+
let sorted = data.sorted { $0.sales > $1.sales }
|
|
342
|
+
let top = Array(sorted.prefix(topN))
|
|
343
|
+
let otherTotal = sorted.dropFirst(topN).reduce(0) { $0 + $1.sales }
|
|
344
|
+
guard otherTotal > 0 else { return top }
|
|
345
|
+
return top + [CategorySales(name: "Other", sales: otherTotal)]
|
|
346
|
+
}
|
|
347
|
+
```
|
|
348
|
+
|
|
349
|
+
---
|
|
350
|
+
|
|
351
|
+
## Combined Chart Patterns
|
|
352
|
+
|
|
353
|
+
### Line + area (trend with fill)
|
|
354
|
+
|
|
355
|
+
```swift
|
|
356
|
+
Chart(data) { item in
|
|
357
|
+
AreaMark(
|
|
358
|
+
x: .value("Date", item.date),
|
|
359
|
+
yStart: .value("Min", item.low),
|
|
360
|
+
yEnd: .value("Max", item.high)
|
|
361
|
+
)
|
|
362
|
+
.foregroundStyle(.blue.opacity(0.15))
|
|
363
|
+
|
|
364
|
+
LineMark(
|
|
365
|
+
x: .value("Date", item.date),
|
|
366
|
+
y: .value("Average", item.average)
|
|
367
|
+
)
|
|
368
|
+
.foregroundStyle(.blue)
|
|
369
|
+
.lineStyle(StrokeStyle(lineWidth: 2))
|
|
370
|
+
}
|
|
371
|
+
```
|
|
372
|
+
|
|
373
|
+
### Bar + threshold rule
|
|
374
|
+
|
|
375
|
+
```swift
|
|
376
|
+
Chart {
|
|
377
|
+
ForEach(data) { item in
|
|
378
|
+
BarMark(
|
|
379
|
+
x: .value("Month", item.month),
|
|
380
|
+
y: .value("Revenue", item.revenue)
|
|
381
|
+
)
|
|
382
|
+
}
|
|
383
|
+
RuleMark(y: .value("Target", targetRevenue))
|
|
384
|
+
.foregroundStyle(.red)
|
|
385
|
+
.lineStyle(StrokeStyle(lineWidth: 1, dash: [5, 3]))
|
|
386
|
+
.annotation(position: .top, alignment: .leading) {
|
|
387
|
+
Text("Target: \(targetRevenue, format: .number)")
|
|
388
|
+
.font(.caption)
|
|
389
|
+
.foregroundStyle(.red)
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
```
|
|
393
|
+
|
|
394
|
+
### Scatter + trend line
|
|
395
|
+
|
|
396
|
+
```swift
|
|
397
|
+
Chart {
|
|
398
|
+
ForEach(data) { item in
|
|
399
|
+
PointMark(
|
|
400
|
+
x: .value("Experience", item.yearsExperience),
|
|
401
|
+
y: .value("Salary", item.salary)
|
|
402
|
+
)
|
|
403
|
+
.opacity(0.6)
|
|
404
|
+
}
|
|
405
|
+
LinePlot(x: "Experience", y: "Salary", domain: 0...20) { x in
|
|
406
|
+
baseSalary + x * salaryPerYear // linear trend
|
|
407
|
+
}
|
|
408
|
+
.foregroundStyle(.red)
|
|
409
|
+
.lineStyle(StrokeStyle(lineWidth: 1.5, dash: [4, 2]))
|
|
410
|
+
}
|
|
411
|
+
```
|
|
412
|
+
|
|
413
|
+
---
|
|
414
|
+
|
|
415
|
+
## Chart Selection with Overlay Annotation
|
|
416
|
+
|
|
417
|
+
Show a tooltip at the selected position using `chartOverlay`:
|
|
418
|
+
|
|
419
|
+
```swift
|
|
420
|
+
@State private var selectedDate: Date?
|
|
421
|
+
|
|
422
|
+
var body: some View {
|
|
423
|
+
Chart(data) { item in
|
|
424
|
+
LineMark(
|
|
425
|
+
x: .value("Date", item.date),
|
|
426
|
+
y: .value("Value", item.value)
|
|
427
|
+
)
|
|
428
|
+
if let selectedDate,
|
|
429
|
+
let match = data.first(where: { Calendar.current.isDate($0.date, inSameDayAs: selectedDate) }) {
|
|
430
|
+
RuleMark(x: .value("Selected", match.date))
|
|
431
|
+
.foregroundStyle(.secondary)
|
|
432
|
+
PointMark(
|
|
433
|
+
x: .value("Date", match.date),
|
|
434
|
+
y: .value("Value", match.value)
|
|
435
|
+
)
|
|
436
|
+
.symbolSize(60)
|
|
437
|
+
.annotation(position: .top) {
|
|
438
|
+
Text("\(match.value, format: .number)")
|
|
439
|
+
.font(.caption)
|
|
440
|
+
.padding(4)
|
|
441
|
+
.background(.ultraThinMaterial, in: RoundedRectangle(cornerRadius: 4))
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
.chartXSelection(value: $selectedDate)
|
|
446
|
+
}
|
|
447
|
+
```
|
|
448
|
+
|
|
449
|
+
---
|
|
450
|
+
|
|
451
|
+
## Scrollable Chart with Visible Domain
|
|
452
|
+
|
|
453
|
+
```swift
|
|
454
|
+
@State private var scrollPosition: Date?
|
|
455
|
+
|
|
456
|
+
var body: some View {
|
|
457
|
+
Chart(dailySteps) { item in
|
|
458
|
+
BarMark(
|
|
459
|
+
x: .value("Date", item.date, unit: .day),
|
|
460
|
+
y: .value("Steps", item.steps)
|
|
461
|
+
)
|
|
462
|
+
}
|
|
463
|
+
.chartScrollableAxes(.horizontal)
|
|
464
|
+
.chartXVisibleDomain(length: 3600 * 24 * 7) // 7 days
|
|
465
|
+
.chartScrollPosition(x: $scrollPosition)
|
|
466
|
+
.chartScrollTargetBehavior(
|
|
467
|
+
.valueAligned(matching: DateComponents(hour: 0), majorAlignment: .page)
|
|
468
|
+
)
|
|
469
|
+
.chartXAxis {
|
|
470
|
+
AxisMarks(values: .stride(by: .day)) { value in
|
|
471
|
+
AxisGridLine()
|
|
472
|
+
AxisValueLabel(format: .dateTime.weekday(.abbreviated))
|
|
473
|
+
}
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
```
|
|
477
|
+
|
|
478
|
+
---
|
|
479
|
+
|
|
480
|
+
## Function Plotting (LinePlot, iOS 18+)
|
|
481
|
+
|
|
482
|
+
### Standard function y = f(x)
|
|
483
|
+
|
|
484
|
+
```swift
|
|
485
|
+
Chart {
|
|
486
|
+
LinePlot(x: "x", y: "y", domain: -2 * .pi ... 2 * .pi) { x in
|
|
487
|
+
sin(x)
|
|
488
|
+
}
|
|
489
|
+
.foregroundStyle(.blue)
|
|
490
|
+
}
|
|
491
|
+
.chartYScale(domain: -1.5...1.5)
|
|
492
|
+
```
|
|
493
|
+
|
|
494
|
+
### Parametric function (x, y) = f(t)
|
|
495
|
+
|
|
496
|
+
```swift
|
|
497
|
+
Chart {
|
|
498
|
+
LinePlot(x: "x", y: "y", t: "t", domain: 0 ... 2 * .pi) { t in
|
|
499
|
+
(x: cos(t), y: sin(t))
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
.chartXScale(domain: -1.5...1.5)
|
|
503
|
+
.chartYScale(domain: -1.5...1.5)
|
|
504
|
+
```
|
|
505
|
+
|
|
506
|
+
### Range area function
|
|
507
|
+
|
|
508
|
+
```swift
|
|
509
|
+
Chart {
|
|
510
|
+
AreaPlot(x: "x", yStart: "min", yEnd: "max", domain: 0...10) { x in
|
|
511
|
+
(yStart: sin(x) - 0.5, yEnd: sin(x) + 0.5)
|
|
512
|
+
}
|
|
513
|
+
.foregroundStyle(.blue.opacity(0.2))
|
|
514
|
+
}
|
|
515
|
+
```
|
|
516
|
+
|
|
517
|
+
---
|
|
518
|
+
|
|
519
|
+
## Accessibility
|
|
520
|
+
|
|
521
|
+
### Automatic VoiceOver support
|
|
522
|
+
|
|
523
|
+
Swift Charts provides automatic VoiceOver descriptions for chart elements. The
|
|
524
|
+
framework reads axis labels and values to visually impaired users without
|
|
525
|
+
additional code. Ensure `.value("Label", ...)` strings are descriptive.
|
|
526
|
+
|
|
527
|
+
### Custom accessibility labels
|
|
528
|
+
|
|
529
|
+
```swift
|
|
530
|
+
Chart(data) { item in
|
|
531
|
+
BarMark(
|
|
532
|
+
x: .value("Month", item.month),
|
|
533
|
+
y: .value("Sales", item.sales)
|
|
534
|
+
)
|
|
535
|
+
.accessibilityLabel("Sales for \(item.month)")
|
|
536
|
+
.accessibilityValue("\(item.sales) units sold")
|
|
537
|
+
}
|
|
538
|
+
```
|
|
539
|
+
|
|
540
|
+
### Accessibility on vectorized plots (KeyPath-based)
|
|
541
|
+
|
|
542
|
+
```swift
|
|
543
|
+
BarPlot(data, x: .value("Month", \.month), y: .value("Sales", \.sales))
|
|
544
|
+
.accessibilityLabel(\.accessibilityDescription)
|
|
545
|
+
.accessibilityValue(\.formattedSales)
|
|
546
|
+
```
|
|
547
|
+
|
|
548
|
+
### Audio graphs
|
|
549
|
+
|
|
550
|
+
The system automatically generates audio representations of chart data for
|
|
551
|
+
VoiceOver users. Use clear, consistent data labels to ensure audio graphs
|
|
552
|
+
convey meaningful patterns.
|
|
553
|
+
|
|
554
|
+
### Best practices
|
|
555
|
+
|
|
556
|
+
- Use descriptive strings in `.value("Label", ...)` -- these become VoiceOver labels.
|
|
557
|
+
- Add `.accessibilityLabel` and `.accessibilityValue` for context beyond raw numbers.
|
|
558
|
+
- Test with VoiceOver enabled: navigate the chart and verify each element is announced.
|
|
559
|
+
- Avoid `.accessibilityHidden(true)` on data-bearing marks.
|
|
560
|
+
|
|
561
|
+
---
|
|
562
|
+
|
|
563
|
+
## Dynamic Type and Color Considerations
|
|
564
|
+
|
|
565
|
+
### Dynamic Type
|
|
566
|
+
|
|
567
|
+
Charts automatically adjust axis label sizes with Dynamic Type. Avoid fixed
|
|
568
|
+
frame heights that clip labels at larger text sizes.
|
|
569
|
+
|
|
570
|
+
```swift
|
|
571
|
+
// WRONG -- clips at large text sizes
|
|
572
|
+
Chart(data) { ... }
|
|
573
|
+
.frame(height: 200)
|
|
574
|
+
|
|
575
|
+
// CORRECT -- adaptive height
|
|
576
|
+
Chart(data) { ... }
|
|
577
|
+
.frame(minHeight: 200)
|
|
578
|
+
.frame(maxHeight: 400)
|
|
579
|
+
```
|
|
580
|
+
|
|
581
|
+
Test charts at the "Accessibility Extra Extra Extra Large" text size to verify
|
|
582
|
+
axis labels, annotations, and legends remain readable.
|
|
583
|
+
|
|
584
|
+
### Color
|
|
585
|
+
|
|
586
|
+
- Avoid encoding meaning solely in color. Pair `.foregroundStyle(by:)` with
|
|
587
|
+
`.symbol(by:)` or `.lineStyle(by:)` for distinguishability.
|
|
588
|
+
- Use system colors that adapt to both light and dark modes.
|
|
589
|
+
- Test with color blindness simulations in the Accessibility Inspector.
|
|
590
|
+
|
|
591
|
+
```swift
|
|
592
|
+
LineMark(x: .value("Date", item.date), y: .value("Value", item.value))
|
|
593
|
+
.foregroundStyle(by: .value("Category", item.category))
|
|
594
|
+
.symbol(by: .value("Category", item.category))
|
|
595
|
+
.lineStyle(by: .value("Category", item.category))
|
|
596
|
+
```
|
|
597
|
+
|
|
598
|
+
---
|
|
599
|
+
|
|
600
|
+
## Performance: Vectorized Plots for Large Datasets
|
|
601
|
+
|
|
602
|
+
For datasets exceeding 1000 data points, use vectorized plot types instead of
|
|
603
|
+
individual marks. Vectorized plots accept entire collections and render
|
|
604
|
+
efficiently.
|
|
605
|
+
|
|
606
|
+
### When to use vectorized plots
|
|
607
|
+
|
|
608
|
+
| Data Points | Recommended Approach |
|
|
609
|
+
|---|---|
|
|
610
|
+
| < 100 | Individual marks (`BarMark`, `LineMark`, etc.) |
|
|
611
|
+
| 100 - 1000 | Either approach; profile if performance matters |
|
|
612
|
+
| > 1000 | Vectorized plots (`BarPlot`, `LinePlot`, etc.) |
|
|
613
|
+
|
|
614
|
+
### Data-driven vectorized plot
|
|
615
|
+
|
|
616
|
+
```swift
|
|
617
|
+
struct SensorReading: Identifiable {
|
|
618
|
+
let id: Int
|
|
619
|
+
let timestamp: Date
|
|
620
|
+
let temperature: Double
|
|
621
|
+
var color: Color { temperature > 30 ? .red : .blue }
|
|
622
|
+
var accessibilityDescription: Text {
|
|
623
|
+
Text("\(timestamp.formatted(.dateTime.hour().minute())): \(temperature, specifier: "%.1f") degrees")
|
|
624
|
+
}
|
|
625
|
+
}
|
|
626
|
+
|
|
627
|
+
Chart {
|
|
628
|
+
LinePlot(
|
|
629
|
+
readings,
|
|
630
|
+
x: .value("Time", \.timestamp),
|
|
631
|
+
y: .value("Temperature", \.temperature)
|
|
632
|
+
)
|
|
633
|
+
.foregroundStyle(.blue)
|
|
634
|
+
}
|
|
635
|
+
```
|
|
636
|
+
|
|
637
|
+
### KeyPath modifier ordering
|
|
638
|
+
|
|
639
|
+
Apply KeyPath-based modifiers before simple-value modifiers:
|
|
640
|
+
|
|
641
|
+
```swift
|
|
642
|
+
// WRONG
|
|
643
|
+
BarPlot(data, x: .value("X", \.x), y: .value("Y", \.y))
|
|
644
|
+
.opacity(0.8) // value modifier
|
|
645
|
+
.foregroundStyle(\.color) // KeyPath -- compiler error
|
|
646
|
+
|
|
647
|
+
// CORRECT
|
|
648
|
+
BarPlot(data, x: .value("X", \.x), y: .value("Y", \.y))
|
|
649
|
+
.foregroundStyle(\.color) // KeyPath first
|
|
650
|
+
.opacity(0.8) // value modifier second
|
|
651
|
+
```
|
|
652
|
+
|
|
653
|
+
### Available vectorized plot types
|
|
654
|
+
|
|
655
|
+
| Plot Type | Mark Equivalent | Available From |
|
|
656
|
+
|---|---|---|
|
|
657
|
+
| `BarPlot` | `BarMark` | iOS 18+ |
|
|
658
|
+
| `LinePlot` | `LineMark` | iOS 18+ |
|
|
659
|
+
| `PointPlot` | `PointMark` | iOS 18+ |
|
|
660
|
+
| `AreaPlot` | `AreaMark` | iOS 18+ |
|
|
661
|
+
| `RulePlot` | `RuleMark` | iOS 18+ |
|
|
662
|
+
| `RectanglePlot` | `RectangleMark` | iOS 18+ |
|
|
663
|
+
| `SectorPlot` | `SectorMark` | iOS 18+ |
|
|
664
|
+
|
|
665
|
+
---
|
|
666
|
+
|
|
667
|
+
## Dark Mode and Theming
|
|
668
|
+
|
|
669
|
+
### Automatic adaptation
|
|
670
|
+
|
|
671
|
+
Swift Charts inherits the current color scheme automatically. System colors
|
|
672
|
+
(`.blue`, `.orange`, `.green`) adapt to light and dark modes without extra code.
|
|
673
|
+
|
|
674
|
+
### Custom color palettes
|
|
675
|
+
|
|
676
|
+
Use `.chartForegroundStyleScale` to define a consistent palette:
|
|
677
|
+
|
|
678
|
+
```swift
|
|
679
|
+
Chart(data) { item in
|
|
680
|
+
BarMark(
|
|
681
|
+
x: .value("Category", item.category),
|
|
682
|
+
y: .value("Value", item.value)
|
|
683
|
+
)
|
|
684
|
+
.foregroundStyle(by: .value("Category", item.category))
|
|
685
|
+
}
|
|
686
|
+
.chartForegroundStyleScale([
|
|
687
|
+
"Electronics": .blue,
|
|
688
|
+
"Clothing": .purple,
|
|
689
|
+
"Food": .orange,
|
|
690
|
+
"Books": .green,
|
|
691
|
+
"Other": .gray
|
|
692
|
+
])
|
|
693
|
+
```
|
|
694
|
+
|
|
695
|
+
### Background and plot area styling
|
|
696
|
+
|
|
697
|
+
```swift
|
|
698
|
+
Chart(data) { ... }
|
|
699
|
+
.chartPlotStyle { plotArea in
|
|
700
|
+
plotArea
|
|
701
|
+
.background(.quaternary.opacity(0.3))
|
|
702
|
+
.border(.quaternary, width: 0.5)
|
|
703
|
+
}
|
|
704
|
+
```
|
|
705
|
+
|
|
706
|
+
### Axis styling
|
|
707
|
+
|
|
708
|
+
```swift
|
|
709
|
+
.chartXAxisStyle { axis in
|
|
710
|
+
axis.background(.blue.opacity(0.05))
|
|
711
|
+
}
|
|
712
|
+
```
|
|
713
|
+
|
|
714
|
+
### Testing dark mode
|
|
715
|
+
|
|
716
|
+
Always preview charts in both light and dark color schemes. In Xcode previews:
|
|
717
|
+
|
|
718
|
+
```swift
|
|
719
|
+
#Preview {
|
|
720
|
+
ChartView()
|
|
721
|
+
.preferredColorScheme(.dark)
|
|
722
|
+
}
|
|
723
|
+
```
|
|
724
|
+
|
|
725
|
+
Verify:
|
|
726
|
+
- Axis labels and grid lines are readable.
|
|
727
|
+
- Data colors maintain sufficient contrast.
|
|
728
|
+
- Annotations and legend text adapt properly.
|
|
729
|
+
|
|
730
|
+
---
|
|
731
|
+
|
|
732
|
+
## Heat Map Pattern
|
|
733
|
+
|
|
734
|
+
```swift
|
|
735
|
+
Chart(heatMapData) { item in
|
|
736
|
+
RectangleMark(
|
|
737
|
+
x: .value("Hour", item.hour),
|
|
738
|
+
y: .value("Day", item.day)
|
|
739
|
+
)
|
|
740
|
+
.foregroundStyle(by: .value("Count", item.count))
|
|
741
|
+
}
|
|
742
|
+
.chartForegroundStyleScale(range: Gradient(colors: [.blue, .yellow, .red]))
|
|
743
|
+
```
|
|
744
|
+
|
|
745
|
+
---
|
|
746
|
+
|
|
747
|
+
## Stacking Methods
|
|
748
|
+
|
|
749
|
+
| Method | Behavior |
|
|
750
|
+
|---|---|
|
|
751
|
+
| `.standard` | Default. Regions stack on top showing absolute values. |
|
|
752
|
+
| `.normalized` | Scales to 0-100% proportional view. |
|
|
753
|
+
| `.center` | Baseline centered (streamgraph). |
|
|
754
|
+
| `.unstacked` | Overlapping; no stacking. |
|
|
755
|
+
|
|
756
|
+
```swift
|
|
757
|
+
AreaMark(
|
|
758
|
+
x: .value("Date", item.date),
|
|
759
|
+
y: .value("Revenue", item.revenue),
|
|
760
|
+
stacking: .normalized
|
|
761
|
+
)
|
|
762
|
+
.foregroundStyle(by: .value("Category", item.category))
|
|
763
|
+
```
|
|
764
|
+
|
|
765
|
+
---
|
|
766
|
+
|
|
767
|
+
## MarkDimension Options
|
|
768
|
+
|
|
769
|
+
| Dimension | Description |
|
|
770
|
+
|---|---|
|
|
771
|
+
| `.automatic` | Framework decides |
|
|
772
|
+
| `.fixed(CGFloat)` | Exact pixel size |
|
|
773
|
+
| `.inset(CGFloat)` | Inset from available space |
|
|
774
|
+
| `.ratio(CGFloat)` | Proportion of available space (0...1) |
|
|
775
|
+
|
|
776
|
+
Use for `width`, `height` on `BarMark` and `innerRadius`, `outerRadius` on `SectorMark`.
|
|
777
|
+
|
|
778
|
+
---
|
|
779
|
+
|
|
780
|
+
## Symbol Configuration
|
|
781
|
+
|
|
782
|
+
### Built-in shapes
|
|
783
|
+
|
|
784
|
+
`circle`, `square`, `triangle`, `diamond`, `pentagon`, `plus`, `cross`, `asterisk`
|
|
785
|
+
|
|
786
|
+
```swift
|
|
787
|
+
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
788
|
+
.symbol(.diamond)
|
|
789
|
+
.symbolSize(80)
|
|
790
|
+
```
|
|
791
|
+
|
|
792
|
+
### Data-driven symbol encoding
|
|
793
|
+
|
|
794
|
+
```swift
|
|
795
|
+
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
796
|
+
.symbol(by: .value("Category", item.category))
|
|
797
|
+
```
|
|
798
|
+
|
|
799
|
+
### Custom symbol view
|
|
800
|
+
|
|
801
|
+
```swift
|
|
802
|
+
PointMark(x: .value("X", item.x), y: .value("Y", item.y))
|
|
803
|
+
.symbol {
|
|
804
|
+
Image(systemName: "star.fill")
|
|
805
|
+
.font(.caption2)
|
|
806
|
+
}
|
|
807
|
+
```
|
|
808
|
+
|
|
809
|
+
---
|
|
810
|
+
|
|
811
|
+
## ChartProxy and Coordinate Conversion
|
|
812
|
+
|
|
813
|
+
Use `chartOverlay` or `chartBackground` to access `ChartProxy`:
|
|
814
|
+
|
|
815
|
+
```swift
|
|
816
|
+
.chartOverlay { proxy in
|
|
817
|
+
GeometryReader { geometry in
|
|
818
|
+
Rectangle()
|
|
819
|
+
.fill(.clear)
|
|
820
|
+
.contentShape(Rectangle())
|
|
821
|
+
.gesture(
|
|
822
|
+
DragGesture()
|
|
823
|
+
.onChanged { value in
|
|
824
|
+
let origin = geometry[proxy.plotAreaFrame].origin
|
|
825
|
+
let location = CGPoint(
|
|
826
|
+
x: value.location.x - origin.x,
|
|
827
|
+
y: value.location.y - origin.y
|
|
828
|
+
)
|
|
829
|
+
if let date: Date = proxy.value(atX: location.x) {
|
|
830
|
+
selectedDate = date
|
|
831
|
+
}
|
|
832
|
+
}
|
|
833
|
+
)
|
|
834
|
+
}
|
|
835
|
+
}
|
|
836
|
+
```
|
|
837
|
+
|
|
838
|
+
### Key ChartProxy methods
|
|
839
|
+
|
|
840
|
+
| Method | Purpose |
|
|
841
|
+
|---|---|
|
|
842
|
+
| `position(forX:)` | Data value to screen x-coordinate |
|
|
843
|
+
| `position(forY:)` | Data value to screen y-coordinate |
|
|
844
|
+
| `value(atX:as:)` | Screen x-coordinate to data value |
|
|
845
|
+
| `value(atY:as:)` | Screen y-coordinate to data value |
|
|
846
|
+
| `plotAreaSize` | Size of the plot area |
|
|
847
|
+
| `plotAreaFrame` | Anchor for the plot area frame |
|
|
848
|
+
|
|
849
|
+
---
|
|
850
|
+
|
|
851
|
+
## Quick Reference: Chart View Modifiers
|
|
852
|
+
|
|
853
|
+
### Axes
|
|
854
|
+
- `chartXAxis(_:)` / `chartXAxis(content:)`
|
|
855
|
+
- `chartYAxis(_:)` / `chartYAxis(content:)`
|
|
856
|
+
- `chartXAxisLabel(...)` / `chartYAxisLabel(...)`
|
|
857
|
+
- `chartXAxisStyle(content:)` / `chartYAxisStyle(content:)`
|
|
858
|
+
|
|
859
|
+
### Scales
|
|
860
|
+
- `chartXScale(domain:range:type:)` and variants
|
|
861
|
+
- `chartYScale(domain:range:type:)` and variants
|
|
862
|
+
- `chartForegroundStyleScale(_:)` -- custom color mapping
|
|
863
|
+
|
|
864
|
+
### Legend
|
|
865
|
+
- `chartLegend(_:)` -- visibility
|
|
866
|
+
- `chartLegend(position:alignment:spacing:)` -- positioning
|
|
867
|
+
- `chartLegend(position:alignment:spacing:content:)` -- custom content
|
|
868
|
+
|
|
869
|
+
### Selection (iOS 17+)
|
|
870
|
+
- `chartXSelection(value:)` / `chartXSelection(range:)`
|
|
871
|
+
- `chartYSelection(value:)` / `chartYSelection(range:)`
|
|
872
|
+
- `chartAngleSelection(value:)` -- for `SectorMark`
|
|
873
|
+
|
|
874
|
+
### Scrolling (iOS 17+)
|
|
875
|
+
- `chartScrollableAxes(_:)`
|
|
876
|
+
- `chartXVisibleDomain(length:)` / `chartYVisibleDomain(length:)`
|
|
877
|
+
- `chartScrollPosition(initialX:)` / `chartScrollPosition(x:)`
|
|
878
|
+
- `chartScrollTargetBehavior(_:)`
|
|
879
|
+
|
|
880
|
+
### Overlay and Background
|
|
881
|
+
- `chartOverlay(alignment:content:)` -- with `ChartProxy`
|
|
882
|
+
- `chartBackground(alignment:content:)` -- with `ChartProxy`
|
|
883
|
+
- `chartPlotStyle(content:)` -- plot area styling
|
|
884
|
+
|
|
885
|
+
---
|
|
886
|
+
|
|
887
|
+
## Apple Documentation Links
|
|
888
|
+
|
|
889
|
+
- [Swift Charts](https://sosumi.ai/documentation/Charts)
|
|
890
|
+
- [Creating a chart using Swift Charts](https://sosumi.ai/documentation/Charts/Creating-a-chart-using-Swift-Charts)
|
|
891
|
+
- [BarMark](https://sosumi.ai/documentation/Charts/BarMark)
|
|
892
|
+
- [LineMark](https://sosumi.ai/documentation/Charts/LineMark)
|
|
893
|
+
- [SectorMark](https://sosumi.ai/documentation/Charts/SectorMark)
|
|
894
|
+
- [LinePlot](https://sosumi.ai/documentation/Charts/LinePlot)
|
|
895
|
+
- [AxisMarks](https://sosumi.ai/documentation/Charts/AxisMarks)
|