summaryrefslogtreecommitdiffstats
path: root/src/translators/audiofileimporter.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/translators/audiofileimporter.cpp')
-rw-r--r--src/translators/audiofileimporter.cpp424
1 files changed, 424 insertions, 0 deletions
diff --git a/src/translators/audiofileimporter.cpp b/src/translators/audiofileimporter.cpp
new file mode 100644
index 0000000..f825964
--- /dev/null
+++ b/src/translators/audiofileimporter.cpp
@@ -0,0 +1,424 @@
+/***************************************************************************
+ copyright : (C) 2004-2007 by Robby Stephenson
+ email : robby@periapsis.org
+ ***************************************************************************/
+
+/***************************************************************************
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of version 2 of the GNU General Public License as *
+ * published by the Free Software Foundation; *
+ * *
+ ***************************************************************************/
+
+#include <config.h>
+
+#include "audiofileimporter.h"
+#include "../collections/musiccollection.h"
+#include "../entry.h"
+#include "../field.h"
+#include "../latin1literal.h"
+#include "../imagefactory.h"
+#include "../tellico_utils.h"
+#include "../tellico_kernel.h"
+#include "../progressmanager.h"
+#include "../tellico_debug.h"
+
+#ifdef HAVE_TAGLIB
+#include <taglib/fileref.h>
+#include <taglib/tag.h>
+#include <taglib/id3v2tag.h>
+#include <taglib/mpegfile.h>
+#include <taglib/vorbisfile.h>
+#include <taglib/flacfile.h>
+#include <taglib/audioproperties.h>
+#endif
+
+#include <klocale.h>
+#include <kapplication.h>
+
+#include <qlabel.h>
+#include <qlayout.h>
+#include <qvgroupbox.h>
+#include <qcheckbox.h>
+#include <qdir.h>
+#include <qwhatsthis.h>
+
+using Tellico::Import::AudioFileImporter;
+
+AudioFileImporter::AudioFileImporter(const KURL& url_) : Tellico::Import::Importer(url_)
+ , m_coll(0)
+ , m_widget(0)
+ , m_cancelled(false) {
+}
+
+bool AudioFileImporter::canImport(int type) const {
+ return type == Data::Collection::Album;
+}
+
+Tellico::Data::CollPtr AudioFileImporter::collection() {
+#ifndef HAVE_TAGLIB
+ return 0;
+#else
+
+ if(m_coll) {
+ return m_coll;
+ }
+
+ ProgressItem& item = ProgressManager::self()->newProgressItem(this, i18n("Scanning audio files..."), true);
+ item.setTotalSteps(100);
+ connect(&item, SIGNAL(signalCancelled(ProgressItem*)), SLOT(slotCancel()));
+ ProgressItem::Done done(this);
+
+ // TODO: allow remote audio file importing
+ QStringList dirs = url().path();
+ if(m_recursive->isChecked()) {
+ dirs += Tellico::findAllSubDirs(dirs[0]);
+ }
+
+ if(m_cancelled) {
+ return 0;
+ }
+
+ const bool showProgress = options() & ImportProgress;
+
+ QStringList files;
+ for(QStringList::ConstIterator it = dirs.begin(); !m_cancelled && it != dirs.end(); ++it) {
+ if((*it).isEmpty()) {
+ continue;
+ }
+
+ QDir dir(*it);
+ dir.setFilter(QDir::Files | QDir::Readable | QDir::Hidden); // hidden since I want directory files
+ const QStringList list = dir.entryList();
+ for(QStringList::ConstIterator it2 = list.begin(); it2 != list.end(); ++it2) {
+ files += dir.absFilePath(*it2);
+ }
+// kapp->processEvents(); not needed ?
+ }
+
+ if(m_cancelled) {
+ return 0;
+ }
+ item.setTotalSteps(files.count());
+
+ const QString title = QString::fromLatin1("title");
+ const QString artist = QString::fromLatin1("artist");
+ const QString year = QString::fromLatin1("year");
+ const QString genre = QString::fromLatin1("genre");
+ const QString track = QString::fromLatin1("track");
+ const QString comments = QString::fromLatin1("comments");
+ const QString file = QString::fromLatin1("file");
+
+ m_coll = new Data::MusicCollection(true);
+
+ const bool addFile = m_addFilePath->isChecked();
+ const bool addBitrate = m_addBitrate->isChecked();
+
+ Data::FieldPtr f;
+ if(addFile) {
+ f = m_coll->fieldByName(file);
+ if(!f) {
+ f = new Data::Field(file, i18n("Files"), Data::Field::Table);
+ m_coll->addField(f);
+ }
+ f->setProperty(QString::fromLatin1("column1"), i18n("Files"));
+ if(addBitrate) {
+ f->setProperty(QString::fromLatin1("columns"), QChar('2'));
+ f->setProperty(QString::fromLatin1("column2"), i18n("Bitrate"));
+ } else {
+ f->setProperty(QString::fromLatin1("columns"), QChar('1'));
+ }
+ }
+
+ QMap<QString, Data::EntryPtr> albumMap;
+
+ QStringList directoryFiles;
+ const uint stepSize = QMAX(static_cast<size_t>(1), files.count() / 100);
+
+ bool changeTrackTitle = true;
+ uint j = 0;
+ for(QStringList::ConstIterator it = files.begin(); !m_cancelled && it != files.end(); ++it, ++j) {
+ TagLib::FileRef f(QFile::encodeName(*it));
+ if(f.isNull() || !f.tag()) {
+ if((*it).endsWith(QString::fromLatin1("/.directory"))) {
+ directoryFiles += *it;
+ }
+ continue;
+ }
+
+ TagLib::Tag* tag = f.tag();
+ QString album = TStringToQString(tag->album()).stripWhiteSpace();
+ if(album.isEmpty()) {
+ // can't do anything since tellico entries are by album
+ kdWarning() << "Skipping: no album listed for " << *it << endl;
+ continue;
+ }
+ int disc = discNumber(f);
+ if(disc > 1 && !m_coll->hasField(QString::fromLatin1("track%1").arg(disc))) {
+ Data::FieldPtr f2 = new Data::Field(QString::fromLatin1("track%1").arg(disc),
+ i18n("Tracks (Disc %1)").arg(disc),
+ Data::Field::Table);
+ f2->setFormatFlag(Data::Field::FormatTitle);
+ f2->setProperty(QString::fromLatin1("columns"), QChar('3'));
+ f2->setProperty(QString::fromLatin1("column1"), i18n("Title"));
+ f2->setProperty(QString::fromLatin1("column2"), i18n("Artist"));
+ f2->setProperty(QString::fromLatin1("column3"), i18n("Length"));
+ m_coll->addField(f2);
+ if(changeTrackTitle) {
+ Data::FieldPtr newTrack = new Data::Field(*m_coll->fieldByName(track));
+ newTrack->setTitle(i18n("Tracks (Disc %1)").arg(1));
+ m_coll->modifyField(newTrack);
+ changeTrackTitle = false;
+ }
+ }
+ bool various = false;
+ bool exists = true;
+ Data::EntryPtr entry = 0;
+ if(!(entry = albumMap[album.lower()])) {
+ entry = new Data::Entry(m_coll);
+ albumMap.insert(album.lower(), entry);
+ exists = false;
+ }
+ // album entries use the album name as the title
+ entry->setField(title, album);
+ QString a = TStringToQString(tag->artist()).stripWhiteSpace();
+ if(!a.isEmpty()) {
+ if(exists && entry->field(artist).lower() != a.lower()) {
+ various = true;
+ entry->setField(artist, i18n("(Various)"));
+ } else {
+ entry->setField(artist, a);
+ }
+ }
+ if(tag->year() > 0) {
+ entry->setField(year, QString::number(tag->year()));
+ }
+ if(!tag->genre().isEmpty()) {
+ entry->setField(genre, TStringToQString(tag->genre()).stripWhiteSpace());
+ }
+
+ if(!tag->title().isEmpty()) {
+ int trackNum = tag->track();
+ if(trackNum <= 0) { // try to figure out track number from file name
+ QFileInfo f(*it);
+ QString fileName = f.baseName();
+ QString numString;
+ int i = 0;
+ const int len = fileName.length();
+ while(fileName[i].isNumber() && i < len) {
+ i++;
+ }
+ if(i == 0) { // does not start with a number
+ i = len - 1;
+ while(i >= 0 && fileName[i].isNumber()) {
+ i--;
+ }
+ // file name ends with a number
+ if(i != len - 1) {
+ numString = fileName.mid(i + 1);
+ }
+ } else {
+ numString = fileName.mid(0, i);
+ }
+ bool ok;
+ int number = numString.toInt(&ok);
+ if(ok) {
+ trackNum = number;
+ }
+ }
+ if(trackNum > 0) {
+ QString t = TStringToQString(tag->title()).stripWhiteSpace();
+ t += "::" + a;
+ const int len = f.audioProperties()->length();
+ if(len > 0) {
+ t += "::" + Tellico::minutes(len);
+ }
+ QString realTrack = disc > 1 ? track + QString::number(disc) : track;
+ entry->setField(realTrack, insertValue(entry->field(realTrack), t, trackNum));
+ if(addFile) {
+ QString fileValue = *it;
+ if(addBitrate) {
+ fileValue += "::" + QString::number(f.audioProperties()->bitrate());
+ }
+ entry->setField(file, insertValue(entry->field(file), fileValue, trackNum));
+ }
+ } else {
+ myDebug() << *it << " contains no track number and track number cannot be determined, so the track is not imported." << endl;
+ }
+ } else {
+ myDebug() << *it << " has an empty title, so the track is not imported." << endl;
+ }
+ if(!tag->comment().stripWhiteSpace().isEmpty()) {
+ QString c = entry->field(comments);
+ if(!c.isEmpty()) {
+ c += QString::fromLatin1("<br/>");
+ }
+ if(!tag->title().isEmpty()) {
+ c += QString::fromLatin1("<em>") + TStringToQString(tag->title()).stripWhiteSpace() + QString::fromLatin1("</em> - ");
+ }
+ c += TStringToQString(tag->comment().stripWhiteSpace()).stripWhiteSpace();
+ entry->setField(comments, c);
+ }
+
+ if(!exists) {
+ m_coll->addEntries(entry);
+ }
+ if(showProgress && j%stepSize == 0) {
+ ProgressManager::self()->setTotalSteps(this, files.count() + directoryFiles.count());
+ ProgressManager::self()->setProgress(this, j);
+ kapp->processEvents();
+ }
+
+/* kdDebug() << "-- TAG --" << endl;
+ kdDebug() << "title - \"" << tag->title().to8Bit() << "\"" << endl;
+ kdDebug() << "artist - \"" << tag->artist().to8Bit() << "\"" << endl;
+ kdDebug() << "album - \"" << tag->album().to8Bit() << "\"" << endl;
+ kdDebug() << "year - \"" << tag->year() << "\"" << endl;
+ kdDebug() << "comment - \"" << tag->comment().to8Bit() << "\"" << endl;
+ kdDebug() << "track - \"" << tag->track() << "\"" << endl;
+ kdDebug() << "genre - \"" << tag->genre().to8Bit() << "\"" << endl;*/
+ }
+
+ if(m_cancelled) {
+ m_coll = 0;
+ return 0;
+ }
+
+ QTextStream ts;
+ QRegExp iconRx(QString::fromLatin1("Icon\\s*=\\s*(.*)"));
+ for(QStringList::ConstIterator it = directoryFiles.begin(); !m_cancelled && it != directoryFiles.end(); ++it, ++j) {
+ QFile file(*it);
+ if(!file.open(IO_ReadOnly)) {
+ continue;
+ }
+ ts.unsetDevice();
+ ts.setDevice(&file);
+ for(QString line = ts.readLine(); !line.isNull(); line = ts.readLine()) {
+ if(!iconRx.exactMatch(line)) {
+ continue;
+ }
+ QDir thisDir(*it);
+ thisDir.cdUp();
+ QFileInfo fi(thisDir, iconRx.cap(1));
+ Data::EntryPtr entry = albumMap[thisDir.dirName()];
+ if(!entry) {
+ continue;
+ }
+ KURL u;
+ u.setPath(fi.absFilePath());
+ QString id = ImageFactory::addImage(u, true);
+ if(!id.isEmpty()) {
+ entry->setField(QString::fromLatin1("cover"), id);
+ }
+ break;
+ }
+
+ if(showProgress && j%stepSize == 0) {
+ ProgressManager::self()->setProgress(this, j);
+ kapp->processEvents();
+ }
+ }
+
+ if(m_cancelled) {
+ m_coll = 0;
+ return 0;
+ }
+
+ return m_coll;
+#endif
+}
+
+QWidget* AudioFileImporter::widget(QWidget* parent_, const char* name_) {
+ if(m_widget) {
+ return m_widget;
+ }
+
+ m_widget = new QWidget(parent_, name_);
+ QVBoxLayout* l = new QVBoxLayout(m_widget);
+
+ QVGroupBox* box = new QVGroupBox(i18n("Audio File Options"), m_widget);
+
+ m_recursive = new QCheckBox(i18n("Recursive &folder search"), box);
+ QWhatsThis::add(m_recursive, i18n("If checked, folders are recursively searched for audio files."));
+ // by default, make it checked
+ m_recursive->setChecked(true);
+
+ m_addFilePath = new QCheckBox(i18n("Include file &location"), box);
+ QWhatsThis::add(m_addFilePath, i18n("If checked, the file names for each track are added to the entries."));
+ m_addFilePath->setChecked(false);
+ connect(m_addFilePath, SIGNAL(toggled(bool)), SLOT(slotAddFileToggled(bool)));
+
+ m_addBitrate = new QCheckBox(i18n("Include &bitrate"), box);
+ QWhatsThis::add(m_addBitrate, i18n("If checked, the bitrate for each track is added to the entries."));
+ m_addBitrate->setChecked(false);
+ m_addBitrate->setEnabled(false);
+
+ l->addWidget(box);
+ l->addStretch(1);
+ return m_widget;
+}
+
+// pos_ is NOT zero-indexed!
+QString AudioFileImporter::insertValue(const QString& str_, const QString& value_, uint pos_) {
+ QStringList list = Data::Field::split(str_, true);
+ for(uint i = list.count(); i < pos_; ++i) {
+ list += QString::null;
+ }
+ if(!list[pos_-1].isNull()) {
+ myDebug() << "AudioFileImporter::insertValue() - overwriting track " << pos_ << endl;
+ myDebug() << "*** Old value: " << list[pos_-1] << endl;
+ myDebug() << "*** New value: " << value_ << endl;
+ }
+ list[pos_-1] = value_;
+ return list.join(QString::fromLatin1("; "));
+}
+
+void AudioFileImporter::slotCancel() {
+ m_cancelled = true;
+}
+
+void AudioFileImporter::slotAddFileToggled(bool on_) {
+ m_addBitrate->setEnabled(on_);
+ if(!on_) {
+ m_addBitrate->setChecked(false);
+ }
+}
+
+int AudioFileImporter::discNumber(const TagLib::FileRef& ref_) const {
+ // default to 1 unless otherwise
+ int num = 1;
+#ifdef HAVE_TAGLIB
+ QString disc;
+ if(TagLib::MPEG::File* file = dynamic_cast<TagLib::MPEG::File*>(ref_.file())) {
+ if(file->ID3v2Tag() && !file->ID3v2Tag()->frameListMap()["TPOS"].isEmpty()) {
+ disc = TStringToQString(file->ID3v2Tag()->frameListMap()["TPOS"].front()->toString()).stripWhiteSpace();
+ }
+ } else if(TagLib::Ogg::Vorbis::File* file = dynamic_cast<TagLib::Ogg::Vorbis::File*>(ref_.file())) {
+ if(file->tag() && !file->tag()->fieldListMap()["DISCNUMBER"].isEmpty()) {
+ disc = TStringToQString(file->tag()->fieldListMap()["DISCNUMBER"].front()).stripWhiteSpace();
+ }
+ } else if(TagLib::FLAC::File* file = dynamic_cast<TagLib::FLAC::File*>(ref_.file())) {
+ if(file->xiphComment() && !file->xiphComment()->fieldListMap()["DISCNUMBER"].isEmpty()) {
+ disc = TStringToQString(file->xiphComment()->fieldListMap()["DISCNUMBER"].front()).stripWhiteSpace();
+ }
+ }
+
+ if(!disc.isEmpty()) {
+ int pos = disc.find('/');
+ int n;
+ bool ok;
+ if(pos == -1) {
+ n = disc.toInt(&ok);
+ } else {
+ n = disc.left(pos).toInt(&ok);
+ }
+ if(ok && n > 0) {
+ num = n;
+ }
+ }
+#endif
+ return num;
+}
+
+#include "audiofileimporter.moc"