photokit
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
ChinesePhotoKit
PhotoKit
Modern patterns for photo picking, camera capture, image loading, and media permissions targeting iOS 26+ with Swift 6.3. Patterns are backward-compatible to iOS 16 unless noted. See references/photokit-patterns.md for complete picker recipes and references/camera-capture.md for AVCaptureSession patterns.
针对iOS 26+和Swift 6.3的照片选择、相机拍摄、图片加载及媒体权限处理的现代方案。除非特别说明,这些方案向后兼容至iOS 16。完整的选择器方案请查看references/photokit-patterns.md,AVCaptureSession相关方案请查看references/camera-capture.md。
Contents
目录
PhotosPicker (SwiftUI, iOS 16+)
PhotosPicker(SwiftUI,iOS 16+)
PhotosPickerUIImagePickerControllerPhotosPickerUIImagePickerControllerSingle Selection
单选择
swift
import SwiftUI
import PhotosUI
struct SinglePhotoPicker: View {
@State private var selectedItem: PhotosPickerItem?
@State private var selectedImage: Image?
var body: some View {
VStack {
if let selectedImage {
selectedImage
.resizable()
.scaledToFit()
.frame(maxHeight: 300)
}
PhotosPicker("Select Photo", selection: $selectedItem, matching: .images)
}
.onChange(of: selectedItem) { _, newItem in
Task {
if let data = try? await newItem?.loadTransferable(type: Data.self),
let uiImage = UIImage(data: data) {
selectedImage = Image(uiImage: uiImage)
}
}
}
}
}swift
import SwiftUI
import PhotosUI
struct SinglePhotoPicker: View {
@State private var selectedItem: PhotosPickerItem?
@State private var selectedImage: Image?
var body: some View {
VStack {
if let selectedImage {
selectedImage
.resizable()
.scaledToFit()
.frame(maxHeight: 300)
}
PhotosPicker("Select Photo", selection: $selectedItem, matching: .images)
}
.onChange(of: selectedItem) { _, newItem in
Task {
if let data = try? await newItem?.loadTransferable(type: Data.self),
let uiImage = UIImage(data: data) {
selectedImage = Image(uiImage: uiImage)
}
}
}
}
}Multi-Selection
多选择
swift
struct MultiPhotoPicker: View {
@State private var selectedItems: [PhotosPickerItem] = []
@State private var selectedImages: [Image] = []
var body: some View {
VStack {
ScrollView(.horizontal) {
HStack {
ForEach(selectedImages.indices, id: \.self) { index in
selectedImages[index]
.resizable()
.scaledToFill()
.frame(width: 100, height: 100)
.clipShape(RoundedRectangle(cornerRadius: 8))
}
}
}
PhotosPicker(
"Select Photos",
selection: $selectedItems,
maxSelectionCount: 5,
matching: .images
)
}
.onChange(of: selectedItems) { _, newItems in
Task {
selectedImages = []
for item in newItems {
if let data = try? await item.loadTransferable(type: Data.self),
let uiImage = UIImage(data: data) {
selectedImages.append(Image(uiImage: uiImage))
}
}
}
}
}
}swift
struct MultiPhotoPicker: View {
@State private var selectedItems: [PhotosPickerItem] = []
@State private var selectedImages: [Image] = []
var body: some View {
VStack {
ScrollView(.horizontal) {
HStack {
ForEach(selectedImages.indices, id: \.self) { index in
selectedImages[index]
.resizable()
.scaledToFill()
.frame(width: 100, height: 100)
.clipShape(RoundedRectangle(cornerRadius: 8))
}
}
}
PhotosPicker(
"Select Photos",
selection: $selectedItems,
maxSelectionCount: 5,
matching: .images
)
}
.onChange(of: selectedItems) { _, newItems in
Task {
selectedImages = []
for item in newItems {
if let data = try? await item.loadTransferable(type: Data.self),
let uiImage = UIImage(data: data) {
selectedImages.append(Image(uiImage: uiImage))
}
}
}
}
}
}Media Type Filtering
媒体类型过滤
Filter with composites to restrict selectable media:
PHPickerFilterswift
// Images only
PhotosPicker(selection: $items, matching: .images)
// Videos only
PhotosPicker(selection: $items, matching: .videos)
// Live Photos only
PhotosPicker(selection: $items, matching: .livePhotos)
// Screenshots only
PhotosPicker(selection: $items, matching: .screenshots)
// Images and videos combined
PhotosPicker(selection: $items, matching: .any(of: [.images, .videos]))
// Images excluding screenshots
PhotosPicker(selection: $items, matching: .all(of: [.images, .not(.screenshots)]))使用组合来限制可选择的媒体类型:
PHPickerFilterswift
// 仅图片
PhotosPicker(selection: $items, matching: .images)
// 仅视频
PhotosPicker(selection: $items, matching: .videos)
// 仅实况照片
PhotosPicker(selection: $items, matching: .livePhotos)
// 仅截图
PhotosPicker(selection: $items, matching: .screenshots)
// 图片和视频
PhotosPicker(selection: $items, matching: .any(of: [.images, .videos]))
// 图片(排除截图)
PhotosPicker(selection: $items, matching: .all(of: [.images, .not(.screenshots)]))Loading Selected Items with Transferable
使用Transferable加载选中项
PhotosPickerItemloadTransferable(type:)Transferableswift
struct PickedImage: Transferable {
let data: Data
let image: Image
static var transferRepresentation: some TransferRepresentation {
DataRepresentation(importedContentType: .image) { data in
guard let uiImage = UIImage(data: data) else {
throw TransferError.importFailed
}
return PickedImage(data: data, image: Image(uiImage: uiImage))
}
}
}
enum TransferError: Error {
case importFailed
}
// Usage
if let picked = try? await item.loadTransferable(type: PickedImage.self) {
selectedImage = picked.image
}Always load in a to avoid blocking the main thread. Handle returns and thrown errors -- the user may select a format that cannot be decoded.
TasknilPhotosPickerItemloadTransferable(type:)Transferableswift
struct PickedImage: Transferable {
let data: Data
let image: Image
static var transferRepresentation: some TransferRepresentation {
DataRepresentation(importedContentType: .image) { data in
guard let uiImage = UIImage(data: data) else {
throw TransferError.importFailed
}
return PickedImage(data: data, image: Image(uiImage: uiImage))
}
}
}
enum TransferError: Error {
case importFailed
}
// 使用示例
if let picked = try? await item.loadTransferable(type: PickedImage.self) {
selectedImage = picked.image
}务必在中加载内容,避免阻塞主线程。处理返回值和抛出的错误——用户可能选择了无法解码的格式。
TasknilPrivacy and Permissions
隐私与权限
Photo Library Access Levels
相册访问级别
iOS provides two access levels for the photo library. The system automatically presents the limited-library picker when an app requests access -- users choose which photos to share.
.readWrite| Access Level | Description | Info.plist Key |
|---|---|---|
| Add-only | Write photos to the library without reading | |
| Read-write | Full or limited read access plus write | |
PhotosPickeriOS为相册提供两种访问级别。当应用请求权限时,系统会自动弹出受限相册选择器,用户可选择要共享的照片。
.readWrite| 访问级别 | 描述 | Info.plist键 |
|---|---|---|
| 仅添加 | 仅向相册写入照片,无需读取权限 | |
| 读写 | 完整或受限读取权限,加上写入权限 | |
PhotosPickerChecking and Requesting Photo Library Permission
检查并请求相册权限
swift
import Photos
func requestPhotoLibraryAccess() async -> PHAuthorizationStatus {
let status = PHPhotoLibrary.authorizationStatus(for: .readWrite)
switch status {
case .notDetermined:
return await PHPhotoLibrary.requestAuthorization(for: .readWrite)
case .authorized, .limited:
return status
case .denied, .restricted:
return status
@unknown default:
return status
}
}swift
import Photos
func requestPhotoLibraryAccess() async -> PHAuthorizationStatus {
let status = PHPhotoLibrary.authorizationStatus(for: .readWrite)
switch status {
case .notDetermined:
return await PHPhotoLibrary.requestAuthorization(for: .readWrite)
case .authorized, .limited:
return status
case .denied, .restricted:
return status
@unknown default:
return status
}
}Camera Permission
相机权限
Add to Info.plist. Check and request access before configuring a capture session:
NSCameraUsageDescriptionswift
import AVFoundation
func requestCameraAccess() async -> Bool {
let status = AVCaptureDevice.authorizationStatus(for: .video)
switch status {
case .notDetermined:
return await AVCaptureDevice.requestAccess(for: .video)
case .authorized:
return true
case .denied, .restricted:
return false
@unknown default:
return false
}
}在Info.plist中添加。在配置拍摄会话前检查并请求权限:
NSCameraUsageDescriptionswift
import AVFoundation
func requestCameraAccess() async -> Bool {
let status = AVCaptureDevice.authorizationStatus(for: .video)
switch status {
case .notDetermined:
return await AVCaptureDevice.requestAccess(for: .video)
case .authorized:
return true
case .denied, .restricted:
return false
@unknown default:
return false
}
}Handling Denied Permissions
处理权限被拒绝的情况
When the user denies access, guide them to Settings. Never repeatedly prompt or hide functionality silently.
swift
struct PermissionDeniedView: View {
let message: String
@Environment(\.openURL) private var openURL
var body: some View {
ContentUnavailableView {
Label("Access Denied", systemImage: "lock.shield")
} description: {
Text(message)
} actions: {
Button("Open Settings") {
if let url = URL(string: UIApplication.openSettingsURLString) {
openURL(url)
}
}
}
}
}当用户拒绝权限时,引导他们前往设置。切勿反复弹出提示或隐藏功能而不告知用户。
swift
struct PermissionDeniedView: View {
let message: String
@Environment(\.openURL) private var openURL
var body: some View {
ContentUnavailableView {
Label("Access Denied", systemImage: "lock.shield")
} description: {
Text(message)
} actions: {
Button("Open Settings") {
if let url = URL(string: UIApplication.openSettingsURLString) {
openURL(url)
}
}
}
}
}Required Info.plist Keys
必填的Info.plist键
| Key | When Required |
|---|---|
| Reading photos from the library |
| Saving photos/videos to the library |
| Accessing the camera |
| Recording audio (video with sound) |
Omitting a required key causes a runtime crash when the permission dialog would appear.
| 键 | 必填场景 |
|---|---|
| 从相册读取照片时 |
| 向相册保存照片/视频时 |
| 访问相机时 |
| 录制音频(带声音的视频)时 |
如果遗漏必填键,当系统要弹出权限对话框时会导致运行时崩溃。
Camera Capture Basics
相机拍摄基础
Manage camera sessions in a dedicated model. The representable view only displays the preview. See references/camera-capture.md for complete patterns.
@Observable在专门的模型中管理相机会话,可表示视图仅用于显示预览。完整方案请查看references/camera-capture.md。
@ObservableMinimal Camera Manager
极简相机管理器
swift
import AVFoundation
@available(iOS 17.0, *)
@Observable
@MainActor
final class CameraManager {
let session = AVCaptureSession()
private let photoOutput = AVCapturePhotoOutput()
private var currentDevice: AVCaptureDevice?
var isRunning = false
var capturedImage: Data?
func configure() async {
guard await requestCameraAccess() else { return }
session.beginConfiguration()
session.sessionPreset = .photo
// Add camera input
guard let device = AVCaptureDevice.default(.builtInWideAngleCamera,
for: .video,
position: .back) else { return }
currentDevice = device
guard let input = try? AVCaptureDeviceInput(device: device),
session.canAddInput(input) else { return }
session.addInput(input)
// Add photo output
guard session.canAddOutput(photoOutput) else { return }
session.addOutput(photoOutput)
session.commitConfiguration()
}
func start() {
guard !session.isRunning else { return }
Task.detached { [session] in
session.startRunning()
}
isRunning = true
}
func stop() {
guard session.isRunning else { return }
Task.detached { [session] in
session.stopRunning()
}
isRunning = false
}
private func requestCameraAccess() async -> Bool {
let status = AVCaptureDevice.authorizationStatus(for: .video)
if status == .notDetermined {
return await AVCaptureDevice.requestAccess(for: .video)
}
return status == .authorized
}
}Start and stop on a background queue. The and methods are synchronous and block the calling thread.
AVCaptureSessionstartRunning()stopRunning()swift
import AVFoundation
@available(iOS 17.0, *)
@Observable
@MainActor
final class CameraManager {
let session = AVCaptureSession()
private let photoOutput = AVCapturePhotoOutput()
private var currentDevice: AVCaptureDevice?
var isRunning = false
var capturedImage: Data?
func configure() async {
guard await requestCameraAccess() else { return }
session.beginConfiguration()
session.sessionPreset = .photo
// 添加相机输入
guard let device = AVCaptureDevice.default(.builtInWideAngleCamera,
for: .video,
position: .back) else { return }
currentDevice = device
guard let input = try? AVCaptureDeviceInput(device: device),
session.canAddInput(input) else { return }
session.addInput(input)
// 添加照片输出
guard session.canAddOutput(photoOutput) else { return }
session.addOutput(photoOutput)
session.commitConfiguration()
}
func start() {
guard !session.isRunning else { return }
Task.detached { [session] in
session.startRunning()
}
isRunning = true
}
func stop() {
guard session.isRunning else { return }
Task.detached { [session] in
session.stopRunning()
}
isRunning = false
}
private func requestCameraAccess() async -> Bool {
let status = AVCaptureDevice.authorizationStatus(for: .video)
if status == .notDetermined {
return await AVCaptureDevice.requestAccess(for: .video)
}
return status == .authorized
}
}在后台队列中启动和停止。和方法是同步的,会阻塞调用线程。
AVCaptureSessionstartRunning()stopRunning()Camera Preview in SwiftUI
SwiftUI中的相机预览
Wrap in a . Override for automatic resizing:
AVCaptureVideoPreviewLayerUIViewRepresentablelayerClassswift
import SwiftUI
import AVFoundation
struct CameraPreview: UIViewRepresentable {
let session: AVCaptureSession
func makeUIView(context: Context) -> PreviewView {
let view = PreviewView()
view.previewLayer.session = session
view.previewLayer.videoGravity = .resizeAspectFill
return view
}
func updateUIView(_ uiView: PreviewView, context: Context) {
if uiView.previewLayer.session !== session {
uiView.previewLayer.session = session
}
}
}
final class PreviewView: UIView {
override class var layerClass: AnyClass { AVCaptureVideoPreviewLayer.self }
var previewLayer: AVCaptureVideoPreviewLayer { layer as! AVCaptureVideoPreviewLayer }
}将包装在中。重写以实现自动调整大小:
AVCaptureVideoPreviewLayerUIViewRepresentablelayerClassswift
import SwiftUI
import AVFoundation
struct CameraPreview: UIViewRepresentable {
let session: AVCaptureSession
func makeUIView(context: Context) -> PreviewView {
let view = PreviewView()
view.previewLayer.session = session
view.previewLayer.videoGravity = .resizeAspectFill
return view
}
func updateUIView(_ uiView: PreviewView, context: Context) {
if uiView.previewLayer.session !== session {
uiView.previewLayer.session = session
}
}
}
final class PreviewView: UIView {
override class var layerClass: AnyClass { AVCaptureVideoPreviewLayer.self }
var previewLayer: AVCaptureVideoPreviewLayer { layer as! AVCaptureVideoPreviewLayer }
}Using the Camera in a View
在视图中使用相机
swift
struct CameraScreen: View {
@State private var cameraManager = CameraManager()
var body: some View {
ZStack(alignment: .bottom) {
CameraPreview(session: cameraManager.session)
.ignoresSafeArea()
Button {
// Capture photo -- see references/camera-capture.md
} label: {
Circle()
.fill(.white)
.frame(width: 72, height: 72)
.overlay(Circle().stroke(.gray, lineWidth: 3))
}
.padding(.bottom, 32)
}
.task {
await cameraManager.configure()
cameraManager.start()
}
.onDisappear {
cameraManager.stop()
}
}
}Always call in . A running capture session holds the camera exclusively and drains battery.
stop()onDisappearswift
struct CameraScreen: View {
@State private var cameraManager = CameraManager()
var body: some View {
ZStack(alignment: .bottom) {
CameraPreview(session: cameraManager.session)
.ignoresSafeArea()
Button {
// 拍摄照片——请查看references/camera-capture.md
} label: {
Circle()
.fill(.white)
.frame(width: 72, height: 72)
.overlay(Circle().stroke(.gray, lineWidth: 3))
}
.padding(.bottom, 32)
}
.task {
await cameraManager.configure()
cameraManager.start()
}
.onDisappear {
cameraManager.stop()
}
}
}务必在中调用。运行中的拍摄会话会独占相机,并且消耗电量。
onDisappearstop()Image Loading and Display
图片加载与显示
AsyncImage for Remote Images
远程图片使用AsyncImage
swift
AsyncImage(url: imageURL) { phase in
switch phase {
case .empty:
ProgressView()
case .success(let image):
image
.resizable()
.scaledToFill()
case .failure:
Image(systemName: "photo")
.foregroundStyle(.secondary)
@unknown default:
EmptyView()
}
}
.frame(width: 200, height: 200)
.clipShape(RoundedRectangle(cornerRadius: 12))AsyncImageURLCacheswift
AsyncImage(url: imageURL) { phase in
switch phase {
case .empty:
ProgressView()
case .success(let image):
image
.resizable()
.scaledToFill()
case .failure:
Image(systemName: "photo")
.foregroundStyle(.secondary)
@unknown default:
EmptyView()
}
}
.frame(width: 200, height: 200)
.clipShape(RoundedRectangle(cornerRadius: 12))AsyncImageURLCacheDownsampling Large Images
大图片降采样
Load full-resolution photos from the library into a display-sized to avoid memory spikes. A 48MP photo can consume over 200 MB uncompressed.
CGImageswift
import ImageIO
import UIKit
func downsample(data: Data, to pointSize: CGSize, scale: CGFloat = UITraitCollection.current.displayScale) -> UIImage? {
let maxDimensionInPixels = max(pointSize.width, pointSize.height) * scale
let options: [CFString: Any] = [
kCGImageSourceCreateThumbnailFromImageAlways: true,
kCGImageSourceShouldCacheImmediately: true,
kCGImageSourceCreateThumbnailWithTransform: true,
kCGImageSourceThumbnailMaxPixelSize: maxDimensionInPixels
]
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, options as CFDictionary) else {
return nil
}
return UIImage(cgImage: cgImage)
}Use this whenever displaying user-selected photos in lists, grids, or thumbnails. Pass the raw from directly to the downsampler before creating a .
DataPhotosPickerItemUIImage将相册中的全分辨率照片加载为适合显示的,避免内存峰值。一张4800万像素的照片未压缩时会占用超过200MB内存。
CGImageswift
import ImageIO
import UIKit
func downsample(data: Data, to pointSize: CGSize, scale: CGFloat = UITraitCollection.current.displayScale) -> UIImage? {
let maxDimensionInPixels = max(pointSize.width, pointSize.height) * scale
let options: [CFString: Any] = [
kCGImageSourceCreateThumbnailFromImageAlways: true,
kCGImageSourceShouldCacheImmediately: true,
kCGImageSourceCreateThumbnailWithTransform: true,
kCGImageSourceThumbnailMaxPixelSize: maxDimensionInPixels
]
guard let source = CGImageSourceCreateWithData(data as CFData, nil),
let cgImage = CGImageSourceCreateThumbnailAtIndex(source, 0, options as CFDictionary) else {
return nil
}
return UIImage(cgImage: cgImage)
}在列表、网格或缩略图中显示用户选择的照片时,请使用此方法。直接将的原始传入降采样器,再创建。
PhotosPickerItemDataUIImageImage Rendering Modes
图片渲染模式
swift
// Original: display the image as-is with its original colors
Image("photo")
.renderingMode(.original)
// Template: treat the image as a mask, colored by foregroundStyle
Image(systemName: "heart.fill")
.renderingMode(.template)
.foregroundStyle(.red)Use for photos and artwork. Use for icons that should adopt the current tint color.
.original.templateswift
// 原始模式:按图片原样显示,保留原始颜色
Image("photo")
.renderingMode(.original)
// 模板模式:将图片视为遮罩,使用前景色填充
Image(systemName: "heart.fill")
.renderingMode(.template)
.foregroundStyle(.red)照片和艺术作品使用模式。需要适配当前色调颜色的图标使用模式。
.original.templateCommon Mistakes
常见误区
DON'T: Use for photo picking.
DO: Use (SwiftUI) or (UIKit).
Why: is legacy API with limited functionality. runs out-of-process, supports multi-selection, and requires no library permission for browsing.
UIImagePickerControllerPhotosPickerPHPickerViewControllerUIImagePickerControllerPhotosPickerDON'T: Request full photo library access when you only need the user to pick photos.
DO: Use which requires no permission, or request and let the system handle limited access.
Why: Full access is unnecessary for most pick-and-use workflows. The system's limited-library picker respects user privacy and still grants access to selected items.
PhotosPicker.readWriteDON'T: Load full-resolution images into memory for thumbnails.
DO: Use with to downsample. A 48MP image is over 200 MB uncompressed.
CGImageSourcekCGImageSourceThumbnailMaxPixelSizeDON'T: Block the main thread loading data.
DO: Use in a .
PhotosPickerItemasync loadTransferable(type:)TaskDON'T: Forget to stop when the view disappears.
DO: Call in or .
AVCaptureSessionsession.stopRunning()onDisappeardismantleUIViewDON'T: Assume camera access is granted without checking.
DO: Check and handle /.
AVCaptureDevice.authorizationStatus(for: .video).denied.restrictedDON'T: Call on the main thread.
DO: Dispatch to a background thread with or a dedicated serial queue.
Why: is a synchronous blocking call that can take hundreds of milliseconds while the hardware initializes.
session.startRunning()Task.detachedstartRunning()DON'T: Create inside a .
DO: Own the session in a separate model.
AVCaptureSessionUIViewRepresentable@Observable不要: 使用进行照片选择。
要: 使用(SwiftUI)或(UIKit)。
原因: 是遗留API,功能有限。在进程外运行,支持多选,浏览时无需相册权限。
UIImagePickerControllerPhotosPickerPHPickerViewControllerUIImagePickerControllerPhotosPicker不要: 仅需用户选择照片时请求完整相册访问权限。
要: 使用无需权限的,或请求权限并让系统处理受限访问。
原因: 大多数选择即用的工作流无需完整访问权限。系统的受限相册选择器尊重用户隐私,同时仍会授予对选中项的访问权限。
PhotosPicker.readWrite不要: 将全分辨率图片加载到内存中用作缩略图。
要: 使用并设置进行降采样。一张4800万像素的图片未压缩时超过200MB。
CGImageSourcekCGImageSourceThumbnailMaxPixelSize不要: 加载数据时阻塞主线程。
要: 在中使用。
PhotosPickerItemTaskasync loadTransferable(type:)不要: 视图消失时忘记停止。
要: 在或中调用。
AVCaptureSessiononDisappeardismantleUIViewsession.stopRunning()不要: 未检查权限就假设相机访问已被授予。
要: 检查并处理/状态。
AVCaptureDevice.authorizationStatus(for: .video).denied.restricted不要: 在主线程调用。
要: 使用或专用串行队列调度到后台线程。
原因: 是同步阻塞调用,硬件初始化时可能需要数百毫秒。
session.startRunning()Task.detachedstartRunning()不要: 在内部创建。
要: 在单独的模型中管理会话。
UIViewRepresentableAVCaptureSession@ObservableReview Checklist
评审检查清单
- used instead of deprecated
PhotosPickerUIImagePickerController - Privacy descriptions in Info.plist for camera/photo library
- Loading states handled for async image/video loading
- Large images downsampled with before display
CGImageSource - Camera session started on background thread; stopped in
onDisappear - Permission denial handled with Settings deep link
- owned by model, not created inside
AVCaptureSessionUIViewRepresentable - Media asset types and picker results are across concurrency boundaries
Sendable
- 使用替代已废弃的
PhotosPickerUIImagePickerController - Info.plist中包含相机/相册的隐私描述
- 异步加载图片/视频时处理加载状态
- 大图片在显示前使用进行降采样
CGImageSource - 相机会话在后台线程启动,在中停止
onDisappear - 权限被拒绝时提供设置跳转链接
- 由模型管理,而非在
AVCaptureSession内部创建UIViewRepresentable - 媒体资源类型和选择器结果在并发边界上是类型
Sendable
References
参考资料
- references/photokit-patterns.md — Picker patterns, media loading, HEIC handling
- references/camera-capture.md — AVCaptureSession, photo/video capture, QR scanning
- references/image-loading-caching.md — AsyncImage, caching, downsampling
- references/av-playback.md — AVPlayer, streaming, audio
- references/photokit-patterns.md — 选择器方案、媒体加载、HEIC格式处理
- references/camera-capture.md — AVCaptureSession、照片/视频拍摄、二维码扫描
- references/image-loading-caching.md — AsyncImage、缓存、降采样
- references/av-playback.md — AVPlayer、流媒体、音频