概述MediaRecorder 可将媒体流录制为分段 Blob。本文展示录制回调、持久化与上传恢复流程。能力检测与录制const supportsMR = typeof MediaRecorder === 'function';

async function startRecord(stream) {

if (!supportsMR) throw new Error('unsupported');

const mr = new MediaRecorder(stream, { mimeType: 'video/webm' });

mr.ondataavailable = e => e.data && saveChunk(e.data);

mr.start(1000);

return mr;

}

IndexedDB 持久化function openDB(name) { return new Promise((resolve, reject) => { const r = indexedDB.open(name, 1); r.onupgradeneeded = () => { const db = r.result; if (!db.objectStoreNames.contains('chunks')) db.createObjectStore('chunks', { autoIncrement: true }); }; r.onsuccess = () => resolve(r.result); r.onerror = () => reject(r.error); }); }

async function saveChunk(blob) {

const db = await openDB('rec');

const tx = db.transaction('chunks','readwrite');

tx.objectStore('chunks').add(blob);

await new Promise((res, rej) => { tx.oncomplete = res; tx.onerror = () => rej(tx.error); });

db.close();

}

上传与恢复async function loadChunks() {

const db = await openDB('rec');

const tx = db.transaction('chunks','readonly');

const store = tx.objectStore('chunks');

const req = store.openCursor();

const list = [];

await new Promise((resolve, reject) => { req.onsuccess = e => { const c = e.target.result; if (c) { list.push(c.value); c.continue(); } else resolve(); }; req.onerror = () => reject(req.error); });

db.close();

return list;

}

async function uploadAll(url) {

const chunks = await loadChunks();

for (let i=0;i<chunks.length;i++) {

const res = await fetch(url, { method:'POST', headers:{ 'X-Index': String(i) }, body: chunks[i] });

if (!res.ok) throw new Error('upload failed');

}

}

点赞(0) 打赏

评论列表 共有 0 条评论

暂无评论
立即
投稿

微信公众账号

微信扫一扫加关注

发表
评论
返回
顶部