/////////////////////////////////////////////////////////////////////////////// // LameXP - Audio Encoder Front-End // Copyright (C) 2004-2011 LoRd_MuldeR // // 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. // // http://www.gnu.org/licenses/gpl-2.0.txt /////////////////////////////////////////////////////////////////////////////// #include "Dialog_CueImport.h" #include "Global.h" #include "Model_CueSheet.h" #include "Model_AudioFile.h" #include "Model_FileList.h" #include "Dialog_WorkingBanner.h" #include "Thread_FileAnalyzer.h" #include "Thread_CueSplitter.h" #include "LockedFile.h" #include #include #include #include #include #include #define SET_FONT_BOLD(WIDGET,BOLD) { QFont _font = WIDGET->font(); _font.setBold(BOLD); WIDGET->setFont(_font); } //////////////////////////////////////////////////////////// // Constructor & Destructor //////////////////////////////////////////////////////////// CueImportDialog::CueImportDialog(QWidget *parent, FileListModel *fileList, const QString &cueFile) : QDialog(parent), m_cueFileName(cueFile), m_fileList(fileList) { //Init the dialog, from the .ui file setupUi(this); //Fix size setMinimumSize(this->size()); setMaximumHeight(this->height()); //Create model m_model = new CueSheetModel(); connect(m_model, SIGNAL(modelReset()), this, SLOT(modelChanged())); //Setup table view treeView->setModel(m_model); treeView->header()->setStretchLastSection(false); treeView->header()->setResizeMode(QHeaderView::ResizeToContents); treeView->header()->setResizeMode(1, QHeaderView::Stretch); treeView->header()->setMovable(false); treeView->setItemsExpandable(false); //Enable up/down button connect(imprtButton, SIGNAL(clicked()), this, SLOT(importButtonClicked())); connect(browseButton, SIGNAL(clicked()), this, SLOT(browseButtonClicked())); //Translate labelHeaderText->setText(QString("%1
%2").arg(tr("Import Cue Sheet"), tr("The following Cue Sheet will be split and imported into LameXP."))); } CueImportDialog::~CueImportDialog(void) { LAMEXP_DELETE(m_model); } //////////////////////////////////////////////////////////// // EVENTS //////////////////////////////////////////////////////////// void CueImportDialog::showEvent(QShowEvent *event) { QDialog::showEvent(event); modelChanged(); } //////////////////////////////////////////////////////////// // Slots //////////////////////////////////////////////////////////// int CueImportDialog::exec(void) { WorkingBanner *progress = new WorkingBanner(dynamic_cast(parent())); progress->show(tr("Loading Cue Sheet file, please be patient...")); QFileInfo cueFileInfo(m_cueFileName); if(!cueFileInfo.exists() || !cueFileInfo.isFile()) { QString text = QString("%1
%2

%3").arg(tr("Failed to load the Cue Sheet file:"), QDir::toNativeSeparators(m_cueFileName), tr("The specified file could not be found!")).replace("-", "−"); QMessageBox::warning(progress, tr("Cue Sheet Error"), text); progress->close(); LAMEXP_DELETE(progress); return CueSheetModel::ErrorIOFailure; } m_outputDir = QString("%1/%2").arg(cueFileInfo.canonicalPath(), cueFileInfo.completeBaseName()); setWindowTitle(QString("%1: %2").arg(windowTitle().split(":", QString::SkipEmptyParts).first().trimmed(), cueFileInfo.fileName())); int iResult = m_model->loadCueSheet(m_cueFileName, QApplication::instance()); if(iResult != CueSheetModel::ErrorSuccess) { QString errorMsg = tr("An unknown error has occured!"); switch(iResult) { case CueSheetModel::ErrorIOFailure: errorMsg = tr("The file could not be opened for reading. Make sure you have the required rights!"); break; case CueSheetModel::ErrorBadFile: errorMsg = tr("The provided file does not look like a valid Cue Sheet disc image file!"); break; case CueSheetModel::ErrorUnsupported: errorMsg = QString("%1
%2").arg(tr("Could not find any supported audio track in the Cue Sheet image!"), tr("Note that LameXP can not handle \"binary\" Cue Sheet images.")); break; case CueSheetModel::ErrorInconsistent: errorMsg = tr("The selected Cue Sheet file contains inconsistent information. Take care!"); break; } QString text = QString("%1
%2

%3").arg(tr("Failed to load the Cue Sheet file:"), QDir::toNativeSeparators(m_cueFileName), errorMsg).replace("-", "−"); QMessageBox::warning(progress, tr("Cue Sheet Error"), text); progress->close(); LAMEXP_DELETE(progress); return iResult; } progress->close(); LAMEXP_DELETE(progress); return QDialog::exec(); } void CueImportDialog::modelChanged(void) { treeView->expandAll(); editOutputDir->setText(QDir::toNativeSeparators(m_outputDir)); } void CueImportDialog::browseButtonClicked(void) { QString newOutDir = QFileDialog::getExistingDirectory(this, tr("Choose Output Directory")); if(!newOutDir.isEmpty()) { m_outputDir = newOutDir; modelChanged(); } } void CueImportDialog::importButtonClicked(void) { static const __int64 oneGigabyte = 1073741824i64; static const __int64 minimumFreeDiskspaceMultiplier = 2i64; static const char *writeTestBuffer = "LAMEXP_WRITE_TEST"; QDir outputDir(m_outputDir); outputDir.mkpath("."); if(!(outputDir.exists() && outputDir.isReadable())) { QMessageBox::warning(this, tr("LameXP"), QString("%2").arg(tr("Error: The selected output directory could not be created!"))); return; } QFile writeTest(QString("%1/~%2.txt").arg(m_outputDir, lamexp_rand_str())); if(!(writeTest.open(QIODevice::ReadWrite) && (writeTest.write(writeTestBuffer) == strlen(writeTestBuffer)))) { QMessageBox::warning(this, tr("LameXP"), QString("%2").arg(tr("Error: The selected output directory is not writable!"))); return; } else { writeTest.close(); writeTest.remove(); } qint64 currentFreeDiskspace = lamexp_free_diskspace(m_outputDir); if(currentFreeDiskspace < (oneGigabyte * minimumFreeDiskspaceMultiplier)) { QMessageBox::warning(this, tr("Low Diskspace Warning"), QString("%1
%2").arg(tr("There are less than %1 GB of free diskspace available in the selected output directory.").arg(QString::number(minimumFreeDiskspaceMultiplier)), tr("It is highly recommend to free up more diskspace before proceeding with the import!"))); return; } importCueSheet(); accept(); } void CueImportDialog::analyzedFile(const AudioFileModel &file) { qDebug("Received result: <%s> <%s/%s>", file.filePath().toLatin1().constData(), file.formatContainerType().toLatin1().constData(), file.formatAudioType().toLatin1().constData()); m_fileInfo << file; } //////////////////////////////////////////////////////////// // Private Functions //////////////////////////////////////////////////////////// void CueImportDialog::importCueSheet(void) { QStringList files; //Fetch all files that are referenced in the Cue Sheet and lock them int nFiles = m_model->getFileCount(); for(int i = 0; i < nFiles; i++) { QString temp = m_model->getFileName(i); try { m_locks << new LockedFile(temp); } catch(char *err) { qWarning("Failed to lock file: %s", err); continue; } files << temp; } //Analyze all source files analyzeFiles(files); //Now split files according to Cue Sheet splitFiles(); //Release locks while(!m_locks.isEmpty()) { delete m_locks.takeFirst(); } } void CueImportDialog::analyzeFiles(QStringList &files) { m_fileInfo.clear(); WorkingBanner *progress = new WorkingBanner(this); FileAnalyzer *analyzer = new FileAnalyzer(files); connect(analyzer, SIGNAL(fileSelected(QString)), progress, SLOT(setText(QString)), Qt::QueuedConnection); connect(analyzer, SIGNAL(fileAnalyzed(AudioFileModel)), this, SLOT(analyzedFile(AudioFileModel)), Qt::QueuedConnection); progress->show(tr("Analyzing file(s), please wait..."), analyzer); progress->close(); LAMEXP_DELETE(progress); } void CueImportDialog::splitFiles(void) { int nTracksSkipped = 0; WorkingBanner *progress = new WorkingBanner(this); CueSplitter *splitter = new CueSplitter(m_outputDir, QFileInfo(m_cueFileName).completeBaseName().replace(".", " ").left(42).trimmed(), m_fileInfo); splitter->setAlbumInfo(m_model->getAlbumPerformer(), m_model->getAlbumTitle()); connect(splitter, SIGNAL(fileSelected(QString)), progress, SLOT(setText(QString)), Qt::QueuedConnection); connect(splitter, SIGNAL(fileSplit(AudioFileModel)), m_fileList, SLOT(addFile(AudioFileModel)), Qt::QueuedConnection); int nFiles = m_model->getFileCount(); for(int i = 0; i < nFiles; i++) { QString currentFileName = m_model->getFileName(i); int nTracks = m_model->getTrackCount(i); for(int j = 0; j < nTracks; j++) { int trackNo = m_model->getTrackNo(i, j); double startIndex = std::numeric_limits::quiet_NaN(); double duration = std::numeric_limits::quiet_NaN(); m_model->getTrackIndex(i, j, &startIndex, &duration); AudioFileModel metaInfo(QString().sprintf("cue://File%02d/Track%02d", i, j)); metaInfo.setFileName(m_model->getTrackTitle(i, j)); metaInfo.setFileArtist(m_model->getTrackPerformer(i, j)); metaInfo.setFilePosition(trackNo); try { splitter->addTrack(trackNo, currentFileName, startIndex, duration, metaInfo); } catch(char *err) { qWarning("Failed to add track #%02d: %s", trackNo, err); nTracksSkipped++; } } } progress->show(tr("Splitting file(s), please wait..."), splitter); progress->close(); if(!splitter->getSuccess()) { QMessageBox::warning(this, tr("Cue Sheet Error"), tr("An unexpected error has occured while splitting the Cue Sheet!")); } else { QString text = tr("Imported %1 track(s) from the Cue Sheet and skipped %2 track(s).").arg(QString::number(splitter->getTracksSuccess()), QString::number(splitter->getTracksSkipped() + nTracksSkipped)); QMessageBox::information(this, tr("Cue Sheet Completed"), text); } LAMEXP_DELETE(splitter); LAMEXP_DELETE(progress); }