From zoom-skills
Builds custom real-time video experiences using Zoom's infrastructure across web, mobile, and desktop platforms. Use for full control over video UI, screen sharing, virtual backgrounds, and audio-only rooms.
How this skill is triggered — by the user, by Claude, or both
Slash command
/zoom-skills:video-sdkThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Build custom video experiences powered by Zoom's infrastructure.
RUNBOOK.mdandroid/RUNBOOK.mdandroid/android.mdandroid/concepts/architecture.mdandroid/concepts/lifecycle-workflow.mdandroid/examples/session-join-pattern.mdandroid/references/android-reference-map.mdandroid/references/environment-variables.mdandroid/references/versioning-and-compatibility.mdandroid/scenarios/high-level-scenarios.mdandroid/troubleshooting/common-issues.mdflutter/RUNBOOK.mdflutter/concepts/high-level-scenarios.mdflutter/concepts/lifecycle-workflow.mdflutter/concepts/sdk-architecture-pattern.mdflutter/examples/event-handling-pattern.mdflutter/examples/session-join-pattern.mdflutter/examples/setup-guide.mdflutter/references/flutter-reference.mdflutter/references/module-map.mdBuild custom video experiences powered by Zoom's infrastructure.
join_url, or Meeting SDK join payload fields (meetingNumber, passWord).| User context | Primary child skill |
|---|---|
| Browser, JavaScript, TypeScript, React, Vue, Angular, Svelte | web/SKILL.md |
| React Native mobile wrapper | react-native/SKILL.md |
| Flutter mobile wrapper | flutter/SKILL.md |
| Native Android, Java, Kotlin, AAR | android/SKILL.md |
| Native iOS, Objective-C, Swift, XCFramework | ios/SKILL.md |
| Native macOS, Objective-C, Swift, Cocoa | macos/SKILL.md |
| Windows native, C++, Win32, C++/CLI | windows/SKILL.md |
| Linux native, C++, headless bot, Docker, Qt/GTK | linux/SKILL.md |
| Unity wrapper or scene integration | unity/SKILL.md |
Do not answer from this root's Web quick start when the user selected another platform.
Use this order so workflow guidance and version-specific APIs remain distinct:
SKILL.md for architecture, lifecycle, implementation patterns, and troubleshooting.| Cross-product need | Chain to |
|---|---|
| Session JWT contract | references/authorization.md and references/token-contract-test-spec.md |
| Browser/device/network preflight | probe-sdk |
| RTMS signaling, media sockets, protocol, and backend processing | zoom-rtms |
| OAuth or REST administration outside the in-session SDK | zoom-oauth |
For React Native, Flutter, and Unity, never invent wrapper APIs from native Android/iOS/Windows documentation. Use native documentation only to understand behavior after confirming the wrapper exports the feature.
| Feature | Meeting SDK | Video SDK |
|---|---|---|
| UI | Default Zoom UI or Custom UI | Fully custom UI (you build it) |
| Experience | Zoom meetings | Video sessions |
| Branding | Limited customization | Full branding control |
| Features | Full Zoom features | Core video features |
Video SDK gives you full control over the UI:
| Option | Description |
|---|---|
| UI Toolkit | Pre-built React components (low-code) |
| Custom UI | Build your own UI using the SDK APIs |
Need a Video SDK session JWT? Start with authorization. Use zoom-oauth only for OAuth or REST API flows outside the session join contract.
Need pre-join diagnostics on web? Use probe-sdk before Video SDK
join()to reduce first-minute failures.
Start troubleshooting fast: Use the 5-Minute Runbook before deep debugging.
Video SDK 2.5.10 packages include version-matched API skills and paired Markdown/JSON documentation for several native platforms. Keep the platform skill in this repository as the workflow entry point, then use the SDK-bundled skill to verify exact signatures, enums, callback timing, role requirements, and object lifetime.
| Platform | SDK-bundled skill |
|---|---|
| Android | Docs/skills/zm-videosdk-android-api/SKILL.md |
| iOS | Sample-Libs/Docs/zm-videosdk-ios-api/SKILL.md |
| macOS | Docs/skills/zm-videosdk-macos-api/SKILL.md |
| Windows | Sample-Libs/<arch>/Docs/skills/zm-videosdk-windows-api/SKILL.md |
| Linux | Docs/videosdk/skills/zm-videosdk-linux-api/SKILL.md |
| React Native | docs/ai-docs/skills/zm-videosdk-react-native-api/SKILL.md |
Flutter, Unity, and Web packages in this review do not include an equivalent bundled SKILL.md. The Web 2.4.5 archive is a React/Vite sample application; use web/references/sample-app-2.4.5.md for its implementation map. Do not infer platform parity from another package; verify against that platform's exported API.
import ZoomVideo from '@zoom/videosdk';
const client = ZoomVideo.createClient();
await client.init('en-US', 'Global', { patchJsMedia: true });
await client.join(topic, signature, userName, password);
// IMPORTANT: getMediaStream() ONLY works AFTER join()
const stream = client.getMediaStream();
await stream.startVideo();
await stream.startAudio();
WARNING: Ad blockers block
source.zoom.us. Self-host the SDK to avoid issues.
# Download SDK locally example VERSION=2.4.5
curl "https://source.zoom.us/videosdk/zoom-video-{VERSION}.min.js" -o js/zoom-video-sdk.min.js
<script src="js/zoom-video-sdk.min.js"></script>
// CDN exports as WebVideoSDK, NOT ZoomVideo
// Must use .default property
const ZoomVideo = WebVideoSDK.default;
const client = ZoomVideo.createClient();
await client.init('en-US', 'Global', { patchJsMedia: true });
await client.join(topic, signature, userName, password);
// IMPORTANT: getMediaStream() ONLY works AFTER join()
const stream = client.getMediaStream();
await stream.startVideo();
await stream.startAudio();
When using <script type="module"> with CDN, SDK may not be loaded yet:
// Wait for SDK to load before using
function waitForSDK(timeout = 10000) {
return new Promise((resolve, reject) => {
if (typeof WebVideoSDK !== 'undefined') {
resolve();
return;
}
const start = Date.now();
const check = setInterval(() => {
if (typeof WebVideoSDK !== 'undefined') {
clearInterval(check);
resolve();
} else if (Date.now() - start > timeout) {
clearInterval(check);
reject(new Error('SDK failed to load'));
}
}, 100);
});
}
// Usage
await waitForSDK();
const ZoomVideo = WebVideoSDK.default;
const client = ZoomVideo.createClient();
The SDK has a strict lifecycle. Violating it causes silent failures.
1. Create client: client = ZoomVideo.createClient()
2. Initialize: await client.init('en-US', 'Global', options)
3. Join session: await client.join(topic, signature, userName, password)
4. Get stream: stream = client.getMediaStream() ← ONLY AFTER JOIN
5. Start media: await stream.startVideo() / await stream.startAudio()
Common Mistake (Silent Failure):
// ❌ WRONG: Getting stream before joining
const client = ZoomVideo.createClient();
await client.init('en-US', 'Global');
const stream = client.getMediaStream(); // Returns undefined!
await client.join(...);
// ✅ CORRECT: Get stream after joining
const client = ZoomVideo.createClient();
await client.init('en-US', 'Global');
await client.join(...);
const stream = client.getMediaStream(); // Works!
The SDK is event-driven. You must listen for events and render videos accordingly.
attachVideo() NOT renderVideo()import { VideoQuality } from '@zoom/videosdk';
// Start your camera
await stream.startVideo();
// Attach video - returns element to append to DOM
const element = await stream.attachVideo(userId, VideoQuality.Video_360P);
container.appendChild(element);
// Detach when done
await stream.detachVideo(userId);
// When other participant's video turns on/off
client.on('peer-video-state-change', async (payload) => {
const { action, userId } = payload;
if (action === 'Start') {
const el = await stream.attachVideo(userId, VideoQuality.Video_360P);
container.appendChild(el);
} else {
await stream.detachVideo(userId);
}
});
// When participants join/leave
client.on('user-added', (payload) => { /* check bVideoOn */ });
client.on('user-removed', (payload) => { stream.detachVideo(payload.userId); });
See web/references/web.md for complete event handling patterns.
| Concept | Description |
|---|---|
| Session | Video session (not a meeting) |
| Topic | Session identifier (any string you choose) |
| Signature | JWT for authorization |
| MediaStream | Audio/video stream control |
Important: Video SDK sessions are created just-in-time, not in advance.
| Aspect | Video SDK | Meeting SDK |
|---|---|---|
| Pre-creation | NOT required | Create meeting via API first |
| Session start | First participant joins with topic | Join existing meeting ID |
| Topic | Any string (you define it) | Meeting ID from API |
| Scheduling | N/A - sessions are ad-hoc | Meetings can be scheduled |
topic string join the same session// Session is created on-the-fly when first user joins
// Any string can be the topic - it becomes the session identifier
await client.join('my-custom-session-123', signature, 'User Name');
// Other participants join the SAME session by using the SAME topic
await client.join('my-custom-session-123', signature, 'Another User');
The signature endpoint must be accessible from your frontend without CORS issues.
Option 1: Same-Origin Proxy (Recommended)
# Nginx config
location /api/ {
proxy_pass http://YOUR_BACKEND_HOST:3005/api/;
proxy_http_version 1.1;
proxy_set_header Host $host;
}
// Frontend uses relative URL (same origin)
const response = await fetch('/api/signature', { ... });
Option 2: CORS Configuration
// Express.js backend
const cors = require('cors');
app.use(cors({
origin: ['https://your-domain.com'],
credentials: true
}));
WARNING: Mixed content (HTTPS page → HTTP API) will be blocked by browsers.
| Use Case | Description |
|---|---|
| Video SDK BYOS (Bring Your Own Storage) | Save recordings directly to your S3 bucket |
Video SDK feature - Zoom saves cloud recordings directly to your Amazon S3 bucket. No downloading required.
Official docs: https://developers.zoom.us/docs/build/storage/
Prerequisites:
Authentication options:
S3 path structure:
Buckets/{bucketName}/cmr/byos/{YYYY}/{MM}/{DD}/{GUID}/cmr_byos/
Key benefits:
Setup location: Developer Portal → Account Settings → General → Communications Content Storage Location
See ../general/use-cases/video-sdk-bring-your-own-storage.md for complete setup guide.
| Type | Repository | Stars |
|---|---|---|
| Web | videosdk-web-sample | 137 |
| Web NPM | videosdk-web | 56 |
| Auth | videosdk-auth-endpoint-sample | 23 |
| UI Toolkit Web | videosdk-ui-toolkit-web | 17 |
| UI Toolkit React | videosdk-ui-toolkit-react-sample | 17 |
| Next.js | videosdk-nextjs-quickstart | 16 |
| Telehealth | VideoSDK-Web-Telehealth | 11 |
| Linux | videosdk-linux-raw-recording-sample | - |
Full list: See general/references/community-repos.md
.env keys and where to find each value.npx claudepluginhub zoom/skills --plugin zoom-skillsEmbeds Zoom meetings into web, mobile, desktop, and headless apps using the Zoom Meeting SDK. Use when integrating full meeting experiences into your own application UI.
Records, composes, and publishes short desktop validation videos for UX interaction proof. Useful when a screenshot cannot show motion, timing, or audio/visual state.
Guides building voice/video apps with Agora SDKs, including AI agents, calls, live streaming, screen sharing, messaging, recording, and CLI operations.