/* * Strawberry Music Player * This code was part of Clementine (GlobalSearch) * Copyright 2012, 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 "core/mimedata.h" #include "core/iconloader.h" #include "internetsongmimedata.h" #include "internetservice.h" #include "internetsearchmodel.h" #include "internetsearchview.h" InternetSearchModel::InternetSearchModel(InternetService *service, QObject *parent) : QStandardItemModel(parent), service_(service), proxy_(nullptr), use_pretty_covers_(true), artist_icon_(IconLoader::Load("folder-sound")), album_icon_(IconLoader::Load("cdcase")) { group_by_[0] = CollectionModel::GroupBy_AlbumArtist; group_by_[1] = CollectionModel::GroupBy_AlbumDisc; group_by_[2] = CollectionModel::GroupBy_None; QList nocover_sizes = album_icon_.availableSizes(); no_cover_icon_ = album_icon_.pixmap(nocover_sizes.last()).scaled(CollectionModel::kPrettyCoverSize, CollectionModel::kPrettyCoverSize, Qt::KeepAspectRatio, Qt::SmoothTransformation); } void InternetSearchModel::AddResults(const InternetSearchView::ResultList &results) { for (const InternetSearchView::Result &result : results) { QStandardItem *parent = invisibleRootItem(); // Find (or create) the container nodes for this result if we can. ContainerKey key; parent = BuildContainers(result.metadata_, parent, &key); // Create the item QStandardItem *item = new QStandardItem; item->setText(result.metadata_.TitleWithCompilationArtist()); item->setData(QVariant::fromValue(result), Role_Result); parent->appendRow(item); } } QStandardItem *InternetSearchModel::BuildContainers(const Song &s, QStandardItem *parent, ContainerKey *key, const int level) { if (level >= 3) { return parent; } bool has_artist_icon = false; bool has_album_icon = false; QString display_text; QString sort_text; QString unique_tag; switch (group_by_[level]) { case CollectionModel::GroupBy_AlbumArtist: if (s.is_compilation()) { display_text = tr("Various artists"); sort_text = "aaaaaa"; } else { display_text = CollectionModel::TextOrUnknown(s.effective_albumartist()); sort_text = CollectionModel::SortTextForArtist(s.effective_albumartist()); } has_artist_icon = true; break; case CollectionModel::GroupBy_Artist: if (s.is_compilation()) { display_text = tr("Various artists"); sort_text = "aaaaaa"; } else { display_text = CollectionModel::TextOrUnknown(s.artist()); sort_text = CollectionModel::SortTextForArtist(s.artist()); } has_artist_icon = true; break; case CollectionModel::GroupBy_Album: display_text = CollectionModel::TextOrUnknown(s.album()); sort_text = CollectionModel::SortTextForArtist(s.album()); unique_tag = s.album_id(); has_album_icon = true; break; case CollectionModel::GroupBy_AlbumDisc:{ int disc = qMax(0, s.disc()); display_text = CollectionModel::PrettyAlbumDisc(s.album(), disc); sort_text = s.album() + CollectionModel::SortTextForNumber(disc); unique_tag = s.album_id(); has_album_icon = true; break; } case CollectionModel::GroupBy_YearAlbum:{ int year = qMax(0, s.year()); display_text = CollectionModel::PrettyYearAlbum(year, s.album()); sort_text = CollectionModel::SortTextForNumber(year) + s.album(); unique_tag = s.album_id(); has_album_icon = true; break; } case CollectionModel::GroupBy_YearAlbumDisc:{ int year = qMax(0, s.year()); int disc = qMax(0, s.disc()); display_text = CollectionModel::PrettyYearAlbumDisc(year, s.album(), disc); sort_text = CollectionModel::SortTextForNumber(year) + s.album() + CollectionModel::SortTextForNumber(disc); unique_tag = s.album_id(); has_album_icon = true; break; } case CollectionModel::GroupBy_OriginalYearAlbum:{ int year = qMax(0, s.effective_originalyear()); display_text = CollectionModel::PrettyYearAlbum(year, s.album()); sort_text = CollectionModel::SortTextForNumber(year) + s.album(); unique_tag = s.album_id(); has_album_icon = true; break; } case CollectionModel::GroupBy_OriginalYearAlbumDisc:{ const int year = qMax(0, s.effective_originalyear()); const int disc = qMax(0, s.disc()); display_text = CollectionModel::PrettyYearAlbumDisc(year, s.album(), disc); sort_text = CollectionModel::SortTextForNumber(year) + s.album() + CollectionModel::SortTextForNumber(disc); unique_tag = s.album_id(); has_album_icon = true; break; } case CollectionModel::GroupBy_Disc: display_text = CollectionModel::PrettyDisc(s.disc()); sort_text = CollectionModel::SortTextForArtist(display_text); has_album_icon = true; break; case CollectionModel::GroupBy_Year:{ const int year = qMax(0, s.year()); display_text = QString::number(year); sort_text = CollectionModel::SortTextForNumber(year) + " "; break; } case CollectionModel::GroupBy_OriginalYear:{ const int year = qMax(0, s.effective_originalyear()); display_text = QString::number(year); sort_text = CollectionModel::SortTextForNumber(year) + " "; break; } case CollectionModel::GroupBy_Genre: display_text = CollectionModel::TextOrUnknown(s.genre()); sort_text = CollectionModel::SortTextForArtist(s.genre()); has_album_icon = true; break; case CollectionModel::GroupBy_Composer: display_text = CollectionModel::TextOrUnknown(s.composer()); sort_text = CollectionModel::SortTextForArtist(s.composer()); has_album_icon = true; break; case CollectionModel::GroupBy_Performer: display_text = CollectionModel::TextOrUnknown(s.performer()); sort_text = CollectionModel::SortTextForArtist(s.performer()); has_album_icon = true; break; case CollectionModel::GroupBy_Grouping: display_text = CollectionModel::TextOrUnknown(s.grouping()); sort_text = CollectionModel::SortTextForArtist(s.grouping()); has_album_icon = true; break; case CollectionModel::GroupBy_FileType: display_text = s.TextForFiletype(); sort_text = display_text; break; case CollectionModel::GroupBy_Format: if (s.samplerate() <= 0) { display_text = s.TextForFiletype(); } else { if (s.bitdepth() <= 0) { display_text = QString("%1 (%2)").arg(s.TextForFiletype(), QString::number(s.samplerate() / 1000.0, 'G', 5)); } else { display_text = QString("%1 (%2/%3)").arg(s.TextForFiletype(), QString::number(s.samplerate() / 1000.0, 'G', 5), QString::number(s.bitdepth())); } } sort_text = display_text; break; case CollectionModel::GroupBy_Samplerate: display_text = QString::number(s.samplerate()); sort_text = display_text; break; case CollectionModel::GroupBy_Bitdepth: display_text = QString::number(s.bitdepth()); sort_text = display_text; break; case CollectionModel::GroupBy_Bitrate: display_text = QString::number(s.bitrate()); sort_text = display_text; break; case CollectionModel::GroupBy_None: case CollectionModel::GroupByCount: return parent; } if (display_text.isEmpty()) display_text = "Unknown"; if (sort_text.isEmpty()) sort_text = "Unknown"; // Find a container for this level key->group_[level] = display_text + unique_tag; QStandardItem *container = containers_[*key]; if (!container) { container = new QStandardItem(display_text); container->setData(sort_text, CollectionModel::Role_SortText); container->setData(group_by_[level], CollectionModel::Role_ContainerType); if (has_artist_icon) { container->setIcon(artist_icon_); } else if (has_album_icon) { if (use_pretty_covers_) { container->setData(no_cover_icon_, Qt::DecorationRole); } else { container->setIcon(album_icon_); } } parent->appendRow(container); containers_[*key] = container; } // Create the container for the next level. return BuildContainers(s, container, key, level + 1); } void InternetSearchModel::Clear() { containers_.clear(); clear(); } InternetSearchView::ResultList InternetSearchModel::GetChildResults(const QModelIndexList &indexes) const { QList items; items.reserve(indexes.count()); for (const QModelIndex &idx : indexes) { items << itemFromIndex(idx); } return GetChildResults(items); } InternetSearchView::ResultList InternetSearchModel::GetChildResults(const QList &items) const { InternetSearchView::ResultList results; QSet visited; for (QStandardItem *item : items) { GetChildResults(item, &results, &visited); } return results; } void InternetSearchModel::GetChildResults(const QStandardItem *item, InternetSearchView::ResultList *results, QSet *visited) const { if (visited->contains(item)) { return; } visited->insert(item); // Does this item have children? if (item->rowCount()) { const QModelIndex parent_proxy_index = proxy_->mapFromSource(item->index()); // Yes - visit all the children, but do so through the proxy so we get them in the right order. for (int i = 0 ; i < item->rowCount() ; ++i) { const QModelIndex proxy_index = parent_proxy_index.model()->index(i, 0, parent_proxy_index); const QModelIndex idx = proxy_->mapToSource(proxy_index); GetChildResults(itemFromIndex(idx), results, visited); } } else { // No - maybe it's a song, add its result if valid QVariant result = item->data(Role_Result); if (result.isValid()) { results->append(result.value()); } } } QMimeData *InternetSearchModel::mimeData(const QModelIndexList &indexes) const { return LoadTracks(GetChildResults(indexes)); } namespace { void GatherResults(const QStandardItem *parent, InternetSearchView::ResultList *results) { QVariant result_variant = parent->data(InternetSearchModel::Role_Result); if (result_variant.isValid()) { InternetSearchView::Result result = result_variant.value(); (*results).append(result); } for (int i = 0 ; i < parent->rowCount() ; ++i) { GatherResults(parent->child(i), results); } } } // namespace void InternetSearchModel::SetGroupBy(const CollectionModel::Grouping grouping, const bool regroup_now) { const CollectionModel::Grouping old_group_by = group_by_; group_by_ = grouping; if (regroup_now && group_by_ != old_group_by) { // Walk the tree gathering the results we have already InternetSearchView::ResultList results; GatherResults(invisibleRootItem(), &results); // Reset the model and re-add all the results using the new grouping. Clear(); AddResults(results); } } MimeData *InternetSearchModel::LoadTracks(const InternetSearchView::ResultList &results) const { if (results.isEmpty()) { return nullptr; } SongList songs; QList urls; songs.reserve(results.count()); urls.reserve(results.count()); for (const InternetSearchView::Result &result : results) { songs << result.metadata_; urls << result.metadata_.url(); } InternetSongMimeData *internet_song_mime_data = new InternetSongMimeData(service_); internet_song_mime_data->songs = songs; MimeData *mime_data = internet_song_mime_data; mime_data->setUrls(urls); return mime_data; }