-
Notifications
You must be signed in to change notification settings - Fork 7
/
speech.js
103 lines (84 loc) · 3.09 KB
/
speech.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/**
* Copyright 2018 Google Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* @author ebidel@ (Eric Bidelman)
*/
/**
* Uses the web speech synth API to make the browser talk.
*
* Run it:
* node speech.js -t Hello there, my name is Jarvis.
* node speech.js -t Read anything good lately?
* CHROME_PATH=/path/to/chrome node speech.js -t hi and bye!
*/
// Inspiration: https://github.com/GoogleChromeLabs/puppeteer-examples/blob/master/speech.js
const argv = require('yargs').argv;
const puppeteer = require('puppeteer');
const DEFAULT_TXT = 'Hello there, my name is Puppeteer. I am controlling your browser.';
const DEFAULT_VOICE = 'Milena';
const executablePath = process.env.CHROME_PATH ||
'/Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome';
const html = (voiceName) => `
<button>speak</button>
<script>
async function speak(text) {
const msg = new SpeechSynthesisUtterance();
msg.text = text;
// msg.volume = 1; // 0 to 1
// msg.rate = 1; // 0.1 to 10
// msg.pitch = 1; //0 to 2
// msg.lang = this.DEST_LANG;
msg.voice = await new Promise(resolve => {
// Voice are populated, async.
speechSynthesis.onvoiceschanged = (e) => {
const voices = window.speechSynthesis.getVoices();
// for (const voice of voices) {
// console.log(voice.name, voice.localService)
// }
const name = "${voiceName}"; // Note: only works in Google Chrome.
resolve(voices.find(voice => voice.name === name));
};
});
msg.onend = e => console.log('SPEECH_DONE');
speechSynthesis.speak(msg);
}
const button = document.querySelector('button');
button.addEventListener('click', e => speak(TEXT2SPEECH));
</script>`;
(async() => {
const browser = await puppeteer.launch({
executablePath, // Note: need Chrome (not Chromium) to use non-default voices.
headless: false, // Speech synth API doesn't work in headless. crbug.com/815388
args: [
'--window-size=0,0', // Launch baby window for fun.
'--window-position=0,0',
'--enable-speech-dispatcher', // Needed for Linux?
],
});
const page = await browser.newPage();
// Clever way to "communicate with page". Know when speech is done.
page.on('console', async msg => {
if (msg.text() === 'SPEECH_DONE') {
await browser.close();
}
});
const text = argv.text || DEFAULT_TXT;
const voiceName = argv.voice || DEFAULT_VOICE;
await page.evaluateOnNewDocument(text => window.TEXT2SPEECH = text, text);
// Cause a navigation so the evaluateOnNewDocument kicks in.
await page.goto(`data:text/html,${html(voiceName)}`);
const button = await page.$('button');
button.click();
})();