Этот код запрашивает у пользователя подтверждение для своего аудиоустройства. После нажатия кнопки запускается startRecord (), и начинается запись. stopRecording () останавливает запись, и как только данные становятся доступными, вызывается функция отправки и Blob отправляется на сервер.
Теперь странная часть: каждый раз, когда данные готовы, функция отправки вызывается i + 1 раз. Таким образом, после второй отправки на сервер приложение пытается отправить большой двоичный объект два раза.
Первое время работает безотказно.
Я также могу заставить его работать, когда
stream = navigator.mediaDevices
.getUserMedia({ audio: true })
.then(stream => {
recorder = new MediaRecorder(stream);
});
Вызывается внутри startRecording (). Но затем Firefox запрашивает разрешение каждый раз, когда Пользователь нажимает кнопку, что также не является хорошим UX.
Есть ли способ спросить пользователя только один раз без постоянной повторной отправки сообщения? Также странным поведением является то, что слушатель в «dataavailable» запускается только один раз (проверено с помощью console.info ()), но код из отправляющей функции вызывается i + 1 раз.
Я надеюсь кто-нибудь сможет помочь
export class Speech extends React.Component {
constructor(props) {
super(props);
this.state = {
isRecording: "false"
};
}
componentDidMount() {
window.MediaRecorder = require("audio-recorder-polyfill");
if (!window.MediaRecorder) {
document.write(decodeURI('%3Cscript
src = "/polyfill.js">%3C/script>'));
}
//I´ve put the navigator.MediaRecorder in the ComponentDidMount, that
it only asks the user one time for microphone confirmation.
stream = navigator.mediaDevices
.getUserMedia({ audio: true })
.then(stream => {
recorder = new MediaRecorder(stream);
});
}
recordHandler(e) {
e.preventDefault();
e.stopPropagation();
let myReader = new FileReader();
if (this.state.isRecording === "false") {
// Request permissions to record audio
if (recorder) {
// Set record to <audio> when recording will be finished
recorder.addEventListener("dataavailable", e => {
blobby = URL.createObjectURL(e.data);
reader = myReader.readAsText(e.data);
this.sendMeBaby(e.data);//This is the function that gets
called i+1 time after every click (send)
});
}
if (recorder) {
// Start recording
recorder.start();
MessageDisplay.addRecordSign();
app.updateText();
console.info("recording started");
this.startRecording();
return;
}
}
if (this.state.isRecording === "true") {
this.stopRecording();
return;
}
}
startRecording = () => {
buttonText = "Recording...";
this.setState({
isRecording: "true"
});
};
stopRecording = () => {
// Stop recording
if (recorder) {
recorder.stop();
// Remove “recording” icon from browser tab
recorder.stream.getTracks().forEach(i => i.stop());
}
buttonText = "Record";
this.setState({
isRecording: "false"
});
};
sendFunctionWithAjaxCall(){
sampleCode()
}
render() {
return (
<Button
variant = "contained"
color = "secondary"
onClick = {e => {
this.recordHandler(e);
}}
>
{buttonText}
</Button>
);
}
Привет, спасибо за новое понимание. Это вроде как делает то, что должно. Единственная проблема в том, что голосовая запись впоследствии выдает ошибочные (непригодные для использования) данные. Но это может быть другая проблема
Так что раньше это были пригодные для использования данные? И он отправлял два одинаковых данных или первый был непригоден для использования, как при подходе once?
Также проверьте документацию MDN по этому поводу: developer.mozilla.org/en-US/docs/Web/API/MediaRecorder/…. Похоже, его нужно вызывать несколько раз.
Я вижу сейчас. Кажется, единственный способ заставить это работать - позволить приложению все время слушать. При нажатии на запись можно запустить временной отрезок для записи и вырезать отрезок с помощью метода stop (). Спасибо за помощь и понимание медиа API.



![Безумие обратных вызовов в javascript [JS]](https://i.imgur.com/WsjO6zJb.png)


Согласно документации MediaRecorder событие dataavailable предназначено для многократного вызова, его можно использовать для срезки записи с определенным интервалом, чтобы данные были разбиты на более мелкие части.
Я предполагаю, что он должен быть инициализирован один раз, а затем при щелчке только начать новую запись, это также устранит проблему FF (он будет запрашивать разрешения один раз при загрузке страницы). Таким образом, с stop будет вызвано событие, которое позволит вам сохранить часть записывающего устройства (между вызовами start и stop).
Да, это то, что я сразу же попробовал. Но это действительно работает, если посмотреть на Демо: mdn.github.io/web-dictaphone Он явно заставил его работать. ТАК есть способ. Может не хватает концентрации. Завтра будет еще один день, на который я откажусь от сегодняшнего. Очень признателен за вашу помощь!
Попробуйте добавить прослушиватель событий с опцией
once: true, чтобы гарантировать, что он вызывается один раз:addEventListener('event', callback, {once: true}). Как если бы метод, который вызывается дважды, вызывается только в прослушивателе событий, тогда прослушиватель событий вызывается дважды. Консоль группирует одинаковые выходные данные, поэтому, вероятно, было два вызова.