You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdemultimedia/juk/filerenamer.cpp

1048 lines
31 KiB

/***************************************************************************
begin : Thu Oct 28 2004
copyright : (C) 2004 by Michael Pyne
: (c) 2003 Frerich Raabe <raabe@kde.org>
email : michael.pyne@kdemail.net
***************************************************************************/
/***************************************************************************
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
#include <algorithm>
#include <kdebug.h>
#include <kcombobox.h>
#include <kurl.h>
#include <kurlrequester.h>
#include <kiconloader.h>
#include <knuminput.h>
#include <kstandarddirs.h>
#include <kio/netaccess.h>
#include <kconfigbase.h>
#include <kconfig.h>
#include <kglobal.h>
#include <klineedit.h>
#include <klocale.h>
#include <kpushbutton.h>
#include <kapplication.h>
#include <kmessagebox.h>
#include <ksimpleconfig.h>
#include <tqfile.h>
#include <tqhbox.h>
#include <tqvbox.h>
#include <tqscrollview.h>
#include <tqobjectlist.h>
#include <tqtimer.h>
#include <tqregexp.h>
#include <tqcheckbox.h>
#include <tqdir.h>
#include <tqlabel.h>
#include <tqlayout.h>
#include <tqsignalmapper.h>
#include <tqheader.h>
#include "tag.h"
#include "filehandle.h"
#include "filerenamer.h"
#include "exampleoptions.h"
#include "playlistitem.h"
#include "playlist.h"
#include "coverinfo.h"
class ConfirmationDialog : public KDialogBase
{
public:
ConfirmationDialog(const TQMap<TQString, TQString> &files,
TQWidget *parent = 0, const char *name = 0)
: KDialogBase(parent, name, true, i18n("Warning"), Ok | Cancel)
{
TQVBox *vbox = makeVBoxMainWidget();
TQHBox *hbox = new TQHBox(vbox);
TQLabel *l = new TQLabel(hbox);
l->setPixmap(SmallIcon("messagebox_warning", 32));
l = new TQLabel(i18n("You are about to rename the following files. "
"Are you sure you want to continue?"), hbox);
hbox->setStretchFactor(l, 1);
KListView *lv = new KListView(vbox);
lv->addColumn(i18n("Original Name"));
lv->addColumn(i18n("New Name"));
int lvHeight = 0;
TQMap<TQString, TQString>::ConstIterator it = files.begin();
for(; it != files.end(); ++it) {
KListViewItem *i = it.key() != it.data()
? new KListViewItem(lv, it.key(), it.data())
: new KListViewItem(lv, it.key(), i18n("No Change"));
lvHeight += i->height();
}
lvHeight += lv->horizontalScrollBar()->height() + lv->header()->height();
lv->setMinimumHeight(QMIN(lvHeight, 400));
resize(QMIN(width(), 500), QMIN(minimumHeight(), 400));
}
};
//
// Implementation of ConfigCategoryReader
//
ConfigCategoryReader::ConfigCategoryReader() : CategoryReaderInterface(),
m_currentItem(0)
{
KConfigGroup config(KGlobal::config(), "FileRenamer");
TQValueList<int> categoryOrder = config.readIntListEntry("CategoryOrder");
unsigned categoryCount[NumTypes] = { 0 }; // Keep track of each category encountered.
// Set a default:
if(categoryOrder.isEmpty())
categoryOrder << Artist << Album << Title << Track;
TQValueList<int>::ConstIterator catIt = categoryOrder.constBegin();
for(; catIt != categoryOrder.constEnd(); ++catIt)
{
unsigned catCount = categoryCount[*catIt]++;
TagType category = static_cast<TagType>(*catIt);
CategoryID catId(category, catCount);
m_options[catId] = TagRenamerOptions(catId);
m_categoryOrder << catId;
}
m_folderSeparators.resize(m_categoryOrder.count() - 1, false);
TQValueList<int> checkedSeparators = config.readIntListEntry("CheckedDirSeparators");
TQValueList<int>::ConstIterator it = checkedSeparators.constBegin();
for(; it != checkedSeparators.constEnd(); ++it) {
unsigned index = static_cast<unsigned>(*it);
if(index < m_folderSeparators.count())
m_folderSeparators[index] = true;
}
m_musicFolder = config.readPathEntry("MusicFolder", "${HOME}/music");
m_separator = config.readEntry("Separator", " - ");
}
TQString ConfigCategoryReader::categoryValue(TagType type) const
{
if(!m_currentItem)
return TQString::null;
Tag *tag = m_currentItem->file().tag();
switch(type) {
case Track:
return TQString::number(tag->track());
case Year:
return TQString::number(tag->year());
case Title:
return tag->title();
case Artist:
return tag->artist();
case Album:
return tag->album();
case Genre:
return tag->genre();
default:
return TQString::null;
}
}
TQString ConfigCategoryReader::prefix(const CategoryID &category) const
{
return m_options[category].prefix();
}
TQString ConfigCategoryReader::suffix(const CategoryID &category) const
{
return m_options[category].suffix();
}
TagRenamerOptions::EmptyActions ConfigCategoryReader::emptyAction(const CategoryID &category) const
{
return m_options[category].emptyAction();
}
TQString ConfigCategoryReader::emptyText(const CategoryID &category) const
{
return m_options[category].emptyText();
}
TQValueList<CategoryID> ConfigCategoryReader::categoryOrder() const
{
return m_categoryOrder;
}
TQString ConfigCategoryReader::separator() const
{
return m_separator;
}
TQString ConfigCategoryReader::musicFolder() const
{
return m_musicFolder;
}
int ConfigCategoryReader::trackWidth(unsigned categoryNum) const
{
return m_options[CategoryID(Track, categoryNum)].trackWidth();
}
bool ConfigCategoryReader::hasFolderSeparator(unsigned index) const
{
if(index >= m_folderSeparators.count())
return false;
return m_folderSeparators[index];
}
bool ConfigCategoryReader::isDisabled(const CategoryID &category) const
{
return m_options[category].disabled();
}
//
// Implementation of FileRenamerWidget
//
FileRenamerWidget::FileRenamerWidget(TQWidget *parent) :
FileRenamerBase(parent), CategoryReaderInterface(),
m_exampleFromFile(false)
{
TQLabel *temp = new TQLabel(0);
m_exampleText->setPaletteBackgroundColor(temp->paletteBackgroundColor());
delete temp;
layout()->setMargin(0); // We'll be wrapped by KDialogBase
// This must be created before createTagRows() is called.
m_exampleDialog = new ExampleOptionsDialog(this);
createTagRows();
loadConfig();
// Add correct text to combo box.
m_category->clear();
for(unsigned i = StartTag; i < NumTypes; ++i) {
TQString category = TagRenamerOptions::tagTypeText(static_cast<TagType>(i));
m_category->insertItem(category);
}
connect(m_exampleDialog, TQT_SIGNAL(signalShown()), TQT_SLOT(exampleDialogShown()));
connect(m_exampleDialog, TQT_SIGNAL(signalHidden()), TQT_SLOT(exampleDialogHidden()));
connect(m_exampleDialog, TQT_SIGNAL(dataChanged()), TQT_SLOT(dataSelected()));
connect(m_exampleDialog, TQT_SIGNAL(fileChanged(const TQString &)),
this, TQT_SLOT(fileSelected(const TQString &)));
exampleTextChanged();
}
void FileRenamerWidget::loadConfig()
{
TQValueList<int> checkedSeparators;
KConfigGroup config(KGlobal::config(), "FileRenamer");
for(unsigned i = 0; i < m_rows.count(); ++i)
m_rows[i].options = TagRenamerOptions(m_rows[i].category);
checkedSeparators = config.readIntListEntry("CheckedDirSeparators");
TQValueList<int>::ConstIterator it = checkedSeparators.begin();
for(; it != checkedSeparators.end(); ++it) {
unsigned separator = static_cast<unsigned>(*it);
if(separator < m_folderSwitches.count())
m_folderSwitches[separator]->setChecked(true);
}
TQString url = config.readPathEntry("MusicFolder", "${HOME}/music");
m_musicFolder->setURL(url);
m_separator->setCurrentText(config.readEntry("Separator", " - "));
}
void FileRenamerWidget::saveConfig()
{
KConfigGroup config(KGlobal::config(), "FileRenamer");
TQValueList<int> checkedSeparators;
TQValueList<int> categoryOrder;
for(unsigned i = 0; i < m_rows.count(); ++i) {
unsigned rowId = idOfPosition(i); // Write out in GUI order, not m_rows order
m_rows[rowId].options.saveConfig(m_rows[rowId].category.categoryNumber);
categoryOrder += m_rows[rowId].category.category;
}
for(unsigned i = 0; i < m_folderSwitches.count(); ++i)
if(m_folderSwitches[i]->isChecked() == true)
checkedSeparators += i;
config.writeEntry("CheckedDirSeparators", checkedSeparators);
config.writeEntry("CategoryOrder", categoryOrder);
config.writePathEntry("MusicFolder", m_musicFolder->url());
config.writeEntry("Separator", m_separator->currentText());
config.sync();
}
FileRenamerWidget::~FileRenamerWidget()
{
}
unsigned FileRenamerWidget::addRowCategory(TagType category)
{
static TQPixmap up = SmallIcon("up");
static TQPixmap down = SmallIcon("down");
// Find number of categories already of this type.
unsigned categoryCount = 0;
for(unsigned i = 0; i < m_rows.count(); ++i)
if(m_rows[i].category.category == category)
++categoryCount;
Row row;
row.category = CategoryID(category, categoryCount);
row.position = m_rows.count();
unsigned id = row.position;
TQHBox *frame = new TQHBox(m_mainFrame);
frame->setPaletteBackgroundColor(frame->paletteBackgroundColor().dark(110));
row.widget = frame;
frame->setFrameShape(TQFrame::Box);
frame->setLineWidth(1);
frame->setMargin(3);
m_mainFrame->setStretchFactor(frame, 1);
TQVBox *buttons = new TQVBox(frame);
buttons->setFrameStyle(TQFrame::Plain | TQFrame::Box);
buttons->setLineWidth(1);
row.upButton = new KPushButton(buttons);
row.downButton = new KPushButton(buttons);
row.upButton->setPixmap(up);
row.downButton->setPixmap(down);
row.upButton->setFlat(true);
row.downButton->setFlat(true);
upMapper->connect(row.upButton, TQT_SIGNAL(clicked()), TQT_SLOT(map()));
upMapper->setMapping(row.upButton, id);
downMapper->connect(row.downButton, TQT_SIGNAL(clicked()), TQT_SLOT(map()));
downMapper->setMapping(row.downButton, id);
TQString labelText = TQString("<b>%1</b>").arg(TagRenamerOptions::tagTypeText(category));
TQLabel *label = new TQLabel(labelText, frame);
frame->setStretchFactor(label, 1);
label->setAlignment(AlignCenter);
TQVBox *options = new TQVBox(frame);
row.enableButton = new KPushButton(i18n("Remove"), options);
toggleMapper->connect(row.enableButton, TQT_SIGNAL(clicked()), TQT_SLOT(map()));
toggleMapper->setMapping(row.enableButton, id);
row.optionsButton = new KPushButton(i18n("Options"), options);
mapper->connect(row.optionsButton, TQT_SIGNAL(clicked()), TQT_SLOT(map()));
mapper->setMapping(row.optionsButton, id);
row.widget->show();
m_rows.append(row);
// Disable add button if there's too many rows.
if(m_rows.count() == MAX_CATEGORIES)
m_insertCategory->setEnabled(false);
return id;
}
void FileRenamerWidget::moveSignalMappings(unsigned oldId, unsigned newId)
{
mapper->setMapping(m_rows[oldId].optionsButton, newId);
downMapper->setMapping(m_rows[oldId].downButton, newId);
upMapper->setMapping(m_rows[oldId].upButton, newId);
toggleMapper->setMapping(m_rows[oldId].enableButton, newId);
}
bool FileRenamerWidget::removeRow(unsigned id)
{
if(id >= m_rows.count()) {
kdWarning(65432) << "Trying to remove row, but " << id << " is out-of-range.\n";
return false;
}
if(m_rows.count() == 1) {
kdError(65432) << "Can't remove last row of File Renamer.\n";
return false;
}
// Remove widget. Don't delete it since it appears TQSignalMapper may still need it.
m_rows[id].widget->deleteLater();
m_rows[id].widget = 0;
m_rows[id].enableButton = 0;
m_rows[id].upButton = 0;
m_rows[id].optionsButton = 0;
m_rows[id].downButton = 0;
unsigned checkboxPosition = 0; // Remove first checkbox.
// If not the first row, remove the checkbox before it.
if(m_rows[id].position > 0)
checkboxPosition = m_rows[id].position - 1;
// The checkbox is contained within a layout widget, so the layout
// widget is the one the needs to die.
delete m_folderSwitches[checkboxPosition]->parent();
m_folderSwitches.erase(&m_folderSwitches[checkboxPosition]);
// Go through all the rows and if they have the same category and a
// higher categoryNumber, decrement the number. Also update the
// position identifier.
for(unsigned i = 0; i < m_rows.count(); ++i) {
if(i == id)
continue; // Don't mess with ourself.
if((m_rows[id].category.category == m_rows[i].category.category) &&
(m_rows[id].category.categoryNumber < m_rows[i].category.categoryNumber))
{
--m_rows[i].category.categoryNumber;
}
// Items are moving up.
if(m_rows[id].position < m_rows[i].position)
--m_rows[i].position;
}
// Every row after the one we delete will have a different identifier, since
// the identifier is simply its index into m_rows. So we need to re-do the
// signal mappings for the affected rows.
for(unsigned i = id + 1; i < m_rows.count(); ++i)
moveSignalMappings(i, i - 1);
m_rows.erase(&m_rows[id]);
// Make sure we update the buttons of affected rows.
m_rows[idOfPosition(0)].upButton->setEnabled(false);
m_rows[idOfPosition(m_rows.count() - 1)].downButton->setEnabled(false);
// We can insert another row now, make sure GUI is updated to match.
m_insertCategory->setEnabled(true);
TQTimer::singleShot(0, this, TQT_SLOT(exampleTextChanged()));
return true;
}
void FileRenamerWidget::addFolderSeparatorCheckbox()
{
TQWidget *temp = new TQWidget(m_mainFrame);
TQHBoxLayout *l = new TQHBoxLayout(temp);
TQCheckBox *cb = new TQCheckBox(i18n("Insert folder separator"), temp);
m_folderSwitches.append(cb);
l->addWidget(cb, 0, AlignCenter);
cb->setChecked(false);
connect(cb, TQT_SIGNAL(toggled(bool)),
TQT_SLOT(exampleTextChanged()));
temp->show();
}
void FileRenamerWidget::createTagRows()
{
KConfigGroup config(KGlobal::config(), "FileRenamer");
TQValueList<int> categoryOrder = config.readIntListEntry("CategoryOrder");
if(categoryOrder.isEmpty())
categoryOrder << Artist << Album << Artist << Title << Track;
// Setup arrays.
m_rows.reserve(categoryOrder.count());
m_folderSwitches.reserve(categoryOrder.count() - 1);
mapper = new TQSignalMapper(this, "signal mapper");
toggleMapper = new TQSignalMapper(this, "toggle mapper");
upMapper = new TQSignalMapper(this, "up button mapper");
downMapper = new TQSignalMapper(this, "down button mapper");
connect(mapper, TQT_SIGNAL(mapped(int)), TQT_SLOT(showCategoryOption(int)));
connect(toggleMapper, TQT_SIGNAL(mapped(int)), TQT_SLOT(slotRemoveRow(int)));
connect(upMapper, TQT_SIGNAL(mapped(int)), TQT_SLOT(moveItemUp(int)));
connect(downMapper, TQT_SIGNAL(mapped(int)), TQT_SLOT(moveItemDown(int)));
m_mainFrame = new TQVBox(m_mainView->viewport());
m_mainFrame->setMargin(10);
m_mainFrame->setSpacing(5);
m_mainView->addChild(m_mainFrame);
m_mainView->setResizePolicy(TQScrollView::AutoOneFit);
// OK, the deal with the categoryOrder variable is that we need to create
// the rows in the order that they were saved in (the order given by categoryOrder).
// The signal mappers operate according to the row identifier. To find the position of
// a row given the identifier, use m_rows[id].position. To find the id of a given
// position, use idOfPosition(position).
TQValueList<int>::ConstIterator it = categoryOrder.constBegin();
for(; it != categoryOrder.constEnd(); ++it) {
if(*it < StartTag || *it >= NumTypes) {
kdError(65432) << "Invalid category encountered in file renamer configuration.\n";
continue;
}
if(m_rows.count() == MAX_CATEGORIES) {
kdError(65432) << "Maximum number of File Renamer tags reached, bailing.\n";
break;
}
TagType i = static_cast<TagType>(*it);
addRowCategory(i);
// Insert the directory separator checkbox if this isn't the last
// item.
TQValueList<int>::ConstIterator dup(it);
// Check for last item
if(++dup != categoryOrder.constEnd())
addFolderSeparatorCheckbox();
}
m_rows.first().upButton->setEnabled(false);
m_rows.last().downButton->setEnabled(false);
// If we have maximum number of categories already, don't let the user
// add more.
if(m_rows.count() >= MAX_CATEGORIES)
m_insertCategory->setEnabled(false);
}
void FileRenamerWidget::exampleTextChanged()
{
// Just use .mp3 as an example
if(m_exampleFromFile && (m_exampleFile.isEmpty() ||
!FileHandle(m_exampleFile).tag()->isValid()))
{
m_exampleText->setText(i18n("No file selected, or selected file has no tags."));
return;
}
m_exampleText->setText(FileRenamer::fileName(*this) + ".mp3");
}
TQString FileRenamerWidget::fileCategoryValue(TagType category) const
{
FileHandle file(m_exampleFile);
Tag *tag = file.tag();
switch(category) {
case Track:
return TQString::number(tag->track());
case Year:
return TQString::number(tag->year());
case Title:
return tag->title();
case Artist:
return tag->artist();
case Album:
return tag->album();
case Genre:
return tag->genre();
default:
return TQString::null;
}
}
TQString FileRenamerWidget::categoryValue(TagType category) const
{
if(m_exampleFromFile)
return fileCategoryValue(category);
const ExampleOptions *example = m_exampleDialog->widget();
switch (category) {
case Track:
return example->m_exampleTrack->text();
case Year:
return example->m_exampleYear->text();
case Title:
return example->m_exampleTitle->text();
case Artist:
return example->m_exampleArtist->text();
case Album:
return example->m_exampleAlbum->text();
case Genre:
return example->m_exampleGenre->text();
default:
return TQString::null;
}
}
TQValueList<CategoryID> FileRenamerWidget::categoryOrder() const
{
TQValueList<CategoryID> list;
// Iterate in GUI row order.
for(unsigned i = 0; i < m_rows.count(); ++i) {
unsigned rowId = idOfPosition(i);
list += m_rows[rowId].category;
}
return list;
}
bool FileRenamerWidget::hasFolderSeparator(unsigned index) const
{
if(index >= m_folderSwitches.count())
return false;
return m_folderSwitches[index]->isChecked();
}
void FileRenamerWidget::moveItem(unsigned id, MovementDirection direction)
{
TQWidget *l = m_rows[id].widget;
unsigned bottom = m_rows.count() - 1;
unsigned pos = m_rows[id].position;
unsigned newPos = (direction == MoveUp) ? pos - 1 : pos + 1;
// Item we're moving can't go further down after this.
if((pos == (bottom - 1) && direction == MoveDown) ||
(pos == bottom && direction == MoveUp))
{
unsigned idBottomRow = idOfPosition(bottom);
unsigned idAboveBottomRow = idOfPosition(bottom - 1);
m_rows[idBottomRow].downButton->setEnabled(true);
m_rows[idAboveBottomRow].downButton->setEnabled(false);
}
// We're moving the top item, do some button switching.
if((pos == 0 && direction == MoveDown) || (pos == 1 && direction == MoveUp)) {
unsigned idTopItem = idOfPosition(0);
unsigned idBelowTopItem = idOfPosition(1);
m_rows[idTopItem].upButton->setEnabled(true);
m_rows[idBelowTopItem].upButton->setEnabled(false);
}
// This is the item we're swapping with.
unsigned idSwitchWith = idOfPosition(newPos);
TQWidget *w = m_rows[idSwitchWith].widget;
// Update the table of widget rows.
std::swap(m_rows[id].position, m_rows[idSwitchWith].position);
// Move the item two spaces above/below its previous position. It has to
// be 2 spaces because of the checkbox.
TQBoxLayout *layout = dynamic_cast<TQBoxLayout *>(m_mainFrame->layout());
layout->remove(l);
layout->insertWidget(2 * newPos, l);
// Move the top item two spaces in the opposite direction, for a similar
// reason.
layout->remove(w);
layout->insertWidget(2 * pos, w);
layout->invalidate();
TQTimer::singleShot(0, this, TQT_SLOT(exampleTextChanged()));
}
unsigned FileRenamerWidget::idOfPosition(unsigned position) const
{
if(position >= m_rows.count()) {
kdError(65432) << "Search for position " << position << " out-of-range.\n";
return static_cast<unsigned>(-1);
}
for(unsigned i = 0; i < m_rows.count(); ++i)
if(m_rows[i].position == position)
return i;
kdError(65432) << "Unable to find identifier for position " << position << endl;
return static_cast<unsigned>(-1);
}
unsigned FileRenamerWidget::findIdentifier(const CategoryID &category) const
{
for(unsigned index = 0; index < m_rows.count(); ++index)
if(m_rows[index].category == category)
return index;
kdError(65432) << "Unable to find match for category " <<
TagRenamerOptions::tagTypeText(category.category) <<
", number " << category.categoryNumber << endl;
return MAX_CATEGORIES;
}
void FileRenamerWidget::enableAllUpButtons()
{
for(unsigned i = 0; i < m_rows.count(); ++i)
m_rows[i].upButton->setEnabled(true);
}
void FileRenamerWidget::enableAllDownButtons()
{
for(unsigned i = 0; i < m_rows.count(); ++i)
m_rows[i].downButton->setEnabled(true);
}
void FileRenamerWidget::showCategoryOption(int id)
{
TagOptionsDialog *dialog = new TagOptionsDialog(this, m_rows[id].options, m_rows[id].category.categoryNumber);
if(dialog->exec() == TQDialog::Accepted) {
m_rows[id].options = dialog->options();
exampleTextChanged();
}
delete dialog;
}
void FileRenamerWidget::moveItemUp(int id)
{
moveItem(static_cast<unsigned>(id), MoveUp);
}
void FileRenamerWidget::moveItemDown(int id)
{
moveItem(static_cast<unsigned>(id), MoveDown);
}
void FileRenamerWidget::toggleExampleDialog()
{
m_exampleDialog->setShown(!m_exampleDialog->isShown());
}
void FileRenamerWidget::insertCategory()
{
TagType category = TagRenamerOptions::tagFromCategoryText(m_category->currentText());
if(category == Unknown) {
kdError(65432) << "Trying to add unknown category somehow.\n";
return;
}
// We need to enable the down button of the current bottom row since it
// can now move down.
unsigned idBottom = idOfPosition(m_rows.count() - 1);
m_rows[idBottom].downButton->setEnabled(true);
addFolderSeparatorCheckbox();
// Identifier of new row.
unsigned id = addRowCategory(category);
// Set its down button to be disabled.
m_rows[id].downButton->setEnabled(false);
m_mainFrame->layout()->invalidate();
m_mainView->update();
// Now update according to the code in loadConfig().
m_rows[id].options = TagRenamerOptions(m_rows[id].category);
exampleTextChanged();
}
void FileRenamerWidget::exampleDialogShown()
{
m_showExample->setText(i18n("Hide Renamer Test Dialog"));
}
void FileRenamerWidget::exampleDialogHidden()
{
m_showExample->setText(i18n("Show Renamer Test Dialog"));
}
void FileRenamerWidget::fileSelected(const TQString &file)
{
m_exampleFromFile = true;
m_exampleFile = file;
exampleTextChanged();
}
void FileRenamerWidget::dataSelected()
{
m_exampleFromFile = false;
exampleTextChanged();
}
TQString FileRenamerWidget::separator() const
{
return m_separator->currentText();
}
TQString FileRenamerWidget::musicFolder() const
{
return m_musicFolder->url();
}
void FileRenamerWidget::slotRemoveRow(int id)
{
// Remove the given identified row.
if(!removeRow(id))
kdError(65432) << "Unable to remove row " << id << endl;
}
//
// Implementation of FileRenamer
//
FileRenamer::FileRenamer()
{
}
void FileRenamer::rename(PlaylistItem *item)
{
PlaylistItemList list;
list.append(item);
rename(list);
}
void FileRenamer::rename(const PlaylistItemList &items)
{
ConfigCategoryReader reader;
TQStringList errorFiles;
TQMap<TQString, TQString> map;
TQMap<TQString, PlaylistItem *> itemMap;
for(PlaylistItemList::ConstIterator it = items.begin(); it != items.end(); ++it) {
reader.setPlaylistItem(*it);
TQString oldFile = (*it)->file().absFilePath();
TQString extension = (*it)->file().fileInfo().extension(false);
TQString newFile = fileName(reader) + "." + extension;
if(oldFile != newFile) {
map[oldFile] = newFile;
itemMap[oldFile] = *it;
}
}
if(itemMap.isEmpty() || ConfirmationDialog(map).exec() != TQDialog::Accepted)
return;
KApplication::setOverrideCursor(Qt::waitCursor);
for(TQMap<TQString, TQString>::ConstIterator it = map.begin();
it != map.end(); ++it)
{
if(moveFile(it.key(), it.data())) {
itemMap[it.key()]->setFile(it.data());
itemMap[it.key()]->refresh();
setFolderIcon(it.data(), itemMap[it.key()]);
}
else
errorFiles << i18n("%1 to %2").arg(it.key()).arg(it.data());
processEvents();
}
KApplication::restoreOverrideCursor();
if(!errorFiles.isEmpty())
KMessageBox::errorList(0, i18n("The following rename operations failed:\n"), errorFiles);
}
bool FileRenamer::moveFile(const TQString &src, const TQString &dest)
{
kdDebug(65432) << "Moving file " << src << " to " << dest << endl;
if(src == dest)
return false;
// Escape URL.
KURL srcURL = KURL::fromPathOrURL(src);
KURL dstURL = KURL::fromPathOrURL(dest);
// Clean it.
srcURL.cleanPath();
dstURL.cleanPath();
// Make sure it is valid.
if(!srcURL.isValid() || !dstURL.isValid())
return false;
// Get just the directory.
KURL dir = dstURL;
dir.setFileName(TQString::null);
// Create the directory.
if(!KStandardDirs::exists(dir.path()))
if(!KStandardDirs::makeDir(dir.path())) {
kdError() << "Unable to create directory " << dir.path() << endl;
return false;
}
// Move the file.
return KIO::NetAccess::file_move(srcURL, dstURL);
}
void FileRenamer::setFolderIcon(const KURL &dst, const PlaylistItem *item)
{
if(item->file().tag()->album().isEmpty() ||
!item->file().coverInfo()->hasCover())
{
return;
}
KURL dstURL = dst;
dstURL.cleanPath();
// Split path, and go through each path element. If a path element has
// the album information, set its folder icon.
TQStringList elements = TQStringList::split("/", dstURL.directory());
TQString path;
for(TQStringList::ConstIterator it = elements.begin(); it != elements.end(); ++it) {
path.append("/" + (*it));
kdDebug() << "Checking path: " << path << endl;
if((*it).find(item->file().tag()->album()) != -1 &&
!TQFile::exists(path + "/.directory"))
{
// Seems to be a match, let's set the folder icon for the current
// path. First we should write out the file.
TQPixmap thumb = item->file().coverInfo()->pixmap(CoverInfo::Thumbnail);
thumb.save(path + "/.juk-thumbnail.png", "PNG");
KSimpleConfig config(path + "/.directory");
config.setGroup("Desktop Entry");
if(!config.hasKey("Icon")) {
config.writeEntry("Icon", TQString("%1/.juk-thumbnail.png").arg(path));
config.sync();
}
return;
}
}
}
/**
* Returns iterator pointing to the last item enabled in the given list with
* a non-empty value (or is required to be included).
*/
TQValueList<CategoryID>::ConstIterator lastEnabledItem(const TQValueList<CategoryID> &list,
const CategoryReaderInterface &interface)
{
TQValueList<CategoryID>::ConstIterator it = list.constBegin();
TQValueList<CategoryID>::ConstIterator last = list.constEnd();
for(; it != list.constEnd(); ++it) {
if(interface.isRequired(*it) || (!interface.isDisabled(*it) &&
!interface.categoryValue((*it).category).isEmpty()))
{
last = it;
}
}
return last;
}
TQString FileRenamer::fileName(const CategoryReaderInterface &interface)
{
const TQValueList<CategoryID> categoryOrder = interface.categoryOrder();
const TQString separator = interface.separator();
const TQString folder = interface.musicFolder();
TQValueList<CategoryID>::ConstIterator lastEnabled;
unsigned i = 0;
TQStringList list;
TQChar dirSeparator = TQChar(TQDir::separator());
// Use lastEnabled to properly handle folder separators.
lastEnabled = lastEnabledItem(categoryOrder, interface);
bool pastLast = false; // Toggles to true once we've passed lastEnabled.
for(TQValueList<CategoryID>::ConstIterator it = categoryOrder.begin();
it != categoryOrder.end();
++it, ++i)
{
if(it == lastEnabled)
pastLast = true;
if(interface.isDisabled(*it))
continue;
TQString value = interface.value(*it);
// The user can use the folder separator checkbox to add folders, so don't allow
// slashes that slip in to accidentally create new folders. Should we filter this
// back out when showing it in the GUI?
value.replace('/', "%2f");
if(!pastLast && interface.hasFolderSeparator(i))
value.append(dirSeparator);
if(interface.isRequired(*it) || !value.isEmpty())
list.append(value);
}
// Construct a single string representation, handling strings ending in
// '/' specially
TQString result;
for(TQStringList::ConstIterator it = list.constBegin(); it != list.constEnd(); /* Empty */) {
result += *it;
++it; // Manually advance iterator to check for end-of-list.
// Add separator unless at a directory boundary
if(it != list.constEnd() &&
!(*it).startsWith(dirSeparator) && // Check beginning of next item.
!result.endsWith(dirSeparator))
{
result += separator;
}
}
return TQString(folder + dirSeparator + result);
}
#include "filerenamer.moc"
// vim: set et sw=4 ts=8: