2019-07-10 10:49:29 +02:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Disable symbol overrides so that we can use system headers.
|
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
|
|
|
|
|
|
|
#include "backends/text-to-speech/linux/linux-text-to-speech.h"
|
|
|
|
|
|
|
|
#if defined(USE_LINUX_TTS)
|
2019-07-11 00:14:28 +02:00
|
|
|
#include <speech-dispatcher/libspeechd.h>
|
|
|
|
|
2019-07-10 10:49:29 +02:00
|
|
|
#include "common/translation.h"
|
2019-07-11 00:14:28 +02:00
|
|
|
#include "common/debug.h"
|
|
|
|
#include "common/system.h"
|
2019-07-14 21:48:26 +02:00
|
|
|
#include "common/ustr.h"
|
|
|
|
#include "common/config-manager.h"
|
2019-07-11 00:14:28 +02:00
|
|
|
SPDConnection *_connection;
|
|
|
|
|
|
|
|
void speech_begin_callback(size_t msg_id, size_t client_id, SPDNotificationType state){
|
|
|
|
LinuxTextToSpeechManager *manager =
|
|
|
|
static_cast<LinuxTextToSpeechManager *> (g_system->getTextToSpeechManager());
|
|
|
|
manager->updateState(LinuxTextToSpeechManager::SPEAKING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void speech_end_callback(size_t msg_id, size_t client_id, SPDNotificationType state){
|
|
|
|
LinuxTextToSpeechManager *manager =
|
|
|
|
static_cast<LinuxTextToSpeechManager *> (g_system->getTextToSpeechManager());
|
|
|
|
manager->updateState(LinuxTextToSpeechManager::READY);
|
|
|
|
}
|
|
|
|
|
|
|
|
void speech_cancel_callback(size_t msg_id, size_t client_id, SPDNotificationType state){
|
|
|
|
LinuxTextToSpeechManager *manager =
|
|
|
|
static_cast<LinuxTextToSpeechManager *> (g_system->getTextToSpeechManager());
|
|
|
|
manager->updateState(LinuxTextToSpeechManager::READY);
|
|
|
|
}
|
|
|
|
|
|
|
|
void speech_resume_callback(size_t msg_id, size_t client_id, SPDNotificationType state){
|
|
|
|
LinuxTextToSpeechManager *manager =
|
|
|
|
static_cast<LinuxTextToSpeechManager *> (g_system->getTextToSpeechManager());
|
|
|
|
manager->updateState(LinuxTextToSpeechManager::SPEAKING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void speech_pause_callback(size_t msg_id, size_t client_id, SPDNotificationType state){
|
|
|
|
LinuxTextToSpeechManager *manager =
|
|
|
|
static_cast<LinuxTextToSpeechManager *> (g_system->getTextToSpeechManager());
|
|
|
|
manager->updateState(LinuxTextToSpeechManager::PAUSED);
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxTextToSpeechManager::LinuxTextToSpeechManager()
|
|
|
|
: _speechState(READY) {
|
2019-07-14 21:01:37 +02:00
|
|
|
init();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxTextToSpeechManager::init() {
|
2019-07-11 00:14:28 +02:00
|
|
|
_connection = spd_open("ScummVM", "main", NULL, SPD_MODE_THREADED);
|
|
|
|
if (_connection == 0) {
|
2019-07-11 23:38:06 +02:00
|
|
|
_speechState = BROKEN;
|
2019-07-14 19:34:29 +02:00
|
|
|
warning("Couldn't initialize text to speech through speech-dispatcher");
|
2019-07-11 00:14:28 +02:00
|
|
|
return;
|
|
|
|
}
|
2019-07-10 10:49:29 +02:00
|
|
|
|
2019-07-11 00:14:28 +02:00
|
|
|
_connection->callback_begin = speech_begin_callback;
|
|
|
|
spd_set_notification_on(_connection, SPD_BEGIN);
|
|
|
|
_connection->callback_end = speech_end_callback;
|
|
|
|
spd_set_notification_on(_connection, SPD_END);
|
|
|
|
_connection->callback_cancel = speech_cancel_callback;
|
|
|
|
spd_set_notification_on(_connection, SPD_CANCEL);
|
|
|
|
_connection->callback_resume = speech_resume_callback;
|
|
|
|
spd_set_notification_on(_connection, SPD_RESUME);
|
|
|
|
_connection->callback_pause = speech_pause_callback;
|
|
|
|
spd_set_notification_on(_connection, SPD_PAUSE);
|
|
|
|
|
|
|
|
updateVoices();
|
2019-07-12 00:46:42 +02:00
|
|
|
_ttsState->_activeVoice = 0;
|
2019-07-17 23:51:58 +02:00
|
|
|
#ifdef USE_TRANSLATION
|
|
|
|
setLanguage(TransMan.getCurrentLanguage());
|
|
|
|
#else
|
|
|
|
setLanguage("en");
|
|
|
|
#endif
|
2019-07-10 10:49:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
LinuxTextToSpeechManager::~LinuxTextToSpeechManager() {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_connection != 0)
|
|
|
|
spd_close(_connection);
|
2019-07-11 00:14:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxTextToSpeechManager::updateState(LinuxTextToSpeechManager::SpeechState state) {
|
|
|
|
_speechState = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::say(Common::String str) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return true;
|
2019-07-14 21:48:26 +02:00
|
|
|
//Convert string, that might have foreign characters to UTF-8
|
|
|
|
if (ConfMan.get("gui_language") != "C") {
|
|
|
|
str = Common::convertUtf32ToUtf8(Common::convertToU32String(str.c_str(), Common::kWindows1250)).c_str();
|
|
|
|
}
|
2019-07-11 00:14:28 +02:00
|
|
|
if (isSpeaking())
|
|
|
|
stop();
|
2019-07-14 19:28:04 +02:00
|
|
|
debug("say: %s", str.c_str());
|
2019-07-14 21:01:37 +02:00
|
|
|
if(spd_say(_connection, SPD_MESSAGE, str.c_str()) == -1) {
|
|
|
|
//restart the connection
|
|
|
|
if (_connection != 0)
|
|
|
|
spd_close(_connection);
|
|
|
|
init();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2019-07-11 00:14:28 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::stop() {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == READY || _speechState == BROKEN)
|
|
|
|
return true;
|
2019-07-11 00:14:28 +02:00
|
|
|
return spd_cancel(_connection) == -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::pause() {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == READY || _speechState == PAUSED || _speechState == BROKEN)
|
|
|
|
return true;
|
2019-07-11 00:14:28 +02:00
|
|
|
return spd_pause(_connection) == -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::resume() {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == READY || _speechState == SPEAKING || _speechState == BROKEN)
|
|
|
|
return true;
|
2019-07-11 00:14:28 +02:00
|
|
|
return spd_resume(_connection) == -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::isSpeaking() {
|
2019-07-11 23:38:06 +02:00
|
|
|
return _speechState == SPEAKING;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::isPaused() {
|
|
|
|
return _speechState == PAUSED;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinuxTextToSpeechManager::isReady() {
|
|
|
|
return _speechState == READY;
|
2019-07-11 00:14:28 +02:00
|
|
|
}
|
|
|
|
|
2019-07-12 00:46:42 +02:00
|
|
|
void LinuxTextToSpeechManager::setVoice(unsigned index) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-12 00:46:42 +02:00
|
|
|
assert(index < _ttsState->_availaibleVoices.size());
|
|
|
|
Common::TTSVoice voice = _ttsState->_availaibleVoices[index];
|
|
|
|
spd_set_voice_type(_connection, *(SPDVoiceType *)(voice.getData()));
|
|
|
|
_ttsState->_activeVoice = index;
|
2019-07-11 00:14:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxTextToSpeechManager::setRate(int rate) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-11 00:14:28 +02:00
|
|
|
assert(rate >= -100 && rate <= 100);
|
|
|
|
spd_set_voice_rate(_connection, rate);
|
|
|
|
_ttsState->_rate = rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxTextToSpeechManager::setPitch(int pitch) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-11 00:14:28 +02:00
|
|
|
assert(pitch >= -100 && pitch <= 100);
|
|
|
|
spd_set_voice_pitch(_connection, pitch);
|
|
|
|
_ttsState->_pitch = pitch;
|
|
|
|
}
|
|
|
|
|
2019-07-12 12:52:18 +02:00
|
|
|
void LinuxTextToSpeechManager::setVolume(unsigned volume) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-12 12:52:18 +02:00
|
|
|
assert(volume <= 100);
|
|
|
|
spd_set_volume(_connection, (volume - 50) * 2);
|
2019-07-11 00:14:28 +02:00
|
|
|
_ttsState->_volume = volume;
|
|
|
|
}
|
|
|
|
|
2019-07-12 12:52:18 +02:00
|
|
|
int LinuxTextToSpeechManager::getVolume() {
|
|
|
|
return (_ttsState->_volume - 50) * 2;
|
|
|
|
}
|
|
|
|
|
2019-07-11 00:14:28 +02:00
|
|
|
void LinuxTextToSpeechManager::setLanguage(Common::String language) {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-11 00:14:28 +02:00
|
|
|
spd_set_language(_connection, language.c_str());
|
|
|
|
_ttsState->_language = language;
|
2019-07-12 00:46:42 +02:00
|
|
|
setVoice(_ttsState->_activeVoice);
|
2019-07-11 00:14:28 +02:00
|
|
|
}
|
|
|
|
|
2019-07-17 13:33:42 +02:00
|
|
|
void LinuxTextToSpeechManager::createVoice(int typeNumber, Common::TTSVoice::Gender gender, Common::TTSVoice::Age age, char *description) {
|
2019-07-12 22:16:44 +02:00
|
|
|
SPDVoiceType *type = (SPDVoiceType *) malloc(sizeof(SPDVoiceType));
|
|
|
|
*type = static_cast<SPDVoiceType>(typeNumber);
|
2019-07-17 13:33:42 +02:00
|
|
|
_ttsState->_availaibleVoices.push_back(Common::TTSVoice(gender, age, (void *) type, description));
|
2019-07-12 22:16:44 +02:00
|
|
|
}
|
|
|
|
|
2019-07-11 00:14:28 +02:00
|
|
|
void LinuxTextToSpeechManager::updateVoices() {
|
2019-07-11 23:38:06 +02:00
|
|
|
if (_speechState == BROKEN)
|
|
|
|
return;
|
2019-07-11 00:14:28 +02:00
|
|
|
/* just use these voices:
|
|
|
|
SPD_MALE1, SPD_MALE2, SPD_MALE3,
|
|
|
|
SPD_FEMALE1, SPD_FEMALE2, SPD_FEMALE3,
|
|
|
|
SPD_CHILD_MALE, SPD_CHILD_FEMALE
|
|
|
|
|
|
|
|
it depends on the user to map them to the right voices in speech-dispatcher
|
|
|
|
configuration
|
|
|
|
*/
|
|
|
|
|
2019-07-12 22:16:44 +02:00
|
|
|
char **voiceInfo = spd_list_voices(_connection);
|
2019-07-11 00:14:28 +02:00
|
|
|
|
2019-07-17 13:33:42 +02:00
|
|
|
createVoice(SPD_MALE1, Common::TTSVoice::MALE, Common::TTSVoice::ADULT, voiceInfo[0]);
|
|
|
|
createVoice(SPD_MALE2, Common::TTSVoice::MALE, Common::TTSVoice::ADULT, voiceInfo[1]);
|
|
|
|
createVoice(SPD_MALE3, Common::TTSVoice::MALE, Common::TTSVoice::ADULT, voiceInfo[2]);
|
|
|
|
createVoice(SPD_FEMALE1, Common::TTSVoice::FEMALE, Common::TTSVoice::ADULT, voiceInfo[3]);
|
|
|
|
createVoice(SPD_FEMALE2, Common::TTSVoice::FEMALE, Common::TTSVoice::ADULT, voiceInfo[4]);
|
|
|
|
createVoice(SPD_FEMALE3, Common::TTSVoice::FEMALE, Common::TTSVoice::ADULT, voiceInfo[5]);
|
|
|
|
createVoice(SPD_CHILD_MALE, Common::TTSVoice::MALE, Common::TTSVoice::CHILD, voiceInfo[6]);
|
|
|
|
createVoice(SPD_CHILD_FEMALE, Common::TTSVoice::FEMALE, Common::TTSVoice::CHILD, voiceInfo[7]);
|
2019-07-11 00:14:28 +02:00
|
|
|
|
2019-07-10 10:49:29 +02:00
|
|
|
}
|
|
|
|
|
2019-07-16 22:09:05 -07:00
|
|
|
bool LinuxTextToSpeechManager::popState() {
|
|
|
|
if (_ttsState->_next == nullptr)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
for (Common::TTSVoice *i = _ttsState->_availaibleVoices.begin(); i < _ttsState->_availaibleVoices.end(); i++) {
|
|
|
|
free(i->getData());
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::TTSState *oldState = _ttsState;
|
|
|
|
_ttsState = _ttsState->_next;
|
|
|
|
|
|
|
|
delete oldState;
|
|
|
|
|
|
|
|
setLanguage(_ttsState->_language);
|
|
|
|
setPitch(_ttsState->_pitch);
|
|
|
|
setVolume(_ttsState->_volume);
|
|
|
|
setRate(_ttsState->_rate);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-10 10:49:29 +02:00
|
|
|
#endif
|