/* * Strawberry Music Player * This file was part of Clementine. * Copyright 2010, David Sansome * Copyright 2018-2021, Jonas Kvinge * * Strawberry 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 3 of the License, or * (at your option) any later version. * * Strawberry is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Strawberry. If not, see . * */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "core/application.h" #include "core/iconloader.h" #include "core/logging.h" #include "core/tagreaderclient.h" #include "utilities/strutils.h" #include "utilities/timeutils.h" #include "utilities/imageutils.h" #include "utilities/cryptutils.h" #include "widgets/busyindicator.h" #include "widgets/lineedit.h" #include "collection/collectionbackend.h" #include "playlist/playlist.h" #include "playlist/playlistdelegates.h" #ifdef HAVE_MUSICBRAINZ # include "musicbrainz/tagfetcher.h" # include "trackselectiondialog.h" #endif #include "covermanager/albumcoverchoicecontroller.h" #include "covermanager/albumcoverloader.h" #include "covermanager/albumcoverloaderoptions.h" #include "covermanager/albumcoverloaderresult.h" #include "covermanager/coverproviders.h" #include "covermanager/currentalbumcoverloader.h" #include "covermanager/albumcoverimageresult.h" #include "edittagdialog.h" #include "ui_edittagdialog.h" #include "tagreadermessages.pb.h" const char *EditTagDialog::kTagsDifferentHintText = QT_TR_NOOP("(different across multiple songs)"); const char *EditTagDialog::kArtDifferentHintText = QT_TR_NOOP("Different art across multiple songs."); const char *EditTagDialog::kSettingsGroup = "EditTagDialog"; EditTagDialog::EditTagDialog(Application *app, QWidget *parent) : QDialog(parent), ui_(new Ui_EditTagDialog), app_(app), album_cover_choice_controller_(new AlbumCoverChoiceController(this)), #ifdef HAVE_MUSICBRAINZ tag_fetcher_(new TagFetcher(this)), results_dialog_(new TrackSelectionDialog(this)), #endif image_no_cover_thumbnail_(ImageUtils::GenerateNoCoverImage(QSize(128, 128))), loading_(false), ignore_edits_(false), summary_cover_art_id_(-1), tags_cover_art_id_(-1), cover_art_is_set_(false), save_tag_pending_(0), save_art_pending_(0) { QObject::connect(app_->album_cover_loader(), &AlbumCoverLoader::AlbumCoverLoaded, this, &EditTagDialog::AlbumCoverLoaded); #ifdef HAVE_MUSICBRAINZ QObject::connect(tag_fetcher_, &TagFetcher::ResultAvailable, results_dialog_, &TrackSelectionDialog::FetchTagFinished, Qt::QueuedConnection); QObject::connect(tag_fetcher_, &TagFetcher::Progress, results_dialog_, &TrackSelectionDialog::FetchTagProgress); QObject::connect(results_dialog_, &TrackSelectionDialog::SongChosen, this, &EditTagDialog::FetchTagSongChosen); QObject::connect(results_dialog_, &TrackSelectionDialog::finished, tag_fetcher_, &TagFetcher::Cancel); #endif album_cover_choice_controller_->Init(app_); ui_->setupUi(this); ui_->splitter->setSizes(QList() << 200 << width() - 200); ui_->loading_label->hide(); ui_->label_lyrics->hide(); ui_->fetch_tag->setIcon(QPixmap::fromImage(QImage(":/pictures/musicbrainz.png"))); #ifdef HAVE_MUSICBRAINZ ui_->fetch_tag->setEnabled(true); #else ui_->fetch_tag->setEnabled(false); #endif // An editable field is one that has a label as a buddy. // The label is important because it gets turned bold when the field is changed. for (QLabel *label : findChildren()) { QWidget *widget = label->buddy(); if (widget) { // Store information about the field fields_ << FieldData(label, widget, widget->objectName()); // clazy:exclude=reserve-candidates // Connect the edited signal if (LineEdit *lineedit = qobject_cast(widget)) { QObject::connect(lineedit, &LineEdit::textChanged, this, &EditTagDialog::FieldValueEdited); QObject::connect(lineedit, &LineEdit::Reset, this, &EditTagDialog::ResetField); } else if (TextEdit *textedit = qobject_cast(widget)) { QObject::connect(textedit, &TextEdit::textChanged, this, &EditTagDialog::FieldValueEdited); QObject::connect(textedit, &TextEdit::Reset, this, &EditTagDialog::ResetField); } else if (SpinBox *spinbox = qobject_cast(widget)) { QObject::connect(spinbox, QOverload::of(&SpinBox::valueChanged), this, &EditTagDialog::FieldValueEdited); QObject::connect(spinbox, &SpinBox::Reset, this, &EditTagDialog::ResetField); } else if (CheckBox *checkbox = qobject_cast(widget)) { QObject::connect(checkbox, &QCheckBox::stateChanged, this, &EditTagDialog::FieldValueEdited); QObject::connect(checkbox, &CheckBox::Reset, this, &EditTagDialog::ResetField); } else if (RatingBox *ratingbox = qobject_cast(widget)) { QObject::connect(ratingbox, &RatingWidget::RatingChanged, this, &EditTagDialog::FieldValueEdited); } } } // Set the colour of all the labels on the summary page const bool light = palette().color(QPalette::Base).value() > 128; const QColor color = palette().color(QPalette::WindowText); QPalette summary_label_palette(palette()); summary_label_palette.setColor(QPalette::WindowText, light ? color.lighter(150) : color.darker(150)); for (QLabel *label : ui_->tab_summary->findChildren()) { if (label->property("field_label").toBool()) { label->setPalette(summary_label_palette); } } QObject::connect(ui_->song_list->selectionModel(), &QItemSelectionModel::selectionChanged, this, &EditTagDialog::SelectionChanged); QObject::connect(ui_->button_box, &QDialogButtonBox::clicked, this, &EditTagDialog::ButtonClicked); QObject::connect(ui_->playcount_reset, &QPushButton::clicked, this, &EditTagDialog::ResetPlayCounts); QObject::connect(ui_->rating, &RatingWidget::RatingChanged, this, &EditTagDialog::SongRated); #ifdef HAVE_MUSICBRAINZ QObject::connect(ui_->fetch_tag, &QPushButton::clicked, this, &EditTagDialog::FetchTag); #endif // Set up the album cover menu cover_menu_ = new QMenu(this); QList actions = album_cover_choice_controller_->GetAllActions(); QObject::connect(album_cover_choice_controller_, &AlbumCoverChoiceController::Error, this, &EditTagDialog::Error); QObject::connect(album_cover_choice_controller_->cover_from_file_action(), &QAction::triggered, this, &EditTagDialog::LoadCoverFromFile); QObject::connect(album_cover_choice_controller_->cover_to_file_action(), &QAction::triggered, this, &EditTagDialog::SaveCoverToFile); QObject::connect(album_cover_choice_controller_->cover_from_url_action(), &QAction::triggered, this, &EditTagDialog::LoadCoverFromURL); QObject::connect(album_cover_choice_controller_->search_for_cover_action(), &QAction::triggered, this, &EditTagDialog::SearchForCover); QObject::connect(album_cover_choice_controller_->unset_cover_action(), &QAction::triggered, this, &EditTagDialog::UnsetCover); QObject::connect(album_cover_choice_controller_->clear_cover_action(), &QAction::triggered, this, &EditTagDialog::ClearCover); QObject::connect(album_cover_choice_controller_->delete_cover_action(), &QAction::triggered, this, &EditTagDialog::DeleteCover); QObject::connect(album_cover_choice_controller_->show_cover_action(), &QAction::triggered, this, &EditTagDialog::ShowCover); QObject::connect(ui_->checkbox_embedded_cover, &QCheckBox::toggled, album_cover_choice_controller_, &AlbumCoverChoiceController::set_save_embedded_cover_override); cover_menu_->addActions(actions); ui_->tags_art_button->setMenu(cover_menu_); ui_->tags_art->installEventFilter(this); ui_->tags_art->setAcceptDrops(true); ui_->summary_art->installEventFilter(this); // Add the next/previous buttons previous_button_ = new QPushButton(IconLoader::Load("go-previous"), tr("Previous"), this); next_button_ = new QPushButton(IconLoader::Load("go-next"), tr("Next"), this); ui_->button_box->addButton(previous_button_, QDialogButtonBox::ResetRole); ui_->button_box->addButton(next_button_, QDialogButtonBox::ResetRole); QObject::connect(previous_button_, &QPushButton::clicked, this, &EditTagDialog::PreviousSong); QObject::connect(next_button_, &QPushButton::clicked, this, &EditTagDialog::NextSong); // Set some shortcuts for the buttons new QShortcut(QKeySequence::Back, previous_button_, SLOT(click())); new QShortcut(QKeySequence::Forward, next_button_, SLOT(click())); new QShortcut(QKeySequence::MoveToPreviousPage, previous_button_, SLOT(click())); new QShortcut(QKeySequence::MoveToNextPage, next_button_, SLOT(click())); // Show the shortcuts as tooltips previous_button_->setToolTip(QString("%1 (%2 / %3)").arg( previous_button_->text(), QKeySequence(QKeySequence::Back).toString(QKeySequence::NativeText), QKeySequence(QKeySequence::MoveToPreviousPage).toString(QKeySequence::NativeText))); next_button_->setToolTip(QString("%1 (%2 / %3)").arg( next_button_->text(), QKeySequence(QKeySequence::Forward).toString(QKeySequence::NativeText), QKeySequence(QKeySequence::MoveToNextPage).toString(QKeySequence::NativeText))); new TagCompleter(app_->collection_backend(), Playlist::Column_Artist, ui_->artist); new TagCompleter(app_->collection_backend(), Playlist::Column_Album, ui_->album); new TagCompleter(app_->collection_backend(), Playlist::Column_AlbumArtist, ui_->albumartist); new TagCompleter(app_->collection_backend(), Playlist::Column_Genre, ui_->genre); new TagCompleter(app_->collection_backend(), Playlist::Column_Composer, ui_->composer); new TagCompleter(app_->collection_backend(), Playlist::Column_Performer, ui_->performer); new TagCompleter(app_->collection_backend(), Playlist::Column_Grouping, ui_->grouping); cover_options_.get_image_data_ = true; cover_options_.get_image_ = true; cover_options_.scale_output_image_ = true; cover_options_.desired_height_ = 128; } EditTagDialog::~EditTagDialog() { delete ui_; } void EditTagDialog::showEvent(QShowEvent *e) { if (!e->spontaneous()) { // Set the dialog's height to the smallest possible resize(width(), sizeHint().height()); // Restore the tab that was current last time. QSettings s; s.beginGroup(kSettingsGroup); if (s.contains("geometry")) { restoreGeometry(s.value("geometry").toByteArray()); } ui_->tab_widget->setCurrentIndex(s.value("current_tab").toInt()); s.endGroup(); album_cover_choice_controller_->ReloadSettings(); } QDialog::showEvent(e); } void EditTagDialog::hideEvent(QHideEvent *e) { // Save the current tab QSettings s; s.beginGroup(kSettingsGroup); s.setValue("geometry", saveGeometry()); s.setValue("current_tab", ui_->tab_widget->currentIndex()); s.endGroup(); QDialog::hideEvent(e); } void EditTagDialog::accept() { // Show the loading indicator if (!SetLoading(tr("Saving tracks") + "...")) return; SaveData(); } bool EditTagDialog::eventFilter(QObject *o, QEvent *e) { if (o == ui_->tags_art) { switch (e->type()) { case QEvent::MouseButtonRelease:{ QMouseEvent *mouse_event = static_cast(e); if (mouse_event && mouse_event->button() == Qt::RightButton) { #if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)) cover_menu_->popup(mouse_event->globalPosition().toPoint()); #else cover_menu_->popup(mouse_event->globalPos()); #endif } break; } case QEvent::MouseButtonDblClick: ShowCover(); break; case QEvent::DragEnter: { QDragEnterEvent *event = static_cast(e); if (AlbumCoverChoiceController::CanAcceptDrag(event)) { event->acceptProposedAction(); } break; } case QEvent::Drop: { const QDropEvent *event = static_cast(e); if (event->mimeData()->hasImage()) { QImage image = qvariant_cast(event->mimeData()->imageData()); if (!image.isNull()) { UpdateCover(UpdateCoverAction_New, AlbumCoverImageResult(image)); } } break; } default: break; } } if (o == ui_->summary_art) { switch (e->type()) { case QEvent::MouseButtonDblClick: ShowCover(); break; default: break; } } return QDialog::eventFilter(o, e); } bool EditTagDialog::SetLoading(const QString &message) { const bool loading = !message.isEmpty(); if (loading == loading_) return false; loading_ = loading; ui_->button_box->setEnabled(!loading); ui_->tab_widget->setEnabled(!loading); ui_->song_list->setEnabled(!loading); #ifdef HAVE_MUSICBRAINZ ui_->fetch_tag->setEnabled(!loading); #endif ui_->loading_label->setVisible(loading); ui_->loading_label->set_text(message); return true; } QList EditTagDialog::LoadData(const SongList &songs) { QList ret; for (const Song &song : songs) { if (song.IsEditable()) { // Try reloading the tags from file Song copy(song); TagReaderClient::Instance()->ReadFileBlocking(copy.url().toLocalFile(), ©); if (copy.is_valid()) { copy.MergeUserSetData(song, false, false); ret << Data(copy); } } } return ret; } void EditTagDialog::SetSongs(const SongList &s, const PlaylistItemList &items) { // Show the loading indicator if (!SetLoading(tr("Loading tracks") + "...")) return; data_.clear(); playlist_items_ = items; ui_->song_list->clear(); collection_songs_.clear(); // Reload tags in the background QFuture> future = QtConcurrent::run(&EditTagDialog::LoadData, s); QFutureWatcher> *watcher = new QFutureWatcher>(); QObject::connect(watcher, &QFutureWatcher>::finished, this, &EditTagDialog::SetSongsFinished); watcher->setFuture(future); } void EditTagDialog::SetSongsFinished() { QFutureWatcher> *watcher = static_cast>*>(sender()); QList result_data = watcher->result(); watcher->deleteLater(); if (!SetLoading(QString())) return; data_ = result_data; if (data_.count() == 0) { // If there were no valid songs, disable everything ui_->song_list->setEnabled(false); ui_->tab_widget->setEnabled(false); // Show a summary with empty information UpdateSummaryTab(Song(), UpdateCoverAction_None); ui_->tab_widget->setCurrentWidget(ui_->tab_summary); SetSongListVisibility(false); return; } // Add the filenames to the list for (const Data &tag_data : data_) { ui_->song_list->addItem(tag_data.current_.basefilename()); } // Select all ui_->song_list->setCurrentRow(0); ui_->song_list->selectAll(); // Hide the list if there's only one song in it SetSongListVisibility(data_.count() != 1); } void EditTagDialog::SetSongListVisibility(bool visible) { ui_->song_list->setVisible(visible); previous_button_->setEnabled(visible); next_button_->setEnabled(visible); } QVariant EditTagDialog::Data::value(const Song &song, const QString &id) { if (id == "title") return song.title(); if (id == "artist") return song.artist(); if (id == "album") return song.album(); if (id == "albumartist") return song.albumartist(); if (id == "composer") return song.composer(); if (id == "performer") return song.performer(); if (id == "grouping") return song.grouping(); if (id == "genre") return song.genre(); if (id == "comment") return song.comment(); if (id == "lyrics") return song.lyrics(); if (id == "track") return song.track(); if (id == "disc") return song.disc(); if (id == "year") return song.year(); if (id == "compilation") return song.compilation(); if (id == "rating") { return song.rating(); } qLog(Warning) << "Unknown ID" << id; return QVariant(); } void EditTagDialog::Data::set_value(const QString &id, const QVariant &value) { if (id == "title") current_.set_title(value.toString()); else if (id == "artist") current_.set_artist(value.toString()); else if (id == "album") current_.set_album(value.toString()); else if (id == "albumartist") current_.set_albumartist(value.toString()); else if (id == "composer") current_.set_composer(value.toString()); else if (id == "performer") current_.set_performer(value.toString()); else if (id == "grouping") current_.set_grouping(value.toString()); else if (id == "genre") current_.set_genre(value.toString()); else if (id == "comment") current_.set_comment(value.toString()); else if (id == "lyrics") current_.set_lyrics(value.toString()); else if (id == "track") current_.set_track(value.toInt()); else if (id == "disc") current_.set_disc(value.toInt()); else if (id == "year") current_.set_year(value.toInt()); else if (id == "compilation") current_.set_compilation(value.toBool()); else if (id == "rating") { current_.set_rating(value.toFloat()); } else qLog(Warning) << "Unknown ID" << id; } bool EditTagDialog::DoesValueVary(const QModelIndexList &sel, const QString &id) const { QVariant value = data_[sel.first().row()].current_value(id); for (int i = 1; i < sel.count(); ++i) { if (value != data_[sel[i].row()].current_value(id)) return true; } return false; } bool EditTagDialog::IsValueModified(const QModelIndexList &sel, const QString &id) const { return std::any_of(sel.begin(), sel.end(), [this, id](const QModelIndex &i) { return data_[i.row()].original_value(id) != data_[i.row()].current_value(id); }); } void EditTagDialog::InitFieldValue(const FieldData &field, const QModelIndexList &sel) { const bool varies = DoesValueVary(sel, field.id_); if (ExtendedEditor *editor = dynamic_cast(field.editor_)) { editor->clear(); editor->clear_hint(); if (varies) { editor->set_hint(tr(kTagsDifferentHintText)); editor->set_partially(); } else { editor->set_value(data_[sel[0].row()].current_value(field.id_)); } } else if (field.editor_) { qLog(Error) << "Missing editor for" << field.editor_->objectName(); } UpdateModifiedField(field, sel); } void EditTagDialog::UpdateFieldValue(const FieldData &field, const QModelIndexList &sel) { // Get the value from the field QVariant value; if (ExtendedEditor *editor = dynamic_cast(field.editor_)) { value = editor->value(); } else if (field.editor_) { qLog(Error) << "Missing editor for" << field.editor_->objectName(); } // Did we get it? if (!value.isValid()) { return; } // Set it in each selected song for (const QModelIndex &i : sel) { data_[i.row()].set_value(field.id_, value); } UpdateModifiedField(field, sel); } void EditTagDialog::UpdateModifiedField(const FieldData &field, const QModelIndexList &sel) { const bool modified = IsValueModified(sel, field.id_); // Update the boldness QFont new_font(font()); new_font.setBold(modified); field.label_->setFont(new_font); if (field.editor_) field.editor_->setFont(new_font); } void EditTagDialog::ResetFieldValue(const FieldData &field, const QModelIndexList &sel) { // Reset each selected song for (const QModelIndex &i : sel) { Data &tag_data = data_[i.row()]; tag_data.set_value(field.id_, tag_data.original_value(field.id_)); } // Reset the field InitFieldValue(field, sel); } void EditTagDialog::SelectionChanged() { const QModelIndexList indexes = ui_->song_list->selectionModel()->selectedIndexes(); if (indexes.isEmpty()) return; // Set the editable fields UpdateUI(indexes); // If we're editing multiple songs then we have to disable certain tabs const bool multiple = indexes.count() > 1; ui_->tab_widget->setTabEnabled(ui_->tab_widget->indexOf(ui_->tab_summary), !multiple); ui_->tab_widget->setTabEnabled(ui_->tab_widget->indexOf(ui_->tab_lyrics), !multiple); if (multiple) { UpdateSummaryTab(Song(), UpdateCoverAction_None); UpdateStatisticsTab(Song()); } else { UpdateSummaryTab(data_[indexes.first().row()].original_, data_[indexes.first().row()].cover_action_); UpdateStatisticsTab(data_[indexes.first().row()].original_); } const Song &first_song = data_[indexes.first().row()].original_; UpdateCoverAction first_cover_action = data_[indexes.first().row()].cover_action_; bool art_different = false; bool action_different = false; bool albumartist_enabled = false; bool composer_enabled = false; bool performer_enabled = false; bool grouping_enabled = false; bool genre_enabled = false; bool compilation_enabled = false; bool rating_enabled = false; bool comment_enabled = false; bool lyrics_enabled = false; for (const QModelIndex &idx : indexes) { if (data_[idx.row()].cover_action_ == UpdateCoverAction_None) { data_[idx.row()].cover_result_ = AlbumCoverImageResult(); } const Song &song = data_[idx.row()].original_; if (data_[idx.row()].cover_action_ != first_cover_action || (first_cover_action != UpdateCoverAction_None && data_[idx.row()].cover_result_.image_data != data_[indexes.first().row()].cover_result_.image_data)) { action_different = true; } if (data_[idx.row()].cover_action_ != first_cover_action || song.art_manual() != first_song.art_manual() || song.has_embedded_cover() != first_song.has_embedded_cover() || (song.art_manual().isEmpty() && song.art_automatic() != first_song.art_automatic()) || (song.has_embedded_cover() && first_song.has_embedded_cover() && (first_song.effective_albumartist() != song.effective_albumartist() || first_song.album() != song.album())) ) { art_different = true; } if (song.albumartist_supported()) { albumartist_enabled = true; } if (song.composer_supported()) { composer_enabled = true; } if (song.performer_supported()) { performer_enabled = true; } if (song.grouping_supported()) { grouping_enabled = true; } if (song.genre_supported()) { genre_enabled = true; } if (song.compilation_supported()) { compilation_enabled = true; } if (song.rating_supported()) { rating_enabled = true; } if (song.comment_supported()) { comment_enabled = true; } if (song.lyrics_supported()) { lyrics_enabled = true; } } QString summary; if (indexes.count() == 1) { summary += "

" + first_song.PrettyTitleWithArtist().toHtmlEscaped() + "

"; } else { summary += "

"; summary += tr("%1 songs selected.").arg(indexes.count()); summary += "

"; } const bool enable_change_art = first_song.is_collection_song(); ui_->tags_art_button->setEnabled(enable_change_art); if ((art_different && first_cover_action != UpdateCoverAction_New) || action_different) { tags_cover_art_id_ = -1; // Cancels any pending art load. ui_->tags_art->clear(); ui_->tags_art->setText(kArtDifferentHintText); album_cover_choice_controller_->show_cover_action()->setEnabled(false); album_cover_choice_controller_->cover_to_file_action()->setEnabled(false); album_cover_choice_controller_->cover_from_file_action()->setEnabled(enable_change_art); album_cover_choice_controller_->cover_from_url_action()->setEnabled(enable_change_art); album_cover_choice_controller_->search_for_cover_action()->setEnabled(enable_change_art); album_cover_choice_controller_->unset_cover_action()->setEnabled(enable_change_art); album_cover_choice_controller_->clear_cover_action()->setEnabled(enable_change_art); album_cover_choice_controller_->delete_cover_action()->setEnabled(enable_change_art); album_cover_choice_controller_->search_for_cover_action()->setEnabled(enable_change_art); } else { ui_->tags_art->clear(); album_cover_choice_controller_->show_cover_action()->setEnabled(first_song.has_valid_art() && !first_song.has_manually_unset_cover()); album_cover_choice_controller_->cover_to_file_action()->setEnabled(first_song.has_valid_art() && !first_song.has_manually_unset_cover()); album_cover_choice_controller_->cover_from_file_action()->setEnabled(enable_change_art); album_cover_choice_controller_->cover_from_url_action()->setEnabled(enable_change_art); album_cover_choice_controller_->search_for_cover_action()->setEnabled(app_->cover_providers()->HasAnyProviders() && enable_change_art); album_cover_choice_controller_->unset_cover_action()->setEnabled(enable_change_art && !first_song.has_manually_unset_cover()); album_cover_choice_controller_->clear_cover_action()->setEnabled(enable_change_art && !first_song.art_manual().isEmpty()); album_cover_choice_controller_->delete_cover_action()->setEnabled(enable_change_art && first_song.has_valid_art() && !first_song.has_manually_unset_cover()); if (data_[indexes.first().row()].cover_action_ == UpdateCoverAction_None) { tags_cover_art_id_ = app_->album_cover_loader()->LoadImageAsync(cover_options_, first_song); } else { tags_cover_art_id_ = app_->album_cover_loader()->LoadImageAsync(cover_options_, data_[indexes.first().row()].cover_result_); } summary += GetArtSummary(first_song, first_cover_action); } ui_->tags_summary->setText(summary); const bool embedded_cover = (first_song.save_embedded_cover_supported() && (first_song.has_embedded_cover() || album_cover_choice_controller_->get_collection_save_album_cover_type() == CollectionSettingsPage::SaveCoverType_Embedded)); ui_->checkbox_embedded_cover->setChecked(embedded_cover); album_cover_choice_controller_->set_save_embedded_cover_override(embedded_cover); ui_->albumartist->setEnabled(albumartist_enabled); ui_->composer->setEnabled(composer_enabled); ui_->performer->setEnabled(performer_enabled); ui_->grouping->setEnabled(grouping_enabled); ui_->genre->setEnabled(genre_enabled); ui_->compilation->setEnabled(compilation_enabled); ui_->rating->setEnabled(rating_enabled); ui_->comment->setEnabled(comment_enabled); ui_->lyrics->setEnabled(lyrics_enabled); } void EditTagDialog::UpdateUI(const QModelIndexList &indexes) { ignore_edits_ = true; for (const FieldData &field : fields_) { InitFieldValue(field, indexes); } ignore_edits_ = false; } void EditTagDialog::SetText(QLabel *label, const int value, const QString &suffix, const QString &def) { label->setText(value <= 0 ? def : (QString::number(value) + " " + suffix)); } void EditTagDialog::SetDate(QLabel *label, const uint time) { if (time == std::numeric_limits::max()) { // -1 label->setText(QObject::tr("Unknown")); } else { label->setText(QDateTime::fromSecsSinceEpoch(time).toString(QLocale::system().dateTimeFormat(QLocale::LongFormat))); } } void EditTagDialog::UpdateSummaryTab(const Song &song, const UpdateCoverAction cover_action) { summary_cover_art_id_ = app_->album_cover_loader()->LoadImageAsync(cover_options_, song); QString summary = "

" + song.PrettyTitleWithArtist().toHtmlEscaped() + "

"; summary += GetArtSummary(song, cover_action); ui_->summary->setText(summary); ui_->length->setText(Utilities::PrettyTimeNanosec(song.length_nanosec())); SetText(ui_->samplerate, song.samplerate(), "Hz"); SetText(ui_->bitdepth, song.bitdepth(), "Bit"); SetText(ui_->bitrate, song.bitrate(), tr("kbps")); SetDate(ui_->mtime, song.mtime()); SetDate(ui_->ctime, song.ctime()); if (song.filesize() == -1) { ui_->filesize->setText(tr("Unknown")); } else { ui_->filesize->setText(Utilities::PrettySize(song.filesize())); } ui_->filetype->setText(song.TextForFiletype()); if (song.url().isLocalFile()) { ui_->filename->setText(song.url().fileName()); ui_->path->setText(QFileInfo(QDir::toNativeSeparators(song.url().toLocalFile())).path()); } else { ui_->filename->setText(song.url().toString()); ui_->path->clear(); } if (song.art_manual().isEmpty()) { ui_->art_manual->setText(tr("None")); } else if (song.has_manually_unset_cover()) { ui_->art_manual->setText(tr("Unset")); } else { ui_->art_manual->setText(song.art_manual().toString()); } if (song.art_automatic().isEmpty()) { ui_->art_automatic->setText(tr("None")); } else if (song.has_embedded_cover()) { ui_->art_automatic->setText(tr("Embedded")); } else { ui_->art_automatic->setText(song.art_automatic().toString()); } } QString EditTagDialog::GetArtSummary(const Song &song, const UpdateCoverAction cover_action) { QString summary; if (cover_action != UpdateCoverAction_None) { switch (cover_action) { case UpdateCoverAction_Clear: summary = tr("Cover changed: Will be cleared when saved.").toHtmlEscaped(); break; case UpdateCoverAction_Unset: summary = tr("Cover changed: Will be unset when saved.").toHtmlEscaped(); break; case UpdateCoverAction_Delete: summary = tr("Cover changed: Will be deleted when saved.").toHtmlEscaped(); break; case UpdateCoverAction_New: summary = tr("Cover changed: Will set new when saved.").toHtmlEscaped(); break; case UpdateCoverAction_None: break; } } else if (song.art_manual().isEmpty() && song.art_automatic().isEmpty()) { summary = tr("Cover art not set").toHtmlEscaped(); } else if (song.has_manually_unset_cover()) { summary = tr("Cover art manually unset").toHtmlEscaped(); } else if (song.art_manual_is_valid()) { summary = tr("Manually set cover art from %1").arg(song.art_manual().toString()).toHtmlEscaped(); } else if (song.has_embedded_cover()) { summary = tr("Cover art from embedded image"); } else if (song.art_automatic_is_valid()) { summary = tr("Cover art automatically loaded from %1").arg(song.art_automatic().toString()).toHtmlEscaped(); } else if (!song.art_manual().isEmpty()) { summary = tr("Manually cover art from %1 is missing").arg(song.art_manual().toString()).toHtmlEscaped(); } else if (!song.art_automatic().isEmpty()) { summary = tr("Automatically cover art from %1 is missing").arg(song.art_automatic().toString()).toHtmlEscaped(); } if (!song.is_collection_song()) { if (!summary.isEmpty()) summary += "
"; summary = tr("Album cover editing is only available for collection songs."); } return summary; } void EditTagDialog::UpdateStatisticsTab(const Song &song) { ui_->playcount->setText(QString::number(song.playcount())); ui_->skipcount->setText(QString::number(song.skipcount())); ui_->lastplayed->setText(song.lastplayed() <= 0 ? tr("Never") : QDateTime::fromSecsSinceEpoch(song.lastplayed()).toString(QLocale::system().dateTimeFormat(QLocale::LongFormat))); } void EditTagDialog::AlbumCoverLoaded(const quint64 id, const AlbumCoverLoaderResult &result) { if (id == tags_cover_art_id_) { ui_->tags_art->clear(); bool enable_change_art = false; if (result.success && !result.image_scaled.isNull() && result.type != AlbumCoverLoaderResult::Type_ManuallyUnset) { ui_->tags_art->setPixmap(QPixmap::fromImage(result.image_scaled)); for (const QModelIndex &idx : ui_->song_list->selectionModel()->selectedIndexes()) { data_[idx.row()].cover_result_ = result.album_cover; enable_change_art = data_[idx.row()].original_.is_collection_song(); } } else { ui_->tags_art->setPixmap(QPixmap::fromImage(image_no_cover_thumbnail_)); for (const QModelIndex &idx : ui_->song_list->selectionModel()->selectedIndexes()) { data_[idx.row()].cover_result_ = AlbumCoverImageResult(); enable_change_art = data_[idx.row()].original_.is_collection_song(); } } tags_cover_art_id_ = -1; album_cover_choice_controller_->show_cover_action()->setEnabled(result.success && result.type != AlbumCoverLoaderResult::Type_ManuallyUnset); album_cover_choice_controller_->cover_to_file_action()->setEnabled(result.success && result.type != AlbumCoverLoaderResult::Type_ManuallyUnset); album_cover_choice_controller_->delete_cover_action()->setEnabled(enable_change_art && result.success && result.type != AlbumCoverLoaderResult::Type_ManuallyUnset); } else if (id == summary_cover_art_id_) { if (result.success && !result.image_scaled.isNull() && result.type != AlbumCoverLoaderResult::Type_ManuallyUnset) { ui_->summary_art->setPixmap(QPixmap::fromImage(result.image_scaled)); } else { ui_->summary_art->setPixmap(QPixmap::fromImage(image_no_cover_thumbnail_)); } summary_cover_art_id_ = -1; } } void EditTagDialog::FieldValueEdited() { if (ignore_edits_) return; const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); if (sel.isEmpty()) { return; } QWidget *w = qobject_cast(sender()); // Find the field for (const FieldData &field : fields_) { if (field.editor_ == w) { UpdateFieldValue(field, sel); return; } } } void EditTagDialog::ResetField() { const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); if (sel.isEmpty()) { return; } QWidget *w = qobject_cast(sender()); // Find the field for (const FieldData &field : fields_) { if (field.editor_ == w) { ignore_edits_ = true; ResetFieldValue(field, sel); ignore_edits_ = false; return; } } } Song *EditTagDialog::GetFirstSelected() { const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); if (sel.isEmpty()) return nullptr; return &data_[sel.first().row()].current_; } void EditTagDialog::LoadCoverFromFile() { Song *song = GetFirstSelected(); if (!song) return; AlbumCoverImageResult result = album_cover_choice_controller_->LoadImageFromFile(song); if (result.is_valid()) UpdateCover(UpdateCoverAction_New, result); } void EditTagDialog::SaveCoverToFile() { if (ui_->song_list->selectionModel()->selectedIndexes().isEmpty()) return; const Data &first_data = data_[ui_->song_list->selectionModel()->selectedIndexes().first().row()]; album_cover_choice_controller_->SaveCoverToFileManual(first_data.current_, first_data.cover_result_); } void EditTagDialog::LoadCoverFromURL() { if (ui_->song_list->selectionModel()->selectedIndexes().isEmpty()) return; AlbumCoverImageResult result = album_cover_choice_controller_->LoadImageFromURL(); if (result.is_valid()) UpdateCover(UpdateCoverAction_New, result); } void EditTagDialog::SearchForCover() { Song *song = GetFirstSelected(); if (!song) return; AlbumCoverImageResult result = album_cover_choice_controller_->SearchForImage(song); if (result.is_valid()) UpdateCover(UpdateCoverAction_New, result); } void EditTagDialog::UnsetCover() { Song *song = GetFirstSelected(); if (!song) return; song->set_manually_unset_cover(); UpdateCover(UpdateCoverAction_Unset); } void EditTagDialog::ClearCover() { Song *song = GetFirstSelected(); if (!song) return; song->clear_art_automatic(); song->clear_art_manual(); UpdateCover(UpdateCoverAction_Clear); } void EditTagDialog::DeleteCover() { UpdateCover(UpdateCoverAction_Delete); } void EditTagDialog::ShowCover() { if (ui_->song_list->selectionModel()->selectedIndexes().isEmpty()) return; const Data &first_data = data_[ui_->song_list->selectionModel()->selectedIndexes().first().row()]; album_cover_choice_controller_->ShowCover(first_data.current_, first_data.cover_result_.image); } void EditTagDialog::UpdateCover(const UpdateCoverAction action, const AlbumCoverImageResult &result) { const QModelIndexList indexes = ui_->song_list->selectionModel()->selectedIndexes(); if (indexes.isEmpty()) return; QString artist = data_[indexes.first().row()].current_.effective_albumartist(); QString album = data_[indexes.first().row()].current_.album(); for (const QModelIndex &idx : indexes) { data_[idx.row()].cover_action_ = action; data_[idx.row()].cover_result_ = result; if (action == UpdateCoverAction_New) { data_[idx.row()].current_.clear_art_manual(); } else if (action == UpdateCoverAction_Unset) { data_[idx.row()].current_.set_manually_unset_cover(); } else if (action == UpdateCoverAction_Clear || action == UpdateCoverAction_Delete) { data_[idx.row()].current_.clear_art_manual(); data_[idx.row()].current_.clear_art_automatic(); } if (artist != data_[idx.row()].current_.effective_albumartist() || album != data_[idx.row()].current_.effective_albumartist()) { artist.clear(); album.clear(); } } // Now check if we have any other songs cached that share that artist and album (and would therefore be changed as well) if (!artist.isEmpty() && !album.isEmpty()) { for (int i = 0; i < data_.count(); ++i) { if (data_[i].current_.effective_albumartist() == artist && data_[i].current_.album() == album) { data_[i].cover_action_ = action; data_[i].cover_result_ = result; if (action == UpdateCoverAction_New) { data_[i].current_.clear_art_manual(); } else if (action == UpdateCoverAction_Unset) { data_[i].current_.set_manually_unset_cover(); } else if (action == UpdateCoverAction_Clear || action == UpdateCoverAction_Delete) { data_[i].current_.clear_art_manual(); data_[i].current_.clear_art_automatic(); } } } } UpdateSummaryTab(data_[indexes.first().row()].current_, data_[indexes.first().row()].cover_action_); SelectionChanged(); } void EditTagDialog::NextSong() { if (ui_->song_list->count() == 0) { return; } int row = (ui_->song_list->currentRow() + 1) % ui_->song_list->count(); ui_->song_list->setCurrentRow(row); } void EditTagDialog::PreviousSong() { if (ui_->song_list->count() == 0) { return; } int row = (ui_->song_list->currentRow() - 1 + ui_->song_list->count()) % ui_->song_list->count(); ui_->song_list->setCurrentRow(row); } void EditTagDialog::ButtonClicked(QAbstractButton *button) { if (button == ui_->button_box->button(QDialogButtonBox::Discard)) { reject(); } } void EditTagDialog::SaveData() { QMap cover_urls; for (int i = 0; i < data_.count(); ++i) { Data &ref = data_[i]; if (!ref.current_.IsMetadataEqual(ref.original_)) { ++save_tag_pending_; TagReaderReply *reply = TagReaderClient::Instance()->SaveFile(ref.current_.url().toLocalFile(), ref.current_); QObject::connect(reply, &TagReaderReply::Finished, this, [this, reply, ref]() { SongSaveTagsComplete(reply, ref.current_.url().toLocalFile(), ref.current_); }, Qt::QueuedConnection); } if (ref.current_.rating() != ref.original_.rating() && ref.current_.is_collection_song()) { app_->collection_backend()->UpdateSongRatingAsync(ref.current_.id(), ref.current_.rating(), true); } QString embedded_cover_from_file; // If embedded album cover is selected, and it isn't saved to the tags, then save it even if no action was done. if (ui_->checkbox_embedded_cover->isChecked() && ref.cover_action_ == UpdateCoverAction_None && !ref.original_.has_embedded_cover() && ref.original_.save_embedded_cover_supported()) { if (ref.original_.art_manual().isValid() && ref.original_.art_manual().isLocalFile() && QFile::exists(ref.original_.art_manual().toLocalFile())) { ref.cover_action_ = UpdateCoverAction_New; embedded_cover_from_file = ref.original_.art_manual().toLocalFile(); } else if (ref.original_.art_automatic().isValid() && ref.original_.art_automatic().isLocalFile() && QFile::exists(ref.original_.art_automatic().toLocalFile())) { ref.cover_action_ = UpdateCoverAction_New; embedded_cover_from_file = ref.original_.art_automatic().toLocalFile(); } } if (ref.cover_action_ != UpdateCoverAction_None) { switch (ref.cover_action_) { case UpdateCoverAction_None: break; case UpdateCoverAction_New:{ if ((!ref.current_.effective_albumartist().isEmpty() && !ref.current_.album().isEmpty()) && (!ui_->checkbox_embedded_cover->isChecked() || !ref.original_.save_embedded_cover_supported())) { QUrl cover_url; if (!ref.cover_result_.cover_url.isEmpty() && ref.cover_result_.cover_url.isLocalFile() && QFile::exists(ref.cover_result_.cover_url.toLocalFile())) { cover_url = ref.cover_result_.cover_url; } else { QString cover_hash = Utilities::Sha1CoverHash(ref.current_.effective_albumartist(), ref.current_.album()).toHex(); if (cover_urls.contains(cover_hash)) { cover_url = cover_urls[cover_hash]; } else { cover_url = album_cover_choice_controller_->SaveCoverToFileAutomatic(&ref.current_, ref.cover_result_); cover_urls.insert(cover_hash, cover_url); } } ref.current_.set_art_manual(cover_url); } break; } case UpdateCoverAction_Unset: ref.current_.set_manually_unset_cover(); break; case UpdateCoverAction_Clear: ref.current_.clear_art_manual(); break; case UpdateCoverAction_Delete:{ if (!ref.original_.art_automatic().isEmpty()) { if (ref.original_.art_automatic().isValid() && !ref.original_.has_embedded_cover() && ref.original_.art_automatic().isLocalFile()) { QString art_automatic = ref.original_.art_automatic().toLocalFile(); if (QFile::exists(art_automatic)) { QFile::remove(art_automatic); } } ref.current_.clear_art_automatic(); } if (!ref.original_.art_manual().isEmpty() && !ref.original_.has_manually_unset_cover()) { if (ref.original_.art_manual().isValid() && ref.original_.art_manual().isLocalFile()) { QString art_manual = ref.original_.art_manual().toLocalFile(); if (QFile::exists(art_manual)) { QFile::remove(art_manual); } } ref.current_.clear_art_manual(); } break; } } if (ui_->checkbox_embedded_cover->isChecked() && ref.original_.save_embedded_cover_supported()) { if (ref.cover_action_ == UpdateCoverAction_New) { if (ref.cover_result_.is_jpeg()) { // Save JPEG data directly. ++save_art_pending_; TagReaderReply *reply = TagReaderClient::Instance()->SaveEmbeddedArt(ref.current_.url().toLocalFile(), ref.cover_result_.image_data); QObject::connect(reply, &TagReaderReply::Finished, this, [this, reply, ref]() { SongSaveArtComplete(reply, ref.current_.url().toLocalFile(), ref.current_, ref.cover_action_); }, Qt::QueuedConnection); } else if (!ref.cover_result_.image.isNull()) { // Convert image data to JPEG. ++save_art_pending_; QFuture future = QtConcurrent::run(&ImageUtils::SaveImageToJpegData, ref.cover_result_.image); QFutureWatcher *watcher = new QFutureWatcher(); QObject::connect(watcher, &QFutureWatcher::finished, this, [this, watcher, ref]() { TagReaderReply *reply = TagReaderClient::Instance()->SaveEmbeddedArt(ref.current_.url().toLocalFile(), watcher->result()); QObject::connect(reply, &TagReaderReply::Finished, this, [this, reply, ref]() { SongSaveArtComplete(reply, ref.current_.url().toLocalFile(), ref.current_, ref.cover_action_); }, Qt::QueuedConnection); watcher->deleteLater(); }); watcher->setFuture(future); } else if (!embedded_cover_from_file.isEmpty()) { // Save existing file on disk as embedded cover. ++save_art_pending_; QFuture future = QtConcurrent::run(&ImageUtils::FileToJpegData, embedded_cover_from_file); QFutureWatcher *watcher = new QFutureWatcher(); QObject::connect(watcher, &QFutureWatcher::finished, this, [this, watcher, ref]() { TagReaderReply *reply = TagReaderClient::Instance()->SaveEmbeddedArt(ref.current_.url().toLocalFile(), watcher->result()); QObject::connect(reply, &TagReaderReply::Finished, this, [this, reply, ref]() { SongSaveArtComplete(reply, ref.current_.url().toLocalFile(), ref.current_, ref.cover_action_); }, Qt::QueuedConnection); watcher->deleteLater(); }); watcher->setFuture(future); } } else if (ref.cover_action_ == UpdateCoverAction_Delete) { ++save_art_pending_; TagReaderReply *reply = TagReaderClient::Instance()->SaveEmbeddedArt(ref.current_.url().toLocalFile(), QByteArray()); QObject::connect(reply, &TagReaderReply::Finished, this, [this, reply, ref]() { SongSaveArtComplete(reply, ref.current_.url().toLocalFile(), ref.current_, ref.cover_action_); }, Qt::QueuedConnection); } } else if (!ref.current_.effective_albumartist().isEmpty() && !ref.current_.album().isEmpty()) { if (ref.current_.is_collection_song()) { collection_songs_.insert(ref.current_.id(), ref.current_); } if (ref.current_ == app_->current_albumcover_loader()->last_song()) { app_->current_albumcover_loader()->LoadAlbumCover(ref.current_); } } } } if (save_tag_pending_ <= 0 && save_art_pending_ <= 0) AcceptFinished(); } void EditTagDialog::AcceptFinished() { if (!collection_songs_.isEmpty()) { app_->collection_backend()->AddOrUpdateSongsAsync(collection_songs_.values()); collection_songs_.clear(); } if (!SetLoading(QString())) return; QDialog::accept(); } void EditTagDialog::ResetPlayCounts() { const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); if (sel.isEmpty()) return; Song *song = &data_[sel.first().row()].original_; if (!song->is_valid() || song->id() == -1) return; if (QMessageBox::question(this, tr("Reset play counts"), tr("Are you sure you want to reset this song's statistics?"), QMessageBox::Reset, QMessageBox::Cancel) != QMessageBox::Reset) { return; } song->set_playcount(0); song->set_skipcount(0); song->set_lastplayed(-1); if (song->is_collection_song()) { app_->collection_backend()->ResetStatisticsAsync(song->id()); } UpdateStatisticsTab(*song); } void EditTagDialog::SongRated(const float rating) { const QModelIndexList indexes = ui_->song_list->selectionModel()->selectedIndexes(); if (indexes.isEmpty()) return; for (const QModelIndex &idx : indexes) { if (!data_[idx.row()].current_.is_valid() || data_[idx.row()].current_.id() == -1) continue; data_[idx.row()].current_.set_rating(rating); } } void EditTagDialog::FetchTag() { #ifdef HAVE_MUSICBRAINZ const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); SongList songs; for (const QModelIndex &idx : sel) { Song song = data_[idx.row()].original_; if (!song.is_valid()) { continue; } songs << song; } if (songs.isEmpty()) return; results_dialog_->Init(songs); tag_fetcher_->StartFetch(songs); results_dialog_->show(); #endif } void EditTagDialog::FetchTagSongChosen(const Song &original_song, const Song &new_metadata) { #ifdef HAVE_MUSICBRAINZ const QString filename = original_song.url().toLocalFile(); // Find the song with this filename auto data_it = std::find_if(data_.begin(), data_.end(), [&filename](const Data &d) { return d.original_.url().toLocalFile() == filename; }); if (data_it == data_.end()) { qLog(Warning) << "Could not find song to filename: " << filename; return; } // Update song data data_it->current_.set_title(new_metadata.title()); data_it->current_.set_artist(new_metadata.artist()); data_it->current_.set_album(new_metadata.album()); data_it->current_.set_track(new_metadata.track()); data_it->current_.set_year(new_metadata.year()); // Is it currently being displayed in the UI? if (ui_->song_list->currentRow() == std::distance(data_.begin(), data_it)) { // Yes! Additionally, update UI const QModelIndexList sel = ui_->song_list->selectionModel()->selectedIndexes(); UpdateUI(sel); } #else Q_UNUSED(original_song) Q_UNUSED(new_metadata) #endif } void EditTagDialog::SongSaveTagsComplete(TagReaderReply *reply, const QString &filename, Song song) { --save_tag_pending_; if (!reply->message().save_file_response().success()) { QString message = tr("An error occurred writing metadata to '%1'").arg(filename); emit Error(message); } else if (song.is_collection_song()) { if (collection_songs_.contains(song.id())) { Song old_song = collection_songs_.take(song.id()); song.set_art_automatic(old_song.art_automatic()); song.set_art_manual(old_song.art_manual()); } collection_songs_.insert(song.id(), song); } QMetaObject::invokeMethod(reply, "deleteLater", Qt::QueuedConnection); if (save_tag_pending_ <= 0 && save_art_pending_ <= 0) AcceptFinished(); } void EditTagDialog::SongSaveArtComplete(TagReaderReply *reply, const QString &filename, Song song, const UpdateCoverAction cover_action) { --save_art_pending_; if (!reply->message().save_embedded_art_response().success()) { QString message = tr("An error occurred writing cover art to '%1'").arg(filename); emit Error(message); } else if (song.is_collection_song()) { if (collection_songs_.contains(song.id())) { song = collection_songs_.take(song.id()); } switch (cover_action) { case UpdateCoverAction_None: break; case UpdateCoverAction_New: song.clear_art_manual(); song.set_embedded_cover(); break; case UpdateCoverAction_Clear: case UpdateCoverAction_Delete: song.clear_art_automatic(); song.clear_art_manual(); break; case UpdateCoverAction_Unset: song.clear_art_automatic(); song.set_manually_unset_cover(); break; } collection_songs_.insert(song.id(), song); } if (song == app_->current_albumcover_loader()->last_song()) { app_->current_albumcover_loader()->LoadAlbumCover(song); } QMetaObject::invokeMethod(reply, "deleteLater", Qt::QueuedConnection); if (save_tag_pending_ <= 0 && save_art_pending_ <= 0) AcceptFinished(); }