2009-06-26 20:40:51 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 by Dominik Wenger
|
|
|
|
*
|
|
|
|
* All files in this archive are subject to the GNU General Public License.
|
|
|
|
* See the file COPYING in the source tree root for full license agreement.
|
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "voicefile.h"
|
|
|
|
#include "utils.h"
|
2010-04-02 21:24:19 +00:00
|
|
|
#include "rockboxinfo.h"
|
2009-06-26 20:40:51 +00:00
|
|
|
#include "rbsettings.h"
|
2010-01-24 21:12:11 +00:00
|
|
|
#include "systeminfo.h"
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
VoiceFileCreator::VoiceFileCreator(QObject* parent) :QObject(parent)
|
|
|
|
{
|
|
|
|
m_wavtrimThreshold=500;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoiceFileCreator::abort()
|
|
|
|
{
|
|
|
|
m_abort = true;
|
|
|
|
emit aborted();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VoiceFileCreator::createVoiceFile()
|
|
|
|
{
|
|
|
|
m_talkList.clear();
|
|
|
|
m_abort = false;
|
|
|
|
emit logItem(tr("Starting Voicefile generation"),LOGINFO);
|
|
|
|
|
|
|
|
// test if tempdir exists
|
|
|
|
if(!QDir(QDir::tempPath()+"/rbvoice/").exists())
|
|
|
|
{
|
|
|
|
QDir(QDir::tempPath()).mkdir("rbvoice");
|
|
|
|
}
|
|
|
|
m_path = QDir::tempPath() + "/rbvoice/";
|
|
|
|
|
|
|
|
// read rockbox-info.txt
|
|
|
|
RockboxInfo info(m_mountpoint);
|
2009-08-10 19:20:53 +00:00
|
|
|
if(!info.success())
|
2009-06-26 20:40:51 +00:00
|
|
|
{
|
|
|
|
emit logItem(tr("could not find rockbox-info.txt"),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString target = info.target();
|
|
|
|
QString features = info.features();
|
|
|
|
QString version = info.version();
|
2009-08-22 14:05:20 +00:00
|
|
|
m_targetid = info.targetID().toInt();
|
2009-08-08 13:06:01 +00:00
|
|
|
version = version.left(version.indexOf("-")).remove("r");
|
|
|
|
|
2009-06-26 20:40:51 +00:00
|
|
|
//prepare download url
|
2011-04-02 08:59:00 +00:00
|
|
|
QString genlang = SystemInfo::value(SystemInfo::GenlangUrl).toString();
|
|
|
|
genlang.replace("%LANG%", m_lang);
|
|
|
|
genlang.replace("%TARGET%", target);
|
|
|
|
genlang.replace("%REVISION%", version);
|
|
|
|
genlang.replace("%FEATURES%", features);
|
|
|
|
QUrl genlangUrl(genlang);
|
2010-09-26 11:43:34 +00:00
|
|
|
qDebug() << "[VoiceFileCreator] downloading " << genlangUrl;
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
//download the correct genlang output
|
|
|
|
QTemporaryFile *downloadFile = new QTemporaryFile(this);
|
|
|
|
downloadFile->open();
|
|
|
|
filename = downloadFile->fileName();
|
|
|
|
downloadFile->close();
|
|
|
|
// get the real file.
|
|
|
|
getter = new HttpGet(this);
|
|
|
|
getter->setFile(downloadFile);
|
|
|
|
|
|
|
|
connect(getter, SIGNAL(done(bool)), this, SLOT(downloadDone(bool)));
|
|
|
|
connect(getter, SIGNAL(dataReadProgress(int, int)), this, SIGNAL(logProgress(int, int)));
|
|
|
|
connect(this, SIGNAL(aborted()), getter, SLOT(abort()));
|
2009-09-25 15:51:17 +00:00
|
|
|
emit logItem(tr("Downloading voice info..."),LOGINFO);
|
2009-06-26 20:40:51 +00:00
|
|
|
getter->getFile(genlangUrl);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void VoiceFileCreator::downloadDone(bool error)
|
|
|
|
{
|
2010-09-26 11:43:34 +00:00
|
|
|
qDebug() << "[VoiceFileCreator] download done, error:" << error;
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
// update progress bar
|
|
|
|
emit logProgress(1,1);
|
|
|
|
if(getter->httpResponse() != 200 && !getter->isCached()) {
|
|
|
|
emit logItem(tr("Download error: received HTTP error %1.").arg(getter->httpResponse()),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(getter->isCached())
|
2010-01-10 17:13:32 +00:00
|
|
|
emit logItem(tr("Cached file used."), LOGINFO);
|
2009-06-26 20:40:51 +00:00
|
|
|
if(error)
|
|
|
|
{
|
|
|
|
emit logItem(tr("Download error: %1").arg(getter->errorString()),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
2010-01-10 17:13:32 +00:00
|
|
|
emit logItem(tr("Download finished."),LOGINFO);
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
QCoreApplication::processEvents();
|
|
|
|
|
|
|
|
//open downloaded file
|
|
|
|
QFile genlang(filename);
|
|
|
|
if(!genlang.open(QIODevice::ReadOnly))
|
|
|
|
{
|
|
|
|
emit logItem(tr("failed to open downloaded file"),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QCoreApplication::processEvents();
|
|
|
|
|
|
|
|
//read in downloaded file
|
|
|
|
emit logItem(tr("Reading strings..."),LOGINFO);
|
|
|
|
QTextStream in(&genlang);
|
|
|
|
in.setCodec("UTF-8");
|
|
|
|
QString id, voice;
|
|
|
|
bool idfound = false;
|
|
|
|
bool voicefound=false;
|
2011-10-01 19:48:58 +00:00
|
|
|
bool useCorrection = RbSettings::value(RbSettings::UseTtsCorrections).toBool();
|
2009-06-26 20:40:51 +00:00
|
|
|
while (!in.atEnd())
|
|
|
|
{
|
|
|
|
QString line = in.readLine();
|
|
|
|
if(line.contains("id:")) //ID found
|
|
|
|
{
|
|
|
|
id = line.remove("id:").remove('"').trimmed();
|
|
|
|
idfound = true;
|
|
|
|
}
|
|
|
|
else if(line.contains("voice:")) // voice found
|
|
|
|
{
|
|
|
|
voice = line.remove("voice:").remove('"').trimmed();
|
|
|
|
voicefound=true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(idfound && voicefound)
|
|
|
|
{
|
|
|
|
TalkGenerator::TalkEntry entry;
|
|
|
|
entry.toSpeak = voice;
|
|
|
|
entry.wavfilename = m_path + "/" + id + ".wav";
|
2011-10-01 19:48:58 +00:00
|
|
|
//voicefont wants them with .mp3 extension
|
|
|
|
entry.talkfilename = m_path + "/" + id + ".mp3";
|
2009-06-26 20:40:51 +00:00
|
|
|
entry.voiced = false;
|
|
|
|
entry.encoded = false;
|
|
|
|
if(id == "VOICE_PAUSE")
|
|
|
|
{
|
|
|
|
QFile::copy(":/builtin/VOICE_PAUSE.wav",m_path + "/VOICE_PAUSE.wav");
|
|
|
|
entry.wavfilename = m_path + "/VOICE_PAUSE.wav";
|
|
|
|
entry.voiced = true;
|
2012-04-06 13:55:48 +00:00
|
|
|
m_talkList.append(entry);
|
|
|
|
}
|
|
|
|
else if(entry.toSpeak.isEmpty()) {
|
|
|
|
qDebug() << "[Voicefile] Empty voice string for ID" << id;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
m_talkList.append(entry);
|
2009-06-26 20:40:51 +00:00
|
|
|
}
|
|
|
|
idfound=false;
|
|
|
|
voicefound=false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
genlang.close();
|
|
|
|
|
|
|
|
// check for empty list
|
|
|
|
if(m_talkList.size() == 0)
|
|
|
|
{
|
|
|
|
emit logItem(tr("The downloaded file was empty!"),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate files
|
|
|
|
{
|
|
|
|
TalkGenerator generator(this);
|
2011-10-01 19:48:58 +00:00
|
|
|
// set language for string correction. If not set no correction will be made.
|
|
|
|
if(useCorrection)
|
|
|
|
generator.setLang(m_lang);
|
2009-06-26 20:40:51 +00:00
|
|
|
connect(&generator,SIGNAL(done(bool)),this,SIGNAL(done(bool)));
|
|
|
|
connect(&generator,SIGNAL(logItem(QString,int)),this,SIGNAL(logItem(QString,int)));
|
|
|
|
connect(&generator,SIGNAL(logProgress(int,int)),this,SIGNAL(logProgress(int,int)));
|
|
|
|
connect(this,SIGNAL(aborted()),&generator,SLOT(abort()));
|
|
|
|
|
2012-01-15 11:24:54 +00:00
|
|
|
if(generator.process(&m_talkList, m_wavtrimThreshold) == TalkGenerator::eERROR)
|
2009-06-26 20:40:51 +00:00
|
|
|
{
|
|
|
|
cleanup();
|
|
|
|
emit logProgress(0,1);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//make voicefile
|
|
|
|
emit logItem(tr("Creating voicefiles..."),LOGINFO);
|
|
|
|
FILE* ids2 = fopen(filename.toLocal8Bit(), "r");
|
|
|
|
if (ids2 == NULL)
|
|
|
|
{
|
|
|
|
cleanup();
|
|
|
|
emit logItem(tr("Error opening downloaded file"),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-04-01 22:06:52 +00:00
|
|
|
FILE* output = fopen(QString(m_mountpoint + "/.rockbox/langs/" + m_lang
|
|
|
|
+ ".voice").toLocal8Bit(), "wb");
|
2009-06-26 20:40:51 +00:00
|
|
|
if (output == NULL)
|
|
|
|
{
|
|
|
|
cleanup();
|
2010-04-01 22:06:52 +00:00
|
|
|
fclose(ids2);
|
2009-06-26 20:40:51 +00:00
|
|
|
emit logItem(tr("Error opening output file"),LOGERROR);
|
|
|
|
emit done(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
voicefont(ids2,m_targetid,m_path.toLocal8Bit().data(), output);
|
2010-04-01 22:06:52 +00:00
|
|
|
// ids2 and output are closed by voicefont().
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
//cleanup
|
|
|
|
cleanup();
|
|
|
|
|
|
|
|
// Add Voice file to the install log
|
|
|
|
QSettings installlog(m_mountpoint + "/.rockbox/rbutil.log", QSettings::IniFormat, 0);
|
|
|
|
installlog.beginGroup("selfcreated Voice");
|
2010-04-01 22:06:52 +00:00
|
|
|
installlog.setValue("/.rockbox/langs/" + m_lang + ".voice",
|
|
|
|
QDate::currentDate().toString("yyyyMMdd"));
|
2009-06-26 20:40:51 +00:00
|
|
|
installlog.endGroup();
|
|
|
|
installlog.sync();
|
|
|
|
|
|
|
|
emit logProgress(1,1);
|
|
|
|
emit logItem(tr("successfully created."),LOGOK);
|
|
|
|
|
|
|
|
emit done(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
//! \brief Cleans up Files potentially left in the temp dir
|
|
|
|
//!
|
|
|
|
void VoiceFileCreator::cleanup()
|
|
|
|
{
|
2009-09-25 15:51:17 +00:00
|
|
|
emit logItem(tr("Cleaning up..."),LOGINFO);
|
2009-06-26 20:40:51 +00:00
|
|
|
|
|
|
|
for(int i=0; i < m_talkList.size(); i++)
|
|
|
|
{
|
|
|
|
if(QFile::exists(m_talkList[i].wavfilename))
|
|
|
|
QFile::remove(m_talkList[i].wavfilename);
|
|
|
|
if(QFile::exists(m_talkList[i].talkfilename))
|
|
|
|
QFile::remove(m_talkList[i].talkfilename);
|
|
|
|
|
|
|
|
QCoreApplication::processEvents();
|
|
|
|
}
|
|
|
|
emit logItem(tr("Finished"),LOGINFO);
|
2012-01-08 11:50:04 +00:00
|
|
|
|
2009-06-26 20:40:51 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|