Add option for text-to-speech
This commit is contained in:
parent
2f88bcf82c
commit
21a2730cde
@ -287,7 +287,8 @@ function profileOptionsCreateDefaults() {
|
||||
sources: ['jpod101'],
|
||||
volume: 100,
|
||||
autoPlay: false,
|
||||
customSourceUrl: ''
|
||||
customSourceUrl: '',
|
||||
textToSpeechVoice: ''
|
||||
},
|
||||
|
||||
scanning: {
|
||||
|
@ -48,6 +48,7 @@ async function formRead(options) {
|
||||
options.audio.autoPlay = $('#auto-play-audio').prop('checked');
|
||||
options.audio.volume = parseFloat($('#audio-playback-volume').val());
|
||||
options.audio.customSourceUrl = $('#audio-custom-source').val();
|
||||
options.audio.textToSpeechVoice = $('#text-to-speech-voice').val();
|
||||
|
||||
options.scanning.middleMouse = $('#middle-mouse-button-scan').prop('checked');
|
||||
options.scanning.touchInputEnabled = $('#touch-input-enabled').prop('checked');
|
||||
@ -119,6 +120,7 @@ async function formWrite(options) {
|
||||
$('#auto-play-audio').prop('checked', options.audio.autoPlay);
|
||||
$('#audio-playback-volume').val(options.audio.volume);
|
||||
$('#audio-custom-source').val(options.audio.customSourceUrl);
|
||||
$('#text-to-speech-voice').val(options.audio.textToSpeechVoice).attr('data-value', options.audio.textToSpeechVoice);
|
||||
|
||||
$('#middle-mouse-button-scan').prop('checked', options.scanning.middleMouse);
|
||||
$('#touch-input-enabled').prop('checked', options.scanning.touchInputEnabled);
|
||||
@ -326,6 +328,77 @@ async function audioSettingsInitialize() {
|
||||
const options = await apiOptionsGet(optionsContext);
|
||||
audioSourceUI = new AudioSourceUI.Container(options.audio.sources, $('.audio-source-list'), $('.audio-source-add'));
|
||||
audioSourceUI.save = () => apiOptionsSave();
|
||||
|
||||
textToSpeechInitialize();
|
||||
}
|
||||
|
||||
function textToSpeechInitialize() {
|
||||
if (typeof speechSynthesis === 'undefined') { return; }
|
||||
|
||||
speechSynthesis.addEventListener('voiceschanged', () => updateTextToSpeechVoices(), false);
|
||||
updateTextToSpeechVoices();
|
||||
|
||||
$('#text-to-speech-voice-test').on('click', () => textToSpeechTest());
|
||||
}
|
||||
|
||||
function updateTextToSpeechVoices() {
|
||||
const voices = Array.prototype.map.call(speechSynthesis.getVoices(), (voice, index) => ({voice, index}));
|
||||
voices.sort(textToSpeechVoiceCompare);
|
||||
if (voices.length > 0) {
|
||||
$('#text-to-speech-voice-container').css('display', '');
|
||||
}
|
||||
|
||||
const select = $('#text-to-speech-voice');
|
||||
select.empty();
|
||||
select.append($('<option>').val('').text('None'));
|
||||
for (const {voice} of voices) {
|
||||
select.append($('<option>').val(voice.voiceURI).text(`${voice.name} (${voice.lang})`));
|
||||
}
|
||||
|
||||
select.val(select.attr('data-value'));
|
||||
}
|
||||
|
||||
function compareLanguageTags(a, b) {
|
||||
if (a.substr(0, 3) === 'ja-') {
|
||||
return (b.substr(0, 3) === 'ja-') ? 0 : -1;
|
||||
} else {
|
||||
return (b.substr(0, 3) === 'ja-') ? 1 : 0;
|
||||
}
|
||||
}
|
||||
|
||||
function textToSpeechVoiceCompare(a, b) {
|
||||
const i = compareLanguageTags(a.voice.lang, b.voice.lang);
|
||||
if (i !== 0) { return i; }
|
||||
|
||||
if (a.voice.default) {
|
||||
if (!b.voice.default) {
|
||||
return -1;
|
||||
}
|
||||
} else if (b.voice.default) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (a.index < b.index) { return -1; }
|
||||
if (a.index > b.index) { return 1; }
|
||||
return 0;
|
||||
}
|
||||
|
||||
function textToSpeechTest() {
|
||||
try {
|
||||
const text = $('#text-to-speech-voice-test').attr('data-speech-text') || '';
|
||||
const voiceURI = $('#text-to-speech-voice').val();
|
||||
const voice = audioGetTextToSpeechVoice(voiceURI);
|
||||
if (voice === null) { return; }
|
||||
|
||||
const utterance = new SpeechSynthesisUtterance(text);
|
||||
utterance.lang = 'ja-JP';
|
||||
utterance.voice = voice;
|
||||
utterance.volume = 1.0;
|
||||
|
||||
speechSynthesis.speak(utterance);
|
||||
} catch (e) {
|
||||
// NOP
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@ -293,6 +293,16 @@
|
||||
<input type="number" min="0" max="100" id="audio-playback-volume" class="form-control">
|
||||
</div>
|
||||
|
||||
<div class="form-group" style="display: none;" id="text-to-speech-voice-container">
|
||||
<label for="text-to-speech-voice">Text-to-speech voice</label>
|
||||
<div class="input-group">
|
||||
<select class="form-control" id="text-to-speech-voice"></select>
|
||||
<div class="input-group-btn">
|
||||
<button class="btn btn-default" id="text-to-speech-voice-test" title="Test voice" data-speech-text="よみちゃん"><span class="glyphicon glyphicon-volume-up"></span></button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group options-advanced">
|
||||
<label for="audio-custom-source">Custom audio source <span class="label-light">(URL)</span></label>
|
||||
<input type="text" id="audio-custom-source" class="form-control" placeholder="Example: http://localhost/audio.mp3?expression={expression}&reading={reading}">
|
||||
@ -658,6 +668,7 @@
|
||||
<script src="/bg/js/profile-conditions.js"></script>
|
||||
<script src="/bg/js/templates.js"></script>
|
||||
<script src="/bg/js/util.js"></script>
|
||||
<script src="/mixed/js/audio.js"></script>
|
||||
|
||||
<script src="/bg/js/settings-profiles.js"></script>
|
||||
<script src="/bg/js/settings.js"></script>
|
||||
|
@ -58,3 +58,16 @@ async function audioGetFromSources(expression, sources, optionsContext, createAu
|
||||
}
|
||||
return {audio: null, source: null};
|
||||
}
|
||||
|
||||
function audioGetTextToSpeechVoice(voiceURI) {
|
||||
try {
|
||||
for (const voice of speechSynthesis.getVoices()) {
|
||||
if (voice.voiceURI === voiceURI) {
|
||||
return voice;
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
// NOP
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user