/*************************************************************************** * Copyright (C) 2008-2010 by Ilya Kotov * * forkotov02@hotmail.ru * * * * 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., * * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * ***************************************************************************/ #include #include #include #include #include #include #include #include #ifdef WITH_ENCA #include #endif #include "cueparser.h" CUEParser::CUEParser(const QString &url) { QString fileName = url; if(url.contains("://")) { QString p = QUrl(url).path(); p.replace(QString(QUrl::toPercentEncoding("#")), "#"); p.replace(QString(QUrl::toPercentEncoding("?")), "?"); p.replace(QString(QUrl::toPercentEncoding("%")), "%"); p.replace(QString(QUrl::toPercentEncoding(":")), ":"); fileName = p; } QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) { qDebug("CUEParser: error: %s", qPrintable(file.errorString())); return; } QTextStream textStream (&file); QSettings settings(Qmmp::configFile(), QSettings::IniFormat); settings.beginGroup("CUE"); QTextCodec *codec = QTextCodec::codecForName(settings.value("encoding","ISO-8859-1").toByteArray ()); if(!codec) codec = QTextCodec::codecForName("UTF-8"); m_dirty = settings.value("dirty_cue", false).toBool(); #ifdef WITH_ENCA EncaAnalyser analyser = 0; if(settings.value("use_enca", false).toBool()) { analyser = enca_analyser_alloc(settings.value("enca_lang").toByteArray ().constData()); if(analyser) { EncaEncoding encoding = enca_analyse(analyser, (uchar *)file.readAll().constData(), file.size()); file.reset(); if(encoding.charset != ENCA_CS_UNKNOWN) { codec = QTextCodec::codecForName(enca_charset_name(encoding.charset,ENCA_NAME_STYLE_ENCA)); //qDebug("CUEParser: detected charset: %s", // enca_charset_name(encoding.charset,ENCA_NAME_STYLE_ENCA)); } } } #endif settings.endGroup(); //qDebug("CUEParser: using %s encoding", codec->name().constData()); textStream.setCodec(codec); QString album, genre, date, comment, artist, file_path; bool new_file = false; while (!textStream.atEnd()) { QString line = textStream.readLine().trimmed(); QStringList words = splitLine(line); if (words.size() < 2) continue; if (words[0] == "FILE") { if(!m_infoList.isEmpty()) { QList f_list = MetaDataManager::instance()->createPlayList(file_path, false); qint64 l = f_list.isEmpty() ? 0 : f_list.at(0)->length() * 1000; if (l > m_offsets.last()) m_infoList.last().setLength(l - m_offsets.last()); else m_infoList.last().setLength(0); } file_path = getDirtyPath(fileName, QFileInfo(fileName).dir().filePath(words[1])); new_file = true; } else if (words[0] == "PERFORMER") { if(m_infoList.isEmpty()) artist = words[1]; else m_infoList.last().setMetaData(Qmmp::ARTIST, words[1]); } else if (words[0] == "TITLE") { if(m_infoList.isEmpty()) album = words[1]; else m_infoList.last().setMetaData(Qmmp::TITLE, words[1]); } else if (words[0] == "TRACK") { QString path = fileName; path.replace("%", QString(QUrl::toPercentEncoding("%"))); //replace special symbols path.replace("#", QString(QUrl::toPercentEncoding("#"))); path.replace("?", QString(QUrl::toPercentEncoding("?"))); path.replace(":", QString(QUrl::toPercentEncoding(":"))); FileInfo info("cue://" + path + QString("#%1").arg(words[1].toInt())); info.setMetaData(Qmmp::TRACK, words[1].toInt()); info.setMetaData(Qmmp::ALBUM, album); info.setMetaData(Qmmp::GENRE, genre); info.setMetaData(Qmmp::YEAR, date); info.setMetaData(Qmmp::COMMENT, comment); info.setMetaData(Qmmp::ARTIST, artist); m_infoList << info; m_offsets << 0; m_files << file_path; } else if (words[0] == "INDEX" && words[1] == "01") { if (m_infoList.isEmpty()) continue; m_offsets.last() = getLength(words[2]); int c = m_infoList.count(); if(c > 1 && !new_file) m_infoList[c - 2].setLength(m_offsets[c - 1] - m_offsets[c - 2]); new_file = false; } else if (words[0] == "REM") { if (words.size() < 3) continue; if (words[1] == "GENRE") genre = words[2]; else if (words[1] == "DATE") date = words[2]; else if (words[1] == "COMMENT") comment = words[2]; } } file.close(); #ifdef WITH_ENCA if(analyser) enca_analyser_free(analyser); #endif if(m_infoList.isEmpty()) { qWarning("CUEParser: invalid cue file"); return; } //calculate last item length QList f_list = MetaDataManager::instance()->createPlayList(file_path, false); qint64 l = f_list.isEmpty() ? 0 : f_list.at(0)->length() * 1000; if (l > m_offsets.last()) m_infoList.last().setLength(l - m_offsets.last()); else m_infoList.last().setLength(0); } CUEParser::~CUEParser() { } QList CUEParser::createPlayList() { QList list; foreach(FileInfo info, m_infoList) { list << new FileInfo(info); list.last()->setLength(list.last()->length()/1000); } return list; } const QString CUEParser::filePath(int track) { return (track <= m_files.size()) ? m_files[track - 1] : QString(); } qint64 CUEParser::offset(int track) { return m_offsets.at(track - 1); } qint64 CUEParser::length(int track) { return m_infoList.at(track - 1).length(); } int CUEParser::count() { return m_infoList.count(); } FileInfo *CUEParser::info(int track) { return &m_infoList[track - 1]; } const QString CUEParser::trackURL(int track) { return m_infoList[track - 1].path(); } QStringList CUEParser::splitLine(const QString &line) { //qDebug("row string = %s",qPrintable(line)); QStringList list; QString buf = line.trimmed(); if (buf.isEmpty()) return list; while (!buf.isEmpty()) { //qDebug(qPrintable(buf)); if (buf.startsWith('"')) { int end = buf.indexOf('"',1); list << buf.mid (1, end - 1); buf.remove (0, end+1); } else { int end = buf.indexOf(' ', 0); if (end < 0) end = buf.size(); list << buf.mid (0, end); buf.remove (0, end); } buf = buf.trimmed(); } return list; } qint64 CUEParser::getLength(const QString &str) { QStringList list = str.split(":"); if (list.size() == 2) return (qint64)list.at(0).toInt()*60000 + list.at(1).toInt()*1000; else if (list.size() == 3) return (qint64)list.at(0).toInt()*60000 + list.at(1).toInt()*1000 + list.at(2).toInt()*1000/75; return 0; } QString CUEParser::getDirtyPath(const QString &cue, const QString &path) { if (Decoder::findByPath(path) || ! m_dirty) return path; QStringList candidates; QDirIterator it(QFileInfo(path).dir().path(), QDir::Files); while (it.hasNext()) { it.next(); QString f = it.filePath(); if ((f != cue) && Decoder::findByPath(f)) candidates.push_back(f); } if (candidates.empty()) return path; else if (candidates.count() == 1) return candidates.first(); int dot = cue.lastIndexOf('.'); if (dot != -1) { QRegExp r(QRegExp::escape(cue.left(dot)) + "\\.[^\\.]+$"); int index = candidates.indexOf(r); int rindex = candidates.lastIndexOf(r); if ((index != -1) && (index == rindex)) return candidates[index]; } dot = path.lastIndexOf('.'); if (dot != -1) { QRegExp r(QRegExp::escape(path.left(dot)) + "\\.[^\\.]+$"); int index = candidates.indexOf(r); int rindex = candidates.lastIndexOf(r); if ((index != -1) && (index == rindex)) return candidates[index]; } return path; }