Запись аудио в браузере
Ольга Маланова
<embed>
data:image/s3,"s3://crabby-images/7a1b6/7a1b699489d03e7ee4c679a431f4a81da2ce32f2" alt=""
<audio>
data:image/s3,"s3://crabby-images/520c9/520c9fefa7a7ef94d0596536e46ff077a8c5596f" alt=""
Плагины
data:image/s3,"s3://crabby-images/456df/456df6451ba8f2d37bc68339e372e78d014f3750" alt=""
data:image/s3,"s3://crabby-images/fe89d/fe89dc71fe93b0fb531c0496295941abfbdef5a8" alt=""
data:image/s3,"s3://crabby-images/bea55/bea55348382deaeee838ffed68af9ce3726b87a0" alt=""
data:image/s3,"s3://crabby-images/75ba8/75ba816381d5bbb9490c76601987deb157d6c29b" alt=""
Воспроизведение аудио
Web Audio API
data:image/s3,"s3://crabby-images/d928e/d928e84fa2a53d30b73bb2ddb7f1abdf093141c9" alt=""
const AudioContext = window.AudioContext || window.webkitAudioContext
const audioCtx = new AudioContext()
const sourceNode = audioCtx.createMediaStreamSource(stream)
const gainNode = audioCtx.createGain()
const finishNode = audioCtx.destination
sourceNode.connect(gainNode);
gainNode.connect(finishNode)
Web Audio API
data:image/s3,"s3://crabby-images/d928e/d928e84fa2a53d30b73bb2ddb7f1abdf093141c9" alt=""
const AudioContext = window.AudioContext || window.webkitAudioContext
const audioCtx = new AudioContext()
const sourceNode = audioCtx.createMediaStreamSource(stream)
const gainNode = audioCtx.createGain()
const finishNode = audioCtx.destination
sourceNode.connect(gainNode);
gainNode.connect(finishNode)
data:image/s3,"s3://crabby-images/b9d17/b9d1734bb1d27db4f144da1408aeb0cb2d45476c" alt=""
data:image/s3,"s3://crabby-images/ada15/ada15adbda3bd25d17727e09363007c82d95dc44" alt=""
data:image/s3,"s3://crabby-images/b9df6/b9df6026a86816cb291563964d911350ea49b293" alt=""
Доступ к потоку медиаданных
data:image/s3,"s3://crabby-images/512f4/512f496f662cb12699f8c9cf1b3193c55f8fb2e7" alt=""
<input type="file"/>
<input type="file" accept="audio/*;capture=microphone">
data:image/s3,"s3://crabby-images/d2bdf/d2bdf6de3bafb978c17b841cbde943b69ddc9535" alt=""
browser
data:image/s3,"s3://crabby-images/a82e2/a82e29568016f50666575520e8f77d3d62746500" alt=""
data:image/s3,"s3://crabby-images/f4de3/f4de3302212bbcda8c323612a317899eb09c03f9" alt=""
Android
data:image/s3,"s3://crabby-images/8f7c1/8f7c10a9645c73f116480676c009354ec0473bf9" alt=""
data:image/s3,"s3://crabby-images/31b17/31b17022d59a999ad921537d8300476feda6957f" alt=""
data:image/s3,"s3://crabby-images/19bf5/19bf563bca2d5dac82e77db7d69a61147337af65" alt=""
iOS
data:image/s3,"s3://crabby-images/4109f/4109f20e3d7fd3d674b846c53fc987741c007afd" alt=""
getUserMedia
data:image/s3,"s3://crabby-images/f4e3c/f4e3cc0a82ab231850c28ecf1120b0eaba0f6241" alt=""
<device type="media" onchange="update(this.data)"></device>
<audio></auidio>
<script>
function update(stream) {
document.querySelector('audio').src = stream.url;
}
</script>
<device>
data:image/s3,"s3://crabby-images/8cbdf/8cbdfd3bb2e44f8ab8b1ece6ec62009846ffdbe3" alt=""
getUserMedia
async getMedia = () => {
try {
const stream = await navigator.mediaDevices.getUserMedia({ audio: true })
successCallback(stream)
} catch (err) {
handleError(err)
}
}
async getDevices = () => {
try {
const devices = await navigator.mediaDevices.enumerateDevices()
devices.map(device => { const { label, kind, deviceInfo) = device } // kind === 'audioinput'
} catch (err) {
handleError(err)
}
}
}
Пользовательский интерфейс
data:image/s3,"s3://crabby-images/188a6/188a64c0f9c210a2ecfc899f92b7bcc8ba9b536f" alt=""
data:image/s3,"s3://crabby-images/ae1d9/ae1d98915afe9860e09aa01af3bd3025c36ce748" alt=""
data:image/s3,"s3://crabby-images/32162/32162ae989d7997f1c95b0ba94ff4df0448c9bb3" alt=""
Обработка и сохранение
data:image/s3,"s3://crabby-images/b3e20/b3e20b53579bf4edcd47878ff98000dbc47c2ea3" alt=""
data:image/s3,"s3://crabby-images/8a550/8a550f238d70e5ae477943d97023fc3d56b55156" alt=""
MediaStream
ondataavailable
Blob
MediaRecorder API
MediaRecorder
export default {
name: 'MediaRecorderComponent',
data () {
return {
mediaRecorder: null
isAudioAccessReceived: false,
isRecording: false,
chunks: [],
audio: this.$refs.audio
}
},
methods: {
toggleRecord () {
if (!this.isAudioAccessReceived) {
try {
const stream = await navigator.mediaDevices.getUserMedia({ audio: true })
this.mediaRecorder = new window.MediaRecorder(stream, { type: 'audio/ogg' })
this.mediaRecorder.ondataavailable = () => this.chunks.push(event.data)
mediaRecorder.onstop = (e) => {
const blob = new Blob(chunks, { 'type' : 'audio/ogg; codecs=opus' });
this.chunks = [];
const audioURL = window.URL.createObjectURL(blob);
audio.src = audioURL;
}
this.isAudioAccessReceived = true
} catch (e) { /* handle error */ }
}
if (!this.isRecording) {
this.mediaRecorder.start()
} else {
this.mediaRecorder.stop()
}
this.isRecording = !this.isRecording
}
}
}
MediaRecorder API
export default {
name: 'MediaRecorderComponent',
data () {
return {
mediaRecorder: null
isAudioAccessReceived: false,
isRecording: false,
chunks: [],
audio: this.$refs.audio
}
},
methods: {
toggleRecord () {
if (!this.isAudioAccessReceived) {
try {
const stream = await navigator.mediaDevices.getUserMedia({ audio: true })
this.mediaRecorder = new window.MediaRecorder(stream, { type: 'audio/ogg' })
this.mediaRecorder.ondataavailable = () => this.chunks.push(event.data)
mediaRecorder.onstop = (e) => {
const blob = new Blob(chunks, { 'type' : 'audio/ogg; codecs=opus' });
this.chunks = [];
const audioURL = window.URL.createObjectURL(blob);
audio.src = audioURL;
}
this.isAudioAccessReceived = true
} catch (e) { /* handle error */ }
}
if (!this.isRecording) {
this.mediaRecorder.start()
} else {
this.mediaRecorder.stop()
}
this.isRecording = !this.isRecording
}
}
}
MediaRecorder API
MediaRecorder Support
data:image/s3,"s3://crabby-images/d446f/d446f72a236c0f41d18f34c0ac54b0ea51e9136c" alt=""
Мобильная история
data:image/s3,"s3://crabby-images/71976/71976a6f97cf97f58fdbd12e56a9573e2a0b226a" alt=""
Android
data:image/s3,"s3://crabby-images/87d37/87d37b9296bd88697b57ed4cbbe302f1608ec6f6" alt=""
data:image/s3,"s3://crabby-images/9c8b3/9c8b31f4b6be1ddeb1477b7de054933ca7dc0afb" alt=""
data:image/s3,"s3://crabby-images/60744/60744813dc813391e04bccf6ec92d33d31f69279" alt=""
data:image/s3,"s3://crabby-images/6a688/6a68889723dd332ad5efa1a1546ad88c3ccd9d51" alt=""
iOS
data:image/s3,"s3://crabby-images/2935b/2935b8d4d255d946e3fb8d07cbf042c0b6bb4afc" alt=""
data:image/s3,"s3://crabby-images/6ed19/6ed19ee8be8fb7a055d15ae4e12be8d5fe9afd62" alt=""
data:image/s3,"s3://crabby-images/563b9/563b957fd786d0055e75fca6636b1423cb6a45e0" alt=""
Движки
data:image/s3,"s3://crabby-images/a5f42/a5f42d3a03638f54a0b273a9479bd955ee8e3579" alt=""
Chrome on iOS
data:image/s3,"s3://crabby-images/ce671/ce67140156fb871c191e83f245ad2d7a117a8d1d" alt=""
Определение API
export default async ({ app, Vue, store }) => {
if (!navigator.mediaDevices) navigator.mediaDevices = {}
if (!navigator.mediaDevices.getUserMedia) {
var getUserMedia = (
navigator.webkitGetUserMedia ||
navigator.mozGetUserMedia ||
navigator.msGetUserMedia ||
navigator.getUserMedia
)
if (getUserMedia) {
navigator.mediaDevices.getUserMedia = function (constraints) {
return new Promise(function (resolve, reject) {
getUserMedia(constraints,
function (stream) { resolve(stream) },
function (error) { reject(error) }
)
})
}
} else {
navigator.mediaDevices.getUserMedia = function () {
return new Promise(function (resolve, reject) {
reject('getUserMedia is not supported in this browser.')
})
}
}
}
}
Альтернативы
data:image/s3,"s3://crabby-images/86694/866949b389c441aeac4bf02b1b20d65d2b6a6c1c" alt=""
RecorderJS
export default {
name: 'AudioRecorder',
data () { return { recorder: null, audioContext: null } },
created () { this.audioContext = new this.AudioContextClass() },
beforeDestroy () { this.audioContext = null },
methods: {
toggleRecord () {
if (this.isRecording) {
this.isRecording = false
setTimeout(() => {
this.recorder.stop()
this.stream.getAudioTracks()[0].stop()
this.recorder.exportWAV(finalUserData => { successCallback(finalUserData) })
}, 700)
} else {
navigator.mediaDevices.getUserMedia({
audio: true
})
.then((stream) => {
this.recorder = new window.Recorder(
this.audioContext.createMediaStreamSource(stream), {
numChannels: 1, type: 'audio/wav'
}
)
this.recorder.record()
this.isRecording = true
})
.catch(err => {
console.error('The following getUserMedia error occured: ' + err)
})
}
},
}
}
WebAudioRecorder
let webAudioRecorder = new WebAudioRecorder(source, {
workerDir: 'web_audio_recorder_js/',
encoding: 'mp3',
options: {
encodeAfterRecord: true,
mp3: { bitRate: '320' }
}
});
webAudioRecorder.onComplete = (webAudioRecorder, blob) => {
/* on complete handler */
}
webAudioRecorder.onError = (webAudioRecorder, err) => {
/* on error handler */
}
webAudioRecorder.startRecording();
webAudioRecorder.finishRecording();
MediaStreamRecorder
const mediaConstraints = {
audio: true
}
navigator.getUserMedia(mediaConstraints, onMediaSuccess, onMediaError)
function onMediaSuccess(stream) {
/* media success handler */
}
function onMediaError(e) {
console.error('media error', e)
}
mediaRecorder.stop()
mediaRecorder.start()
mediaRecorder.save()
data:image/s3,"s3://crabby-images/8f7b8/8f7b850e34f54977679065325bda3d8d039939a0" alt=""
RecordRTC
let stream = await navigator.mediaDevices.getUserMedia({video: true, audio: true});
let recorder = new RecordRTCPromisesHandler(stream, {
type: 'video'
});
recorder.startRecording();
const sleep = m => new Promise(r => setTimeout(r, m));
await sleep(3000);
await recorder.stopRecording();
let blob = await recorder.getBlob();
invokeSaveAsDialog(blob);
data:image/s3,"s3://crabby-images/d7e47/d7e471363b6f8ebef2d08d26a15823a630705586" alt=""
AudioRecorderPolyfill
data:image/s3,"s3://crabby-images/365b8/365b8861c15ff30b5ec6650ada3b16ca9b4dc7f0" alt=""
window.MediaRecorder = require('audio-recorder-polyfill')
navigator.mediaDevices.getUserMedia({ audio: true }).then(stream => {
recorder = new MediaRecorder(stream)
recorder.addEventListener('dataavailable', e => {
audio.src = URL.createObjectURL(e.data)
})
recorder.start()
})
data:image/s3,"s3://crabby-images/5c4e6/5c4e61a92a62ab635b1f3a5253ef912248705c26" alt=""
WebRTC adapter
Готовые компоненты
data:image/s3,"s3://crabby-images/cd453/cd453ef7f1476d1c8e560c609d41112011e54b9f" alt=""
VueAudioRecorder
/* boot/audioRecorder.js */
import AudioRecorder from 'vue-audio-recorder'
export default ({ Vue }) => {
Vue.use(AudioRecorder)
}
data:image/s3,"s3://crabby-images/2a83d/2a83dbe4e0f6b3a8a9bbb26532525a932466efd9" alt=""
VueAudioRecorder
/* MyVueAudioRecorder.vue */
<template>
<audio-recorder
upload-url="some url"
:attempts="3"
:time="2"
:before-recording="callback"
:after-recording="callback"
:before-upload="callback"
:successful-upload="callback"
:failed-upload="callback"></audio-recorder>
</template>
<script>
export default {
name: 'MyVueAudioRecorder',
methods: {
callback (msg) {
console.debug('Event: ', msg)
}
}
}
</script>
Полезные ссылки
- https://github.com/yandex/audio-js/blob/master/tutorial/sound.md
- https://developers.google.com/web/fundamentals/media/recording-audio
- https://gist.github.com/borismus/1032746
- https://github.com/mattdiamond/Recorderjs
- https://github.com/higuma/web-audio-recorder-js
- https://recordrtc.org/
- https://github.com/streamproc/MediaStreamRecorder
- https://developer.apple.com/app-store/review/guidelines/
- https://github.com/ai/audio-recorder-polyfill
- https://webrtc.github.io/samples/
- https://webaudioapi.com/samples/
Спасибо за внимание!
tg @omalanova
Copy of Запись аудио в браузере
By Olga Malanova
Copy of Запись аудио в браузере
- 461