OSDN Git Service

Updated license info.
[lamexp/LameXP.git] / src / Dialog_MainWindow.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // LameXP - Audio Encoder Front-End
3 // Copyright (C) 2004-2013 LoRd_MuldeR <MuldeR2@GMX.de>
4 //
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; either version 2 of the License, or
8 // (at your option) any later version, but always including the *additional*
9 // restrictions defined in the "License.txt" file.
10 //
11 // This program is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 // GNU General Public License for more details.
15 //
16 // You should have received a copy of the GNU General Public License along
17 // with this program; if not, write to the Free Software Foundation, Inc.,
18 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 //
20 // http://www.gnu.org/licenses/gpl-2.0.txt
21 ///////////////////////////////////////////////////////////////////////////////
22
23 #include "Dialog_MainWindow.h"
24
25 //UIC includes
26 #include "../tmp/UIC_MainWindow.h"
27
28 //LameXP includes
29 #include "Global.h"
30 #include "Resource.h"
31 #include "Dialog_WorkingBanner.h"
32 #include "Dialog_MetaInfo.h"
33 #include "Dialog_About.h"
34 #include "Dialog_Update.h"
35 #include "Dialog_DropBox.h"
36 #include "Dialog_CueImport.h"
37 #include "Dialog_LogView.h"
38 #include "Thread_FileAnalyzer.h"
39 #include "Thread_MessageHandler.h"
40 #include "Model_MetaInfo.h"
41 #include "Model_Settings.h"
42 #include "Model_FileList.h"
43 #include "Model_FileSystem.h"
44 #include "WinSevenTaskbar.h"
45 #include "Registry_Encoder.h"
46 #include "Registry_Decoder.h"
47 #include "Encoder_Abstract.h"
48 #include "ShellIntegration.h"
49 #include "CustomEventFilter.h"
50
51 //Qt includes
52 #include <QMessageBox>
53 #include <QTimer>
54 #include <QDesktopWidget>
55 #include <QDate>
56 #include <QFileDialog>
57 #include <QInputDialog>
58 #include <QFileSystemModel>
59 #include <QDesktopServices>
60 #include <QUrl>
61 #include <QPlastiqueStyle>
62 #include <QCleanlooksStyle>
63 #include <QWindowsVistaStyle>
64 #include <QWindowsStyle>
65 #include <QSysInfo>
66 #include <QDragEnterEvent>
67 #include <QMimeData>
68 #include <QProcess>
69 #include <QUuid>
70 #include <QProcessEnvironment>
71 #include <QCryptographicHash>
72 #include <QTranslator>
73 #include <QResource>
74 #include <QScrollBar>
75
76 ////////////////////////////////////////////////////////////
77 // Helper macros
78 ////////////////////////////////////////////////////////////
79
80 #define ABORT_IF_BUSY do \
81 { \
82         if(m_banner->isVisible() || m_delayedFileTimer->isActive()) \
83         { \
84                 lamexp_beep(lamexp_beep_warning); \
85                 return; \
86         } \
87 } \
88 while(0)
89
90 #define SET_TEXT_COLOR(WIDGET, COLOR) do \
91 { \
92         QPalette _palette = WIDGET->palette(); \
93         _palette.setColor(QPalette::WindowText, (COLOR)); \
94         _palette.setColor(QPalette::Text, (COLOR)); \
95         WIDGET->setPalette(_palette); \
96 } \
97 while(0)
98
99 #define SET_FONT_BOLD(WIDGET,BOLD) do \
100 { \
101         QFont _font = WIDGET->font(); \
102         _font.setBold(BOLD); \
103         WIDGET->setFont(_font); \
104 } \
105 while(0)
106
107 #define TEMP_HIDE_DROPBOX(CMD) do \
108 { \
109         bool _dropBoxVisible = m_dropBox->isVisible(); \
110         if(_dropBoxVisible) m_dropBox->hide(); \
111         do { CMD } while(0); \
112         if(_dropBoxVisible) m_dropBox->show(); \
113 } \
114 while(0)
115
116 #define SET_MODEL(VIEW, MODEL) do \
117 { \
118         QItemSelectionModel *_tmp = (VIEW)->selectionModel(); \
119         (VIEW)->setModel(MODEL); \
120         LAMEXP_DELETE(_tmp); \
121 } \
122 while(0)
123
124 #define SET_CHECKBOX_STATE(CHCKBX, STATE) do \
125 { \
126         if((CHCKBX)->isChecked() != (STATE)) \
127         { \
128                 (CHCKBX)->click(); \
129         } \
130         if((CHCKBX)->isChecked() != (STATE)) \
131         { \
132                 qWarning("Warning: Failed to set checkbox " #CHCKBX " state!"); \
133         } \
134 } \
135 while(0)
136
137 #define TRIM_STRING_RIGHT(STR) do \
138 { \
139         while((STR.length() > 0) && STR[STR.length()-1].isSpace()) STR.chop(1); \
140 } \
141 while(0)
142
143 #define MAKE_TRANSPARENT(WIDGET, FLAG) do \
144 { \
145         QPalette _p = (WIDGET)->palette(); \
146         _p.setColor(QPalette::Background, Qt::transparent); \
147         (WIDGET)->setPalette(FLAG ? _p : QPalette()); \
148 } \
149 while(0)
150
151 #define WITH_BLOCKED_SIGNALS(WIDGET, CMD, ...) do \
152 { \
153         const bool _flag = (WIDGET)->blockSignals(true); \
154         (WIDGET)->CMD(__VA_ARGS__); \
155         if(!(_flag)) { (WIDGET)->blockSignals(false); } \
156 } \
157 while(0)
158
159 #define LINK(URL) QString("<a href=\"%1\">%2</a>").arg(URL).arg(QString(URL).replace("-", "&minus;"))
160 #define FSLINK(PATH) QString("<a href=\"file:///%1\">%2</a>").arg(PATH).arg(QString(PATH).replace("-", "&minus;"))
161 //#define USE_NATIVE_FILE_DIALOG (lamexp_themes_enabled() || ((QSysInfo::windowsVersion() & QSysInfo::WV_NT_based) < QSysInfo::WV_XP))
162 #define CENTER_CURRENT_OUTPUT_FOLDER_DELAYED QTimer::singleShot(125, this, SLOT(centerOutputFolderModel()))
163
164 static const unsigned int IDM_ABOUTBOX = 0xEFF0;
165
166 ////////////////////////////////////////////////////////////
167 // Constructor
168 ////////////////////////////////////////////////////////////
169
170 MainWindow::MainWindow(FileListModel *fileListModel, AudioFileModel_MetaInfo *metaInfo, SettingsModel *settingsModel, QWidget *parent)
171 :
172         QMainWindow(parent),
173         ui(new Ui::MainWindow),
174         m_fileListModel(fileListModel),
175         m_metaData(metaInfo),
176         m_settings(settingsModel),
177         m_fileSystemModel(NULL),
178         m_accepted(false),
179         m_firstTimeShown(true),
180         m_outputFolderViewCentering(false),
181         m_outputFolderViewInitCounter(0)
182 {
183         //Init the dialog, from the .ui file
184         ui->setupUi(this);
185         setWindowFlags(windowFlags() ^ Qt::WindowMaximizeButtonHint);
186         
187         //Register meta types
188         qRegisterMetaType<AudioFileModel>("AudioFileModel");
189
190         //Enabled main buttons
191         connect(ui->buttonAbout, SIGNAL(clicked()), this, SLOT(aboutButtonClicked()));
192         connect(ui->buttonStart, SIGNAL(clicked()), this, SLOT(encodeButtonClicked()));
193         connect(ui->buttonQuit, SIGNAL(clicked()), this, SLOT(closeButtonClicked()));
194
195         //Setup tab widget
196         ui->tabWidget->setCurrentIndex(0);
197         connect(ui->tabWidget, SIGNAL(currentChanged(int)), this, SLOT(tabPageChanged(int)));
198
199         //Add system menu
200         lamexp_append_sysmenu(this, IDM_ABOUTBOX, "About...");
201
202         //--------------------------------
203         // Setup "Source" tab
204         //--------------------------------
205
206         ui->sourceFileView->setModel(m_fileListModel);
207         ui->sourceFileView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
208         ui->sourceFileView->horizontalHeader()->setResizeMode(QHeaderView::ResizeToContents);
209         ui->sourceFileView->setContextMenuPolicy(Qt::CustomContextMenu);
210         ui->sourceFileView->viewport()->installEventFilter(this);
211         m_dropNoteLabel = new QLabel(ui->sourceFileView);
212         m_dropNoteLabel->setAlignment(Qt::AlignHCenter | Qt::AlignVCenter);
213         SET_FONT_BOLD(m_dropNoteLabel, true);
214         SET_TEXT_COLOR(m_dropNoteLabel, Qt::darkGray);
215         m_sourceFilesContextMenu = new QMenu();
216         m_showDetailsContextAction = m_sourceFilesContextMenu->addAction(QIcon(":/icons/zoom.png"), "N/A");
217         m_previewContextAction = m_sourceFilesContextMenu->addAction(QIcon(":/icons/sound.png"), "N/A");
218         m_findFileContextAction = m_sourceFilesContextMenu->addAction(QIcon(":/icons/folder_go.png"), "N/A");
219         m_sourceFilesContextMenu->addSeparator();
220         m_exportCsvContextAction = m_sourceFilesContextMenu->addAction(QIcon(":/icons/table_save.png"), "N/A");
221         m_importCsvContextAction = m_sourceFilesContextMenu->addAction(QIcon(":/icons/folder_table.png"), "N/A");
222         SET_FONT_BOLD(m_showDetailsContextAction, true);
223         connect(ui->buttonAddFiles, SIGNAL(clicked()), this, SLOT(addFilesButtonClicked()));
224         connect(ui->buttonRemoveFile, SIGNAL(clicked()), this, SLOT(removeFileButtonClicked()));
225         connect(ui->buttonClearFiles, SIGNAL(clicked()), this, SLOT(clearFilesButtonClicked()));
226         connect(ui->buttonFileUp, SIGNAL(clicked()), this, SLOT(fileUpButtonClicked()));
227         connect(ui->buttonFileDown, SIGNAL(clicked()), this, SLOT(fileDownButtonClicked()));
228         connect(ui->buttonShowDetails, SIGNAL(clicked()), this, SLOT(showDetailsButtonClicked()));
229         connect(m_fileListModel, SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(sourceModelChanged()));
230         connect(m_fileListModel, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(sourceModelChanged()));
231         connect(m_fileListModel, SIGNAL(modelReset()), this, SLOT(sourceModelChanged()));
232         connect(ui->sourceFileView, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(sourceFilesContextMenu(QPoint)));
233         connect(ui->sourceFileView->verticalScrollBar(), SIGNAL(sliderMoved(int)), this, SLOT(sourceFilesScrollbarMoved(int)));
234         connect(ui->sourceFileView->verticalScrollBar(), SIGNAL(valueChanged(int)), this, SLOT(sourceFilesScrollbarMoved(int)));
235         connect(m_showDetailsContextAction, SIGNAL(triggered(bool)), this, SLOT(showDetailsButtonClicked()));
236         connect(m_previewContextAction, SIGNAL(triggered(bool)), this, SLOT(previewContextActionTriggered()));
237         connect(m_findFileContextAction, SIGNAL(triggered(bool)), this, SLOT(findFileContextActionTriggered()));
238         connect(m_exportCsvContextAction, SIGNAL(triggered(bool)), this, SLOT(exportCsvContextActionTriggered()));
239         connect(m_importCsvContextAction, SIGNAL(triggered(bool)), this, SLOT(importCsvContextActionTriggered()));
240
241         //--------------------------------
242         // Setup "Output" tab
243         //--------------------------------
244
245         ui->outputFolderView->setHeaderHidden(true);
246         ui->outputFolderView->setAnimated(false);
247         ui->outputFolderView->setMouseTracking(false);
248         ui->outputFolderView->setContextMenuPolicy(Qt::CustomContextMenu);
249         ui->outputFolderView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
250
251         m_evenFilterOutputFolderMouse = new CustomEventFilter;
252         ui->outputFoldersGoUpLabel->installEventFilter(m_evenFilterOutputFolderMouse);
253         ui->outputFoldersEditorLabel->installEventFilter(m_evenFilterOutputFolderMouse);
254         ui->outputFoldersFovoritesLabel->installEventFilter(m_evenFilterOutputFolderMouse);
255         ui->outputFolderLabel->installEventFilter(m_evenFilterOutputFolderMouse);
256
257         m_evenFilterOutputFolderView = new CustomEventFilter;
258         ui->outputFolderView->installEventFilter(m_evenFilterOutputFolderView);
259
260         SET_CHECKBOX_STATE(ui->saveToSourceFolderCheckBox, m_settings->outputToSourceDir());
261         ui->prependRelativePathCheckBox->setChecked(m_settings->prependRelativeSourcePath());
262         
263         connect(ui->outputFolderView, SIGNAL(clicked(QModelIndex)), this, SLOT(outputFolderViewClicked(QModelIndex)));
264         connect(ui->outputFolderView, SIGNAL(activated(QModelIndex)), this, SLOT(outputFolderViewClicked(QModelIndex)));
265         connect(ui->outputFolderView, SIGNAL(pressed(QModelIndex)), this, SLOT(outputFolderViewClicked(QModelIndex)));
266         connect(ui->outputFolderView, SIGNAL(entered(QModelIndex)), this, SLOT(outputFolderViewMoved(QModelIndex)));
267         connect(ui->outputFolderView, SIGNAL(expanded(QModelIndex)), this, SLOT(outputFolderItemExpanded(QModelIndex)));
268         connect(ui->buttonMakeFolder, SIGNAL(clicked()), this, SLOT(makeFolderButtonClicked()));
269         connect(ui->buttonGotoHome, SIGNAL(clicked()), SLOT(gotoHomeFolderButtonClicked()));
270         connect(ui->buttonGotoDesktop, SIGNAL(clicked()), this, SLOT(gotoDesktopButtonClicked()));
271         connect(ui->buttonGotoMusic, SIGNAL(clicked()), this, SLOT(gotoMusicFolderButtonClicked()));
272         connect(ui->saveToSourceFolderCheckBox, SIGNAL(clicked()), this, SLOT(saveToSourceFolderChanged()));
273         connect(ui->prependRelativePathCheckBox, SIGNAL(clicked()), this, SLOT(prependRelativePathChanged()));
274         connect(ui->outputFolderEdit, SIGNAL(editingFinished()), this, SLOT(outputFolderEditFinished()));
275         connect(m_evenFilterOutputFolderMouse, SIGNAL(eventOccurred(QWidget*, QEvent*)), this, SLOT(outputFolderMouseEventOccurred(QWidget*, QEvent*)));
276         connect(m_evenFilterOutputFolderView, SIGNAL(eventOccurred(QWidget*, QEvent*)), this, SLOT(outputFolderViewEventOccurred(QWidget*, QEvent*)));
277
278         if(m_outputFolderContextMenu = new QMenu())
279         {
280                 m_showFolderContextAction = m_outputFolderContextMenu->addAction(QIcon(":/icons/zoom.png"), "N/A");
281                 m_goUpFolderContextAction = m_outputFolderContextMenu->addAction(QIcon(":/icons/folder_up.png"), "N/A");
282                 m_outputFolderContextMenu->addSeparator();
283                 m_refreshFolderContextAction = m_outputFolderContextMenu->addAction(QIcon(":/icons/arrow_refresh.png"), "N/A");
284                 m_outputFolderContextMenu->setDefaultAction(m_showFolderContextAction);
285                 connect(ui->outputFolderView, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(outputFolderContextMenu(QPoint)));
286                 connect(m_showFolderContextAction, SIGNAL(triggered(bool)), this, SLOT(showFolderContextActionTriggered()));
287                 connect(m_refreshFolderContextAction, SIGNAL(triggered(bool)), this, SLOT(refreshFolderContextActionTriggered()));
288                 connect(m_goUpFolderContextAction, SIGNAL(triggered(bool)), this, SLOT(goUpFolderContextActionTriggered()));
289         }
290
291         if(m_outputFolderFavoritesMenu = new QMenu())
292         {
293                 m_addFavoriteFolderAction = m_outputFolderFavoritesMenu->addAction(QIcon(":/icons/add.png"), "N/A");
294                 m_outputFolderFavoritesMenu->insertSeparator(m_addFavoriteFolderAction);
295                 connect(m_addFavoriteFolderAction, SIGNAL(triggered(bool)), this, SLOT(addFavoriteFolderActionTriggered()));
296         }
297
298         ui->outputFolderEdit->setVisible(false);
299         if(m_outputFolderNoteBox = new QLabel(ui->outputFolderView))
300         {
301                 m_outputFolderNoteBox->setAutoFillBackground(true);
302                 m_outputFolderNoteBox->setAlignment(Qt::AlignHCenter | Qt::AlignVCenter);
303                 m_outputFolderNoteBox->setFrameShape(QFrame::StyledPanel);
304                 SET_FONT_BOLD(m_outputFolderNoteBox, true);
305                 m_outputFolderNoteBox->hide();
306
307         }
308
309         outputFolderViewClicked(QModelIndex());
310         refreshFavorites();
311         
312         //--------------------------------
313         // Setup "Meta Data" tab
314         //--------------------------------
315
316         m_metaInfoModel = new MetaInfoModel(m_metaData);
317         m_metaInfoModel->clearData();
318         m_metaInfoModel->setData(m_metaInfoModel->index(4, 1), m_settings->metaInfoPosition());
319         ui->metaDataView->setModel(m_metaInfoModel);
320         ui->metaDataView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
321         ui->metaDataView->verticalHeader()->hide();
322         ui->metaDataView->horizontalHeader()->setResizeMode(QHeaderView::ResizeToContents);
323         SET_CHECKBOX_STATE(ui->writeMetaDataCheckBox, m_settings->writeMetaTags());
324         ui->generatePlaylistCheckBox->setChecked(m_settings->createPlaylist());
325         connect(ui->buttonEditMeta, SIGNAL(clicked()), this, SLOT(editMetaButtonClicked()));
326         connect(ui->buttonClearMeta, SIGNAL(clicked()), this, SLOT(clearMetaButtonClicked()));
327         connect(ui->writeMetaDataCheckBox, SIGNAL(clicked()), this, SLOT(metaTagsEnabledChanged()));
328         connect(ui->generatePlaylistCheckBox, SIGNAL(clicked()), this, SLOT(playlistEnabledChanged()));
329
330         //--------------------------------
331         //Setup "Compression" tab
332         //--------------------------------
333
334         m_encoderButtonGroup = new QButtonGroup(this);
335         m_encoderButtonGroup->addButton(ui->radioButtonEncoderMP3, SettingsModel::MP3Encoder);
336         m_encoderButtonGroup->addButton(ui->radioButtonEncoderVorbis, SettingsModel::VorbisEncoder);
337         m_encoderButtonGroup->addButton(ui->radioButtonEncoderAAC, SettingsModel::AACEncoder);
338         m_encoderButtonGroup->addButton(ui->radioButtonEncoderAC3, SettingsModel::AC3Encoder);
339         m_encoderButtonGroup->addButton(ui->radioButtonEncoderFLAC, SettingsModel::FLACEncoder);
340         m_encoderButtonGroup->addButton(ui->radioButtonEncoderOpus, SettingsModel::OpusEncoder);
341         m_encoderButtonGroup->addButton(ui->radioButtonEncoderDCA, SettingsModel::DCAEncoder);
342         m_encoderButtonGroup->addButton(ui->radioButtonEncoderPCM, SettingsModel::PCMEncoder);
343
344         const int aacEncoder = EncoderRegistry::getAacEncoder();
345         ui->radioButtonEncoderAAC->setEnabled(aacEncoder > SettingsModel::AAC_ENCODER_NONE);
346
347         m_modeButtonGroup = new QButtonGroup(this);
348         m_modeButtonGroup->addButton(ui->radioButtonModeQuality, SettingsModel::VBRMode);
349         m_modeButtonGroup->addButton(ui->radioButtonModeAverageBitrate, SettingsModel::ABRMode);
350         m_modeButtonGroup->addButton(ui->radioButtonConstBitrate, SettingsModel::CBRMode);
351
352         ui->radioButtonEncoderMP3->setChecked(true);
353         foreach(QAbstractButton *currentButton, m_encoderButtonGroup->buttons())
354         {
355                 if(currentButton->isEnabled() && (m_encoderButtonGroup->id(currentButton) == m_settings->compressionEncoder()))
356                 {
357                         currentButton->setChecked(true);
358                         break;
359                 }
360         }
361
362         m_evenFilterCompressionTab = new CustomEventFilter();
363         ui->labelCompressionHelp->installEventFilter(m_evenFilterCompressionTab);
364         ui->labelResetEncoders ->installEventFilter(m_evenFilterCompressionTab);
365
366         connect(m_encoderButtonGroup, SIGNAL(buttonClicked(int)), this, SLOT(updateEncoder(int)));
367         connect(m_modeButtonGroup, SIGNAL(buttonClicked(int)), this, SLOT(updateRCMode(int)));
368         connect(m_evenFilterCompressionTab, SIGNAL(eventOccurred(QWidget*, QEvent*)), this, SLOT(compressionTabEventOccurred(QWidget*, QEvent*)));
369         connect(ui->sliderBitrate, SIGNAL(valueChanged(int)), this, SLOT(updateBitrate(int)));
370
371         updateEncoder(m_encoderButtonGroup->checkedId());
372
373         //--------------------------------
374         //Setup "Advanced Options" tab
375         //--------------------------------
376
377         ui->sliderLameAlgoQuality->setValue(m_settings->lameAlgoQuality());
378         if(m_settings->maximumInstances() > 0) ui->sliderMaxInstances->setValue(m_settings->maximumInstances());
379
380         ui->spinBoxBitrateManagementMin->setValue(m_settings->bitrateManagementMinRate());
381         ui->spinBoxBitrateManagementMax->setValue(m_settings->bitrateManagementMaxRate());
382         ui->spinBoxNormalizationFilter->setValue(static_cast<double>(m_settings->normalizationFilterMaxVolume()) / 100.0);
383         ui->spinBoxToneAdjustBass->setValue(static_cast<double>(m_settings->toneAdjustBass()) / 100.0);
384         ui->spinBoxToneAdjustTreble->setValue(static_cast<double>(m_settings->toneAdjustTreble()) / 100.0);
385         ui->spinBoxAftenSearchSize->setValue(m_settings->aftenExponentSearchSize());
386         ui->spinBoxOpusComplexity->setValue(m_settings->opusComplexity());
387         
388         ui->comboBoxMP3ChannelMode->setCurrentIndex(m_settings->lameChannelMode());
389         ui->comboBoxSamplingRate->setCurrentIndex(m_settings->samplingRate());
390         ui->comboBoxAACProfile->setCurrentIndex(m_settings->aacEncProfile());
391         ui->comboBoxAftenCodingMode->setCurrentIndex(m_settings->aftenAudioCodingMode());
392         ui->comboBoxAftenDRCMode->setCurrentIndex(m_settings->aftenDynamicRangeCompression());
393         ui->comboBoxNormalizationMode->setCurrentIndex(m_settings->normalizationFilterEQMode());
394         //comboBoxOpusOptimize->setCurrentIndex(m_settings->opusOptimizeFor());
395         ui->comboBoxOpusFramesize->setCurrentIndex(m_settings->opusFramesize());
396         
397         SET_CHECKBOX_STATE(ui->checkBoxBitrateManagement, m_settings->bitrateManagementEnabled());
398         SET_CHECKBOX_STATE(ui->checkBoxNeroAAC2PassMode, m_settings->neroAACEnable2Pass());
399         SET_CHECKBOX_STATE(ui->checkBoxAftenFastAllocation, m_settings->aftenFastBitAllocation());
400         SET_CHECKBOX_STATE(ui->checkBoxNormalizationFilter, m_settings->normalizationFilterEnabled());
401         SET_CHECKBOX_STATE(ui->checkBoxAutoDetectInstances, (m_settings->maximumInstances() < 1));
402         SET_CHECKBOX_STATE(ui->checkBoxUseSystemTempFolder, !m_settings->customTempPathEnabled());
403         SET_CHECKBOX_STATE(ui->checkBoxRenameOutput, m_settings->renameOutputFilesEnabled());
404         SET_CHECKBOX_STATE(ui->checkBoxForceStereoDownmix, m_settings->forceStereoDownmix());
405         SET_CHECKBOX_STATE(ui->checkBoxOpusDisableResample, m_settings->opusDisableResample());
406         ui->checkBoxNeroAAC2PassMode->setEnabled(aacEncoder == SettingsModel::AAC_ENCODER_NERO);
407         
408         ui->lineEditCustomParamLAME   ->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::MP3Encoder));
409         ui->lineEditCustomParamOggEnc ->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::VorbisEncoder));
410         ui->lineEditCustomParamNeroAAC->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::AACEncoder));
411         ui->lineEditCustomParamFLAC   ->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::FLACEncoder));
412         ui->lineEditCustomParamAften  ->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::AC3Encoder));
413         ui->lineEditCustomParamOpus   ->setText(EncoderRegistry::loadEncoderCustomParams(m_settings, SettingsModel::OpusEncoder));
414         ui->lineEditCustomTempFolder  ->setText(QDir::toNativeSeparators(m_settings->customTempPath()));
415         ui->lineEditRenamePattern     ->setText(m_settings->renameOutputFilesPattern());
416         
417         m_evenFilterCustumParamsHelp = new CustomEventFilter();
418         ui->helpCustomParamLAME->installEventFilter(m_evenFilterCustumParamsHelp);
419         ui->helpCustomParamOggEnc->installEventFilter(m_evenFilterCustumParamsHelp);
420         ui->helpCustomParamNeroAAC->installEventFilter(m_evenFilterCustumParamsHelp);
421         ui->helpCustomParamFLAC->installEventFilter(m_evenFilterCustumParamsHelp);
422         ui->helpCustomParamAften->installEventFilter(m_evenFilterCustumParamsHelp);
423         ui->helpCustomParamOpus->installEventFilter(m_evenFilterCustumParamsHelp);
424         
425         m_overwriteButtonGroup = new QButtonGroup(this);
426         m_overwriteButtonGroup->addButton(ui->radioButtonOverwriteModeKeepBoth, SettingsModel::Overwrite_KeepBoth);
427         m_overwriteButtonGroup->addButton(ui->radioButtonOverwriteModeSkipFile, SettingsModel::Overwrite_SkipFile);
428         m_overwriteButtonGroup->addButton(ui->radioButtonOverwriteModeReplaces, SettingsModel::Overwrite_Replaces);
429
430         ui->radioButtonOverwriteModeKeepBoth->setChecked(m_settings->overwriteMode() == SettingsModel::Overwrite_KeepBoth);
431         ui->radioButtonOverwriteModeSkipFile->setChecked(m_settings->overwriteMode() == SettingsModel::Overwrite_SkipFile);
432         ui->radioButtonOverwriteModeReplaces->setChecked(m_settings->overwriteMode() == SettingsModel::Overwrite_Replaces);
433
434         connect(ui->sliderLameAlgoQuality, SIGNAL(valueChanged(int)), this, SLOT(updateLameAlgoQuality(int)));
435         connect(ui->checkBoxBitrateManagement, SIGNAL(clicked(bool)), this, SLOT(bitrateManagementEnabledChanged(bool)));
436         connect(ui->spinBoxBitrateManagementMin, SIGNAL(valueChanged(int)), this, SLOT(bitrateManagementMinChanged(int)));
437         connect(ui->spinBoxBitrateManagementMax, SIGNAL(valueChanged(int)), this, SLOT(bitrateManagementMaxChanged(int)));
438         connect(ui->comboBoxMP3ChannelMode, SIGNAL(currentIndexChanged(int)), this, SLOT(channelModeChanged(int)));
439         connect(ui->comboBoxSamplingRate, SIGNAL(currentIndexChanged(int)), this, SLOT(samplingRateChanged(int)));
440         connect(ui->checkBoxNeroAAC2PassMode, SIGNAL(clicked(bool)), this, SLOT(neroAAC2PassChanged(bool)));
441         connect(ui->comboBoxAACProfile, SIGNAL(currentIndexChanged(int)), this, SLOT(neroAACProfileChanged(int)));
442         connect(ui->checkBoxNormalizationFilter, SIGNAL(clicked(bool)), this, SLOT(normalizationEnabledChanged(bool)));
443         connect(ui->comboBoxAftenCodingMode, SIGNAL(currentIndexChanged(int)), this, SLOT(aftenCodingModeChanged(int)));
444         connect(ui->comboBoxAftenDRCMode, SIGNAL(currentIndexChanged(int)), this, SLOT(aftenDRCModeChanged(int)));
445         connect(ui->spinBoxAftenSearchSize, SIGNAL(valueChanged(int)), this, SLOT(aftenSearchSizeChanged(int)));
446         connect(ui->checkBoxAftenFastAllocation, SIGNAL(clicked(bool)), this, SLOT(aftenFastAllocationChanged(bool)));
447         connect(ui->spinBoxNormalizationFilter, SIGNAL(valueChanged(double)), this, SLOT(normalizationMaxVolumeChanged(double)));
448         connect(ui->comboBoxNormalizationMode, SIGNAL(currentIndexChanged(int)), this, SLOT(normalizationModeChanged(int)));
449         connect(ui->spinBoxToneAdjustBass, SIGNAL(valueChanged(double)), this, SLOT(toneAdjustBassChanged(double)));
450         connect(ui->spinBoxToneAdjustTreble, SIGNAL(valueChanged(double)), this, SLOT(toneAdjustTrebleChanged(double)));
451         connect(ui->buttonToneAdjustReset, SIGNAL(clicked()), this, SLOT(toneAdjustTrebleReset()));
452         connect(ui->lineEditCustomParamLAME, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
453         connect(ui->lineEditCustomParamOggEnc, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
454         connect(ui->lineEditCustomParamNeroAAC, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
455         connect(ui->lineEditCustomParamFLAC, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
456         connect(ui->lineEditCustomParamAften, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
457         connect(ui->lineEditCustomParamOpus, SIGNAL(editingFinished()), this, SLOT(customParamsChanged()));
458         connect(ui->sliderMaxInstances, SIGNAL(valueChanged(int)), this, SLOT(updateMaximumInstances(int)));
459         connect(ui->checkBoxAutoDetectInstances, SIGNAL(clicked(bool)), this, SLOT(autoDetectInstancesChanged(bool)));
460         connect(ui->buttonBrowseCustomTempFolder, SIGNAL(clicked()), this, SLOT(browseCustomTempFolderButtonClicked()));
461         connect(ui->lineEditCustomTempFolder, SIGNAL(textChanged(QString)), this, SLOT(customTempFolderChanged(QString)));
462         connect(ui->checkBoxUseSystemTempFolder, SIGNAL(clicked(bool)), this, SLOT(useCustomTempFolderChanged(bool)));
463         connect(ui->buttonResetAdvancedOptions, SIGNAL(clicked()), this, SLOT(resetAdvancedOptionsButtonClicked()));
464         connect(ui->checkBoxRenameOutput, SIGNAL(clicked(bool)), this, SLOT(renameOutputEnabledChanged(bool)));
465         connect(ui->lineEditRenamePattern, SIGNAL(editingFinished()), this, SLOT(renameOutputPatternChanged()));
466         connect(ui->lineEditRenamePattern, SIGNAL(textChanged(QString)), this, SLOT(renameOutputPatternChanged(QString)));
467         connect(ui->labelShowRenameMacros, SIGNAL(linkActivated(QString)), this, SLOT(showRenameMacros(QString)));
468         connect(ui->checkBoxForceStereoDownmix, SIGNAL(clicked(bool)), this, SLOT(forceStereoDownmixEnabledChanged(bool)));
469         connect(ui->comboBoxOpusFramesize, SIGNAL(currentIndexChanged(int)), this, SLOT(opusSettingsChanged()));
470         connect(ui->spinBoxOpusComplexity, SIGNAL(valueChanged(int)), this, SLOT(opusSettingsChanged()));
471         connect(ui->checkBoxOpusDisableResample, SIGNAL(clicked(bool)), SLOT(opusSettingsChanged()));
472         connect(m_overwriteButtonGroup, SIGNAL(buttonClicked(int)), this, SLOT(overwriteModeChanged(int)));
473         connect(m_evenFilterCustumParamsHelp, SIGNAL(eventOccurred(QWidget*, QEvent*)), this, SLOT(customParamsHelpRequested(QWidget*, QEvent*)));
474
475         //--------------------------------
476         // Force initial GUI update
477         //--------------------------------
478
479         updateLameAlgoQuality(ui->sliderLameAlgoQuality->value());
480         updateMaximumInstances(ui->sliderMaxInstances->value());
481         toneAdjustTrebleChanged(ui->spinBoxToneAdjustTreble->value());
482         toneAdjustBassChanged(ui->spinBoxToneAdjustBass->value());
483         customParamsChanged();
484         
485         //--------------------------------
486         // Initialize actions
487         //--------------------------------
488
489         //Activate file menu actions
490         ui->actionOpenFolder->setData(QVariant::fromValue<bool>(false));
491         ui->actionOpenFolderRecursively->setData(QVariant::fromValue<bool>(true));
492         connect(ui->actionOpenFolder, SIGNAL(triggered()), this, SLOT(openFolderActionActivated()));
493         connect(ui->actionOpenFolderRecursively, SIGNAL(triggered()), this, SLOT(openFolderActionActivated()));
494
495         //Activate view menu actions
496         m_tabActionGroup = new QActionGroup(this);
497         m_tabActionGroup->addAction(ui->actionSourceFiles);
498         m_tabActionGroup->addAction(ui->actionOutputDirectory);
499         m_tabActionGroup->addAction(ui->actionCompression);
500         m_tabActionGroup->addAction(ui->actionMetaData);
501         m_tabActionGroup->addAction(ui->actionAdvancedOptions);
502         ui->actionSourceFiles->setData(0);
503         ui->actionOutputDirectory->setData(1);
504         ui->actionMetaData->setData(2);
505         ui->actionCompression->setData(3);
506         ui->actionAdvancedOptions->setData(4);
507         ui->actionSourceFiles->setChecked(true);
508         connect(m_tabActionGroup, SIGNAL(triggered(QAction*)), this, SLOT(tabActionActivated(QAction*)));
509
510         //Activate style menu actions
511         m_styleActionGroup = new QActionGroup(this);
512         m_styleActionGroup->addAction(ui->actionStylePlastique);
513         m_styleActionGroup->addAction(ui->actionStyleCleanlooks);
514         m_styleActionGroup->addAction(ui->actionStyleWindowsVista);
515         m_styleActionGroup->addAction(ui->actionStyleWindowsXP);
516         m_styleActionGroup->addAction(ui->actionStyleWindowsClassic);
517         ui->actionStylePlastique->setData(0);
518         ui->actionStyleCleanlooks->setData(1);
519         ui->actionStyleWindowsVista->setData(2);
520         ui->actionStyleWindowsXP->setData(3);
521         ui->actionStyleWindowsClassic->setData(4);
522         ui->actionStylePlastique->setChecked(true);
523         ui->actionStyleWindowsXP->setEnabled((QSysInfo::windowsVersion() & QSysInfo::WV_NT_based) >= QSysInfo::WV_XP && lamexp_themes_enabled());
524         ui->actionStyleWindowsVista->setEnabled((QSysInfo::windowsVersion() & QSysInfo::WV_NT_based) >= QSysInfo::WV_VISTA && lamexp_themes_enabled());
525         connect(m_styleActionGroup, SIGNAL(triggered(QAction*)), this, SLOT(styleActionActivated(QAction*)));
526         styleActionActivated(NULL);
527
528         //Populate the language menu
529         m_languageActionGroup = new QActionGroup(this);
530         QStringList translations = lamexp_query_translations();
531         while(!translations.isEmpty())
532         {
533                 QString langId = translations.takeFirst();
534                 QAction *currentLanguage = new QAction(this);
535                 currentLanguage->setData(langId);
536                 currentLanguage->setText(lamexp_translation_name(langId));
537                 currentLanguage->setIcon(QIcon(QString(":/flags/%1.png").arg(langId)));
538                 currentLanguage->setCheckable(true);
539                 currentLanguage->setChecked(false);
540                 m_languageActionGroup->addAction(currentLanguage);
541                 ui->menuLanguage->insertAction(ui->actionLoadTranslationFromFile, currentLanguage);
542         }
543         ui->menuLanguage->insertSeparator(ui->actionLoadTranslationFromFile);
544         connect(ui->actionLoadTranslationFromFile, SIGNAL(triggered(bool)), this, SLOT(languageFromFileActionActivated(bool)));
545         connect(m_languageActionGroup, SIGNAL(triggered(QAction*)), this, SLOT(languageActionActivated(QAction*)));
546         ui->actionLoadTranslationFromFile->setChecked(false);
547
548         //Activate tools menu actions
549         ui->actionDisableUpdateReminder->setChecked(!m_settings->autoUpdateEnabled());
550         ui->actionDisableSounds->setChecked(!m_settings->soundsEnabled());
551         ui->actionDisableNeroAacNotifications->setChecked(!m_settings->neroAacNotificationsEnabled());
552         ui->actionDisableSlowStartupNotifications->setChecked(!m_settings->antivirNotificationsEnabled());
553         ui->actionDisableShellIntegration->setChecked(!m_settings->shellIntegrationEnabled());
554         ui->actionDisableShellIntegration->setDisabled(lamexp_portable_mode() && ui->actionDisableShellIntegration->isChecked());
555         ui->actionCheckForBetaUpdates->setChecked(m_settings->autoUpdateCheckBeta() || lamexp_version_demo());
556         ui->actionCheckForBetaUpdates->setEnabled(!lamexp_version_demo());
557         ui->actionHibernateComputer->setChecked(m_settings->hibernateComputer());
558         ui->actionHibernateComputer->setEnabled(lamexp_is_hibernation_supported());
559         connect(ui->actionDisableUpdateReminder, SIGNAL(triggered(bool)), this, SLOT(disableUpdateReminderActionTriggered(bool)));
560         connect(ui->actionDisableSounds, SIGNAL(triggered(bool)), this, SLOT(disableSoundsActionTriggered(bool)));
561         connect(ui->actionDisableNeroAacNotifications, SIGNAL(triggered(bool)), this, SLOT(disableNeroAacNotificationsActionTriggered(bool)));
562         connect(ui->actionDisableSlowStartupNotifications, SIGNAL(triggered(bool)), this, SLOT(disableSlowStartupNotificationsActionTriggered(bool)));
563         connect(ui->actionDisableShellIntegration, SIGNAL(triggered(bool)), this, SLOT(disableShellIntegrationActionTriggered(bool)));
564         connect(ui->actionShowDropBoxWidget, SIGNAL(triggered(bool)), this, SLOT(showDropBoxWidgetActionTriggered(bool)));
565         connect(ui->actionHibernateComputer, SIGNAL(triggered(bool)), this, SLOT(hibernateComputerActionTriggered(bool)));
566         connect(ui->actionCheckForBetaUpdates, SIGNAL(triggered(bool)), this, SLOT(checkForBetaUpdatesActionTriggered(bool)));
567         connect(ui->actionImportCueSheet, SIGNAL(triggered(bool)), this, SLOT(importCueSheetActionTriggered(bool)));
568                 
569         //Activate help menu actions
570         ui->actionVisitHomepage->setData(QString::fromLatin1(lamexp_website_url()));
571         ui->actionVisitSupport->setData(QString::fromLatin1(lamexp_support_url()));
572         ui->actionVisitMuldersSite->setData(QString::fromLatin1(lamexp_mulders_url()));
573         ui->actionDocumentFAQ->setData(QString("%1/FAQ.html").arg(QApplication::applicationDirPath()));
574         ui->actionDocumentChangelog->setData(QString("%1/Changelog.html").arg(QApplication::applicationDirPath()));
575         ui->actionDocumentTranslate->setData(QString("%1/Translate.html").arg(QApplication::applicationDirPath()));
576         connect(ui->actionCheckUpdates, SIGNAL(triggered()), this, SLOT(checkUpdatesActionActivated()));
577         connect(ui->actionVisitHomepage, SIGNAL(triggered()), this, SLOT(visitHomepageActionActivated()));
578         connect(ui->actionVisitMuldersSite, SIGNAL(triggered()), this, SLOT(visitHomepageActionActivated()));
579         connect(ui->actionVisitSupport, SIGNAL(triggered()), this, SLOT(visitHomepageActionActivated()));
580         connect(ui->actionDocumentFAQ, SIGNAL(triggered()), this, SLOT(documentActionActivated()));
581         connect(ui->actionDocumentChangelog, SIGNAL(triggered()), this, SLOT(documentActionActivated()));
582         connect(ui->actionDocumentTranslate, SIGNAL(triggered()), this, SLOT(documentActionActivated()));
583         
584         //--------------------------------
585         // Prepare to show window
586         //--------------------------------
587
588         //Center window in screen
589         QRect desktopRect = QApplication::desktop()->screenGeometry();
590         QRect thisRect = this->geometry();
591         move((desktopRect.width() - thisRect.width()) / 2, (desktopRect.height() - thisRect.height()) / 2);
592         setMinimumSize(thisRect.width(), thisRect.height());
593
594         //Create banner
595         m_banner = new WorkingBanner(this);
596
597         //Create DropBox widget
598         m_dropBox = new DropBox(this, m_fileListModel, m_settings);
599         connect(m_fileListModel, SIGNAL(modelReset()), m_dropBox, SLOT(modelChanged()));
600         connect(m_fileListModel, SIGNAL(rowsInserted(QModelIndex,int,int)), m_dropBox, SLOT(modelChanged()));
601         connect(m_fileListModel, SIGNAL(rowsRemoved(QModelIndex,int,int)), m_dropBox, SLOT(modelChanged()));
602         connect(m_fileListModel, SIGNAL(rowAppended()), m_dropBox, SLOT(modelChanged()));
603
604         //Create message handler thread
605         m_messageHandler = new MessageHandlerThread();
606         m_delayedFileList = new QStringList();
607         m_delayedFileTimer = new QTimer();
608         m_delayedFileTimer->setSingleShot(true);
609         m_delayedFileTimer->setInterval(5000);
610         connect(m_messageHandler, SIGNAL(otherInstanceDetected()), this, SLOT(notifyOtherInstance()), Qt::QueuedConnection);
611         connect(m_messageHandler, SIGNAL(fileReceived(QString)), this, SLOT(addFileDelayed(QString)), Qt::QueuedConnection);
612         connect(m_messageHandler, SIGNAL(folderReceived(QString, bool)), this, SLOT(addFolderDelayed(QString, bool)), Qt::QueuedConnection);
613         connect(m_messageHandler, SIGNAL(killSignalReceived()), this, SLOT(close()), Qt::QueuedConnection);
614         connect(m_delayedFileTimer, SIGNAL(timeout()), this, SLOT(handleDelayedFiles()));
615         m_messageHandler->start();
616
617         //Load translation
618         initializeTranslation();
619
620         //Re-translate (make sure we translate once)
621         QEvent languageChangeEvent(QEvent::LanguageChange);
622         changeEvent(&languageChangeEvent);
623
624         //Enable Drag & Drop
625         this->setAcceptDrops(true);
626 }
627
628 ////////////////////////////////////////////////////////////
629 // Destructor
630 ////////////////////////////////////////////////////////////
631
632 MainWindow::~MainWindow(void)
633 {
634         //Stop message handler thread
635         if(m_messageHandler && m_messageHandler->isRunning())
636         {
637                 m_messageHandler->stop();
638                 if(!m_messageHandler->wait(2500))
639                 {
640                         m_messageHandler->terminate();
641                         m_messageHandler->wait();
642                 }
643         }
644
645         //Unset models
646         SET_MODEL(ui->sourceFileView, NULL);
647         SET_MODEL(ui->outputFolderView, NULL);
648         SET_MODEL(ui->metaDataView, NULL);
649
650         //Free memory
651         LAMEXP_DELETE(m_tabActionGroup);
652         LAMEXP_DELETE(m_styleActionGroup);
653         LAMEXP_DELETE(m_languageActionGroup);
654         LAMEXP_DELETE(m_banner);
655         LAMEXP_DELETE(m_fileSystemModel);
656         LAMEXP_DELETE(m_messageHandler);
657         LAMEXP_DELETE(m_delayedFileList);
658         LAMEXP_DELETE(m_delayedFileTimer);
659         LAMEXP_DELETE(m_metaInfoModel);
660         LAMEXP_DELETE(m_encoderButtonGroup);
661         LAMEXP_DELETE(m_modeButtonGroup);
662         LAMEXP_DELETE(m_overwriteButtonGroup);
663         LAMEXP_DELETE(m_sourceFilesContextMenu);
664         LAMEXP_DELETE(m_outputFolderFavoritesMenu);
665         LAMEXP_DELETE(m_outputFolderContextMenu);
666         LAMEXP_DELETE(m_dropBox);
667         LAMEXP_DELETE(m_evenFilterCustumParamsHelp);
668         LAMEXP_DELETE(m_evenFilterOutputFolderMouse);
669         LAMEXP_DELETE(m_evenFilterOutputFolderView);
670         LAMEXP_DELETE(m_evenFilterCompressionTab);
671         
672         //Un-initialize the dialog
673         LAMEXP_DELETE(ui);
674 }
675
676 ////////////////////////////////////////////////////////////
677 // PRIVATE FUNCTIONS
678 ////////////////////////////////////////////////////////////
679
680 /*
681  * Add file to source list
682  */
683 void MainWindow::addFiles(const QStringList &files)
684 {
685         if(files.isEmpty())
686         {
687                 return;
688         }
689
690         ui->tabWidget->setCurrentIndex(0);
691
692         FileAnalyzer *analyzer = new FileAnalyzer(files);
693         connect(analyzer, SIGNAL(fileSelected(QString)), m_banner, SLOT(setText(QString)), Qt::QueuedConnection);
694         connect(analyzer, SIGNAL(progressValChanged(unsigned int)), m_banner, SLOT(setProgressVal(unsigned int)), Qt::QueuedConnection);
695         connect(analyzer, SIGNAL(progressMaxChanged(unsigned int)), m_banner, SLOT(setProgressMax(unsigned int)), Qt::QueuedConnection);
696         connect(analyzer, SIGNAL(fileAnalyzed(AudioFileModel)), m_fileListModel, SLOT(addFile(AudioFileModel)), Qt::QueuedConnection);
697         connect(m_banner, SIGNAL(userAbort()), analyzer, SLOT(abortProcess()), Qt::DirectConnection);
698
699         try
700         {
701                 m_fileListModel->setBlockUpdates(true);
702                 QTime startTime = QTime::currentTime();
703                 m_banner->show(tr("Adding file(s), please wait..."), analyzer);
704         }
705         catch(...)
706         {
707                 /* ignore any exceptions that may occur */
708         }
709
710         m_fileListModel->setBlockUpdates(false);
711         qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
712         ui->sourceFileView->update();
713         qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
714         ui->sourceFileView->scrollToBottom();
715         qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
716
717         if(analyzer->filesDenied())
718         {
719                 QMessageBox::warning(this, tr("Access Denied"), QString("%1<br>%2").arg(NOBR(tr("%n file(s) have been rejected, because read access was not granted!", "", analyzer->filesDenied())), NOBR(tr("This usually means the file is locked by another process."))));
720         }
721         if(analyzer->filesDummyCDDA())
722         {
723                 QMessageBox::warning(this, tr("CDDA Files"), QString("%1<br><br>%2<br>%3").arg(NOBR(tr("%n file(s) have been rejected, because they are dummy CDDA files!", "", analyzer->filesDummyCDDA())), NOBR(tr("Sorry, LameXP cannot extract audio tracks from an Audio-CD at present.")), NOBR(tr("We recommend using %1 for that purpose.").arg("<a href=\"http://www.exactaudiocopy.de/\">Exact Audio Copy</a>"))));
724         }
725         if(analyzer->filesCueSheet())
726         {
727                 QMessageBox::warning(this, tr("Cue Sheet"), QString("%1<br>%2").arg(NOBR(tr("%n file(s) have been rejected, because they appear to be Cue Sheet images!", "",analyzer->filesCueSheet())), NOBR(tr("Please use LameXP's Cue Sheet wizard for importing Cue Sheet files."))));
728         }
729         if(analyzer->filesRejected())
730         {
731                 QMessageBox::warning(this, tr("Files Rejected"), QString("%1<br>%2").arg(NOBR(tr("%n file(s) have been rejected, because the file format could not be recognized!", "", analyzer->filesRejected())), NOBR(tr("This usually means the file is damaged or the file format is not supported."))));
732         }
733
734         LAMEXP_DELETE(analyzer);
735         m_banner->close();
736 }
737
738 /*
739  * Add folder to source list
740  */
741 void MainWindow::addFolder(const QString &path, bool recursive, bool delayed)
742 {
743         QFileInfoList folderInfoList;
744         folderInfoList << QFileInfo(path);
745         QStringList fileList;
746         
747         m_banner->show(tr("Scanning folder(s) for files, please wait..."));
748         
749         QApplication::processEvents();
750         lamexp_check_escape_state();
751
752         while(!folderInfoList.isEmpty())
753         {
754                 if(lamexp_check_escape_state())
755                 {
756                         lamexp_beep(lamexp_beep_error);
757                         qWarning("Operation cancelled by user!");
758                         fileList.clear();
759                         break;
760                 }
761                 
762                 QDir currentDir(folderInfoList.takeFirst().canonicalFilePath());
763                 QFileInfoList fileInfoList = currentDir.entryInfoList(QDir::Files | QDir::NoSymLinks);
764
765                 while(!fileInfoList.isEmpty())
766                 {
767                         fileList << fileInfoList.takeFirst().canonicalFilePath();
768                 }
769
770                 QApplication::processEvents();
771
772                 if(recursive)
773                 {
774                         folderInfoList.append(currentDir.entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot | QDir::NoSymLinks));
775                         QApplication::processEvents();
776                 }
777         }
778         
779         m_banner->close();
780         QApplication::processEvents();
781
782         if(!fileList.isEmpty())
783         {
784                 if(delayed)
785                 {
786                         addFilesDelayed(fileList);
787                 }
788                 else
789                 {
790                         addFiles(fileList);
791                 }
792         }
793 }
794
795 /*
796  * Check for updates
797  */
798 bool MainWindow::checkForUpdates(void)
799 {
800         bool bReadyToInstall = false;
801         
802         UpdateDialog *updateDialog = new UpdateDialog(m_settings, this);
803         updateDialog->exec();
804
805         if(updateDialog->getSuccess())
806         {
807                 m_settings->autoUpdateLastCheck(QDate::currentDate().toString(Qt::ISODate));
808                 bReadyToInstall = updateDialog->updateReadyToInstall();
809         }
810
811         LAMEXP_DELETE(updateDialog);
812         return bReadyToInstall;
813 }
814
815 /*
816  * Refresh list of favorites
817  */
818 void MainWindow::refreshFavorites(void)
819 {
820         QList<QAction*> folderList = m_outputFolderFavoritesMenu->actions();
821         QStringList favorites = m_settings->favoriteOutputFolders().split("|", QString::SkipEmptyParts);
822         while(favorites.count() > 6) favorites.removeFirst();
823
824         while(!folderList.isEmpty())
825         {
826                 QAction *currentItem = folderList.takeFirst();
827                 if(currentItem->isSeparator()) break;
828                 m_outputFolderFavoritesMenu->removeAction(currentItem);
829                 LAMEXP_DELETE(currentItem);
830         }
831
832         QAction *lastItem = m_outputFolderFavoritesMenu->actions().first();
833
834         while(!favorites.isEmpty())
835         {
836                 QString path = favorites.takeLast();
837                 if(QDir(path).exists())
838                 {
839                         QAction *action = new QAction(QIcon(":/icons/folder_go.png"), QDir::toNativeSeparators(path), this);
840                         action->setData(path);
841                         m_outputFolderFavoritesMenu->insertAction(lastItem, action);
842                         connect(action, SIGNAL(triggered(bool)), this, SLOT(gotoFavoriteFolder()));
843                         lastItem = action;
844                 }
845         }
846 }
847
848 /*
849  * Initilaize translation
850  */
851 void MainWindow::initializeTranslation(void)
852 {
853         bool translationLoaded = false;
854
855         //Try to load "external" translation file
856         if(!m_settings->currentLanguageFile().isEmpty())
857         {
858                 const QString qmFilePath = QFileInfo(m_settings->currentLanguageFile()).canonicalFilePath();
859                 if((!qmFilePath.isEmpty()) && QFileInfo(qmFilePath).exists() && QFileInfo(qmFilePath).isFile() && (QFileInfo(qmFilePath).suffix().compare("qm", Qt::CaseInsensitive) == 0))
860                 {
861                         if(lamexp_install_translator_from_file(qmFilePath))
862                         {
863                                 QList<QAction*> actions = m_languageActionGroup->actions();
864                                 while(!actions.isEmpty()) actions.takeFirst()->setChecked(false);
865                                 ui->actionLoadTranslationFromFile->setChecked(true);
866                                 translationLoaded = true;
867                         }
868                 }
869         }
870
871         //Try to load "built-in" translation file
872         if(!translationLoaded)
873         {
874                 QList<QAction*> languageActions = m_languageActionGroup->actions();
875                 while(!languageActions.isEmpty())
876                 {
877                         QAction *currentLanguage = languageActions.takeFirst();
878                         if(currentLanguage->data().toString().compare(m_settings->currentLanguage(), Qt::CaseInsensitive) == 0)
879                         {
880                                 currentLanguage->setChecked(true);
881                                 languageActionActivated(currentLanguage);
882                                 translationLoaded = true;
883                         }
884                 }
885         }
886
887         //Fallback to default translation
888         if(!translationLoaded)
889         {
890                 QList<QAction*> languageActions = m_languageActionGroup->actions();
891                 while(!languageActions.isEmpty())
892                 {
893                         QAction *currentLanguage = languageActions.takeFirst();
894                         if(currentLanguage->data().toString().compare(LAMEXP_DEFAULT_LANGID, Qt::CaseInsensitive) == 0)
895                         {
896                                 currentLanguage->setChecked(true);
897                                 languageActionActivated(currentLanguage);
898                                 translationLoaded = true;
899                         }
900                 }
901         }
902
903         //Make sure we loaded some translation
904         if(!translationLoaded)
905         {
906                 qFatal("Failed to load any translation, this is NOT supposed to happen!");
907         }
908 }
909
910 ////////////////////////////////////////////////////////////
911 // EVENTS
912 ////////////////////////////////////////////////////////////
913
914 /*
915  * Window is about to be shown
916  */
917 void MainWindow::showEvent(QShowEvent *event)
918 {
919         m_accepted = false;
920         resizeEvent(NULL);
921         sourceModelChanged();
922         
923         if(!event->spontaneous())
924         {
925                 ui->tabWidget->setCurrentIndex(0);
926         }
927
928         if(m_firstTimeShown)
929         {
930                 m_firstTimeShown = false;
931                 QTimer::singleShot(0, this, SLOT(windowShown()));
932         }
933         else
934         {
935                 if(m_settings->dropBoxWidgetEnabled())
936                 {
937                         m_dropBox->setVisible(true);
938                 }
939         }
940 }
941
942 /*
943  * Re-translate the UI
944  */
945 void MainWindow::changeEvent(QEvent *e)
946 {
947         if(e->type() == QEvent::LanguageChange)
948         {
949                 /*qWarning("\nMainWindow::changeEvent()\n");*/
950
951                 int comboBoxIndex[8];
952                 
953                 //Backup combobox indices, as retranslateUi() resets
954                 comboBoxIndex[0] = ui->comboBoxMP3ChannelMode->currentIndex();
955                 comboBoxIndex[1] = ui->comboBoxSamplingRate->currentIndex();
956                 comboBoxIndex[2] = ui->comboBoxAACProfile->currentIndex();
957                 comboBoxIndex[3] = ui->comboBoxAftenCodingMode->currentIndex();
958                 comboBoxIndex[4] = ui->comboBoxAftenDRCMode->currentIndex();
959                 comboBoxIndex[5] = ui->comboBoxNormalizationMode->currentIndex();
960                 comboBoxIndex[6] = 0; //comboBoxOpusOptimize->currentIndex();
961                 comboBoxIndex[7] = ui->comboBoxOpusFramesize->currentIndex();
962                 
963                 //Re-translate from UIC
964                 ui->retranslateUi(this);
965
966                 //Restore combobox indices
967                 ui->comboBoxMP3ChannelMode->setCurrentIndex(comboBoxIndex[0]);
968                 ui->comboBoxSamplingRate->setCurrentIndex(comboBoxIndex[1]);
969                 ui->comboBoxAACProfile->setCurrentIndex(comboBoxIndex[2]);
970                 ui->comboBoxAftenCodingMode->setCurrentIndex(comboBoxIndex[3]);
971                 ui->comboBoxAftenDRCMode->setCurrentIndex(comboBoxIndex[4]);
972                 ui->comboBoxNormalizationMode->setCurrentIndex(comboBoxIndex[5]);
973                 //comboBoxOpusOptimize->setCurrentIndex(comboBoxIndex[6]);
974                 ui->comboBoxOpusFramesize->setCurrentIndex(comboBoxIndex[7]);
975
976                 //Update the window title
977                 if(LAMEXP_DEBUG)
978                 {
979                         setWindowTitle(QString("%1 [!!! DEBUG BUILD !!!]").arg(windowTitle()));
980                 }
981                 else if(lamexp_version_demo())
982                 {
983                         setWindowTitle(QString("%1 [%2]").arg(windowTitle(), tr("DEMO VERSION")));
984                 }
985
986                 //Manually re-translate widgets that UIC doesn't handle
987                 m_outputFolderNoteBox->setText(tr("Initializing directory outline, please be patient..."));
988                 m_dropNoteLabel->setText(QString("<br><br>» %1 Â«<br><br><br><img src=\":/images/Sound.png\">").arg(tr("You can drop in audio files here!")));
989                 m_showDetailsContextAction->setText(tr("Show Details"));
990                 m_previewContextAction->setText(tr("Open File in External Application"));
991                 m_findFileContextAction->setText(tr("Browse File Location"));
992                 m_showFolderContextAction->setText(tr("Browse Selected Folder"));
993                 m_refreshFolderContextAction->setText(tr("Refresh Directory Outline"));
994                 m_goUpFolderContextAction->setText(tr("Go To Parent Directory"));
995                 m_addFavoriteFolderAction->setText(tr("Bookmark Current Output Folder"));
996                 m_exportCsvContextAction->setText(tr("Export Meta Tags to CSV File"));
997                 m_importCsvContextAction->setText(tr("Import Meta Tags from CSV File"));
998
999                 //Force GUI update
1000                 m_metaInfoModel->clearData();
1001                 m_metaInfoModel->setData(m_metaInfoModel->index(4, 1), m_settings->metaInfoPosition());
1002                 updateEncoder(m_settings->compressionEncoder());
1003                 updateLameAlgoQuality(ui->sliderLameAlgoQuality->value());
1004                 updateMaximumInstances(ui->sliderMaxInstances->value());
1005                 renameOutputPatternChanged(ui->lineEditRenamePattern->text(), true);
1006
1007                 //Re-install shell integration
1008                 if(m_settings->shellIntegrationEnabled())
1009                 {
1010                         ShellIntegration::install();
1011                 }
1012
1013                 //Translate system menu
1014                 lamexp_update_sysmenu(this, IDM_ABOUTBOX, ui->buttonAbout->text());
1015                         
1016                 //Force resize, if needed
1017                 tabPageChanged(ui->tabWidget->currentIndex());
1018         }
1019 }
1020
1021 /*
1022  * File dragged over window
1023  */
1024 void MainWindow::dragEnterEvent(QDragEnterEvent *event)
1025 {
1026         QStringList formats = event->mimeData()->formats();
1027         
1028         if(formats.contains("application/x-qt-windows-mime;value=\"FileNameW\"", Qt::CaseInsensitive) && formats.contains("text/uri-list", Qt::CaseInsensitive))
1029         {
1030                 event->acceptProposedAction();
1031         }
1032 }
1033
1034 /*
1035  * File dropped onto window
1036  */
1037 void MainWindow::dropEvent(QDropEvent *event)
1038 {
1039         ABORT_IF_BUSY;
1040
1041         QStringList droppedFiles;
1042         QList<QUrl> urls = event->mimeData()->urls();
1043
1044         while(!urls.isEmpty())
1045         {
1046                 QUrl currentUrl = urls.takeFirst();
1047                 QFileInfo file(currentUrl.toLocalFile());
1048                 if(!file.exists())
1049                 {
1050                         continue;
1051                 }
1052                 if(file.isFile())
1053                 {
1054                         qDebug("Dropped File: %s", QUTF8(file.canonicalFilePath()));
1055                         droppedFiles << file.canonicalFilePath();
1056                         continue;
1057                 }
1058                 if(file.isDir())
1059                 {
1060                         qDebug("Dropped Folder: %s", QUTF8(file.canonicalFilePath()));
1061                         QList<QFileInfo> list = QDir(file.canonicalFilePath()).entryInfoList(QDir::Files | QDir::NoSymLinks);
1062                         if(list.count() > 0)
1063                         {
1064                                 for(int j = 0; j < list.count(); j++)
1065                                 {
1066                                         droppedFiles << list.at(j).canonicalFilePath();
1067                                 }
1068                         }
1069                         else
1070                         {
1071                                 list = QDir(file.canonicalFilePath()).entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot | QDir::NoSymLinks);
1072                                 for(int j = 0; j < list.count(); j++)
1073                                 {
1074                                         qDebug("Descending to Folder: %s", QUTF8(list.at(j).canonicalFilePath()));
1075                                         urls.prepend(QUrl::fromLocalFile(list.at(j).canonicalFilePath()));
1076                                 }
1077                         }
1078                 }
1079         }
1080         
1081         if(!droppedFiles.isEmpty())
1082         {
1083                 addFilesDelayed(droppedFiles, true);
1084         }
1085 }
1086
1087 /*
1088  * Window tries to close
1089  */
1090 void MainWindow::closeEvent(QCloseEvent *event)
1091 {
1092         if(m_banner->isVisible() || m_delayedFileTimer->isActive())
1093         {
1094                 lamexp_beep(lamexp_beep_warning);
1095                 event->ignore();
1096         }
1097         
1098         if(m_dropBox)
1099         {
1100                 m_dropBox->hide();
1101         }
1102 }
1103
1104 /*
1105  * Window was resized
1106  */
1107 void MainWindow::resizeEvent(QResizeEvent *event)
1108 {
1109         if(event) QMainWindow::resizeEvent(event);
1110
1111         if(QWidget *port = ui->sourceFileView->viewport())
1112         {
1113                 m_dropNoteLabel->setGeometry(port->geometry());
1114         }
1115
1116         if(QWidget *port = ui->outputFolderView->viewport())
1117         {
1118                 m_outputFolderNoteBox->setGeometry(16, (port->height() - 64) / 2, port->width() - 32,  64);
1119         }
1120 }
1121
1122 /*
1123  * Key press event filter
1124  */
1125 void MainWindow::keyPressEvent(QKeyEvent *e)
1126 {
1127         if(e->key() == Qt::Key_Delete)
1128         {
1129                 if(ui->sourceFileView->isVisible())
1130                 {
1131                         QTimer::singleShot(0, this, SLOT(removeFileButtonClicked()));
1132                         return;
1133                 }
1134         }
1135
1136         if(e->modifiers().testFlag(Qt::ControlModifier) && (e->key() == Qt::Key_F5))
1137         {
1138                 initializeTranslation();
1139                 lamexp_beep(lamexp_beep_info);
1140                 return;
1141         }
1142
1143         if(e->key() == Qt::Key_F5)
1144         {
1145                 if(ui->outputFolderView->isVisible())
1146                 {
1147                         QTimer::singleShot(0, this, SLOT(refreshFolderContextActionTriggered()));
1148                         return;
1149                 }
1150         }
1151
1152         QMainWindow::keyPressEvent(e);
1153 }
1154
1155 /*
1156  * Event filter
1157  */
1158 bool MainWindow::eventFilter(QObject *obj, QEvent *event)
1159 {
1160         if(obj == m_fileSystemModel)
1161         {
1162                 if(QApplication::overrideCursor() == NULL)
1163                 {
1164                         QApplication::setOverrideCursor(QCursor(Qt::WaitCursor));
1165                         QTimer::singleShot(250, this, SLOT(restoreCursor()));
1166                 }
1167         }
1168
1169         return QMainWindow::eventFilter(obj, event);
1170 }
1171
1172 bool MainWindow::event(QEvent *e)
1173 {
1174         switch(e->type())
1175         {
1176         case lamexp_event_queryendsession:
1177                 qWarning("System is shutting down, main window prepares to close...");
1178                 if(m_banner->isVisible()) m_banner->close();
1179                 if(m_delayedFileTimer->isActive()) m_delayedFileTimer->stop();
1180                 return true;
1181         case lamexp_event_endsession:
1182                 qWarning("System is shutting down, main window will close now...");
1183                 if(isVisible())
1184                 {
1185                         while(!close())
1186                         {
1187                                 QApplication::processEvents(QEventLoop::WaitForMoreEvents & QEventLoop::ExcludeUserInputEvents);
1188                         }
1189                 }
1190                 m_fileListModel->clearFiles();
1191                 return true;
1192         case QEvent::MouseButtonPress:
1193                 if(ui->outputFolderEdit->isVisible())
1194                 {
1195                         QTimer::singleShot(0, this, SLOT(outputFolderEditFinished()));
1196                 }
1197         default:
1198                 return QMainWindow::event(e);
1199         }
1200 }
1201
1202 bool MainWindow::winEvent(MSG *message, long *result)
1203 {
1204         if(lamexp_check_sysmenu_msg(message, IDM_ABOUTBOX))
1205         {
1206                 QTimer::singleShot(0, ui->buttonAbout, SLOT(click()));
1207                 *result = 0;
1208                 return true;
1209         }
1210         return WinSevenTaskbar::handleWinEvent(message, result);
1211 }
1212
1213 ////////////////////////////////////////////////////////////
1214 // Slots
1215 ////////////////////////////////////////////////////////////
1216
1217 // =========================================================
1218 // Show window slots
1219 // =========================================================
1220
1221 /*
1222  * Window shown
1223  */
1224 void MainWindow::windowShown(void)
1225 {
1226         const QStringList &arguments = lamexp_arguments(); //QApplication::arguments();
1227
1228         //Force resize event
1229         resizeEvent(NULL);
1230
1231         //First run?
1232         bool firstRun = false;
1233         for(int i = 0; i < arguments.count(); i++)
1234         {
1235                 /*QMessageBox::information(this, QString::number(i), arguments[i]);*/
1236                 if(!arguments[i].compare("--first-run", Qt::CaseInsensitive)) firstRun = true;
1237         }
1238
1239         //Check license
1240         if((m_settings->licenseAccepted() <= 0) || firstRun)
1241         {
1242                 int iAccepted = m_settings->licenseAccepted();
1243
1244                 if((iAccepted == 0) || firstRun)
1245                 {
1246                         AboutDialog *about = new AboutDialog(m_settings, this, true);
1247                         iAccepted = about->exec();
1248                         if(iAccepted <= 0) iAccepted = -2;
1249                         LAMEXP_DELETE(about);
1250                 }
1251
1252                 if(iAccepted <= 0)
1253                 {
1254                         m_settings->licenseAccepted(++iAccepted);
1255                         m_settings->syncNow();
1256                         QApplication::processEvents();
1257                         lamexp_play_sound(IDR_WAVE_WHAMMY, false);
1258                         QMessageBox::critical(this, tr("License Declined"), tr("You have declined the license. Consequently the application will exit now!"), tr("Goodbye!"));
1259                         QFileInfo uninstallerInfo = QFileInfo(QString("%1/Uninstall.exe").arg(QApplication::applicationDirPath()));
1260                         if(uninstallerInfo.exists())
1261                         {
1262                                 QString uninstallerDir = uninstallerInfo.canonicalPath();
1263                                 QString uninstallerPath = uninstallerInfo.canonicalFilePath();
1264                                 for(int i = 0; i < 3; i++)
1265                                 {
1266                                         if(lamexp_exec_shell(this, QDir::toNativeSeparators(uninstallerPath), "/Force", QDir::toNativeSeparators(uninstallerDir))) break;
1267                                 }
1268                         }
1269                         QApplication::quit();
1270                         return;
1271                 }
1272                 
1273                 lamexp_play_sound(IDR_WAVE_WOOHOO, false);
1274                 m_settings->licenseAccepted(1);
1275                 m_settings->syncNow();
1276                 if(lamexp_version_demo()) showAnnounceBox();
1277         }
1278         
1279         //Check for expiration
1280         if(lamexp_version_demo())
1281         {
1282                 if(lamexp_current_date_safe() >= lamexp_version_expires())
1283                 {
1284                         qWarning("Binary has expired !!!");
1285                         lamexp_play_sound(IDR_WAVE_WHAMMY, false);
1286                         if(QMessageBox::warning(this, tr("LameXP - Expired"), QString("%1<br>%2").arg(NOBR(tr("This demo (pre-release) version of LameXP has expired at %1.").arg(lamexp_version_expires().toString(Qt::ISODate))), NOBR(tr("LameXP is free software and release versions won't expire."))), tr("Check for Updates"), tr("Exit Program")) == 0)
1287                         {
1288                                 checkForUpdates();
1289                         }
1290                         QApplication::quit();
1291                         return;
1292                 }
1293         }
1294
1295         //Slow startup indicator
1296         if(m_settings->slowStartup() && m_settings->antivirNotificationsEnabled())
1297         {
1298                 QString message;
1299                 message += NOBR(tr("It seems that a bogus anti-virus software is slowing down the startup of LameXP.")).append("<br>");
1300                 message += NOBR(tr("Please refer to the %1 document for details and solutions!")).arg("<a href=\"http://lamexp.sourceforge.net/doc/FAQ.html#df406578\">F.A.Q.</a>").append("<br>");
1301                 if(QMessageBox::warning(this, tr("Slow Startup"), message, tr("Discard"), tr("Don't Show Again")) == 1)
1302                 {
1303                         m_settings->antivirNotificationsEnabled(false);
1304                         ui->actionDisableSlowStartupNotifications->setChecked(!m_settings->antivirNotificationsEnabled());
1305                 }
1306         }
1307
1308         //Update reminder
1309         if(lamexp_current_date_safe() >= lamexp_version_date().addYears(1))
1310         {
1311                 qWarning("Binary is more than a year old, time to update!");
1312                 int ret = QMessageBox::warning(this, tr("Urgent Update"), NOBR(tr("Your version of LameXP is more than a year old. Time for an update!")), tr("Check for Updates"), tr("Exit Program"), tr("Ignore"));
1313                 switch(ret)
1314                 {
1315                 case 0:
1316                         if(checkForUpdates())
1317                         {
1318                                 QApplication::quit();
1319                                 return;
1320                         }
1321                         break;
1322                 case 1:
1323                         QApplication::quit();
1324                         return;
1325                 default:
1326                         QEventLoop loop; QTimer::singleShot(7000, &loop, SLOT(quit()));
1327                         lamexp_play_sound(IDR_WAVE_WAITING, true);
1328                         m_banner->show(tr("Skipping update check this time, please be patient..."), &loop);
1329                         break;
1330                 }
1331         }
1332         else if(m_settings->autoUpdateEnabled())
1333         {
1334                 QDate lastUpdateCheck = QDate::fromString(m_settings->autoUpdateLastCheck(), Qt::ISODate);
1335                 if(!firstRun && (!lastUpdateCheck.isValid() || lamexp_current_date_safe() >= lastUpdateCheck.addDays(14)))
1336                 {
1337                         if(QMessageBox::information(this, tr("Update Reminder"), NOBR(lastUpdateCheck.isValid() ? tr("Your last update check was more than 14 days ago. Check for updates now?") : tr("Your did not check for LameXP updates yet. Check for updates now?")), tr("Check for Updates"), tr("Postpone")) == 0)
1338                         {
1339                                 if(checkForUpdates())
1340                                 {
1341                                         QApplication::quit();
1342                                         return;
1343                                 }
1344                         }
1345                 }
1346         }
1347
1348         //Check for AAC support
1349         const int aacEncoder = EncoderRegistry::getAacEncoder();
1350         if(aacEncoder == SettingsModel::AAC_ENCODER_NERO)
1351         {
1352                 if(m_settings->neroAacNotificationsEnabled())
1353                 {
1354                         if(lamexp_tool_version("neroAacEnc.exe") < lamexp_toolver_neroaac())
1355                         {
1356                                 QString messageText;
1357                                 messageText += NOBR(tr("LameXP detected that your version of the Nero AAC encoder is outdated!")).append("<br>");
1358                                 messageText += NOBR(tr("The current version available is %1 (or later), but you still have version %2 installed.").arg(lamexp_version2string("?.?.?.?", lamexp_toolver_neroaac(), tr("n/a")), lamexp_version2string("?.?.?.?", lamexp_tool_version("neroAacEnc.exe"), tr("n/a")))).append("<br><br>");
1359                                 messageText += NOBR(tr("You can download the latest version of the Nero AAC encoder from the Nero website at:")).append("<br>");
1360                                 messageText += "<nobr><tt>" + LINK(AboutDialog::neroAacUrl) + "</tt></nobr><br><br>";
1361                                 messageText += NOBR(tr("(Hint: Please ignore the name of the downloaded ZIP file and check the included 'changelog.txt' instead!)")).append("<br>");
1362                                 QMessageBox::information(this, tr("AAC Encoder Outdated"), messageText);
1363                         }
1364                 }
1365         }
1366         else
1367         {
1368                 if(m_settings->neroAacNotificationsEnabled() && (aacEncoder <= SettingsModel::AAC_ENCODER_NONE))
1369                 {
1370                         QString appPath = QDir(QCoreApplication::applicationDirPath()).canonicalPath();
1371                         if(appPath.isEmpty()) appPath = QCoreApplication::applicationDirPath();
1372                         QString messageText;
1373                         messageText += NOBR(tr("The Nero AAC encoder could not be found. AAC encoding support will be disabled.")).append("<br>");
1374                         messageText += NOBR(tr("Please put 'neroAacEnc.exe', 'neroAacDec.exe' and 'neroAacTag.exe' into the LameXP directory!")).append("<br><br>");
1375                         messageText += NOBR(tr("Your LameXP directory is located here:")).append("<br>");
1376                         messageText += QString("<nobr><tt>%1</tt></nobr><br><br>").arg(FSLINK(QDir::toNativeSeparators(appPath)));
1377                         messageText += NOBR(tr("You can download the Nero AAC encoder for free from the official Nero website at:")).append("<br>");
1378                         messageText += "<nobr><tt>" + LINK(AboutDialog::neroAacUrl) + "</tt></nobr><br>";
1379                         if(QMessageBox::information(this, tr("AAC Support Disabled"), messageText, tr("Discard"), tr("Don't Show Again")) == 1)
1380                         {
1381                                 m_settings->neroAacNotificationsEnabled(false);
1382                                 ui->actionDisableNeroAacNotifications->setChecked(!m_settings->neroAacNotificationsEnabled());
1383                         }
1384                 }
1385         }
1386
1387         //Add files from the command-line
1388         for(int i = 0; i < arguments.count() - 1; i++)
1389         {
1390                 QStringList addedFiles;
1391                 if(!arguments[i].compare("--add", Qt::CaseInsensitive))
1392                 {
1393                         QFileInfo currentFile(arguments[++i].trimmed());
1394                         qDebug("Adding file from CLI: %s", QUTF8(currentFile.absoluteFilePath()));
1395                         addedFiles.append(currentFile.absoluteFilePath());
1396                 }
1397                 if(!addedFiles.isEmpty())
1398                 {
1399                         addFilesDelayed(addedFiles);
1400                 }
1401         }
1402
1403         //Add folders from the command-line
1404         for(int i = 0; i < arguments.count() - 1; i++)
1405         {
1406                 if(!arguments[i].compare("--add-folder", Qt::CaseInsensitive))
1407                 {
1408                         QFileInfo currentFile(arguments[++i].trimmed());
1409                         qDebug("Adding folder from CLI: %s", QUTF8(currentFile.absoluteFilePath()));
1410                         addFolder(currentFile.absoluteFilePath(), false, true);
1411                 }
1412                 if(!arguments[i].compare("--add-recursive", Qt::CaseInsensitive))
1413                 {
1414                         QFileInfo currentFile(arguments[++i].trimmed());
1415                         qDebug("Adding folder recursively from CLI: %s", QUTF8(currentFile.absoluteFilePath()));
1416                         addFolder(currentFile.absoluteFilePath(), true, true);
1417                 }
1418         }
1419
1420         //Enable shell integration
1421         if(m_settings->shellIntegrationEnabled())
1422         {
1423                 ShellIntegration::install();
1424         }
1425
1426         //Make DropBox visible
1427         if(m_settings->dropBoxWidgetEnabled())
1428         {
1429                 m_dropBox->setVisible(true);
1430         }
1431 }
1432
1433 /*
1434  * Show announce box
1435  */
1436 void MainWindow::showAnnounceBox(void)
1437 {
1438         const unsigned int timeout = 8U;
1439
1440         const QString announceText = QString("%1<br><br>%2<br><nobr><tt>%3</tt></nobr><br>").arg
1441         (
1442                 NOBR("We are still looking for LameXP translators!"),
1443                 NOBR("If you are willing to translate LameXP to your language or to complete an existing translation, please refer to:"),
1444                 LINK("http://lamexp.sourceforge.net/doc/Translate.html")
1445         );
1446
1447         QMessageBox *announceBox = new QMessageBox(QMessageBox::Warning, "We want you!", announceText, QMessageBox::NoButton, this);
1448         announceBox->setWindowFlags(Qt::Window | Qt::WindowTitleHint | Qt::CustomizeWindowHint);
1449         announceBox->setIconPixmap(QIcon(":/images/Announcement.png").pixmap(64,79));
1450         
1451         QTimer *timers[timeout+1];
1452         QPushButton *buttons[timeout+1];
1453
1454         for(unsigned int i = 0; i <= timeout; i++)
1455         {
1456                 QString text = (i > 0) ? QString("%1 (%2)").arg(tr("Discard"), QString::number(i)) : tr("Discard");
1457                 buttons[i] = announceBox->addButton(text, (i > 0) ? QMessageBox::NoRole : QMessageBox::AcceptRole);
1458         }
1459
1460         for(unsigned int i = 0; i <= timeout; i++)
1461         {
1462                 buttons[i]->setEnabled(i == 0);
1463                 buttons[i]->setVisible(i == timeout);
1464         }
1465
1466         for(unsigned int i = 0; i < timeout; i++)
1467         {
1468                 timers[i] = new QTimer(this);
1469                 timers[i]->setSingleShot(true);
1470                 timers[i]->setInterval(1000);
1471                 connect(timers[i], SIGNAL(timeout()), buttons[i+1], SLOT(hide()));
1472                 connect(timers[i], SIGNAL(timeout()), buttons[i], SLOT(show()));
1473                 if(i > 0)
1474                 {
1475                         connect(timers[i], SIGNAL(timeout()), timers[i-1], SLOT(start()));
1476                 }
1477         }
1478
1479         timers[timeout-1]->start();
1480         announceBox->exec();
1481
1482         for(unsigned int i = 0; i < timeout; i++)
1483         {
1484                 timers[i]->stop();
1485                 LAMEXP_DELETE(timers[i]);
1486         }
1487
1488         LAMEXP_DELETE(announceBox);
1489 }
1490
1491 // =========================================================
1492 // Main button solots
1493 // =========================================================
1494
1495 /*
1496  * Encode button
1497  */
1498 void MainWindow::encodeButtonClicked(void)
1499 {
1500         static const unsigned __int64 oneGigabyte = 1073741824ui64; 
1501         static const unsigned __int64 minimumFreeDiskspaceMultiplier = 2ui64;
1502         static const char *writeTestBuffer = "LAMEXP_WRITE_TEST";
1503         
1504         ABORT_IF_BUSY;
1505
1506         if(m_fileListModel->rowCount() < 1)
1507         {
1508                 QMessageBox::warning(this, tr("LameXP"), NOBR(tr("You must add at least one file to the list before proceeding!")));
1509                 ui->tabWidget->setCurrentIndex(0);
1510                 return;
1511         }
1512         
1513         QString tempFolder = m_settings->customTempPathEnabled() ? m_settings->customTempPath() : lamexp_temp_folder2();
1514         if(!QFileInfo(tempFolder).exists() || !QFileInfo(tempFolder).isDir())
1515         {
1516                 if(QMessageBox::warning(this, tr("Not Found"), QString("%1<br><tt>%2</tt>").arg(NOBR(tr("Your currently selected TEMP folder does not exist anymore:")), NOBR(QDir::toNativeSeparators(tempFolder))), tr("Restore Default"), tr("Cancel")) == 0)
1517                 {
1518                         SET_CHECKBOX_STATE(ui->checkBoxUseSystemTempFolder, m_settings->customTempPathEnabledDefault());
1519                 }
1520                 return;
1521         }
1522
1523         bool ok = false;
1524         unsigned __int64 currentFreeDiskspace = lamexp_free_diskspace(tempFolder, &ok);
1525
1526         if(ok && (currentFreeDiskspace < (oneGigabyte * minimumFreeDiskspaceMultiplier)))
1527         {
1528                 QStringList tempFolderParts = tempFolder.split("/", QString::SkipEmptyParts, Qt::CaseInsensitive);
1529                 tempFolderParts.takeLast();
1530                 if(m_settings->soundsEnabled()) lamexp_play_sound(IDR_WAVE_WHAMMY, false);
1531                 QString lowDiskspaceMsg = QString("%1<br>%2<br><br>%3<br>%4<br>").arg
1532                 (
1533                         NOBR(tr("There are less than %1 GB of free diskspace available on your system's TEMP folder.").arg(QString::number(minimumFreeDiskspaceMultiplier))),
1534                         NOBR(tr("It is highly recommend to free up more diskspace before proceeding with the encode!")),
1535                         NOBR(tr("Your TEMP folder is located at:")),
1536                         QString("<nobr><tt>%1</tt></nobr>").arg(FSLINK(tempFolderParts.join("\\")))
1537                 );
1538                 switch(QMessageBox::warning(this, tr("Low Diskspace Warning"), lowDiskspaceMsg, tr("Abort Encoding Process"), tr("Clean Disk Now"), tr("Ignore")))
1539                 {
1540                 case 1:
1541                         QProcess::startDetached(QString("%1/cleanmgr.exe").arg(lamexp_known_folder(lamexp_folder_systemfolder)), QStringList() << "/D" << tempFolderParts.first());
1542                 case 0:
1543                         return;
1544                         break;
1545                 default:
1546                         QMessageBox::warning(this, tr("Low Diskspace"), NOBR(tr("You are proceeding with low diskspace. Problems might occur!")));
1547                         break;
1548                 }
1549         }
1550
1551         switch(m_settings->compressionEncoder())
1552         {
1553         case SettingsModel::MP3Encoder:
1554         case SettingsModel::VorbisEncoder:
1555         case SettingsModel::AACEncoder:
1556         case SettingsModel::AC3Encoder:
1557         case SettingsModel::FLACEncoder:
1558         case SettingsModel::OpusEncoder:
1559         case SettingsModel::DCAEncoder:
1560         case SettingsModel::PCMEncoder:
1561                 break;
1562         default:
1563                 QMessageBox::warning(this, tr("LameXP"), tr("Sorry, an unsupported encoder has been chosen!"));
1564                 ui->tabWidget->setCurrentIndex(3);
1565                 return;
1566         }
1567
1568         if(!m_settings->outputToSourceDir())
1569         {
1570                 QFile writeTest(QString("%1/~%2.txt").arg(m_settings->outputDir(), lamexp_rand_str()));
1571                 if(!(writeTest.open(QIODevice::ReadWrite) && (writeTest.write(writeTestBuffer) == strlen(writeTestBuffer))))
1572                 {
1573                         QMessageBox::warning(this, tr("LameXP"), QString("%1<br><nobr>%2</nobr><br><br>%3").arg(tr("Cannot write to the selected output directory."), m_settings->outputDir(), tr("Please choose a different directory!")));
1574                         ui->tabWidget->setCurrentIndex(1);
1575                         return;
1576                 }
1577                 else
1578                 {
1579                         writeTest.close();
1580                         writeTest.remove();
1581                 }
1582         }
1583
1584         m_accepted = true;
1585         close();
1586 }
1587
1588 /*
1589  * About button
1590  */
1591 void MainWindow::aboutButtonClicked(void)
1592 {
1593         ABORT_IF_BUSY;
1594
1595         TEMP_HIDE_DROPBOX
1596         (
1597                 AboutDialog *aboutBox = new AboutDialog(m_settings, this);
1598                 aboutBox->exec();
1599                 LAMEXP_DELETE(aboutBox);
1600         );
1601 }
1602
1603 /*
1604  * Close button
1605  */
1606 void MainWindow::closeButtonClicked(void)
1607 {
1608         ABORT_IF_BUSY;
1609         close();
1610 }
1611
1612 // =========================================================
1613 // Tab widget slots
1614 // =========================================================
1615
1616 /*
1617  * Tab page changed
1618  */
1619 void MainWindow::tabPageChanged(int idx)
1620 {
1621         resizeEvent(NULL);
1622         
1623         QList<QAction*> actions = m_tabActionGroup->actions();
1624         for(int i = 0; i < actions.count(); i++)
1625         {
1626                 bool ok = false;
1627                 int actionIndex = actions.at(i)->data().toInt(&ok);
1628                 if(ok && actionIndex == idx)
1629                 {
1630                         actions.at(i)->setChecked(true);
1631                 }
1632         }
1633
1634         int initialWidth = this->width();
1635         int maximumWidth = QApplication::desktop()->availableGeometry().width();
1636
1637         //Make sure all tab headers are fully visible
1638         if(this->isVisible())
1639         {
1640                 int delta = ui->tabWidget->sizeHint().width() - ui->tabWidget->width();
1641                 if(delta > 0)
1642                 {
1643                         this->resize(qMin(this->width() + delta, maximumWidth), this->height());
1644                 }
1645         }
1646
1647         //Tab specific operations
1648         if(idx == ui->tabWidget->indexOf(ui->tabOptions) && ui->scrollArea->widget() && this->isVisible())
1649         {
1650                 ui->scrollArea->widget()->updateGeometry();
1651                 ui->scrollArea->viewport()->updateGeometry();
1652                 qApp->processEvents();
1653                 int delta = ui->scrollArea->widget()->width() - ui->scrollArea->viewport()->width();
1654                 if(delta > 0)
1655                 {
1656                         this->resize(qMin(this->width() + delta, maximumWidth), this->height());
1657                 }
1658         }
1659         else if(idx == ui->tabWidget->indexOf(ui->tabSourceFiles))
1660         {
1661                 m_dropNoteLabel->setGeometry(0, 0, ui->sourceFileView->width(), ui->sourceFileView->height());
1662         }
1663         else if(idx == ui->tabWidget->indexOf(ui->tabOutputDir))
1664         {
1665                 if(!m_fileSystemModel)
1666                 {
1667                         QTimer::singleShot(125, this, SLOT(initOutputFolderModel()));
1668                 }
1669                 else
1670                 {
1671                         CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
1672                 }
1673         }
1674
1675         //Center window around previous position
1676         if(initialWidth < this->width())
1677         {
1678                 QPoint prevPos = this->pos();
1679                 int delta = (this->width() - initialWidth) >> 2;
1680                 move(prevPos.x() - delta, prevPos.y());
1681         }
1682 }
1683
1684 /*
1685  * Tab action triggered
1686  */
1687 void MainWindow::tabActionActivated(QAction *action)
1688 {
1689         if(action && action->data().isValid())
1690         {
1691                 bool ok = false;
1692                 int index = action->data().toInt(&ok);
1693                 if(ok)
1694                 {
1695                         ui->tabWidget->setCurrentIndex(index);
1696                 }
1697         }
1698 }
1699
1700 // =========================================================
1701 // View menu slots
1702 // =========================================================
1703
1704 /*
1705  * Style action triggered
1706  */
1707 void MainWindow::styleActionActivated(QAction *action)
1708 {
1709         //Change style setting
1710         if(action && action->data().isValid())
1711         {
1712                 bool ok = false;
1713                 int actionIndex = action->data().toInt(&ok);
1714                 if(ok)
1715                 {
1716                         m_settings->interfaceStyle(actionIndex);
1717                 }
1718         }
1719
1720         //Set up the new style
1721         switch(m_settings->interfaceStyle())
1722         {
1723         case 1:
1724                 if(ui->actionStyleCleanlooks->isEnabled())
1725                 {
1726                         ui->actionStyleCleanlooks->setChecked(true);
1727                         QApplication::setStyle(new QCleanlooksStyle());
1728                         break;
1729                 }
1730         case 2:
1731                 if(ui->actionStyleWindowsVista->isEnabled())
1732                 {
1733                         ui->actionStyleWindowsVista->setChecked(true);
1734                         QApplication::setStyle(new QWindowsVistaStyle());
1735                         break;
1736                 }
1737         case 3:
1738                 if(ui->actionStyleWindowsXP->isEnabled())
1739                 {
1740                         ui->actionStyleWindowsXP->setChecked(true);
1741                         QApplication::setStyle(new QWindowsXPStyle());
1742                         break;
1743                 }
1744         case 4:
1745                 if(ui->actionStyleWindowsClassic->isEnabled())
1746                 {
1747                         ui->actionStyleWindowsClassic->setChecked(true);
1748                         QApplication::setStyle(new QWindowsStyle());
1749                         break;
1750                 }
1751         default:
1752                 ui->actionStylePlastique->setChecked(true);
1753                 QApplication::setStyle(new QPlastiqueStyle());
1754                 break;
1755         }
1756
1757         //Force re-translate after style change
1758         if(QEvent *e = new QEvent(QEvent::LanguageChange))
1759         {
1760                 changeEvent(e);
1761                 LAMEXP_DELETE(e);
1762         }
1763
1764         //Make transparent
1765         const type_info &styleType = typeid(*qApp->style());
1766         const bool bTransparent = ((typeid(QWindowsVistaStyle) == styleType) || (typeid(QWindowsXPStyle) == styleType));
1767         MAKE_TRANSPARENT(ui->scrollArea, bTransparent);
1768 }
1769
1770 /*
1771  * Language action triggered
1772  */
1773 void MainWindow::languageActionActivated(QAction *action)
1774 {
1775         if(action->data().type() == QVariant::String)
1776         {
1777                 QString langId = action->data().toString();
1778
1779                 if(lamexp_install_translator(langId))
1780                 {
1781                         action->setChecked(true);
1782                         ui->actionLoadTranslationFromFile->setChecked(false);
1783                         m_settings->currentLanguage(langId);
1784                         m_settings->currentLanguageFile(QString());
1785                 }
1786         }
1787 }
1788
1789 /*
1790  * Load language from file action triggered
1791  */
1792 void MainWindow::languageFromFileActionActivated(bool checked)
1793 {
1794         QFileDialog dialog(this, tr("Load Translation"));
1795         dialog.setFileMode(QFileDialog::ExistingFile);
1796         dialog.setNameFilter(QString("%1 (*.qm)").arg(tr("Translation Files")));
1797
1798         if(dialog.exec())
1799         {
1800                 QStringList selectedFiles = dialog.selectedFiles();
1801                 const QString qmFile = QFileInfo(selectedFiles.first()).canonicalFilePath();
1802                 if(lamexp_install_translator_from_file(qmFile))
1803                 {
1804                         QList<QAction*> actions = m_languageActionGroup->actions();
1805                         while(!actions.isEmpty())
1806                         {
1807                                 actions.takeFirst()->setChecked(false);
1808                         }
1809                         ui->actionLoadTranslationFromFile->setChecked(true);
1810                         m_settings->currentLanguageFile(qmFile);
1811                 }
1812                 else
1813                 {
1814                         languageActionActivated(m_languageActionGroup->actions().first());
1815                 }
1816         }
1817 }
1818
1819 // =========================================================
1820 // Tools menu slots
1821 // =========================================================
1822
1823 /*
1824  * Disable update reminder action
1825  */
1826 void MainWindow::disableUpdateReminderActionTriggered(bool checked)
1827 {
1828         if(checked)
1829         {
1830                 if(0 == QMessageBox::question(this, tr("Disable Update Reminder"), NOBR(tr("Do you really want to disable the update reminder?")), tr("Yes"), tr("No"), QString(), 1))
1831                 {
1832                         QMessageBox::information(this, tr("Update Reminder"), QString("%1<br>%2").arg(NOBR(tr("The update reminder has been disabled.")), NOBR(tr("Please remember to check for updates at regular intervals!"))));
1833                         m_settings->autoUpdateEnabled(false);
1834                 }
1835                 else
1836                 {
1837                         m_settings->autoUpdateEnabled(true);
1838                 }
1839         }
1840         else
1841         {
1842                         QMessageBox::information(this, tr("Update Reminder"), NOBR(tr("The update reminder has been re-enabled.")));
1843                         m_settings->autoUpdateEnabled(true);
1844         }
1845
1846         ui->actionDisableUpdateReminder->setChecked(!m_settings->autoUpdateEnabled());
1847 }
1848
1849 /*
1850  * Disable sound effects action
1851  */
1852 void MainWindow::disableSoundsActionTriggered(bool checked)
1853 {
1854         if(checked)
1855         {
1856                 if(0 == QMessageBox::question(this, tr("Disable Sound Effects"), NOBR(tr("Do you really want to disable all sound effects?")), tr("Yes"), tr("No"), QString(), 1))
1857                 {
1858                         QMessageBox::information(this, tr("Sound Effects"), NOBR(tr("All sound effects have been disabled.")));
1859                         m_settings->soundsEnabled(false);
1860                 }
1861                 else
1862                 {
1863                         m_settings->soundsEnabled(true);
1864                 }
1865         }
1866         else
1867         {
1868                         QMessageBox::information(this, tr("Sound Effects"), NOBR(tr("The sound effects have been re-enabled.")));
1869                         m_settings->soundsEnabled(true);
1870         }
1871
1872         ui->actionDisableSounds->setChecked(!m_settings->soundsEnabled());
1873 }
1874
1875 /*
1876  * Disable Nero AAC encoder action
1877  */
1878 void MainWindow::disableNeroAacNotificationsActionTriggered(bool checked)
1879 {
1880         if(checked)
1881         {
1882                 if(0 == QMessageBox::question(this, tr("Nero AAC Notifications"), NOBR(tr("Do you really want to disable all Nero AAC Encoder notifications?")), tr("Yes"), tr("No"), QString(), 1))
1883                 {
1884                         QMessageBox::information(this, tr("Nero AAC Notifications"), NOBR(tr("All Nero AAC Encoder notifications have been disabled.")));
1885                         m_settings->neroAacNotificationsEnabled(false);
1886                 }
1887                 else
1888                 {
1889                         m_settings->neroAacNotificationsEnabled(true);
1890                 }
1891         }
1892         else
1893         {
1894                         QMessageBox::information(this, tr("Nero AAC Notifications"), NOBR(tr("The Nero AAC Encoder notifications have been re-enabled.")));
1895                         m_settings->neroAacNotificationsEnabled(true);
1896         }
1897
1898         ui->actionDisableNeroAacNotifications->setChecked(!m_settings->neroAacNotificationsEnabled());
1899 }
1900
1901 /*
1902  * Disable slow startup action
1903  */
1904 void MainWindow::disableSlowStartupNotificationsActionTriggered(bool checked)
1905 {
1906         if(checked)
1907         {
1908                 if(0 == QMessageBox::question(this, tr("Slow Startup Notifications"), NOBR(tr("Do you really want to disable the slow startup notifications?")), tr("Yes"), tr("No"), QString(), 1))
1909                 {
1910                         QMessageBox::information(this, tr("Slow Startup Notifications"), NOBR(tr("The slow startup notifications have been disabled.")));
1911                         m_settings->antivirNotificationsEnabled(false);
1912                 }
1913                 else
1914                 {
1915                         m_settings->antivirNotificationsEnabled(true);
1916                 }
1917         }
1918         else
1919         {
1920                         QMessageBox::information(this, tr("Slow Startup Notifications"), NOBR(tr("The slow startup notifications have been re-enabled.")));
1921                         m_settings->antivirNotificationsEnabled(true);
1922         }
1923
1924         ui->actionDisableSlowStartupNotifications->setChecked(!m_settings->antivirNotificationsEnabled());
1925 }
1926
1927 /*
1928  * Import a Cue Sheet file
1929  */
1930 void MainWindow::importCueSheetActionTriggered(bool checked)
1931 {
1932         ABORT_IF_BUSY;
1933         
1934         TEMP_HIDE_DROPBOX
1935         (
1936                 while(true)
1937                 {
1938                         int result = 0;
1939                         QString selectedCueFile;
1940
1941                         if(lamexp_themes_enabled())
1942                         {
1943                                 selectedCueFile = QFileDialog::getOpenFileName(this, tr("Open Cue Sheet"), m_settings->mostRecentInputPath(), QString("%1 (*.cue)").arg(tr("Cue Sheet File")));
1944                         }
1945                         else
1946                         {
1947                                 QFileDialog dialog(this, tr("Open Cue Sheet"));
1948                                 dialog.setFileMode(QFileDialog::ExistingFile);
1949                                 dialog.setNameFilter(QString("%1 (*.cue)").arg(tr("Cue Sheet File")));
1950                                 dialog.setDirectory(m_settings->mostRecentInputPath());
1951                                 if(dialog.exec())
1952                                 {
1953                                         selectedCueFile = dialog.selectedFiles().first();
1954                                 }
1955                         }
1956
1957                         if(!selectedCueFile.isEmpty())
1958                         {
1959                                 m_settings->mostRecentInputPath(QFileInfo(selectedCueFile).canonicalPath());
1960                                 CueImportDialog *cueImporter  = new CueImportDialog(this, m_fileListModel, selectedCueFile, m_settings);
1961                                 result = cueImporter->exec();
1962                                 LAMEXP_DELETE(cueImporter);
1963                         }
1964
1965                         if(result != (-1)) break;
1966                 }
1967         );
1968 }
1969
1970 /*
1971  * Show the "drop box" widget
1972  */
1973 void MainWindow::showDropBoxWidgetActionTriggered(bool checked)
1974 {
1975         m_settings->dropBoxWidgetEnabled(true);
1976         
1977         if(!m_dropBox->isVisible())
1978         {
1979                 m_dropBox->show();
1980         }
1981         
1982         lamexp_blink_window(m_dropBox);
1983 }
1984
1985 /*
1986  * Check for beta (pre-release) updates
1987  */
1988 void MainWindow::checkForBetaUpdatesActionTriggered(bool checked)
1989 {       
1990         bool checkUpdatesNow = false;
1991         
1992         if(checked)
1993         {
1994                 if(0 == QMessageBox::question(this, tr("Beta Updates"), NOBR(tr("Do you really want LameXP to check for Beta (pre-release) updates?")), tr("Yes"), tr("No"), QString(), 1))
1995                 {
1996                         if(0 == QMessageBox::information(this, tr("Beta Updates"), NOBR(tr("LameXP will check for Beta (pre-release) updates from now on.")), tr("Check Now"), tr("Discard")))
1997                         {
1998                                 checkUpdatesNow = true;
1999                         }
2000                         m_settings->autoUpdateCheckBeta(true);
2001                 }
2002                 else
2003                 {
2004                         m_settings->autoUpdateCheckBeta(false);
2005                 }
2006         }
2007         else
2008         {
2009                         QMessageBox::information(this, tr("Beta Updates"), NOBR(tr("LameXP will <i>not</i> check for Beta (pre-release) updates from now on.")));
2010                         m_settings->autoUpdateCheckBeta(false);
2011         }
2012
2013         ui->actionCheckForBetaUpdates->setChecked(m_settings->autoUpdateCheckBeta());
2014
2015         if(checkUpdatesNow)
2016         {
2017                 if(checkForUpdates())
2018                 {
2019                         QApplication::quit();
2020                 }
2021         }
2022 }
2023
2024 /*
2025  * Hibernate computer action
2026  */
2027 void MainWindow::hibernateComputerActionTriggered(bool checked)
2028 {
2029         if(checked)
2030         {
2031                 if(0 == QMessageBox::question(this, tr("Hibernate Computer"), NOBR(tr("Do you really want the computer to be hibernated on shutdown?")), tr("Yes"), tr("No"), QString(), 1))
2032                 {
2033                         QMessageBox::information(this, tr("Hibernate Computer"), NOBR(tr("LameXP will hibernate the computer on shutdown from now on.")));
2034                         m_settings->hibernateComputer(true);
2035                 }
2036                 else
2037                 {
2038                         m_settings->hibernateComputer(false);
2039                 }
2040         }
2041         else
2042         {
2043                         QMessageBox::information(this, tr("Hibernate Computer"), NOBR(tr("LameXP will <i>not</i> hibernate the computer on shutdown from now on.")));
2044                         m_settings->hibernateComputer(false);
2045         }
2046
2047         ui->actionHibernateComputer->setChecked(m_settings->hibernateComputer());
2048 }
2049
2050 /*
2051  * Disable shell integration action
2052  */
2053 void MainWindow::disableShellIntegrationActionTriggered(bool checked)
2054 {
2055         if(checked)
2056         {
2057                 if(0 == QMessageBox::question(this, tr("Shell Integration"), NOBR(tr("Do you really want to disable the LameXP shell integration?")), tr("Yes"), tr("No"), QString(), 1))
2058                 {
2059                         ShellIntegration::remove();
2060                         QMessageBox::information(this, tr("Shell Integration"), NOBR(tr("The LameXP shell integration has been disabled.")));
2061                         m_settings->shellIntegrationEnabled(false);
2062                 }
2063                 else
2064                 {
2065                         m_settings->shellIntegrationEnabled(true);
2066                 }
2067         }
2068         else
2069         {
2070                         ShellIntegration::install();
2071                         QMessageBox::information(this, tr("Shell Integration"), NOBR(tr("The LameXP shell integration has been re-enabled.")));
2072                         m_settings->shellIntegrationEnabled(true);
2073         }
2074
2075         ui->actionDisableShellIntegration->setChecked(!m_settings->shellIntegrationEnabled());
2076         
2077         if(lamexp_portable_mode() && ui->actionDisableShellIntegration->isChecked())
2078         {
2079                 ui->actionDisableShellIntegration->setEnabled(false);
2080         }
2081 }
2082
2083 // =========================================================
2084 // Help menu slots
2085 // =========================================================
2086
2087 /*
2088  * Visit homepage action
2089  */
2090 void MainWindow::visitHomepageActionActivated(void)
2091 {
2092         if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
2093         {
2094                 if(action->data().isValid() && (action->data().type() == QVariant::String))
2095                 {
2096                         QDesktopServices::openUrl(QUrl(action->data().toString()));
2097                 }
2098         }
2099 }
2100
2101 /*
2102  * Show document
2103  */
2104 void MainWindow::documentActionActivated(void)
2105 {
2106         if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
2107         {
2108                 if(action->data().isValid() && (action->data().type() == QVariant::String))
2109                 {
2110                         QFileInfo document(action->data().toString());
2111                         QFileInfo resource(QString(":/doc/%1.html").arg(document.baseName()));
2112                         if(document.exists() && document.isFile() && (document.size() == resource.size()))
2113                         {
2114                                 QDesktopServices::openUrl(QUrl::fromLocalFile(document.canonicalFilePath()));
2115                         }
2116                         else
2117                         {
2118                                 QFile source(resource.filePath());
2119                                 QFile output(QString("%1/%2.%3.html").arg(lamexp_temp_folder2(), document.baseName(), lamexp_rand_str().left(8)));
2120                                 if(source.open(QIODevice::ReadOnly) && output.open(QIODevice::ReadWrite))
2121                                 {
2122                                         output.write(source.readAll());
2123                                         action->setData(output.fileName());
2124                                         source.close();
2125                                         output.close();
2126                                         QDesktopServices::openUrl(QUrl::fromLocalFile(output.fileName()));
2127                                 }
2128                         }
2129                 }
2130         }
2131 }
2132
2133 /*
2134  * Check for updates action
2135  */
2136 void MainWindow::checkUpdatesActionActivated(void)
2137 {
2138         ABORT_IF_BUSY;
2139         bool bFlag = false;
2140
2141         TEMP_HIDE_DROPBOX
2142         (
2143                 bFlag = checkForUpdates();
2144         );
2145         
2146         if(bFlag)
2147         {
2148                 QApplication::quit();
2149         }
2150 }
2151
2152 // =========================================================
2153 // Source file slots
2154 // =========================================================
2155
2156 /*
2157  * Add file(s) button
2158  */
2159 void MainWindow::addFilesButtonClicked(void)
2160 {
2161         ABORT_IF_BUSY;
2162
2163         TEMP_HIDE_DROPBOX
2164         (
2165                 if(lamexp_themes_enabled())
2166                 {
2167                         QStringList fileTypeFilters = DecoderRegistry::getSupportedTypes();
2168                         QStringList selectedFiles = QFileDialog::getOpenFileNames(this, tr("Add file(s)"), m_settings->mostRecentInputPath(), fileTypeFilters.join(";;"));
2169                         if(!selectedFiles.isEmpty())
2170                         {
2171                                 m_settings->mostRecentInputPath(QFileInfo(selectedFiles.first()).canonicalPath());
2172                                 addFiles(selectedFiles);
2173                         }
2174                 }
2175                 else
2176                 {
2177                         QFileDialog dialog(this, tr("Add file(s)"));
2178                         QStringList fileTypeFilters = DecoderRegistry::getSupportedTypes();
2179                         dialog.setFileMode(QFileDialog::ExistingFiles);
2180                         dialog.setNameFilter(fileTypeFilters.join(";;"));
2181                         dialog.setDirectory(m_settings->mostRecentInputPath());
2182                         if(dialog.exec())
2183                         {
2184                                 QStringList selectedFiles = dialog.selectedFiles();
2185                                 if(!selectedFiles.isEmpty())
2186                                 {
2187                                         m_settings->mostRecentInputPath(QFileInfo(selectedFiles.first()).canonicalPath());
2188                                         addFiles(selectedFiles);
2189                                 }
2190                         }
2191                 }
2192         );
2193 }
2194
2195 /*
2196  * Open folder action
2197  */
2198 void MainWindow::openFolderActionActivated(void)
2199 {
2200         ABORT_IF_BUSY;
2201         QString selectedFolder;
2202         
2203         if(QAction *action = dynamic_cast<QAction*>(QObject::sender()))
2204         {
2205                 TEMP_HIDE_DROPBOX
2206                 (
2207                         if(lamexp_themes_enabled())
2208                         {
2209                                 selectedFolder = QFileDialog::getExistingDirectory(this, tr("Add Folder"), m_settings->mostRecentInputPath());
2210                         }
2211                         else
2212                         {
2213                                 QFileDialog dialog(this, tr("Add Folder"));
2214                                 dialog.setFileMode(QFileDialog::DirectoryOnly);
2215                                 dialog.setDirectory(m_settings->mostRecentInputPath());
2216                                 if(dialog.exec())
2217                                 {
2218                                         selectedFolder = dialog.selectedFiles().first();
2219                                 }
2220                         }
2221                         
2222                         if(!selectedFolder.isEmpty())
2223                         {
2224                                 m_settings->mostRecentInputPath(QDir(selectedFolder).canonicalPath());
2225                                 addFolder(selectedFolder, action->data().toBool());
2226                         }
2227                 );
2228         }
2229 }
2230
2231 /*
2232  * Remove file button
2233  */
2234 void MainWindow::removeFileButtonClicked(void)
2235 {
2236         if(ui->sourceFileView->currentIndex().isValid())
2237         {
2238                 int iRow = ui->sourceFileView->currentIndex().row();
2239                 m_fileListModel->removeFile(ui->sourceFileView->currentIndex());
2240                 ui->sourceFileView->selectRow(iRow < m_fileListModel->rowCount() ? iRow : m_fileListModel->rowCount()-1);
2241         }
2242 }
2243
2244 /*
2245  * Clear files button
2246  */
2247 void MainWindow::clearFilesButtonClicked(void)
2248 {
2249         m_fileListModel->clearFiles();
2250 }
2251
2252 /*
2253  * Move file up button
2254  */
2255 void MainWindow::fileUpButtonClicked(void)
2256 {
2257         if(ui->sourceFileView->currentIndex().isValid())
2258         {
2259                 int iRow = ui->sourceFileView->currentIndex().row() - 1;
2260                 m_fileListModel->moveFile(ui->sourceFileView->currentIndex(), -1);
2261                 ui->sourceFileView->selectRow(iRow >= 0 ? iRow : 0);
2262         }
2263 }
2264
2265 /*
2266  * Move file down button
2267  */
2268 void MainWindow::fileDownButtonClicked(void)
2269 {
2270         if(ui->sourceFileView->currentIndex().isValid())
2271         {
2272                 int iRow = ui->sourceFileView->currentIndex().row() + 1;
2273                 m_fileListModel->moveFile(ui->sourceFileView->currentIndex(), 1);
2274                 ui->sourceFileView->selectRow(iRow < m_fileListModel->rowCount() ? iRow : m_fileListModel->rowCount()-1);
2275         }
2276 }
2277
2278 /*
2279  * Show details button
2280  */
2281 void MainWindow::showDetailsButtonClicked(void)
2282 {
2283         ABORT_IF_BUSY;
2284
2285         int iResult = 0;
2286         MetaInfoDialog *metaInfoDialog = new MetaInfoDialog(this);
2287         QModelIndex index = ui->sourceFileView->currentIndex();
2288
2289         while(index.isValid())
2290         {
2291                 if(iResult > 0)
2292                 {
2293                         index = m_fileListModel->index(index.row() + 1, index.column()); 
2294                         ui->sourceFileView->selectRow(index.row());
2295                 }
2296                 if(iResult < 0)
2297                 {
2298                         index = m_fileListModel->index(index.row() - 1, index.column()); 
2299                         ui->sourceFileView->selectRow(index.row());
2300                 }
2301
2302                 AudioFileModel &file = (*m_fileListModel)[index];
2303                 TEMP_HIDE_DROPBOX
2304                 (
2305                         iResult = metaInfoDialog->exec(file, index.row() > 0, index.row() < m_fileListModel->rowCount() - 1);
2306                 );
2307                 
2308                 if(iResult == INT_MAX)
2309                 {
2310                         m_metaInfoModel->assignInfoFrom(file);
2311                         ui->tabWidget->setCurrentIndex(ui->tabWidget->indexOf(ui->tabMetaData));
2312                         break;
2313                 }
2314
2315                 if(!iResult) break;
2316         }
2317
2318         LAMEXP_DELETE(metaInfoDialog);
2319         QApplication::processEvents(QEventLoop::ExcludeUserInputEvents);
2320         sourceFilesScrollbarMoved(0);
2321 }
2322
2323 /*
2324  * Show context menu for source files
2325  */
2326 void MainWindow::sourceFilesContextMenu(const QPoint &pos)
2327 {
2328         QAbstractScrollArea *scrollArea = dynamic_cast<QAbstractScrollArea*>(QObject::sender());
2329         QWidget *sender = scrollArea ? scrollArea->viewport() : dynamic_cast<QWidget*>(QObject::sender());
2330
2331         if(sender)
2332         {
2333                 if(pos.x() <= sender->width() && pos.y() <= sender->height() && pos.x() >= 0 && pos.y() >= 0)
2334                 {
2335                         m_sourceFilesContextMenu->popup(sender->mapToGlobal(pos));
2336                 }
2337         }
2338 }
2339
2340 /*
2341  * Scrollbar of source files moved
2342  */
2343 void MainWindow::sourceFilesScrollbarMoved(int)
2344 {
2345         ui->sourceFileView->resizeColumnToContents(0);
2346 }
2347
2348 /*
2349  * Open selected file in external player
2350  */
2351 void MainWindow::previewContextActionTriggered(void)
2352 {
2353         QModelIndex index = ui->sourceFileView->currentIndex();
2354         if(!index.isValid())
2355         {
2356                 return;
2357         }
2358
2359         if(!lamexp_open_media_file(m_fileListModel->getFile(index).filePath()))
2360         {
2361                 qDebug("Player not found, falling back to default application...");
2362                 QDesktopServices::openUrl(QString("file:///").append(m_fileListModel->getFile(index).filePath()));
2363         }
2364 }
2365
2366 /*
2367  * Find selected file in explorer
2368  */
2369 void MainWindow::findFileContextActionTriggered(void)
2370 {
2371         QModelIndex index = ui->sourceFileView->currentIndex();
2372         if(index.isValid())
2373         {
2374                 QString systemRootPath;
2375
2376                 QDir systemRoot(lamexp_known_folder(lamexp_folder_systemfolder));
2377                 if(systemRoot.exists() && systemRoot.cdUp())
2378                 {
2379                         systemRootPath = systemRoot.canonicalPath();
2380                 }
2381
2382                 if(!systemRootPath.isEmpty())
2383                 {
2384                         QFileInfo explorer(QString("%1/explorer.exe").arg(systemRootPath));
2385                         if(explorer.exists() && explorer.isFile())
2386                         {
2387                                 QProcess::execute(explorer.canonicalFilePath(), QStringList() << "/select," << QDir::toNativeSeparators(m_fileListModel->getFile(index).filePath()));
2388                                 return;
2389                         }
2390                 }
2391                 else
2392                 {
2393                         qWarning("SystemRoot directory could not be detected!");
2394                 }
2395         }
2396 }
2397
2398 /*
2399  * Add all pending files
2400  */
2401 void MainWindow::handleDelayedFiles(void)
2402 {
2403         m_delayedFileTimer->stop();
2404
2405         if(m_delayedFileList->isEmpty())
2406         {
2407                 return;
2408         }
2409
2410         if(m_banner->isVisible())
2411         {
2412                 m_delayedFileTimer->start(5000);
2413                 return;
2414         }
2415
2416         QStringList selectedFiles;
2417         ui->tabWidget->setCurrentIndex(0);
2418
2419         while(!m_delayedFileList->isEmpty())
2420         {
2421                 QFileInfo currentFile = QFileInfo(m_delayedFileList->takeFirst());
2422                 if(!currentFile.exists() || !currentFile.isFile())
2423                 {
2424                         continue;
2425                 }
2426                 selectedFiles << currentFile.canonicalFilePath();
2427         }
2428         
2429         addFiles(selectedFiles);
2430 }
2431
2432 /*
2433  * Export Meta tags to CSV file
2434  */
2435 void MainWindow::exportCsvContextActionTriggered(void)
2436 {
2437         TEMP_HIDE_DROPBOX
2438         (
2439                 QString selectedCsvFile;
2440         
2441                 if(lamexp_themes_enabled())
2442                 {
2443                         selectedCsvFile = QFileDialog::getSaveFileName(this, tr("Save CSV file"), m_settings->mostRecentInputPath(), QString("%1 (*.csv)").arg(tr("CSV File")));
2444                 }
2445                 else
2446                 {
2447                         QFileDialog dialog(this, tr("Save CSV file"));
2448                         dialog.setFileMode(QFileDialog::AnyFile);
2449                         dialog.setAcceptMode(QFileDialog::AcceptSave);
2450                         dialog.setNameFilter(QString("%1 (*.csv)").arg(tr("CSV File")));
2451                         dialog.setDirectory(m_settings->mostRecentInputPath());
2452                         if(dialog.exec())
2453                         {
2454                                 selectedCsvFile = dialog.selectedFiles().first();
2455                         }
2456                 }
2457
2458                 if(!selectedCsvFile.isEmpty())
2459                 {
2460                         m_settings->mostRecentInputPath(QFileInfo(selectedCsvFile).canonicalPath());
2461                         switch(m_fileListModel->exportToCsv(selectedCsvFile))
2462                         {
2463                         case FileListModel::CsvError_NoTags:
2464                                 QMessageBox::critical(this, tr("CSV Export"), NOBR(tr("Sorry, there are no meta tags that can be exported!")));
2465                                 break;
2466                         case FileListModel::CsvError_FileOpen:
2467                                 QMessageBox::critical(this, tr("CSV Export"), NOBR(tr("Sorry, failed to open CSV file for writing!")));
2468                                 break;
2469                         case FileListModel::CsvError_FileWrite:
2470                                 QMessageBox::critical(this, tr("CSV Export"), NOBR(tr("Sorry, failed to write to the CSV file!")));
2471                                 break;
2472                         case FileListModel::CsvError_OK:
2473                                 QMessageBox::information(this, tr("CSV Export"), NOBR(tr("The CSV files was created successfully!")));
2474                                 break;
2475                         default:
2476                                 qWarning("exportToCsv: Unknown return code!");
2477                         }
2478                 }
2479         );
2480 }
2481
2482
2483 /*
2484  * Import Meta tags from CSV file
2485  */
2486 void MainWindow::importCsvContextActionTriggered(void)
2487 {
2488         TEMP_HIDE_DROPBOX
2489         (
2490                 QString selectedCsvFile;
2491         
2492                 if(lamexp_themes_enabled())
2493                 {
2494                         selectedCsvFile = QFileDialog::getOpenFileName(this, tr("Open CSV file"), m_settings->mostRecentInputPath(), QString("%1 (*.csv)").arg(tr("CSV File")));
2495                 }
2496                 else
2497                 {
2498                         QFileDialog dialog(this, tr("Open CSV file"));
2499                         dialog.setFileMode(QFileDialog::ExistingFile);
2500                         dialog.setNameFilter(QString("%1 (*.csv)").arg(tr("CSV File")));
2501                         dialog.setDirectory(m_settings->mostRecentInputPath());
2502                         if(dialog.exec())
2503                         {
2504                                 selectedCsvFile = dialog.selectedFiles().first();
2505                         }
2506                 }
2507
2508                 if(!selectedCsvFile.isEmpty())
2509                 {
2510                         m_settings->mostRecentInputPath(QFileInfo(selectedCsvFile).canonicalPath());
2511                         switch(m_fileListModel->importFromCsv(this, selectedCsvFile))
2512                         {
2513                         case FileListModel::CsvError_FileOpen:
2514                                 QMessageBox::critical(this, tr("CSV Import"), NOBR(tr("Sorry, failed to open CSV file for reading!")));
2515                                 break;
2516                         case FileListModel::CsvError_FileRead:
2517                                 QMessageBox::critical(this, tr("CSV Import"), NOBR(tr("Sorry, failed to read from the CSV file!")));
2518                                 break;
2519                         case FileListModel::CsvError_NoTags:
2520                                 QMessageBox::critical(this, tr("CSV Import"), NOBR(tr("Sorry, the CSV file does not contain any known fields!")));
2521                                 break;
2522                         case FileListModel::CsvError_Incomplete:
2523                                 QMessageBox::warning(this, tr("CSV Import"), NOBR(tr("CSV file is incomplete. Not all files were updated!")));
2524                                 break;
2525                         case FileListModel::CsvError_OK:
2526                                 QMessageBox::information(this, tr("CSV Import"), NOBR(tr("The CSV files was imported successfully!")));
2527                                 break;
2528                         case FileListModel::CsvError_Aborted:
2529                                 /* User aborted, ignore! */
2530                                 break;
2531                         default:
2532                                 qWarning("exportToCsv: Unknown return code!");
2533                         }
2534                 }
2535         );
2536 }
2537
2538 /*
2539  * Show or hide Drag'n'Drop notice after model reset
2540  */
2541 void MainWindow::sourceModelChanged(void)
2542 {
2543         m_dropNoteLabel->setVisible(m_fileListModel->rowCount() <= 0);
2544 }
2545
2546 // =========================================================
2547 // Output folder slots
2548 // =========================================================
2549
2550 /*
2551  * Output folder changed (mouse clicked)
2552  */
2553 void MainWindow::outputFolderViewClicked(const QModelIndex &index)
2554 {
2555         if(index.isValid() && (ui->outputFolderView->currentIndex() != index))
2556         {
2557                 ui->outputFolderView->setCurrentIndex(index);
2558         }
2559         
2560         if(m_fileSystemModel && index.isValid())
2561         {
2562                 QString selectedDir = m_fileSystemModel->filePath(index);
2563                 if(selectedDir.length() < 3) selectedDir.append(QDir::separator());
2564                 ui->outputFolderLabel->setText(QDir::toNativeSeparators(selectedDir));
2565                 ui->outputFolderLabel->setToolTip(ui->outputFolderLabel->text());
2566                 m_settings->outputDir(selectedDir);
2567         }
2568         else
2569         {
2570                 ui->outputFolderLabel->setText(QDir::toNativeSeparators(m_settings->outputDir()));
2571                 ui->outputFolderLabel->setToolTip(ui->outputFolderLabel->text());
2572         }
2573 }
2574
2575 /*
2576  * Output folder changed (mouse moved)
2577  */
2578 void MainWindow::outputFolderViewMoved(const QModelIndex &index)
2579 {
2580         if(QApplication::mouseButtons() & Qt::LeftButton)
2581         {
2582                 outputFolderViewClicked(index);
2583         }
2584 }
2585
2586 /*
2587  * Goto desktop button
2588  */
2589 void MainWindow::gotoDesktopButtonClicked(void)
2590 {
2591         if(!m_fileSystemModel)
2592         {
2593                 qWarning("File system model not initialized yet!");
2594                 return;
2595         }
2596         
2597         QString desktopPath = QDesktopServices::storageLocation(QDesktopServices::DesktopLocation);
2598         
2599         if(!desktopPath.isEmpty() && QDir(desktopPath).exists())
2600         {
2601                 ui->outputFolderView->setCurrentIndex(m_fileSystemModel->index(desktopPath));
2602                 outputFolderViewClicked(ui->outputFolderView->currentIndex());
2603                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2604         }
2605         else
2606         {
2607                 ui->buttonGotoDesktop->setEnabled(false);
2608         }
2609 }
2610
2611 /*
2612  * Goto home folder button
2613  */
2614 void MainWindow::gotoHomeFolderButtonClicked(void)
2615 {
2616         if(!m_fileSystemModel)
2617         {
2618                 qWarning("File system model not initialized yet!");
2619                 return;
2620         }
2621
2622         QString homePath = QDesktopServices::storageLocation(QDesktopServices::HomeLocation);
2623         
2624         if(!homePath.isEmpty() && QDir(homePath).exists())
2625         {
2626                 ui->outputFolderView->setCurrentIndex(m_fileSystemModel->index(homePath));
2627                 outputFolderViewClicked(ui->outputFolderView->currentIndex());
2628                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2629         }
2630         else
2631         {
2632                 ui->buttonGotoHome->setEnabled(false);
2633         }
2634 }
2635
2636 /*
2637  * Goto music folder button
2638  */
2639 void MainWindow::gotoMusicFolderButtonClicked(void)
2640 {
2641         if(!m_fileSystemModel)
2642         {
2643                 qWarning("File system model not initialized yet!");
2644                 return;
2645         }
2646
2647         QString musicPath = QDesktopServices::storageLocation(QDesktopServices::MusicLocation);
2648         
2649         if(!musicPath.isEmpty() && QDir(musicPath).exists())
2650         {
2651                 ui->outputFolderView->setCurrentIndex(m_fileSystemModel->index(musicPath));
2652                 outputFolderViewClicked(ui->outputFolderView->currentIndex());
2653                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2654         }
2655         else
2656         {
2657                 ui->buttonGotoMusic->setEnabled(false);
2658         }
2659 }
2660
2661 /*
2662  * Goto music favorite output folder
2663  */
2664 void MainWindow::gotoFavoriteFolder(void)
2665 {
2666         if(!m_fileSystemModel)
2667         {
2668                 qWarning("File system model not initialized yet!");
2669                 return;
2670         }
2671
2672         QAction *item = dynamic_cast<QAction*>(QObject::sender());
2673         
2674         if(item)
2675         {
2676                 QDir path(item->data().toString());
2677                 if(path.exists())
2678                 {
2679                         ui->outputFolderView->setCurrentIndex(m_fileSystemModel->index(path.canonicalPath()));
2680                         outputFolderViewClicked(ui->outputFolderView->currentIndex());
2681                         CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2682                 }
2683                 else
2684                 {
2685                         lamexp_beep(lamexp_beep_error);
2686                         m_outputFolderFavoritesMenu->removeAction(item);
2687                         item->deleteLater();
2688                 }
2689         }
2690 }
2691
2692 /*
2693  * Make folder button
2694  */
2695 void MainWindow::makeFolderButtonClicked(void)
2696 {
2697         ABORT_IF_BUSY;
2698
2699         if(!m_fileSystemModel)
2700         {
2701                 qWarning("File system model not initialized yet!");
2702                 return;
2703         }
2704
2705         QDir basePath(m_fileSystemModel->fileInfo(ui->outputFolderView->currentIndex()).absoluteFilePath());
2706         QString suggestedName = tr("New Folder");
2707
2708         if(!m_metaData->artist().isEmpty() && !m_metaData->album().isEmpty())
2709         {
2710                 suggestedName = QString("%1 - %2").arg(m_metaData->artist(),m_metaData->album());
2711         }
2712         else if(!m_metaData->artist().isEmpty())
2713         {
2714                 suggestedName = m_metaData->artist();
2715         }
2716         else if(!m_metaData->album().isEmpty())
2717         {
2718                 suggestedName =m_metaData->album();
2719         }
2720         else
2721         {
2722                 for(int i = 0; i < m_fileListModel->rowCount(); i++)
2723                 {
2724                         const AudioFileModel &audioFile = m_fileListModel->getFile(m_fileListModel->index(i, 0));
2725                         const AudioFileModel_MetaInfo &fileMetaInfo = audioFile.metaInfo();
2726
2727                         if(!fileMetaInfo.album().isEmpty() || !fileMetaInfo.artist().isEmpty())
2728                         {
2729                                 if(!fileMetaInfo.artist().isEmpty() && !fileMetaInfo.album().isEmpty())
2730                                 {
2731                                         suggestedName = QString("%1 - %2").arg(fileMetaInfo.artist(), fileMetaInfo.album());
2732                                 }
2733                                 else if(!fileMetaInfo.artist().isEmpty())
2734                                 {
2735                                         suggestedName = fileMetaInfo.artist();
2736                                 }
2737                                 else if(!fileMetaInfo.album().isEmpty())
2738                                 {
2739                                         suggestedName = fileMetaInfo.album();
2740                                 }
2741                                 break;
2742                         }
2743                 }
2744         }
2745         
2746         suggestedName = lamexp_clean_filename(suggestedName);
2747
2748         while(true)
2749         {
2750                 bool bApplied = false;
2751                 QString folderName = QInputDialog::getText(this, tr("New Folder"), tr("Enter the name of the new folder:").leftJustified(96, ' '), QLineEdit::Normal, suggestedName, &bApplied, Qt::WindowStaysOnTopHint).simplified();
2752
2753                 if(bApplied)
2754                 {
2755                         folderName = lamexp_clean_filepath(folderName.simplified());
2756
2757                         if(folderName.isEmpty())
2758                         {
2759                                 lamexp_beep(lamexp_beep_error);
2760                                 continue;
2761                         }
2762
2763                         int i = 1;
2764                         QString newFolder = folderName;
2765
2766                         while(basePath.exists(newFolder))
2767                         {
2768                                 newFolder = QString(folderName).append(QString().sprintf(" (%d)", ++i));
2769                         }
2770                         
2771                         if(basePath.mkpath(newFolder))
2772                         {
2773                                 QDir createdDir = basePath;
2774                                 if(createdDir.cd(newFolder))
2775                                 {
2776                                         QModelIndex newIndex = m_fileSystemModel->index(createdDir.canonicalPath());
2777                                         ui->outputFolderView->setCurrentIndex(newIndex);
2778                                         outputFolderViewClicked(newIndex);
2779                                         CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2780                                 }
2781                         }
2782                         else
2783                         {
2784                                 QMessageBox::warning(this, tr("Failed to create folder"), QString("%1<br><nobr>%2</nobr><br><br>%3").arg(tr("The new folder could not be created:"), basePath.absoluteFilePath(newFolder), tr("Drive is read-only or insufficient access rights!")));
2785                         }
2786                 }
2787                 break;
2788         }
2789 }
2790
2791 /*
2792  * Output to source dir changed
2793  */
2794 void MainWindow::saveToSourceFolderChanged(void)
2795 {
2796         m_settings->outputToSourceDir(ui->saveToSourceFolderCheckBox->isChecked());
2797 }
2798
2799 /*
2800  * Prepend relative source file path to output file name changed
2801  */
2802 void MainWindow::prependRelativePathChanged(void)
2803 {
2804         m_settings->prependRelativeSourcePath(ui->prependRelativePathCheckBox->isChecked());
2805 }
2806
2807 /*
2808  * Show context menu for output folder
2809  */
2810 void MainWindow::outputFolderContextMenu(const QPoint &pos)
2811 {
2812         QAbstractScrollArea *scrollArea = dynamic_cast<QAbstractScrollArea*>(QObject::sender());
2813         QWidget *sender = scrollArea ? scrollArea->viewport() : dynamic_cast<QWidget*>(QObject::sender());      
2814
2815         if(pos.x() <= sender->width() && pos.y() <= sender->height() && pos.x() >= 0 && pos.y() >= 0)
2816         {
2817                 m_outputFolderContextMenu->popup(sender->mapToGlobal(pos));
2818         }
2819 }
2820
2821 /*
2822  * Show selected folder in explorer
2823  */
2824 void MainWindow::showFolderContextActionTriggered(void)
2825 {
2826         if(!m_fileSystemModel)
2827         {
2828                 qWarning("File system model not initialized yet!");
2829                 return;
2830         }
2831
2832         QString path = QDir::toNativeSeparators(m_fileSystemModel->filePath(ui->outputFolderView->currentIndex()));
2833         if(!path.endsWith(QDir::separator())) path.append(QDir::separator());
2834         lamexp_exec_shell(this, path, true);
2835 }
2836
2837 /*
2838  * Refresh the directory outline
2839  */
2840 void MainWindow::refreshFolderContextActionTriggered(void)
2841 {
2842         //force re-initialization
2843         QTimer::singleShot(0, this, SLOT(initOutputFolderModel()));
2844 }
2845
2846 /*
2847  * Go one directory up
2848  */
2849 void MainWindow::goUpFolderContextActionTriggered(void)
2850 {
2851         QModelIndex current = ui->outputFolderView->currentIndex();
2852         if(current.isValid())
2853         {
2854                 QModelIndex parent = current.parent();
2855                 if(parent.isValid())
2856                 {
2857                         
2858                         ui->outputFolderView->setCurrentIndex(parent);
2859                         outputFolderViewClicked(parent);
2860                 }
2861                 else
2862                 {
2863                         lamexp_beep(lamexp_beep_warning);
2864                 }
2865                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2866         }
2867 }
2868
2869 /*
2870  * Add current folder to favorites
2871  */
2872 void MainWindow::addFavoriteFolderActionTriggered(void)
2873 {
2874         QString path = m_fileSystemModel->filePath(ui->outputFolderView->currentIndex());
2875         QStringList favorites = m_settings->favoriteOutputFolders().split("|", QString::SkipEmptyParts);
2876
2877         if(!favorites.contains(path, Qt::CaseInsensitive))
2878         {
2879                 favorites.append(path);
2880                 while(favorites.count() > 6) favorites.removeFirst();
2881         }
2882         else
2883         {
2884                 lamexp_beep(lamexp_beep_warning);
2885         }
2886
2887         m_settings->favoriteOutputFolders(favorites.join("|"));
2888         refreshFavorites();
2889 }
2890
2891 /*
2892  * Output folder edit finished
2893  */
2894 void MainWindow::outputFolderEditFinished(void)
2895 {
2896         if(ui->outputFolderEdit->isHidden())
2897         {
2898                 return; //Not currently in edit mode!
2899         }
2900         
2901         bool ok = false;
2902         
2903         QString text = QDir::fromNativeSeparators(ui->outputFolderEdit->text().trimmed());
2904         while(text.startsWith('"') || text.startsWith('/')) text = text.right(text.length() - 1).trimmed();
2905         while(text.endsWith('"') || text.endsWith('/')) text = text.left(text.length() - 1).trimmed();
2906
2907         static const char *str = "?*<>|\"";
2908         for(size_t i = 0; str[i]; i++) text.replace(str[i], "_");
2909
2910         if(!((text.length() >= 2) && text.at(0).isLetter() && text.at(1) == QChar(':')))
2911         {
2912                 text = QString("%1/%2").arg(QDir::fromNativeSeparators(ui->outputFolderLabel->text()), text);
2913         }
2914
2915         if(text.length() == 2) text += "/"; /* "X:" => "X:/" */
2916
2917         while(text.length() > 2)
2918         {
2919                 QFileInfo info(text);
2920                 if(info.exists() && info.isDir())
2921                 {
2922                         QModelIndex index = m_fileSystemModel->index(QFileInfo(info.canonicalFilePath()).absoluteFilePath());
2923                         if(index.isValid())
2924                         {
2925                                 ok = true;
2926                                 ui->outputFolderView->setCurrentIndex(index);
2927                                 outputFolderViewClicked(index);
2928                                 break;
2929                         }
2930                 }
2931                 else if(info.exists() && info.isFile())
2932                 {
2933                         QModelIndex index = m_fileSystemModel->index(QFileInfo(info.canonicalPath()).absoluteFilePath());
2934                         if(index.isValid())
2935                         {
2936                                 ok = true;
2937                                 ui->outputFolderView->setCurrentIndex(index);
2938                                 outputFolderViewClicked(index);
2939                                 break;
2940                         }
2941                 }
2942
2943                 text = text.left(text.length() - 1).trimmed();
2944         }
2945
2946         ui->outputFolderEdit->setVisible(false);
2947         ui->outputFolderLabel->setVisible(true);
2948         ui->outputFolderView->setEnabled(true);
2949
2950         if(!ok) lamexp_beep(lamexp_beep_error);
2951         CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2952 }
2953
2954 /*
2955  * Initialize file system model
2956  */
2957 void MainWindow::initOutputFolderModel(void)
2958 {
2959         if(m_outputFolderNoteBox->isHidden())
2960         {
2961                 m_outputFolderNoteBox->show();
2962                 m_outputFolderNoteBox->repaint();
2963                 m_outputFolderViewInitCounter = 4;
2964
2965                 if(m_fileSystemModel)
2966                 {
2967                         SET_MODEL(ui->outputFolderView, NULL);
2968                         LAMEXP_DELETE(m_fileSystemModel);
2969                         ui->outputFolderView->repaint();
2970                 }
2971
2972                 if(m_fileSystemModel = new QFileSystemModelEx())
2973                 {
2974                         m_fileSystemModel->installEventFilter(this);
2975                         connect(m_fileSystemModel, SIGNAL(directoryLoaded(QString)), this, SLOT(outputFolderDirectoryLoaded(QString)));
2976                         connect(m_fileSystemModel, SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(outputFolderRowsInserted(QModelIndex,int,int)));
2977
2978                         SET_MODEL(ui->outputFolderView, m_fileSystemModel);
2979                         ui->outputFolderView->header()->setStretchLastSection(true);
2980                         ui->outputFolderView->header()->hideSection(1);
2981                         ui->outputFolderView->header()->hideSection(2);
2982                         ui->outputFolderView->header()->hideSection(3);
2983                 
2984                         m_fileSystemModel->setRootPath("");
2985                         QModelIndex index = m_fileSystemModel->index(m_settings->outputDir());
2986                         if(index.isValid()) ui->outputFolderView->setCurrentIndex(index);
2987                         outputFolderViewClicked(ui->outputFolderView->currentIndex());
2988                 }
2989
2990                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
2991                 QTimer::singleShot(125, this, SLOT(initOutputFolderModel_doAsync()));
2992         }
2993 }
2994
2995 /*
2996  * Initialize file system model (do NOT call this one directly!)
2997  */
2998 void MainWindow::initOutputFolderModel_doAsync(void)
2999 {
3000         if(m_outputFolderViewInitCounter > 0)
3001         {
3002                 m_outputFolderViewInitCounter--;
3003                 QTimer::singleShot(125, this, SLOT(initOutputFolderModel_doAsync()));
3004         }
3005         else
3006         {
3007                 QTimer::singleShot(125, m_outputFolderNoteBox, SLOT(hide()));
3008                 ui->outputFolderView->setFocus();
3009         }
3010 }
3011
3012 /*
3013  * Center current folder in view
3014  */
3015 void MainWindow::centerOutputFolderModel(void)
3016 {
3017         if(ui->outputFolderView->isVisible())
3018         {
3019                 centerOutputFolderModel_doAsync();
3020                 QTimer::singleShot(125, this, SLOT(centerOutputFolderModel_doAsync()));
3021         }
3022 }
3023
3024 /*
3025  * Center current folder in view (do NOT call this one directly!)
3026  */
3027 void MainWindow::centerOutputFolderModel_doAsync(void)
3028 {
3029         if(ui->outputFolderView->isVisible())
3030         {
3031                 m_outputFolderViewCentering = true;
3032                 const QModelIndex index = ui->outputFolderView->currentIndex();
3033                 ui->outputFolderView->scrollTo(index, QAbstractItemView::PositionAtCenter);
3034                 ui->outputFolderView->setFocus();
3035         }
3036 }
3037
3038 /*
3039  * File system model asynchronously loaded a dir
3040  */
3041 void MainWindow::outputFolderDirectoryLoaded(const QString &path)
3042 {
3043         if(m_outputFolderViewCentering)
3044         {
3045                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
3046         }
3047 }
3048
3049 /*
3050  * File system model inserted new items
3051  */
3052 void MainWindow::outputFolderRowsInserted(const QModelIndex &parent, int start, int end)
3053 {
3054         if(m_outputFolderViewCentering)
3055         {
3056                 CENTER_CURRENT_OUTPUT_FOLDER_DELAYED;
3057         }
3058 }
3059
3060 /*
3061  * Directory view item was expanded by user
3062  */
3063 void MainWindow::outputFolderItemExpanded(const QModelIndex &item)
3064 {
3065         //We need to stop centering as soon as the user has expanded an item manually!
3066         m_outputFolderViewCentering = false;
3067 }
3068
3069 /*
3070  * View event for output folder control occurred
3071  */
3072 void MainWindow::outputFolderViewEventOccurred(QWidget *sender, QEvent *event)
3073 {
3074         switch(event->type())
3075         {
3076         case QEvent::Enter:
3077         case QEvent::Leave:
3078         case QEvent::KeyPress:
3079         case QEvent::KeyRelease:
3080         case QEvent::FocusIn:
3081         case QEvent::FocusOut:
3082         case QEvent::TouchEnd:
3083                 outputFolderViewClicked(ui->outputFolderView->currentIndex());
3084                 break;
3085         }
3086 }
3087
3088 /*
3089  * Mouse event for output folder control occurred
3090  */
3091 void MainWindow::outputFolderMouseEventOccurred(QWidget *sender, QEvent *event)
3092 {
3093         QMouseEvent *mouseEvent = dynamic_cast<QMouseEvent*>(event);
3094         QPoint pos = (mouseEvent) ? mouseEvent->pos() : QPoint();
3095
3096         if(sender == ui->outputFolderLabel)
3097         {
3098                 switch(event->type())
3099                 {
3100                 case QEvent::MouseButtonPress:
3101                         if(mouseEvent && (mouseEvent->button() == Qt::LeftButton))
3102                         {
3103                                 QString path = ui->outputFolderLabel->text();
3104                                 if(!path.endsWith(QDir::separator())) path.append(QDir::separator());
3105                                 lamexp_exec_shell(this, path, true);
3106                         }
3107                         break;
3108                 case QEvent::Enter:
3109                         ui->outputFolderLabel->setForegroundRole(QPalette::Link);
3110                         break;
3111                 case QEvent::Leave:
3112                         ui->outputFolderLabel->setForegroundRole(QPalette::WindowText);
3113                         break;
3114                 }
3115         }
3116
3117         if((sender == ui->outputFoldersFovoritesLabel) || (sender == ui->outputFoldersEditorLabel) || (sender == ui->outputFoldersGoUpLabel))
3118         {
3119                 const type_info &styleType = typeid(*qApp->style());
3120                 if((typeid(QPlastiqueStyle) == styleType) || (typeid(QWindowsStyle) == styleType))
3121                 {
3122                         switch(event->type())
3123                         {
3124                         case QEvent::Enter:
3125                                 dynamic_cast<QLabel*>(sender)->setFrameShadow(ui->outputFolderView->isEnabled() ? QFrame::Raised : QFrame::Plain);
3126                                 break;
3127                         case QEvent::MouseButtonPress:
3128                                 dynamic_cast<QLabel*>(sender)->setFrameShadow(ui->outputFolderView->isEnabled() ? QFrame::Sunken : QFrame::Plain);
3129                                 break;
3130                         case QEvent::MouseButtonRelease:
3131                                 dynamic_cast<QLabel*>(sender)->setFrameShadow(ui->outputFolderView->isEnabled() ? QFrame::Raised : QFrame::Plain);
3132                                 break;
3133                         case QEvent::Leave:
3134                                 dynamic_cast<QLabel*>(sender)->setFrameShadow(ui->outputFolderView->isEnabled() ? QFrame::Plain : QFrame::Plain);
3135                                 break;
3136                         }
3137                 }
3138                 else
3139                 {
3140                         dynamic_cast<QLabel*>(sender)->setFrameShadow(QFrame::Plain);
3141                 }
3142
3143                 if((event->type() == QEvent::MouseButtonRelease) && ui->outputFolderView->isEnabled() && (mouseEvent))
3144                 {
3145                         if(pos.x() <= sender->width() && pos.y() <= sender->height() && pos.x() >= 0 && pos.y() >= 0 && mouseEvent->button() != Qt::MidButton)
3146                         {
3147                                 if(sender == ui->outputFoldersFovoritesLabel)
3148                                 {
3149                                         m_outputFolderFavoritesMenu->popup(sender->mapToGlobal(pos));
3150                                 }
3151                                 else if(sender == ui->outputFoldersEditorLabel)
3152                                 {
3153                                         ui->outputFolderView->setEnabled(false);
3154                                         ui->outputFolderLabel->setVisible(false);
3155                                         ui->outputFolderEdit->setVisible(true);
3156                                         ui->outputFolderEdit->setText(ui->outputFolderLabel->text());
3157                                         ui->outputFolderEdit->selectAll();
3158                                         ui->outputFolderEdit->setFocus();
3159                                 }
3160                                 else if(sender == ui->outputFoldersGoUpLabel)
3161                                 {
3162                                         QTimer::singleShot(0, this, SLOT(goUpFolderContextActionTriggered()));
3163                                 }
3164                                 else
3165                                 {
3166                                         THROW("Oups, this is not supposed to happen!");
3167                                 }
3168                         }
3169                 }
3170         }
3171 }
3172
3173 // =========================================================
3174 // Metadata tab slots
3175 // =========================================================
3176
3177 /*
3178  * Edit meta button clicked
3179  */
3180 void MainWindow::editMetaButtonClicked(void)
3181 {
3182         ABORT_IF_BUSY;
3183
3184         const QModelIndex index = ui->metaDataView->currentIndex();
3185
3186         if(index.isValid())
3187         {
3188                 m_metaInfoModel->editItem(index, this);
3189         
3190                 if(index.row() == 4)
3191                 {
3192                         m_settings->metaInfoPosition(m_metaData->position());
3193                 }
3194         }
3195 }
3196
3197 /*
3198  * Reset meta button clicked
3199  */
3200 void MainWindow::clearMetaButtonClicked(void)
3201 {
3202         ABORT_IF_BUSY;
3203         m_metaInfoModel->clearData();
3204 }
3205
3206 /*
3207  * Meta tags enabled changed
3208  */
3209 void MainWindow::metaTagsEnabledChanged(void)
3210 {
3211         m_settings->writeMetaTags(ui->writeMetaDataCheckBox->isChecked());
3212 }
3213
3214 /*
3215  * Playlist enabled changed
3216  */
3217 void MainWindow::playlistEnabledChanged(void)
3218 {
3219         m_settings->createPlaylist(ui->generatePlaylistCheckBox->isChecked());
3220 }
3221
3222 // =========================================================
3223 // Compression tab slots
3224 // =========================================================
3225
3226 /*
3227  * Update encoder
3228  */
3229 void MainWindow::updateEncoder(int id)
3230 {
3231         /*qWarning("\nupdateEncoder(%d)", id);*/
3232
3233         m_settings->compressionEncoder(id);
3234         const AbstractEncoderInfo *info = EncoderRegistry::getEncoderInfo(id);
3235
3236         //Update UI controls
3237         ui->radioButtonModeQuality       ->setEnabled(info->isModeSupported(SettingsModel::VBRMode));
3238         ui->radioButtonModeAverageBitrate->setEnabled(info->isModeSupported(SettingsModel::ABRMode));
3239         ui->radioButtonConstBitrate      ->setEnabled(info->isModeSupported(SettingsModel::CBRMode));
3240         
3241         //Initialize checkbox state
3242         if(ui->radioButtonModeQuality->isEnabled())             ui->radioButtonModeQuality->setChecked(true);
3243         else if(ui->radioButtonModeAverageBitrate->isEnabled()) ui->radioButtonModeAverageBitrate->setChecked(true);
3244         else if(ui->radioButtonConstBitrate->isEnabled())       ui->radioButtonConstBitrate->setChecked(true);
3245         else THROW("It appears that the encoder does not support *any* RC mode!");
3246
3247         //Apply current RC mode
3248         const int currentRCMode = EncoderRegistry::loadEncoderMode(m_settings, id);
3249         switch(currentRCMode)
3250         {
3251                 case SettingsModel::VBRMode: if(ui->radioButtonModeQuality->isEnabled())        ui->radioButtonModeQuality->setChecked(true);        break;
3252                 case SettingsModel::ABRMode: if(ui->radioButtonModeAverageBitrate->isEnabled()) ui->radioButtonModeAverageBitrate->setChecked(true); break;
3253                 case SettingsModel::CBRMode: if(ui->radioButtonConstBitrate->isEnabled())       ui->radioButtonConstBitrate->setChecked(true);       break;
3254                 default: THROW("updateEncoder(): Unknown rc-mode encountered!");
3255         }
3256
3257         //Display encoder description
3258         if(const char* description = info->description())
3259         {
3260                 ui->labelEncoderInfo->setVisible(true);
3261                 ui->labelEncoderInfo->setText(tr("Current Encoder: %1").arg(QString::fromUtf8(description)));
3262         }
3263         else
3264         {
3265                 ui->labelEncoderInfo->setVisible(false);
3266         }
3267
3268         //Update RC mode!
3269         updateRCMode(m_modeButtonGroup->checkedId());
3270 }
3271
3272 /*
3273  * Update rate-control mode
3274  */
3275 void MainWindow::updateRCMode(int id)
3276 {
3277         /*qWarning("updateRCMode(%d)", id);*/
3278
3279         //Store new RC mode
3280         const int currentEncoder = m_encoderButtonGroup->checkedId();
3281         EncoderRegistry::saveEncoderMode(m_settings, currentEncoder, id);
3282
3283         //Fetch encoder info
3284         const AbstractEncoderInfo *info = EncoderRegistry::getEncoderInfo(currentEncoder);
3285         const int valueCount = info->valueCount(id);
3286
3287         //Sanity check
3288         if(!info->isModeSupported(id))
3289         {
3290                 qWarning("Attempting to use an unsupported RC mode (%d) with current encoder (%d)!", id, currentEncoder);
3291                 ui->labelBitrate->setText("(ERROR)");
3292                 return;
3293         }
3294
3295         //Update slider min/max values
3296         if(valueCount > 0)
3297         {
3298                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setEnabled, true);
3299                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setMinimum, 0);
3300                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setMaximum, valueCount-1);
3301         }
3302         else
3303         {
3304                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setEnabled, false);
3305                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setMinimum, 0);
3306                 WITH_BLOCKED_SIGNALS(ui->sliderBitrate, setMaximum, 2);
3307         }
3308
3309         //Now update bitrate/quality value!
3310         if(valueCount > 0)
3311         {
3312                 const int currentValue = EncoderRegistry::loadEncoderValue(m_settings, currentEncoder, id);
3313                 ui->sliderBitrate->setValue(qBound(0, currentValue, valueCount-1));
3314                 updateBitrate(qBound(0, currentValue, valueCount-1));
3315         }
3316         else
3317         {
3318                 ui->sliderBitrate->setValue(1);
3319                 updateBitrate(0);
3320         }
3321 }
3322
3323 /*
3324  * Update bitrate
3325  */
3326 void MainWindow::updateBitrate(int value)
3327 {
3328         /*qWarning("updateBitrate(%d)", value);*/
3329
3330         //Load current encoder and RC mode
3331         const int currentEncoder = m_encoderButtonGroup->checkedId();
3332         const int currentRCMode = m_modeButtonGroup->checkedId();
3333
3334         //Fetch encoder info
3335         const AbstractEncoderInfo *info = EncoderRegistry::getEncoderInfo(currentEncoder);
3336         const int valueCount = info->valueCount(currentRCMode);
3337
3338         //Sanity check
3339         if(!info->isModeSupported(currentRCMode))
3340         {
3341                 qWarning("Attempting to use an unsupported RC mode (%d) with current encoder (%d)!", currentRCMode, currentEncoder);
3342                 ui->labelBitrate->setText("(ERROR)");
3343                 return;
3344         }
3345
3346         //Store new bitrate value
3347         if(valueCount > 0)
3348         {
3349                 EncoderRegistry::saveEncoderValue(m_settings, currentEncoder, currentRCMode, qBound(0, value, valueCount-1));
3350         }
3351
3352         //Update bitrate value
3353         const int displayValue = (valueCount > 0) ? info->valueAt(currentRCMode, qBound(0, value, valueCount-1)) : INT_MAX;
3354         switch(info->valueType(currentRCMode))
3355         {
3356         case AbstractEncoderInfo::TYPE_BITRATE:
3357                 ui->labelBitrate->setText(QString("%1 kbps").arg(QString::number(displayValue)));
3358                 break;
3359         case AbstractEncoderInfo::TYPE_APPROX_BITRATE:
3360                 ui->labelBitrate->setText(QString("&asymp; %1 kbps").arg(QString::number(displayValue)));
3361                 break;
3362         case AbstractEncoderInfo::TYPE_QUALITY_LEVEL_INT:
3363                 ui->labelBitrate->setText(tr("Quality Level %1").arg(QString::number(displayValue)));
3364                 break;
3365         case AbstractEncoderInfo::TYPE_QUALITY_LEVEL_FLT:
3366                 ui->labelBitrate->setText(tr("Quality Level %1").arg(QString().sprintf("%.2f", double(displayValue)/100.0)));
3367                 break;
3368         case AbstractEncoderInfo::TYPE_COMPRESSION_LEVEL:
3369                 ui->labelBitrate->setText(tr("Compression %1").arg(QString::number(displayValue)));
3370                 break;
3371         case AbstractEncoderInfo::TYPE_UNCOMPRESSED:
3372                 ui->labelBitrate->setText(tr("Uncompressed"));
3373                 break;
3374         default:
3375                 THROW("Unknown display value type encountered!");
3376                 break;
3377         }
3378 }
3379
3380 /*
3381  * Event for compression tab occurred
3382  */
3383 void MainWindow::compressionTabEventOccurred(QWidget *sender, QEvent *event)
3384 {
3385         static const QUrl helpUrl("http://lamexp.sourceforge.net/doc/FAQ.html#054010d9");
3386         
3387         if((sender == ui->labelCompressionHelp) && (event->type() == QEvent::MouseButtonPress))
3388         {
3389                 QDesktopServices::openUrl(helpUrl);
3390         }
3391         else if((sender == ui->labelResetEncoders) && (event->type() == QEvent::MouseButtonPress))
3392         {
3393                 if(m_settings->soundsEnabled())
3394                 {
3395                         lamexp_play_sound(IDR_WAVE_BLAST, true);
3396                 }
3397
3398                 EncoderRegistry::resetAllEncoders(m_settings);
3399                 m_settings->compressionEncoder(SettingsModel::MP3Encoder);
3400                 ui->radioButtonEncoderMP3->setChecked(true);
3401                 QTimer::singleShot(0, this, SLOT(updateEncoder()));
3402         }
3403 }
3404
3405 // =========================================================
3406 // Advanced option slots
3407 // =========================================================
3408
3409 /*
3410  * Lame algorithm quality changed
3411  */
3412 void MainWindow::updateLameAlgoQuality(int value)
3413 {
3414         QString text;
3415
3416         switch(value)
3417         {
3418         case 3:
3419                 text = tr("Best Quality (Slow)");
3420                 break;
3421         case 2:
3422                 text = tr("High Quality (Recommended)");
3423                 break;
3424         case 1:
3425                 text = tr("Acceptable Quality (Fast)");
3426                 break;
3427         case 0:
3428                 text = tr("Poor Quality (Very Fast)");
3429                 break;
3430         }
3431
3432         if(!text.isEmpty())
3433         {
3434                 m_settings->lameAlgoQuality(value);
3435                 ui->labelLameAlgoQuality->setText(text);
3436         }
3437
3438         bool warning = (value == 0), notice = (value == 3);
3439         ui->labelLameAlgoQualityWarning->setVisible(warning);
3440         ui->labelLameAlgoQualityWarningIcon->setVisible(warning);
3441         ui->labelLameAlgoQualityNotice->setVisible(notice);
3442         ui->labelLameAlgoQualityNoticeIcon->setVisible(notice);
3443         ui->labelLameAlgoQualitySpacer->setVisible(warning || notice);
3444 }
3445
3446 /*
3447  * Bitrate management endabled/disabled
3448  */
3449 void MainWindow::bitrateManagementEnabledChanged(bool checked)
3450 {
3451         m_settings->bitrateManagementEnabled(checked);
3452 }
3453
3454 /*
3455  * Minimum bitrate has changed
3456  */
3457 void MainWindow::bitrateManagementMinChanged(int value)
3458 {
3459         if(value > ui->spinBoxBitrateManagementMax->value())
3460         {
3461                 ui->spinBoxBitrateManagementMin->setValue(ui->spinBoxBitrateManagementMax->value());
3462                 m_settings->bitrateManagementMinRate(ui->spinBoxBitrateManagementMax->value());
3463         }
3464         else
3465         {
3466                 m_settings->bitrateManagementMinRate(value);
3467         }
3468 }
3469
3470 /*
3471  * Maximum bitrate has changed
3472  */
3473 void MainWindow::bitrateManagementMaxChanged(int value)
3474 {
3475         if(value < ui->spinBoxBitrateManagementMin->value())
3476         {
3477                 ui->spinBoxBitrateManagementMax->setValue(ui->spinBoxBitrateManagementMin->value());
3478                 m_settings->bitrateManagementMaxRate(ui->spinBoxBitrateManagementMin->value());
3479         }
3480         else
3481         {
3482                 m_settings->bitrateManagementMaxRate(value);
3483         }
3484 }
3485
3486 /*
3487  * Channel mode has changed
3488  */
3489 void MainWindow::channelModeChanged(int value)
3490 {
3491         if(value >= 0) m_settings->lameChannelMode(value);
3492 }
3493
3494 /*
3495  * Sampling rate has changed
3496  */
3497 void MainWindow::samplingRateChanged(int value)
3498 {
3499         if(value >= 0) m_settings->samplingRate(value);
3500 }
3501
3502 /*
3503  * Nero AAC 2-Pass mode changed
3504  */
3505 void MainWindow::neroAAC2PassChanged(bool checked)
3506 {
3507         m_settings->neroAACEnable2Pass(checked);
3508 }
3509
3510 /*
3511  * Nero AAC profile mode changed
3512  */
3513 void MainWindow::neroAACProfileChanged(int value)
3514 {
3515         if(value >= 0) m_settings->aacEncProfile(value);
3516 }
3517
3518 /*
3519  * Aften audio coding mode changed
3520  */
3521 void MainWindow::aftenCodingModeChanged(int value)
3522 {
3523         if(value >= 0) m_settings->aftenAudioCodingMode(value);
3524 }
3525
3526 /*
3527  * Aften DRC mode changed
3528  */
3529 void MainWindow::aftenDRCModeChanged(int value)
3530 {
3531         if(value >= 0) m_settings->aftenDynamicRangeCompression(value);
3532 }
3533
3534 /*
3535  * Aften exponent search size changed
3536  */
3537 void MainWindow::aftenSearchSizeChanged(int value)
3538 {
3539         if(value >= 0) m_settings->aftenExponentSearchSize(value);
3540 }
3541
3542 /*
3543  * Aften fast bit allocation changed
3544  */
3545 void MainWindow::aftenFastAllocationChanged(bool checked)
3546 {
3547         m_settings->aftenFastBitAllocation(checked);
3548 }
3549
3550
3551 /*
3552  * Opus encoder settings changed
3553  */
3554 void MainWindow::opusSettingsChanged(void)
3555 {
3556         m_settings->opusFramesize(ui->comboBoxOpusFramesize->currentIndex());
3557         m_settings->opusComplexity(ui->spinBoxOpusComplexity->value());
3558         m_settings->opusDisableResample(ui->checkBoxOpusDisableResample->isChecked());
3559 }
3560
3561 /*
3562  * Normalization filter enabled changed
3563  */
3564 void MainWindow::normalizationEnabledChanged(bool checked)
3565 {
3566         m_settings->normalizationFilterEnabled(checked);
3567 }
3568
3569 /*
3570  * Normalization max. volume changed
3571  */
3572 void MainWindow::normalizationMaxVolumeChanged(double value)
3573 {
3574         m_settings->normalizationFilterMaxVolume(static_cast<int>(value * 100.0));
3575 }
3576
3577 /*
3578  * Normalization equalization mode changed
3579  */
3580 void MainWindow::normalizationModeChanged(int mode)
3581 {
3582         m_settings->normalizationFilterEQMode(mode);
3583 }
3584
3585 /*
3586  * Tone adjustment has changed (Bass)
3587  */
3588 void MainWindow::toneAdjustBassChanged(double value)
3589 {
3590         m_settings->toneAdjustBass(static_cast<int>(value * 100.0));
3591         ui->spinBoxToneAdjustBass->setPrefix((value > 0) ? "+" : QString());
3592 }
3593
3594 /*
3595  * Tone adjustment has changed (Treble)
3596  */
3597 void MainWindow::toneAdjustTrebleChanged(double value)
3598 {
3599         m_settings->toneAdjustTreble(static_cast<int>(value * 100.0));
3600         ui->spinBoxToneAdjustTreble->setPrefix((value > 0) ? "+" : QString());
3601 }
3602
3603 /*
3604  * Tone adjustment has been reset
3605  */
3606 void MainWindow::toneAdjustTrebleReset(void)
3607 {
3608         ui->spinBoxToneAdjustBass->setValue(m_settings->toneAdjustBassDefault());
3609         ui->spinBoxToneAdjustTreble->setValue(m_settings->toneAdjustTrebleDefault());
3610         toneAdjustBassChanged(ui->spinBoxToneAdjustBass->value());
3611         toneAdjustTrebleChanged(ui->spinBoxToneAdjustTreble->value());
3612 }
3613
3614 /*
3615  * Custom encoder parameters changed
3616  */
3617 void MainWindow::customParamsChanged(void)
3618 {
3619         ui->lineEditCustomParamLAME->setText(ui->lineEditCustomParamLAME->text().simplified());
3620         ui->lineEditCustomParamOggEnc->setText(ui->lineEditCustomParamOggEnc->text().simplified());
3621         ui->lineEditCustomParamNeroAAC->setText(ui->lineEditCustomParamNeroAAC->text().simplified());
3622         ui->lineEditCustomParamFLAC->setText(ui->lineEditCustomParamFLAC->text().simplified());
3623         ui->lineEditCustomParamAften->setText(ui->lineEditCustomParamAften->text().simplified());
3624         ui->lineEditCustomParamOpus->setText(ui->lineEditCustomParamOpus->text().simplified());
3625
3626         bool customParamsUsed = false;
3627         if(!ui->lineEditCustomParamLAME->text().isEmpty()) customParamsUsed = true;
3628         if(!ui->lineEditCustomParamOggEnc->text().isEmpty()) customParamsUsed = true;
3629         if(!ui->lineEditCustomParamNeroAAC->text().isEmpty()) customParamsUsed = true;
3630         if(!ui->lineEditCustomParamFLAC->text().isEmpty()) customParamsUsed = true;
3631         if(!ui->lineEditCustomParamAften->text().isEmpty()) customParamsUsed = true;
3632         if(!ui->lineEditCustomParamOpus->text().isEmpty()) customParamsUsed = true;
3633
3634         ui->labelCustomParamsIcon->setVisible(customParamsUsed);
3635         ui->labelCustomParamsText->setVisible(customParamsUsed);
3636         ui->labelCustomParamsSpacer->setVisible(customParamsUsed);
3637
3638         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::MP3Encoder,    ui->lineEditCustomParamLAME->text());
3639         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::VorbisEncoder, ui->lineEditCustomParamOggEnc->text());
3640         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::AACEncoder,    ui->lineEditCustomParamNeroAAC->text());
3641         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::FLACEncoder,   ui->lineEditCustomParamFLAC->text());
3642         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::AC3Encoder,    ui->lineEditCustomParamAften->text());
3643         EncoderRegistry::saveEncoderCustomParams(m_settings, SettingsModel::OpusEncoder,   ui->lineEditCustomParamOpus->text());
3644 }
3645
3646 /*
3647  * Rename output files enabled changed
3648  */
3649 void MainWindow::renameOutputEnabledChanged(bool checked)
3650 {
3651         m_settings->renameOutputFilesEnabled(checked);
3652 }
3653
3654 /*
3655  * Rename output files patterm changed
3656  */
3657 void MainWindow::renameOutputPatternChanged(void)
3658 {
3659         QString temp = ui->lineEditRenamePattern->text().simplified();
3660         ui->lineEditRenamePattern->setText(temp.isEmpty() ? m_settings->renameOutputFilesPatternDefault() : temp);
3661         m_settings->renameOutputFilesPattern(ui->lineEditRenamePattern->text());
3662 }
3663
3664 /*
3665  * Rename output files patterm changed
3666  */
3667 void MainWindow::renameOutputPatternChanged(const QString &text, bool silent)
3668 {
3669         QString pattern(text.simplified());
3670         
3671         pattern.replace("<BaseName>", "The_White_Stripes_-_Fell_In_Love_With_A_Girl", Qt::CaseInsensitive);
3672         pattern.replace("<TrackNo>", "04", Qt::CaseInsensitive);
3673         pattern.replace("<Title>", "Fell In Love With A Girl", Qt::CaseInsensitive);
3674         pattern.replace("<Artist>", "The White Stripes", Qt::CaseInsensitive);
3675         pattern.replace("<Album>", "White Blood Cells", Qt::CaseInsensitive);
3676         pattern.replace("<Year>", "2001", Qt::CaseInsensitive);
3677         pattern.replace("<Comment>", "Encoded by LameXP", Qt::CaseInsensitive);
3678
3679         const QString patternClean = lamexp_clean_filename(pattern);
3680
3681         if(pattern.compare(patternClean))
3682         {
3683                 if(ui->lineEditRenamePattern->palette().color(QPalette::Text) != Qt::red)
3684                 {
3685                         if(!silent) lamexp_beep(lamexp_beep_error);
3686                         SET_TEXT_COLOR(ui->lineEditRenamePattern, Qt::red);
3687                 }
3688         }
3689         else
3690         {
3691                 if(ui->lineEditRenamePattern->palette() != QPalette())
3692                 {
3693                         if(!silent) lamexp_beep(lamexp_beep_info);
3694                         ui->lineEditRenamePattern->setPalette(QPalette());
3695                 }
3696         }
3697
3698         ui->labelRanameExample->setText(patternClean);
3699 }
3700
3701 /*
3702  * Show list of rename macros
3703  */
3704 void MainWindow::showRenameMacros(const QString &text)
3705 {
3706         if(text.compare("reset", Qt::CaseInsensitive) == 0)
3707         {
3708                 ui->lineEditRenamePattern->setText(m_settings->renameOutputFilesPatternDefault());
3709                 return;
3710         }
3711
3712         const QString format = QString("<tr><td><tt>&lt;%1&gt;</tt></td><td>&nbsp;&nbsp;</td><td>%2</td></tr>");
3713
3714         QString message = QString("<table>");
3715         message += QString(format).arg("BaseName", tr("File name without extension"));
3716         message += QString(format).arg("TrackNo", tr("Track number with leading zero"));
3717         message += QString(format).arg("Title", tr("Track title"));
3718         message += QString(format).arg("Artist", tr("Artist name"));
3719         message += QString(format).arg("Album", tr("Album name"));
3720         message += QString(format).arg("Year", tr("Year with (at least) four digits"));
3721         message += QString(format).arg("Comment", tr("Comment"));
3722         message += "</table><br><br>";
3723         message += QString("%1<br>").arg(tr("Characters forbidden in file names:"));
3724         message += "<b><tt>\\ / : * ? &lt; &gt; |<br>";
3725         
3726         QMessageBox::information(this, tr("Rename Macros"), message, tr("Discard"));
3727 }
3728
3729 void MainWindow::forceStereoDownmixEnabledChanged(bool checked)
3730 {
3731         m_settings->forceStereoDownmix(checked);
3732 }
3733
3734 /*
3735  * Maximum number of instances changed
3736  */
3737 void MainWindow::updateMaximumInstances(int value)
3738 {
3739         ui->labelMaxInstances->setText(tr("%n Instance(s)", "", value));
3740         m_settings->maximumInstances(ui->checkBoxAutoDetectInstances->isChecked() ? NULL : value);
3741 }
3742
3743 /*
3744  * Auto-detect number of instances
3745  */
3746 void MainWindow::autoDetectInstancesChanged(bool checked)
3747 {
3748         m_settings->maximumInstances(checked ? NULL : ui->sliderMaxInstances->value());
3749 }
3750
3751 /*
3752  * Browse for custom TEMP folder button clicked
3753  */
3754 void MainWindow::browseCustomTempFolderButtonClicked(void)
3755 {
3756         QString newTempFolder;
3757
3758         if(lamexp_themes_enabled())
3759         {
3760                 newTempFolder = QFileDialog::getExistingDirectory(this, QString(), m_settings->customTempPath());
3761         }
3762         else
3763         {
3764                 QFileDialog dialog(this);
3765                 dialog.setFileMode(QFileDialog::DirectoryOnly);
3766                 dialog.setDirectory(m_settings->customTempPath());
3767                 if(dialog.exec())
3768                 {
3769                         newTempFolder = dialog.selectedFiles().first();
3770                 }
3771         }
3772
3773         if(!newTempFolder.isEmpty())
3774         {
3775                 QFile writeTest(QString("%1/~%2.tmp").arg(newTempFolder, lamexp_rand_str()));
3776                 if(writeTest.open(QIODevice::ReadWrite))
3777                 {
3778                         writeTest.remove();
3779                         ui->lineEditCustomTempFolder->setText(QDir::toNativeSeparators(newTempFolder));
3780                 }
3781                 else
3782                 {
3783                         QMessageBox::warning(this, tr("Access Denied"), tr("Cannot write to the selected directory. Please choose another directory!"));
3784                 }
3785         }
3786 }
3787
3788 /*
3789  * Custom TEMP folder changed
3790  */
3791 void MainWindow::customTempFolderChanged(const QString &text)
3792 {
3793         m_settings->customTempPath(QDir::fromNativeSeparators(text));
3794 }
3795
3796 /*
3797  * Use custom TEMP folder option changed
3798  */
3799 void MainWindow::useCustomTempFolderChanged(bool checked)
3800 {
3801         m_settings->customTempPathEnabled(!checked);
3802 }
3803
3804 /*
3805  * Help for custom parameters was requested
3806  */
3807 void MainWindow::customParamsHelpRequested(QWidget *obj, QEvent *event)
3808 {
3809         if(event->type() != QEvent::MouseButtonRelease)
3810         {
3811                 return;
3812         }
3813
3814         if(QMouseEvent *mouseEvent = dynamic_cast<QMouseEvent*>(event))
3815         {
3816                 QPoint pos = mouseEvent->pos();
3817                 if(!(pos.x() <= obj->width() && pos.y() <= obj->height() && pos.x() >= 0 && pos.y() >= 0 && mouseEvent->button() != Qt::MidButton))
3818                 {
3819                         return;
3820                 }
3821         }
3822
3823         if(obj == ui->helpCustomParamLAME)         showCustomParamsHelpScreen("lame.exe", "--longhelp");
3824         else if(obj == ui->helpCustomParamOggEnc)  showCustomParamsHelpScreen("oggenc2.exe", "--help");
3825         else if(obj == ui->helpCustomParamNeroAAC)
3826         {
3827                 switch(EncoderRegistry::getAacEncoder())
3828                 {
3829                         case SettingsModel::AAC_ENCODER_QAAC: showCustomParamsHelpScreen("qaac.exe", "--help"); break;
3830                         case SettingsModel::AAC_ENCODER_FHG : showCustomParamsHelpScreen("fhgaacenc.exe", ""); break;
3831                         case SettingsModel::AAC_ENCODER_NERO: showCustomParamsHelpScreen("neroAacEnc.exe", "-help"); break;
3832                         default: lamexp_beep(lamexp_beep_error); break;
3833                 }
3834         }
3835         else if(obj == ui->helpCustomParamFLAC)    showCustomParamsHelpScreen("flac.exe", "--help");
3836         else if(obj == ui->helpCustomParamAften)   showCustomParamsHelpScreen("aften.exe", "-h");
3837         else if(obj == ui->helpCustomParamOpus)    showCustomParamsHelpScreen("opusenc.exe", "--help");
3838         else lamexp_beep(lamexp_beep_error);
3839 }
3840
3841 /*
3842  * Show help for custom parameters
3843  */
3844 void MainWindow::showCustomParamsHelpScreen(const QString &toolName, const QString &command)
3845 {
3846         const QString binary = lamexp_lookup_tool(toolName);
3847         if(binary.isEmpty())
3848         {
3849                 lamexp_beep(lamexp_beep_error);
3850                 qWarning("customParamsHelpRequested: Binary could not be found!");
3851                 return;
3852         }
3853
3854         QProcess *process = new QProcess();
3855         process->setProcessChannelMode(QProcess::MergedChannels);
3856         process->setReadChannel(QProcess::StandardOutput);
3857         process->start(binary, command.isEmpty() ? QStringList() : QStringList() << command);
3858         qApp->setOverrideCursor(QCursor(Qt::WaitCursor));
3859
3860         if(process->waitForStarted(15000))
3861         {
3862                 qApp->processEvents();
3863                 process->waitForFinished(15000);
3864         }
3865         
3866         if(process->state() != QProcess::NotRunning)
3867         {
3868                 process->kill();
3869                 process->waitForFinished(-1);
3870         }
3871
3872         qApp->restoreOverrideCursor();
3873         QStringList output; bool spaceFlag = true;
3874
3875         while(process->canReadLine())
3876         {
3877                 QString temp = QString::fromUtf8(process->readLine());
3878                 TRIM_STRING_RIGHT(temp);
3879                 if(temp.isEmpty())
3880                 {
3881                         if(!spaceFlag) { output << temp; spaceFlag = true; }
3882                 }
3883                 else
3884                 {
3885                         output << temp; spaceFlag = false;
3886                 }
3887         }
3888
3889         LAMEXP_DELETE(process);
3890
3891         if(output.count() < 1)
3892         {
3893                 qWarning("Empty output, cannot show help screen!");
3894                 lamexp_beep(lamexp_beep_error);
3895         }
3896
3897         LogViewDialog *dialog = new LogViewDialog(this);
3898         TEMP_HIDE_DROPBOX( dialog->exec(output); );
3899         LAMEXP_DELETE(dialog);
3900 }
3901
3902 void MainWindow::overwriteModeChanged(int id)
3903 {
3904         if((id == SettingsModel::Overwrite_Replaces) && (m_settings->overwriteMode() != SettingsModel::Overwrite_Replaces))
3905         {
3906                 if(QMessageBox::warning(this, tr("Overwrite Mode"), tr("Warning: This mode may overwrite existing files with no way to revert!"), tr("Continue"), tr("Revert"), QString(), 1) != 0)
3907                 {
3908                         ui->radioButtonOverwriteModeKeepBoth->setChecked(m_settings->overwriteMode() == SettingsModel::Overwrite_KeepBoth);
3909                         ui->radioButtonOverwriteModeSkipFile->setChecked(m_settings->overwriteMode() == SettingsModel::Overwrite_SkipFile);
3910                         return;
3911                 }
3912         }
3913
3914         m_settings->overwriteMode(id);
3915 }
3916
3917 /*
3918  * Reset all advanced options to their defaults
3919  */
3920 void MainWindow::resetAdvancedOptionsButtonClicked(void)
3921 {
3922         if(m_settings->soundsEnabled())
3923         {
3924                 lamexp_play_sound(IDR_WAVE_BLAST, true);
3925         }
3926
3927         ui->sliderLameAlgoQuality->setValue(m_settings->lameAlgoQualityDefault());
3928         ui->spinBoxBitrateManagementMin->setValue(m_settings->bitrateManagementMinRateDefault());
3929         ui->spinBoxBitrateManagementMax->setValue(m_settings->bitrateManagementMaxRateDefault());
3930         ui->spinBoxNormalizationFilter->setValue(static_cast<double>(m_settings->normalizationFilterMaxVolumeDefault()) / 100.0);
3931         ui->spinBoxToneAdjustBass->setValue(static_cast<double>(m_settings->toneAdjustBassDefault()) / 100.0);
3932         ui->spinBoxToneAdjustTreble->setValue(static_cast<double>(m_settings->toneAdjustTrebleDefault()) / 100.0);
3933         ui->spinBoxAftenSearchSize->setValue(m_settings->aftenExponentSearchSizeDefault());
3934         ui->spinBoxOpusComplexity->setValue(m_settings->opusComplexityDefault());
3935         ui->comboBoxMP3ChannelMode->setCurrentIndex(m_settings->lameChannelModeDefault());
3936         ui->comboBoxSamplingRate->setCurrentIndex(m_settings->samplingRateDefault());
3937         ui->comboBoxAACProfile->setCurrentIndex(m_settings->aacEncProfileDefault());
3938         ui->comboBoxAftenCodingMode->setCurrentIndex(m_settings->aftenAudioCodingModeDefault());
3939         ui->comboBoxAftenDRCMode->setCurrentIndex(m_settings->aftenDynamicRangeCompressionDefault());
3940         ui->comboBoxNormalizationMode->setCurrentIndex(m_settings->normalizationFilterEQModeDefault());
3941         ui->comboBoxOpusFramesize->setCurrentIndex(m_settings->opusFramesizeDefault());
3942
3943         SET_CHECKBOX_STATE(ui->checkBoxBitrateManagement, m_settings->bitrateManagementEnabledDefault());
3944         SET_CHECKBOX_STATE(ui->checkBoxNeroAAC2PassMode, m_settings->neroAACEnable2PassDefault());
3945         SET_CHECKBOX_STATE(ui->checkBoxNormalizationFilter, m_settings->normalizationFilterEnabledDefault());
3946         SET_CHECKBOX_STATE(ui->checkBoxAutoDetectInstances, (m_settings->maximumInstancesDefault() < 1));
3947         SET_CHECKBOX_STATE(ui->checkBoxUseSystemTempFolder, !m_settings->customTempPathEnabledDefault());
3948         SET_CHECKBOX_STATE(ui->checkBoxAftenFastAllocation, m_settings->aftenFastBitAllocationDefault());
3949         SET_CHECKBOX_STATE(ui->checkBoxRenameOutput, m_settings->renameOutputFilesEnabledDefault());
3950         SET_CHECKBOX_STATE(ui->checkBoxForceStereoDownmix, m_settings->forceStereoDownmixDefault());
3951         SET_CHECKBOX_STATE(ui->checkBoxOpusDisableResample, m_settings->opusDisableResampleDefault());
3952         
3953         ui->lineEditCustomParamLAME   ->setText(m_settings->customParametersLAMEDefault());
3954         ui->lineEditCustomParamOggEnc ->setText(m_settings->customParametersOggEncDefault());
3955         ui->lineEditCustomParamNeroAAC->setText(m_settings->customParametersAacEncDefault());
3956         ui->lineEditCustomParamFLAC   ->setText(m_settings->customParametersFLACDefault());
3957         ui->lineEditCustomParamOpus   ->setText(m_settings->customParametersOpusEncDefault());
3958         ui->lineEditCustomTempFolder  ->setText(QDir::toNativeSeparators(m_settings->customTempPathDefault()));
3959         ui->lineEditRenamePattern     ->setText(m_settings->renameOutputFilesPatternDefault());
3960
3961         if(m_settings->overwriteModeDefault() == SettingsModel::Overwrite_KeepBoth) ui->radioButtonOverwriteModeKeepBoth->click();
3962         if(m_settings->overwriteModeDefault() == SettingsModel::Overwrite_SkipFile) ui->radioButtonOverwriteModeSkipFile->click();
3963         if(m_settings->overwriteModeDefault() == SettingsModel::Overwrite_Replaces) ui->radioButtonOverwriteModeReplaces->click();
3964
3965         customParamsChanged();
3966         ui->scrollArea->verticalScrollBar()->setValue(0);
3967 }
3968
3969 // =========================================================
3970 // Multi-instance handling slots
3971 // =========================================================
3972
3973 /*
3974  * Other instance detected
3975  */
3976 void MainWindow::notifyOtherInstance(void)
3977 {
3978         if(!m_banner->isVisible())
3979         {
3980                 QMessageBox msgBox(QMessageBox::Warning, tr("Already Running"), tr("LameXP is already running, please use the running instance!"), QMessageBox::NoButton, this, Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint | Qt::WindowStaysOnTopHint);
3981                 msgBox.exec();
3982         }
3983 }
3984
3985 /*
3986  * Add file from another instance
3987  */
3988 void MainWindow::addFileDelayed(const QString &filePath, bool tryASAP)
3989 {
3990         if(tryASAP && !m_delayedFileTimer->isActive())
3991         {
3992                 qDebug("Received file: %s", QUTF8(filePath));
3993                 m_delayedFileList->append(filePath);
3994                 QTimer::singleShot(0, this, SLOT(handleDelayedFiles()));
3995         }
3996         
3997         m_delayedFileTimer->stop();
3998         qDebug("Received file: %s", QUTF8(filePath));
3999         m_delayedFileList->append(filePath);
4000         m_delayedFileTimer->start(5000);
4001 }
4002
4003 /*
4004  * Add files from another instance
4005  */
4006 void MainWindow::addFilesDelayed(const QStringList &filePaths, bool tryASAP)
4007 {
4008         if(tryASAP && !m_delayedFileTimer->isActive())
4009         {
4010                 qDebug("Received %d file(s).", filePaths.count());
4011                 m_delayedFileList->append(filePaths);
4012                 QTimer::singleShot(0, this, SLOT(handleDelayedFiles()));
4013         }
4014         else
4015         {
4016                 m_delayedFileTimer->stop();
4017                 qDebug("Received %d file(s).", filePaths.count());
4018                 m_delayedFileList->append(filePaths);
4019                 m_delayedFileTimer->start(5000);
4020         }
4021 }
4022
4023 /*
4024  * Add folder from another instance
4025  */
4026 void MainWindow::addFolderDelayed(const QString &folderPath, bool recursive)
4027 {
4028         if(!m_banner->isVisible())
4029         {
4030                 addFolder(folderPath, recursive, true);
4031         }
4032 }
4033
4034 // =========================================================
4035 // Misc slots
4036 // =========================================================
4037
4038 /*
4039  * Restore the override cursor
4040  */
4041 void MainWindow::restoreCursor(void)
4042 {
4043         QApplication::restoreOverrideCursor();
4044 }