tageditor.cpp 21.2 KB
Newer Older
1 2 3 4
/***************************************************************************
                          tageditor.cpp  -  description
                             -------------------
    begin                : Sat Sep 7 2002
5
    copyright            : (C) 2002, 2003 by Scott Wheeler
6
    email                : wheeler@kde.org
7 8 9 10 11 12 13 14 15 16 17
 ***************************************************************************/

/***************************************************************************
 *                                                                         *
 *   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.                                   *
 *                                                                         *
 ***************************************************************************/

18 19 20 21 22
#include "tageditor.h"
#include "collectionlist.h"
#include "playlistitem.h"
#include "tag.h"

23 24 25 26
#include <kcombobox.h>
#include <klineedit.h>
#include <knuminput.h>
#include <keditcl.h>
27
#include <kmessagebox.h>
28
#include <kconfig.h>
29 30
#include <klocale.h>
#include <kdebug.h>
31
#include <kiconloader.h>
32 33

#include <qlabel.h>
34
#include <qcheckbox.h>
35 36
#include <qlayout.h>
#include <qdir.h>
Scott Wheeler's avatar
Scott Wheeler committed
37
#include <qvalidator.h>
38
#include <qtooltip.h>
39

40 41
#include <id3v1genres.h>

Scott Wheeler's avatar
Scott Wheeler committed
42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
class FileNameValidator : public QValidator
{
public:
    FileNameValidator(QObject *parent, const char *name = 0) :
	QValidator(parent, name) {}

    virtual void fixup(QString &s) const
    {
	s.remove('/');
    }

    virtual State validate(QString &s, int &) const
    {
	if(s.find('/' != -1))
	   return Invalid;
	return Acceptable;
    }
};

61 62 63 64
class FileBoxToolTip : public QToolTip
{
public:
    FileBoxToolTip(TagEditor *editor, QWidget *widget) :
65
	QToolTip(widget), m_editor(editor) {}
66 67 68
protected:
    virtual void maybeTip(const QPoint &)
    {
69
	tip(parentWidget()->rect(), m_editor->items().first()->file().absFilePath());
70 71 72 73 74
    }
private:
    TagEditor *m_editor;
};

75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
class FixedHLayout : public QHBoxLayout
{
public:
    FixedHLayout(QWidget *parent, int margin = 0, int spacing = -1, const char *name = 0) :
	QHBoxLayout(parent, margin, spacing, name),
	m_width(-1) {}
    FixedHLayout(QLayout *parentLayout, int spacing = -1, const char *name = 0) :
	QHBoxLayout(parentLayout, spacing, name),
	m_width(-1) {}
    void setWidth(int w = -1)
    {
	m_width = w == -1 ? QHBoxLayout::minimumSize().width() : w;
    }
    virtual QSize minimumSize() const
    {
	QSize s = QHBoxLayout::minimumSize();
	s.setWidth(m_width);
	return s;
    }
private:
    int m_width;
};

98 99 100 101
////////////////////////////////////////////////////////////////////////////////
// public members
////////////////////////////////////////////////////////////////////////////////

102 103 104
TagEditor::TagEditor(QWidget *parent, const char *name) :
    QWidget(parent, name),
    m_currentPlaylist(0)
105 106 107
{
    setupLayout();
    readConfig();
108
    m_dataChanged = false;
109 110 111 112 113 114 115 116 117 118 119
}

TagEditor::~TagEditor()
{
    saveConfig();
}

////////////////////////////////////////////////////////////////////////////////
// public slots
////////////////////////////////////////////////////////////////////////////////

120
void TagEditor::slotSetItems(const PlaylistItemList &list)
121 122
{
    saveChangesPrompt();
123 124 125 126 127 128 129 130 131 132 133 134 135

    if(m_currentPlaylist) {
	disconnect(m_currentPlaylist, SIGNAL(signalAboutToRemove(PlaylistItem *)),
		   this, SLOT(slotItemRemoved(PlaylistItem *)));
    }

    m_currentPlaylist = list.isEmpty() ? 0 : static_cast<Playlist *>(list.first()->listView());

    if(m_currentPlaylist) {
	connect(m_currentPlaylist, SIGNAL(signalAboutToRemove(PlaylistItem *)),
		this, SLOT(slotItemRemoved(PlaylistItem *)));
    }

136
    m_items = list;
137

138
    if(isVisible())
139
	slotRefresh();
140 141
}

142
void TagEditor::slotRefresh()
143
{
144
    // This method takes the list of currently selected m_items and tries to 
145 146
    // figure out how to show that in the tag editor.  The current strategy --
    // the most common case -- is to just process the first item.  Then we
147
    // check after that to see if there are other m_items and adjust accordingly.
148

149
    if(m_items.isEmpty() || !m_items.first()->file().tag()) {
150
	slotClear();
151
	setEnabled(false);
152 153 154
	return;
    }
    
155 156
    setEnabled(true);

157
    PlaylistItem *item = m_items.first();
158

159
    Tag *tag = item->file().tag();
160
	
161
    m_artistNameBox->setEditText(tag->artist());
162
    m_trackNameBox->setText(tag->title());
163
    m_albumNameBox->setEditText(tag->album());
164

165
    m_fileNameBox->setText(item->file().fileInfo().fileName());
166
    new FileBoxToolTip(this, m_fileNameBox);
167
    m_bitrateBox->setText(QString::number(tag->bitrate()));
168 169
    m_lengthBox->setText(tag->lengthString());

170 171
    if(m_genreList.findIndex(tag->genre()) >= 0)
	m_genreBox->setCurrentItem(m_genreList.findIndex(tag->genre()) + 1);
172 173
    else {
	m_genreBox->setCurrentItem(0);
174
	m_genreBox->setEditText(tag->genre());
175
    }
176

177
    m_trackSpin->setValue(tag->track());
178 179 180 181 182 183 184 185 186 187 188 189
    m_yearSpin->setValue(tag->year());
    
    m_commentBox->setText(tag->comment());
    
    // Start at the second item, since we've already processed the first.
    
    PlaylistItemList::Iterator it = m_items.begin();
    ++it;

    // If there is more than one item in the m_items that we're dealing with...
    
    if(it != m_items.end()) {
190 191 192 193

	QValueListIterator<QWidget *> hideIt = m_hideList.begin();
	for(; hideIt != m_hideList.end(); ++hideIt)
	    (*hideIt)->hide();
Scott Wheeler's avatar
Scott Wheeler committed
194 195 196

	BoxMap::Iterator boxIt = m_enableBoxes.begin();
	for(; boxIt != m_enableBoxes.end(); boxIt++) {
197 198
	    (*boxIt)->setChecked(true);
	    (*boxIt)->show();
199 200
	}
	
201 202 203
	// Yep, this is ugly.  Loop through all of the files checking to see
	// if their fields are the same.  If so, by default, enable their 
	// checkbox.
204
	
205
	// Also, if there are more than 50 m_items, don't scan all of them.
206
	
207 208 209 210 211 212 213 214
	if(m_items.count() > 50) {
	    m_enableBoxes[m_artistNameBox]->setChecked(false);
	    m_enableBoxes[m_trackNameBox]->setChecked(false);
	    m_enableBoxes[m_albumNameBox]->setChecked(false);
	    m_enableBoxes[m_genreBox]->setChecked(false);
	    m_enableBoxes[m_trackSpin]->setChecked(false);
	    m_enableBoxes[m_yearSpin]->setChecked(false);
	    m_enableBoxes[m_commentBox]->setChecked(false);
215 216
	}
	else {
217
	    for(; it != m_items.end(); ++it) {
218
		tag = (*it)->file().tag();
219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263

		if(tag) {

		    if(m_artistNameBox->currentText() != tag->artist() &&
		       m_enableBoxes.contains(m_artistNameBox))
		    {
			m_artistNameBox->lineEdit()->clear();
			m_enableBoxes[m_artistNameBox]->setChecked(false);
		    }
		    if(m_trackNameBox->text() != tag->title() &&
		       m_enableBoxes.contains(m_trackNameBox))
		    {
			m_trackNameBox->clear();
			m_enableBoxes[m_trackNameBox]->setChecked(false);
		    }
		    if(m_albumNameBox->currentText() != tag->album() &&
		       m_enableBoxes.contains(m_albumNameBox))
		    {
			m_albumNameBox->lineEdit()->clear();
			m_enableBoxes[m_albumNameBox]->setChecked(false);
		    }
		    if(m_genreBox->currentText() != tag->genre() &&
		       m_enableBoxes.contains(m_genreBox))
		    {
			m_genreBox->lineEdit()->clear();
			m_enableBoxes[m_genreBox]->setChecked(false);
		    }		
		    if(m_trackSpin->value() != tag->track() &&
		       m_enableBoxes.contains(m_trackSpin))
		    {
			m_trackSpin->setValue(0);
			m_enableBoxes[m_trackSpin]->setChecked(false);
		    }		
		    if(m_yearSpin->value() != tag->year() &&
		       m_enableBoxes.contains(m_yearSpin))
		    {
			m_yearSpin->setValue(0);
			m_enableBoxes[m_yearSpin]->setChecked(false);
		    }
		    if(m_commentBox->text() != tag->comment() &&
		       m_enableBoxes.contains(m_commentBox))
		    {
			m_commentBox->clear();
			m_enableBoxes[m_commentBox]->setChecked(false);
		    }
264
		}
265 266
	    }
	}
267
    }
268 269
    else {
	// Clean up in the case that we are only handling one item.
270 271 272 273

	QValueListIterator<QWidget *> showIt = m_hideList.begin();
	for(; showIt != m_hideList.end(); ++showIt)
	    (*showIt)->show();
Scott Wheeler's avatar
Scott Wheeler committed
274 275 276

	BoxMap::iterator boxIt = m_enableBoxes.begin();
	for(; boxIt != m_enableBoxes.end(); boxIt++) {
277 278 279 280 281
	    (*boxIt)->setChecked(true);
	    (*boxIt)->hide();
	}
    }
    m_dataChanged = false;
282 283
}

284
void TagEditor::slotClear()
285
{
286 287 288 289 290 291 292 293 294 295
    m_artistNameBox->lineEdit()->clear();
    m_trackNameBox->clear();
    m_albumNameBox->lineEdit()->clear();
    m_genreBox->setCurrentItem(0);
    m_fileNameBox->clear();
    m_trackSpin->setValue(0);
    m_yearSpin->setValue(0);
    m_lengthBox->clear();
    m_bitrateBox->clear();
    m_commentBox->clear();    
296 297
}

298
void TagEditor::slotUpdateCollection()
299
{
300 301 302 303 304
    CollectionList *list = CollectionList::instance();

    if(!list)
	return;
    
305 306 307 308 309 310 311 312 313
    QStringList artistList = list->uniqueSet(CollectionList::Artists);
    m_artistNameBox->listBox()->clear();
    m_artistNameBox->listBox()->insertStringList(artistList);
    m_artistNameBox->completionObject()->setItems(artistList);

    QStringList albumList = list->uniqueSet(CollectionList::Albums);
    m_albumNameBox->listBox()->clear();
    m_albumNameBox->listBox()->insertStringList(albumList);
    m_albumNameBox->completionObject()->setItems(albumList);
314 315 316 317 318 319
}

////////////////////////////////////////////////////////////////////////////////
// private members
////////////////////////////////////////////////////////////////////////////////

320
void TagEditor::readConfig()
321
{
322 323
    KConfig *config = KGlobal::config();
    { // combo box completion modes
324 325
	KConfigGroupSaver saver(config, "TagEditor");
	if(m_artistNameBox && m_albumNameBox) {
326 327 328
	    readCompletionMode(config, m_artistNameBox, "ArtistNameBoxMode");
	    readCompletionMode(config, m_albumNameBox, "AlbumNameBoxMode");
	    readCompletionMode(config, m_genreBox, "GenreBoxMode");
329 330
        }
    }
331

332
    TagLib::StringList genres = TagLib::ID3v1::genreList();
333

334 335 336
    for(TagLib::StringList::ConstIterator it = genres.begin(); it != genres.end(); ++it)
	m_genreList.append(TStringToQString((*it)));
    m_genreList.sort();
337

338 339 340
    m_genreBox->clear();
    m_genreBox->insertItem(QString::null);
    m_genreBox->insertStringList(m_genreList);
341 342 343 344 345 346 347 348 349
    m_genreBox->completionObject()->setItems(m_genreList);
}

void TagEditor::readCompletionMode(KConfig *config, KComboBox *box, const QString &key)
{
    KGlobalSettings::Completion mode =
	KGlobalSettings::Completion(config->readNumEntry(key, KGlobalSettings::CompletionAuto));

    box->setCompletionMode(mode);
350
}
351

352 353 354 355 356
void TagEditor::saveConfig()
{
    KConfig *config = KGlobal::config();
    { // combo box completion modes
        KConfigGroupSaver saver(config, "TagEditor");
357 358 359
        if(m_artistNameBox && m_albumNameBox) {
	    config->writeEntry("ArtistNameBoxMode", m_artistNameBox->completionMode());
	    config->writeEntry("AlbumNameBoxMode", m_albumNameBox->completionMode());
360
	    config->writeEntry("GenreBoxMode", m_genreBox->completionMode());
361 362
        }
    }
363

364
}
365

366 367
void TagEditor::setupLayout()
{
368
    static const int horizontalSpacing = 12;
369
    static const int verticalSpacing = 2;
370

371
    QHBoxLayout *layout = new QHBoxLayout(this, 6, horizontalSpacing);
372 373 374 375 376 377 378 379 380 381 382 383 384 385

    //////////////////////////////////////////////////////////////////////////////
    // define two columns of the bottem layout
    //////////////////////////////////////////////////////////////////////////////
    QVBoxLayout *leftColumnLayout = new QVBoxLayout(layout, verticalSpacing);
    QVBoxLayout *rightColumnLayout = new QVBoxLayout(layout, verticalSpacing);

    layout->setStretchFactor(leftColumnLayout, 2);
    layout->setStretchFactor(rightColumnLayout, 3);

    //////////////////////////////////////////////////////////////////////////////
    // put stuff in the left column -- all of the field names are class wide
    //////////////////////////////////////////////////////////////////////////////
    { // just for organization
386 387
    	
	m_artistNameBox = new KComboBox(true, this, "artistNameBox");
388
	m_artistNameBox->setCompletionMode(KGlobalSettings::CompletionAuto);
389
	addItem(i18n("&Artist name:"), m_artistNameBox, leftColumnLayout, "personal");
390

391
        m_trackNameBox = new KLineEdit(this, "trackNameBox");
392
	addItem(i18n("&Track name:"), m_trackNameBox, leftColumnLayout, "player_play");
393

394
	m_albumNameBox = new KComboBox(true, this, "albumNameBox");
395
	m_albumNameBox->setCompletionMode(KGlobalSettings::CompletionAuto);
396
	addItem(i18n("Album &name:"), m_albumNameBox, leftColumnLayout, "cdrom_unmount");
397

398
        m_genreBox = new KComboBox(true, this, "genreBox");
399
	addItem(i18n("&Genre:"), m_genreBox, leftColumnLayout, "knotify");
400 401

        // this fills the space at the bottem of the left column
Scott Wheeler's avatar
Scott Wheeler committed
402 403
        leftColumnLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Minimum,
						  QSizePolicy::Expanding));
404 405 406 407 408
    }
    //////////////////////////////////////////////////////////////////////////////
    // put stuff in the right column
    //////////////////////////////////////////////////////////////////////////////
    { // just for organization
409 410
	
	QHBoxLayout *fileNameLayout = new QHBoxLayout(rightColumnLayout,
411
						      horizontalSpacing);
412

413
	m_fileNameBox = new KLineEdit(this, "fileNameBox");
Scott Wheeler's avatar
Scott Wheeler committed
414 415
	m_fileNameBox->setValidator(new FileNameValidator(m_fileNameBox));	

416 417
	QLabel *fileNameIcon = new QLabel(this);
	fileNameIcon->setPixmap(SmallIcon("sound"));
Scott Wheeler's avatar
Scott Wheeler committed
418 419
	QWidget *fileNameLabel = addHidden(new QLabel(m_fileNameBox, i18n("&File name:"), this));

420 421 422 423 424 425
	fileNameLayout->addWidget(addHidden(fileNameIcon));
	fileNameLayout->addWidget(fileNameLabel);
	fileNameLayout->setStretchFactor(fileNameIcon, 0);
	fileNameLayout->setStretchFactor(fileNameLabel, 0);
	fileNameLayout->insertStretch(-1, 1);
	rightColumnLayout->addWidget(addHidden(m_fileNameBox));
426 427

        { // lay out the track row
428 429
            FixedHLayout *trackRowLayout = new FixedHLayout(rightColumnLayout,
							    horizontalSpacing);
430

431
	    m_trackSpin = new KIntSpinBox(0, 255, 1, 0, 10, this, "trackSpin");
432
	    addItem(i18n("T&rack:"), m_trackSpin, trackRowLayout);
433
	    m_trackSpin->installEventFilter(this);
434

435
	    trackRowLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Expanding,
Scott Wheeler's avatar
Scott Wheeler committed
436
						    QSizePolicy::Minimum));
437

438
	    m_yearSpin = new KIntSpinBox(0, 9999, 1, 0, 10, this, "yearSpin");
439
	    addItem(i18n("&Year:"), m_yearSpin, trackRowLayout);
440
	    m_yearSpin->installEventFilter(this);
441

442
	    trackRowLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Expanding,
Scott Wheeler's avatar
Scott Wheeler committed
443
						    QSizePolicy::Minimum));
444

445 446 447 448 449 450 451
	    trackRowLayout->addWidget(addHidden(new QLabel(i18n("Length:"), this)));
	    m_lengthBox = new KLineEdit(this, "lengthBox");
	    // addItem(i18n("Length:"), m_lengthBox, trackRowLayout);
	    m_lengthBox->setMaximumWidth(50);
	    m_lengthBox->setAlignment(Qt::AlignRight);
	    m_lengthBox->setReadOnly(true);
	    trackRowLayout->addWidget(addHidden(m_lengthBox));
452

453
	    trackRowLayout->addItem(new QSpacerItem(0, 0, QSizePolicy::Expanding,
Scott Wheeler's avatar
Scott Wheeler committed
454
						    QSizePolicy::Minimum));
455

456 457 458 459 460 461 462
	    trackRowLayout->addWidget(addHidden(new QLabel(i18n("Bitrate:"), this)));
	    m_bitrateBox = new KLineEdit(this, "bitrateBox");
	    // addItem(i18n("Bitrate:"), m_bitrateBox, trackRowLayout);
	    m_bitrateBox->setMaximumWidth(50);
	    m_bitrateBox->setAlignment(Qt::AlignRight);
	    m_bitrateBox->setReadOnly(true);
	    trackRowLayout->addWidget(addHidden(m_bitrateBox));
463 464

	    trackRowLayout->setWidth();
465 466
        }

467 468
        m_commentBox = new KEdit(this, "commentBox");
	m_commentBox->setTextFormat(Qt::PlainText);
469 470 471
	addItem(i18n("&Comment:"), m_commentBox, rightColumnLayout, "edit");
    	fileNameLabel->setMinimumHeight(m_trackSpin->height());

472 473
    }

Scott Wheeler's avatar
Scott Wheeler committed
474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499
    connect(m_artistNameBox, SIGNAL(textChanged(const QString&)),
	    this, SLOT(slotDataChanged()));

    connect(m_trackNameBox, SIGNAL(textChanged(const QString&)),
	    this, SLOT(slotDataChanged()));

    connect(m_albumNameBox, SIGNAL(textChanged(const QString&)),
	    this, SLOT(slotDataChanged()));

    connect(m_genreBox, SIGNAL(activated(int)),
	    this, SLOT(slotDataChanged()));

    connect(m_genreBox, SIGNAL(textChanged(const QString&)),
	    this, SLOT(slotDataChanged()));

    connect(m_fileNameBox, SIGNAL(textChanged(const QString&)),
	    this, SLOT(slotDataChanged()));

    connect(m_yearSpin, SIGNAL(valueChanged(int)),
	    this, SLOT(slotDataChanged()));

    connect(m_trackSpin, SIGNAL(valueChanged(int)),
	    this, SLOT(slotDataChanged()));

    connect(m_commentBox, SIGNAL(textChanged()),
	    this, SLOT(slotDataChanged()));
500 501 502 503
}

void TagEditor::save(const PlaylistItemList &list)
{
504
    if(!list.isEmpty() && m_dataChanged) {
505 506 507 508 509 510 511
	
	KApplication::setOverrideCursor(Qt::waitCursor);

	// To keep track of the files that don't cooperate...

	QStringList errorFiles;
	
512
	for(PlaylistItemList::ConstIterator it = list.begin(); it != list.end(); ++it) {
513 514
	    PlaylistItem *item = *it;
	    
515 516 517
	    QFileInfo newFile(item->file().fileInfo().dirPath() + QDir::separator() +
			      m_fileNameBox->text());
	    QFileInfo directory(item->file().fileInfo().dirPath());
518 519 520 521 522 523
	    
	    // If (the new file is writable or the new file doesn't exist and
	    // it's directory is writable) and the old file is writable...  
	    // If not we'll append it to errorFiles to tell the user which
	    // files we couldn't write to.
	    
Scott Wheeler's avatar
Scott Wheeler committed
524 525
	    if(item &&
	       item->file().tag() &&
526
	       (newFile.isWritable() || (!newFile.exists() && directory.isWritable())) &&
527
	       item->file().fileInfo().isWritable())
Scott Wheeler's avatar
Scott Wheeler committed
528
	    {
529 530 531 532
		
		// If the file name in the box doesn't match the current file
		// name...
		
533 534 535
		if(list.count() == 1 &&
		   item->file().fileInfo().fileName() != newFile.fileName())
		{
536
		    
537 538
		    // Rename the file if it doesn't exist or the user says
		    // that it's ok.
539
		    
540
		    if(!newFile.exists() ||
Scott Wheeler's avatar
Scott Wheeler committed
541 542 543 544
		       KMessageBox::warningYesNo(
			   this, 
			   i18n("This file already exists.\nDo you want to replace it?"),
			   i18n("File Exists")) == KMessageBox::Yes)
545 546
		    {
			QDir currentDir;
547 548
			currentDir.rename(item->file().absFilePath(), newFile.filePath());
			item->file().setFile(newFile.filePath());
549 550
		    }
		}
551 552 553 554 555 556 557
		
		// A bit more ugliness.  If there are multiple files that are
		// being modified, they each have a "enabled" checkbox that
		// says if that field is to be respected for the multiple 
		// files.  We have to check to see if that is enabled before
		// each field that we write.
		
558
		if(m_enableBoxes[m_artistNameBox]->isOn())
559
		    item->file().tag()->setArtist(m_artistNameBox->currentText());
560
		if(m_enableBoxes[m_trackNameBox]->isOn())
561
		    item->file().tag()->setTitle(m_trackNameBox->text());
562
		if(m_enableBoxes[m_albumNameBox]->isOn())
563
		    item->file().tag()->setAlbum(m_albumNameBox->currentText());
564
		if(m_enableBoxes[m_trackSpin]->isOn())
565
		    item->file().tag()->setTrack(m_trackSpin->value());
566
		if(m_enableBoxes[m_yearSpin]->isOn())
567
		    item->file().tag()->setYear(m_yearSpin->value());
568
		if(m_enableBoxes[m_commentBox]->isOn())
569
		    item->file().tag()->setComment(m_commentBox->text());
570
		
571
		if(m_enableBoxes[m_genreBox]->isOn())
572
		    item->file().tag()->setGenre(m_genreBox->currentText());
573
		
574
		item->file().tag()->save();
575
		
576
		item->refresh();
577
	    }
Scott Wheeler's avatar
Scott Wheeler committed
578
	    else if(item)
579
		errorFiles.append(item->file().absFilePath());
580 581

	    kapp->processEvents();
582 583 584 585 586
	}
	
	if(!errorFiles.isEmpty())
	    KMessageBox::detailedSorry(this,
				       i18n("Could not save to specified file(s)."), 
587
				       i18n("Could Not Write to:\n") + errorFiles.join("\n"));
588
	m_dataChanged = false;
589 590

	KApplication::restoreOverrideCursor();
591 592 593 594 595
    }
}

void TagEditor::saveChangesPrompt()
{
596 597 598 599 600 601
    if(!isVisible() || !m_dataChanged || m_items.isEmpty())
	return;

    QStringList files;

    for(PlaylistItemList::Iterator it = m_items.begin(); it != m_items.end(); it++)
602
	files.append((*it)->file().absFilePath());
603 604 605 606 607 608 609 610 611 612

    if(KMessageBox::questionYesNoList(this,
				      i18n("Do you want to save your changes to:\n"), 
				      files, 
				      i18n("Save Changes"),
				      KStdGuiItem::yes(),
				      KStdGuiItem::no(),
				      "tagEditor_showSaveChangesBox") == KMessageBox::Yes)
    {
	save(m_items);
613 614 615
    }
}

616
void TagEditor::addItem(const QString &text, QWidget *item, QBoxLayout *layout, const QString &iconName)
617 618 619 620
{
    if(!item || !layout)
	return;

621
    QLabel *label = new QLabel(item, text, this);
622 623 624 625 626
    QLabel *iconLabel = new QLabel(item, 0, this);
    
    if(!iconName.isNull())
	iconLabel->setPixmap(SmallIcon(iconName));
    
627 628 629 630 631 632
    QCheckBox *enableBox = new QCheckBox(i18n("Enable"), this);
    enableBox->setChecked(true);

    label->setMinimumHeight(enableBox->height());

    if(layout->direction() == QBoxLayout::LeftToRight) {
633
    	layout->addWidget(iconLabel);
634 635 636 637 638 639
	layout->addWidget(label);
	layout->addWidget(item);
	layout->addWidget(enableBox);
    }
    else {
	QHBoxLayout *l = new QHBoxLayout(layout);
640 641
	
	l->addWidget(iconLabel);
642 643 644 645 646 647 648 649 650 651 652 653 654 655
	l->addWidget(label);
	l->setStretchFactor(label, 1);

	l->insertStretch(-1, 1);

	l->addWidget(enableBox);
	l->setStretchFactor(enableBox, 0);

	layout->addWidget(item);
    }

    enableBox->hide();

    connect(enableBox, SIGNAL(toggled(bool)), item, SLOT(setEnabled(bool)));
656
    m_enableBoxes.insert(item, enableBox);
657 658
}

659 660
void TagEditor::showEvent(QShowEvent *e)
{
661
    slotRefresh();
662 663 664
    QWidget::showEvent(e);
}

665 666 667 668 669 670 671 672
bool TagEditor::eventFilter(QObject *watched, QEvent *e)
{
    if(watched->inherits("QSpinBox") && e->type() == QEvent::KeyRelease)
	slotDataChanged();

    return false;
}

673 674 675 676
////////////////////////////////////////////////////////////////////////////////
// private slots
////////////////////////////////////////////////////////////////////////////////

677
void TagEditor::slotDataChanged(bool c)
678
{
679
    m_dataChanged = c;
680 681
}

682 683 684 685 686 687 688
void TagEditor::slotItemRemoved(PlaylistItem *item)
{
    m_items.remove(item);
    if(m_items.isEmpty())
	slotRefresh();
}

689
#include "tageditor.moc"