MuseScore/mscore/scoreBrowser.cpp

361 lines
13 KiB
C++

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2014 Werner Schweer
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2
// as published by the Free Software Foundation and appearing in
// the file LICENCE.GPL
//=============================================================================
#include "scoreBrowser.h"
#include "musescore.h"
#include "icons.h"
#include "libmscore/score.h"
namespace Ms {
//---------------------------------------------------------
// sizeHint
//---------------------------------------------------------
QSize ScoreListWidget::sizeHint() const
{
int cols = (width()-SPACE) / (CELLW + SPACE);
int n = 0;
for (int i = 0; i < count(); ++i) {
if (!item(i)->isHidden())
n++;
}
int rows = 1;
if (cols > 0)
rows = (n+cols-1) / cols;
if (rows <= 0)
rows = 1;
return QSize(cols * CELLW, rows * (CELLH + SPACE) + SPACE);
}
//---------------------------------------------------------
// ScoreItem
//---------------------------------------------------------
class ScoreItem : public QListWidgetItem
{
ScoreInfo _info;
public:
ScoreItem(const ScoreInfo& i) : QListWidgetItem(), _info(i) {}
const ScoreInfo& info() const { return _info; }
};
//---------------------------------------------------------
// ScoreBrowser
//---------------------------------------------------------
ScoreBrowser::ScoreBrowser(QWidget* parent)
: QWidget(parent)
{
setupUi(this);
scoreList->setLayout(new QVBoxLayout);
scoreList->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
scoreList->layout()->setMargin(0);
_noMatchedScoresLabel = new QLabel(tr("There are no templates matching the current search."));
_noMatchedScoresLabel->setHidden(true);
_noMatchedScoresLabel->setObjectName("noMatchedScoresLabel");
scoreList->layout()->addWidget(_noMatchedScoresLabel);
connect(preview, SIGNAL(doubleClicked(QString)), SIGNAL(scoreActivated(QString)));
if (!_showPreview)
preview->setVisible(false);
}
//---------------------------------------------------------
// createScoreList
//---------------------------------------------------------
ScoreListWidget* ScoreBrowser::createScoreList()
{
ScoreListWidget* sl = new ScoreListWidget;
sl->setWrapping(true);
sl->setViewMode(QListView::IconMode);
sl->setIconSize(QSize(sl->cellWidth(), sl->cellHeight() - 30));
sl->setSpacing(sl->space());
sl->setResizeMode(QListView::Adjust);
sl->setFlow(QListView::LeftToRight);
sl->setMovement(QListView::Static);
sl->setTextElideMode(Qt::ElideRight);
sl->setWordWrap(true);
sl->setUniformItemSizes(true);
sl->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::MinimumExpanding);
sl->setLineWidth(0);
sl->setFrameStyle(QFrame::NoFrame | QFrame::Plain);
sl->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
sl->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
sl->setLayoutMode(QListView::SinglePass);
sl->setSizeAdjustPolicy(QAbstractScrollArea::AdjustToContents);
if (!_showPreview)
sl->setSelectionMode(QAbstractItemView::NoSelection);
if (!style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick)) {
// Set our handler for item clicks only if Qt
// doesn't treat a click as an item activation.
connect(sl, SIGNAL(itemClicked(QListWidgetItem*)), this, SLOT(scoreClicked(QListWidgetItem*)), Qt::QueuedConnection);
}
connect(sl, SIGNAL(itemActivated(QListWidgetItem*)), SLOT(setScoreActivated(QListWidgetItem*)));
scoreLists.append(sl);
return sl;
}
//---------------------------------------------------------
// genScoreItem
//---------------------------------------------------------
ScoreItem* ScoreBrowser::genScoreItem(const QFileInfo& fi, ScoreListWidget* l)
{
ScoreInfo si(fi);
QPixmap pm(l->iconSize() * qApp->devicePixelRatio());
if (!QPixmapCache::find(fi.filePath(), &pm)) {
//load and scale pixmap
QPixmap pixmap = mscore->extractThumbnail(fi.filePath());
if (pixmap.isNull())
pixmap = icons[int(Icons::file_ICON)]->pixmap(QSize(50,60));
pixmap = pixmap.scaled(pm.width() - 2, pm.height() - 2, Qt::KeepAspectRatio, Qt::SmoothTransformation);
// draw pixmap and add border
pm.fill(Qt::transparent);
QPainter painter( &pm );
painter.setRenderHint(QPainter::Antialiasing);
painter.setRenderHint(QPainter::TextAntialiasing);
painter.drawPixmap(0, 0, pixmap);
painter.setPen(QPen(QColor(0, 0, 0, 128), 1));
painter.setBrush(Qt::white);
if (fi.completeBaseName() == "00-Blank" || fi.completeBaseName() == "Create_New_Score") {
qreal round = 8.0 * qApp->devicePixelRatio();
painter.drawRoundedRect(QRectF(0, 0, pm.width() - 1 , pm.height() - 1), round, round);
}
else
painter.drawRect(0, 0, pm.width() - 1, pm.height() - 1);
if (fi.completeBaseName() != "00-Blank")
painter.drawPixmap(1, 1, pixmap);
painter.end();
QPixmapCache::insert(fi.filePath(), pm);
}
si.setPixmap(pm);
ScoreItem* item = new ScoreItem(si);
item->setTextAlignment(Qt::AlignHCenter | Qt::AlignBottom);
QFont f = item->font();
f.setPixelSize(11);
f.setBold(_boldTitle);
if (fi.completeBaseName() == "00-Blank") {
item->setText(tr("Choose Instruments"));
f.setBold(true);
}
else if (fi.completeBaseName() == "Create_New_Score") {
item->setText(tr("Create New Score…"));
f.setBold(true);
}
else {
QString s(si.completeBaseName());
if (!s.isEmpty() && s[0].isNumber() && _stripNumbers)
s = s.mid(3);
s = s.replace('_', ' ');
item->setText(s);
}
item->setFont(f);
item->setTextAlignment(Qt::AlignHCenter | Qt::AlignTop);
item->setIcon(QIcon(pm));
item->setSizeHint(l->cellSize());
return item;
}
//---------------------------------------------------------
// setScores
//---------------------------------------------------------
void ScoreBrowser::setScores(QFileInfoList& s)
{
qDeleteAll(scoreLists);
scoreLists.clear();
QVBoxLayout* l = static_cast<QVBoxLayout*>(scoreList->layout());
QLayoutItem* child;
while (l->count()) {
child = l->takeAt(0);
if (child->widget() != 0) {
if (child->widget()->objectName() == "noMatchedScoresLabel") // do not delete
continue;
delete child->widget();
}
delete child;
}
ScoreListWidget* sl = 0;
QStringList filter = { "*.mscz", "*.mscx" };
if (_showCustomCategory)
std::sort(s.begin(), s.end(), [](QFileInfo a, QFileInfo b)->bool { return a.fileName() < b.fileName(); });
QSet<QString> entries; //to avoid duplicates
for (const QFileInfo& fil : s) {
if (fil.isDir()) {
QString st(fil.fileName());
if (!st.isEmpty() && st[0].isNumber() && _stripNumbers)
st = st.mid(3);
st = st.replace('_', ' ');
QLabel* label = new QLabel(st);
QFont f = label->font();
f.setBold(true);
label->setFont(f);
static_cast<QVBoxLayout*>(l)->addWidget(label);
QDir dir(fil.filePath());
sl = createScoreList();
l->addWidget(sl);
unsigned count = 0; //nbr of entries added
for (const QFileInfo& fi : dir.entryInfoList(filter, QDir::Files, QDir::Name)) {
if (entries.contains(fi.filePath()))
continue;
sl->addItem(genScoreItem(fi, sl));
count++;
entries.insert(fi.filePath());
}
if (count == 0) {
delete label;
delete sl;
}
sl = 0;
}
}
for (const QFileInfo& fi : s) {
if (fi.isFile()) {
QString st = fi.filePath();
if (entries.contains(st))
continue;
if (st.endsWith(".mscz") || st.endsWith(".mscx")) {
if (!sl) {
if (_showCustomCategory) {
QLabel* label = new QLabel(tr("Custom Templates"));
QFont f = label->font();
f.setBold(true);
label->setFont(f);
l->insertWidget(2,label);
}
sl = createScoreList();
l->insertWidget(3,sl);
}
sl->addItem(genScoreItem(fi, sl));
entries.insert(st);
}
}
}
}
//---------------------------------------------------------
// selectFirst
//---------------------------------------------------------
void ScoreBrowser::selectFirst()
{
if (scoreLists.isEmpty())
return;
ScoreListWidget* w = scoreLists.front();
if (w->count() == 0)
return;
ScoreItem* item = static_cast<ScoreItem*>(w->item(0));
w->setCurrentItem(item);
preview->setScore(item->info());
}
//---------------------------------------------------------
// selectLast
//---------------------------------------------------------
void ScoreBrowser::selectLast()
{
if (scoreLists.isEmpty())
return;
ScoreListWidget* w = scoreLists.front();
if (w->count() == 0)
return;
ScoreItem* item = static_cast<ScoreItem*>(w->item(w->count()-1));
w->setCurrentItem(item);
preview->setScore(item->info());
}
//---------------------------------------------------------
// filter
// filter which scores are visible based on searchString
//---------------------------------------------------------
void ScoreBrowser::filter(const QString &searchString)
{
int numCategoriesWithMathingScores = 0;
for (ScoreListWidget* list : scoreLists) {
int numMatchedScores = 0;
for (int i = 0; i < list->count(); ++i) {
ScoreItem* score = static_cast<ScoreItem*>(list->item(i));
QString title = score->text();
bool isMatch = title.contains(searchString, Qt::CaseInsensitive);
score->setHidden(!isMatch);
if (isMatch)
numMatchedScores++;
}
int listindex = scoreList->layout()->indexOf(list);
QLayoutItem *label = scoreList->layout()->itemAt(listindex - 1);
if (label && label->widget()) {
label->widget()->setHidden(numMatchedScores == 0);
}
list->setHidden(numMatchedScores == 0);
if (numMatchedScores > 0) {
numCategoriesWithMathingScores++;
}
}
_noMatchedScoresLabel->setHidden(numCategoriesWithMathingScores > 0);
}
//---------------------------------------------------------
// scoreClicked
//---------------------------------------------------------
void ScoreBrowser::scoreClicked(QListWidgetItem* current)
{
if (!current)
return;
ScoreItem* item = static_cast<ScoreItem*>(current);
if (!_showPreview)
emit scoreActivated(item->info().filePath());
else {
preview->setScore(item->info());
emit scoreSelected(item->info().filePath());
for (ScoreListWidget* sl : scoreLists) {
if (static_cast<QListWidget*>(sl) != item->listWidget()) {
sl->clearSelection();
}
}
}
}
//---------------------------------------------------------
// setScoreActivated
//---------------------------------------------------------
void ScoreBrowser::setScoreActivated(QListWidgetItem* val)
{
ScoreItem* item = static_cast<ScoreItem*>(val);
emit scoreActivated(item->info().filePath());
}
}