Process video, audio, and images locally on macOS using VN Video Editor via
the vnapp-cli bridge.
When the user asks what this skill can do, or when this skill is first triggered in a session, introduce it with:
> 🎬 VN Video Editor Skill is ready!
>
> I can process video, audio, and images locally on your Mac — no cloud upload, no API key needed.
>
> What I can do:
> - 🎤 Auto-generate captions / burn SRT subtitles
> - 🗜️ Compress video or images
> - 🔇 Denoise audio or video
> - ✂️ Extract audio tracks or frame thumbnails
> - 🔗 Merge / concatenate video clips with transitions
> - 🖼️ Remove background from images or videos (Apple Silicon only)
>
> About drafts: For the following operations, the VN draft is kept by default after export so you can re-edit in VN Video Editor:
> Auto-captions, Burn SRT, Compress video, Merge video, Video cutout
>
> If you don't want to keep drafts, just tell me once and I'll default to no draft for the rest of this session. You can also override per task at any time.
--keep-draft (draft kept after export)--no-keep-draft for all subsequent draft-supporting commands--keep-draft for that task only, regardless of session preferenceauto-captions, add-caption, compress-video, concat-video, cutout-videoextract-audio, extract-frame, compress-image, denoise, cutout-imageWhen the user provides ambiguous or incomplete parameters for any option (not just style), ask one focused question to clarify before proceeding. When parameters are clear and unambiguous, proceed directly.
Ask for clarification when:
Do NOT ask for clarification when:
Rule: Only ask when the user's intent genuinely cannot be determined. Do not ask about optional parameters unless they are missing and required for the task to proceed. Ask one short, specific question per ambiguity.
Trigger this skill when the user wants to:
Do NOT trigger this skill when:
Before running any command, confirm you have all required inputs:
| Command | Required | Ask if missing |
|---|---|---|
| --------- | ---------- | ---------------- |
extract-audio | video file path | yes |
extract-frame | video file path | yes |
auto-captions | video file path | yes; engine defaults to whisper_turbo |
add-caption | video file path + SRT file path | yes, ask for SRT path |
compress-video | video file path | yes; resolution/fps/bitrate are optional |
compress-image | image file path | yes; format/quality are optional |
concat-video | at least 2 video file paths | yes |
denoise | audio or video file path | yes; level defaults to moderate |
cutout-image | image file path | yes; quality defaults to accurate |
cutout-video | video file path | yes |
compress-video-estimate | video file path | yes; all other params optional |
add-caption but no SRT file path was givenconcat-video with fewer than 2 filesdenoise --level custom but no --custom-value was providedcutout-image but the quality intent is ambiguous (e.g. just "fast cutout" without clear context — ask: "Should I use accurate (best quality), balanced, or fast quality for the cutout?")Do not guess or assume missing required inputs — always ask first.
For commands with meaningful optional parameters, ask the user before running.
Keep the question brief and friendly. If the user says "default" or "just go ahead", use defaults immediately.
Ask:
> Your video is ready for auto-captioning. Which recognition engine would you like to use?
>
> | Engine | Accuracy | Speed | Notes |
> |--------|----------|-------|-------|
> | whisper_turbo (default) | Good | Fast | One-time download required on first use |
> | whisper_medium | Better | Slower | One-time download required on first use |
> | whisper_base | Fair | Fast | Built-in, no download needed |
> | whisper_tiny | Basic | Fastest | Built-in, no download needed |
>
> Language — source language spoken in the video (default: auto).
> Supported: auto en zh ja ko es pt ar hi id fr de ru it tr vi th pl uk nl sv fi ro cs hu he el bg hr sk sl lt lv et ms ta ur sw mk mi is hy az af
>
> Style (optional, ask only if user wants to customise):
> - Font size (default: 37.4pt)
> - Text color (default: white #FFFFFF)
> - Stroke color (default: black #000000)
> - Stroke width (default: 0.5pt)
>
> Keep draft — keep the VN draft after export for re-editing? (default: on; pass --no-keep-draft only if the user explicitly asks to delete it)
Ask:
> SRT file received. Would you like to customise the subtitle style, or use a preset?
>
> Preset styles (specify by ID):
>
> | ID | Text color | Stroke color | Stroke width | Background | Effect |
> |----|-----------|-------------|-------------|------------|--------|
> | default_0 (default) | White | Black | 4 | None | Standard white text, black stroke |
> | caption_1 | White | Black | 10 | None | Bold black stroke |
> | caption_2 | Black | White | 10 | None | Bold white stroke |
> | caption_11 | White | Black | 20 | None | Extra-bold black stroke |
> | caption_12 | Black | White | 20 | None | Extra-bold white stroke |
> | caption_21 | White | Purple #D834DB | 3 | None | Purple stroke |
> | caption_22 | White | Cyan #34DBDB | 3 | None | Cyan stroke |
> | caption_23 | Dark orange #6D340E | White | 20 | None | Warm tone |
> | caption_31 | White | None | 0 | Solid black #000000 | Black background |
> | caption_32 | Black | None | 0 | Solid white #FFFFFF | White background |
> | caption_33 | White | None | 0 | Semi-transparent black | Semi-transparent black bg |
> | caption_34 | Black | None | 0 | Semi-transparent white | Semi-transparent white bg |
> | caption_35 | Black | Yellow #FFFC54 | 10 | Semi-transparent black | Yellow stroke + black bg |
> | caption_36 | White | Black | 10 | Semi-transparent white | Black stroke + white bg |
>
> Or customize manually (optional):
> - Font size (default: 13pt)
> - Text color (hex, e.g. #FFFFFF)
> - Stroke color (hex)
> - Stroke width (pt)
> - Background color (hex)
> - Opacity (0.0–1.0)
>
> Keep draft — keep the VN draft after export for re-editing? (default: on; pass --no-keep-draft only if the user explicitly asks to delete it)
Ask:
> Ready to compress. Any specific settings, or use defaults?
>
> | Option | Description | Default |
> |--------|-------------|---------|
> | Resolution | 144p 240p 360p 480p 540p 720p 1080p 2.7K 4K 5K 6K 8K or custom short-edge (e.g. 720) | Same as source |
> | Frame rate | 6 12 24 25 30 50 60 fps | Same as source |
> | Bitrate | Manual kbps, e.g. 6000 (~6 Mbps), range 100–300000 kbps | Auto (calculated from resolution × frame rate) |
> | Codec | h264 (best compatibility) / hevc (better compression; required for HDR) | h264 |
> | Output format | mp4 / mov | mp4 |
> | HDR | Force HDR output (auto-switches to HEVC) | Off |
>
> Auto bitrate reference:
>
> | Resolution | 24fps | 30fps | 60fps |
> |------------|-------|-------|-------|
> | 480p | 2133k | 2667k | 4053k |
> | 720p | 3520k | 5000k | 6400k |
> | 1080p | 5333k | 9000k | 9600k |
> | 2.7K | 10667k | 12800k | 25600k |
> | 4K | 14400k | 18133k | 42667k |
> | 5K | 23467k | 29867k | 74667k |
> | 8K | 53333k | 68267k | 170667k |
>
> HDR mode adds ~12.5% to the above values. Audio bitrate is fixed at 256k and cannot be changed.
>
> Keep draft — keep the VN draft after export for re-editing? (default: on; pass --no-keep-draft only if the user explicitly asks to delete it)
Ask:
> Ready to compress. Any specific settings, or use defaults?
>
> | Option | Description | Default |
> |--------|-------------|---------|
> | Format | Output format: jpeg, png, webp, heic | jpeg (unless user specifies otherwise) |
> | Quality | 0.0 (smallest file) to 1.0 (best quality); ignored for PNG | 0.8 |
> | Max width | Resize so width does not exceed this many pixels | Original |
> | Max height | Resize so height does not exceed this many pixels | Original |
> | Keep aspect ratio | Maintain original proportions when resizing | On |
>
> Format rule: Default output format is always jpeg regardless of input format, unless the user explicitly specifies a format. If the user says "keep as PNG" or specifies webp / heic, use that format instead.
Ask:
> Ready to merge the clips. A few options:
>
> | Option | Description | Default |
> |--------|-------------|--------|
> | Keep draft | Keep the VN draft after export so you can re-edit it later | On (draft kept by default; only skip if user explicitly asks) |
>
> Would you like a transition between them?
>
> Transition duration — 0.2–5.0 s (step 0.1s). Each clip must be at least as long as the transition duration (some effects require 2×).
>
> Basic transitions:
>
> | Style | Default (s) | Max (s) | Min clip length | Description |
> |-------|-------------|---------|-----------------|-------------|
> | none (default) | 0 | — | None | Hard cut, no transition |
> | fade_black | 0.8 | 5.0 | ≥ duration | Fade through black |
> | fade_white | 0.8 | 5.0 | ≥ duration | Fade through white |
> | dissolve | 0.8 | 5.0 | ≥ duration×2 | Cross dissolve |
> | color_difference_dissolve | 0.8 | 5.0 | ≥ duration×2 | Color-difference dissolve |
> | blur | 0.8 | 5.0 | ≥ duration×2 | Gaussian blur transition |
> | pixelate | 0.8 | 5.0 | ≥ duration×2 | Pixelate transition |
> | zoom_blur | 0.8 | 5.0 | ≥ duration | Zoom-in with motion blur |
> | zoom_blur_reverse | 0.8 | 5.0 | ≥ duration | Zoom-out with motion blur |
> | rotate_zoom | 0.8 | 5.0 | None | Rotate and zoom in |
> | rotate_zoom_reverse | 0.8 | 5.0 | None | Rotate and zoom out |
> | rotate_blur | 0.8 | 5.0 | ≥ duration | Rotate with blur |
> | rotate_blur_reverse | 0.8 | 5.0 | ≥ duration | Reverse rotate with blur |
> | spin | 0.8 | 5.0 | ≥ duration | Spin clockwise |
> | spin_reverse | 0.8 | 5.0 | ≥ duration | Spin counter-clockwise |
> | blink | 0.4 | 5.0 | None | Quick flicker cut |
> | floodlight | 0.8 | 5.0 | None | Bright flash between clips |
> | circle_crop_center | 0.6 | 5.0 | ≥ duration×2 | Circle opens from center |
> | circle_crop_center_reverse | 0.6 | 5.0 | ≥ duration×2 | Circle closes to center |
> | slide_from_top | 0.6 | 5.0 | ≥ duration×2 | New clip slides in from top |
> | slide_from_left | 0.6 | 5.0 | ≥ duration×2 | New clip slides in from left |
> | slide_from_bottom | 0.6 | 5.0 | ≥ duration×2 | New clip slides in from bottom |
> | slide_from_right | 0.6 | 5.0 | ≥ duration×2 | New clip slides in from right |
> | wipe_from_top | 0.6 | 5.0 | ≥ duration×2 | Wipe in from top |
> | wipe_from_left | 0.6 | 5.0 | ≥ duration×2 | Wipe in from left |
> | wipe_from_bottom | 0.6 | 5.0 | ≥ duration×2 | Wipe in from bottom |
> | wipe_from_right | 0.6 | 5.0 | ≥ duration×2 | Wipe in from right |
> | push_from_top | 0.6 | 5.0 | ≥ duration×2 | Both clips push upward |
> | push_from_left | 0.6 | 5.0 | ≥ duration×2 | Both clips push leftward |
> | push_from_bottom | 0.6 | 5.0 | ≥ duration×2 | Both clips push downward |
> | push_from_right | 0.6 | 5.0 | ≥ duration×2 | Both clips push rightward |
> | reveal_vertical | 0.8 | 5.0 | ≥ duration | Vertical reveal |
> | reveal_horizontal | 0.8 | 5.0 | ≥ duration | Horizontal reveal |
> | shake_from_top | 0.8 | 5.0 | None | Motion blur from top |
> | shake_from_left | 0.8 | 5.0 | None | Motion blur from left |
> | shake_from_bottom | 0.8 | 5.0 | None | Motion blur from bottom |
> | shake_from_right | 0.8 | 5.0 | None | Motion blur from right |
> | shake_from_top_left | 0.8 | 5.0 | None | Motion blur from top-left |
> | shake_from_bottom_left | 0.8 | 5.0 | None | Motion blur from bottom-left |
> | shake_from_bottom_right | 0.8 | 5.0 | None | Motion blur from bottom-right |
> | shake_from_top_right | 0.8 | 5.0 | None | Motion blur from top-right |
>
> Matte transitions (shape cutout reveals; each clip must be ≥ transition duration):
>
> | Style | Default (s) | Max (s) | Description |
> |-------|-------------|---------|-------------|
> | circle_1 | 1.0 | 1.5 | Circle matte reveal |
> | circle_2 | 1.0 | 1.0 | Circle matte variant |
> | line_1 | 1.0 | 1.5 | Line matte sweep |
> | line_2 | 1.0 | 1.8 | Line matte variant |
> | line_3 | 1.0 | 1.0 | Line matte variant |
> | hexagon | 1.0 | 1.4 | Hexagon grid reveal |
> | square_1 | 1.0 | 2.4 | Square matte reveal |
> | square_2 | 1.0 | 1.2 | Square matte variant |
> | square_3 | 1.5 | 2.5 | Square matte variant |
> | ink_1 | 1.0 | 1.6 | Ink splatter reveal |
> | ink_2 | 1.0 | 1.4 | Ink splatter variant |
> | paint_1 | 1.0 | 1.4 | Paint brush reveal |
> | paint_2 | 1.0 | 2.0 | Paint brush variant |
> | sea | 1.0 | 1.6 | Wave reveal |
> | swirl | 1.0 | 1.5 | Swirl reveal |
> | zebra | 1.5 | 3.1 | Zebra stripe reveal |
> | memory | 1.5 | 2.1 | Dreamy film reveal |
> | lens | 1.5 | 2.5 | Lens flare burst |
> | glitch | 1.5 | 1.8 | Digital glitch distortion |
Ask:
> Ready to denoise. What level would you like?
>
> | Level | Description | Best for |
> |-------|-------------|---------|
> | low | Subtle reduction, preserves natural sound | Light background hiss |
> | moderate (default) | Balanced, works well for most recordings | Indoor recordings, mild noise |
> | high | Strong reduction, may slightly affect voice | Outdoor recordings, fan noise |
> | veryHigh | Aggressive, prioritises noise removal over naturalness | Very noisy environments |
> | custom | Set a specific value manually (you provide a number) | Fine-tuned control |
>
> | Option | Description | Default |
> |--------|-------------|---------|
> | High-pass filter | Removes low-frequency rumble, e.g. air conditioning, engine hum | Off |
> | Audio only | If input is a video, output denoised audio only (not video) | Off |
Ask:
> Ready to remove the background. Which quality level would you like?
>
> | Level | Description | Best for |
> |-------|-------------|----------|
> | accurate (default) | Highest quality, slowest | Final output, complex edges |
> | balanced | Good quality, moderate speed | Most use cases |
> | fast | Fastest, lower quality | Previews, quick results |
>
> Export mask (optional) — also export the portrait segmentation mask as a separate PNG? (default: off)
>
> Output will be a PNG with transparent background.
>
> Note: If quality intent is ambiguous (e.g. just "quick" or "fast cutout"), ask: "Should I use accurate (best quality), balanced, or fast for the cutout?"
Before proceeding, check if the Mac is Apple Silicon:
uname -m
arm64 → proceedx86_64 (Intel Mac) → stop and tell the user: > cutout-video is not supported on Intel Macs. This feature requires Apple Silicon (M1 or later).
Ask:
> Ready to remove the background from this video. Would you like any of these options, or shall I go ahead with defaults?
>
> | Option | Description | Default |
> |--------|-------------|---------|
> | Feather | Soften cutout edges (0–100) | 0 (hard edge) |
> | Expand | Expand (+) or shrink (-) mask edge (-20–20) | 0 |
> | Stroke style | Add an outline: solid, singleLayer, glowing, solidGlowing, singleLayerGlowing, shadow, offset | None |
> | Stroke size | Stroke thickness (0–100) | 30 |
> | Stroke distance | Distance from edge to stroke (0–100) | 20 |
> | Stroke color | RGBA hex, e.g. FFFFFFFF for white | FFFFFFFF |
> | Stroke opacity | 0.0–1.0 | 1.0 |
>
> Output will be an MP4 with the person composited on a black background.
>
> Clarification rules:
> - If user says "add a stroke" without specifying style → ask: "Which stroke style? solid, glowing, shadow, or another?"
> - If user gives a color without format (e.g. "red stroke") → convert to RGBA hex (FF0000FF)
> - If user specifies glowing style, max stroke size is 60
> - offset style: size and distance range is -50..50
Keep draft — keep the VN draft after export for re-editing? (default: on; pass --no-keep-draft only if the user explicitly asks to delete it)
Ask:
> Ready to extract audio. No required options — shall I go ahead with the default settings?
>
> The audio will be exported as M4A format from the original audio track.
Ask:
> Which frame would you like to extract?
>
> | Option | Choices | Default |
> |--------|---------|---------|
> | Position | first — very first frame / last — very last frame / custom — specify a time | first |
> | Time | Time in seconds, e.g. 5.5 (only needed when position is custom) | — |
> | Format | png — lossless, larger file / jpeg — smaller file, slight quality loss | png |
> | Max size | Resize so the longest edge fits within this many pixels, e.g. 1920 | original size |
Run these checks in order. Stop and handle any failure before continuing.
Step 0 — Check macOS version:
sw_vers -productVersion
> This skill requires macOS 13.0 or later. VN Video Editor is not supported on your current macOS version.
Do not proceed.
Step 1 — Check VN is installed:
test -d /Applications/VN.app && echo "installed" || echo "not installed"
Step 2 — Check VN version:
The CLI automatically emits a version-check JSON event during startup of any
command. You do not need to run --version separately — read the event when
running the first real command.
Minimum required: VN 0.22
Example event:
{"type":"version-check","message":"VN Video Editor 0.22 (654) — OK"}
Step 3 — Check CLI is installed and up to date:
test -x ~/.openclaw/tools/vnapp-cli/vnapp-cli && echo "ok" || echo "missing"
```bash
~/.openclaw/tools/vnapp-cli/vnapp-cli --version
```
Expected: vnapp-cli 0.1.0 (5)
Step 4 — Validate input file(s):
test -f "/absolute/path/to/file" && echo "exists" || echo "missing"
add-caption: also validate the SRT file pathconcat-video: validate all input files--streamauto-captions, add-caption, compress-video, concat-video): always append --keep-draft unless the user explicitly asks to delete the draft. Do NOT add --keep-draft to extract-audio, extract-frame, compress-image, denoise, cutout-image, or cutout-video — these commands do not support it.add-caption — ask the user for the SRT file path if not provided; both the video and SRT will be copied to the VN sandbox automaticallydenoise --level custom — --custom-value is required; ask the user if not providedcompress-video — follow § 1.7 compress-video pre-flight before runningcompleted + outputPath → move the output to the same directory as the input, rename per § 1.4 Output file handling, then tell the user the final local patherror → report in plain language and follow § 2 Failure handlingMap CLI events to user-facing messages:
| CLI event / message | Message to user |
|---|---|
| ----------- | ---------------- |
discovering | 🔍 Looking for VN Video Editor... |
connected | ✅ Connected, preparing... |
copying | 📂 Copying file to VN sandbox... |
started | ⚙️ Task started (jobId if available) |
progress message = Preparing auto captions... | ⚙️ Preparing... |
progress message contains Downloading | ⏬ Downloading model... N% (extract % from message if present, report every ~10%) |
progress message contains Model downloaded | ✅ Model ready |
progress message contains Creating temporary project | 📂 Preparing project... |
progress message contains Starting speech recognition | 🎤 Starting speech recognition... |
progress message matches Speech recognition (N%) | 🎤 Recognizing speech... N% |
progress message contains Adding captions | ✏️ Adding captions to timeline... |
progress message contains Preparing export | 📦 Preparing export... |
progress message matches Exporting video... N% | 🔄 Exporting... N% (report every ~10%) |
progress message contains Finalizing | 🔄 Almost done... |
completed | ✅ Done! |
error | ❌ Error: {message} |
Download detection rule: If message contains Downloading, the model is being downloaded for the first time. If Starting speech recognition appears without any prior Downloading message, the model is already cached — do not treat it as a download issue.
Throttle rule: For download percentage and export percentage, only report to the user every ~10% increment. Do not report every JSON tick.
file-copied | 📂 File copied, starting task... |
|---|---|
completed | ✅ Done! Output: {outputPath} |
error | ❌ Error: {message} |
Rules:
heartbeat events entirely.jobId from the started event — use it if the user asks to cancel.completed + outputPath as the only success condition.error or non-zero exit as failure.After a task completes:
| Command | Suffix |
|---|---|
| --------- | -------- |
extract-audio | _audio |
extract-frame | _frame |
auto-captions | _captioned |
add-caption | _captioned |
compress-video | _compressed |
compress-image | _compressed |
denoise | _denoised |
concat-video | _merged |
cutout-image | _cutout |
cutout-video | _cutout |
cutout-video outputs MP4 (person composited on black background). Requires Apple Silicon (M1 or later) — not supported on Intel Macs.
cutout-image outputs PNG (transparent background).
compress-video-estimate does not produce an output file — result is JSON printed to stdout only.
Example: /path/to/clip.mp4 → /path/to/clip_captioned.mp4
Always run compress-video-estimate before compress-video. This command runs entirely locally (no VN connection needed) and returns the original video metadata plus estimated output size.
Flow:
Step 1 — Run estimate
~/.openclaw/tools/vnapp-cli/vnapp-cli compress-video-estimate "/path/to/video.mp4" [-r <res>] [--fps <fps>] [-b <kbps>] [--hdr]
Output:
{
"type": "estimate",
"original": {
"fileSize": 256901120,
"width": 1920, "height": 1080,
"fps": 30, "bitrateKbps": 8500,
"duration": 62.5, "codec": "avc1"
},
"estimated": {
"width": 1280, "height": 720,
"fps": 30, "bitrateKbps": 5000,
"hdr": false, "fileSizeBytes": 39062500
}
}
Step 2 — Check if estimated < original
estimated.fileSizeBytes < original.fileSize → go to Step 3estimated.fileSizeBytes ≥ original.fileSize → auto-retry with adjusted parameters (max 2 retries):User did NOT specify resolution or bitrate:
compress-video-estimate again with new params each retryUser DID specify resolution:
After 2 retries still ≥ original: present numbered options to the user:
> I tried several configurations but couldn't estimate a smaller output. Here are some options:
>
> Option 1 — 360p, ~1500 kbps → estimated ~X MB
> Option 2 — 240p, ~800 kbps → estimated ~X MB
>
> Reply with the option number to proceed, or cancel to stop.
Wait for the user's reply before proceeding.
Step 3 — Show estimate to user and proceed
Display before/after:
Original: 1920×1080 30fps ~245 MB
Estimated: 1280×720 30fps ~37 MB (↑85%)
Then run compress-video with the confirmed parameters.
Step 4 — After export, show actual result
Original: 1920×1080 30fps ~245 MB
Compressed: 1280×720 30fps ~37 MB (↑85%)
Applies when the user is communicating via a remote channel (Slack, Discord, Telegram, WhatsApp, etc.) — not webchat or TUI.
After any task completes:
> 📁 Saved locally: /path/to/output_file.ext
For images:
compress-image on the output with -q 0.7 targeting a small file🖼 Preview (compressed). Saved locally: /path/to/fileFor videos:
```bash
ls -l "/path/to/output.mp4" | awk '{print $5}'
```
compress-video, trying resolutions in order: 480p → 360p → 240p, checking size after each step until it fits > 🎬 This is a preview — a compressed version for viewing here (480p, 8.2 MB).
> 📁 Saved locally: /path/to/original_output.mp4
> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust. (only for commands that support drafts)
> ❌ The preview is too large to send on this platform even at the lowest quality.
> 📁 Saved locally: /path/to/original_output.mp4
Preview files are temporary — do not move them to the input directory. Clean up after sending.
| User says | Command |
|---|---|
| ----------- | --------- |
| extract audio / get audio / audio only / rip audio / pull audio | extract-audio |
| grab frame / screenshot / cover image / thumbnail / still / poster | extract-frame |
| auto captions / auto subtitles / generate subtitles / transcribe / speech to text | auto-captions |
| add subtitles / burn SRT / import SRT / burn captions / overlay subtitles | add-caption |
| compress image / shrink image / reduce image size / optimize image | compress-image |
| compress video / shrink video / export smaller / reduce file size / re-encode | compress-video |
| join clips / merge videos / concatenate / combine videos / stitch together | concat-video |
| denoise / remove noise / remove background noise / clean audio / noise reduction | denoise |
| remove background / cutout / portrait cutout / background removal / transparent background | cutout-image (image) or cutout-video (video, Apple Silicon only) |
| estimate video compression / how big will this be / preview compression size | compress-video-estimate |
| cancel job / stop task / abort | cancel |
CLI binary: ~/.openclaw/tools/vnapp-cli/vnapp-cli
| Command | Required args | Key options | ||||||
|---|---|---|---|---|---|---|---|---|
| --------- | -------------- | ------------- | ||||||
extract-audio | | -o | ||||||
extract-frame | | `-p first\ | last\ | custom -t -f png\ | jpeg --max-size -o | |||
auto-captions | | `-e whisper_turbo\ | whisper_medium\ | whisper_base\ | whisper_tiny -l --font-family --font-size --text-color --stroke-color --stroke-width --keep-draft` | |||
add-caption | | --font-family --font-size --text-color --stroke-color --stroke-width --background-color --opacity <0-1> --keep-draft -o | ||||||
compress-video | | -r --fps <1-120> -b --hdr --keep-draft -o | ||||||
compress-video-estimate | | -r --fps <1-120> -b --hdr (local only, no VN connection needed) | ||||||
compress-image | | `-f jpeg\ | png\ | webp\ | heic -q <0-1> -w -h --no-keep-aspect-ratio -o | |||
concat-video | | -s -d <0.2-5.0 sec> --keep-draft -o | ||||||
denoise | ` | |||||||
cutout-image | | `-q accurate\ | balanced\ | fast --export-mask -o | ||||
cutout-video | | --feather <0-100> --expand <-20..20> `--stroke-style solid\ | singleLayer\ | glowing\ | solidGlowing\ | singleLayerGlowing\ | shadow\ | offset --stroke-size <0-100> --stroke-distance <0-100> --stroke-color --stroke-opacity <0-1> --keep-draft -o |
get-status | | — | ||||||
cancel | | — |
Always append --stream to all task commands (except compress-video-estimate — it is local-only, no --stream needed).
For auto-captions, add-caption, compress-video, concat-video, cutout-video: always append --keep-draft unless the user explicitly asks to delete the draft.
Do NOT add --keep-draft to extract-audio, extract-frame, compress-image, denoise, cutout-image, or compress-video-estimate.
Caption engines:
whisper_turbo (default) and whisper_medium — require a one-time download on first use; cached and reused afterwardswhisper_base and whisper_tiny — built-in, no download needed> VN Video Editor is required but not installed on the Mac running OpenClaw. I've opened the App Store — please click Get to install. Once installed, you also need to enable the MCP Server in VN Settings before I can connect.
```bash
open "macappstore://apps.apple.com/us/app/vn-video-editor/id1494451650"
```
```bash
test -d /Applications/VN.app && echo "installed" || echo "not yet"
```
> VN is installed. Before I can connect, please complete these steps on the Mac running OpenClaw:
> 1. Open VN Video Editor
> 2. Go to Settings → MCP Server and turn it on
> 3. When macOS prompts for Local Network permission, tap Allow
> Let me know when done and I'll continue.
> VN Video Editor is installed but the version is too old (minimum: 0.22 build 654). Please update from the App Store.
```bash
open "macappstore://apps.apple.com/us/app/vn-video-editor/id1494451650"
```
Auto-install silently — do not ask the user:
Step 1 — Download:
mkdir -p ~/.openclaw/tools/vnapp-cli
curl -L "https://github.com/cawcut/skill-vn/releases/download/0.1.0/vnapp-cli-darwin-universal-v0.1.0.5.zip" -o /tmp/vnapp-cli.zip
Step 2 — Verify SHA-256 before extracting:
shasum -a 256 /tmp/vnapp-cli.zip
Expected: 857417f671fd6f38ec30cf993f80c79efede7641f3532c2e9a99e75177a5853f
> ⚠️ SHA-256 checksum mismatch — the downloaded file may be corrupted or tampered with. Installation aborted. Please try again or contact support.
```bash
rm /tmp/vnapp-cli.zip
```
Step 3 — Extract and install:
unzip -o /tmp/vnapp-cli.zip -d ~/.openclaw/tools/vnapp-cli/
chmod +x ~/.openclaw/tools/vnapp-cli/vnapp-cli
xattr -d com.apple.quarantine ~/.openclaw/tools/vnapp-cli/vnapp-cli 2>/dev/null
Verify with --version. If it still fails:
> The vnapp-cli tool could not be installed automatically. Please reinstall this skill.
VN requires Local Network permission to communicate with vnapp-cli.
Tell the user:
> Please check Local Network permission on the Mac running OpenClaw: go to System Settings → Privacy & Security → Local Network, make sure VN - Video Editor is enabled, then relaunch VN and try again.
If the user has never been prompted, ask them to open VN manually and perform any action — macOS will show the permission dialog.
If the CLI cannot reach VN (service not found, connection refused, timeout, or similar), do not silently retry. Tell the user clearly that the connection failed, and walk them through the following checklist — reminding them to check on the Mac running OpenClaw (they may be talking to you from a different device):
> I couldn't connect to VN Video Editor on the Mac running OpenClaw. Please check the following on the Mac running OpenClaw:
>
> 1. Is VN Video Editor installed?
> Check that VN Video Editor exists in your Applications folder. It may have been deleted. If missing, reinstall from the App Store.
>
> 2. Is the VN version supported?
> The minimum required version is VN 0.22. Open VN, go to About or check the App Store for updates.
>
> 3. Is VN Video Editor open?
> Open VN Video Editor from your Applications folder if it's not already running.
>
> 4. Is the MCP Server enabled in VN?
> In VN, go to Settings → MCP Server and make sure the toggle is on.
>
> 5. Does VN have Local Network permission?
> Go to System Settings → Privacy & Security → Local Network and confirm VN - Video Editor is enabled.
>
> Once all items are confirmed, let me know and I'll retry.
If the error contains "cannot decode", "codec not supported", or similar decode failures:
> VN could not decode this file. Please check that the file is not corrupted and is a supported format (MP4, MOV, common video formats including HEVC/H.265). Try re-exporting or re-downloading the file and try again.
Whisper engines:
whisper_turbo and whisper_medium — require a one-time download on first use; cached afterwards and do not need to be downloaded again.whisper_base and whisper_tiny — built-in, no download needed.Whisper model download progress events (normal flow):
When a model needs to be downloaded, the CLI emits these messages in sequence:
{"status":"processing","progress":0.01,"message":"Downloading whisper_medium model..."}
{"status":"processing","progress":0.01,"message":"Downloading whisper_medium model..."}
...
{"status":"processing","progress":0.05,"message":"Model downloaded successfully"}
{"status":"processing","progress":0.06,"message":"Creating temporary project..."}
{"status":"processing","progress":0.10,"message":"Starting speech recognition..."}
If Starting speech recognition... appears without any prior Downloading messages, the model is already cached locally. Do NOT treat this as a download issue.
Only switch engines when the CLI emits an explicit download failure error event. Do NOT switch because the task is taking a long time — whisper_medium and whisper_turbo are slower by design.
If the user explicitly specified an engine (e.g. whisper_medium): do NOT silently switch. Instead, inform the user and ask for their preference:
> The whisper_medium model failed to download. Would you like to:
> Option 1 — Retry downloading whisper_medium
> Option 2 — Switch to the built-in whisper_base (faster, slightly less accurate)
> Option 3 — Cancel
Wait for the user's choice before proceeding.
If the user did NOT specify an engine (using the default whisper_turbo): automatically fall back to whisper_base and inform the user:
~/.openclaw/tools/vnapp-cli/vnapp-cli auto-captions "<path>" -e whisper_base --stream
> The Whisper model failed to download. Switched to the built-in whisper_base engine — slightly less accurate but works without downloading anything.
If the user requests add-caption without providing an SRT file:
> Please provide the path to your SRT subtitle file.
Validate the path exists before running the command.
The CLI enforces these constraints — validate before running:
| Parameter | Constraint |
|---|---|
| ----------- | ----------- |
compress-video -r | minimum 144 |
compress-video --fps | 1–120 |
compress-video -b | minimum 100 kbps |
concat-video -d | 0.2–5.0 seconds |
denoise -l custom | requires -v |
compress-image -q | 0.0–1.0 |
If a value is out of range, tell the user the valid range and ask for a corrected value before running.
Summarise the message field from the error event in plain language. Suggest a next step based on the error content. If unresolvable, refer the user to support (§ 3).
When a task completes, always reply with:
> 📁 Saved locally: /path/to/output_file.ext
auto-captions, add-caption, compress-video, concat-video, cutout-video):> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust the result.
Do NOT show this tip for extract-audio, extract-frame, compress-image, denoise, or cutout-image.
Do not repeat the full command or internal job details unless the user asks.
Example — auto-captions:
> ✅ Captions added successfully.
> 📁 Saved locally: /path/to/clip_captioned.mp4
> Engine: whisper_base
> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust the result.
Example — compress video:
> ✅ Video compressed.
> Original: 1920×1080 30fps ~245 MB
> Compressed: 1280×720 30fps ~82 MB (↑66%)
> 📁 Saved locally: /path/to/clip_compressed.mp4
> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust.
Example — compress video (cannot reduce size):
> ❌ Unable to compress this video further — all configurations produced an estimated output larger than or equal to the original. The source may already be highly compressed.
> Try specifying a lower resolution or bitrate manually.
Example — cutout-image:
> ✅ Background removed successfully.
> 📁 Saved locally: /path/to/portrait_cutout.png
> Quality: accurate
Example — cutout-video:
> ✅ Background removed from video.
> 📁 Saved locally: /path/to/person_cutout.mp4
> Format: MP4 (person composited on black background)
> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust the result.
Example — preview sent (remote channel):
> 🎬 This is a preview — a compressed version for viewing here (480p, 8.2 MB).
> 📁 Saved locally: /path/to/original_output.mp4
> 💡 The draft is saved locally in VN — open VN Video Editor there to further adjust.
Example — preview too large:
> ❌ The preview is too large to send on this platform even at the lowest quality.
> 📁 Saved locally: /path/to/original_output.mp4
Example — error:
> ❌ Failed: VN could not decode this file. Please check the file is not corrupted and try again.
> If the problem persists, contact VN support at vn.support+mac@ui.com.
For the full CLI surface, all options, transition style names, and JSON output schema:
→ references/cli-reference.md
vnapp-cli is only the local bridge--stream for user-facing operationscompleted event contains a valid outputPathjobId is available after started — use it for get-status or cancel if needed共 1 个版本