Socket
Socket
Sign inDemoInstall

react-say

Package Overview
Dependencies
Maintainers
1
Versions
52
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

react-say - npm Package Compare versions

Comparing version 1.0.1-master.cfdb77a to 1.0.1-master.de311fd

5

CHANGELOG.md

@@ -16,6 +16,9 @@ # Changelog

- Lerna bootstrap will no longer hoist
- Updated playground
- Playground: Bump to `react@16.6.0`, `react-dom@16.6.0`, and `react-scripts@2.0.5`
### Fixed
- Null reference on `props.speechSynthesis`
## [1.0.0] - 2018-07-09
### Added
- Initial release

16

lib/Composer.js

@@ -205,4 +205,9 @@ 'use strict';

props.speechSynthesis.addEventListener && props.speechSynthesis.addEventListener('voiceschanged', _this2.handleVoicesChanged);
var voices = [];
if (props.speechSynthesis) {
props.speechSynthesis.addEventListener && props.speechSynthesis.addEventListener('voiceschanged', _this2.handleVoicesChanged);
voices = props.speechSynthesis.getVoices();
}
_this2.mergeContext = (0, _memoizeOne2.default)(function (_ref5, voices) {

@@ -223,3 +228,3 @@ var cancel = _ref5.cancel,

}),
voices: props.speechSynthesis.getVoices()
voices: voices
};

@@ -248,8 +253,11 @@ return _this2;

var nextVoices = [];
if (nextProps.speechSynthesis) {
nextProps.speechSynthesis.addEventListener && nextProps.speechSynthesis.addEventListener('voiceschanged', this.handleVoicesChanged);
nextVoices = nextProps.speechSynthesis.getVoices() || [];
}
this.setState(function () {
return { voices: nextProps.speechSynthesis.getVoices() };
return { voices: nextVoices };
});

@@ -319,2 +327,2 @@ }

};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Composer.js"],"names":["SpeechContext","ponyfill","queueWithCurrent","cancel","bind","cancelAll","speak","setPonyfill","speechSynthesis","SpeechSynthesisUtterance","id","index","findIndex","utterance","utteranceLike","find","Utterance","length","_next","deferred","promise","then","filter","Composer","props","handleVoicesChanged","addEventListener","mergeContext","voices","state","context","speechSynthesisUtterance","getVoices","nextProps","changed","some","name","removeEventListener","setState","target","children","React","Component","defaultProps","window","webkitSpeechSynthesis","webkitSpeechSynthesisUtterance","propTypes","PropTypes","any"],"mappings":"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","file":"Composer.js","sourcesContent":["import memoize from 'memoize-one';\nimport PropTypes from 'prop-types';\nimport React from 'react';\n\nimport Context from './Context';\nimport Utterance from './Utterance';\n\nclass SpeechContext {\n  constructor(ponyfill) {\n    this.queueWithCurrent = [];\n\n    this.cancel = this.cancel.bind(this);\n    this.cancelAll = this.cancelAll.bind(this);\n    this.speak = this.speak.bind(this);\n\n    this.setPonyfill(ponyfill);\n  }\n\n  setPonyfill({ speechSynthesis, SpeechSynthesisUtterance }) {\n    this.ponyfill = { speechSynthesis, SpeechSynthesisUtterance };\n  }\n\n  async cancel(id) {\n    const index = this.queueWithCurrent.findIndex(utterance => utterance.id === id);\n\n    if (~index) {\n      return this.queueWithCurrent[index].cancel();\n    }\n  }\n\n  async cancelAll() {\n    // console.debug(`CANCELLING QUEUED ITEMS: ${ this.queueWithCurrent.length }`);\n\n    // this.queueWithCurrent.forEach(entry => entry.cancelled = true);\n\n    // const cancelAll = Promise.all(this.queueWithCurrent.map(({ deferred: { promise } }) => promise.catch(err => 0)));\n\n    // this.ponyfill.speechSynthesis.cancel();\n\n    // try {\n    //   await cancelAll;\n    // } catch (err) {}\n\n    // console.debug(`ALL CANCELLED OR FINISHED`);\n  }\n\n  speak(utteranceLike) {\n    // console.debug(`QUEUED: ${ utteranceLike.text }`);\n\n    if (\n      utteranceLike.id\n      && this.queueWithCurrent.find(({ id }) => id === utteranceLike.id)\n    ) {\n      // Do not queue duplicated speak with same unique ID\n      // console.debug('NOT QUEUEING DUPE');\n\n      return;\n    }\n\n    const utterance = new Utterance(utteranceLike);\n\n    this.queueWithCurrent = [...this.queueWithCurrent, utterance];\n\n    if (this.queueWithCurrent.length === 1) {\n      this._next();\n    }\n\n    return utterance.deferred.promise;\n  }\n\n  _next() {\n    const utterance = this.queueWithCurrent[0];\n\n    if (!utterance) { return; }\n\n    const { id } = utterance;\n    const promise = utterance.speak(this.ponyfill);\n\n    promise.then(() => {\n      this.queueWithCurrent = this.queueWithCurrent.filter(utterance => utterance.id !== id);\n      this._next();\n    }, () => {\n      // TODO: If the error is due to Safari restriction on user touch\n      //       The next loop on the next audio will also fail because it was not queued with a user touch\n      this.queueWithCurrent = this.queueWithCurrent.filter(utterance => utterance.id !== id);\n      this._next();\n    });\n  }\n}\n\nexport default class Composer extends React.Component {\n  constructor(props) {\n    super(props);\n\n    this.handleVoicesChanged = this.handleVoicesChanged.bind(this);\n\n    props.speechSynthesis.addEventListener && props.speechSynthesis.addEventListener('voiceschanged', this.handleVoicesChanged);\n\n    this.mergeContext = memoize(({ cancel, speak }, voices) => ({\n      cancel,\n      speak,\n      voices\n    }));\n\n    this.state = {\n      context: new SpeechContext({\n        speechSynthesis: props.speechSynthesis,\n        SpeechSynthesisUtterance: props.speechSynthesisUtterance\n      }),\n      voices: props.speechSynthesis.getVoices()\n    };\n  }\n\n  componentWillReceiveProps(nextProps) {\n    const { props } = this;\n    const changed = [\n      'speechSynthesis',\n      'speechSynthesisUtterance'\n    ].some(name => nextProps[name] !== props[name]);\n\n    if (changed) {\n      if (props.speechSynthesis) {\n        props.speechSynthesis.removeEventListener && props.speechSynthesis.removeEventListener('voiceschanged', this.handleVoicesChanged);\n      }\n\n      this.state.context.setPonyfill({\n        speechSynthesis: nextProps.speechSynthesis,\n        SpeechSynthesisUtterance: nextProps.speechSynthesisUtterance\n      });\n\n      if (nextProps.speechSynthesis) {\n        nextProps.speechSynthesis.addEventListener && nextProps.speechSynthesis.addEventListener('voiceschanged', this.handleVoicesChanged);\n      }\n\n      this.setState(() => ({ voices: nextProps.speechSynthesis.getVoices() }));\n    }\n  }\n\n  componentWillUnmount() {\n    const { speechSynthesis } = this.props;\n\n    speechSynthesis && speechSynthesis.removeEventListener && speechSynthesis.removeEventListener('voiceschanged', this.handleVoicesChanged);\n  }\n\n  handleVoicesChanged({ target }) {\n    this.setState(() => ({ voices: target.getVoices() }));\n  }\n\n  render() {\n    const { props, state } = this;\n    const { children } = props;\n\n    return (\n      <Context.Consumer>\n        { context => context ?\n            typeof children === 'function' ? children(context) : children\n          :\n            <Context.Provider value={ this.mergeContext(state.context, state.voices) }>\n              {\n                typeof children === 'function' ?\n                  <Context.Consumer>\n                    { context => children(context) }\n                  </Context.Consumer>\n                :\n                  children\n              }\n            </Context.Provider>\n        }\n      </Context.Consumer>\n    );\n  }\n}\n\nComposer.defaultProps = {\n  speechSynthesis: window.speechSynthesis || window.webkitSpeechSynthesis,\n  speechSynthesisUtterance: window.SpeechSynthesisUtterance || window.webkitSpeechSynthesisUtterance\n};\n\nComposer.propTypes = {\n  speechSynthesis: PropTypes.any,\n  speechSynthesisUtterance: PropTypes.any\n};\n"]}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Composer.js"],"names":["SpeechContext","ponyfill","queueWithCurrent","cancel","bind","cancelAll","speak","setPonyfill","speechSynthesis","SpeechSynthesisUtterance","id","index","findIndex","utterance","utteranceLike","find","Utterance","length","_next","deferred","promise","then","filter","Composer","props","handleVoicesChanged","voices","addEventListener","getVoices","mergeContext","state","context","speechSynthesisUtterance","nextProps","changed","some","name","removeEventListener","nextVoices","setState","target","children","React","Component","defaultProps","window","webkitSpeechSynthesis","webkitSpeechSynthesisUtterance","propTypes","PropTypes","any"],"mappings":"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","file":"Composer.js","sourcesContent":["import memoize from 'memoize-one';\nimport PropTypes from 'prop-types';\nimport React from 'react';\n\nimport Context from './Context';\nimport Utterance from './Utterance';\n\nclass SpeechContext {\n  constructor(ponyfill) {\n    this.queueWithCurrent = [];\n\n    this.cancel = this.cancel.bind(this);\n    this.cancelAll = this.cancelAll.bind(this);\n    this.speak = this.speak.bind(this);\n\n    this.setPonyfill(ponyfill);\n  }\n\n  setPonyfill({ speechSynthesis, SpeechSynthesisUtterance }) {\n    this.ponyfill = { speechSynthesis, SpeechSynthesisUtterance };\n  }\n\n  async cancel(id) {\n    const index = this.queueWithCurrent.findIndex(utterance => utterance.id === id);\n\n    if (~index) {\n      return this.queueWithCurrent[index].cancel();\n    }\n  }\n\n  async cancelAll() {\n    // console.debug(`CANCELLING QUEUED ITEMS: ${ this.queueWithCurrent.length }`);\n\n    // this.queueWithCurrent.forEach(entry => entry.cancelled = true);\n\n    // const cancelAll = Promise.all(this.queueWithCurrent.map(({ deferred: { promise } }) => promise.catch(err => 0)));\n\n    // this.ponyfill.speechSynthesis.cancel();\n\n    // try {\n    //   await cancelAll;\n    // } catch (err) {}\n\n    // console.debug(`ALL CANCELLED OR FINISHED`);\n  }\n\n  speak(utteranceLike) {\n    // console.debug(`QUEUED: ${ utteranceLike.text }`);\n\n    if (\n      utteranceLike.id\n      && this.queueWithCurrent.find(({ id }) => id === utteranceLike.id)\n    ) {\n      // Do not queue duplicated speak with same unique ID\n      // console.debug('NOT QUEUEING DUPE');\n\n      return;\n    }\n\n    const utterance = new Utterance(utteranceLike);\n\n    this.queueWithCurrent = [...this.queueWithCurrent, utterance];\n\n    if (this.queueWithCurrent.length === 1) {\n      this._next();\n    }\n\n    return utterance.deferred.promise;\n  }\n\n  _next() {\n    const utterance = this.queueWithCurrent[0];\n\n    if (!utterance) { return; }\n\n    const { id } = utterance;\n    const promise = utterance.speak(this.ponyfill);\n\n    promise.then(() => {\n      this.queueWithCurrent = this.queueWithCurrent.filter(utterance => utterance.id !== id);\n      this._next();\n    }, () => {\n      // TODO: If the error is due to Safari restriction on user touch\n      //       The next loop on the next audio will also fail because it was not queued with a user touch\n      this.queueWithCurrent = this.queueWithCurrent.filter(utterance => utterance.id !== id);\n      this._next();\n    });\n  }\n}\n\nexport default class Composer extends React.Component {\n  constructor(props) {\n    super(props);\n\n    this.handleVoicesChanged = this.handleVoicesChanged.bind(this);\n\n    let voices = [];\n\n    if (props.speechSynthesis) {\n      props.speechSynthesis.addEventListener && props.speechSynthesis.addEventListener('voiceschanged', this.handleVoicesChanged);\n      voices = props.speechSynthesis.getVoices();\n    }\n\n    this.mergeContext = memoize(({ cancel, speak }, voices) => ({\n      cancel,\n      speak,\n      voices\n    }));\n\n    this.state = {\n      context: new SpeechContext({\n        speechSynthesis: props.speechSynthesis,\n        SpeechSynthesisUtterance: props.speechSynthesisUtterance\n      }),\n      voices\n    };\n  }\n\n  componentWillReceiveProps(nextProps) {\n    const { props } = this;\n    const changed = [\n      'speechSynthesis',\n      'speechSynthesisUtterance'\n    ].some(name => nextProps[name] !== props[name]);\n\n    if (changed) {\n      if (props.speechSynthesis) {\n        props.speechSynthesis.removeEventListener && props.speechSynthesis.removeEventListener('voiceschanged', this.handleVoicesChanged);\n      }\n\n      this.state.context.setPonyfill({\n        speechSynthesis: nextProps.speechSynthesis,\n        SpeechSynthesisUtterance: nextProps.speechSynthesisUtterance\n      });\n\n      let nextVoices = [];\n\n      if (nextProps.speechSynthesis) {\n        nextProps.speechSynthesis.addEventListener && nextProps.speechSynthesis.addEventListener('voiceschanged', this.handleVoicesChanged);\n        nextVoices = nextProps.speechSynthesis.getVoices() || [];\n      }\n\n      this.setState(() => ({ voices: nextVoices }));\n    }\n  }\n\n  componentWillUnmount() {\n    const { speechSynthesis } = this.props;\n\n    speechSynthesis && speechSynthesis.removeEventListener && speechSynthesis.removeEventListener('voiceschanged', this.handleVoicesChanged);\n  }\n\n  handleVoicesChanged({ target }) {\n    this.setState(() => ({ voices: target.getVoices() }));\n  }\n\n  render() {\n    const { props, state } = this;\n    const { children } = props;\n\n    return (\n      <Context.Consumer>\n        { context => context ?\n            typeof children === 'function' ? children(context) : children\n          :\n            <Context.Provider value={ this.mergeContext(state.context, state.voices) }>\n              {\n                typeof children === 'function' ?\n                  <Context.Consumer>\n                    { context => children(context) }\n                  </Context.Consumer>\n                :\n                  children\n              }\n            </Context.Provider>\n        }\n      </Context.Consumer>\n    );\n  }\n}\n\nComposer.defaultProps = {\n  speechSynthesis: window.speechSynthesis || window.webkitSpeechSynthesis,\n  speechSynthesisUtterance: window.SpeechSynthesisUtterance || window.webkitSpeechSynthesisUtterance\n};\n\nComposer.propTypes = {\n  speechSynthesis: PropTypes.any,\n  speechSynthesisUtterance: PropTypes.any\n};\n"]}

@@ -50,3 +50,4 @@ 'use strict';

function createNativeUtterance(utteranceLike, ponyfill) {
var SpeechSynthesisUtterance = ponyfill.SpeechSynthesisUtterance;
var speechSynthesis = ponyfill.speechSynthesis,
SpeechSynthesisUtterance = ponyfill.SpeechSynthesisUtterance;
var lang = utteranceLike.lang,

@@ -99,3 +100,5 @@ onBoundary = utteranceLike.onBoundary,

if (utterance.addEventListener) {
utterance.addEventListener('boundary', onBoundary);
if (onBoundary) {
utterance.addEventListener('boundary', onBoundary);
}

@@ -308,2 +311,2 @@ // Since browser quirks, start/error/end events are emulated for best compatibility

exports.default = Utterance;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Utterance.js"],"names":["createNativeUtterance","utteranceLike","ponyfill","SpeechSynthesisUtterance","lang","onBoundary","pitch","rate","text","voice","volume","utterance","targetVoice","call","speechSynthesis","getVoices","voiceURI","find","slice","v","addEventListener","Utterance","cancelled","deferred","id","speaking","cancel","promise","_speak","then","onEnd","type","resolve","onError","error","reject","Error","startDeferred","errorDeferred","endDeferred","speak","race","onStart","artificial","endEvent"],"mappings":"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","file":"Utterance.js","sourcesContent":["import createDeferred from './createDeferred';\nimport retry from './retry';\nimport sleep from './sleep';\nimport spinWaitUntil from './spinWaitUntil';\nimport timeout from './timeout';\n\nfunction createNativeUtterance(utteranceLike, ponyfill) {\n  const { SpeechSynthesisUtterance } = ponyfill;\n  const {\n    lang,\n    onBoundary,\n    pitch = 1,\n    rate = 1,\n    text,\n    voice,\n    volume = 1\n  } = utteranceLike;\n  const utterance = new SpeechSynthesisUtterance(text);\n  let targetVoice;\n\n  if (typeof voice === 'function') {\n    targetVoice = voice.call(speechSynthesis, speechSynthesis.getVoices());\n  } else {\n    const { voiceURI } = voice || {};\n\n    targetVoice = voiceURI && [].find.call([].slice.call(speechSynthesis.getVoices()), v => v.voiceURI === voiceURI);\n  }\n\n  // Edge will mute if \"lang\" is set to \"\"\n  utterance.lang = lang || '';\n\n  if (utterance.pitch || utterance.pitch === 0) {\n    utterance.pitch = pitch;\n  }\n\n  if (utterance.rate || utterance.rate === 0) {\n    utterance.rate = rate;\n  }\n\n  // Cognitive Services will error when \"voice\" is set to \"null\"\n  // Edge will error when \"voice\" is set to \"undefined\"\n  if (targetVoice) {\n    utterance.voice = targetVoice;\n  }\n\n  if (utterance.volume || utterance.volume === 0) {\n    utterance.volume = volume;\n  }\n\n  if (utterance.addEventListener) {\n    utterance.addEventListener('boundary', onBoundary);\n\n    // Since browser quirks, start/error/end events are emulated for best compatibility\n  }\n\n  return utterance;\n}\n\nexport default class Utterance {\n  constructor(utteranceLike) {\n    this.cancelled = false;\n    this.deferred = createDeferred();\n    this.id = utteranceLike.id;\n    this.speaking = false;\n    this.utteranceLike = utteranceLike;\n  }\n\n  cancel() {\n    this.cancelled = true;\n\n    this.speaking && this.ponyfill.speechSynthesis.cancel();\n\n    return this.deferred.promise;\n  }\n\n  speak(ponyfill) {\n    this._speak(ponyfill).then(() => {\n      this.utteranceLike.onEnd && this.utteranceLike.onEnd({ type: 'end' });\n      this.deferred.resolve();\n    }, error => {\n      this.utteranceLike.onError && this.utteranceLike.onError({ type: 'error', error });\n      this.deferred.reject(error);\n    });\n\n    return this.deferred.promise;\n  }\n\n  async _speak(ponyfill) {\n    if (this.cancelled) {\n      throw new Error('cancelled');\n    }\n\n    this.ponyfill = ponyfill;\n\n    const { speechSynthesis } = ponyfill;\n\n    const utterance = createNativeUtterance(this.utteranceLike, ponyfill);\n    const startDeferred = createDeferred();\n    const errorDeferred = createDeferred();\n    const endDeferred = createDeferred();\n\n    utterance.addEventListener('end', endDeferred.resolve);\n    utterance.addEventListener('error', errorDeferred.resolve);\n    utterance.addEventListener('start', startDeferred.resolve);\n\n    // if (speechSynthesis.speaking) {\n    //   console.warn(`ASSERTION: speechSynthesis.speaking should not be truthy before we call speak`);\n    // }\n\n    // Chrome quirks:\n    // 1. Speak an utterance\n    // 2. Cancel in the midway\n    // 3. Speak another utterance\n    // Expected: speaking is falsy, then turn to truthy, then receive \"start\" event, and audio played\n    // Actual: speaking is falsy, then turn to truthy (which is wrong), but receive no \"start\" event, and no audio played\n    // Workaround: retry 2 times with a second\n\n    // Safari quirks:\n    // - Audio doesn't play if the speech is started from a user event\n    // - If no audio is played, the \"start\" event won't fire\n\n    // For Chrome quirks, we need a custom queue, because we need to definitely know when to expect a \"start\" event.\n    // If we don't have a queue, the \"start\" event could be happening long time later because it's still pending in the queue.\n\n    // But with the custom queue, the first item might be started from non-user event. That means in Safari, the first item is muted.\n    // And after the first fail, the custom queue will play the second item from a non-user event code path. That means, all subsequent\n    // items are blocked until Safari has the very first item queued from user event.\n\n    // console.debug(`STARTING: ${ utterance.text }`);\n\n    this.speaking = true;\n\n    await retry(async () => {\n      if (this.cancelled) {\n        throw new Error('cancelled');\n      }\n\n      speechSynthesis.speak(utterance);\n\n      try {\n        await Promise.race([\n          startDeferred.promise,\n          timeout(1000)\n        ]);\n      } catch (error) {\n        // This is required for Chrome quirks.\n        // Chrome doesn't know it can't start speech, and it just wait there forever.\n        // We need to cancel it out.\n        speechSynthesis.cancel();\n\n        throw error;\n      }\n    }, 2, 0);\n\n    // console.debug(`STARTED: ${ utterance.text }`);\n\n    utterance.onStart && utterance.onStart({ type: 'start' });\n\n    const endEvent = await Promise.race([\n      errorDeferred.promise,\n      endDeferred.promise,\n      spinWaitUntil(() => !speechSynthesis.speaking).then(() => sleep(500)).then(() => ({ type: 'end', artificial: true }))\n    ]);\n\n    // if (speechSynthesis.speaking) {\n    //   console.warn(`ASSERTION: speechSynthesis.speaking should not be truthy after speak is stopped`);\n    // }\n\n    // console.debug(`ENDED: ${ utterance.text }`);\n\n    this.speaking = false;\n\n    if (endEvent.type === 'error') {\n      throw endEvent.error;\n    } else if (this.cancelled) {\n      throw new Error('cancelled');\n    }\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Utterance.js"],"names":["createNativeUtterance","utteranceLike","ponyfill","speechSynthesis","SpeechSynthesisUtterance","lang","onBoundary","pitch","rate","text","voice","volume","utterance","targetVoice","call","getVoices","voiceURI","find","slice","v","addEventListener","Utterance","cancelled","deferred","id","speaking","cancel","promise","_speak","then","onEnd","type","resolve","onError","error","reject","Error","startDeferred","errorDeferred","endDeferred","speak","race","onStart","artificial","endEvent"],"mappings":"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","file":"Utterance.js","sourcesContent":["import createDeferred from './createDeferred';\nimport retry from './retry';\nimport sleep from './sleep';\nimport spinWaitUntil from './spinWaitUntil';\nimport timeout from './timeout';\n\nfunction createNativeUtterance(utteranceLike, ponyfill) {\n  const { speechSynthesis, SpeechSynthesisUtterance } = ponyfill;\n  const {\n    lang,\n    onBoundary,\n    pitch = 1,\n    rate = 1,\n    text,\n    voice,\n    volume = 1\n  } = utteranceLike;\n  const utterance = new SpeechSynthesisUtterance(text);\n  let targetVoice;\n\n  if (typeof voice === 'function') {\n    targetVoice = voice.call(speechSynthesis, speechSynthesis.getVoices());\n  } else {\n    const { voiceURI } = voice || {};\n\n    targetVoice = voiceURI && [].find.call([].slice.call(speechSynthesis.getVoices()), v => v.voiceURI === voiceURI);\n  }\n\n  // Edge will mute if \"lang\" is set to \"\"\n  utterance.lang = lang || '';\n\n  if (utterance.pitch || utterance.pitch === 0) {\n    utterance.pitch = pitch;\n  }\n\n  if (utterance.rate || utterance.rate === 0) {\n    utterance.rate = rate;\n  }\n\n  // Cognitive Services will error when \"voice\" is set to \"null\"\n  // Edge will error when \"voice\" is set to \"undefined\"\n  if (targetVoice) {\n    utterance.voice = targetVoice;\n  }\n\n  if (utterance.volume || utterance.volume === 0) {\n    utterance.volume = volume;\n  }\n\n  if (utterance.addEventListener) {\n    if (onBoundary) {\n      utterance.addEventListener('boundary', onBoundary);\n    }\n\n    // Since browser quirks, start/error/end events are emulated for best compatibility\n  }\n\n  return utterance;\n}\n\nexport default class Utterance {\n  constructor(utteranceLike) {\n    this.cancelled = false;\n    this.deferred = createDeferred();\n    this.id = utteranceLike.id;\n    this.speaking = false;\n    this.utteranceLike = utteranceLike;\n  }\n\n  cancel() {\n    this.cancelled = true;\n\n    this.speaking && this.ponyfill.speechSynthesis.cancel();\n\n    return this.deferred.promise;\n  }\n\n  speak(ponyfill) {\n    this._speak(ponyfill).then(() => {\n      this.utteranceLike.onEnd && this.utteranceLike.onEnd({ type: 'end' });\n      this.deferred.resolve();\n    }, error => {\n      this.utteranceLike.onError && this.utteranceLike.onError({ type: 'error', error });\n      this.deferred.reject(error);\n    });\n\n    return this.deferred.promise;\n  }\n\n  async _speak(ponyfill) {\n    if (this.cancelled) {\n      throw new Error('cancelled');\n    }\n\n    this.ponyfill = ponyfill;\n\n    const { speechSynthesis } = ponyfill;\n\n    const utterance = createNativeUtterance(this.utteranceLike, ponyfill);\n    const startDeferred = createDeferred();\n    const errorDeferred = createDeferred();\n    const endDeferred = createDeferred();\n\n    utterance.addEventListener('end', endDeferred.resolve);\n    utterance.addEventListener('error', errorDeferred.resolve);\n    utterance.addEventListener('start', startDeferred.resolve);\n\n    // if (speechSynthesis.speaking) {\n    //   console.warn(`ASSERTION: speechSynthesis.speaking should not be truthy before we call speak`);\n    // }\n\n    // Chrome quirks:\n    // 1. Speak an utterance\n    // 2. Cancel in the midway\n    // 3. Speak another utterance\n    // Expected: speaking is falsy, then turn to truthy, then receive \"start\" event, and audio played\n    // Actual: speaking is falsy, then turn to truthy (which is wrong), but receive no \"start\" event, and no audio played\n    // Workaround: retry 2 times with a second\n\n    // Safari quirks:\n    // - Audio doesn't play if the speech is started from a user event\n    // - If no audio is played, the \"start\" event won't fire\n\n    // For Chrome quirks, we need a custom queue, because we need to definitely know when to expect a \"start\" event.\n    // If we don't have a queue, the \"start\" event could be happening long time later because it's still pending in the queue.\n\n    // But with the custom queue, the first item might be started from non-user event. That means in Safari, the first item is muted.\n    // And after the first fail, the custom queue will play the second item from a non-user event code path. That means, all subsequent\n    // items are blocked until Safari has the very first item queued from user event.\n\n    // console.debug(`STARTING: ${ utterance.text }`);\n\n    this.speaking = true;\n\n    await retry(async () => {\n      if (this.cancelled) {\n        throw new Error('cancelled');\n      }\n\n      speechSynthesis.speak(utterance);\n\n      try {\n        await Promise.race([\n          startDeferred.promise,\n          timeout(1000)\n        ]);\n      } catch (error) {\n        // This is required for Chrome quirks.\n        // Chrome doesn't know it can't start speech, and it just wait there forever.\n        // We need to cancel it out.\n        speechSynthesis.cancel();\n\n        throw error;\n      }\n    }, 2, 0);\n\n    // console.debug(`STARTED: ${ utterance.text }`);\n\n    utterance.onStart && utterance.onStart({ type: 'start' });\n\n    const endEvent = await Promise.race([\n      errorDeferred.promise,\n      endDeferred.promise,\n      spinWaitUntil(() => !speechSynthesis.speaking).then(() => sleep(500)).then(() => ({ type: 'end', artificial: true }))\n    ]);\n\n    // if (speechSynthesis.speaking) {\n    //   console.warn(`ASSERTION: speechSynthesis.speaking should not be truthy after speak is stopped`);\n    // }\n\n    // console.debug(`ENDED: ${ utterance.text }`);\n\n    this.speaking = false;\n\n    if (endEvent.type === 'error') {\n      throw endEvent.error;\n    } else if (this.cancelled) {\n      throw new Error('cancelled');\n    }\n  }\n}\n"]}
{
"name": "react-say",
"version": "1.0.1-master.cfdb77a",
"version": "1.0.1-master.de311fd",
"description": "A React component that synthesis text into speech using Web Speech API",

@@ -22,4 +22,5 @@ "keywords": [

"bootstrap": "lerna bootstrap",
"build": "lerna run build",
"test": "lerna run test"
"build": "lerna run --stream build",
"test": "lerna run --stream test",
"watch": "lerna run --parallel --stream watch"
},

@@ -26,0 +27,0 @@ "author": "William Wong <compulim@hotmail.com> (http://compulim.info/)",

@@ -102,10 +102,15 @@ # react-say

* If `<Say>` or `<SayButton>` is unmounted, the utterance will continue to speak
* Web Speech does not support stopping or dequeueing a pending utterance, only to cancel all utterances at once
* Since the utterance is already queued, speaking, or spoken, we cannot stop it and not cancelling other "innocent" utterance
* We decided to let it continue but not firing any events after unmount
* Instead of using the native queue for utterances, we implement our own speech queue for browser compatibility reasons
* Queue is managed by `<Composer>`, all `<Say>` and `<SayButton>` inside the same `<Composer>` will share the same queue
* Native queue does not support partial cancel, when `cancel` is called, all pending utterances are stopped
* If `<Say>` or `<SayButton>` is unmounted, the utterance can be stopped without affecting other pending utterances
* Utterance order can be changed on-the-fly
* Browser quirks
* Chrome: if `cancel` and `speak` are called repeatedly, `speak` will appear to succeed (`speaking === true`) but audio is never played (`start` event is never fired)
* Safari: when speech is not triggered by user event (e.g. mouse click or tap), the speech will not be played
* Workaround: on page load, prime the speech engine by any user events
# Roadmap
* [ ] Prettify playground page
* [x] Prettify playground page

@@ -112,0 +117,0 @@ # Contributions

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc