Функция Reactjs отправляет данные i + 1 раз

Этот код запрашивает у пользователя подтверждение для своего аудиоустройства. После нажатия кнопки запускается 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: true, чтобы гарантировать, что он вызывается один раз: addEventListener('event', callback, {once: true}). Как если бы метод, который вызывается дважды, вызывается только в прослушивателе событий, тогда прослушиватель событий вызывается дважды. Консоль группирует одинаковые выходные данные, поэтому, вероятно, было два вызова.

extempl 25.10.2018 19:00

Привет, спасибо за новое понимание. Это вроде как делает то, что должно. Единственная проблема в том, что голосовая запись впоследствии выдает ошибочные (непригодные для использования) данные. Но это может быть другая проблема

Derek Haynes 26.10.2018 10:48

Так что раньше это были пригодные для использования данные? И он отправлял два одинаковых данных или первый был непригоден для использования, как при подходе once?

extempl 26.10.2018 10:50

Также проверьте документацию MDN по этому поводу: developer.mozilla.org/en-US/docs/Web/API/MediaRecorder/…. Похоже, его нужно вызывать несколько раз.

extempl 26.10.2018 10:55

Я вижу сейчас. Кажется, единственный способ заставить это работать - позволить приложению все время слушать. При нажатии на запись можно запустить временной отрезок для записи и вырезать отрезок с помощью метода stop (). Спасибо за помощь и понимание медиа API.

Derek Haynes 26.10.2018 13:02
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
В настоящее время производительность загрузки веб-сайта имеет решающее значение не только для удобства пользователей, но и для ранжирования в...
Безумие обратных вызовов в javascript [JS]
Безумие обратных вызовов в javascript [JS]
Здравствуйте! Юный падаван 🚀. Присоединяйся ко мне, чтобы разобраться в одной из самых запутанных концепций, когда вы начинаете изучать мир...
Система управления парковками с использованием HTML, CSS и JavaScript
Система управления парковками с использованием HTML, CSS и JavaScript
Веб-сайт по управлению парковками был создан с использованием HTML, CSS и JavaScript. Это простой сайт, ничего вычурного. Основная цель -...
JavaScript Вопросы с множественным выбором и ответы
JavaScript Вопросы с множественным выбором и ответы
Если вы ищете платформу, которая предоставляет вам бесплатный тест JavaScript MCQ (Multiple Choice Questions With Answers) для оценки ваших знаний,...
0
5
108
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Согласно документации MediaRecorder событие dataavailable предназначено для многократного вызова, его можно использовать для срезки записи с определенным интервалом, чтобы данные были разбиты на более мелкие части.

Я предполагаю, что он должен быть инициализирован один раз, а затем при щелчке только начать новую запись, это также устранит проблему FF (он будет запрашивать разрешения один раз при загрузке страницы). Таким образом, с stop будет вызвано событие, которое позволит вам сохранить часть записывающего устройства (между вызовами start и stop).

Да, это то, что я сразу же попробовал. Но это действительно работает, если посмотреть на Демо: mdn.github.io/web-dictaphone Он явно заставил его работать. ТАК есть способ. Может не хватает концентрации. Завтра будет еще один день, на который я откажусь от сегодняшнего. Очень признателен за вашу помощь!

Derek Haynes 26.10.2018 15:44

Другие вопросы по теме