Javascript заставить работать события клика и распознавания речи с несколькими кнопками - PullRequest
0 голосов
/ 21 марта 2020

Итак, я использую webkitSpeechRecognition, и он отлично работает, но проблема в том, что я хочу использовать его одновременно в нескольких местах, и когда я добавляю его в другой раздел на моей странице, он работает неправильно.

Вот оно работает само по себе:

var final_transcript = '';
var recognizing = false;
var ignore_onend;
var start_timestamp;

//get languages
// https://gist.githubusercontent.com/onigetoc/d5dc63320c4d08633eb7b7daf6c1ddeb/raw/e9c3da4a40db8edf4992db9219526617da811c4c/lang.json
$.get("//raw.githubusercontent.com/the-creature/language-json/master/data.json", function(data) {
  var cList = $('#language');
  var data = $.parseJSON(data);

  $.each(data, function(i) {
    var option = $('<option/>')
      .attr('value', data[i].code)
      .html(data[i].name)
      .appendTo(cList);
  });

  var userLang = navigator.language || navigator.userLanguage;
  $("#language").val(userLang);

});

// Speech Recognition
if (!('webkitSpeechRecognition' in window)) {
  message.innerHTML = 'Web Speech API is not supported by this browser. Upgrade to <a href="//www.google.com/chrome">Chrome</a> version 25 or later.';
} else {
  var recognition = new webkitSpeechRecognition();
  recognition.continuous = true;
  recognition.interimResults = true;

  recognition.onstart = function() {
    recognizing = true;
    message.innerHTML = 'Speak now.';
    talk_button.innerHTML = 'Listen';
  };

  recognition.onresult = function(event) {
    var interim_transcript = '';
    for (var i = event.resultIndex; i < event.results.length; ++i) {
      if (event.results[i].isFinal) {
        final_transcript += event.results[i][0].transcript;
        vc_search(final_transcript);
      } else {
        interim_transcript += event.results[i][0].transcript;
      }
    }
    final_span.innerHTML = final_transcript;
    interim_span.innerHTML = interim_transcript;
  };

  recognition.onend = function() {

    recognizing = false;
    if (ignore_onend) {
      return;
    }
    speechMyText(final_transcript);
    if (!final_transcript) {
      message.innerHTML = 'Click "Talk" and begin speaking.';
      talk_button.innerHTML = 'Talk';
      return;
    }
  };

  recognition.onerror = function(event) {
    if (event.error == 'no-speech') {
      message.innerHTML = 'No speech was detected.';
      ignore_onend = true;
    }
    if (event.error == 'audio-capture') {
      message.innerHTML = 'No microphone was found. Ensure that a microphone is installed.';
      ignore_onend = true;
    }
    if (event.error == 'not-allowed') {
      if (event.timeStamp - start_timestamp < 100) {
        message.innerHTML = 'Permission to use microphone is blocked. To change, go to chrome://settings/contentExceptions#media-stream';
      } else {
        message.innerHTML = 'Permission to use microphone was denied.';
      }
      ignore_onend = true;
    }
  };

}

function talkWithApp(event) {
  if (recognizing) {
    recognition.stop();
    message.innerHTML = 'Click "Talk" and begin speaking.';
    talk_button.innerHTML = 'Talk';
    return;
  }
  final_transcript = '';
  recognition.lang = language.value;
  recognition.start();
  ignore_onend = false;
  final_span.innerHTML = '';
  interim_span.innerHTML = '';
  message.innerHTML = 'Click the "Allow" button above to enable your microphone.';
  start_timestamp = event.timeStamp;
}

// Speech Synthesis
function speechMyText(textToSpeech) {
  var u = new SpeechSynthesisUtterance();
  u.text = textToSpeech;
  u.lang = language.value;
  u.rate = 1.0;
  u.onend = function(event) {}
  speechSynthesis.speak(u);
}
.message {
    color:#999;
    padding: 1em 0;
}
.todo, .response {
    min-height:50px;
    background-color: #fff;
    margin-bottom: 0.5em;
    padding: 1px;
    color:#555;
    -webkit-box-shadow: 0 0px 3px #BDBDBD;
    box-shadow: 0 0px 3px #BDBDBD;
    transition: all 0.3s ease-in-out;
}
<div class="container">
  <div id="message" class="message">Click "Talk" and begin speaking.</div>
  <div class="todo"> <span id="final_span" class="final"></span>
    <span id="interim_span" class="interim"></span>
  </div>
  <div class="controls">
    <button id="talk_button" type="button" class="btn btn-default" onclick="talkWithApp(event)">Talk</button>
    
    <select id="language" class="select">
      <option selected="selected">Select Your Language</option>
    </select>
  </div>
</div>

И вот он работает несколько раз на одной странице:

var final_transcript = '';
var recognizing = false;
var ignore_onend;
var start_timestamp;

//get languages
// https://gist.githubusercontent.com/onigetoc/d5dc63320c4d08633eb7b7daf6c1ddeb/raw/e9c3da4a40db8edf4992db9219526617da811c4c/lang.json
$.get("//raw.githubusercontent.com/the-creature/language-json/master/data.json", function(data) {
  var cList = $('#language');
  var data = $.parseJSON(data);

  $.each(data, function(i) {
    var option = $('<option/>')
      .attr('value', data[i].code)
      .html(data[i].name)
      .appendTo(cList);
  });

  var userLang = navigator.language || navigator.userLanguage;
  $("#language").val(userLang);

});

// Speech Recognition
if (!('webkitSpeechRecognition' in window)) {
  message.innerHTML = 'Web Speech API is not supported by this browser. Upgrade to <a href="//www.google.com/chrome">Chrome</a> version 25 or later.';
} else {
  var recognition = new webkitSpeechRecognition();
  recognition.continuous = true;
  recognition.interimResults = true;

  recognition.onstart = function() {
    recognizing = true;
    message.innerHTML = 'Speak now.';
    talk_button.innerHTML = 'Listen';
  };

  recognition.onresult = function(event) {
    var interim_transcript = '';
    for (var i = event.resultIndex; i < event.results.length; ++i) {
      if (event.results[i].isFinal) {
        final_transcript += event.results[i][0].transcript;
        vc_search(final_transcript);
      } else {
        interim_transcript += event.results[i][0].transcript;
      }
    }
    final_span.innerHTML = final_transcript;
    interim_span.innerHTML = interim_transcript;
  };

  recognition.onend = function() {

    recognizing = false;
    if (ignore_onend) {
      return;
    }
    speechMyText(final_transcript);
    if (!final_transcript) {
      message.innerHTML = 'Click "Talk" and begin speaking.';
      talk_button.innerHTML = 'Talk';
      return;
    }
  };

  recognition.onerror = function(event) {
    if (event.error == 'no-speech') {
      message.innerHTML = 'No speech was detected.';
      ignore_onend = true;
    }
    if (event.error == 'audio-capture') {
      message.innerHTML = 'No microphone was found. Ensure that a microphone is installed.';
      ignore_onend = true;
    }
    if (event.error == 'not-allowed') {
      if (event.timeStamp - start_timestamp < 100) {
        message.innerHTML = 'Permission to use microphone is blocked. To change, go to chrome://settings/contentExceptions#media-stream';
      } else {
        message.innerHTML = 'Permission to use microphone was denied.';
      }
      ignore_onend = true;
    }
  };

}

function talkWithApp(event) {
  if (recognizing) {
    recognition.stop();
    message.innerHTML = 'Click "Talk" and begin speaking.';
    talk_button.innerHTML = 'Talk';
    return;
  }
  final_transcript = '';
  recognition.lang = language.value;
  recognition.start();
  ignore_onend = false;
  final_span.innerHTML = '';
  interim_span.innerHTML = '';
  message.innerHTML = 'Click the "Allow" button above to enable your microphone.';
  start_timestamp = event.timeStamp;
}

// Speech Synthesis
function speechMyText(textToSpeech) {
  var u = new SpeechSynthesisUtterance();
  u.text = textToSpeech;
  u.lang = language.value;
  u.rate = 1.0;
  u.onend = function(event) {}
  speechSynthesis.speak(u);
}
.message {
    color:#999;
    padding: 1em 0;
}
.todo, .response {
    min-height:50px;
    background-color: #fff;
    margin-bottom: 0.5em;
    padding: 1px;
    color:#555;
    -webkit-box-shadow: 0 0px 3px #BDBDBD;
    box-shadow: 0 0px 3px #BDBDBD;
    transition: all 0.3s ease-in-out;
}
<div class="container">
  <div id="message" class="message">Click "Talk" and begin speaking.</div>
  <div class="todo"> <span id="final_span" class="final"></span>
    <span id="interim_span" class="interim"></span>
  </div>
  <div class="controls">
    <button id="talk_button" type="button" class="btn btn-default" onclick="talkWithApp(event)">Talk</button>
    
    <select id="language" class="select">
      <option selected="selected">Select Your Language</option>
    </select>
  </div>
</div>

<div class="container">
  <div id="message" class="message">Click "Talk" and begin speaking.</div>
  <div class="todo"> <span id="final_span" class="final"></span>
    <span id="interim_span" class="interim"></span>
  </div>
  <div class="controls">
    <button id="talk_button" type="button" class="btn btn-default" onclick="talkWithApp(event)">Talk</button>
    
    <select id="language" class="select">
      <option selected="selected">Select Your Language</option>
    </select>
  </div>
</div>

Как вы можете видеть, он работает неправильно на обоих при многократном запуске. Как я могу сделать так, чтобы он использовал только тот раздел, на который нажали? но все еще работает в нескольких местах на странице.

Вот два разных способа:

https://jsfiddle.net/k5cm8ypg/

https://jsfiddle.net/ygpbfLso/1/

1 Ответ

0 голосов
/ 21 марта 2020

Кажется, что в вашем HTML вы используете ID несколько раз; Идентификаторы должны быть уникальными для определенной страницы или фрейма.

Тогда в вашем сценарии и объект распознавания, и ваше событие не могут правильно получить и назначить значения, поскольку идентификатор, к которому они пытаются получить доступ, был объявлен более чем один раз.

То, что я сделал, было;

  1. Удалите все идентификаторы для использования классов

  2. Добавьте глобальный var activeButton

  3. Привязка события разговора ко всему документу и срабатывание при нажатии .talk_button.

  4. При нажатии .talk_button контекст имеет вид установить вокруг этой кнопки. Это сделано с activeButton = $(this);. Вы заметите, что я использовал комбинацию .parent(), .find(), чтобы найти элементы в контексте или контейнере этой кнопки.

Запустите демонстрацию на jsfiddle. С ним было весело работать, я не знал, что в браузере есть готовый речевой API.

https://jsfiddle.net/1pvqx2am/2/

...