A TypeScript library for implementing read aloud features with Web technologies, following best practices for digital publishing.
npm install @readium/speechReadium Speech is a TypeScript library for implementing a read aloud feature with Web technologies. It follows best practices gathered through interviews with members of the digital publishing industry.
While this project is still in a very early stage, it is meant to power the read aloud feature for two different Readium projects: Readium Web and Thorium.
Readium Speech was spun out as a separate project in order to facilitate its integration as a shared component, but also because of its potential outside of the realm of ebook reading apps.
* Extracting Guided Navigation objects from a document (or a fragment of a document)
* Generating utterances from these Guided Navigation objects
* Processing utterances (prepending/appending text to utterances based on context, pronunciation through SSML/PLS…)
* Voice selection
* TTS playback
* Highlighting
For our initial work on this project, we focused on voice selection based on recommended voices.
The outline of this work has been explored in a GitHub discussion and through a best practices document.
In the second phase, we focused on implementing a WebSpeech API-based solution with an architecture designed for future extensibility:
- Engine Layer: Core TTS functionality through ReadiumSpeechPlaybackEngine
- Navigator Layer: Content and playback management via (a temporary) ReadiumSpeechNavigator
- Current Implementation: WebSpeech API with cross-browser compatibility
- Future-Proof Design: Architecture prepared for additional TTS service adapters
Key features include advanced voice selection, cross-browser playback control, flexible content loading, and comprehensive event handling for UI feedback. The architecture is designed to be extensible for different TTS backends while maintaining TypeScript-first development practices.
Two live demos are available:
1. Voice selection with playback demo
2. In-context demo
The first demo showcases the following features:
- fetching a list of all available languages, translating them to the user's locale and sorting them based on these translations
- returning a list of voices for a given language, grouped by region and sorted based on quality
- filtering languages and voices based on gender and offline availability
- using embedded test utterances to demo voices
- using the current Navigator for playback control
The second demo focuses on in-context reading with seamless voice selection (grouped by region and sorted based on quality), and playback control, providing an optional read-along experience that integrates naturally with the content.
Install the package using npm:
``bash`
npm install @readium/speech
Or using yarn:
`bash`
yarn add @readium/speech
`typescript
import { WebSpeechVoiceManager, WebSpeechReadAloudNavigator } from "@readium/speech";
// Initialize voice manager
const voiceManager = await WebSpeechVoiceManager.initialize({
languages: ["en", "fr", "es"] // List of languages to fetch voices for
});
// Get the best available voice for a specific language
const voice = await voiceManager.getDefaultVoice("en-US");
// Create a navigator instance
const navigator = new WebSpeechReadAloudNavigator();
await navigator.setVoice(voice);
// Handle playback events
navigator.on("play", () => console.log("Playback started"));
navigator.on("pause", () => console.log("Playback paused"));
navigator.on("end", () => console.log("Playback completed"));
// Load and play content
const content = document.getElementById("content");
navigator.loadContent(content);
navigator.play();
`
Documentation provides guides for:
- SpeechSynthesis in browsers and OSes
- Voices and Filtering
- API Reference
We are trying to use a test-driven development approach as much as possible, where we write tests before implementing the code. Currently, this is true for the WebSpeechVoiceManager class as it deals primarily with voice selection and management, where mocking is straightforward.
The playback logic is more complex and may not be suitable for this approach yet, as it involves more intricate state management and user interactions that is difficult to handle through mock objects, especially as browsers vary significantly in their implementation of the Web Speech API.
To build the library:
`bash`
npm run build
This will compile the TypeScript code and generate the following outputs in the build/ directory:index.js
- (ES modules)index.cjs
- (CommonJS)
- TypeScript type definitions
The project includes two demo applications that can be served locally:
1. Start the local development server:
`bash`
npm run start
2. Open your browser to:
- Voice selection demo
- In-context reading demo
For ChromeOS development, the project includes a debug mode that mocks the Web Speech API with the set of voices exported from the ChromeOS browser:
1. Open the debug page: http://localhost:8080/debug
2. The debug page loads mock voices from a json file which contains a snapshot of ChromeOS voices.
To run the test suite for WebSpeechVoiceManager:`bash``
npm test