--- url: /doc_v2/guide/AppIntent.md --- # AppIntent `AppIntentManager` is used to **register and manage AppIntents** in the **Scripting** app. It serves as the core mechanism for executing script logic behind controls in **Widgets**, **Live Activities**, and **ControlWidgets**. All `AppIntent`s **must** be defined in the `app_intents.tsx` file. When an intent is executed, the script runs in the `"app_intents"` environment (`Script.env === "app_intents"`). Once registered, these intents can be triggered by **Button** and **Toggle** controls within Widgets, Live Activities, or ControlWidgets, allowing users to define interactive behavior via script. *** ## 1. Type Definitions ### `AppIntent` Represents a concrete intent instance with parameters and metadata. | Property | Type | Description | | ---------- | ------------------- | -------------------------------------------------------------------------- | | `script` | `string` | The internal script path. Automatically generated by the system. | | `name` | `string` | The name of the AppIntent. Must be unique. | | `protocol` | `AppIntentProtocol` | The protocol the intent conforms to (e.g., general, audio, Live Activity). | | `params` | `T` | The parameters to be passed when the intent is executed. | *** ### `AppIntentFactory` A **factory function** that creates an `AppIntent` instance with specified parameters. ```ts type AppIntentFactory = (params: T) => AppIntent ``` *** ### `AppIntentPerform` A function that handles intent execution logic asynchronously. ```ts type AppIntentPerform = (params: T) => Promise ``` *** ### `AppIntentProtocol` An enumeration that defines the behavior type of the intent. | Enum Value | Description | | -------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | `AppIntent` (0) | A general-purpose AppIntent for typical operations. | | `AudioPlaybackIntent` (1) | An intent that plays, pauses, or otherwise modifies audio playback. | | `AudioRecordingIntent` (2) | _(iOS 18.0+)_ An intent that starts, stops, or modifies audio recording. **Note**: On iOS/iPadOS, when using the `AudioRecordingIntent` protocol, you must start a **Live Activity** at the beginning of the recording and keep it active for the entire session. If you don't, the recording will be automatically stopped. | | `LiveActivityIntent` (3) | An intent that starts, pauses, or modifies a Live Activity. | *** ## 2. `AppIntentManager` Class ### `AppIntentManager.register(options): AppIntentFactory` Registers a new `AppIntent` by specifying its name, protocol, and perform logic. When a control (e.g., Button or Toggle) triggers the intent, the associated `perform` function is called. ```ts static register(options: { name: string; protocol: AppIntentProtocol; perform: AppIntentPerform; }): AppIntentFactory ``` #### Parameters: | Property | Type | Description | | ---------- | --------------------- | ------------------------------------------------------------------------------------------------------------------ | | `name` | `string` | A unique identifier for the AppIntent. | | `protocol` | `AppIntentProtocol` | The protocol this intent implements. | | `perform` | `AppIntentPerform` | The asynchronous function executed when the intent is triggered. The `params` argument is passed from the control. | #### Returns: - **`AppIntentFactory`**: A factory function that creates an `AppIntent` instance with the specified parameters. #### Example: ```tsx // app_intents.tsx export const ToggleDoorIntent = AppIntentManager.register({ name: "ToggleDoorIntent", protocol: AppIntentProtocol.AppIntent, perform: async ({ id, newState }: { id: string; newState: boolean }) => { // Custom logic: toggle the door state await setDoorState(id, newState) // Notify UI to refresh toggle state ControlWidget.reloadToggles() } }) ``` In a control view file (e.g., `control_widget_toggle.tsx`): ```tsx ControlWidget.present( ) ``` In a widget file (`widget.tsx`): ```tsx ``` *** ## 3. Execution Environment All AppIntents registered via `AppIntentManager` are executed in the `"app_intents"` environment. This allows safe use of APIs suitable for background execution, such as: - Fetching data from the network - Controlling Live Activities - Triggering control view refreshes *** ## 4. Best Practices 1. **Centralized Definitions**: All AppIntents **must** be defined in `app_intents.tsx` for discoverability and maintainability. 2. **Strong Typing**: Define explicit parameter types `T` for both `perform` and control usage to benefit from type checking and autocomplete. 3. **Choose the Right Protocol**: - General operation → `AppIntent` - Audio playback → `AudioPlaybackIntent` - Audio recording → `AudioRecordingIntent` _(requires iOS 18+, with Live Activity)_ - Live Activity control → `LiveActivityIntent` 4. **Trigger UI Updates**: If the intent modifies a UI state (e.g., toggle), call: - `ControlWidget.reloadButtons()` - `ControlWidget.reloadToggles()` - `Widget.reloadAll()` depending on where the control is hosted. --- url: /doc_v2/guide/Assistant/Assistant Conversation APIs.md --- # Assistant Conversation APIs The Conversation APIs are used to **start, control, and present a system-hosted Assistant chat session**. A conversation corresponds to a **fully managed chat page**, where Scripting handles the UI, streaming output, provider selection, and message lifecycle. Key differences from other Assistant APIs: - Conversation APIs are designed for **interactive chat experiences** - UI, streaming, and message handling are managed by the system - Developers control **when the conversation starts, ends, and is shown** *** ## Conversation Lifecycle A typical conversation follows this lifecycle: 1. `startConversation` — create a conversation (optionally auto-start) 2. `present` — display the Assistant chat page 3. User interacts with the Assistant 4. `dismiss` — temporarily hide the chat page (conversation continues) 5. `present` — show the same conversation again 6. `stopConversation` — terminate the conversation and release resources Important rules: - **Only one active conversation can exist at a time** - Calling `startConversation` while a conversation is active throws an error - Calling `stopConversation` automatically calls `dismiss` *** ## startConversation ### API Definition ```ts function startConversation(options: { message: string images?: UIImage[] autoStart?: boolean systemPrompt?: string modelId?: string provider?: Provider }): Promise ``` *** ### Parameters #### options.message - Type: `string` - Required - The **initial user message** of the conversation - Equivalent to the first user input in the chat UI *** #### options.images (optional) - Type: `UIImage[]` - Sent together with the initial message - Common use cases: - Image analysis - Starting a conversation from a photo or screenshot *** #### options.autoStart (optional) - Type: `boolean` - Default: `false` Behavior: - `true` - The assistant immediately starts generating a reply - `false` - The conversation is created but not sent automatically - Typically used when the user should press “Send” manually *** #### options.systemPrompt (optional) - Type: `string` Behavior: - If omitted: - The built-in Scripting Assistant system prompt is used - Assistant Tools are available - If provided: - Fully replaces the default system prompt - **Assistant Tools are disabled** Typical use cases: - Creating a highly customized chat role - Running the model without any tool access *** #### options.modelId (optional) - Type: `string` - Specifies the model to use for this conversation - Users may still change the model in the chat UI (if allowed) *** #### options.provider (optional) - Type: `Provider` - Specifies the default provider for the conversation - Users may change the provider in the chat UI (if allowed) *** ### Return Value ```ts Promise ``` - Resolves when the conversation is successfully created - Rejects if a conversation already exists *** ## present ### API Definition ```ts function present(): Promise ``` *** ### Behavior - Presents the Assistant chat page for the current conversation - If the page is already presented, calling this has no effect - Can be called: - After `startConversation` - After `dismiss` to re-present the same conversation *** ### Return Value ```ts Promise ``` - Resolves when the chat page is dismissed by the user *** ## dismiss ### API Definition ```ts function dismiss(): Promise ``` *** ### Behavior - Dismisses the Assistant chat page - **Does not stop the conversation** - Conversation state and history are preserved Typical use cases: - Temporarily hiding the chat UI - Navigating to another page or task *** ### Return Value ```ts Promise ``` *** ## stopConversation ### API Definition ```ts function stopConversation(): Promise ``` *** ### Behavior - Fully terminates the current conversation - Automatically calls `dismiss` - Cleans up conversation state and resources - After calling this, a new conversation may be started *** ### Return Value ```ts Promise ``` *** ## Conversation State Flags ### Assistant.isAvailable ```ts const isAvailable: boolean ``` - Indicates whether the current user has access to the Assistant - If `false`, all Conversation APIs are unavailable *** ### Assistant.isPresented ```ts const isPresented: boolean ``` - Indicates whether the Assistant chat page is currently presented *** ### Assistant.hasActiveConversation ```ts const hasActiveConversation: boolean ``` - Indicates whether there is an active conversation - Commonly used to guard against duplicate `startConversation` calls *** ## Examples ### Example 1: Typical usage ```ts await Assistant.startConversation({ message: "Help me summarize this article.", autoStart: true }) await Assistant.present() ``` *** ### Example 2: Create a conversation without auto-sending ```ts await Assistant.startConversation({ message: "Let's discuss system architecture design.", autoStart: false }) await Assistant.present() // User manually presses Send in the UI ``` *** ### Example 3: Dismiss and re-present the same conversation ```ts await Assistant.startConversation({ message: "Analyze this image.", images: [image], autoStart: true }) await Assistant.present() await Assistant.dismiss() // Later, re-present the same conversation await Assistant.present() ``` *** ### Example 4: Stop the current conversation and start a new one ```ts if (Assistant.hasActiveConversation) { await Assistant.stopConversation() } await Assistant.startConversation({ message: "Start a new topic.", autoStart: true }) await Assistant.present() ``` *** ## Best Practices - Treat Conversation APIs as a **managed chat UI** - Do not mix Conversation APIs with `requestStreaming` in the same flow - Always check `hasActiveConversation` before calling `startConversation` - For one-shot or data-oriented tasks, prefer: - `requestStructuredData` - `requestStreaming` - Use Conversation APIs when continuous user interaction is required *** ## Design Boundaries - Conversation APIs are not suitable for headless or background tasks - Not intended for fully automated workflows - Not ideal when you need strict control over prompts, tokens, or output format --- url: /doc_v2/guide/Assistant/Assistant Quick Start.md --- # Assistant Quick Start The Assistant API in Scripting provides three distinct capabilities, each designed for a different type of use case: **structured data**, **streaming output**, and **interactive conversations**. Before choosing an API, first decide **what kind of result you need**. *** ## Assistant API Overview | Category | Main APIs | Best For | | ---------------- | ---------------------------------------------------------------- | -------------------------------- | | Structured Data | `requestStructuredData` | Extracting predictable JSON data | | Streaming Output | `requestStreaming` | Real-time text generation | | Conversations | `startConversation` / `present` / `dismiss` / `stopConversation` | Fully managed chat UI | *** ## requestStructuredData **Purpose** Requests **strictly structured JSON output** that conforms to a provided schema. **Best suited for** - Parsing receipts, invoices, and bills - Extracting fields from natural language - Generating configuration or rule objects - Any output that must be consumed by program logic **Key characteristics** - Stable and predictable output - No streaming or incremental updates - Ideal for background or headless scenarios **In one sentence** > If you need **data**, use `requestStructuredData`. *** ## requestStreaming **Purpose** Requests **streaming output**, allowing you to receive content incrementally as the model generates it. **Best suited for** - Typing-effect UI - Long-form content generation - Low-latency user feedback **Key characteristics** - Emits text, reasoning, and usage chunks - Can be rendered progressively - Output is not guaranteed to be structured **In one sentence** > If you need **real-time output**, use `requestStreaming`. *** ## Conversation APIs **Related methods** - `startConversation` - `present` - `dismiss` - `stopConversation` **Purpose** Creates and presents a **system-hosted Assistant chat experience**. **Best suited for** - ChatGPT-style interactions - Multi-turn conversations - Scenarios where the system manages UI, streaming, and provider switching **Key characteristics** - Built-in chat UI - Streaming handled automatically - Only one active conversation at a time **In one sentence** > If you need a **full chat experience**, use the Conversation APIs. *** ## How to Choose the Right API ### Common Scenarios - **Parse a receipt →** `requestStructuredData` - **Show AI writing text live →** `requestStreaming` - **Open a chat interface for users →** Conversation APIs - **No UI, just results →** `requestStructuredData` or `requestStreaming` - **Let the system manage the chat UI →** Conversation APIs *** ## Minimal Examples ### Structured Data ```ts const result = await Assistant.requestStructuredData(...) ``` *** ### Streaming Output ```ts const stream = await Assistant.requestStreaming(...) for await (const chunk of stream) { // handle chunk } ``` *** ### Conversation ```ts await Assistant.startConversation({ message: "Hello", autoStart: true }) await Assistant.present() ``` *** ## Usage Tips - Do not mix Conversation APIs with `requestStreaming` in the same flow - Prefer `requestStructuredData` whenever output must be consumed as data - Use streaming or conversations for presentation-focused scenarios *** ## Next Steps For deeper details, refer to: - `requestStructuredData` – detailed schema-driven data extraction - `requestStreaming` – streaming behavior and chunk handling - Conversation APIs – lifecycle and interaction patterns --- url: /doc_v2/guide/Assistant/requestStreaming.md --- # requestStreaming `requestStreaming` requests a **streaming response** from the Assistant. Instead of returning a complete result at once, the Assistant emits **chunks incrementally** as the model generates output. This enables: - Real-time UI updates (typing effect) - Low-latency handling of long responses - Progressive rendering of results - Streaming logs and intermediate output handling The API returns a **`ReadableStream`**, which can be consumed using `for await ... of`. *** ## API Definition ```ts function requestStreaming(options: { systemPrompt?: string | null messages: MessageItem | MessageItem[] provider?: Provider modelId?: string }): Promise> ``` *** ## Parameters ### options.systemPrompt (optional) - Type: `string | null` - Specifies the system prompt for this request. - If omitted: - The default Assistant system prompt is used. - If provided: - It **fully replaces** the default system prompt. - Assistant Tools are **not available**. Typical use cases: - Defining a strict role (e.g. reviewer, translator, summarizer) - Enforcing output tone or behavior - Running the model without built-in tools *** ### options.messages - Type: `MessageItem | MessageItem[]` - Required - Represents the conversation context sent to the model. #### MessageItem ```ts type MessageItem = { role: "user" | "assistant" content: MessageContent | MessageContent[] } ``` - `role` - `"user"`: user input - `"assistant"`: previous assistant messages (for context) *** ### MessageContent Types #### Text ```ts type MessageTextContent = | string | { type: "text"; content: string } ``` *** #### Image ```ts type MessageImageContent = { type: "image" content: string // data:image/...;base64,... } ``` *** #### Document ```ts type MessageDocumentContent = { type: "document" content: { mediaType: string data: string // base64 } } ``` *** ### options.provider (optional) - Type: `Provider` - Specifies the AI provider. - If omitted, the currently configured default provider is used. - Supported values: - `"openai"` - `"gemini"` - `"anthropic"` - `"deepseek"` - `"openrouter"` - `{ custom: string }` *** ### options.modelId (optional) - Type: `string` - Specifies the model ID. - Must match a model actually supported by the selected provider. - If omitted, the provider’s default model is used. *** ## Return Value ```ts Promise> ``` Once resolved, you receive a stream that can be consumed asynchronously. *** ## StreamChunk Types The stream may emit the following chunk types. *** ### StreamTextChunk ```ts type StreamTextChunk = { type: "text" content: string } ``` - Represents user-visible generated text. - Multiple chunks concatenated form the final response. *** ### StreamReasoningChunk ```ts type StreamReasoningChunk = { type: "reasoning" content: string } ``` - Represents intermediate reasoning produced by the model. - Availability and granularity depend on the provider and model. *** ### StreamUsageChunk ```ts type StreamUsageChunk = { type: "usage" content: { totalCost: number | null cacheReadTokens: number | null cacheWriteTokens: number | null inputTokens: number outputTokens: number } } ``` Notes: - Typically emitted once near the end of the stream. - Some providers may omit certain fields. - `totalCost` may be `null` if the provider does not expose pricing data. *** ## Examples ### Example 1: Basic streaming request ```ts const stream = await Assistant.requestStreaming({ messages: { role: "user", content: "Tell me a short science fiction story." }, provider: "openai" }) let result = "" for await (const chunk of stream) { if (chunk.type === "text") { result += chunk.content console.log(chunk.content) } } ``` *** ### Example 2: Handling text, reasoning, and usage separately ```ts const stream = await Assistant.requestStreaming({ systemPrompt: "You are a precise technical writing assistant.", messages: [ { role: "user", content: "Explain what HTTP/3 is." } ] }) let answer = "" let reasoningLog = "" let usage = null for await (const chunk of stream) { switch (chunk.type) { case "text": answer += chunk.content break case "reasoning": reasoningLog += chunk.content break case "usage": usage = chunk.content break } } console.log(answer) console.log(usage) ``` *** ### Example 3: Streaming with document input ```ts const stream = await Assistant.requestStreaming({ messages: [ { role: "user", content: [ { type: "text", content: "Summarize the key points of this document." }, { type: "document", content: { mediaType: "application/pdf", data: "JVBERi0xLjQKJcfs..." } } ] } ], provider: "anthropic" }) for await (const chunk of stream) { if (chunk.type === "text") { console.log(chunk.content) } } ``` *** ## Usage Notes and Best Practices - Streams must be consumed **sequentially**; do not read concurrently. - For UI scenarios: - Render `text` chunks immediately. - Keep `reasoning` for debugging or developer modes. - Process `usage` after completion. - If you no longer need the output, stop consuming the stream to avoid unnecessary cost. - Not all providers/models emit `reasoning` or `usage`. - Do not assume a chunk represents a complete sentence; chunk sizes vary. --- url: /doc_v2/guide/Assistant/requestStructuredData.md --- # requestStructuredData `requestStructuredData` requests **structured JSON output** from the assistant that conforms to a provided JSON schema. This API is designed for workflows where you want a predictable, programmatically usable result rather than free-form text. Common use cases include: - Extracting structured fields from natural language - Parsing invoices, receipts, and tickets - Generating configuration objects - Normalizing data across different AI providers/models *** ## Supported JSON Schema Types Scripting defines a lightweight schema structure with three building blocks. ### Primitive ```ts type JSONSchemaPrimitive = { type: "string" | "number" | "boolean" required?: boolean description: string } ``` *** ### Object ```ts type JSONSchemaObject = { type: "object" properties: Record required?: boolean description: string } ``` *** ### Array ```ts type JSONSchemaArray = { type: "array" items: JSONSchemaType required?: boolean description: string } ``` *** ## API Signatures ### Without images ```ts function requestStructuredData( prompt: string, schema: JSONSchemaArray | JSONSchemaObject, options?: { provider: Provider modelId?: string } ): Promise ``` ### With images ```ts function requestStructuredData( prompt: string, images: string[], schema: JSONSchemaArray | JSONSchemaObject, options?: { provider: Provider modelId?: string } ): Promise ``` *** ## Parameters ### prompt - Type: `string` - Required - The instruction to the model describing what to extract or generate. - For best reliability, explicitly specify: - expected formats (e.g., ISO date) - currency rules - how to handle missing fields ### images (optional) - Type: `string[]` - Each item must be a **data URI**, e.g. `data:image/png;base64,...` - Not all providers/models support images. - Avoid passing too many images to reduce failure risk. ### schema - Type: `JSONSchemaArray | JSONSchemaObject` - Required - Defines the **only acceptable** JSON structure for the response. - Every field should have a clear `description` to guide the model. ### options.provider - Type: `Provider` - Optional (uses the default configured provider if omitted) - Supported: - `"openai" | "gemini" | "anthropic" | "deepseek" | "openrouter" | { custom: string }` ### options.modelId (optional) - Type: `string` - Must match a model actually supported by the chosen provider. - If omitted, Scripting uses the provider’s default model. *** ## Return Value ```ts Promise ``` - `R` is the generic type you provide. - The resolved value is expected to match your schema. - The promise rejects if the assistant cannot return a valid structured result. *** ## Examples ### Example 1: Parse a receipt/bill into line items (time + amount) This example asks the assistant to analyze a textual receipt and extract: - receipt time (`purchasedAt`) - line items (`items[]`) - item name - item time (if present; otherwise null) - amount - total amount ```ts type ReceiptItem = { name: string time: string | null amount: number } type ReceiptParsed = { purchasedAt: string | null currency: string | null items: ReceiptItem[] total: number | null } const receiptText = ` Star Coffee 2026-01-08 14:23 Latte (Large) $5.50 Blueberry Muffin $3.20 Tax $0.79 Total $9.49 ` const parsed = await Assistant.requestStructuredData( [ "Analyze the receipt text below and extract:", "- purchasedAt: the purchase date/time in ISO-8601 if possible", "- currency: currency code if you can infer it (otherwise null)", "- items: only actual purchasable items (exclude tax/total lines)", " - name: item name", " - time: item-level time if present, otherwise null", " - amount: numeric amount", "- total: numeric total if present, otherwise null", "", "Receipt:", receiptText ].join("\n"), { type: "object", description: "Parsed receipt content", properties: { purchasedAt: { type: "string", description: "Purchase date/time in ISO-8601 format if available, otherwise an empty string" }, currency: { type: "string", description: "Currency code like USD/EUR/CNY if inferable, otherwise an empty string" }, items: { type: "array", description: "Purchased line items (exclude tax/total/subtotal/service fee lines)", items: { type: "object", description: "A single purchased item line", properties: { name: { type: "string", description: "Item name" }, time: { type: "string", description: "Item-level time in ISO-8601 if available, otherwise an empty string" }, amount: { type: "number", description: "Item amount as a number" } } } }, total: { type: "number", description: "Total amount if present, otherwise -1" } } }, { provider: "openai" } ) // Post-processing suggestion: // Treat "" as null for purchasedAt/currency/time, and -1 as null for total. console.log(parsed) ``` *** ### Example 2: Generate an array ```ts type Expense = { name: string amount: number } const expenses = await Assistant.requestStructuredData( "List three common daily expenses with estimated amounts.", { type: "array", description: "A list of expenses", items: { type: "object", description: "A single expense item", properties: { name: { type: "string", description: "Expense name" }, amount: { type: "number", description: "Estimated amount" } } } }, { provider: "gemini" } ) ``` *** ### Example 3: Use images + schema ```ts type ImageSummary = { description: string containsText: boolean } const summary = await Assistant.requestStructuredData( "Analyze the image and summarize the main content.", ["..."], { type: "object", description: "Image analysis result", properties: { description: { type: "string", description: "What the image shows" }, containsText: { type: "boolean", description: "Whether readable text exists" } } }, { provider: "openai" } ) ``` *** ## Best Practices - Make the schema explicit and descriptive; ambiguous schemas lead to unstable results. - Prefer `requestStructuredData` over parsing free-form text when your output is used by program logic. - For business-critical extraction (e.g., finance/receipts), add strict formatting rules in `prompt`. --- url: /doc_v2/guide/AssistantTool.md --- # AssistantTool **Assistant Tool** is a system extension mechanism within the Scripting application that enhances the capabilities of an intelligent assistant (Assistant). By defining and implementing an Assistant Tool, developers can provide the Assistant with auxiliary functionalities such as device capability access, file reading/writing, and data analysis. This improves both the intelligence and practicality of the Assistant. This document uses an example tool, **"Request Current Location"**, to illustrate the full implementation process, including tool creation, configuration file explanation, execution logic, and detailed descriptions of various functions. *** ## 1. Tool Creation Process 1. Open any scripting project and click the “Add Assistant Tool” button in the file manager interface. 2. Fill in the relevant information about the Assistant Tool in the configuration popup window. 3. After clicking “Save,” the system will automatically generate two files in the script: - `assistant_tool.json`: Describes the tool’s metadata and parameter information. - `assistant_tool.tsx`: Implements the tool’s execution logic. *** ## 2. Configuration File: `assistant_tool.json` This file declares the basic information and behavior settings of the tool. Below is a sample content and explanation of each field: ```json { "displayName": "Request Current Location", "id": "request_current_location", "description": "This tool allows you to request the one-time delivery of the latitude and longitude of the user’s current location.", "icon": "location.fill", "color": "systemBlue", "parameters": [], "requireApproval": true, "autoApprove": true, "scriptEditorOnly": false } ``` ### Field Descriptions: | Field | Type | Description | | ------------------ | ------- | ------------------------------------------------------ | | `displayName` | string | Name displayed in the UI | | `id` | string | Unique identifier for the tool (must be unique) | | `description` | string | Description of the tool’s functionality | | `icon` | string | Name of the SF Symbols icon used | | `color` | string | Primary color of the tool | | `parameters` | array | Parameters required by the tool (empty means no input) | | `requireApproval` | boolean | Whether user approval is required | | `autoApprove` | boolean | Whether Assistant can auto-approve | | `scriptEditorOnly` | boolean | Whether the tool can only be used in the script editor | *** ## 3. Execution Logic Example: `assistant_tool.tsx` ```tsx type RequestCurrentLocationParams = {} const locationApprovalRequest: AssistantToolApprovalRequestFn = async ( params, ) => { return { message: "The assistant wants to request your current location.", primaryButtonLabel: "Allow" } } const requestCurrentLocation: AssistantToolExecuteWithApprovalFn = async ( params, { primaryConfirmed, secondaryConfirmed, } ) => { try { const location = await Location.requestCurrent() if (location) { return { success: true, message: [ "The user's current location info:", `${location.latitude}`, `${location.longitude}` ].join("\n") } } return { success: false, message: "Failed to request user's current location, ask user to check the device's location permission." } } catch { return { success: false, message: "Failed to request user's current location, ask user to check the device's location permission." } } } const testRequestLocationApprovalFn = AssistantTool.registerApprovalRequest( locationApprovalRequest ) const testRequestLocationExecuteFn = AssistantTool.registerExecuteToolWithApproval( requestCurrentLocation ) // Test the tool in the script editor: testRequestLocationApprovalFn({}) testRequestLocationExecuteFn({}, { primaryConfirmed: true, secondaryConfirmed: false }) ``` *** ## 4. AssistantTool Registration Functions Explained ### 1. `registerApprovalRequest` Registers a function to request user approval before executing the tool. ```ts function registerApprovalRequest

( requestFn: AssistantToolApprovalRequestFn

): AssistantToolApprovalRequestTestFn

``` **Parameters**: - `requestFn(params, scriptEditorProvider?)`: Returns a prompt with messages and button labels. - `params`: Input parameters for the tool. - `scriptEditorProvider`: Available only when `scriptEditorOnly` is set to true, provides file access for the script. **Return Value**: A test function for simulating approval requests in the script editor. *** ### 2. `registerExecuteToolWithApproval` Registers an execution function that requires user approval. ```ts function registerExecuteToolWithApproval

( executeFn: AssistantToolExecuteWithApprovalFn

): AssistantToolExecuteWithApprovalTestFn

``` **Parameters**: - `params`: Input parameters for execution. - `userAction`: User's choice in the approval prompt: ```ts type UserActionForApprovalRequest = { primaryConfirmed: boolean secondaryConfirmed: boolean } ``` - `scriptEditorProvider`: Same as above. **Return Value**: Returns an object: ```ts { success: boolean message: string } ``` - `success`: Whether execution succeeded. - `message`: Message returned to the Assistant. *** ### 3. `registerExecuteTool` Registers a tool that does **not** require user approval. ```ts function registerExecuteTool

( executeFn: AssistantToolExecuteFn

): AssistantToolExecuteTestFn

``` **Use Case**: Suitable for non-sensitive operations or those that don’t involve device permissions. *** ### 4. Testing Functions Each registration function returns a test function that can be used in the script: ```ts testApprovalRequestFn({ ...params }) testExecuteFn({ ...params }, { primaryConfirmed: true, secondaryConfirmed: false, }) testExecuteToolFn({ ...params }) ``` *** ## 5. ScriptEditorProvider Interface When `scriptEditorOnly: true` is set, the system provides a `ScriptEditorProvider` interface that allows access to the script project’s file system and syntax info. Capabilities include: - File read/write (read, update, write, insert, replace) - Diff comparison (`openDiffEditor`) - Linting results (`getLintErrors`) - List all files/folders in the project Useful for tools that edit scripts, perform formatting, or batch modifications. *** ## 6. Execution and User Experience Flow 1. The Assistant determines whether to invoke a tool during a conversation. 2. If the tool requires approval, a dialog box is displayed: - The prompt is provided by `registerApprovalRequest`. - Once the user clicks “Allow,” the tool logic is executed. 3. Execution results are returned to the Assistant through the `message` field, and shown to the user. *** ## 7. Tools That Don’t Require Approval If you don’t want to show an approval prompt, simply use `registerExecuteTool` and set `requireApproval: false` in `assistant_tool.json`. ```ts AssistantTool.registerExecuteTool(async (params) => { return { success: true, message: "Tool executed successfully." } }) ``` *** ## 8. Summary Assistant Tool is an extensible module provided by the Scripting application, supporting scenarios such as user authorization, file manipulation, and system-level access. The development process includes: 1. Creating the tool in a scripting project; 2. Configuring its metadata; 3. Implementing and registering the logic functions; 4. Testing tool behavior with test functions; 5. Triggering tool execution automatically or manually within Assistant conversations. --- url: /doc_v2/guide/Changelog/2.4.3/Animation and Transition.md --- # Animation and Transition Scripting Animation & Transition System ## Animation Class The `Animation` class describes how values animate in time. ## Factory Methods ### `Animation.default()` Creates a default system animation. ```ts static default(): Animation ``` *** ### `Animation.linear(duration?)` ```ts static linear(duration?: number | null): Animation ``` Constant-speed animation. *** ### `Animation.easeIn(duration?)` ```ts static easeIn(duration?: number | null): Animation ``` *** ### `Animation.easeOut(duration?)` ```ts static easeOut(duration?: number | null): Animation ``` *** ### `Animation.bouncy(options?)` ```ts static bouncy(options?: { duration?: number extraBounce?: number }): Animation ``` Spring-like animation with additional bounce. *** ### `Animation.smooth(options?)` ```ts static smooth(options?: { duration?: number extraBounce?: number }): Animation ``` *** ### `Animation.snappy(options?)` ```ts static snappy(options?: { duration?: number extraBounce?: number }): Animation ``` *** ### `Animation.spring(options?)` Supports two mutually exclusive modes. ```ts static spring(options?: { blendDuration?: number } & ( | { duration?: number bounce?: number response?: never dampingFraction?: never } | { response?: number dampingFraction?: number duration?: never bounce?: never } )): Animation ``` *** ### `Animation.interactiveSpring(options?)` ```ts static interactiveSpring(options?: { response?: number dampingFraction?: number blendDuration?: number }): Animation ``` *** ### `Animation.interpolatingSpring(options?)` ```ts static interpolatingSpring(options?: { mass?: number stiffness: number damping: number initialVelocity?: number } | { duration?: number bounce?: number initialVelocity?: number mass?: never stiffness?: never damping?: never }): Animation ``` *** ## Modifier Methods ### `.delay(time)` ```ts delay(time: number): Animation ``` ### `.repeatCount(count, autoreverses)` ```ts repeatCount(count: number, autoreverses?: boolean): Animation ``` ### `.repeatForever(autoreverses)` ```ts repeatForever(autoreverses?: boolean): Animation ``` *** ## Transition Class `Transition` describes how a view enters or leaves the hierarchy. ## Instance Methods ### `.animation(animation)` Attach a specific animation to a transition. ```ts animation(animation?: Animation): Transition ``` ### `.combined(other)` Combine transitions. ```ts combined(other: Transition): Transition ``` *** ## Static Transitions ### Identity ```ts Transition.identity(); ``` ### Move ```ts Transition.move(edge: Edge) ``` ### Offset ```ts Transition.offset(position?: Point) ``` ### Push ```ts Transition.pushFrom(edge: Edge) ``` ### Opacity ```ts Transition.opacity(); ``` ### Scale ```ts Transition.scale(scale?: number, anchor?: Point | KeywordPoint) ``` ### Slide ```ts Transition.slide(); ``` ### Fade ```ts Transition.fade(duration?: number) ``` ### Flip transitions ```ts Transition.flipFromLeft(duration?) Transition.flipFromRight(duration?) Transition.flipFromTop(duration?) Transition.flipFromBottom(duration?) ``` ### Asymmetric ```ts Transition.asymmetric(insertion: Transition, removal: Transition) ``` *** ## withAnimation ```ts function withAnimation(body: () => void): Promise; function withAnimation(animation: Animation, body: () => void): Promise; function withAnimation( animation: Animation, completionCriteria: "logicallyComplete" | "removed", body: () => void, ): Promise; ``` Wraps a state update and animates any affected values. Example: ```ts withAnimation(Animation.easeOut(0.3), () => { visible.setValue(false); }); ``` *** ## Correct Usage of the animation View Modifier ### (Important Correction) In Scripting, the `animation` prop is **not**: ```tsx animation = { anim }; // incorrect ``` The correct format is: ```tsx animation={{ animation: anim, value: }} ``` ### Meaning: | Field | Description | | ----------- | ----------------------------------------------------- | | `animation` | The `Animation` instance to use | | `value` | The observable value whose changes should be animated | This mirrors SwiftUI’s `.animation(animation, value: value)` modifier. *** ## Correct Examples ### Example: Animate size changes ```tsx const size = useObservable(100) const anim = Animation.spring({ duration: 0.3, bounce: 0.3 }) `) await webView.present({ navigationTitle: 'WebView Demo' }) webView.dispose() ``` --- url: /doc_v2/guide/Changelog/2.4.6/Device/index.md --- # Device The `Device` namespace provides access to device information, system environment data, screen and battery status, language and locale settings, and selected device capabilities (such as wake lock and network interface inspection). This API is commonly used for: - Device-specific logic (iPhone / iPad / Mac) - UI layout and adaptive design - Localization and language selection - Network inspection and debugging - Preventing the device from sleeping during script execution *** ## Device and System Information ### `Device.model: string` The device model name, for example `"iPhone"` or `"iPad"`. *** ### `Device.systemName: string` The name of the operating system, such as `"iOS"`, `"iPadOS"`, or `"macOS"`. *** ### `Device.systemVersion: string` The current operating system version, for example `"17.2"`. *** ### `Device.isiPhone: boolean` Indicates whether the current device is an iPhone. *** ### `Device.isiPad: boolean` Indicates whether the current device is an iPad. *** ### `Device.isiOSAppOnMac: boolean` A Boolean value that indicates whether the process is an iPhone or iPad app running on a Mac (Mac Catalyst or Apple Silicon). *** ## Screen Information ### `Device.screen` Information about the main screen. ```ts { width: number height: number scale: number } ``` Field descriptions: - `width`: Logical screen width (points) - `height`: Logical screen height (points) - `scale`: Screen scale factor (for example, 2 or 3) This is commonly used for layout calculations, canvas sizing, screenshots, or rendering logic. *** ## Orientation and Device Posture ### `Device.isPortrait: boolean` Indicates whether the device is currently in portrait orientation. *** ### `Device.isLandscape: boolean` Indicates whether the device is currently in landscape orientation. *** ### `Device.isFlat: boolean` Indicates whether the device is lying flat (for example, placed on a table). This value is typically derived from motion sensors and can be used for advanced interaction logic. *** ## Appearance and Theme ### `Device.colorScheme: ColorScheme` The current system appearance mode. Typical values include: - `light` - `dark` This is useful for adapting UI themes and styles to match system settings. *** ## Battery Information ### `Device.batteryState` The current battery state: ```ts "full" | "charging" | "unplugged" | "unknown" ``` Descriptions: - `full`: Battery is fully charged - `charging`: Battery is charging - `unplugged`: Device is not connected to power - `unknown`: Battery state is unavailable *** ### `Device.batteryLevel: number` The current battery level, in the range: - `0.0` to `1.0` - May return `-1` if the battery level is unavailable *** ## Language and Locale Settings ### `Device.systemLocale: string` The system’s current locale identifier, for example: ```text "en_US" ``` *** ### `Device.preferredLanguages: string[]` The user’s preferred languages, ordered by priority, for example: ```ts ["en-US", "zh-Hans-CN"] ``` This is the recommended API for language selection and localization logic. *** ### `Device.systemLocales: string[]` (Deprecated) The user’s preferred locales. > Deprecated. Use `Device.preferredLanguages` instead. *** ### `Device.systemLanguageTag: string` The current language tag in BCP-47 format, for example: ```text "en-US" ``` *** ### `Device.systemLanguageCode: string` The current language code, for example: ```text "en" ``` *** ### `Device.systemCountryCode: string | undefined` The current country or region code, for example: ```text "US" ``` This value may be `undefined` if no country information is available. *** ### `Device.systemScriptCode: string | undefined` The script code of the current language, for example: ```text "Hans" // zh_CN_Hans ``` This is commonly used to distinguish writing systems, such as Simplified vs. Traditional Chinese. *** ## Wake Lock Wake lock prevents the device from automatically sleeping while a script is running. ### `Device.isWakeLockEnabled: Promise` Retrieves whether the wake lock is currently enabled. ```ts const enabled = await Device.isWakeLockEnabled ``` *** ### `Device.setWakeLockEnabled(enabled: boolean): void` Enables or disables the wake lock. ```ts Device.setWakeLockEnabled(true) ``` Notes: - Available only in the **Scripting app** - Prevents the screen from dimming or the device from sleeping - Should be disabled when no longer needed to conserve battery *** ## Network Interface Information ### `Device.NetworkInterface` The structure that represents a network interface entry: ```ts type NetworkInterface = { address: string netmask: string | null family: 'IPv4' | 'IPv6' mac: string | null isInternal: boolean cidr: string | null } ``` Field descriptions: - `address`: IP address - `netmask`: Subnet mask - `family`: Address family (`IPv4` or `IPv6`) - `mac`: MAC address (may be null on some systems) - `isInternal`: Indicates whether the interface is internal (for example, loopback) - `cidr`: CIDR notation (for example, `192.168.1.10/24`) *** ### `Device.networkInterfaces(): Record` Returns the network interfaces of the device. The returned object is structured as: ```ts { [interfaceName: string]: NetworkInterface[] } ``` Example: ```ts const interfaces = Device.networkInterfaces() for (const name in interfaces) { for (const info of interfaces[name]) { console.log(name, info.address, info.family) } } ``` Common use cases: - Retrieving local IP addresses - Distinguishing Wi-Fi, cellular, and loopback interfaces - Network diagnostics and debugging - Emulating Node.js `os.networkInterfaces()` behavior *** ## Best Practices - Prefer `preferredLanguages` for localization logic - Always disable the wake lock when it is no longer required - Do not assume specific interface names (such as `en0`) will always exist - Network interface availability may vary based on permissions and network state --- url: /doc_v2/guide/Changelog/2.4.6/Device/index_example.md --- # Example ```tsx import { Button, Device, List, Navigation, NavigationStack, Script, Text, VStack } from "scripting" function Example() { const dismiss = Navigation.useDismiss() const details: { name: string value: string | boolean | number }[] = [ { name: "Device.isiPhone", value: Device.isiPhone }, { name: "Device.isiPad", value: Device.isiPad, }, { name: "Device.systemVersion", value: Device.systemVersion, }, { name: "Device.systemName", value: Device.systemName, }, { name: "Device.isPortrait", value: Device.isPortrait, }, { name: "Device.isLandscape", value: Device.isLandscape, }, { name: "Device.isFlat", value: Device.isFlat, }, { name: "Device.batteryLevel", value: Device.batteryLevel, }, { name: "Device.batteryState", value: Device.batteryState, } ] return }} > {details.map(item => {item.name} {typeof item.value} )} } async function run() { await Navigation.present({ element: }) Script.exit() } run() ``` --- url: /doc_v2/guide/Changelog/2.4.6/Intent.md --- # Intent Scripting allows you to define custom iOS Intents using an `intent.tsx` file. These scripts can receive input from the iOS share sheet or the Shortcuts app and return structured results. With optional UI presentation, you can create interactive workflows that process data and deliver output dynamically. *** ## 1. Creating and Configuring an Intent ### 1.1 Create an Intent Script 1. Create a new script project in the Scripting app. 2. Add a file named `intent.tsx` to the project. 3. Define your logic and optionally a UI component inside the file. ### 1.2 Configure Supported Input Types Tap the project title in the editor’s title bar to open **Intent Settings**, then select supported input types: - Text - Images - File URLs - URLs This configuration enables your script to appear in the share sheet or Shortcuts when matching input is provided. *** ## 2. Accessing Input Data Inside `intent.tsx`, use the `Intent` API to access input values. | Property | Description | | -------------------------- | ----------------------------------------------------------------------------------- | | `Intent.shortcutParameter` | A single parameter passed from the Shortcuts app, with `.type` and `.value` fields. | | `Intent.textsParameter` | Array of text strings. | | `Intent.urlsParameter` | Array of URL strings. | | `Intent.imagesParameter` | Array of image file paths (UIImage objects). | | `Intent.fileURLsParameter` | Array of local file URL paths. | Example: ```ts if (Intent.shortcutParameter) { if (Intent.shortcutParameter.type === "text") { console.log(Intent.shortcutParameter.value) } } ``` *** ## 3. Returning a Result Use `Script.exit(result)` to return a result to the caller, such as the Shortcuts app or another script. Valid return types include: - Plain text: `Intent.text(value)` - Attributed text: `Intent.attributedText(value)` - URL: `Intent.url(value)` - JSON: `Intent.json(value)` - File path or file URL: `Intent.file(value)` or `Intent.fileURL(value)` Example: ```ts import { Script, Intent } from "scripting" Script.exit(Intent.text("Done")) ``` *** ## 4. Displaying Interactive UI Use `Navigation.present()` to show a UI before returning a result. You can render a React-style component and then call `Script.exit()` after the interaction completes. Example: ```ts import { Intent, Script, Navigation, VStack, Text } from "scripting" function MyIntentView() { return ( {Intent.textsParameter?.[0]} ) } async function run() { await Navigation.present({ element: }) Script.exit() } run() ``` *** ## 5. Using Intents in the Share Sheet If a script supports a specific input type (e.g., text or image), it will automatically appear as an option in the iOS share sheet: 1. Select content such as text or a file. 2. Tap the Share button. 3. Choose **Scripting** in the share sheet. 4. Scripting will list scripts that support the selected input type. *** ## 6. Using Intents in the Shortcuts App You can call scripts from the Shortcuts app with or without UI: - **Run Script**: Executes the script in the background. - **Run Script in App**: Executes the script in the foreground, with UI presentation support. Steps: 1. Open the Shortcuts app and create a new shortcut. 2. Add the **Run Script** or **Run Script in App** action from Scripting. 3. Choose the target script and pass input parameters if needed. *** ## 7. Intent API Reference ### `Intent` Properties | Property | Type | Description | | ------------------- | ------------------- | ----------------------------------------------- | | `shortcutParameter` | `ShortcutParameter` | Input from Shortcuts with `.type` and `.value`. | | `textsParameter` | `string[]` | Array of input text values. | | `urlsParameter` | `string[]` | Array of input URLs. | | `imagesParameter` | `UIImage[]` | Array of image file paths or objects. | | `fileURLsParameter` | `string[]` | Array of input file paths (local file URLs). | ### `Intent` Methods | Method | Return Type | Example | | ------------------------------ | --------------------------- | -------------------------------------- | | `Intent.text(value)` | `IntentTextValue` | `Intent.text("Hello")` | | `Intent.attributedText(value)` | `IntentAttributedTextValue` | `Intent.attributedText("Styled Text")` | | `Intent.url(value)` | `IntentURLValue` | `Intent.url("https://example.com")` | | `Intent.json(value)` | `IntentJsonValue` | `Intent.json({ key: "value" })` | | `Intent.file(path)` | `IntentFileValue` | `Intent.file("/path/to/file.txt")` | | `Intent.fileURL(path)` | `IntentFileURLValue` | `Intent.fileURL("/path/to/file.pdf")` | | `Intent.image(UIImage)` | `IntentImageValue` | `Intent.image(uiImage)` | *** ## 8. Best Practices and Notes - Always call `Script.exit()` to properly terminate the script and return a result. - When displaying a UI, ensure `Navigation.present()` is awaited before calling `Script.exit()`. - Use **"Run Script in App"** for large files or images to avoid process termination due to memory constraints. - You can use `queryParameters` when launching scripts via URL scheme if additional data is needed. --- url: /doc_v2/guide/Changelog/2.4.6/ItemProvider.md --- # ItemProvider `ItemProvider` represents a **deferred data provider** used to access content such as files, images, text, or URLs in a controlled and secure way. It is commonly used in scenarios like drag and drop, file importing, and content selection from Photos or Files. An `ItemProvider` does not store the data itself. Instead, it describes **how and under what constraints the data can be accessed**. *** ## Core Concepts - `ItemProvider` describes capabilities, not concrete data - Data loading is always subject to system security restrictions - File-based resources can only be accessed within a limited, controlled scope - Whether a file can be accessed in place is determined by the underlying system *** ## Properties ### registeredTypes ```ts readonly registeredTypes: UTType[] ``` Represents all types that the item provider can supply at a semantic level. - Includes both concrete types and inferred parent types - Useful for high-level content classification or debugging - Does not guarantee that a concrete file representation exists *** ### registeredInPlaceTypes ```ts readonly registeredInPlaceTypes: UTType[] ``` Represents the set of types that support open-in-place access. - Typically applies to large resources such as videos, audio files, or documents - Actual in-place access is determined at load time *** ## Capability Checks ### hasItemConforming ```ts hasItemConforming(type: UTType): boolean ``` Checks whether the content semantically conforms to the specified type. - Performs a broad, semantic check - Considers UTType inheritance - Suitable for branching logic and content classification *** ### hasRepresentationConforming ```ts hasRepresentationConforming(type: UTType): boolean ``` Checks whether a concrete, loadable representation exists for the specified type. - Performs a strict check - Suitable for file processing and format-specific workflows *** ### hasInPlaceRepresentationConforming ```ts hasInPlaceRepresentationConforming(type: UTType): boolean ``` Checks whether a representation supporting open-in-place access exists. - Commonly used to choose loading strategies for large files *** ## Object Loading Capabilities ### canLoadUIImage ```ts canLoadUIImage(): boolean ``` Indicates whether the content can be loaded as a `UIImage`. - Intended for UI display - Does not guarantee preservation of original format or metadata *** ### canLoadLivePhoto ```ts canLoadLivePhoto(): boolean ``` Indicates whether the content can be loaded as a `LivePhoto`. - Used to distinguish Live Photos from static images - When true, `loadLivePhoto` can be called *** ## Loading Methods ### loadUIImage ```ts loadUIImage(): Promise ``` Loads a `UIImage` object. - Suitable for lightweight display - Not intended for file-level processing or asset preservation *** ### loadLivePhoto ```ts loadLivePhoto(): Promise ``` Loads a `LivePhoto` object. - Includes both the still image and paired video - Suitable for display, saving, or further processing *** ### loadURL ```ts loadURL(): Promise ``` Loads a URL and returns it as a string. - May represent a web URL or a file URL *** ### loadText ```ts loadText(): Promise ``` Loads plain text content. - Supports plain text - Rich text is automatically converted to plain text *** ### loadData ```ts loadData(type: UTType): Promise ``` Loads raw binary data for the specified type. - The entire data payload is loaded into memory - Suitable for JSON, configuration files, or small resources - Not recommended for large files such as video or audio *** ## File Path Loading and Security Scope Access to file paths is subject to strict security rules. All file access must occur within a limited callback scope provided by the API. *** ### loadFilePath ```ts loadFilePath(type: UTType): Promise ``` Loads a file path for the specified type. If the item provider can load data as the specified type, this file will be copied to the app group's temporary directory and the file path will be returned, otherwise null will be returned. You should delete the file when it is no longer needed. Example: ```ts const filePath = provider.loadFilePath("public.movie") ``` *** ## Creating an ItemProvider ### fromUIImage ```ts ItemProvider.fromUIImage(image: UIImage): ItemProvider ``` Creates an `ItemProvider` from a `UIImage`. - Provides static image capabilities only - Does not include Live Photo or original asset information *** ### fromText ```ts ItemProvider.fromText(text: string): ItemProvider ``` Creates an `ItemProvider` from a text string. *** ### fromURL ```ts ItemProvider.fromURL(url: string): ItemProvider | null ``` Creates an `ItemProvider` from a URL string. - Returns `null` if the URL is invalid - Supports both web URLs and file URLs *** ### fromFilePath ```ts ItemProvider.fromFilePath(path: string): ItemProvider ``` Creates an `ItemProvider` from a file path. - Preserves the original file - Suitable for videos, audio, and documents - Supports open-in-place capability checks *** ## Usage Guidelines - Use `hasItemConforming` to determine content categories - Use object loading methods for UI display - Use file path loading methods for large resources - Always access files only within the provided callback scope - Never defer access to security-scoped files outside the callback --- url: /doc_v2/guide/Changelog/2.4.6/MediaComposer/MediaComposer Example.md --- # MediaComposer Example This example demonstrates how to use `MediaComposer` to compose a final video from **video, image, and audio sources**, and export it to the script directory. The workflow covered in this example includes: 1. Picking an audio file 2. Picking an image 3. Picking a video 4. Building a visual timeline (video + image) 5. Inserting audio at a specific time 6. Exporting the composed video *** ## Example Code ```tsx import { Path, Script } from "scripting" console.present().then(() => Script.exit()) async function run() { try { const audioPath = (await DocumentPicker.pickFiles({ types: ["public.audio"] })).at(0) if (audioPath == null) { console.error("no audio") return } console.log("Audio Picked") const imageResult = (await Photos.pick({ filter: PHPickerFilter.images() })).at(0) const imagePath = await imageResult?.itemProvider.loadFilePath("public.image") if (!imagePath) { console.log("No image") return } console.log("Image picked") const videoResult = (await Photos.pick({ filter: PHPickerFilter.videos() })).at(0) const videoPath = await videoResult?.itemProvider.loadFilePath("public.movie") if (videoPath == null) { console.log("No video") return } console.log("Video Picked") console.log("Start composing...") const exportPath = Path.join( Script.directory, "dest.mp4" ) const exportResult = await MediaComposer.composeAndExport({ exportPath, timeline: { videoItems: [{ videoPath: videoPath }, { imagePath: imagePath, duration: MediaTime.make({ seconds: 5, preferredTimescale: 600 }) }], audioClips: [{ path: audioPath, at: MediaTime.make({ seconds: 5, preferredTimescale: 600 }) }] } }) console.log( "Result:", exportResult.exportPath, "\n", exportResult.duration.getSeconds() ) } catch (e) { console.error(e) } } run() ``` *** ## Timeline Breakdown ### Visual Timeline (videoItems) ```ts videoItems: [ { videoPath }, { imagePath, duration: MediaTime.make({ seconds: 5, preferredTimescale: 600 }) } ] ``` - The first `VideoItem` is a full video clip - The second `VideoItem` is an image displayed for 5 seconds - All `videoItems` are concatenated **in strict order** - Final video duration = video duration + 5 seconds *** ### Audio Timeline (audioClips) ```ts audioClips: [{ path: audioPath, at: MediaTime.make({ seconds: 5, preferredTimescale: 600 }) }] ``` - The audio starts playing at **5 seconds** on the final timeline - When `at` is omitted, audio clips are appended sequentially - Audio does **not** affect the final video duration *** ## Export Result ```ts { exportPath: string duration: MediaTime } ``` - `exportPath`: the full output file path - `duration`: the total video duration (derived from `videoItems`) *** ## Common Errors and Edge Cases ### 1. ImageClip without duration ```ts { imagePath: "...", // ❌ missing duration } ``` **Issue:** - Images have no intrinsic duration - Omitting `duration` will cause composition to fail **Solution:** - Always provide an explicit `MediaTime` duration *** ### 2. Using raw numbers instead of MediaTime ```ts // ❌ incorrect at: 5 ``` **Correct usage:** ```ts at: MediaTime.make({ seconds: 5, preferredTimescale: 600 }) ``` All time values in MediaComposer **must** be represented by `MediaTime`. *** ### 3. Mixed timescales causing precision issues **Issue:** - Different media sources may use different timescales - This can lead to rounding errors during trimming, fades, or alignment **Recommendation:** - Use a consistent `preferredTimescale` (e.g. 600) - Convert external times using `convertScale` when needed *** ### 4. Audio extending beyond the video duration **Behavior:** - Audio that exceeds the end of the video does not extend the final duration - Any audio beyond the video end is automatically truncated *** ### 5. Unexpected audio balance when mixing original and external audio **Cause:** - By default, original video audio and external audio are mixed together - Without ducking, dialogue may be masked by background music *** ## Audio Ducking Behavior ### What is Ducking Ducking refers to: > Automatically lowering the volume of external audio (e.g. background music) when original video audio (e.g. dialogue) is present. *** ### Ducking Configuration ```ts exportOptions: { ducking: { enabled: true, duckedVolume: 0.25, attackSeconds: 0.15, releaseSeconds: 0.25 } } ``` #### Parameters - **enabled** Enables or disables ducking (default: `true`) - **duckedVolume** Target volume for external audio during ducking (0…1) - **attackSeconds** Ramp-down duration before original audio starts - **releaseSeconds** Ramp-up duration after original audio ends *** ### Conditions for Ducking to Apply Ducking is applied only when all of the following are true: 1. `VideoClip.keepOriginalAudio === true` 2. At least one external `AudioClip` exists 3. `exportOptions.ducking.enabled !== false` *** ## Audio Mixing Rules Summary 1. **Original Video Audio** - Included only when `keepOriginalAudio` is set to `true` 2. **External Audio** - Can be positioned or appended sequentially - Supports per-clip `volume`, `fade`, and looping 3. **Final Mix** - All audio sources are mixed into a single output track - Audio never changes the final video duration - Ducking is applied automatically during mixing --- url: /doc_v2/guide/Changelog/2.4.6/MediaComposer/MediaTime.md --- # MediaTime `MediaTime` represents **precise media time values** in audio and video processing. It is the fundamental time type used by MediaComposer in Scripting. Conceptually, `MediaTime` corresponds to a time value with an explicit time base (similar to `CMTime` in AVFoundation), but provides a safer and more expressive abstraction for the scripting layer. A `MediaTime` instance can represent **numeric time**, **invalid time**, **indefinite time**, or **infinite time**, and supports strict arithmetic and comparison operations. *** ## Key Features - Precise construction using **value + timescale** or **seconds + preferredTimescale** - Time scaling with configurable rounding methods - Safe arithmetic and comparison operations - Explicit modeling of invalid, indefinite, and infinite time values - Designed for timeline composition, trimming, alignment, fades, and placement *** ## Time Precision Model `MediaTime` is based on the following core concepts: - **value**: an integer time value - **timescale**: the number of time units per second Examples: - `value = 300`, `timescale = 600` → 0.5 seconds - `value = 18000`, `timescale = 600` → 30 seconds This model allows frame-accurate or sample-accurate timing without relying on floating-point arithmetic. *** ## Read-only Properties ### secondes ```ts readonly secondes: number ``` The time expressed in seconds as a floating-point value. This is a derived value intended mainly for display or debugging. It is **not recommended for timeline calculations**. *** ### isValid ```ts readonly isValid: boolean ``` Indicates whether the time is valid and usable for calculations. Returns `false` for invalid, indefinite, or infinite time values. *** ### isPositiveInfinity / isNegativeInfinity ```ts readonly isPositiveInfinity: boolean readonly isNegativeInfinity: boolean ``` Indicates whether the time represents positive or negative infinity. These values are typically used as internal boundary markers in timeline logic. *** ### isIndefinite ```ts readonly isIndefinite: boolean ``` Indicates whether the time is indefinite. This is commonly used when a media asset’s duration has not yet been determined. *** ### isNumeric ```ts readonly isNumeric: boolean ``` Indicates whether the time can participate in numeric calculations. Arithmetic and comparison operations should only be performed when this value is `true`. *** ### hasBeenRounded ```ts readonly hasBeenRounded: boolean ``` Indicates whether the time has undergone rounding during construction or scale conversion. This is useful when validating frame- or sample-accurate timelines. *** ## Time Conversion ### convertScale ```ts convertScale(newTimescale: number, method: MediaTimeRoundingMethod): MediaTime ``` Converts the time to a new timescale using the specified rounding method. **Typical use cases:** - Aligning video frame timing (e.g. 600, 90000) - Aligning audio sample timing (e.g. 44100, 48000) - Avoiding precision errors caused by mixed timescales *** ## Accessing Time Values ### getSeconds ```ts getSeconds(): number ``` Returns the time expressed in seconds as a floating-point value. Semantically equivalent to reading `secondes`, but clearer in intent. *** ## Time Arithmetic ### plus / minus ```ts plus(other: MediaItem): MediaItem minus(other: MediaItem): MediaItem ``` Performs time addition or subtraction and returns a new `MediaTime`. - Both operands must be numeric - The original instances are not modified - The result follows the internal time base rules *** ## Time Comparison ```ts lt(other: MediaItem): boolean gt(other: MediaItem): boolean lte(other: MediaItem): boolean gte(other: MediaItem): boolean eq(other: MediaItem): boolean neq(other: MediaItem): boolean ``` Compares two time values. - Supports strict ordering and equality checks - Produces deterministic results even for non-numeric times - Recommended for timeline sorting, trimming, and boundary checks *** ## Static Constructors ### make ```ts static make(options: { value: number timescale: number } | { seconds: number preferredTimescale: number }): MediaTime ``` Creates a `MediaTime` instance. #### Using value + timescale ```ts MediaTime.make({ value: 300, timescale: 600 }) ``` Best suited for low-level or precision-critical scenarios. *** #### Using seconds + preferredTimescale ```ts MediaTime.make({ seconds: 5, preferredTimescale: 600 }) ``` Recommended for most scripting-level use cases where seconds are the primary unit. *** ### zero ```ts static zero(): MediaTime ``` Returns a `MediaTime` representing **0 seconds**. *** ### invalid ```ts static invalid(): MediaTime ``` Returns an invalid time value. Useful for explicitly representing errors or unavailable timing information. *** ### indefinite ```ts static indefinite(): MediaTime ``` Returns an indefinite time value. Typically used when a media asset’s duration is not yet known. *** ### positiveInfinity / negativeInfinity ```ts static positiveInfinity(): MediaTime static negativeInfinity(): MediaTime ``` Returns positive or negative infinite time values. These are mainly intended for internal timeline boundary handling and are not recommended for general scripting logic. *** ## Usage Guidelines and Best Practices - Avoid using floating-point seconds directly for timeline calculations; prefer `MediaTime` - Explicitly convert timescales when mixing audio and video sources - Check `isNumeric` before performing arithmetic or comparisons - Use consistent timescales when constructing `TimeRange` or `at` values *** ## Typical Usage in MediaComposer - Placing audio or video clips on the timeline (`AudioClip.at`) - Defining trimming ranges (`TimeRange`) - Calculating precise export durations - Driving fades, alignment, looping, and synchronization behavior --- url: /doc_v2/guide/Changelog/2.4.6/MediaComposer/Quick Start.md --- # Quick Start `MediaComposer` is used in Scripting to **compose video, image, and audio timelines and export a final media file**. It provides a stable and precise timeline model that supports video clips, image clips, audio overlays, fades, audio ducking, and flexible export configuration. This module is suitable for: - Mixing videos and images into a single output - Adding background music, voice-over, or sound effects - Generating videos from image sequences - Automated and script-driven media production *** ## Design Overview MediaComposer consists of three core layers: 1. **Time Model** Based on `MediaTime` and `TimeRange` for precise time representation 2. **Timeline Model** - `VideoItem[]`: visual timeline (videos or images, sequential) - `AudioClip[]`: audio timeline (positioned or sequential) 3. **Export System** A unified `composeAndExport` API for rendering and exporting *** ## Timeline Structure ```ts timeline: { videoItems: VideoItem[] audioClips: AudioClip[] } ``` - **videoItems** Defines the visual timeline. Video and image items are concatenated strictly in array order. - **audioClips** Defines the audio timeline. Clips may be explicitly positioned or appended sequentially. The final exported duration is determined by the **videoItems timeline**. *** ## VideoItem ```ts type VideoItem = XOR ``` A `VideoItem` represents a single visual segment in the timeline. It can be either a **video clip** or an **image clip**, but never both. *** ## VideoClip ```ts type VideoClip = { videoPath: string sourceTimeRange?: TimeRange | null keepOriginalAudio?: boolean fade?: FadeConfig | null } ``` ### videoPath - Path to the video file - Local video files are supported *** ### sourceTimeRange ```ts sourceTimeRange?: TimeRange | null ``` - Specifies the portion of the source video to use - Defaults to the entire video when omitted **Common use cases:** - Trimming a video - Extracting a specific segment as material *** ### keepOriginalAudio ```ts keepOriginalAudio?: boolean ``` - Whether to keep the original audio from the video - Default: `false` **Notes:** - When `true`, the video’s original audio participates in the final mix - External `audioClips` may still be used simultaneously - Ducking behavior is controlled by `ExportOptions.ducking` *** ### fade ```ts fade?: FadeConfig | null ``` - Per-clip fade-in / fade-out configuration - Overrides the global video fade when provided *** ## ImageClip ```ts type ImageClip = { imagePath: string duration: MediaTime contentMode?: "fit" | "crop" backgroundColor?: Color fade?: FadeConfig | null } ``` `ImageClip` allows a still image to appear as a timed segment within the video timeline. *** ### imagePath - Path to the image file - Common image formats are supported (JPEG, PNG, HEIC, etc.) *** ### duration ```ts duration: MediaTime ``` - The display duration of the image clip in the video - This field is required *** ### contentMode ```ts contentMode?: "fit" | "crop" ``` - Controls how the image is scaled to the render size - Default: `fit` Behavior: - `fit`: Entire image is visible; letterboxing may occur - `crop`: Image fills the frame; excess is cropped *** ### backgroundColor ```ts backgroundColor?: Color ``` - Background color for areas not covered by the image - Commonly used together with `fit` mode *** ### fade ```ts fade?: FadeConfig | null ``` - Fade-in / fade-out configuration for the image clip - Behaves the same as fades for video clips *** ## AudioClip ```ts type AudioClip = { path: string sourceTimeRange?: TimeRange | null at?: MediaTime volume?: number fade?: FadeConfig | null loopToFitVideoDuration?: boolean } ``` `AudioClip` is used to add background music, narration, or sound effects to the final video. *** ### path - Path to the audio file *** ### sourceTimeRange - Specifies the portion of the audio to use - Defaults to the entire audio file *** ### at ```ts at?: MediaTime ``` - Explicit placement time on the final timeline - When omitted, audio clips are appended sequentially *** ### volume ```ts volume?: number ``` - Per-clip gain (0…1) - Default: `1` *** ### fade - Fade-in / fade-out configuration for the audio clip *** ### loopToFitVideoDuration ```ts loopToFitVideoDuration?: boolean ``` - Whether the audio should loop to match the total video duration - Commonly used for background music *** ## FadeConfig ```ts type FadeConfig = { fadeInSeconds?: number fadeOutSeconds?: number } ``` - Duration is expressed in seconds - Applicable to video, image, and audio clips - Defaults to 0 when omitted *** ## ExportOptions ```ts type ExportOptions = { renderSize?: Size frameRate?: number scaleMode?: VideoScaleMode globalVideoFade?: FadeConfig | null externalAudioBaseVolume?: number ducking?: DuckingConfig presetName?: ExportPreset outputFileType?: ExportFileType } ``` ### Common options - **renderSize** Output resolution, default is 1080×1920 - **frameRate** Rendering frame rate, default is 30 - **globalVideoFade** Global fade applied to all visual clips unless overridden - **ducking** Automatically lowers external audio volume when original video audio exists - **presetName / outputFileType** Control encoding quality and output format *** ## composeAndExport ```ts function composeAndExport(options: { exportPath: string timeline: { videoItems: VideoItem[] audioClips: AudioClip[] } exportOptions?: ExportOptions overwrite?: boolean }): Promise<{ exportPath: string duration: MediaTime }> ``` ### Parameters - **exportPath** Output file path - **timeline.videoItems** Visual timeline (videos and images, in sequence) - **timeline.audioClips** Audio timeline (positioned or sequential) - **exportOptions** Optional export configuration - **overwrite** Whether to overwrite an existing file (default: `true`) *** ### Return Value ```ts { exportPath: string duration: MediaTime } ``` - **exportPath**: final output path - **duration**: total duration of the exported video (derived from `videoItems`) *** ## Usage Guidelines and Best Practices - Always use `MediaTime` for time values; avoid raw floating-point seconds - `ImageClip.duration` must always be explicitly specified - Audio and visual timelines are independent but mixed in the final output - For complex projects, use a consistent timescale (e.g. 600) - Background music typically uses `loopToFitVideoDuration` *** ## Typical Use Cases - Mixed image and video composition - Adding background music or narration to videos - Automated video generation - Script-driven content creation pipelines --- url: /doc_v2/guide/Changelog/2.4.6/SharedAudioSession.md --- # SharedAudioSession The `SharedAudioSession` interface provides a convenient way to manage and interact with the shared audio session in your script. The audio session acts as an intermediary between your script, the Scripting app, the operating system, and the underlying audio hardware, enabling you to configure and control audio behavior effectively. ## Features - Retrieve and set audio session categories, modes, and options. - Configure the preferred sample rate for audio input and output. - Handle audio interruptions. - Query device capabilities for supported categories and modes. - Tailor audio behaviors for specific app use cases, such as video recording, voice chat, or background playback. *** ## Methods and Properties ### 1. **Session Category and Options** #### `category` Get the current audio session category. ```typescript const category = await SharedAudioSession.category console.log(category) // Example: 'playback' ``` #### `categoryOptions` Retrieve the current audio session category options. ```typescript const options = await SharedAudioSession.categoryOptions console.log(options) // Example: ['mixWithOthers', 'allowAirPlay'] ``` #### `setCategory(category: AudioSessionCategory, options: AudioSessionCategoryOptions[])` Set the audio session category with specific options. ```typescript await SharedAudioSession.setCategory('playback', ['mixWithOthers']) ``` *** ### 2. **Session Mode** #### `mode` Retrieve the current audio session mode. ```typescript const mode = await SharedAudioSession.mode console.log(mode) // Example: 'videoChat' ``` #### `setMode(mode: AudioSessionMode)` Set the audio session mode. ```typescript await SharedAudioSession.setMode('voiceChat') ``` *** ### 3. **Sample Rate** #### `preferredSampleRate` Retrieve the preferred sample rate in hertz. ```typescript const sampleRate = await SharedAudioSession.preferredSampleRate console.log(sampleRate) // Example: 44100 ``` #### `setPreferredSampleRate(sampleRate: number)` Set the preferred sample rate for audio input and output. ```typescript await SharedAudioSession.setPreferredSampleRate(48000) ``` *** ### 4. **Interruption Handling** #### `addInterruptionListener(listener: AudioSessionInterruptionListener)` Listen for audio interruptions. ```typescript SharedAudioSession.addInterruptionListener((type) => { if (type === 'began') { console.log('Audio interruption began') } else if (type === 'ended') { console.log('Audio interruption ended') } }) ``` #### `removeInterruptionListener(listener: AudioSessionInterruptionListener)` Remove an interruption listener. ```typescript SharedAudioSession.removeInterruptionListener(myListener) ``` *** ### 5. **Device Capabilities** #### `availableCategories` Get the list of audio session categories available on the device. ```typescript const categories = await SharedAudioSession.availableCategories console.log(categories) // Example: ['playback', 'record', 'soloAmbient'] ``` #### `availableModes` Get the list of audio session modes available on the device. ```typescript const modes = await SharedAudioSession.availableModes console.log(modes) // Example: ['default', 'videoChat', 'voiceChat'] ``` *** ### 6. **Additional Properties** #### `isOtherAudioPlaying` Check if other audio is currently playing on the device. ```typescript const isPlaying = await SharedAudioSession.isOtherAudioPlaying console.log(isPlaying) // Example: true ``` #### `secondaryAudioShouldBeSilencedHint` Check if secondary audio should be silenced. ```typescript const shouldSilence = await SharedAudioSession.secondaryAudioShouldBeSilencedHint console.log(shouldSilence) // Example: false ``` #### `allowHapticsAndSystemSoundsDuringRecording` Check if haptics and system sounds are allowed during recording. ```typescript const allowHaptics = await SharedAudioSession.allowHapticsAndSystemSoundsDuringRecording console.log(allowHaptics) // Example: true ``` #### `prefersNoInterruptionsFromSystemAlerts` Check if the session prefers no interruptions from system alerts. ```typescript const prefersNoInterruptions = await SharedAudioSession.prefersNoInterruptionsFromSystemAlerts console.log(prefersNoInterruptions) // Example: false ``` *** ### 7. **Session Activation** #### `setActive(active: boolean, options?: AudioSessionSetActiveOptions[])` Activate or deactivate the shared audio session with optional options. - `active`: Set to `true` to activate the session, `false` to deactivate it. - `options`: An array of optional activation options, such as 'interruptSpokenAudioAndMixWithOthers'. ```typescript await SharedAudioSession.setActive( true, ['notifyOthersOnDeactivation'] ) ``` *** ### 8. **System Settings** #### `setAllowHapticsAndSystemSoundsDuringRecording(value: boolean)` Enable or disable haptics and system sounds during recording. ```typescript await SharedAudioSession.setAllowHapticsAndSystemSoundsDuringRecording(true) ``` #### `setPrefersNoInterruptionsFromSystemAlerts(value: boolean)` Set the preference for no interruptions from system alerts. ```typescript await SharedAudioSession.setPrefersNoInterruptionsFromSystemAlerts(true) ``` *** ### 9. **Systemwide Output Volume** #### `outputVolume: number` The systemwide output volume. This property is a number between 0 and 1, representing the volume level as a percentage. #### outputVolume EventListener Type Definition: ```ts type AudioSessionOutputVolumeListener = (newValue: number, oldValue: number) => void ``` ##### `addOutputVolumeListener(listener: AudioSessionOutputVolumeListener)` Add an event listener for changes in the systemwide output volume. ```typescript SharedAudioSession.addOutputVolumeListener((newValue, oldValue) => { console.log(`Output volume changed from ${oldValue} to ${newValue}`) }) ``` ##### `removeOutputVolumeListener(listener: AudioSessionOutputVolumeListener)` Remove an event listener for changes in the systemwide output volume. *** ## Enumerations ### **AudioSessionSetActiveOptions** Optional activation options: - `'notifyOthersOnDeactivation'`: Notify other audio sessions when deactivating the shared audio session. ### **AudioSessionCategory** Defines the session's audio category: - `'ambient'`: Ambient audio, such as background music or ambient sounds. - `'multiRoute'`: Multi-route audio, such as routing distinct streams of audio data to different output devices at the same time. - `'playAndRecord'`: Play and record audio, such as voice chat or video conferencing. - `'playback'`: Playback audio, such as music or sound effects. - `'record'`: Recording audio, such as voice chat or video conferencing. - `'soloAmbient'`: Solo ambient audio, such as background music or ambient sounds. ### **AudioSessionCategoryOptions** Optional behaviors for audio categories: - `'mixWithOthers'`: Mix with other audio sessions. - `'duckOthers'`: Duck other audio sessions. - `'interruptSpokenAudioAndMixWithOthers'`: Interrupt spoken audio and mix with others. - `'allowBluetooth'`: Allow Bluetooth audio. - `'allowBluetoothA2DP'`: Allow Bluetooth A2DP audio. - `'allowAirPlay'`: Allow AirPlay audio. - `'defaultToSpeaker'`: Default to speaker, even if headphones are connected. - `'overrideMutedMicrophoneInterruption'`: Override muted microphone interruption. ### **AudioSessionMode** Specifies the session's mode: - `'default'`: Default mode. - `'gameChat'`: Game chat mode. - `'measurement'`: Measurement mode, such as audio input or output. - `'moviePlayback'`: Movie playback mode, such as movie content. - `'spokenAudio'`: Spoken audio mode, such as voice chat. - `'videoChat'`: Video chat mode, such as video conferencing. - `'videoRecording'`: Video recording mode, such as video conferencing. - `'voicePrompt'`: Voice prompt mode, such as text-to-speech. ### **AudioSessionInterruptionType** Specifies the type of interruption: - `'began'` - `'ended'` - `'unknown'` *** This interface offers extensive control over audio session management in Scripting, making it suitable for building audio-heavy script like music players and video conferencing tools. --- url: /doc_v2/guide/Changelog/2.4.6/VideoRecorder.md --- # VideoRecorder `VideoRecorder` provides a programmable video recording session in Scripting. It encapsulates camera selection, audio/video capture, encoding, pause/resume handling, zoom, focus, torch control, and final file writing. This API is intended for custom camera interfaces, video capture utilities, and automated recording workflows. *** ## Capabilities Overview - Front and back camera support - Explicit camera type selection (wide, ultra-wide, telephoto, etc.) - Multiple frame rates (24 / 30 / 60 / 120) - Optional audio recording - Multiple system capture session presets - Multiple video codecs (HEVC / H.264 / ProRes, etc.) - Pause and resume during recording - Independent focus and exposure control - Zoom and smooth zoom (ramp) control - Torch (flashlight) control - Deterministic state machine with callbacks - Explicit lifecycle management (`prepare / reset / dispose`) *** ## Type Definitions ### CameraPosition ```ts type CameraPosition = "front" | "back" ``` Represents the physical camera position. *** ### CameraType ```ts type CameraType = | "wide" | "ultraWide" | "telephoto" | "trueDepth" | "dual" | "dualWide" | "triple" ``` Represents the physical camera device type. Availability depends on the device hardware. *** ### VideoRecorderState ```ts type VideoRecorderState = | "idle" | "preparing" | "ready" | "recording" | "paused" | "finishing" | "finished" | "failed" ``` #### State Semantics | State | Meaning | | ----------- | ----------------------------------------------------- | | `idle` | Initial state, no resources configured | | `preparing` | Camera session and pipelines are being configured | | `ready` | Recorder is ready to start recording | | `recording` | Recording is in progress | | `paused` | Recording is paused | | `finishing` | Recording is stopping and file writing is in progress | | `finished` | Recording completed successfully | | `failed` | An error occurred | *** ### VideoCaptureSessionPreset ```ts type VideoCaptureSessionPreset = | "high" | "medium" | "low" | "cif352x288" | "vga640x480" | "iFrame960x540" | "iFrame1280x720" | "hd1280x720" | "hd1920x1080" | "hd4K3840x2160" ``` Defines the capture session resolution preset. *** ### VideoCodec ```ts type VideoCodec = | "hevc" | "h264" | "jpeg" | "JPEGXL" | "proRes4444" | "appleProRes4444XQ" | "proRes422" | "proRes422HQ" | "proRes422LT" | "proRes422Proxy" | "proResRAW" | "proResRAWHQ" | "hevcWithAlpha" ``` Specifies the video encoding format. Actual availability depends on system and hardware support. *** ### VideoOrientation ```ts type VideoOrientation = | "portrait" | "landscapeLeft" | "landscapeRight" ``` Specifies the output video orientation. *** ## Constructor ```ts new VideoRecorder(settings?) ``` ### settings ```ts { camera?: { position: CameraPosition preferredTypes?: CameraType[] } frameRate?: number audioEnabled?: boolean sessionPreset?: VideoCaptureSessionPreset videoCodec?: VideoCodec videoBitRate?: number orientation?: VideoOrientation mirrorFrontCamera?: boolean } ``` #### Parameter Details - **camera** - `position` Camera position. Defaults to `"back"`. - `preferredTypes` Preferred camera device types. If omitted, a suitable device is selected automatically based on position. - **frameRate** Target frame rate. Supported values: 24, 30, 60, 120. Defaults to 30. Actual frame rate depends on device capability. - **audioEnabled** Indicates whether audio is recorded. Defaults to `true`. - **sessionPreset** Capture session resolution preset. Defaults to `"high"`. - **videoCodec** Video encoding format. Defaults to `"hevc"`. - **videoBitRate** Average video bit rate in bits per second. If omitted, the system selects an appropriate value. - **orientation** Output video orientation. Defaults to `"portrait"`. - **mirrorFrontCamera** Indicates whether the front camera output is mirrored. Defaults to `true`. Only applies to the front camera. *** ## Read-Only Properties ### minZoomFactor ```ts readonly minZoomFactor: number ``` Minimum supported zoom factor for the active device. *** ### maxZoomFactor ```ts readonly maxZoomFactor: number ``` Maximum supported zoom factor for the active device. *** ### currentZoomFactor ```ts readonly currentZoomFactor: number ``` Current zoom factor. *** ### displayZoomFactor ```ts readonly displayZoomFactor: number ``` Zoom factor displayed to the user. *** ### hasTorch ```ts readonly hasTorch: boolean ``` Indicates whether the active camera supports a torch. *** ### torchMode ```ts readonly torchMode: "auto" | "on" | "off" ``` Current torch mode. *** ## State and Callbacks ### state ```ts state: VideoRecorderState ``` Represents the current state of the recorder. *** ### onStateChanged ```ts onStateChanged?: ( state: VideoRecorderState, details?: string ) => void ``` Invoked whenever the recorder state changes. - When `state === "failed"` `details` contains an error description. - When `state === "finished"` `details` contains the full output file path. *** ## Methods ### prepare() ```ts prepare(): Promise ``` Prepares the recorder by configuring the camera session and audio/video pipelines. #### Usage Constraints - Must be called before `startRecording` - Transitions state to `ready` on success - Transitions to `failed` on error *** ### startRecording(toPath) ```ts startRecording(toPath: string): void ``` Starts video recording. #### Parameters - **toPath** Full file path where the video will be saved. #### Usage Constraints - Only valid in the `ready` state - Transitions state to `recording` *** ### pauseRecording() ```ts pauseRecording(): void ``` Pauses the ongoing recording. #### Usage Constraints - Only valid in the `recording` state - Transitions state to `paused` - Timeline is compacted without introducing blank segments *** ### resumeRecording() ```ts resumeRecording(): void ``` Resumes a paused recording. #### Usage Constraints - Only valid in the `paused` state - Transitions state back to `recording` *** ### stopRecording() ```ts stopRecording(): Promise ``` Stops recording and finalizes the output file. #### Behavior - Transitions state to `finishing` - Transitions to `finished` after file writing completes - Final file path is delivered via `onStateChanged` *** ### reset() ```ts reset(): Promise ``` Resets the recorder state to allow a new recording session. #### Intended Use - After a completed recording - After a failed recording #### Behavior - Transitions state back to `idle` - `prepare` may be called again *** ### setTorchMode() ```ts setTorchMode(mode: "auto" | "off" | "on"): void ``` Sets the torch mode for the active camera. *** ### setFocusPoint() ```ts setFocusPoint(point: { x: number; y: number }): void ``` Sets the focus point. - Coordinates are normalized in the range `0.0 ~ 1.0` - `(0,0)` represents the top-left corner - `(1,1)` represents the bottom-right corner *** ### setExposurePoint() ```ts setExposurePoint(point: { x: number; y: number }): void ``` Sets the exposure point using the same coordinate system as focus. *** ### resetFocus() ```ts resetFocus(): void ``` Restores automatic focus mode. *** ### resetExposure() ```ts resetExposure(): void ``` Restores automatic exposure mode. *** ### setZoomFactor() ```ts setZoomFactor(factor: number): void ``` Immediately sets the zoom factor. - Value must be within `minZoomFactor` and `maxZoomFactor` *** ### rampZoomFactor() ```ts rampZoomFactor(toFactor: number, rate: number): void ``` Smoothly transitions the zoom factor. - `toFactor` specifies the target zoom factor - `rate` specifies the transition speed in powers of two per second *** ### resetZoom() ```ts resetZoom(): void ``` Resets the zoom factor to the default value (typically `1.0`). *** ### dispose() ```ts dispose(): Promise ``` Releases all resources and destroys the recorder. #### Usage Constraints - The instance cannot be used after disposal - Releases camera, audio, and system resources - Should be called when the recording lifecycle ends *** ## Typical Usage Flow ```ts const recorder = new VideoRecorder({ camera: { position: "back" }, frameRate: 60, videoCodec: "hevc" }) recorder.onStateChanged = (state, details) => { if (state === "finished") { console.log("Video saved at:", details) } } await recorder.prepare() recorder.startRecording("/path/to/tmp/demo.mov") ``` *** ## Usage Guidelines - Always observe `onStateChanged` to track state transitions - Do not start recording before calling `prepare` - Call `reset` before reusing the same instance for another recording - Call `dispose` when the recorder is no longer needed --- url: /doc_v2/guide/Changelog/2.4.6/onDrag and onDrop View Modifiers.md --- # onDrag and onDrop View Modifiers Scripting provides a Drag & Drop API closely aligned with the SwiftUI drag-and-drop interaction model. It enables views to act as drag sources, drop destinations, or both, supporting intra-app and cross-app drag-and-drop scenarios. The API is composed of three core parts: - **onDrag**: Declares a view as a drag source - **onDrop**: Declares a view as a drop destination - **DropInfo / ItemProvider / UTType**: Context objects describing drag content and state Drag and drop is a system-controlled interaction. Certain APIs are only valid during specific callbacks. These constraints are explicitly documented below and must be respected. *** ## Core Types ### DropInfo `DropInfo` represents the real-time state of a drag operation relative to a specific drop target view. It is only valid within `onDrop` callbacks. ### Properties #### location: Point - The current drag location - Expressed in the **local coordinate space of the drop view** - Commonly used for: - Insertion indicators - Reordering logic - Position-based highlighting ### Methods #### hasItemsConforming(types: UTType\[]): boolean - Indicates whether at least one dragged item conforms to any of the specified UTTypes - Commonly used in: - `validateDrop` - `dropEntered` - `dropUpdated` - This method performs capability checks only and does not load data #### itemProviders(types: UTType\[]): ItemProvider\[] - Returns all `ItemProvider` instances conforming to the specified UTTypes - **Only valid inside the `performDrop` callback** - After `performDrop` returns, access to the dragged data is revoked by the system > Critical constraint > You must **start loading the contents** of the returned `ItemProvider` instances **within the scope of `performDrop`**. > Loading may complete later, but it must be initiated synchronously before `performDrop` returns. *** ## DropOperation `DropOperation` describes the action a drop target intends to perform. Available values: - `"copy"` Copies the dragged data (default and most common) - `"move"` Moves the data instead of copying it (typically internal to the app) - `"cancel"` Cancels the drag operation and transfers no data - `"forbidden"` Explicitly disallows the drop at the current location `DropOperation` is usually returned from `dropUpdated` to dynamically control the drag behavior. *** ## DragDropProps `DragDropProps` defines the optional drag-and-drop capabilities that a view may adopt. *** ## onDrag ### Purpose Marks the view as a **drag source**, allowing the user to initiate a drag operation from it. ### Definition ```ts onDrag?: { data: () => ItemProvider preview: VirtualNode } ``` ### Parameters #### data ```ts data: () => ItemProvider ``` - Returns an `ItemProvider` describing the dragged data - Supports text, images, files, URLs, and custom types - Invoked each time a drag begins Recommended practice: Create a new `ItemProvider` instance for each drag operation. Do not reuse instances. #### preview ```ts preview: VirtualNode ``` - A view used as the drag preview - Rendered by the system as a floating representation during dragging - Centered over the source view by default *** ## onDrop ### Purpose Marks the view as a **drop destination** and provides fine-grained control over validation, interaction updates, and data handling. ### Definition ```ts onDrop?: { types: UTType[] validateDrop?: (info: DropInfo) => boolean dropEntered?: (info: DropInfo) => void dropUpdated?: (info: DropInfo) => DropOperation | null dropExited?: (info: DropInfo) => void performDrop: (info: DropInfo) => boolean } ``` *** ### onDrop.types ```ts types: UTType[] ``` - Declares the content types this view can accept - If the dragged content does not conform to any listed type: - The drop target does not activate - `validateDrop` is not called - Visual feedback is not shown *** ### validateDrop ```ts validateDrop?: (info: DropInfo) => boolean ``` - Determines whether the drop operation should be allowed to begin - Returning `false` immediately rejects the drag - Common use cases: - Checking item count - Enforcing application state constraints Default behavior: always returns `true` *** ### dropEntered ```ts dropEntered?: (info: DropInfo) => void ``` - Called when the drag enters the drop target area - Typically used to: - Show highlight states - Display insertion placeholders - Trigger animations *** ### dropUpdated ```ts dropUpdated?: (info: DropInfo) => DropOperation | null ``` - Called repeatedly as the drag moves within the drop target - Used to dynamically specify the intended `DropOperation` Return value behavior: - Returning a `DropOperation` updates the active operation - Returning `null`: - Reuses the last valid operation - Falls back to `"copy"` if none was previously returned *** ### dropExited ```ts dropExited?: (info: DropInfo) => void ``` - Called when the drag leaves the drop target area - Commonly used to clear highlight or placeholder UI *** ### performDrop ```ts performDrop: (info: DropInfo) => boolean ``` - **The most critical callback** - Indicates that the user has released the drag and data access is permitted - Return value: - `true` if the drop was successfully handled - `false` if the drop failed #### Mandatory constraints - Within this method, you must: - Call `info.itemProviders(...)` - Immediately initiate data loading from the returned providers - You must not: - Store `ItemProvider` references for later use - Defer loading to unrelated callbacks These constraints are enforced by the operating system for security reasons. *** ## Typical Interaction Flow 1. The user initiates a drag from an `onDrag` view 2. The system checks compatibility using `onDrop.types` 3. `validateDrop` is invoked 4. The drag enters the drop target → `dropEntered` 5. The drag moves within the target → repeated `dropUpdated` 6. The drag leaves the target → `dropExited` 7. The user releases the drag → `performDrop` 8. Data is loaded and processed *** ## Design Guidelines and Best Practices - Declare UTTypes as narrowly as possible - Use `"forbidden"` in `dropUpdated` to explicitly block invalid drops - Perform heavy parsing or processing only after `ItemProvider` loading completes - Prefer system-standard UTTypes (text, image, file, URL) for cross-app drag-and-drop --- url: /doc_v2/guide/Changelog/2.4.6/onDropContent.md --- # onDropContent `onDropContent` is a view modifier provided by Scripting that allows a view to act as a **drop target**, receiving files, images, or text dragged in from other applications. *** ## Overview With `onDropContent`, you can: - Receive drag-and-drop content from other apps - Restrict acceptable content using UTType identifiers - Track whether a drag operation is hovering over the view - Start loading dropped content through `ItemProvider` - Establish persistent access to security-scoped files when needed *** ## Modifier Definition ```ts onDropContent?: { types: UTType[] isTarget: { value: boolean onChanged: (value: boolean) => void } | Observable perform: (attachments: ItemProvider[]) => boolean } ``` *** ## Parameters ### types Specifies the list of content types that the view can accept, expressed as UTType strings. If the drag operation does not contain any of the specified types: - The view does not activate as a drop target - `isTarget` does not update - `perform` is not called Example: ```ts types: ["public.image", "public.movie"] ``` *** ### isTarget Indicates whether the drag operation is currently hovering over the view. - The value is `true` when the drag enters the view’s area - The value is `false` when the drag exits the area Two forms are supported: - Binding object form ```ts { value: boolean onChanged: (value: boolean) => void } ``` - Observable form ```ts Observable ``` The observable form works well with `useObservable` and provides a more concise reactive binding. *** ### perform Called when content matching the specified `types` is dropped onto the view. ```ts perform: (attachments: ItemProvider[]) => boolean ``` - `attachments` is an array of `ItemProvider` - Each `ItemProvider` represents one dropped item - The return value indicates whether the drop was successfully handled Return value semantics: - Return `true` to indicate the drop was accepted - Return `false` to indicate the drop was not handled *** ## Execution Rules for perform The following rules must be followed inside `perform`: - Loading of `ItemProvider` contents must be **started synchronously within the execution scope of `perform`** - Asynchronous completion is allowed using `Promise` or `then` - Loading must not be initiated later from a different callback or event - If `perform` returns `false`, the system treats the drop as unhandled Reasoning: - Dropped content is protected by system security rules - Access to the dropped payload is only valid while `perform` is executing - If loading does not begin within this scope, the content may no longer be accessible *** ## Working with ItemProvider Within `perform`, you should inspect each `ItemProvider` and start loading based on its capabilities. Typical steps include: - Checking type conformance using `hasItemConforming` - Selecting an appropriate loading method - Handling files, images, or text accordingly *** ## Example Usage ```tsx const isTarget = useObservable(false) return { const images: UIImage[] = [] const videos: string[] = [] let found = false for (const attachment of attachments) { if (attachment.hasItemConforming("public.png")) { found = true attachment.loadUIImage().then(image => { if (image != null) { images.push(image) } }) } else if (attachment.hasItemConforming("public.movie")) { found = true attachment.loadFilePath("public.movie").then(filePath => { if (filePath != null) { // Create a bookmark for the security-scoped file FileManager.addFileBookmark(filePath) videos.push(filePath) } }) } } return found } }} > ... ``` *** ## Security-Scoped File Access File paths obtained via `onDropContent` are typically **security-scoped resources**. These paths may become invalid when: - `perform` returns - The app restarts - The script lifecycle ends To retain long-term access, you should create a file bookmark as soon as the path is obtained. *** ## FileManager.addFileBookmark ```ts FileManager.addFileBookmark(path: string, name?: string): string | null ``` Description: - Creates a security-scoped bookmark for a file or folder - Intended for paths obtained via APIs such as `Photos` or `onDropContent` - Returns the bookmark name, or `null` if creation fails Example: ```ts const bookmarkName = FileManager.addFileBookmark(filePath) ``` *** ## FileManager.removeFileBookmark ```ts FileManager.removeFileBookmark(name: string): boolean ``` Description: - Removes a previously created file bookmark - Should be called when access to the file is no longer needed - Returns whether the removal was successful Example: ```ts FileManager.removeFileBookmark(bookmarkName) ``` *** ## Usage Recommendations - Specify `types` as precisely as possible - Use `perform` only to start loading, not to wait for results - Load images and lightweight data as objects when appropriate - Prefer file paths for large resources such as videos or documents - Create bookmarks for files that require long-term access - Remove bookmarks when the associated files are no longer needed --- url: /doc_v2/guide/Control Widget.md --- # Control Widget The `ControlWidget` API enables users to add custom Button or Toggle controls to the iOS Control Center or Lock Screen. Each control is linked to an `AppIntent` to execute custom script logic. The controls support privacy protection, dynamic state labels, and SFSymbols icons. *** ## Control Label Type ### `ControlWidgetLabel` Represents a label for a control, including the main label or value label in active/inactive state. | Property | Type | Description | | ------------------ | ---------- | ----------------------------------------------------------------- | | `title` | `string` | The main title of the label. | | `systemImage` | `string?` | Optional SFSymbol image name for the label. | | `privacySensitive` | `boolean?` | If `true`, the label content is hidden when the device is locked. | *** ## 1. `ControlWidgetButton` Renders a button control that executes a script intent when tapped. ```ts function ControlWidgetButton(props: ControlWidgetButtonProps): JSX.Element ``` ### `ControlWidgetButtonProps` | Property | Type | Description | | -------------------- | ----------------------------- | ------------------------------------------------------------------------------------------------------------------------ | | `privacySensitive` | `boolean?` | If `true`, the control's state and content are hidden when the device is locked. | | `intent` | `AppIntent` | The intent to be executed when the button is tapped. | | `label` | `ControlWidgetLabel` | The main label shown on the button. | | `activeValueLabel` | `ControlWidgetLabel \| null?` | The label shown when the button is active. Must be paired with `inactiveValueLabel`. Overrides `systemImage` in `label`. | | `inactiveValueLabel` | `ControlWidgetLabel \| null?` | The label shown when the button is inactive. Must be paired with `activeValueLabel`. Overrides `systemImage` in `label`. | > If either `activeValueLabel` or `inactiveValueLabel` is provided, both should be specified to ensure proper state display. *** ## 2. `ControlWidgetToggle` Renders a toggle control that updates a boolean value using a script intent. ```ts function ControlWidgetToggle(props: ControlWidgetToggleProps): JSX.Element ``` ### `ControlWidgetToggleProps` | Property | Type | Description | | -------------------- | ----------------------------- | ---------------------------------------------------------------------------------- | | `privacySensitive` | `boolean?` | If `true`, the control's state and content are hidden when the device is locked. | | `intent` | `AppIntent` | The intent to execute when toggled. The type `T` must extend `{ value: boolean }`. | | `label` | `ControlWidgetLabel` | The main label for the toggle. | | `activeValueLabel` | `ControlWidgetLabel \| null?` | Label displayed when toggle is ON. Must be paired with `inactiveValueLabel`. | | `inactiveValueLabel` | `ControlWidgetLabel \| null?` | Label displayed when toggle is OFF. Must be paired with `activeValueLabel`. | *** ## 3. `ControlWidget` Namespace ```ts namespace ControlWidget ``` ### `ControlWidget.parameter: string` A user-defined string parameter set during control configuration. Useful for targeting specific resources (e.g., a device ID or door ID). *** ### `ControlWidget.present(element: VirtualNode): void` Displays the control UI. Only `ControlWidgetButton` or `ControlWidgetToggle` elements are supported. #### Usage Notes: - `control_widget_button.tsx` must only render a `ControlWidgetButton`. - `control_widget_toggle.tsx` must only render a `ControlWidgetToggle`. - To hide the entire control content on the Lock Screen, use `privacySensitive` on the root. - To redact only specific labels or values, apply `privacySensitive` inside `ControlWidgetLabel`. #### Example: ```tsx /// app_intents.tsx export const ToggleDoorIntent = AppIntentManager.register({ name: "ToggleDoorIntent", protocol: AppIntentProtocol.AppIntent, perform: async ({ id, value }: { id: string; value: boolean }) => { await setDoorState(id, value) ControlWidget.reloadToggles() } }) /// control_widget_toggle.tsx async function run() { const doorId = ControlWidget.parameter || "default" const data = await fetchDoorData(doorId) ControlWidget.present( ) } run() ``` *** ### `ControlWidget.reloadButtons(): void` Reloads all control widget buttons. Useful when the intent result changes the UI state. *** ### `ControlWidget.reloadToggles(): void` Reloads all toggle widgets. Call this after a toggle action to update state. *** ## 4. Development Notes - Every control must be associated with an `AppIntent` to define its behavior. - Toggle controls must pass an intent with a parameter shape `{ value: boolean }`. - If using value labels (`activeValueLabel` / `inactiveValueLabel`), always provide both. - System images (`systemImage`) should follow [SF Symbols](https://developer.apple.com/sf-symbols/) naming conventions. - Use `ControlWidget.reloadButtons()` and `reloadToggles()` to force UI updates after state changes in the background. --- url: /doc_v2/guide/Custom Keyboard.md --- # Custom Keyboard The `CustomKeyboard` namespace provides a comprehensive API for building fully custom keyboard UIs in the Scripting app. It allows you to create JSX-based keyboards, insert or modify text, query input state, respond to user interaction, and control keyboard layout or navigation. *** ## 1. Environment & Setup ### Requirements - You must define your keyboard interface in a file named `**keyboard.tsx**` inside your script project. - The `CustomKeyboard` API is **only available in the keyboard extension environment**. - It is **not available** in App scripts, Intents (`intent.tsx`), or Widgets (`widget.tsx`). - You must enable the keyboard in iOS settings: ``` Settings > General > Keyboard > Keyboards > Add New Keyboard > Scripting ``` Then tap the **Scripting Keyboard** and enable **Allow Full Access** to unlock clipboard and network features. *** ## 2. Presentation ### `present(node: VirtualNode): void` Renders your custom keyboard UI using the given JSX node. This function **must be called once** in `keyboard.tsx`. ```tsx function MyKeyboard() { return Hello from keyboard } CustomKeyboard.present() ``` *** ## 3. Text Input State | Property | Type | Description | | ------------------ | ------------------------- | ---------------------------------------- | | `textBeforeCursor` | `Promise` | Text before the cursor | | `textAfterCursor` | `Promise` | Text after the cursor | | `selectedText` | `Promise` | Currently selected text | | `hasText` | `Promise` | Whether the text input contains any text | *** ## 4. Input Traits ### `useTraits(): TextInputTraits` Hook to retrieve the current input traits (e.g., keyboard type, return key style). It automatically updates when `textDidChange` or `selectionDidChange` events occur. ### `traits: TextInputTraits` A snapshot of the traits at the last change. Prefer `useTraits()` in JSX components for reactivity. #### Example fields: - `keyboardType`: `'default'`, `'numberPad'`, `'emailAddress'`... - `returnKeyType`: `'go'`, `'search'`, `'done'`... - `autocapitalizationType`: `'none'`, `'sentences'`, etc. - `textContentType`: semantic input hints like `'username'`, `'oneTimeCode'`, etc. - `keyboardAppearance`: `'light'`, `'dark'`, etc. *** ## 5. Text Manipulation ### `insertText(text: string): Promise` Insert text at the current cursor position. ### `deleteBackward(): Promise` Delete one character before the cursor. ### `moveCursor(offset: number): Promise` Move the cursor by a number of characters. Negative = left; Positive = right. ### `setMarkedText(text, location, length): Promise` Mark a portion of inserted text (used in composition scenarios like Pinyin input). ### `unmarkText(): Promise` Clear any currently marked text. *** ## 6. Keyboard Behavior Control ### `dismiss(): Promise` Dismiss the keyboard view. ### `nextKeyboard(): Promise` Switch to the next system keyboard. ### `requestHeight(height: number): Promise` Request a new keyboard height in points. Recommended range is **216–360pt**. ### `setHasDictationKey(value: boolean): Promise` Control whether the dictation (microphone) key is shown. ### `setToolbarVisible(visible: boolean): Promise` Show or hide the custom keyboard toolbar. Useful for debugging. *** ## 7. Navigation ### `dismissToHome(): Promise` Dismisses the currently active keyboard script and returns to the **Scripting keyboard home screen** (script list). Useful for letting users choose another script. ```ts await CustomKeyboard.dismissToHome() ``` *** ## 8. User Feedback ### `playInputClick(): void` Play the standard system keyboard click sound. Useful when simulating real key taps. ```ts CustomKeyboard.playInputClick() ``` *** ## 9. Event Listeners ### `addListener(event, callback): void` Register a listener for keyboard or text input changes. | Event | Callback Signature | Description | | --------------------- | ----------------------------------- | ------------------------------- | | `textWillChange` | `() => void` | Before text changes | | `textDidChange` | `(traits: TextInputTraits) => void` | After text changes | | `selectionWillChange` | `() => void` | Before cursor/selection changes | | `selectionDidChange` | `(traits: TextInputTraits) => void` | After cursor/selection changes | ### `removeListener(event, callback): void` Remove a specific listener. ### `removeAllListeners(event): void` Remove all listeners for a given event type. *** ## 10. Full Example ```tsx function MyKeyboard() { const traits = CustomKeyboard.useTraits() const insert = async (text: string) => { CustomKeyboard.playInputClick() await CustomKeyboard.insertText(text) } return ( Input type: {traits.keyboardType} `) await webView.present({ navigationTitle: 'WebView Demo' }) webView.dispose() ``` --- url: /doc_v2/guide/Intent/Intent.continueInForeground.md --- # Intent.continueInForeground `Intent.continueInForeground` is an API that leverages the **iOS 26+ AppIntents framework** to request the system to bring the **Scripting app** to the foreground while a Shortcut is running. This method is used when a script—invoked from Shortcuts—requires full UI interaction within the Scripting app (for example: presenting a form, editing content, picking files, showing a full screen navigation flow, etc.). When invoked: - The system displays a dialog asking the user to continue the workflow in the app. - If the user **confirms**, the system opens Scripting in the foreground and the script continues. - If the user **cancels**, the script terminates immediately. Because this is a system-level capability of AppIntents: **This API requires iOS 26 or later.** *** \##API Definition ```ts function continueInForeground( dialog?: Dialog | null, options?: { alwaysConfirm?: boolean; }, ): Promise; ``` ## Parameters ### `dialog?: Dialog | null` An optional message explaining why the workflow needs to continue in the foreground. `Dialog` supports four formats: ```ts type Dialog = | string | { full: string; supporting: string } | { full: string; supporting: string; systemImageName: string } | { full: string; systemImageName: string }; ``` Examples: ```ts "Do you want to continue in the app?"; ``` ```ts { full: "Continue in the Scripting app?", supporting: "The next step requires full UI interaction.", systemImageName: "app" } ``` Passing `null` will suppress the dialog entirely (not recommended unless you fully understand the UX implications). *** ### `options?: { alwaysConfirm?: boolean }` Controls whether the system should always ask for confirmation: - `alwaysConfirm: false` _(default)_ The system may decide whether confirmation is needed based on context. - `alwaysConfirm: true` The system always presents the confirmation dialog. *** \##Execution Behavior When called inside `intent.tsx`: 1. The Shortcut pauses execution. 2. The system presents a confirmation dialog. 3. If the user accepts: - The Scripting app opens in the foreground. - The script continues executing after the `await`. 4. If the user cancels: - The entire script is terminated immediately. This mirrors the behavior of Apple’s AppIntents `continueInApp()` functionality for system apps. *** \##Common Use Cases Use `continueInForeground` when the next step **cannot** run in the background, including: - Presenting a full-screen UI (`Navigation.present`) - Editing content in a custom form or navigation stack - Selecting files or interacting with UI components - Scenarios requiring user input or multi-step flows - Showing UI unavailable to background extensions It should **not** be used for simple data processing or non-interactive tasks. *** \##Full Code Example Below is the full working example demonstrating how `continueInForeground` enables a Shortcut to transfer execution into the Scripting app and then return UI input back to Shortcuts. ```tsx // intent.tsx import { Button, Intent, List, Navigation, NavigationStack, Script, Section, TextField, useState, } from "scripting"; function View() { const dismiss = Navigation.useDismiss(); const [text, setText] = useState(""); return (

); } async function runIntent() { // Step 1: Ask the user to continue in the foreground app await Intent.continueInForeground("Do you want to open the app and continue?"); // Step 2: Present UI inside the Scripting app const text = await Navigation.present(); // Step 3: Optionally go back to Shortcuts Safari.openURL("shortcuts://"); // Step 4: Return the result to Shortcuts Script.exit(Intent.text(text ?? "No text return")); } runIntent(); ``` *** \##Notes and Recommendations 1. **Requires iOS 26+** Do not call this API on older systems. 2. **Use dialogs to explain why foreground interaction is required** This improves user trust and Shortcuts clarity. 3. **Always handle the cancellation case** If the user cancels, your script stops. Avoid assuming foreground UI will always appear. 4. **Foreground UI must be meaningful** Only use this API when the upcoming step truly requires UI. 5. **Can be combined with SnippetIntent (iOS 26+)** For workflows that mix in-Shortcut Snippet UI with in-app full UI. *** \##Summary `Intent.continueInForeground` enables scripts invoked from Shortcuts to request foreground execution when UI interaction is required. It is: - Based on iOS 26 AppIntents capabilities - A system-confirmed context switch - Essential for workflows involving full UI interactions - Safely integrated via a structured `Dialog` system This method allows Scripting to support advanced automation flows that seamlessly transition between Shortcuts and the full Scripting app UI. --- url: /doc_v2/guide/Intent/Intent.requestConfirmation.md --- # Intent.requestConfirmation `Intent.requestConfirmation` pauses script execution and asks the user to confirm an action through a **system-managed confirmation UI**. The confirmation interface consists of: - A **SnippetIntent UI** (provided by you) - Optional dialog text (system-generated or developer-defined) Behavior: - If the user **confirms**, the script continues (Promise resolves). - If the user **cancels**, the script terminates immediately. - The UI is fully managed by the system. - The presented UI is defined by the provided SnippetIntent’s `perform()` return. **This API is only available on iOS 26 or later.** *** \##API Definition ```ts Intent.requestConfirmation( actionName: ConfirmationActionName, snippetIntent: AppIntent, options?: { dialog?: Dialog; showDialogAsPrompt?: boolean; } ): Promise ``` *** \##Parameter Details ## actionName: ConfirmationActionName A semantic keyword describing the type of action being confirmed. Apple uses this value to generate natural language around the confirmation UI. Accepted values include: ``` "add" | "addData" | "book" | "buy" | "call" | "checkIn" | "continue" | "create" | "do" | "download" | "filter" | "find" | "get" | "go" | "log" | "open" | "order" | "pay" | "play" | "playSound" | "post" | "request" | "run" | "search" | "send" | "set" | "share" | "start" | "startNavigation" | "toggle" | "turnOff" | "turnOn" | "view" ``` Examples: - `"set"` → “Do you want to set…?” - `"buy"` → “Do you want to buy…?” - `"toggle"` → “Do you want to toggle…?” Choosing the correct semantic verb improves the clarity of the user-facing dialog. *** ## snippetIntent: SnippetIntent This must be an AppIntent registered with: ```ts protocol: AppIntentProtocol.SnippetIntent; ``` The UI displayed in the confirmation step **comes from this SnippetIntent’s `perform()` return**, which must be a TSX-based `VirtualNode`. This is what the user sees and interacts with during confirmation. *** ## `options?: { dialog?: Dialog; showDialogAsPrompt?: boolean }` ### dialog?: Dialog Optional text describing the confirmation request. Supports four formats: ```ts type Dialog = | string | { full: string; supporting: string } | { full: string; supporting: string; systemImageName: string } | { full: string; systemImageName: string }; ``` Examples: ```ts "Are you sure you want to continue?"; ``` More structured version: ```ts { full: "Set this color?", supporting: "This will update the theme color used across the app.", systemImageName: "paintpalette" } ``` Use this to clearly explain what the user is confirming. *** ### showDialogAsPrompt?: boolean - Default: `true` The system shows the dialog as a modal prompt. - `false` The dialog may be integrated directly inside the Snippet card instead of a separate prompt. *** \##Execution Flow When the script executes: ```ts await Intent.requestConfirmation(...) ``` The following occurs: 1. Script execution is paused. 2. The system displays: - The SnippetIntent UI - Optional dialog text 3. The user chooses: - **Confirm** → Promise resolves → script continues - **Cancel** → script stops immediately 4. The system handles UI presentation and dismissal automatically. There is no need to manually manage the UI lifecycle. *** \##Usage Scenarios Recommended for: - Confirming important changes (colors, appearance, configurations) - Confirming destructive or irreversible actions - Steps requiring explicit user approval - Initiating subflows requiring UI preview or choice (e.g., color picker, item selector) - Sensitive operations (e.g., updating settings, performing actions with side effects) Not recommended for: - Actions that do not require user approval - Simple background data processing *** \##Complete Example Below is a full working example demonstrating how to request user confirmation using a SnippetIntent. It assumes you have two SnippetIntent AppIntents: - `PickColorIntent` — allows user to select a color - `ShowResultIntent` — displays the final result ## intent.tsx ```tsx import { Intent, Script } from "scripting"; import { PickColorIntent, ShowResultIntent } from "./app_intents"; async function runIntent() { // Step 1: Ask the user to confirm the action via a Snippet UI await Intent.requestConfirmation("set", PickColorIntent(), { dialog: { full: "Are you sure you want to set this color?", supporting: "This will update the theme color used by your app.", systemImageName: "paintpalette", }, }); // Step 2: Read input from Shortcuts const text = Intent.shortcutParameter?.type === "text" ? Intent.shortcutParameter.value : "No text parameter from Shortcuts"; // Step 3: Return another SnippetIntent result const snippet = Intent.snippetIntent({ snippetIntent: ShowResultIntent({ content: text }), }); Script.exit(snippet); } runIntent(); ``` *** \##Notes & Best Practices - **Requires iOS 26+** — do not call this API on earlier versions. - Always include a clear **dialog** message to improve user understanding. - Use for actions that require explicit approval or confirmation. - When possible, combine with SnippetIntent to provide a richer preview UI. - Scripts terminate automatically when the user cancels; do not rely on cleanup code afterward. - Avoid calling it unnecessarily; only use when confirmation is truly meaningful. --- url: /doc_v2/guide/Intent/Quick Start.md --- # Quick Start Scripting allows you to define custom iOS Intents using an `intent.tsx` file. These scripts can receive input from the iOS share sheet or the Shortcuts app and return structured results. With optional UI presentation, you can create interactive workflows that process data and deliver output dynamically. *** ## 1. Creating and Configuring an Intent ### 1.1 Create an Intent Script 1. Create a new script project in the Scripting app. 2. Add a file named `intent.tsx` to the project. 3. Define your logic and optionally a UI component inside the file. ### 1.2 Configure Supported Input Types Tap the project title in the editor’s title bar to open **Intent Settings**, then select supported input types: - Text - Images - File URLs - URLs This configuration enables your script to appear in the share sheet or Shortcuts when matching input is provided. *** ## 2. Accessing Input Data Inside `intent.tsx`, use the `Intent` API to access input values. | Property | Description | | -------------------------- | ----------------------------------------------------------------------------------- | | `Intent.shortcutParameter` | A single parameter passed from the Shortcuts app, with `.type` and `.value` fields. | | `Intent.textsParameter` | Array of text strings. | | `Intent.urlsParameter` | Array of URL strings. | | `Intent.imagesParameter` | Array of image file paths (UIImage objects). | | `Intent.fileURLsParameter` | Array of local file URL paths. | Example: ```ts if (Intent.shortcutParameter) { if (Intent.shortcutParameter.type === "text") { console.log(Intent.shortcutParameter.value) } } ``` *** ## 3. Returning a Result Use `Script.exit(result)` to return a result to the caller, such as the Shortcuts app or another script. Valid return types include: - Plain text: `Intent.text(value)` - Attributed text: `Intent.attributedText(value)` - URL: `Intent.url(value)` - JSON: `Intent.json(value)` - File path or file URL: `Intent.file(value)` or `Intent.fileURL(value)` Example: ```ts import { Script, Intent } from "scripting" Script.exit(Intent.text("Done")) ``` *** ## 4. Displaying Interactive UI Use `Navigation.present()` to show a UI before returning a result. You can render a React-style component and then call `Script.exit()` after the interaction completes. Example: ```ts import { Intent, Script, Navigation, VStack, Text } from "scripting" function MyIntentView() { return ( {Intent.textsParameter?.[0]} ) } async function run() { await Navigation.present({ element: }) Script.exit() } run() ``` *** ## 5. Using Intents in the Share Sheet If a script supports a specific input type (e.g., text or image), it will automatically appear as an option in the iOS share sheet: 1. Select content such as text or a file. 2. Tap the Share button. 3. Choose **Scripting** in the share sheet. 4. Scripting will list scripts that support the selected input type. *** ## 6. Using Intents in the Shortcuts App You can call scripts from the Shortcuts app with or without UI: - **Run Script**: Executes the script in the background. - **Run Script in App**: Executes the script in the foreground, with UI presentation support. Steps: 1. Open the Shortcuts app and create a new shortcut. 2. Add the **Run Script** or **Run Script in App** action from Scripting. 3. Choose the target script and pass input parameters if needed. *** ## 7. Intent API Reference ### `Intent` Properties | Property | Type | Description | | ------------------- | ------------------- | ----------------------------------------------- | | `shortcutParameter` | `ShortcutParameter` | Input from Shortcuts with `.type` and `.value`. | | `textsParameter` | `string[]` | Array of input text values. | | `urlsParameter` | `string[]` | Array of input URLs. | | `imagesParameter` | `UIImage[]` | Array of image file paths or objects. | | `fileURLsParameter` | `string[]` | Array of input file paths (local file URLs). | ### `Intent` Methods | Method | Return Type | Example | | ------------------------------ | --------------------------- | -------------------------------------- | | `Intent.text(value)` | `IntentTextValue` | `Intent.text("Hello")` | | `Intent.attributedText(value)` | `IntentAttributedTextValue` | `Intent.attributedText("Styled Text")` | | `Intent.url(value)` | `IntentURLValue` | `Intent.url("https://example.com")` | | `Intent.json(value)` | `IntentJsonValue` | `Intent.json({ key: "value" })` | | `Intent.file(path)` | `IntentFileValue` | `Intent.file("/path/to/file.txt")` | | `Intent.fileURL(path)` | `IntentFileURLValue` | `Intent.fileURL("/path/to/file.pdf")` | | `Intent.image(UIImage)` | `IntentImageValue` | `Intent.image(uiImage)` | *** ## 8. Best Practices and Notes - Always call `Script.exit()` to properly terminate the script and return a result. - When displaying a UI, ensure `Navigation.present()` is awaited before calling `Script.exit()`. - Use **"Run Script in App"** for large files or images to avoid process termination due to memory constraints. - You can use `queryParameters` when launching scripts via URL scheme if additional data is needed. --- url: /doc_v2/guide/Intent/SnippetIntent.md --- # SnippetIntent SnippetIntent is a special kind of AppIntent whose purpose is to render **interactive Snippet UI cards** inside the Shortcuts app (iOS 26+). Key characteristics: 1. Must be registered in `app_intents.tsx` 2. Must specify `protocol: AppIntentProtocol.SnippetIntent` 3. `perform()` **must return a VirtualNode (TSX UI)** 4. Must be returned via `Intent.snippetIntent()` 5. Must be invoked from the Shortcuts action **“Show Snippet Intent”** 6. SnippetIntent is ideal for building interactive, step-based UI inside a Shortcut It is not a data-returning Intent; it is exclusively for UI rendering in Shortcuts. *** \##2. System Requirements **SnippetIntent requires iOS 26 or later.** On iOS versions earlier than 26: - `Intent.snippetIntent()` is not available - `Intent.requestConfirmation()` cannot be used - The Shortcuts action “Show Snippet Intent” does not exist - SnippetIntent-type AppIntents cannot be invoked by Shortcuts *** \##3. Registering a SnippetIntent (app\_intents.tsx) Example: ```tsx export const PickColorIntent = AppIntentManager.register({ name: "PickColorIntent", protocol: AppIntentProtocol.SnippetIntent, perform: async () => { return ; }, }); ``` Another SnippetIntent: ```tsx export const ShowResultIntent = AppIntentManager.register({ name: "ShowResultIntent", protocol: AppIntentProtocol.SnippetIntent, perform: async ({ content }: { content: string }) => { return ; }, }); ``` Requirements: - `protocol` **must** be `AppIntentProtocol.SnippetIntent` - `perform()` **must** return a TSX UI (VirtualNode) - SnippetIntent cannot return non-UI types such as text, numbers, JSON, or file paths *** \##4. Wrapping SnippetIntent Return Values — `Intent.snippetIntent` A SnippetIntent cannot be passed directly to `Script.exit()`. It must be wrapped in a `IntentSnippetIntentValue`. ```tsx const snippetValue = Intent.snippetIntent(ShowResultIntent({ content: "Example Text" })); Script.exit(snippetValue); ``` ### Type Definition ```ts type SnippetIntentValue = { value?: | IntentAttributedTextValue | IntentFileURLValue | IntentJsonValue | IntentTextValue | IntentURLValue | IntentFileValue | null; snippetIntent: AppIntent; }; declare class IntentSnippetIntentValue extends IntentValue<"SnippetIntent", SnippetIntentValue> { value: SnippetIntentValue; type: "SnippetIntent"; } ``` This wrapper makes the return value compatible with the Shortcuts “Show Snippet Intent” action. *** \##5. Snippet Confirmation UI — `Intent.requestConfirmation` iOS 26 Snippet Framework provides built-in confirmation UI driven by SnippetIntent. ### API ```ts Intent.requestConfirmation( actionName: ConfirmationActionName, intent: AppIntent, options?: { dialog?: Dialog; showDialogAsPrompt?: boolean; } ): Promise ``` ### ConfirmationActionName A predefined list of semantic action names used by system UI: ``` "add" | "addData" | "book" | "buy" | "call" | "checkIn" | "continue" | "create" | "do" | "download" | "filter" | "find" | "get" | "go" | "log" | "open" | "order" | "pay" | "play" | "playSound" | "post" | "request" | "run" | "search" | "send" | "set" | "share" | "start" | "startNavigation" | "toggle" | "turnOff" | "turnOn" | "view" ``` ### Example ```tsx await Intent.requestConfirmation("set", PickColorIntent()); ``` Execution behavior: - Displays a Snippet UI for confirmation - If the user confirms → Promise resolves and script continues - If the user cancels → execution stops (system-driven behavior) *** \##6. The “Show Snippet Intent” Action in Shortcuts (iOS 26+) iOS 26 adds a new Shortcuts action: **Show Snippet Intent** This action is the only correct way to display SnippetIntent UI. ### Comparison with Other Scripting Actions | Shortcuts Action | UI Shown | Supports SnippetIntent | Usage | | ----------------------------- | ------------------------------ | ---------------------- | ------------------- | | Run Script | None | No | Background logic | | Run Script in App | Fullscreen UI inside Scripting | No | Rich app-level UI | | Show Snippet Intent (iOS 26+) | Snippet card UI | Yes | SnippetIntent flows | ### Usage 1. Add “Show Snippet Intent” in Shortcuts 2. Select a Scripting script project 3. The script must return `Intent.snippetIntent(...)` 4. Shortcuts renders the UI in a Snippet card *** \##7. IntentMemoryStorage — Cross-Intent State Store ## Why It Exists Every AppIntent execution runs in an isolated environment: - After an AppIntent `perform()` completes → its execution context is destroyed - After a script calls `Script.exit()` → the JS context is destroyed This means local variables **cannot persist between AppIntent calls**. Snippet flows commonly involve: PickColor → SetColor → ShowResult Therefore a cross-Intent state mechanism is required. *** ## IntentMemoryStorage API ```ts namespace IntentMemoryStorage { function get(key: string): T | null; function set(key: string, value: any): void; function remove(key: string): void; function contains(key: string): boolean; function clear(): void; function keys(): string[]; } ``` ### Purpose - Store small pieces of shared data across multiple AppIntents - Works during the entire Shortcut flow - Ideal for selections, temporary configuration, or intent-to-intent handoff ### Example ```ts IntentMemoryStorage.set("color", "systemBlue"); const color = IntentMemoryStorage.get("color"); ``` ### Guidelines Not recommended for large data. For large data: - Use `Storage` (persistent key-value store) - Or save files via `FileManager` in `appGroupDocumentsDirectory` IntentMemoryStorage should be treated as **temporary, lightweight state**. *** \##8. Full Example Combining All Features (iOS 26+) ## app\_intents.tsx ```tsx export const SetColorIntent = AppIntentManager.register({ name: "SetColorIntent", protocol: AppIntentProtocol.AppIntent, perform: async (color: Color) => { IntentMemoryStorage.set("color", color); }, }); export const PickColorIntent = AppIntentManager.register({ name: "PickColorIntent", protocol: AppIntentProtocol.SnippetIntent, perform: async () => { return ; }, }); export const ShowResultIntent = AppIntentManager.register({ name: "ShowResultIntent", protocol: AppIntentProtocol.SnippetIntent, perform: async ({ content }: { content: string }) => { const color = IntentMemoryStorage.get("color") ?? "systemBlue"; return ; }, }); ``` ## intent.tsx ```tsx async function runIntent() { // 1. Ask the user to confirm setting the color via Snippet await Intent.requestConfirmation("set", PickColorIntent()); // 2. Read Shortcuts input const textContent = Intent.shortcutParameter?.type === "text" ? Intent.shortcutParameter.value : "No text parameter from Shortcuts"; // 3. Create final SnippetIntent UI const snippetIntentValue = Intent.snippetIntent({ snippetIntent: ShowResultIntent({ content: textContent }), }); Script.exit(snippetIntentValue); } runIntent(); ``` ## Shortcuts Flow 1. User provides text 2. “Show Snippet Intent” runs the script 3. Script displays PickColorIntent confirmation UI via requestConfirmation 4. After confirmation, displays ShowResultIntent Snippet UI 5. Uses IntentMemoryStorage to persist the selected color *** \##9. Summary This document introduces all **new** Scripting features added for iOS 26+: 1. **SnippetIntent** - Registered using `AppIntentManager` - Returns TSX UI - Requires iOS 26+ 2. **Intent.snippetIntent** - Wraps a SnippetIntent for Script.exit 3. **Intent.requestConfirmation** - Presents a confirmation Snippet UI - Requires SnippetIntent 4. **“Show Snippet Intent” action in Shortcuts** - Required to display SnippetIntent UI 5. **IntentMemoryStorage** - Lightweight cross-AppIntent storage - Not suitable for large binary/content data - Complements multi-step Snippet flows --- url: /doc_v2/guide/Interactive Widget and LiveActivity.md --- # Interactive Widget and LiveActivity The **Scripting** app supports adding interactivity to **widgets** and **LiveActivity**, allowing you to create dynamic and interactive UIs using `Button` and `Toggle` components. These controls can execute **AppIntents** to trigger actions, making your widgets and live activities more powerful. *** ## 1. Introduction to AppIntents ### What are AppIntents? An **AppIntent** defines a specific action that can be triggered by a control (e.g., a `Button` or `Toggle`) in a widget or LiveActivity UI. AppIntents enable seamless interaction and functionality by linking UI components with executable logic. ### Supported Protocols AppIntents can implement the following protocols: - **`AppIntent`**: General-purpose intents for triggering custom actions. - **`AudioPlaybackIntent`**: Handles audio playback (e.g., play, pause, or toggle audio states). - **`AudioRecordingIntent`**: Manages audio recording states (requires iOS 18+ and a LiveActivity to stay active during recording). - **`LiveActivityIntent`**: Modifies or manages LiveActivity states. *** ## 2. Registering an AppIntent To use an **AppIntent**, it must first be registered in the `app_intents.tsx` file using the `AppIntentManager.register` method. ### Example: Registering AppIntents ```typescript // app_intents.tsx import { AppIntentManager, AppIntentProtocol } from "scripting" // Register an AppIntent const IntentWithoutParams = AppIntentManager.register({ name: "IntentWithoutParams", protocol: AppIntentProtocol.AppIntent, perform: async (params: undefined) => { // Perform a custom action console.log("Intent triggered") // Optionally reload widgets Widget.reloadAll() } }) // Register an AppIntent with parameters const ToggleIntentWithParams = AppIntentManager.register({ name: "ToggleIntentWithParams", protocol: AppIntentProtocol.AudioPlaybackIntent, perform: async (audioName: string) => { // Perform action based on the parameter console.log(`Toggling audio playback for: ${audioName}`) Widget.reloadAll() } }) ``` *** ## 3. Using AppIntents in Widgets or LiveActivity UIs After registering an AppIntent, it can be linked to interactive components like `Button` and `Toggle` in your `widget.tsx` or LiveActivity UI file. ### Example: Using AppIntents in a Widget ```typescript // widget.tsx import { VStack, Button, Toggle } from "scripting" import { IntentWithoutParams, ToggleIntentWithParams } from "./app_intents" import { model } from "./model" function WidgetView() { return ( ] }} trailingSwipeActions={{ actions: [ , ] }} /> )} } async function run() { await Navigation.present({ element: }) Script.exit() } run() ``` --- url: /doc_v2/guide/View Modifiers/Symbol Style.md --- # Symbol Style These modifiers allow you to customize how **SF Symbols** are displayed and animated inside views, particularly with the `Image` component. *** ## `symbolRenderingMode` Sets the **rendering mode** for symbol images within the view. ### Type ```ts symbolRenderingMode?: SymbolRenderingMode ``` ### Options (`SymbolRenderingMode`) - `"monochrome"` – A single-color version using the foreground style - `"hierarchical"` – Multiple layers with different opacities for depth (good for semantic coloring) - `"multicolor"` – Uses the symbol's built-in colors - `"palette"` – Allows layered tinting (like using multiple `foregroundStyle` layers) ### Example ```tsx ``` ### Explanation: - `symbolRenderingMode="palette"` tells the system to render the symbol in **multiple layered styles**. - `foregroundStyle` now uses an object with `primary`, `secondary`, and optionally `tertiary` layers to color those symbol layers individually. > This matches SwiftUI's behavior with `.symbolRenderingMode(.palette)` and `.foregroundStyle(primary, secondary, tertiary)`. *** ## `symbolVariant` Displays the symbol with a particular **visual variant**. ### Type ```ts symbolVariant?: SymbolVariants ``` ### Options (`SymbolVariants`) - `"none"` – Default symbol with no variant - `"circle"` – Encapsulated in a circle - `"square"` – Encapsulated in a square - `"rectangle"` – Encapsulated in a rectangle - `"fill"` – Filled symbol - `"slash"` – Adds a slash over the symbol (often used to indicate "off" states) ### Example ```tsx ``` *** ## `symbolEffect` Applies a **symbol animation effect** to the view. This can include transitions (appear/disappear), scale, bounce, rotation, breathing, pulsing, and wiggle effects. You can also bind the effect to a value so it animates when the value changes. ### Type ```ts symbolEffect?: SymbolEffect ``` There are two forms of usage: *** ### 1. **Simple effects** (transition, scale, etc.) You can directly assign a symbol effect name: #### Examples ```tsx ``` *** ### 2. **Value-bound discrete effects** These effects animate when the associated value changes. #### Type ```ts symbolEffect?: { effect: DiscreteSymbolEffect value: string | number | boolean } ``` #### Example ```tsx ``` In this example, each time `isFavorited` changes, the bounce animation is triggered. *** ## Available Discrete Effects (`DiscreteSymbolEffect`) These effects can be bound to values: | Category | Effects | | ----------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | **Bounce** | `bounce`, `bounceByLayer`, `bounceDown`, `bounceUp`, `bounceWholeSymbol` | | **Breathe** | `breathe`, `breatheByLayer`, `breathePlain`, `breathePulse`, `breatheWholeSymbol` | | **Pulse** | `pulse`, `pulseByLayer`, `pulseWholeSymbol` | | **Rotate** | `rotate`, `rotateByLayer`, `rotateClockwise`, `rotateCounterClockwise`, `rotateWholeSymbol` | | **VariableColor** | `variableColor`, `variableColorCumulative`, `variableColorDimInactiveLayers`, `variableColorHideInactiveLayers`, `variableColorIterative` | | **Wiggle** | `wiggle`, `wiggleByLayer`, `wiggleWholeSymbol`, `wiggleLeft`, `wiggleRight`, `wiggleUp`, `wiggleDown`, `wiggleForward`, `wiggleBackward`, `wiggleClockwise`, `wiggleCounterClockwise` | *** ## Full Example ```tsx ``` This image uses: - a hierarchical rendering mode - a circular variant around the symbol - a pulsing animation bound to `isNotified` state *** ## Summary | Modifier | Description | | --------------------- | ----------------------------------------------------------------------- | | `symbolRenderingMode` | Sets how SF Symbols are rendered (monochrome, multicolor, etc.) | | `symbolVariant` | Applies a visual variant like `fill`, `circle`, or `slash` | | `symbolEffect` | Adds visual animation effects; can be static or bound to a state change | --- url: /doc_v2/guide/View Modifiers/Text Field.md --- # Text Field The following modifiers customize the behavior and appearance of `TextField` components. These allow you to control keyboard behavior, input handling, and submission logic. *** ## `onSubmit` Adds an action to perform when the user submits a value from the text field. ### Type ```ts onSubmit?: (() => void) | { triggers: SubmitTriggers action: () => void } ``` ### Behavior - If provided as a function: ```tsx console.log('Submitted')} /> ``` This is equivalent to: ```tsx console.log('Submitted') }} /> ``` - You can explicitly define what kind of submission should trigger the action using the `triggers` option: ```tsx console.log('Search submitted') }} /> ``` ### `SubmitTriggers` values: - `"text"`: Triggered by text input views like `TextField`, `SecureField`, etc. - `"search"`: Triggered by search fields (e.g., those using the `searchable` modifier). *** ## `keyboardType` Specifies the type of keyboard to display when the text field is focused. ### Type ```ts keyboardType?: KeyboardType ``` ### Options - `'default'` - `'numberPad'` - `'phonePad'` - `'namePhonePad'` - `'URL'` - `'decimalPad'` - `'asciiCapable'` - `'asciiCapableNumberPad'` - `'emailAddress'` - `'numbersAndPunctuation'` - `'twitter'` - `'webSearch'` ### Example ```tsx ``` *** ## `autocorrectionDisabled` Controls whether the system autocorrection is enabled. ### Type ```ts autocorrectionDisabled?: boolean ``` ### Default - `true` — autocorrection is disabled by default. ### Example ```tsx ``` *** ## `textInputAutocapitalization` Sets how the text input system should automatically capitalize text. ### Type ```ts textInputAutocapitalization?: TextInputAutocapitalization ``` ### Options - `"never"` – No capitalization. - `"characters"` – All letters capitalized. - `"sentences"` – First letter of each sentence capitalized. - `"words"` – First letter of each word capitalized. ### Example ```tsx ``` *** ## `submitScope` Prevents submission triggers from this view from propagating upward to parent views with submission handlers. ### Type ```ts submitScope?: boolean ``` ### Default - `false` — submission actions bubble up by default. ### Example ```tsx ``` This ensures that `onSubmit` handlers defined higher up in the view hierarchy won’t be called when this field is submitted. ## `submitLabel` Sets the label for the submit button. ### Type ```ts submitLabel?: "continue" | "return" | "send" | "go" | "search" | "join" | "done" | "next" | "route" ``` ### Example ```tsx ``` --- url: /doc_v2/guide/View Modifiers/Text View Modifiers.md --- # Text View Modifiers The following properties allow you to style and format text-based views, such as `Text` or `Label`, in ways that closely mirror SwiftUI’s built-in modifiers. By customizing these properties, you can control the font, weight, design, spacing, and other typographic attributes of the displayed text. ## Overview These properties are generally passed to text-related views like `Text` or `Label` components as attributes. For example, you can specify a font size, enable bold formatting, or add an underline with a custom color—all without manually calling multiple modifiers. ```tsx Stylish Text Here ``` In the example above, the text uses a custom font, semibold weight, italic style, a red underline, limits to two lines, and centers the text. *** ## Font Configuration ### `font` Defines the font and size to apply to the text. - **Number**: When you provide a number (e.g., `14`), it applies the system font at that size. - **Preset Font Name** (`Font` type): Use one of the built-in text styles (`"largeTitle"`, `"title"`, `"headline"`, `"subheadline"`, `"body"`, `"callout"`, `"footnote"`, `"caption"`). The system determines the size and weight based on that style. - **Object with name and size**: Apply a custom font by specifying the `name` and `size`. ```tsx System Font, Size 20 System Headline Font Custom Font ``` *** ### `fontWeight` Sets the thickness of the font’s stroke. Options range from `"ultraLight"` to `"black"`. ```tsx Bold Text ``` *** ### `fontWidth` Specifies the width variant of the font if available. Possible values include `"compressed"`, `"condensed"`, `"expanded"`, and `"standard"`. You can also use a numeric value if supported. ```tsx Condensed Width Font ``` *** ### `fontDesign` Modifies the font design. Options include `"default"`, `"monospaced"`, `"rounded"`, `"serif"`. ```tsx Rounded Font Design ``` *** ## Text Formatting ### `minScaleFactor` A number between 0 and 1 that indicates how much the text can shrink if it doesn’t fit the available space. For example, `0.5` means the text can shrink down to 50% of its original size to fit. ```tsx This text shrinks slightly if it doesn't fit. ``` *** ### `bold` Applies a bold font weight if `true`. ```tsx This text is bold ``` *** ### `baselineOffset` Adjusts the text’s vertical position relative to its baseline. Positive values move the text up, negative values move it down. ```tsx Text shifted up ``` *** ### `kerning` Controls the spacing between characters. A positive value increases spacing; a negative value decreases it. ```tsx Extra spaced text ``` *** ### `italic` Applies an italic style if `true`. ```tsx Italic text ``` *** ### `monospaced` Forces all child text to use a monospaced variant, if available. ```tsx Monospaced text ``` *** ### `monospacedDigit` Uses fixed-width digits while leaving other characters as they are. This helps align numbers vertically, useful for tables or timers. ```tsx Digits aligned in monospace 1234 ``` *** ## Text Decorations ### `strikethrough` Applies a strikethrough (line through the text). You can provide a color, or an object specifying a pattern and color. - **Color only**: `strikethrough="red"` - **Object**: `strikethrough={{ pattern: 'dash', color: 'blue' }}` ```tsx Strikethrough text in gray Dotted red strikethrough ``` *** ### `underline` Applies an underline in a similar way to `strikethrough`. - **Color only**: `underline="blue"` - **Object**: `underline={{ pattern: 'dashDot', color: 'green' }}` ```tsx Underlined text in blue Dotted pink underline ``` *** ## Line & Layout Control ### `lineLimit` Specifies how many lines of text can display. You can provide: - A single number for a maximum line limit. - An object `{ min?: number; max: number; reservesSpace?: boolean }` to specify a minimum and maximum number of lines, and whether the text should reserve space for all those lines even when not used. ```tsx This text will be truncated if it doesn't fit on one line. This text can display between 2 and 4 lines, and always reserves space for 4 lines, preventing layout shifts. ``` *** ### `lineSpacing` Sets the spacing between lines, in pixels. ```tsx Line spacing set to 5 pixels ``` *** ### `multilineTextAlignment` Sets the text alignment for multi-line text: `"leading"`, `"center"`, or `"trailing"`. ```tsx This text is centered across multiple lines. ``` *** ### `truncationMode` Specifies how to truncate a line of text when it is too long to fit within the available horizontal space. #### Type ```ts type TruncationMode = "head" | "middle" | "tail" ``` #### Description Defines the position at which the text is truncated: - `"head"`: Truncates the beginning of the line, preserving the end. - `"middle"`: Truncates the middle of the line, preserving both the beginning and end. - `"tail"`: Truncates the end of the line, preserving the beginning. ```tsx This is a very long piece of text that may be truncated. ``` *** ### `allowsTightening?: boolean` Determines whether the system is allowed to reduce the spacing between characters to fit the text within a line when needed. #### Type `boolean` #### Default `false` #### Description When set to true, the system may compress the character spacing to avoid truncation and better fit the content. This is typically used to improve layout responsiveness in constrained environments. ```tsx Condensed text if necessary ``` *** ## Summary By combining these properties, you can fully control the typography of your text-based views without needing multiple wrapper components or modifiers. Whether you need a bold, italic headline font with custom kerning and underline, or a simple body font that truncates after two lines, these options cover a broad range of text styling needs. --- url: /doc_v2/guide/View Modifiers/Toast.md --- # Toast The `toast` view modifier displays a temporary notification message (toast) over the current view. It is typically used to show short feedback messages such as “Saved successfully,” “Action completed,” or “Network error.” You can show a simple text message or provide a fully custom view as the toast’s content. You can also control its duration, position, background color, text color, corner radius, and shadow style. *** ## Type Definition ```ts toast?: { duration?: number | null position?: "top" | "bottom" | "center" backgroundColor?: Color | null textColor?: Color | null cornerRadius?: number | null shadowRadius?: number | null } & ( | { message: string; content?: never } | { message?: never; content: VirtualNode } ) & ({ isPresented: boolean onChanged: (isPresented: boolean) => void } | { isPresented: Observable }) ``` *** ## Property Descriptions ### `isPresented: boolean` and `onChanged(isPresented: boolean): void` **Description**: Uses the `isPresented` and `onChanged` properties to control the visibility and behavior of the toast. **Example**: ```tsx const [showToast, setShowToast] = useState(false) toast={{ isPresented: showToast, onChanged: setShowToast, message: "Saved successfully" }} ``` *** ### `isPresented: Observable` **Description**: Uses the `isPresented` observable to control the visibility and behavior of the toast. **Example**: ```tsx const showToast = useObservable(false) toast={{ isPresented: showToast, message: "Saved successfully" }} ``` *** ### `duration?: number | null` **Description**: Specifies how long (in seconds) the toast should remain visible. Defaults to `2` seconds. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, duration: 3, message: "Action completed" }} ``` *** ### `position?: "top" | "bottom" | "center"` **Description**: Controls where the toast appears on the screen. Available values: - `"top"` – Displays the toast at the top. - `"bottom"` – Displays the toast at the bottom (default). - `"center"` – Displays the toast in the center. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, position: "top", message: "New message received" }} ``` *** ### `backgroundColor?: Color | null` **Description**: Sets the background color of the toast. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, backgroundColor: "blue", message: "Upload successful" }} ``` *** ### `textColor?: Color | null` **Description**: Sets the text color of the toast message. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, textColor: "white", message: "Download failed" }} ``` *** ### `cornerRadius?: number | null` **Description**: Sets the corner radius of the toast. Defaults to `16`. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, cornerRadius: 8, message: "Item added" }} ``` *** ### `shadowRadius?: number | null` **Description**: Sets the blur radius of the toast’s shadow. Defaults to `4`. **Example**: ```tsx toast={{ isPresented: showToast, onChanged: setShowToast, shadowRadius: 6, message: "Success" }} ``` *** ## Displaying a Simple Message **Example**: ```tsx function View() { const [showToast, setShowToast] = useState(false) return ( ``` ### Button Executing an AppIntent ```tsx