MuseScore/mscore/musescore.cpp

6923 lines
254 KiB
C++
Raw Normal View History

2012-05-26 14:49:10 +02:00
//=============================================================================
// MuseScore
// Music Composition & Notation
//
2016-06-30 15:34:02 +02:00
// Copyright (C) 2002-2016 Werner Schweer
2012-05-26 14:49:10 +02:00
//
// This program is free software; you can redistribute it and/or modify
2012-12-06 14:21:53 +01:00
// 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
2012-05-26 14:49:10 +02:00
//=============================================================================
#include <fenv.h>
#include "loginmanager.h"
#include "uploadscoredialog.h"
#include <QStyleFactory>
#include "palettebox.h"
#include "config.h"
2012-05-26 14:49:10 +02:00
#include "musescore.h"
#include "scoreview.h"
#include "libmscore/style.h"
#include "libmscore/score.h"
#include "instrdialog.h"
#include "preferences.h"
#include "prefsdialog.h"
#include "icons.h"
#include "libmscore/xml.h"
#include "seq.h"
#include "libmscore/tempo.h"
#include "libmscore/sym.h"
#include "pagesettings.h"
2013-06-10 11:03:34 +02:00
#include "debugger/debugger.h"
2012-05-26 14:49:10 +02:00
#include "editstyle.h"
#include "playpanel.h"
#include "libmscore/page.h"
#include "mixer.h"
2014-06-02 12:12:53 +02:00
#include "selectionwindow.h"
2012-05-26 14:49:10 +02:00
#include "palette.h"
2013-02-10 14:50:18 +01:00
#include "palettebox.h"
2012-05-26 14:49:10 +02:00
#include "libmscore/part.h"
#include "libmscore/drumset.h"
#include "libmscore/instrtemplate.h"
#include "libmscore/note.h"
#include "libmscore/staff.h"
#include "driver.h"
#include "libmscore/harmony.h"
#include "magbox.h"
#include "libmscore/sig.h"
#include "libmscore/undo.h"
#include "synthcontrol.h"
#include "pianoroll.h"
#include "drumroll.h"
#include "scoretab.h"
#include "timedialog.h"
#include "keyedit.h"
#include "harmonyedit.h"
#include "navigator.h"
#include "timeline.h"
#include "importmidi/importmidi_panel.h"
#include "importmidi/importmidi_operations.h"
2012-05-26 14:49:10 +02:00
#include "libmscore/chord.h"
#include "libmscore/segment.h"
#include "editraster.h"
#include "pianotools.h"
#include "mediadialog.h"
2013-02-06 16:43:25 +01:00
#include "workspace.h"
2012-05-26 14:49:10 +02:00
#include "selectdialog.h"
#include "selectnotedialog.h"
2012-05-26 14:49:10 +02:00
#include "transposedialog.h"
#include "metaedit.h"
2013-06-10 11:03:34 +02:00
#include "inspector/inspector.h"
2015-02-19 23:17:14 +01:00
#ifdef OMR
2012-05-26 14:49:10 +02:00
#include "omrpanel.h"
2015-02-19 23:17:14 +01:00
#endif
2012-05-26 14:49:10 +02:00
#include "shortcut.h"
2015-01-07 13:47:03 +01:00
#ifdef SCRIPT_INTERFACE
2012-07-02 18:05:10 +02:00
#include "pluginCreator.h"
#include "pluginManager.h"
2015-01-07 13:47:03 +01:00
#endif
#include "helpBrowser.h"
#include "drumtools.h"
#include "editstafftype.h"
2013-02-15 22:39:56 +01:00
#include "texttools.h"
#include "textpalette.h"
#include "resourceManager.h"
#include "scoreaccessibility.h"
#include "startupWizard.h"
2012-05-26 14:49:10 +02:00
#include "libmscore/mscore.h"
#include "libmscore/system.h"
#include "libmscore/measure.h"
2012-05-26 14:49:10 +02:00
#include "libmscore/chordlist.h"
#include "libmscore/volta.h"
2012-07-12 15:35:35 +02:00
#include "libmscore/lasso.h"
#include "libmscore/excerpt.h"
#include "libmscore/synthesizerstate.h"
#include "libmscore/utils.h"
#include "driver.h"
2012-05-26 14:49:10 +02:00
#include "effects/zita1/zita.h"
2015-09-08 15:39:04 +02:00
#include "effects/compressor/compressor.h"
#include "effects/noeffect/noeffect.h"
2013-03-26 19:59:51 +01:00
#include "synthesizer/synthesizer.h"
2013-04-03 12:13:23 +02:00
#include "synthesizer/synthesizergui.h"
#include "synthesizer/msynthesizer.h"
#include "fluid/fluid.h"
#include "qmlplugin.h"
#include "accessibletoolbutton.h"
#include "toolbuttonmenu.h"
2015-02-01 20:50:01 +01:00
#include "searchComboBox.h"
2014-11-12 16:36:40 +01:00
#include "startcenter.h"
2015-02-02 09:30:17 +01:00
#include "help.h"
#include "awl/aslider.h"
#include "extension.h"
#include "thirdparty/qzip/qzipreader_p.h"
2014-11-12 16:36:40 +01:00
2017-05-16 05:01:10 +02:00
#ifdef USE_LAME
#include "exportmp3.h"
#endif
#ifdef Q_OS_MAC
#include "macos/cocoabridge.h"
#endif
2013-05-13 18:49:17 +02:00
#ifdef AEOLUS
extern Ms::Synthesizer* createAeolus();
#endif
#ifdef ZERBERUS
extern Ms::Synthesizer* createZerberus();
#endif
#ifdef QT_NO_DEBUG
Q_LOGGING_CATEGORY(undoRedo, "undoRedo", QtCriticalMsg)
#else
Q_LOGGING_CATEGORY(undoRedo, "undoRedo", QtCriticalMsg)
// Q_LOGGING_CATEGORY(undoRedo, "undoRedo")
#endif
2013-05-13 18:49:17 +02:00
namespace Ms {
2012-05-26 14:49:10 +02:00
MuseScore* mscore;
2013-03-26 19:59:51 +01:00
MasterSynthesizer* synti;
2012-05-26 14:49:10 +02:00
bool enableExperimental = false;
QString dataPath;
QString iconPath;
2012-05-26 14:49:10 +02:00
bool converterMode = false;
bool processJob = false;
2012-05-26 14:49:10 +02:00
bool externalIcons = false;
bool pluginMode = false;
2012-05-26 14:49:10 +02:00
static bool startWithNewScore = false;
double guiScaling = 0.0;
static double userDPI = 0.0;
int trimMargin = -1;
bool noWebView = false;
bool exportScoreParts = false;
bool ignoreWarnings = false;
2012-05-26 14:49:10 +02:00
QString mscoreGlobalShare;
2012-05-26 14:49:10 +02:00
static QString outFileName;
static QString jsonFileName;
static QString audioDriver;
2012-05-26 14:49:10 +02:00
static QString pluginName;
static QString styleFile;
static QString extensionName;
static bool scoresOnCommandline { false };
2016-06-30 15:34:02 +02:00
static QList<QTranslator*> translatorList;
QString localeName;
2012-05-26 14:49:10 +02:00
bool useFactorySettings = false;
bool deletePreferences = false;
2012-05-26 14:49:10 +02:00
QString styleName;
QString revision;
2013-07-17 14:36:34 +02:00
QErrorMessage* errorMessage;
const char* voiceActions[] = { "voice-1", "voice-2", "voice-3", "voice-4" };
2012-05-26 14:49:10 +02:00
2016-05-31 17:44:39 +02:00
const std::list<const char*> MuseScore::_allNoteInputMenuEntries {
"note-input",
"pad-note-128",
"pad-note-64",
"pad-note-32",
"pad-note-16",
"pad-note-8",
"pad-note-4",
"pad-note-2",
"pad-note-1",
"note-breve",
"note-longa",
"pad-dot",
"pad-dotdot",
"pad-dot3",
"pad-dot4",
2016-05-31 17:44:39 +02:00
"tie",
"",
"pad-rest",
"",
"sharp2",
"sharp",
"nat",
"flat",
"flat2",
"flip",
"",
"voice-1",
"voice-2",
"voice-3",
"voice-4"
};
const std::list<const char*> MuseScore::_advancedNoteInputMenuEntries {
"note-input",
"pad-note-128",
"pad-note-64",
"pad-note-32",
"pad-note-16",
"pad-note-8",
"pad-note-4",
"pad-note-2",
"pad-note-1",
"note-breve",
"note-longa",
"pad-dot",
"pad-dotdot",
"pad-dot3",
"pad-dot4",
2016-05-31 17:44:39 +02:00
"tie",
"",
"pad-rest",
"",
"sharp2",
"sharp",
"nat",
"flat",
"flat2",
"flip",
"",
"voice-1",
"voice-2",
"voice-3",
"voice-4"
};
const std::list<const char*> MuseScore::_basicNoteInputMenuEntries {
"note-input",
"pad-note-32",
"pad-note-16",
"pad-note-8",
"pad-note-4",
"pad-note-2",
"pad-note-1",
"pad-dot",
"tie",
"",
"pad-rest",
"",
"sharp",
"nat",
"flat",
"flip",
"",
"voice-1",
"voice-2",
"voice-3",
"voice-4"
};
const std::list<const char*> MuseScore::_allFileOperationEntries {
"file-new",
"file-open",
"file-save",
"print",
"redo",
"undo"
};
const std::list<const char*> MuseScore::_allPlaybackControlEntries {
"midi-on",
"rewind",
"play",
"loop",
"repeat",
"pan",
"metronome"
};
2016-05-31 17:44:39 +02:00
2014-07-13 18:33:26 +02:00
extern bool savePositions(Score*, const QString& name, bool segments );
2013-02-25 21:57:11 +01:00
extern TextPalette* textPalette;
2012-05-26 14:49:10 +02:00
2015-11-23 16:02:33 +01:00
static constexpr double SCALE_MAX = 16.0;
static constexpr double SCALE_MIN = 0.05;
static constexpr double SCALE_STEP = 1.7;
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// cmdInsertMeasure
//---------------------------------------------------------
void MuseScore::cmdInsertMeasures()
{
if (cs) {
if (cs->selection().isNone() && !cs->selection().findMeasure()) {
QMessageBox::warning(0, "MuseScore",
tr("No measure selected:\n" "Please select a measure and try again"));
}
else {
insertMeasuresDialog = new InsertMeasuresDialog;
insertMeasuresDialog->show();
}
}
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// InsertMeasuresDialog
//---------------------------------------------------------
InsertMeasuresDialog::InsertMeasuresDialog(QWidget* parent)
: QDialog(parent)
{
setObjectName("InsertMeasuresDialog");
setupUi(this);
setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
setModal(true);
insmeasures->selectAll();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// Insert Measure --> accept
//---------------------------------------------------------
void InsertMeasuresDialog::accept()
{
2013-04-28 21:29:12 +02:00
int n = insmeasures->value();
if (mscore->currentScore())
2017-01-18 14:16:33 +01:00
mscore->currentScoreView()->cmdInsertMeasures(n, ElementType::MEASURE);
2013-04-28 21:29:12 +02:00
done(1);
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// InsertMeasuresDialog hideEvent
//---------------------------------------------------------
void InsertMeasuresDialog::hideEvent(QHideEvent* event)
{
MuseScore::saveGeometry(this);
QDialog::hideEvent(event);
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// getSharePath
//---------------------------------------------------------
2015-02-02 09:30:17 +01:00
QString getSharePath()
2012-05-26 14:49:10 +02:00
{
#ifdef Q_OS_WIN
2012-05-26 14:49:10 +02:00
QDir dir(QCoreApplication::applicationDirPath() + QString("/../" INSTALL_NAME));
return dir.absolutePath() + "/";
#else
#ifdef Q_OS_MAC
2012-05-26 14:49:10 +02:00
QDir dir(QCoreApplication::applicationDirPath() + QString("/../Resources"));
return dir.absolutePath() + "/";
#else
// Try relative path (needed for portable AppImage and non-standard installations)
QDir dir(QCoreApplication::applicationDirPath() + QString("/../share/" INSTALL_NAME));
if (dir.exists())
return dir.absolutePath() + "/";
// Otherwise fall back to default location (e.g. if binary has moved relative to share)
2012-05-26 14:49:10 +02:00
return QString( INSTPREFIX "/share/" INSTALL_NAME);
#endif
#endif
}
//---------------------------------------------------------
// printVersion
//---------------------------------------------------------
static void printVersion(const char* prog)
{
if (MuseScore::unstable())
fprintf(stderr, "%s: Music Score Editor\nUnstable Prerelease for Version %s; Build %s\n",
2012-05-26 14:49:10 +02:00
prog, VERSION, qPrintable(revision));
else
fprintf(stderr, "%s: Music Score Editor; Version %s; Build %s\n", prog, VERSION, qPrintable(revision));
2012-05-26 14:49:10 +02:00
}
2014-11-13 09:52:12 +01:00
static const int RECENT_LIST_SIZE = 20;
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// closeEvent
//---------------------------------------------------------
void MuseScore::closeEvent(QCloseEvent* ev)
{
unloadPlugins();
2016-03-11 12:18:46 +01:00
QList<MasterScore*> removeList;
for (MasterScore* score : scoreList) {
2012-05-26 14:49:10 +02:00
if (score->created() && !score->dirty())
removeList.append(score);
else {
if (checkDirty(score)) { // ask user if file is dirty
ev->ignore();
return;
}
//
// if score is still dirty, then the user has discarded the
// score and we can remove it from the list
//
if (score->created() && score->dirty())
removeList.append(score);
}
}
// remove all new created/not save score so they are
// note saved as session data
2016-03-11 12:18:46 +01:00
for (MasterScore* score : removeList)
2012-05-26 14:49:10 +02:00
scoreList.removeAll(score);
writeSessionFile(true);
2016-03-11 12:18:46 +01:00
for (MasterScore* score : scoreList) {
2012-05-26 14:49:10 +02:00
if (!score->tmpName().isEmpty()) {
QFile f(score->tmpName());
f.remove();
}
}
writeSettings();
2014-11-17 13:06:34 +01:00
_loginManager->save();
2013-04-12 10:56:17 +02:00
2012-05-26 14:49:10 +02:00
ev->accept();
if (Shortcut::dirty)
Shortcut::save();
this->deleteLater(); //this is necessary on windows http://musescore.org/node/16713
2012-05-26 14:49:10 +02:00
qApp->quit();
}
void updateExternalValuesFromPreferences() {
// set values in libmscore
MScore::bgColor = preferences.getColor(PREF_UI_CANVAS_BG_COLOR);
MScore::dropColor = preferences.getColor(PREF_UI_SCORE_NOTE_DROPCOLOR);
MScore::defaultColor = preferences.getColor(PREF_UI_SCORE_DEFAULTCOLOR);
MScore::defaultPlayDuration = preferences.getInt(PREF_SCORE_NOTE_DEFAULTPLAYDURATION);
MScore::panPlayback = preferences.getBool(PREF_APP_PLAYBACK_PANPLAYBACK);
MScore::playRepeats = preferences.getBool(PREF_APP_PLAYBACK_PLAYREPEATS);
MScore::warnPitchRange = preferences.getBool(PREF_SCORE_NOTE_WARNPITCHRANGE);
MScore::layoutBreakColor = preferences.getColor(PREF_UI_SCORE_LAYOUTBREAKCOLOR);
MScore::frameMarginColor = preferences.getColor(PREF_UI_SCORE_FRAMEMARGINCOLOR);
MScore::setVerticalOrientation(preferences.getBool(PREF_UI_CANVAS_SCROLL_VERTICALORIENTATION));
MScore::selectColor[0] = preferences.getColor(PREF_UI_SCORE_VOICE1_COLOR);
MScore::selectColor[1] = preferences.getColor(PREF_UI_SCORE_VOICE2_COLOR);
MScore::selectColor[2] = preferences.getColor(PREF_UI_SCORE_VOICE3_COLOR);
MScore::selectColor[3] = preferences.getColor(PREF_UI_SCORE_VOICE4_COLOR);
MScore::setHRaster(preferences.getInt(PREF_UI_APP_RASTER_HORIZONTAL));
MScore::setVRaster(preferences.getInt(PREF_UI_APP_RASTER_VERTICAL));
MScore::setNudgeStep(.1); // cursor key (default 0.1)
MScore::setNudgeStep10(1.0); // Ctrl + cursor key (default 1.0)
MScore::setNudgeStep50(0.01); // Alt + cursor key (default 0.01)
if (!preferences.getBool(PREF_APP_STARTUP_FIRSTSTART)) {
//Create directories if they are missing
QDir dir;
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYSCORES));
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYSTYLES));
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYIMAGES));
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYTEMPLATES));
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS));
dir.mkpath(preferences.getString(PREF_APP_PATHS_MYPLUGINS));
foreach (QString path, preferences.getString(PREF_APP_PATHS_MYSOUNDFONTS).split(";"))
dir.mkpath(path);
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// preferencesChanged
//---------------------------------------------------------
void MuseScore::preferencesChanged()
{
updateExternalValuesFromPreferences();
setPlayRepeats(MScore::playRepeats);
getAction("pan")->setChecked(MScore::panPlayback);
getAction("follow")->setChecked(preferences.getBool(PREF_APP_PLAYBACK_FOLLOWSONG));
getAction("midi-on")->setEnabled(preferences.getBool(PREF_IO_MIDI_ENABLEINPUT));
getAction("toggle-statusbar")->setChecked(preferences.getBool(PREF_UI_APP_SHOWSTATUSBAR));
_statusBar->setVisible(preferences.getBool(PREF_UI_APP_SHOWSTATUSBAR));
MuseScore::updateUiStyleAndTheme();
QString fgWallpaper = preferences.getString(PREF_UI_CANVAS_FG_WALLPAPER);
2012-05-26 14:49:10 +02:00
for (int i = 0; i < tab1->count(); ++i) {
ScoreView* canvas = tab1->view(i);
if (canvas == 0)
continue;
if (preferences.getBool(PREF_UI_CANVAS_BG_USECOLOR))
canvas->setBackground(preferences.getColor(PREF_UI_CANVAS_BG_COLOR));
2012-05-26 14:49:10 +02:00
else {
QPixmap* pm = new QPixmap(preferences.getString(PREF_UI_CANVAS_BG_WALLPAPER));
2012-05-26 14:49:10 +02:00
canvas->setBackground(pm);
}
if (preferences.getBool(PREF_UI_CANVAS_FG_USECOLOR))
canvas->setForeground(preferences.getColor(PREF_UI_CANVAS_FG_COLOR));
2012-05-26 14:49:10 +02:00
else {
QPixmap* pm = new QPixmap(fgWallpaper);
2012-05-26 14:49:10 +02:00
if (pm == 0 || pm->isNull())
qDebug("no valid pixmap %s", fgWallpaper.toLatin1().data());
2012-05-26 14:49:10 +02:00
canvas->setForeground(pm);
}
}
if (tab2) {
for (int i = 0; i < tab2->count(); ++i) {
ScoreView* canvas = tab2->view(i);
if (canvas == 0)
continue;
if (preferences.getBool(PREF_UI_CANVAS_BG_USECOLOR))
canvas->setBackground(preferences.getColor(PREF_UI_CANVAS_BG_COLOR));
2012-05-26 14:49:10 +02:00
else {
QPixmap* pm = new QPixmap(preferences.getString(PREF_UI_CANVAS_BG_WALLPAPER));
2012-05-26 14:49:10 +02:00
canvas->setBackground(pm);
}
if (preferences.getBool(PREF_UI_CANVAS_FG_USECOLOR))
canvas->setForeground(preferences.getColor(PREF_UI_CANVAS_FG_COLOR));
2012-05-26 14:49:10 +02:00
else {
QPixmap* pm = new QPixmap(fgWallpaper);
2012-05-26 14:49:10 +02:00
if (pm == 0 || pm->isNull())
qDebug("no valid pixmap %s", fgWallpaper.toLatin1().data());
2012-05-26 14:49:10 +02:00
canvas->setForeground(pm);
}
}
}
transportTools->setEnabled(!noSeq && seq && seq->isRunning());
playId->setEnabled(!noSeq && seq && seq->isRunning());
2012-05-26 14:49:10 +02:00
// change workspace
if (preferences.getString(PREF_APP_WORKSPACE) != Workspace::currentWorkspace->name()) {
Workspace* workspace = 0;
for (Workspace* w: Workspace::workspaces()) {
if (w->name() == preferences.getString(PREF_APP_WORKSPACE)) {
workspace = w;
break;
}
}
if (workspace != 0) {
mscore->changeWorkspace(workspace);
mscore->getPaletteBox()->updateWorkspaces();
}
}
delete newWizard;
newWizard = 0;
reloadInstrumentTemplates();
updateInstrumentDialog();
2012-05-26 14:49:10 +02:00
}
2016-05-31 17:44:39 +02:00
//---------------------------------------------------------
// populateNoteInputMenu
//---------------------------------------------------------
void MuseScore::populateNoteInputMenu()
{
entryTools->clear();
for (const auto s : _noteInputMenuEntries) {
if (!*s)
entryTools->addSeparator();
else {
QAction* a = getAction(s);
2016-08-03 17:29:52 +02:00
QWidget* w;
if (strcmp(s, "note-input") == 0) {
//-----------------------------------------------------------------
// Note Entry Modes menu
// ToolButtonMenu to swap between Note Entry Methods
//-----------------------------------------------------------------
QActionGroup* noteEntryMethods = new QActionGroup(entryTools);
noteEntryMethods->addAction(getAction("note-input-steptime"));
noteEntryMethods->addAction(getAction("note-input-repitch"));
noteEntryMethods->addAction(getAction("note-input-rhythm"));
noteEntryMethods->addAction(getAction("note-input-realtime-auto"));
noteEntryMethods->addAction(getAction("note-input-realtime-manual"));
connect(noteEntryMethods, SIGNAL(triggered(QAction*)), this, SLOT(cmd(QAction*)));
2016-08-03 17:29:52 +02:00
w = new ToolButtonMenu(tr("Note Entry Methods"),
ToolButtonMenu::TYPES::ICON_CHANGED,
getAction("note-input"),
noteEntryMethods,
this);
}
else if (strncmp(s, "voice-", 6) == 0) {
2016-06-03 13:40:18 +02:00
AccessibleToolButton* tb = new AccessibleToolButton(this, a);
tb->setObjectName("voice");
2016-05-31 17:44:39 +02:00
tb->setFocusPolicy(Qt::ClickFocus);
tb->setToolButtonStyle(Qt::ToolButtonTextOnly);
tb->setProperty((QString("voice%1").arg(atoi(s+6))).toUtf8().data(), true);
2016-05-31 17:44:39 +02:00
QPalette p(tb->palette());
int i = atoi(s+6) - 1;
2016-05-31 17:44:39 +02:00
p.setColor(QPalette::Base, MScore::selectColor[i]);
tb->setPalette(p);
a->setCheckable(true);
2016-06-03 13:40:18 +02:00
// tb->setDefaultAction(a);
2016-08-03 17:29:52 +02:00
w = tb;
2016-05-31 17:44:39 +02:00
}
else
2016-08-03 17:29:52 +02:00
w = new AccessibleToolButton(entryTools, a);
entryTools->addWidget(w);
2016-05-31 17:44:39 +02:00
}
}
}
//---------------------------------------------------------
// onLongOperationFinished
//---------------------------------------------------------
void MuseScore::onLongOperationFinished()
{
infoMsgBox->accept();
}
//---------------------------------------------------------
// importExtension
//---------------------------------------------------------
bool MuseScore::importExtension(QString path)
{
MQZipReader zipFile(path);
// compute total unzipped size
qint64 totalZipSize = 0;
for (auto fi : zipFile.fileInfoList())
totalZipSize += fi.size;
// check if extension path is writable and has enough space
QStorageInfo storage = QStorageInfo(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS));
if (storage.isReadOnly()) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Cannot import extension on read-only storage: %1").arg(storage.displayName()));
return false;
}
if (totalZipSize >= storage.bytesAvailable()) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Cannot import extension: storage %1 is full").arg(storage.displayName()));
return false;
}
// Check structure of the extension
bool hasMetadata = false;
bool hasAlienDirectory = false;
bool hasAlienFiles = false;
QSet<QString> acceptableFolders = { Extension::sfzsDir, Extension::soundfontsDir, Extension::templatesDir, Extension::instrumentsDir, Extension::workspacesDir };
for (auto fi : zipFile.fileInfoList()) {
if (fi.filePath == "metadata.json")
hasMetadata = true;
else {
// get folders
auto fpath = QDir::cleanPath(fi.filePath);
QStringList folders(fpath);
while ((fpath = QFileInfo(fpath).path()).length() < folders.last().length())
folders << fpath;
if (folders.size() < 2) {
hasAlienFiles = true; // in root dir
break;
}
QString rootDir = folders.at(folders.size() - 2);
if (!acceptableFolders.contains(rootDir)) {
hasAlienDirectory = true; // in root dir
break;
}
}
}
if (!hasMetadata) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Corrupted extension: no metadata.json"));
return false;
}
if (hasAlienDirectory) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Corrupted extension: unsupported directories in root directory"));
return false;
}
if (hasAlienFiles) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Corrupted extension: unsupported files in root directory"));
return false;
}
zipFile.close();
MQZipReader zipFile2(path);
// get extension id from metadata.json
QByteArray mdba = zipFile2.fileData("metadata.json");
zipFile2.close();
QJsonDocument loadDoc = QJsonDocument::fromJson(mdba);
QJsonObject mdObject = loadDoc.object();
QString extensionId = mdObject["id"].toString();
QString version = mdObject["version"].toString();
if (extensionId.isEmpty() || version.isEmpty()) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Corrupted extension: corrupted metadata.json"));
return false;
}
// Check if extension is already installed, ask for uninstall
QDir dir(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS));
auto dirList = dir.entryList(QStringList(extensionId), QDir::Dirs | QDir::NoDotAndDotDot);
bool newerVersion = false;
if (dirList.contains(extensionId)) {
QString extDirName = QString("%1/%2").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId);
QDir extDir(extDirName);
auto versionDirList = extDir.entryList(QDir::Dirs | QDir::NoDotAndDotDot);
if (versionDirList.size() > 0) {
// potentially other versions
// is there a more recent version?
for (auto versionDir : versionDirList) {
if (compareVersion(version, versionDir)) {
qDebug() << "There is a newer version. We don't install";
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("A newer version is already installed"));
newerVersion = true;
return false;
}
}
}
if (!newerVersion) {
qDebug() << "found already install extension without newer version: deleting it";
QDir d(QString("%1/%2").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId));
if (!d.removeRecursively()) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Error while deleting previous version of the extension: %1").arg(extensionId));
return false;
}
}
}
//setup the message box
infoMsgBox = new QMessageBox();
infoMsgBox->setWindowModality(Qt::ApplicationModal);
infoMsgBox->setWindowFlags(Qt::WindowFlags(Qt::Dialog | Qt::FramelessWindowHint | Qt::WindowTitleHint));
infoMsgBox->setTextFormat(Qt::RichText);
infoMsgBox->setMinimumSize(300, 100);
infoMsgBox->setMaximumSize(300, 100);
infoMsgBox->setStandardButtons(0);
infoMsgBox->setText(QString("<p align='center'>") + tr("Please wait, unpacking extension...") + QString("</p>"));
//setup async run of long operations
QFutureWatcher<bool> futureWatcherUnzip;
connect(&futureWatcherUnzip, SIGNAL(finished()), this, SLOT(onLongOperationFinished()));
MQZipReader* zipFile3 = new MQZipReader(path);
// Unzip the extension asynchronously
QFuture<bool> futureUnzip = QtConcurrent::run(zipFile3, &MQZipReader::extractAll, QString("%1/%2/%3").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version));
futureWatcherUnzip.setFuture(futureUnzip);
if (!MScore::noGui)
infoMsgBox->exec();
bool unzipResult = futureUnzip.result();
zipFile3->close();
delete zipFile3;
zipFile3 = nullptr;
if (!unzipResult) {
if (!MScore::noGui)
QMessageBox::critical(mscore, QWidget::tr("Import Extension File"), QWidget::tr("Unable to extract files from the extension"));
return false;
}
delete newWizard;
newWizard = 0;
mscore->reloadInstrumentTemplates();
mscore->updateInstrumentDialog();
auto loadSoundFontAsync = [&]() {
// After install: add sfz to zerberus
QDir sfzDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::sfzsDir));
if (sfzDir.exists()) {
// get all sfz files
QDirIterator it(sfzDir.absolutePath(), QStringList("*.sfz"), QDir::Files, QDirIterator::Subdirectories);
Synthesizer* s = synti->synthesizer("Zerberus");
QStringList sfzs;
while (it.hasNext()) {
it.next();
sfzs.append(it.fileName());
}
sfzs.sort();
for (int sfzNum = 0; sfzNum < sfzs.size(); ++sfzNum)
s->addSoundFont(sfzs[sfzNum]);
if (!sfzs.isEmpty())
synti->storeState();
}
// After install: add soundfont to fluid
QDir sfDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::soundfontsDir));
if (sfDir.exists()) {
// get all soundfont files
QStringList filters("*.sf2");
filters.append("*.sf3");
QDirIterator it(sfzDir.absolutePath(), filters, QDir::Files, QDirIterator::Subdirectories);
Synthesizer* s = synti->synthesizer("Fluid");
QStringList sfs;
while (it.hasNext()) {
it.next();
sfs.append(it.fileName());
}
sfs.sort();
for (auto sf : sfs)
s->addSoundFont(sf);
if (!sfs.isEmpty())
synti->storeState();
}
};
if (!enableExperimental) {
//load soundfonts async
QFuture<void> futureLoadSFs = QtConcurrent::run(loadSoundFontAsync);
QFutureWatcher<void> futureWatcherLoadSFs;
futureWatcherLoadSFs.setFuture(futureLoadSFs);
connect(&futureWatcherLoadSFs, SIGNAL(finished()), this, SLOT(onLongOperationFinished()));
infoMsgBox->setText(QString("<p align='center'>") + tr("Please wait, loading soundfonts...") + QString("</p>"));
if (!MScore::noGui)
infoMsgBox->exec();
else
futureLoadSFs.waitForFinished();
}
// after install: refresh workspaces if needed
QDir workspacesDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::workspacesDir));
if (workspacesDir.exists() && !MScore::noGui) {
auto wsList = workspacesDir.entryInfoList(QStringList("*.workspace"), QDir::Files);
if (!wsList.isEmpty()) {
Workspace::refreshWorkspaces();
paletteBox->updateWorkspaces();
paletteBox->selectWorkspace(wsList.last().absoluteFilePath());
}
}
return true;
}
//---------------------------------------------------------
// uninstallExtension
//---------------------------------------------------------
bool MuseScore::uninstallExtension(QString extensionId)
{
QString version = Extension::getLatestVersion(extensionId);
// Before install: remove sfz from zerberus
QDir sfzDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::sfzsDir));
if (sfzDir.exists()) {
// get all sfz files
QDirIterator it(sfzDir.absolutePath(), QStringList("*.sfz"), QDir::Files, QDirIterator::Subdirectories);
Synthesizer* s = synti->synthesizer("Zerberus");
bool found = it.hasNext();
while (it.hasNext()) {
it.next();
s->removeSoundFont(it.fileInfo().absoluteFilePath());
}
if (found)
synti->storeState();
}
// Before install: remove soundfont from fluid
QDir sfDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::soundfontsDir));
if (sfDir.exists()) {
// get all soundfont files
QStringList filters("*.sf2");
filters.append("*.sf3");
QDirIterator it(sfzDir.absolutePath(), filters, QDir::Files, QDirIterator::Subdirectories);
Synthesizer* s = synti->synthesizer("Fluid");
bool found = it.hasNext();
while (it.hasNext()) {
it.next();
s->removeSoundFont(it.fileInfo().absoluteFilePath());
}
if (found)
synti->storeState();
}
bool refreshWorkspaces = false;
QDir workspacesDir(QString("%1/%2/%3/%4").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId).arg(version).arg(Extension::workspacesDir));
if (workspacesDir.exists()) {
auto wsList = workspacesDir.entryInfoList(QStringList("*.workspace"), QDir::Files);
if (!wsList.isEmpty())
refreshWorkspaces = true;
}
// delete directories
QDir extensionDir(QString("%1/%2").arg(preferences.getString(PREF_APP_PATHS_MYEXTENSIONS)).arg(extensionId));
extensionDir.removeRecursively();
// update UI
delete newWizard;
newWizard = 0;
mscore->reloadInstrumentTemplates();
mscore->updateInstrumentDialog();
if (refreshWorkspaces) {
Workspace::refreshWorkspaces();
paletteBox->updateWorkspaces();
paletteBox->selectWorkspace("Basic");
}
return true;
}
//---------------------------------------------------------
// isInstalled
/// used from javascript in start center webview
//---------------------------------------------------------
bool MuseScore::isInstalledExtension(QString extensionId)
{
return Extension::isInstalled(extensionId);
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// MuseScore
//---------------------------------------------------------
MuseScore::MuseScore()
: QMainWindow()
{
2017-05-02 14:17:31 +02:00
QScreen* screen = QGuiApplication::primaryScreen();
if (userDPI == 0.0) {
#if defined(Q_OS_WIN)
if (QSysInfo::WindowsVersion <= QSysInfo::WV_WINDOWS7)
_physicalDotsPerInch = screen->logicalDotsPerInch() * screen->devicePixelRatio();
else
_physicalDotsPerInch = screen->physicalDotsPerInch(); // physical resolution
#else
2015-11-16 14:24:47 +01:00
_physicalDotsPerInch = screen->physicalDotsPerInch(); // physical resolution
#endif
}
else {
_physicalDotsPerInch = userDPI;
}
if (guiScaling == 0.0) {
// set scale for icons, palette elements, window sizes, etc
// the default values are hard coded in pixel sizes and assume ~96 DPI
if (qAbs(_physicalDotsPerInch - DPI_DISPLAY) > 6.0)
guiScaling = _physicalDotsPerInch / DPI_DISPLAY;
else
guiScaling = 1.0;
}
2015-11-16 14:24:47 +01:00
2016-09-03 17:45:59 +02:00
MScore::pixelRatio = DPI / screen->logicalDotsPerInch();
setObjectName("MuseScore");
2012-05-26 14:49:10 +02:00
_sstate = STATE_INIT;
setWindowTitle(QString(MUSESCORE_NAME_VERSION));
setIconSize(QSize(preferences.getInt(PREF_UI_THEME_ICONWIDTH) * guiScaling, preferences.getInt(PREF_UI_THEME_ICONHEIGHT) * guiScaling));
2012-05-26 14:49:10 +02:00
ucheck = new UpdateChecker(this);
packUChecker = new ExtensionsUpdateChecker(this);
2012-05-26 14:49:10 +02:00
setAcceptDrops(true);
setFocusPolicy(Qt::NoFocus);
2012-05-26 14:49:10 +02:00
#ifdef SCRIPT_INTERFACE
pluginManager = new PluginManager(0);
#endif
if (!converterMode && !pluginMode) {
_loginManager = new LoginManager(this);
#if 0
// initialize help engine
QString lang = mscore->getLocaleISOCode();
if (lang == "en_US") // HACK
lang = "en";
QString s = getSharePath() + "manual/doc_" + lang + ".qhc";
_helpEngine = new QHelpEngine(s, this);
if (!_helpEngine->setupData()) {
qDebug("cannot setup data for help engine: %s", qPrintable(_helpEngine->error()));
delete _helpEngine;
_helpEngine = 0;
}
#endif
}
2012-05-26 14:49:10 +02:00
_positionLabel = new QLabel;
_positionLabel->setObjectName("decoration widget"); // this prevents animations
_modeText = new QLabel;
2013-07-04 21:07:38 +02:00
_modeText->setAutoFillBackground(false);
_modeText->setObjectName("modeLabel");
2014-11-14 11:44:48 +01:00
hRasterAction = getAction("hraster");
vRasterAction = getAction("vraster");
loopAction = getAction("loop");
loopInAction = getAction("loop-in");
loopOutAction = getAction("loop-out");
2012-05-26 14:49:10 +02:00
metronomeAction = getAction("metronome");
2014-11-14 11:44:48 +01:00
countInAction = getAction("countin");
panAction = getAction("pan");
2017-05-02 14:17:31 +02:00
_statusBar = new QStatusBar;
2012-05-26 14:49:10 +02:00
_statusBar->addPermanentWidget(new QWidget(this), 2);
_statusBar->addPermanentWidget(new QWidget(this), 100);
_statusBar->addPermanentWidget(_modeText, 0);
if (enableExperimental) {
layerSwitch = new QComboBox(this);
2014-10-15 20:43:06 +02:00
layerSwitch->setToolTip(tr("Switch layer"));
connect(layerSwitch, SIGNAL(activated(const QString&)), SLOT(switchLayer(const QString&)));
playMode = new QComboBox(this);
2016-06-30 15:34:02 +02:00
playMode->addItem(tr("Synthesizer"));
playMode->addItem(tr("Audio track"));
2014-10-15 20:43:06 +02:00
playMode->setToolTip(tr("Switch play mode"));
connect(playMode, SIGNAL(activated(int)), SLOT(switchPlayMode(int)));
_statusBar->addPermanentWidget(playMode);
_statusBar->addPermanentWidget(layerSwitch);
}
2017-05-02 14:17:31 +02:00
2012-05-26 14:49:10 +02:00
_statusBar->addPermanentWidget(_positionLabel, 0);
setStatusBar(_statusBar);
ScoreAccessibility::createInstance(this);
2012-05-26 14:49:10 +02:00
// otherwise unused actions:
// must be added somewere to work
QActionGroup* ag = Shortcut::getActionGroupForWidget(MsWidget::MAIN_WINDOW);
ag->setParent(this);
2014-11-14 11:44:48 +01:00
addActions(ag->actions());
2012-05-26 14:49:10 +02:00
connect(ag, SIGNAL(triggered(QAction*)), SLOT(cmd(QAction*)));
mainWindow = new QSplitter;
2013-05-08 10:54:06 +02:00
mainWindow->setChildrenCollapsible(false);
2012-05-26 14:49:10 +02:00
QWidget* mainScore = new QWidget;
mainScore->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
mainWindow->addWidget(mainScore);
layout = new QVBoxLayout;
layout->setMargin(0);
layout->setSpacing(0);
mainScore->setLayout(layout);
_navigator = new NScrollArea;
_navigator->setFocusPolicy(Qt::NoFocus);
2012-05-26 14:49:10 +02:00
mainWindow->addWidget(_navigator);
scorePageLayoutChanged();
showNavigator(preferences.getBool(PREF_UI_APP_STARTUP_SHOWNAVIGATOR));
2012-05-26 14:49:10 +02:00
_timeline = new TDockWidget;
_timeline->setFocusPolicy(Qt::NoFocus);
addDockWidget(Qt::BottomDockWidgetArea, _timeline);
scorePageLayoutChanged();
showTimeline(false);
2013-05-09 18:46:51 +02:00
mainWindow->setStretchFactor(0, 1);
mainWindow->setStretchFactor(1, 0);
mainWindow->setSizes(QList<int>({500, 50}));
2013-05-09 18:46:51 +02:00
QSplitter* envelope = new QSplitter;
envelope->setChildrenCollapsible(false);
envelope->setOrientation(Qt::Vertical);
envelope->addWidget(mainWindow);
2015-01-13 17:27:58 +01:00
importmidiPanel = new ImportMidiPanel(this);
importmidiPanel->setVisible(false);
envelope->addWidget(importmidiPanel);
{
importmidiShowPanel = new QFrame;
QHBoxLayout *hl = new QHBoxLayout;
hl->setMargin(0);
hl->setSpacing(0);
importmidiShowPanel->setLayout(hl);
2016-06-30 15:34:02 +02:00
showMidiImportButton = new QPushButton();
showMidiImportButton->setFocusPolicy(Qt::ClickFocus);
importmidiShowPanel->setVisible(false);
2016-06-30 15:34:02 +02:00
connect(showMidiImportButton, SIGNAL(clicked()), SLOT(showMidiImportPanel()));
connect(importmidiPanel, SIGNAL(closeClicked()), importmidiShowPanel, SLOT(show()));
2016-06-30 15:34:02 +02:00
hl->addWidget(showMidiImportButton);
QSpacerItem *item = new QSpacerItem(1, 1, QSizePolicy::Expanding, QSizePolicy::Fixed);
hl->addSpacerItem(item);
envelope->addWidget(importmidiShowPanel);
}
2015-01-13 17:27:58 +01:00
envelope->setSizes(QList<int>({550, 180}));
2012-05-26 14:49:10 +02:00
splitter = new QSplitter;
tab1 = new ScoreTab(&scoreList, this);
2012-05-26 14:49:10 +02:00
tab1->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
connect(tab1, SIGNAL(currentScoreViewChanged(ScoreView*)), SLOT(setCurrentScoreView(ScoreView*)));
connect(tab1, SIGNAL(tabCloseRequested(int)), SLOT(removeTab(int)));
connect(tab1, SIGNAL(actionTriggered(QAction*)), SLOT(cmd(QAction*)));
2012-05-26 14:49:10 +02:00
splitter->addWidget(tab1);
if (splitScreen()) {
tab2 = new ScoreTab(&scoreList, this);
2012-05-26 14:49:10 +02:00
tab2->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
connect(tab2, SIGNAL(currentScoreViewChanged(ScoreView*)), SLOT(setCurrentScoreView(ScoreView*)));
connect(tab2, SIGNAL(tabCloseRequested(int)), SLOT(removeTab(int)));
connect(tab2, SIGNAL(actionTriggered(QAction*)), SLOT(cmd(QAction*)));
2012-05-26 14:49:10 +02:00
splitter->addWidget(tab2);
tab2->setVisible(false);
}
else
tab2 = 0;
layout->addWidget(splitter);
//---------------------------------------------------
// Transport Action
//---------------------------------------------------
QAction* a;
#ifdef HAS_MIDI
a = getAction("midi-on");
a->setEnabled(preferences.getBool(PREF_IO_MIDI_ENABLEINPUT));
2012-05-26 14:49:10 +02:00
a->setChecked(_midiinEnabled);
#endif
2014-11-14 11:44:48 +01:00
getAction("undo")->setEnabled(false);
getAction("redo")->setEnabled(false);
getAction("paste")->setEnabled(false);
getAction("swap")->setEnabled(false);
2014-11-14 11:44:48 +01:00
selectionChanged(SelState::NONE);
2012-05-26 14:49:10 +02:00
//---------------------------------------------------
2016-06-03 13:40:18 +02:00
// File Tool Bar
2012-05-26 14:49:10 +02:00
//---------------------------------------------------
2016-06-30 15:34:02 +02:00
fileTools = addToolBar("");
2012-05-26 14:49:10 +02:00
fileTools->setObjectName("file-operations");
if (qApp->layoutDirection() == Qt::LayoutDirection::LeftToRight) {
for (auto i : { "file-new", "file-open", "file-save", "print", "undo", "redo"})
fileTools->addWidget(new AccessibleToolButton(fileTools, getAction(i)));
}
else {
for (auto i : { "undo", "redo", "print", "file-save", "file-open", "file-new"})
fileTools->addWidget(new AccessibleToolButton(fileTools, getAction(i)));
}
2012-05-26 14:49:10 +02:00
fileTools->addSeparator();
mag = new MagBox;
2015-11-23 16:02:33 +01:00
connect(mag, SIGNAL(magChanged(MagIdx)), SLOT(magChanged(MagIdx)));
fileTools->addWidget(mag);
viewModeCombo = new QComboBox(this);
#if defined(Q_OS_MAC)
viewModeCombo->setFocusPolicy(Qt::StrongFocus);
#else
viewModeCombo->setFocusPolicy(Qt::TabFocus);
#endif
viewModeCombo->setFixedHeight(preferences.getInt(PREF_UI_THEME_ICONHEIGHT) + 8); // hack
2016-06-30 15:34:02 +02:00
viewModeCombo->addItem("", int(LayoutMode::PAGE));
viewModeCombo->addItem("", int(LayoutMode::LINE));
viewModeCombo->addItem("", int(LayoutMode::SYSTEM));
connect(viewModeCombo, SIGNAL(activated(int)), SLOT(switchLayoutMode(int)));
fileTools->addWidget(viewModeCombo);
2012-05-26 14:49:10 +02:00
2016-06-03 13:40:18 +02:00
//---------------------
// Transport Tool Bar
//---------------------
2016-06-30 15:34:02 +02:00
transportTools = addToolBar("");
2012-05-26 14:49:10 +02:00
transportTools->setObjectName("transport-tools");
#ifdef HAS_MIDI
transportTools->addWidget(new AccessibleToolButton(transportTools, getAction("midi-on")));
2012-05-26 14:49:10 +02:00
transportTools->addSeparator();
2014-11-14 11:44:48 +01:00
#endif
transportTools->addWidget(new AccessibleToolButton(transportTools, getAction("rewind")));
_playButton = new AccessibleToolButton(transportTools, getAction("play"));
transportTools->addWidget(_playButton);
transportTools->addWidget(new AccessibleToolButton(transportTools, getAction("loop")));
2012-05-26 14:49:10 +02:00
transportTools->addSeparator();
QAction* repeatAction = getAction("repeat");
repeatAction->setChecked(preferences.getBool(PREF_APP_PLAYBACK_PLAYREPEATS));
transportTools->addWidget(new AccessibleToolButton(transportTools, repeatAction));
2014-11-14 11:44:48 +01:00
transportTools->addWidget(new AccessibleToolButton(transportTools, getAction("pan")));
transportTools->addWidget(new AccessibleToolButton(transportTools, metronomeAction));
2012-05-26 14:49:10 +02:00
2016-06-03 13:40:18 +02:00
//-------------------------------
// Concert Pitch Tool Bar
//-------------------------------
2016-06-30 15:34:02 +02:00
cpitchTools = addToolBar("");
2012-05-26 14:49:10 +02:00
cpitchTools->setObjectName("pitch-tools");
a = getAction("concert-pitch");
a->setCheckable(true);
cpitchTools->addWidget(new AccessibleToolButton(cpitchTools, a));
2012-05-26 14:49:10 +02:00
2016-06-03 13:40:18 +02:00
//-------------------------------
// Image Capture Tool Bar
//-------------------------------
2016-06-30 15:34:02 +02:00
fotoTools = addToolBar("");
fotoTools->setObjectName("foto-tools");
fotoTools->addWidget(new AccessibleToolButton(fotoTools, getAction("fotomode")));
2012-05-26 14:49:10 +02:00
2013-07-04 21:07:38 +02:00
addToolBarBreak();
2012-05-26 14:49:10 +02:00
//-------------------------------
// Note Input Tool Bar
2012-05-26 14:49:10 +02:00
//-------------------------------
2016-06-30 15:34:02 +02:00
entryTools = addToolBar("");
2012-05-26 14:49:10 +02:00
entryTools->setObjectName("entry-tools");
2016-05-31 17:44:39 +02:00
populateNoteInputMenu();
2015-02-02 09:30:17 +01:00
2012-05-26 14:49:10 +02:00
//---------------------
// Menus
//---------------------
QMenuBar* mb = menuBar();
//---------------------
// Menu File
//---------------------
2016-06-30 15:34:02 +02:00
menuFile = mb->addMenu("");
menuFile->setObjectName("File");
2012-05-26 14:49:10 +02:00
a = getAction("startcenter");
a->setCheckable(true);
2016-06-30 15:34:02 +02:00
menuFile->addAction(a);
menuFile->addAction(getAction("file-new"));
menuFile->addAction(getAction("file-open"));
2014-11-14 11:44:48 +01:00
2016-06-30 15:34:02 +02:00
openRecent = menuFile->addMenu("");
2014-11-14 11:44:48 +01:00
2012-05-26 14:49:10 +02:00
connect(openRecent, SIGNAL(aboutToShow()), SLOT(openRecentMenu()));
connect(openRecent, SIGNAL(triggered(QAction*)), SLOT(selectScore(QAction*)));
2014-11-14 11:44:48 +01:00
for (auto i : {
2017-01-31 12:21:44 +01:00
"",
"file-save",
"file-save-as",
"file-save-a-copy",
"file-save-selection",
"file-save-online",
"file-export",
"file-part-export",
"file-import-pdf",
"",
"file-close",
"",
"parts",
"album" }) {
2014-11-14 11:44:48 +01:00
if (!*i)
2016-06-30 15:34:02 +02:00
menuFile->addSeparator();
else if ((MuseScore::unstable() && enableExperimental) || strcmp(i,"file-save-online") != 0)
2016-06-30 15:34:02 +02:00
menuFile->addAction(getAction(i));
2014-11-14 11:44:48 +01:00
}
if (enableExperimental)
2016-06-30 15:34:02 +02:00
menuFile->addAction(getAction("layer"));
menuFile->addSeparator();
menuFile->addAction(getAction("edit-info"));
2013-05-24 16:35:48 +02:00
if (enableExperimental)
2016-06-30 15:34:02 +02:00
menuFile->addAction(getAction("media"));
menuFile->addSeparator();
menuFile->addAction(getAction("print"));
#ifndef Q_OS_MAC
2016-06-30 15:34:02 +02:00
menuFile->addSeparator();
menuFile->addAction(getAction("quit"));
#endif
2014-11-14 11:44:48 +01:00
2012-05-26 14:49:10 +02:00
//---------------------
// Menu Edit
//---------------------
2016-06-30 15:34:02 +02:00
menuEdit = mb->addMenu("");
2012-05-26 14:49:10 +02:00
menuEdit->setObjectName("Edit");
menuEdit->addAction(getAction("undo"));
menuEdit->addAction(getAction("redo"));
menuEdit->addSeparator();
menuEdit->addAction(getAction("cut"));
menuEdit->addAction(getAction("copy"));
2014-11-14 11:44:48 +01:00
menuEdit->addAction(getAction("paste"));
menuEdit->addAction(getAction("swap"));
menuEdit->addAction(getAction("delete"));
2014-11-14 11:44:48 +01:00
2012-05-26 14:49:10 +02:00
menuEdit->addSeparator();
menuEdit->addAction(getAction("select-all"));
menuEdit->addAction(getAction("select-section"));
menuEdit->addAction(getAction("find"));
2012-09-12 16:19:03 +02:00
menuEdit->addSeparator();
2014-11-09 02:02:37 +01:00
2015-02-10 15:45:31 +01:00
menuEdit->addAction(getAction("instruments"));
2014-10-31 14:11:53 +01:00
#ifdef NDEBUG
if (enableExperimental) {
2014-10-31 14:11:53 +01:00
#endif
menuEdit->addSeparator();
menuEdit->addAction(getAction("debugger"));
2014-10-31 14:11:53 +01:00
#ifdef NDEBUG
}
2014-10-31 14:11:53 +01:00
#endif
2012-05-26 14:49:10 +02:00
2012-09-12 16:19:03 +02:00
menuEdit->addSeparator();
2016-06-30 15:34:02 +02:00
pref = menuEdit->addAction("", this, SLOT(startPreferenceDialog()));
2012-05-26 14:49:10 +02:00
pref->setMenuRole(QAction::PreferencesRole);
//---------------------
// Menu View
//---------------------
2016-06-30 15:34:02 +02:00
menuView = mb->addMenu("");
menuView->setObjectName("View");
a = getAction("toggle-palette");
a->setCheckable(true);
menuView->addAction(a);
a = getAction("masterpalette");
a->setCheckable(true);
menuView->addAction(a);
a = getAction("inspector");
a->setCheckable(true);
menuView->addAction(a);
#ifdef OMR
a = getAction("omr");
a->setCheckable(true);
menuView->addAction(a);
#endif
playId = getAction("toggle-playpanel");
playId->setCheckable(true);
menuView->addAction(playId);
a = getAction("toggle-navigator");
a->setCheckable(true);
a->setChecked(preferences.getBool(PREF_UI_APP_STARTUP_SHOWNAVIGATOR));
menuView->addAction(a);
a = getAction("toggle-timeline");
a->setCheckable(true);
menuView->addAction(a);
a = getAction("toggle-mixer");
a->setCheckable(true);
menuView->addAction(a);
a = getAction("synth-control");
a->setCheckable(true);
menuView->addAction(a);
2014-06-02 12:12:53 +02:00
a = getAction("toggle-selection-window");
a->setCheckable(true);
menuView->addAction(a);
2014-12-22 09:56:39 +01:00
a = getAction("toggle-piano");
a->setCheckable(true);
menuView->addAction(a);
menuView->addSeparator();
menuView->addAction(getAction("zoomin"));
menuView->addAction(getAction("zoomout"));
menuView->addSeparator();
menuToolbars = new QMenu();
2017-01-16 20:51:12 +01:00
a = getAction("toggle-fileoperations");
a->setCheckable(true);
a->setChecked(fileTools->isVisible());
connect(fileTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
menuToolbars->addAction(a);
a = getAction("toggle-transport");
a->setCheckable(true);
a->setChecked(transportTools->isVisible());
connect(transportTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
menuToolbars->addAction(a);
a = getAction("toggle-concertpitch");
a->setCheckable(true);
a->setChecked(cpitchTools->isVisible());
connect(cpitchTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
menuToolbars->addAction(a);
a = getAction("toggle-imagecapture");
a->setCheckable(true);
a->setChecked(fotoTools->isVisible());
connect(fotoTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
menuToolbars->addAction(a);
a = getAction("toggle-noteinput");
a->setCheckable(true);
a->setChecked(entryTools->isVisible());
connect(entryTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
menuToolbars->addAction(a);
menuToolbars->addSeparator();
menuToolbars->addAction(getAction("edit-toolbars"));
menuView->addMenu(menuToolbars);
2016-06-30 15:34:02 +02:00
menuWorkspaces = new QMenu();
connect(menuWorkspaces, SIGNAL(aboutToShow()), SLOT(showWorkspaceMenu()));
menuView->addMenu(menuWorkspaces);
a = getAction("toggle-statusbar");
a->setCheckable(true);
a->setChecked(preferences.getBool(PREF_UI_APP_SHOWSTATUSBAR));
menuView->addAction(a);
menuView->addSeparator();
a = getAction("split-h");
a->setCheckable(true);
menuView->addAction(a);
a = getAction("split-v");
a->setCheckable(true);
menuView->addAction(a);
menuView->addSeparator();
menuView->addAction(getAction("show-invisible"));
menuView->addAction(getAction("show-unprintable"));
menuView->addAction(getAction("show-frames"));
menuView->addAction(getAction("show-pageborders"));
2016-08-06 11:36:51 +02:00
menuView->addAction(getAction("mark-irregular"));
menuView->addSeparator();
2016-03-02 13:20:19 +01:00
a = getAction("fullscreen");
a->setCheckable(true);
a->setChecked(false);
#ifndef Q_OS_MAC
menuView->addAction(a);
#endif
2012-05-26 14:49:10 +02:00
//---------------------
2016-06-30 15:34:02 +02:00
// Menu Add
2012-05-26 14:49:10 +02:00
//---------------------
2016-06-30 15:34:02 +02:00
menuAdd = mb->addMenu("");
menuAdd->setObjectName("Add");
menuAddPitch = new QMenu();
menuAddPitch->addAction(getAction("note-input"));
menuAddPitch->addSeparator();
2016-06-30 15:34:02 +02:00
2012-05-26 14:49:10 +02:00
for (int i = 0; i < 7; ++i) {
char buffer[8];
sprintf(buffer, "note-%c", "cdefgab"[i]);
a = getAction(buffer);
menuAddPitch->addAction(a);
}
menuAddPitch->addSeparator();
2012-05-26 14:49:10 +02:00
for (int i = 0; i < 7; ++i) {
char buffer[8];
sprintf(buffer, "chord-%c", "cdefgab"[i]);
a = getAction(buffer);
menuAddPitch->addAction(a);
}
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddPitch);
menuAddInterval = new QMenu();
2012-05-26 14:49:10 +02:00
for (int i = 1; i < 10; ++i) {
char buffer[16];
sprintf(buffer, "interval%d", i);
a = getAction(buffer);
menuAddInterval->addAction(a);
}
menuAddInterval->addSeparator();
for (int i = 2; i < 10; ++i) {
char buffer[16];
sprintf(buffer, "interval-%d", i);
a = getAction(buffer);
menuAddInterval->addAction(a);
}
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddInterval);
2012-05-26 14:49:10 +02:00
2016-06-30 15:34:02 +02:00
menuTuplet = new QMenu();
2014-11-14 11:44:48 +01:00
for (auto i : { "duplet", "triplet", "quadruplet", "quintuplet", "sextuplet",
"septuplet", "octuplet", "nonuplet", "tuplet-dialog" })
menuTuplet->addAction(getAction(i));
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuTuplet);
menuAdd->addSeparator();
menuAddMeasures = new QMenu("");
menuAddMeasures->addAction(getAction("insert-measure"));
menuAddMeasures->addAction(getAction("insert-measures"));
menuAddMeasures->addSeparator();
menuAddMeasures->addAction(getAction("append-measure"));
menuAddMeasures->addAction(getAction("append-measures"));
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddMeasures);
menuAddFrames = new QMenu();
menuAddFrames->addAction(getAction("insert-hbox"));
menuAddFrames->addAction(getAction("insert-vbox"));
menuAddFrames->addAction(getAction("insert-textframe"));
if (enableExperimental)
menuAddFrames->addAction(getAction("insert-fretframe"));
menuAddFrames->addSeparator();
menuAddFrames->addAction(getAction("append-hbox"));
menuAddFrames->addAction(getAction("append-vbox"));
menuAddFrames->addAction(getAction("append-textframe"));
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddFrames);
menuAddText = new QMenu();
menuAddText->addAction(getAction("title-text"));
menuAddText->addAction(getAction("subtitle-text"));
menuAddText->addAction(getAction("composer-text"));
menuAddText->addAction(getAction("poet-text"));
menuAddText->addAction(getAction("part-text"));
menuAddText->addSeparator();
menuAddText->addAction(getAction("system-text"));
menuAddText->addAction(getAction("staff-text"));
menuAddText->addAction(getAction("expression-text"));
menuAddText->addAction(getAction("chord-text"));
menuAddText->addAction(getAction("rehearsalmark-text"));
menuAddText->addAction(getAction("instrument-change-text"));
menuAddText->addAction(getAction("fingering-text"));
menuAddText->addSeparator();
menuAddText->addAction(getAction("lyrics"));
menuAddText->addAction(getAction("figured-bass"));
menuAddText->addAction(getAction("tempo"));
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddText);
menuAddLines = new QMenu();
menuAddLines->addAction(getAction("add-slur"));
menuAddLines->addAction(getAction("add-hairpin"));
menuAddLines->addAction(getAction("add-hairpin-reverse"));
menuAddLines->addAction(getAction("add-8va"));
menuAddLines->addAction(getAction("add-8vb"));
menuAddLines->addAction(getAction("add-noteline"));
2016-06-30 15:34:02 +02:00
menuAdd->addMenu(menuAddLines);
2012-05-26 14:49:10 +02:00
//---------------------
// Menu Format
2012-05-26 14:49:10 +02:00
//---------------------
2016-06-30 15:34:02 +02:00
menuFormat = mb->addMenu("");
menuFormat->setObjectName("Format");
menuFormat->addAction(getAction("edit-style"));
2017-08-17 22:08:09 +02:00
QAction* pageSettingsAction = getAction("page-settings");
// in some locale (fr), page settings ends up in Application menu on mac
// this line prevents it.
pageSettingsAction->setMenuRole(QAction::NoRole);
menuFormat->addAction(pageSettingsAction);
menuFormat->addSeparator();
2016-06-30 15:34:02 +02:00
menuFormat->addAction(getAction("add-remove-breaks"));
2016-06-30 15:34:02 +02:00
QMenu* menuStretch = new QMenu(tr("&Stretch"));
for (auto i : { "stretch+", "stretch-", "reset-stretch" })
menuStretch->addAction(getAction(i));
menuFormat->addMenu(menuStretch);
menuFormat->addSeparator();
2016-06-30 15:34:02 +02:00
menuFormat->addAction(getAction("reset-beammode"));
menuFormat->addAction(getAction("reset"));
menuFormat->addSeparator();
2016-06-30 15:34:02 +02:00
if (enableExperimental)
menuFormat->addAction(getAction("edit-harmony"));
2012-05-26 14:49:10 +02:00
menuFormat->addSeparator();
menuFormat->addAction(getAction("load-style"));
menuFormat->addAction(getAction("save-style"));
2016-06-30 15:34:02 +02:00
2012-05-26 14:49:10 +02:00
//---------------------
// Menu Tools
2012-05-26 14:49:10 +02:00
//---------------------
2016-06-30 15:34:02 +02:00
menuTools = mb->addMenu("");
menuTools->setObjectName("Tools");
2016-06-30 15:34:02 +02:00
menuTools->addAction(getAction("transpose"));
menuTools->addSeparator();
menuTools->addAction(getAction("explode"));
menuTools->addAction(getAction("implode"));
2016-06-30 15:34:02 +02:00
menuVoices = new QMenu("");
for (auto i : { "voice-x12", "voice-x13", "voice-x14", "voice-x23", "voice-x24", "voice-x34" })
menuVoices->addAction(getAction(i));
menuTools->addMenu(menuVoices);
menuTools->addSeparator();
2016-06-30 15:34:02 +02:00
menuTools->addAction(getAction("slash-fill"));
menuTools->addAction(getAction("slash-rhythm"));
menuTools->addSeparator();
2016-06-30 15:34:02 +02:00
menuTools->addAction(getAction("pitch-spell"));
menuTools->addAction(getAction("reset-groupings"));
menuTools->addAction(getAction("resequence-rehearsal-marks"));
menuTools->addSeparator();
2016-06-30 15:34:02 +02:00
menuTools->addAction(getAction("copy-lyrics-to-clipboard"));
menuTools->addAction(getAction("fotomode"));
2012-05-26 14:49:10 +02:00
menuTools->addAction(getAction("del-empty-measures"));
2012-07-13 13:08:54 +02:00
//---------------------
// Menu Plugins
//---------------------
2016-06-30 15:34:02 +02:00
menuPlugins = mb->addMenu("");
2012-07-13 13:08:54 +02:00
menuPlugins->setObjectName("Plugins");
2013-10-05 12:03:30 +02:00
menuPlugins->addAction(getAction("plugin-manager"));
2012-07-13 13:08:54 +02:00
a = getAction("plugin-creator");
a->setCheckable(true);
menuPlugins->addAction(a);
2012-07-13 13:08:54 +02:00
menuPlugins->addSeparator();
2012-07-19 16:47:23 +02:00
//---------------------
// Menu Debug
//---------------------
#ifndef NDEBUG
QMenu* menuDebug = mb->addMenu("Debug");
menuDebug->setObjectName("Debug");
a = getAction("no-horizontal-stretch");
a->setCheckable(true);
menuDebug->addAction(a);
a = getAction("no-vertical-stretch");
a->setCheckable(true);
menuDebug->addAction(a);
menuDebug->addSeparator();
a = getAction("show-segment-shapes");
a->setCheckable(true);
menuDebug->addAction(a);
a = getAction("show-skylines");
a->setCheckable(true);
a->setChecked(MScore::showSkylines);
menuDebug->addAction(a);
a = getAction("show-bounding-rect");
a->setCheckable(true);
menuDebug->addAction(a);
a = getAction("show-corrupted-measures");
a->setCheckable(true);
a->setChecked(true);
menuDebug->addAction(a);
2017-07-17 16:11:21 +02:00
a = getAction("relayout");
menuDebug->addAction(a);
a = getAction("autoplace-slurs");
a->setCheckable(true);
a->setChecked(MScore::autoplaceSlurs);
menuDebug->addAction(a);
#endif
2012-05-26 14:49:10 +02:00
//---------------------
// Menu Help
//---------------------
mb->addSeparator();
2016-06-30 15:34:02 +02:00
menuHelp = mb->addMenu("");
2012-05-26 14:49:10 +02:00
menuHelp->setObjectName("Help");
2015-02-20 14:01:22 +01:00
#if 0
if (_helpEngine) {
HelpQuery* hw = new HelpQuery(menuHelp);
menuHelp->addAction(hw);
connect(menuHelp, SIGNAL(aboutToShow()), hw, SLOT(setFocus()));
}
#endif
//menuHelp->addAction(getAction("help"));
2016-06-30 15:34:02 +02:00
onlineHandbookAction = menuHelp->addAction("", this, SLOT(helpBrowser1()));
2012-05-26 14:49:10 +02:00
menuHelp->addSeparator();
2016-06-30 15:34:02 +02:00
aboutAction = new QAction("", 0);
2012-05-26 14:49:10 +02:00
aboutAction->setMenuRole(QAction::AboutRole);
connect(aboutAction, SIGNAL(triggered()), this, SLOT(about()));
menuHelp->addAction(aboutAction);
2016-06-30 15:34:02 +02:00
aboutQtAction = new QAction("", 0);
aboutQtAction->setMenuRole(QAction::AboutQtRole);
connect(aboutQtAction, SIGNAL(triggered()), this, SLOT(aboutQt()));
menuHelp->addAction(aboutQtAction);
2012-05-26 14:49:10 +02:00
2016-06-30 15:34:02 +02:00
aboutMusicXMLAction = new QAction("", 0);
2012-05-26 14:49:10 +02:00
aboutMusicXMLAction->setMenuRole(QAction::ApplicationSpecificRole);
connect(aboutMusicXMLAction, SIGNAL(triggered()), this, SLOT(aboutMusicXML()));
menuHelp->addAction(aboutMusicXMLAction);
#if defined(Q_OS_MAC) || defined(Q_OS_WIN)
Solved all compilation errors on MSVC This commit contains changes required for MuseScore to compile under MSVC with no errors. There are several general categories of problems that resulted in errors with the compilation. Main issues: - Variable Length Arrays (VLA). This is a non-standard extension to C++, supported by clang toolchains, but not by MSVC. The initial workaround is to use std::vector<> instead of VLAs, eventually (if needed) with the original variable pointing to the beginning of the allocated array. More efficient alternatives are possible if profiling shows any code using VLAs to be timing-critical. - Floating-point constants not suffixed with "f" are doubles by default; in some instances, this leads to narrowing conversion errors (in other instances, just warnings). - MSVC does not support "or"/"and"/"not" in lieu of "||"/"&&"/"!". Changed unconditionally to use standard C++ symbols. - MSVC does not support the "__builtin_unreachable()" compiler hint. A similar, albeit not exactly equal, alternative is "__assume(0)", which is MSVC-specific. - MSVC does not support ranges in case statements. Replaced with list of cases instead of range (non-conditionally) Detailed changes, with per-file comments: - all.h: opt-in to deprecated features, include <io.h> and <process.h> instead of POSIX <unistd.h>, undefine "STRING_NONE" and "small", which Microsoft defines as macros, which result in compilation errors. - effects/compressor/zita.cpp: eliminated narrowing conversion error by appending "f" to float constants. - fluid/voice.cpp: appended "f" to float constants to eliminate narrowing conversion errors/warnings - libmscore/beam.cpp: conditionally replaced VLA - libmscore/edit.cpp: conditionally replaced VLA - libmscore/element.cpp: appended "f" to float constant - libmscore/fret.cpp: changed or -> || - libmscore/layout.cpp: conditionally replaced VLA - libmscore/mscore.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - libmscore/scorefile.coo: use correct char representation conversion for MSVC - libmscore/stringdata.cpp: conditionally replaced VLA - libmscore/system.cpp: conditionally replaced VLA - libmscroe/text.cpp: replaced range in case statement. - manual/genManual.cpp: use getopt() replacement. - midi/midifile.cpp: conditionally replaced VLA. This does not use the default replacement. - mscore/bb.cpp: replaced range in case statement. - mscore/capella.cpp: conditionally replaced VLA. Changed and -> && - mscore/driver:cpp: preclude errors due to macro redefinitions. - mscore/editstyle.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - mscore/importgtp-gp6.cpp: conditionally replaced VLA. - mscore/instrwidget.cpp: conditionally replaced VLA. - mscore/jackaudio.cpp: conditionally replaced VLA. Preclude errors due to macro redefinitions. - mscore/jackweakapi.cpp: Preclude errors due to macro redefinitions. Replacement for __atribute__((constructor)) through static object construction for MSVC. Force use of LoadLibraryA instead of LoadLibrary. - mscore/mididriver.h: Changed not -> ! - mscore/musescore.cpp: Changed not -> !. Conditionally replaced VLA. - mscore/resourceManager.cpp: conditionally replaced VLA. - mscore/timeline.cpp: conditionally replaced VLA. - omr/omrpage.cpp: conditionally replaced VLA. - synthesizer/msynthesizer.cpp: replaced UNIX sleep(1) method with MSVC Sleep(1000) (equivalent, but in ms instead of seconds) - synthesizer/msynthsizer.h: appended "f" to float constant - thirdparty/poppler/config.h: set defines for MSVC to preclude the use of inexistent libraries. - thirdparty/poppler/poppler/poppler-config.h: set defines for MSVC to preclude the use of inexistent libraries. Eliminated #defines for fmin and fmax which where causing problems. - thirdparty/poppler/poppler/PSOutputDev.cc: added #include <algorithm> for std::min() and std::max(). Note this is required per-C++ standard. - thirdparty/portmidi/pm_win/pmwinmm.c: undefined UNICODE to use char-based library functions. - thirdparty/qzip/qzip.cpp: changed or -> || - thirdparty/rtf2html/rtf_keyword.h: file format changed from Apple to UNIX, as MSVC does not supported the former. (NOT error related, just improvement on previous commit; manual/getopt/README.md: added link to source article)
2018-05-01 19:14:44 +02:00
#if !defined(FOR_WINSTORE)
2016-06-30 15:34:02 +02:00
checkForUpdateAction = menuHelp->addAction("", this, SLOT(checkForUpdate()));
#endif
2012-05-26 14:49:10 +02:00
#endif
menuHelp->addSeparator();
2016-06-30 15:34:02 +02:00
askForHelpAction = menuHelp->addAction("", this, SLOT(askForHelp()));
reportBugAction = menuHelp->addAction("", this, SLOT(reportBug()));
menuHelp->addSeparator();
menuHelp->addAction(getAction("resource-manager"));
menuHelp->addSeparator();
2016-06-30 15:34:02 +02:00
revertToFactoryAction = menuHelp->addAction("", this, SLOT(resetAndRestart()));
2016-07-26 22:28:48 +02:00
if (!MScore::noGui) {
2016-06-30 15:34:02 +02:00
retranslate(true);
2016-07-26 22:28:48 +02:00
//accessibility for menus
for (QMenu* menu : mb->findChildren<QMenu*>()) {
menu->setAccessibleName(menu->objectName());
menu->setAccessibleDescription(Shortcut::getMenuShortcutString(menu));
}
}
2013-05-09 18:46:51 +02:00
setCentralWidget(envelope);
if (!MScore::noGui)
preferencesChanged();
2012-05-26 14:49:10 +02:00
if (seq) {
connect(seq, SIGNAL(started()), SLOT(seqStarted()));
connect(seq, SIGNAL(stopped()), SLOT(seqStopped()));
}
loadScoreList();
QClipboard* cb = QApplication::clipboard();
connect(cb, SIGNAL(dataChanged()), SLOT(clipboardChanged()));
connect(cb, SIGNAL(selectionChanged()), SLOT(clipboardChanged()));
2017-05-02 14:17:31 +02:00
2012-05-26 14:49:10 +02:00
autoSaveTimer = new QTimer(this);
autoSaveTimer->setSingleShot(true);
connect(autoSaveTimer, SIGNAL(timeout()), this, SLOT(autoSaveTimerTimeout()));
initOsc();
startAutoSave();
QInputMethod* im = QGuiApplication::inputMethod();
connect(im, SIGNAL(localeChanged()), SLOT(inputMethodLocaleChanged()));
2012-05-26 14:49:10 +02:00
if (enableExperimental) {
cornerLabel = new QLabel(this);
cornerLabel->setScaledContents(true);
cornerLabel->setPixmap(QPixmap(":/data/mscore.png"));
cornerLabel->setGeometry(width() - 48, 0, 48, 48);
}
}
MuseScore::~MuseScore()
{
delete synti;
}
//---------------------------------------------------------
// showError
//---------------------------------------------------------
void MuseScore::showError()
{
static QErrorMessage* msg = 0;
if (msg == 0)
msg = new QErrorMessage(this);
msg->showMessage(tr(MScore::errorMessage()), MScore::errorGroup());
2016-11-28 18:25:27 +01:00
MScore::setError(MS_NO_ERROR);
}
//---------------------------------------------------------
// retranslate
//---------------------------------------------------------
2016-06-30 15:34:02 +02:00
void MuseScore::retranslate(bool firstStart)
{
_positionLabel->setToolTip(tr("Measure:Beat:Tick"));
// retranslate the menu
menuFile->setTitle(tr("&File"));
openRecent->setTitle(tr("Open &Recent"));
menuEdit->setTitle(tr("&Edit"));
menuView->setTitle(tr("&View"));
menuToolbars->setTitle(tr("&Toolbars"));
2016-06-30 15:34:02 +02:00
menuWorkspaces->setTitle(tr("W&orkspaces"));
pref->setText(tr("&Preferences..."));
menuAdd->setTitle(tr("&Add"));
menuAddMeasures->setTitle(tr("&Measures"));
menuAddFrames->setTitle(tr("&Frames"));
menuAddText->setTitle(tr("&Text"));
menuAddLines->setTitle(tr("&Lines"));
menuAddPitch->setTitle(tr("N&otes"));
menuAddInterval->setTitle(tr("&Intervals"));
2016-06-30 15:34:02 +02:00
menuTuplet->setTitle(tr("T&uplets"));
menuFormat->setTitle(tr("F&ormat"));
2016-06-30 15:34:02 +02:00
menuTools->setTitle(tr("&Tools"));
menuVoices->setTitle(tr("&Voices"));
menuPlugins->setTitle(tr("&Plugins"));
menuHelp->setTitle(tr("&Help"));
aboutAction->setText(tr("&About..."));
aboutQtAction->setText(tr("About &Qt..."));
aboutMusicXMLAction->setText(tr("About &MusicXML..."));
onlineHandbookAction->setText(tr("&Online Handbook"));
if (checkForUpdateAction)
checkForUpdateAction->setText(tr("Check for &Update"));
askForHelpAction->setText(tr("Ask for Help"));
reportBugAction->setText(tr("Report a Bug"));
revertToFactoryAction->setText(tr("Revert to Factory Settings"));
fileTools->setWindowTitle(tr("File Operations"));
transportTools->setWindowTitle(tr("Playback Controls"));
2016-06-30 15:34:02 +02:00
cpitchTools->setWindowTitle(tr("Concert Pitch"));
fotoTools->setWindowTitle(tr("Image Capture"));
entryTools->setWindowTitle(tr("Note Input"));
viewModeCombo->setAccessibleName(tr("View Mode"));
viewModeCombo->setItemText(viewModeCombo->findData(int(LayoutMode::PAGE)), tr("Page View"));
viewModeCombo->setItemText(viewModeCombo->findData(int(LayoutMode::LINE)), tr("Continuous View"));
viewModeCombo->setItemText(viewModeCombo->findData(int(LayoutMode::SYSTEM)), tr("Single Page"));
2016-06-30 15:34:02 +02:00
showMidiImportButton->setText(tr("Show MIDI import panel"));
2016-06-30 15:34:02 +02:00
Shortcut::retranslate();
if (!firstStart && Workspace::currentWorkspace->readOnly()) {
changeWorkspace(Workspace::currentWorkspace);
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// resizeEvent
//---------------------------------------------------------
void MuseScore::resizeEvent(QResizeEvent*)
{
if (enableExperimental) {
cornerLabel->setGeometry(width() - 48, 0, 48, 48);
}
}
//---------------------------------------------------------
// startAutoSave
//---------------------------------------------------------
void MuseScore::startAutoSave()
{
if (preferences.getBool(PREF_APP_AUTOSAVE_USEAUTOSAVE)) {
int t = preferences.getInt(PREF_APP_AUTOSAVE_AUTOSAVETIME) * 60 * 1000;
2012-05-26 14:49:10 +02:00
autoSaveTimer->start(t);
}
else
autoSaveTimer->stop();
}
//---------------------------------------------------------
// getLocaleISOCode
//---------------------------------------------------------
QString MuseScore::getLocaleISOCode() const
2012-05-26 14:49:10 +02:00
{
QString lang;
if (localeName.toLower() == "system")
lang = QLocale::system().name();
else
lang = localeName;
return lang;
}
//---------------------------------------------------------
// helpBrowser1
// show online help
//---------------------------------------------------------
void MuseScore::helpBrowser1() const
2012-05-26 14:49:10 +02:00
{
QString lang = getLocaleISOCode();
if (MScore::debugMode)
qDebug("open online handbook for language <%s>", qPrintable(lang));
QString help = QString("https://musescore.org/redirect/help?tag=handbook&locale=%1").arg(getLocaleISOCode());
2012-05-26 14:49:10 +02:00
//try to find an exact match
bool found = false;
foreach (LanguageItem item, _languages) {
if (item.key == lang) {
2012-05-26 14:49:10 +02:00
QString handbook = item.handbook;
if (!handbook.isNull()) {
help = item.handbook;
found = true;
}
break;
}
}
//try a to find a match on first two letters
if (!found && lang.size() > 2) {
lang = lang.left(2);
foreach (LanguageItem item, _languages) {
if (item.key == lang){
QString handbook = item.handbook;
if (!handbook.isNull())
help = item.handbook;
break;
}
}
}
//track visits. see: http://www.google.com/support/googleanalytics/bin/answer.py?answer=55578
help += QString("&utm_source=desktop&utm_medium=menu&utm_content=%1&utm_campaign=MuseScore%2").arg(rev.trimmed()).arg(QString(VERSION));
2015-02-06 16:36:43 +01:00
QDesktopServices::openUrl(QUrl(help));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// resetAndRestart
//---------------------------------------------------------
void MuseScore::resetAndRestart()
{
int ret = QMessageBox::question(0, tr("Are you sure?"),
tr("This will reset all your preferences.\n"
"Custom palettes, custom shortcuts, and the list of recent scores will be deleted. "
"MuseScore will restart with its default settings.\n"
"Reverting will not remove any scores from your computer.\n"
"Are you sure you want to proceed?"),
QMessageBox::Yes|QMessageBox::No, QMessageBox::No);
if (ret == QMessageBox::Yes ) {
close();
QStringList args("-F");
QProcess::startDetached(qApp->arguments()[0], args);
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// aboutQt
//---------------------------------------------------------
void MuseScore::aboutQt()
{
QMessageBox::aboutQt(this, QString("About Qt"));
}
//---------------------------------------------------------
// aboutMusicXML
//---------------------------------------------------------
void MuseScore::aboutMusicXML()
{
AboutMusicXMLBoxDialog ab;
ab.show();
ab.exec();
}
//---------------------------------------------------------
// selectScore
// "open recent"
//---------------------------------------------------------
void MuseScore::selectScore(QAction* action)
{
QString a = action->data().toString();
if (!a.isEmpty()) {
if (a == "clear-recent") {
_recentScores.clear();
if (startcenter)
startcenter->updateRecentScores();
}
else {
2016-03-11 12:18:46 +01:00
MasterScore* score = readScore(a);
if (score) {
setCurrentScoreView(appendScore(score));
addRecentScore(score);
writeSessionFile(false);
}
}
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
// selectionChanged
//---------------------------------------------------------
void MuseScore::selectionChanged(SelState selectionState)
2012-05-26 14:49:10 +02:00
{
2014-05-26 13:21:04 +02:00
bool enable = selectionState != SelState::NONE;
2012-05-26 14:49:10 +02:00
getAction("cut")->setEnabled(enable);
getAction("copy")->setEnabled(enable);
2014-07-14 13:24:47 +02:00
getAction("select-similar-range")->setEnabled(selectionState == SelState::RANGE);
2012-05-26 14:49:10 +02:00
if (pianorollEditor)
pianorollEditor->changeSelection(selectionState);
2012-05-26 14:49:10 +02:00
if (drumrollEditor)
drumrollEditor->changeSelection(selectionState);
if (timeline())
timeline()->changeSelection(selectionState);
if (_pianoTools && _pianoTools->isVisible()) {
if (cs)
_pianoTools->changeSelection(cs->selection());
else
_pianoTools->clearSelection();
}
if (_inspector)
updateInspector();
2012-07-12 15:35:35 +02:00
}
//---------------------------------------------------------
// updateInspector
//---------------------------------------------------------
void MuseScore::updateInspector()
{
2017-03-31 13:03:15 +02:00
if (_inspector)
_inspector->update(cs);
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// updateShadowNote
// update the shadow note in the current ScoreView
//---------------------------------------------------------
void MuseScore::updateShadowNote()
{
currentScoreView()->updateShadowNotes();
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// appendScore
// append score to project list
//---------------------------------------------------------
2016-03-11 12:18:46 +01:00
int MuseScore::appendScore(MasterScore* score)
2012-05-26 14:49:10 +02:00
{
int index = scoreList.size();
for (int i = 0; i < scoreList.size(); ++i) {
if ((!score->importedFilePath().isEmpty()
&& scoreList[i]->importedFilePath() == score->importedFilePath())
2016-03-11 12:18:46 +01:00
|| (scoreList[i]->fileInfo()->canonicalFilePath() == score->fileInfo()->canonicalFilePath() && score->fileInfo()->exists())) {
2012-05-26 14:49:10 +02:00
removeTab(i);
index = i;
break;
}
}
scoreList.insert(index, score);
tab1->blockSignals(true);
if (tab2)
tab2->blockSignals(true);
tab1->insertTab(score);
if (tab2)
tab2->insertTab(score);
tab1->blockSignals(false);
if (tab2)
tab2->blockSignals(false);
return index;
}
//---------------------------------------------------------
// addRecentScore
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
void MuseScore::addRecentScore(Score* score)
2012-05-26 14:49:10 +02:00
{
QString path = score->importedFilePath(); // defined for scores imported from e.g. MIDI files
addRecentScore(path);
2016-03-10 10:41:31 +01:00
path = score->masterScore()->fileInfo()->absoluteFilePath();
addRecentScore(path);
if (startcenter)
startcenter->updateRecentScores();
}
void MuseScore::addRecentScore(const QString& scorePath)
{
if (scorePath.isEmpty())
return;
QFileInfo fi(scorePath);
QString absoluteFilePath = fi.absoluteFilePath();
_recentScores.removeAll(absoluteFilePath);
_recentScores.prepend(absoluteFilePath);
2014-11-14 12:29:17 +01:00
if (_recentScores.size() > RECENT_LIST_SIZE)
_recentScores.removeLast();
2012-05-26 14:49:10 +02:00
}
2016-10-10 18:37:28 +02:00
#if 0
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// updateTabNames
//---------------------------------------------------------
void MuseScore::updateTabNames()
{
for (int i = 0; i < tab1->count(); ++i) {
ScoreView* view = tab1->view(i);
if (view)
2016-10-10 18:37:28 +02:00
tab1->setTabText(i, view->score()->masterScore()->fileInfo()->completeBaseName());
2012-05-26 14:49:10 +02:00
}
if (tab2) {
for (int i = 0; i < tab2->count(); ++i) {
ScoreView* view = tab2->view(i);
if (view)
2016-10-10 18:37:28 +02:00
tab2->setTabText(i, view->score()->masterScore()->fileInfo()->completeBaseName());
2012-05-26 14:49:10 +02:00
}
}
}
2016-10-10 18:37:28 +02:00
#endif
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// loadScoreList
// read list of "Recent Scores"
//---------------------------------------------------------
void MuseScore::loadScoreList()
{
if (useFactorySettings)
return;
QSettings s;
for (int i = RECENT_LIST_SIZE-1; i >= 0; --i) {
QString path = s.value(QString("recent-%1").arg(i),"").toString();
if (!path.isEmpty() && QFileInfo(path).exists()) {
_recentScores.removeAll(path);
_recentScores.prepend(path);
2012-05-26 14:49:10 +02:00
}
}
}
//---------------------------------------------------------
// openRecentMenu
//---------------------------------------------------------
void MuseScore::openRecentMenu()
{
openRecent->clear();
bool one = false;
for (const QFileInfo& fi : recentScores()) {
QAction* action = openRecent->addAction(fi.fileName().replace("&", "&&")); // show filename only
QString data(fi.canonicalFilePath());
action->setData(data);
action->setToolTip(data);
one = true;
}
if (one) {
openRecent->addSeparator();
QAction* action = openRecent->addAction(tr("Clear Recent Files"));
action->setData("clear-recent");
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
// reloadInstrumentTemplates
//---------------------------------------------------------
void MuseScore::reloadInstrumentTemplates()
{
clearInstrumentTemplates();
// load cascading instrument templates
loadInstrumentTemplates(preferences.getString(PREF_APP_PATHS_INSTRUMENTLIST1));
QString list2 = preferences.getString(PREF_APP_PATHS_INSTRUMENTLIST2);
if (!list2.isEmpty())
loadInstrumentTemplates(list2);
// load instrument templates from extension
QStringList extensionDir = Extension::getDirectoriesByType(Extension::instrumentsDir);
QStringList filter("*.xml");
for (QString s : extensionDir) {
QDir extDir(s);
extDir.setNameFilters(filter);
auto instFiles = extDir.entryInfoList(QDir::Files | QDir::NoSymLinks | QDir::Readable);
for (auto instFile : instFiles)
loadInstrumentTemplates(instFile.absoluteFilePath());
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// setCurrentView
//---------------------------------------------------------
void MuseScore::setCurrentScoreView(int idx)
{
setCurrentView(0, idx);
if (tab2)
setCurrentView(1, idx);
2012-05-26 14:49:10 +02:00
}
void MuseScore::setCurrentView(int tabIdx, int idx)
{
if (idx == -1)
2013-02-15 22:39:56 +01:00
setCurrentScoreView(nullptr);
2012-05-26 14:49:10 +02:00
else {
ScoreTab* tab = tabIdx ? tab2 : tab1;
2013-02-15 22:39:56 +01:00
if (tab)
2012-05-26 14:49:10 +02:00
tab->setCurrentIndex(idx);
}
}
//---------------------------------------------------------
// setCurrentScoreView
//---------------------------------------------------------
void MuseScore::setCurrentScoreView(ScoreView* view)
{
cv = view;
if (cv) {
if (timeline())
timeline()->setScoreView(cv);
2014-07-28 10:56:50 +02:00
if (cv->score() && (cs != cv->score())) {
// exit note entry mode
if (cv->noteEntryMode()) {
cv->cmd(getAction("escape"));
qApp->processEvents();
}
2012-05-26 14:49:10 +02:00
updateInputState(cv->score());
2014-07-28 10:56:50 +02:00
}
2012-05-26 14:49:10 +02:00
cs = cv->score();
2015-11-23 16:02:33 +01:00
cv->setFocusRect();
2012-05-26 14:49:10 +02:00
}
else
cs = 0;
// set midi import panel
QString fileName = cs ? cs->importedFilePath() : "";
midiPanelOnSwitchToFile(fileName);
if (enableExperimental) {
updateLayer();
updatePlayMode();
}
2012-05-26 14:49:10 +02:00
if (seq)
seq->setScoreView(cv);
if (playPanel)
playPanel->setScore(cs);
if (synthControl)
synthControl->setScore(cs);
if (selectionWindow)
selectionWindow->setScore(cs);
2013-04-03 12:13:23 +02:00
if (mixer)
mixer->updateAll(cs ? cs->masterScore() : nullptr);
2012-05-26 14:49:10 +02:00
#ifdef OMR
if (omrPanel) {
if (cv && cv->omrView())
omrPanel->setOmrView(cv->omrView());
else
omrPanel->setOmrView(0);
}
#endif
if (!cs) {
setWindowTitle(MUSESCORE_NAME_VERSION);
if (_navigator && _navigator->widget()) {
2015-11-23 16:02:33 +01:00
navigator()->setScoreView(cv);
2013-05-08 10:54:06 +02:00
navigator()->setScore(0);
}
if (timeline()) {
timeline()->setScoreView(cv);
timeline()->setScore(0);
}
2014-08-29 19:18:39 +02:00
if (_inspector)
2017-03-31 13:03:15 +02:00
_inspector->update(0);
2012-05-26 14:49:10 +02:00
viewModeCombo->setEnabled(false);
2013-02-25 21:57:11 +01:00
if (_textTools) {
2013-02-15 22:39:56 +01:00
_textTools->hide();
2013-02-25 21:57:11 +01:00
if (textPalette)
textPalette->hide();
}
2013-02-15 22:39:56 +01:00
if (_pianoTools)
_pianoTools->hide();
if (_drumTools)
_drumTools->hide();
changeState(STATE_DISABLED);
2012-05-26 14:49:10 +02:00
return;
}
2012-05-26 14:49:10 +02:00
viewModeCombo->setEnabled(true);
updateViewModeCombo();
2012-05-26 14:49:10 +02:00
selectionChanged(cs->selection().state());
2013-02-15 22:39:56 +01:00
_sstate = STATE_DISABLED; // defeat optimization
2015-11-23 16:02:33 +01:00
changeState(cv->mscoreState());
2012-05-26 14:49:10 +02:00
2015-11-23 16:02:33 +01:00
cv->setFocus(Qt::OtherFocusReason);
setFocusProxy(cv);
2012-05-26 14:49:10 +02:00
2016-03-10 10:41:31 +01:00
getAction("file-save")->setEnabled(cs->masterScore()->isSavable());
getAction("file-part-export")->setEnabled(cs->masterScore()->excerpts().size() > 0);
2012-05-26 14:49:10 +02:00
getAction("show-invisible")->setChecked(cs->showInvisible());
getAction("show-unprintable")->setChecked(cs->showUnprintable());
getAction("show-frames")->setChecked(cs->showFrames());
getAction("show-pageborders")->setChecked(cs->showPageborders());
2016-08-06 11:36:51 +02:00
getAction("mark-irregular")->setChecked(cs->markIrregularMeasures());
getAction("fotomode")->setChecked(cv->fotoMode());
getAction("join-measures")->setEnabled(cs->masterScore()->excerpts().size() == 0);
2016-03-10 10:41:31 +01:00
getAction("split-measure")->setEnabled(cs->masterScore()->excerpts().size() == 0);
2012-05-26 14:49:10 +02:00
updateUndoRedo();
2015-11-23 16:02:33 +01:00
MagIdx midx = cv->magIdx();
if (midx == MagIdx::MAG_FREE)
2015-11-18 18:09:42 +01:00
mag->setMag(view->lmag());
else {
2015-11-23 16:02:33 +01:00
mag->setMagIdx(midx);
magChanged(midx);
}
2012-05-26 14:49:10 +02:00
updateWindowTitle(cs);
setWindowModified(cs->dirty());
2012-05-26 14:49:10 +02:00
QAction* a = getAction("concert-pitch");
2018-03-27 15:36:00 +02:00
a->setChecked(cs->styleB(Sid::concertPitch));
2012-05-26 14:49:10 +02:00
setPos(cs->inputPos());
//showMessage(cs->filePath(), 2000);
if (_navigator && _navigator->widget()) {
navigator()->setScore(cs);
navigator()->setScoreView(view);
}
if (timeline()) {
timeline()->setScore(cs);
timeline()->setScoreView(view);
}
ScoreAccessibility::instance()->updateAccessibilityInfo();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// updateViewModeCombo
//---------------------------------------------------------
void MuseScore::updateViewModeCombo()
{
int idx;
switch (cs->layoutMode()) {
case LayoutMode::PAGE:
case LayoutMode::FLOAT:
idx = 0;
break;
case LayoutMode::LINE:
idx = 1;
break;
case LayoutMode::SYSTEM:
idx = 2;
break;
default:
idx = 0;
break;
}
viewModeCombo->setCurrentIndex(idx);
}
2013-05-24 18:40:15 +02:00
//---------------------------------------------------------
// showMessage
//---------------------------------------------------------
void MuseScore::showMessage(const QString& s, int timeout)
{
_statusBar->showMessage(s, timeout);
}
2013-05-17 20:11:36 +02:00
//---------------------------------------------------------
// midiPanel
2013-05-17 20:11:36 +02:00
//---------------------------------------------------------
void MuseScore::midiPanelOnSwitchToFile(const QString &file)
2013-05-17 20:11:36 +02:00
{
bool isMidiFile = ImportMidiPanel::isMidiFile(file);
if (isMidiFile) {
importmidiPanel->setMidiFile(file);
2014-07-29 21:58:36 +02:00
if (importmidiPanel->isPreferredVisible())
importmidiPanel->setVisible(true);
2013-05-17 20:11:36 +02:00
}
else
importmidiPanel->setVisible(false);
2014-07-29 21:58:36 +02:00
importmidiShowPanel->setVisible(!importmidiPanel->isPreferredVisible() && isMidiFile);
}
2014-11-14 11:44:48 +01:00
//---------------------------------------------------------
// midiPanelOnCloseFile
//---------------------------------------------------------
void MuseScore::midiPanelOnCloseFile(const QString &file)
{
if (ImportMidiPanel::isMidiFile(file))
importmidiPanel->excludeMidiFile(file);
}
2014-11-14 11:44:48 +01:00
//---------------------------------------------------------
// allowShowMidiPanel
//---------------------------------------------------------
void MuseScore::allowShowMidiPanel(const QString &file)
{
if (ImportMidiPanel::isMidiFile(file))
2014-07-29 21:58:36 +02:00
importmidiPanel->setPreferredVisible(true);
2013-05-17 20:11:36 +02:00
}
2014-11-14 11:44:48 +01:00
//---------------------------------------------------------
// setMidiReopenInProgress
//---------------------------------------------------------
2014-07-29 21:58:36 +02:00
void MuseScore::setMidiReopenInProgress(const QString &file)
2013-07-23 13:05:19 +02:00
{
if (ImportMidiPanel::isMidiFile(file))
importmidiPanel->setReopenInProgress();
}
2014-11-14 11:44:48 +01:00
//---------------------------------------------------------
// showMidiImportPanel
//---------------------------------------------------------
void MuseScore::showMidiImportPanel()
{
2014-07-29 21:58:36 +02:00
importmidiPanel->setPreferredVisible(true);
QString fileName = cs ? cs->importedFilePath() : "";
if (ImportMidiPanel::isMidiFile(fileName))
importmidiPanel->setVisible(true);
importmidiShowPanel->hide();
2013-07-23 13:05:19 +02:00
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// dragEnterEvent
//---------------------------------------------------------
void MuseScore::dragEnterEvent(QDragEnterEvent* event)
{
const QMimeData* data = event->mimeData();
if (data->hasUrls()) {
QList<QUrl>ul = event->mimeData()->urls();
foreach(const QUrl& u, ul) {
if (MScore::debugMode)
2014-10-16 18:12:58 +02:00
qDebug("drag Url: %s scheme <%s>", qPrintable(u.toString()), qPrintable(u.scheme()));
2012-05-26 14:49:10 +02:00
if (u.scheme() == "file") {
2014-10-16 18:12:58 +02:00
// QFileInfo fi(u.toLocalFile());
2012-05-26 14:49:10 +02:00
event->acceptProposedAction();
break;
}
}
}
}
//---------------------------------------------------------
// dropEvent
//---------------------------------------------------------
void MuseScore::dropEvent(QDropEvent* event)
{
const QMimeData* data = event->mimeData();
if (data->hasUrls()) {
int view = -1;
foreach(const QUrl& u, event->mimeData()->urls()) {
if (u.scheme() == "file") {
2013-04-28 21:29:12 +02:00
QString file = u.toLocalFile();
2016-03-11 12:18:46 +01:00
MasterScore* score = readScore(file);
2012-09-29 16:46:45 +02:00
if (score) {
2012-05-26 14:49:10 +02:00
view = appendScore(score);
addRecentScore(score);
}
2012-05-26 14:49:10 +02:00
}
}
2014-10-16 18:12:58 +02:00
if (view != -1) {
2012-05-26 14:49:10 +02:00
setCurrentScoreView(view);
writeSessionFile(false);
}
2012-05-26 14:49:10 +02:00
event->acceptProposedAction();
}
}
2016-06-30 15:34:02 +02:00
//---------------------------------------------------------
// changeEvent
//---------------------------------------------------------
2016-06-30 15:34:02 +02:00
void MuseScore::changeEvent(QEvent *e)
{
QMainWindow::changeEvent(e);
switch (e->type()) {
case QEvent::LanguageChange:
retranslate();
break;
default:
break;
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// showPageSettings
//---------------------------------------------------------
void MuseScore::showPageSettings()
{
if (pageSettings == 0)
pageSettings = new PageSettings();
2016-03-10 10:41:31 +01:00
pageSettings->setScore(cs->masterScore());
2012-05-26 14:49:10 +02:00
pageSettings->show();
pageSettings->raise();
}
//---------------------------------------------------------
// startDebugger
//---------------------------------------------------------
void MuseScore::startDebugger()
{
if (!cs)
return;
if (debugger == 0)
debugger = new Debugger(this);
debugger->updateList(cs);
debugger->show();
}
//---------------------------------------------------------
// showElementContext
//---------------------------------------------------------
void MuseScore::showElementContext(Element* el)
{
if (el == 0)
return;
startDebugger();
debugger->setElement(el);
}
//---------------------------------------------------------
// showPlayPanel
//---------------------------------------------------------
void MuseScore::showPlayPanel(bool visible)
{
if (noSeq || !(seq && seq->isRunning()))
2012-05-26 14:49:10 +02:00
return;
if (playPanel == 0) {
if (!visible)
return;
playPanel = new PlayPanel(this);
2013-04-03 12:49:55 +02:00
connect(playPanel, SIGNAL(gainChange(float)), synti, SLOT(setGain(float)));
connect(playPanel, SIGNAL(metronomeGainChanged(float)), seq, SLOT(setMetronomeGain(float)));
2012-05-26 14:49:10 +02:00
connect(playPanel, SIGNAL(relTempoChanged(double)),seq, SLOT(setRelTempo(double)));
connect(playPanel, SIGNAL(posChange(int)), seq, SLOT(seek(int)));
connect(playPanel, SIGNAL(closed(bool)), playId, SLOT(setChecked(bool)));
2013-04-03 12:49:55 +02:00
connect(synti, SIGNAL(gainChanged(float)), playPanel, SLOT(setGain(float)));
playPanel->setGain(synti->gain());
2012-05-26 14:49:10 +02:00
playPanel->setScore(cs);
addDockWidget(Qt::RightDockWidgetArea, playPanel);
2012-05-26 14:49:10 +02:00
}
playPanel->setVisible(visible);
playPanel->setFloating(false);
2012-05-26 14:49:10 +02:00
playId->setChecked(visible);
}
//---------------------------------------------------------
// cmdAppendMeasures
//---------------------------------------------------------
void MuseScore::cmdAppendMeasures()
{
if (cs) {
2013-04-28 21:29:12 +02:00
if (measuresDialog == 0)
2012-05-26 14:49:10 +02:00
measuresDialog = new MeasuresDialog;
measuresDialog->show();
}
}
//---------------------------------------------------------
// MeasuresDialog
//---------------------------------------------------------
MeasuresDialog::MeasuresDialog(QWidget* parent)
: QDialog(parent)
{
2012-05-26 14:49:10 +02:00
setupUi(this);
setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
setModal(true);
measures->selectAll();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// accept
//---------------------------------------------------------
void MeasuresDialog::accept()
2013-04-28 21:29:12 +02:00
{
int n = measures->value();
2012-05-26 14:49:10 +02:00
if (mscore->currentScore())
2017-01-18 14:16:33 +01:00
mscore->currentScoreView()->cmdAppendMeasures(n, ElementType::MEASURE);
2012-05-26 14:49:10 +02:00
done(1);
2013-04-28 21:29:12 +02:00
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// midiinToggled
//---------------------------------------------------------
void MuseScore::midiinToggled(bool val)
{
_midiinEnabled = val;
}
//---------------------------------------------------------
// midiinEnabled
//---------------------------------------------------------
bool MuseScore::midiinEnabled() const
{
return preferences.getBool(PREF_IO_MIDI_ENABLEINPUT) && _midiinEnabled;
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// processMidiRemote
// return if midi remote command detected
//---------------------------------------------------------
bool MuseScore::processMidiRemote(MidiRemoteType type, int data, int value)
2012-05-26 14:49:10 +02:00
{
if (!preferences.getBool(PREF_IO_MIDI_USEREMOTECONTROL))
2012-05-26 14:49:10 +02:00
return false;
if (!value) {
// This was a "NoteOff" or "CtrlOff" event. Most MidiRemote actions should only
// be triggered by an "On" event, so we need to check if this is one of those.
if (!preferences.getBool(PREF_IO_MIDI_ADVANCEONRELEASE)
|| type != preferences.midiRemote(RMIDI_REALTIME_ADVANCE).type
|| data != preferences.midiRemote(RMIDI_REALTIME_ADVANCE).data)
return false;
}
2012-05-26 14:49:10 +02:00
for (int i = 0; i < MIDI_REMOTES; ++i) {
if (preferences.midiRemote(i).type == type && preferences.midiRemote(i).data == data) {
2012-05-26 14:49:10 +02:00
if (cv == 0)
return false;
QAction* a = 0;
switch (i) {
case RMIDI_REWIND: a = getAction("rewind"); break;
case RMIDI_TOGGLE_PLAY: a = getAction("play"); break;
case RMIDI_PLAY:
a = getAction("play");
if (a->isChecked())
return true;
break;
case RMIDI_STOP:
a = getAction("play");
if (!a->isChecked())
return true;
break;
case RMIDI_NOTE1: a = getAction("pad-note-1"); break;
case RMIDI_NOTE2: a = getAction("pad-note-2"); break;
case RMIDI_NOTE4: a = getAction("pad-note-4"); break;
case RMIDI_NOTE8: a = getAction("pad-note-8"); break;
case RMIDI_NOTE16: a = getAction("pad-note-16"); break;
case RMIDI_NOTE32: a = getAction("pad-note-32"); break;
case RMIDI_NOTE64: a = getAction("pad-note-64"); break;
case RMIDI_REST: a = getAction("rest"); break;
case RMIDI_DOT: a = getAction("pad-dot"); break;
case RMIDI_DOTDOT: a = getAction("pad-dotdot"); break;
2012-05-26 14:49:10 +02:00
case RMIDI_TIE: a = getAction("tie"); break;
case RMIDI_UNDO: a = getAction("undo"); break;
2012-05-26 14:49:10 +02:00
case RMIDI_NOTE_EDIT_MODE: a = getAction("note-input"); break;
2016-08-08 17:46:43 +02:00
case RMIDI_REALTIME_ADVANCE: a = getAction("realtime-advance"); break;
2012-05-26 14:49:10 +02:00
}
if (a)
a->trigger();
return true;
}
}
return false;
}
//---------------------------------------------------------
// midiNoteReceived
//---------------------------------------------------------
void MuseScore::midiNoteReceived(int channel, int pitch, int velo)
{
static const int THRESHOLD_DRUMS = 5; // iterations required before consecutive drum notes
2012-05-26 14:49:10 +02:00
// are not considered part of a chord
static int active = 0;
static int iterDrums = 0;
static int activeDrums = 0;
2012-05-26 14:49:10 +02:00
if (!midiinEnabled())
return;
// qDebug("midiNoteReceived %d %d %d", channel, pitch, velo);
if (_midiRecordId != -1) {
preferences.updateMidiRemote(_midiRecordId, MIDI_REMOTE_TYPE_NOTEON, pitch);
2012-05-26 14:49:10 +02:00
_midiRecordId = -1;
if (preferenceDialog)
preferenceDialog->updateRemote();
return;
}
2012-10-29 16:37:24 +01:00
if (processMidiRemote(MIDI_REMOTE_TYPE_NOTEON, pitch, velo)) {
2012-05-26 14:49:10 +02:00
active = 0;
return;
}
2012-10-29 16:37:24 +01:00
2012-05-26 14:49:10 +02:00
QWidget* w = QApplication::activeModalWidget();
if (!cv || w) {
active = 0;
2012-05-26 14:49:10 +02:00
return;
}
if (velo) {
/*
* Since some drum modules do not overlap note on / off messages
* we need to take a bit of a different tactic to allow chords
* to be entered.
*
* Rather than decrement active for drums (midi on ch10),
* we'll just assume that if read() has been called a couple
* times in a row without a drum note, that this note is not
* part of a cord.
*/
if (channel == 0x09) {
if (iterDrums >= THRESHOLD_DRUMS)
activeDrums = 0;
iterDrums = 0;
cv->midiNoteReceived(pitch, activeDrums > 0, velo);
}
else {
//qDebug(" midiNoteReceived %d active %d", pitch, active);
cv->midiNoteReceived(pitch, active > 0, velo);
++active;
2012-05-26 14:49:10 +02:00
}
}
else {
/*
* Since a note may be assigned to a midi_remote,
* don't decrease active below zero on noteoff.
*/
if ((channel != 0x09) && (active > 0))
2012-05-26 14:49:10 +02:00
--active;
if ((channel == 0x09) && (activeDrums > 0))
--activeDrums;
cv->midiNoteReceived(pitch, false, velo);
2012-05-26 14:49:10 +02:00
}
if (_pianoTools && _pianoTools->isVisible()) {
if (velo)
_pianoTools->pressPitch(pitch);
else
_pianoTools->releasePitch(pitch);
}
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// midiCtrlReceived
//---------------------------------------------------------
void MuseScore::midiCtrlReceived(int controller, int value)
2012-05-26 14:49:10 +02:00
{
if (!midiinEnabled())
return;
if (_midiRecordId != -1) {
preferences.updateMidiRemote(_midiRecordId, MIDI_REMOTE_TYPE_CTRL, controller);
2012-05-26 14:49:10 +02:00
_midiRecordId = -1;
if (preferenceDialog)
preferenceDialog->updateRemote();
return;
}
// when value is 0 (usually when a key is released ) nothing happens
if (processMidiRemote(MIDI_REMOTE_TYPE_CTRL, controller, value))
2012-05-26 14:49:10 +02:00
return;
}
//---------------------------------------------------------
// removeTab
//---------------------------------------------------------
void MuseScore::removeTab()
{
2016-03-11 12:18:46 +01:00
int n = scoreList.indexOf(cs->masterScore());
2012-05-26 14:49:10 +02:00
if (n == -1) {
qDebug("removeTab: %p not found", cs);
return;
}
removeTab(n);
}
void MuseScore::removeTab(int i)
{
2016-03-11 12:18:46 +01:00
MasterScore* score = scoreList.value(i);
2012-09-12 16:19:03 +02:00
2012-05-26 14:49:10 +02:00
if (score == 0)
return;
QString tmpName = score->tmpName();
2012-05-26 14:49:10 +02:00
if (checkDirty(score))
return;
if (seq && seq->score() == score) {
seq->stopWait();
2012-05-26 14:49:10 +02:00
seq->setScoreView(0);
2013-02-07 00:19:53 +01:00
}
2012-05-26 14:49:10 +02:00
int idx1 = tab1->currentIndex();
bool firstTab = tab1->view(idx1) == cv;
midiPanelOnCloseFile(score->importedFilePath());
2012-05-26 14:49:10 +02:00
scoreList.removeAt(i);
2012-05-26 14:49:10 +02:00
tab1->blockSignals(true);
tab1->removeTab(i);
tab1->blockSignals(false);
if (tab2) {
tab2->blockSignals(true);
tab2->removeTab(i);
tab2->blockSignals(false);
}
cs = 0;
cv = 0;
int n = scoreList.size();
2013-02-15 22:39:56 +01:00
if (n == 0)
setCurrentScoreView(nullptr);
else
2012-05-26 14:49:10 +02:00
setCurrentScoreView((firstTab ? tab1 : tab2)->view());
writeSessionFile(false);
if (!tmpName.isEmpty()) {
QFile f(tmpName);
2012-05-26 14:49:10 +02:00
f.remove();
}
delete score;
// Shouldn't be necessary... but fix #21841
update();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// loadTranslation
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
void loadTranslation(QString filename, QString _localeName)
2012-05-26 14:49:10 +02:00
{
2017-02-23 11:41:53 +01:00
QString userPrefix = dataPath + "/locale/"+ filename +"_";
QString defaultPrefix = mscoreGlobalShare + "locale/"+ filename +"_";
QString userlp = userPrefix + _localeName;
QString defaultlp = defaultPrefix + _localeName;
2017-02-23 11:41:53 +01:00
QString lp = defaultlp;
QFileInfo userFi(userlp + ".qm");
QFileInfo defaultFi(defaultlp + ".qm");
2017-02-23 11:41:53 +01:00
if (!defaultFi.exists()) { // try with a shorter locale name
QString shortLocaleName = _localeName.left(_localeName.lastIndexOf("_"));
QString shortDefaultlp = defaultPrefix + shortLocaleName;
QFileInfo shortDefaultFi(shortDefaultlp + ".qm");
2017-02-23 11:41:53 +01:00
if (shortDefaultFi.exists()) {
userlp = userPrefix + shortLocaleName;
userFi = QFileInfo(userlp + ".qm");
defaultFi = shortDefaultFi;
defaultlp = shortDefaultlp;
}
2014-09-06 07:52:20 +02:00
}
2014-08-28 14:50:17 +02:00
if (userFi.exists()) {
if (userFi.lastModified() > defaultFi.lastModified())
lp = userlp;
2017-02-23 11:41:53 +01:00
// else
//REVIEW QFile::remove(userlp + ".qm");
}
2016-07-15 01:01:51 +02:00
if (MScore::debugMode)
qDebug("load translator <%s>", qPrintable(lp));
2017-02-23 11:41:53 +01:00
QTranslator* translator = new QTranslator;
bool success = translator->load(lp);
2016-07-15 01:01:51 +02:00
if (success) {
qApp->installTranslator(translator);
translatorList.append(translator);
}
2016-07-15 01:01:51 +02:00
else {
if (MScore::debugMode)
qDebug("load translator <%s> failed", qPrintable(lp));
delete translator;
}
}
//---------------------------------------------------------
// setLocale
//---------------------------------------------------------
void setMscoreLocale(QString _localeName)
{
2017-02-23 11:41:53 +01:00
for (QTranslator* t : translatorList) {
qApp->removeTranslator(t);
delete t;
}
translatorList.clear();
if (MScore::debugMode)
qDebug("configured localeName <%s>", qPrintable(_localeName));
if (_localeName.toLower() == "system") {
_localeName = QLocale::system().name();
if (MScore::debugMode)
qDebug("real localeName <%s>", qPrintable(_localeName));
if (_localeName == "en_AU") {
_localeName = "en_GB"; // otherwise Australia would fall back to US English
if (MScore::debugMode)
qDebug("modified localeName <%s>", qPrintable(_localeName));
}
}
2012-05-26 14:49:10 +02:00
// find the most recent translation file
// try to replicate QTranslator.load algorithm in our particular case
loadTranslation("mscore", _localeName);
loadTranslation("instruments", _localeName);
2012-05-26 14:49:10 +02:00
QString resourceDir;
#if defined(Q_OS_MAC) || defined(Q_OS_WIN)
2012-05-26 14:49:10 +02:00
resourceDir = mscoreGlobalShare + "locale/";
#else
resourceDir = QLibraryInfo::location(QLibraryInfo::TranslationsPath);
#endif
QTranslator* qtTranslator = new QTranslator;
if (MScore::debugMode)
qDebug("load translator <qt_%s> from <%s>",
qPrintable(_localeName), qPrintable(resourceDir));
2012-05-26 14:49:10 +02:00
if (!qtTranslator->load(QLatin1String("qt_") + _localeName, resourceDir) && MScore::debugMode)
qDebug("load translator <qt_%s> failed", qPrintable(_localeName));
2012-05-26 14:49:10 +02:00
else {
qApp->installTranslator(qtTranslator);
translatorList.append(qtTranslator);
}
QLocale locale(_localeName);
QLocale::setDefault(locale);
qApp->setLayoutDirection(locale.textDirection());
2012-05-26 14:49:10 +02:00
// initShortcuts();
}
//---------------------------------------------------------
// loadScores
// load scores for a new session
//---------------------------------------------------------
static void loadScores(const QStringList& argv)
{
int currentScoreView = 0;
if (argv.isEmpty()) {
if (startWithNewScore)
mscore->newFile();
else {
switch (preferences.sessionStart()) {
2014-07-01 14:44:45 +02:00
case SessionStart::LAST:
2012-05-26 14:49:10 +02:00
{
QSettings settings;
int n = settings.value("scores", 0).toInt();
int c = settings.value("currentScore", 0).toInt();
for (int i = 0; i < n; ++i) {
QString s = settings.value(QString("score-%1").arg(i),"").toString();
2016-03-11 12:18:46 +01:00
MasterScore* score = mscore->readScore(s);
2012-09-29 16:46:45 +02:00
if (score) {
2012-05-26 14:49:10 +02:00
int view = mscore->appendScore(score);
if (i == c)
currentScoreView = view;
}
}
}
break;
2014-07-01 14:44:45 +02:00
case SessionStart::EMPTY:
2012-05-26 14:49:10 +02:00
break;
2014-07-01 14:44:45 +02:00
case SessionStart::NEW:
2012-05-26 14:49:10 +02:00
mscore->newFile();
break;
2014-07-01 14:44:45 +02:00
case SessionStart::SCORE:
2012-05-26 14:49:10 +02:00
{
QString startScore = preferences.getString(PREF_APP_STARTUP_STARTSCORE);
MasterScore* score = mscore->readScore(startScore);
if (startScore.startsWith(":/") && score) {
2017-03-30 15:36:32 +02:00
score->setName(mscore->createDefaultName());
// TODO score->setPageFormat(*MScore::defaultStyle().pageFormat());
score->doLayout();
score->setCreated(true);
}
if (score == 0) {
2014-11-14 12:29:17 +01:00
score = mscore->readScore(":/data/My_First_Score.mscz");
if (score) {
2017-03-30 15:36:32 +02:00
score->setName(mscore->createDefaultName());
// TODO score->setPageFormat(*MScore::defaultStyle().pageFormat());
score->doLayout();
score->setCreated(true);
}
}
2012-09-29 16:46:45 +02:00
if (score)
2012-05-26 14:49:10 +02:00
currentScoreView = mscore->appendScore(score);
}
break;
}
}
}
else {
foreach(const QString& name, argv) {
if (name.isEmpty())
continue;
2016-03-11 12:18:46 +01:00
MasterScore* score = mscore->readScore(name);
2012-09-29 16:46:45 +02:00
if (score) {
2012-05-26 14:49:10 +02:00
mscore->appendScore(score);
scoresOnCommandline = true;
2014-02-28 11:14:43 +01:00
if(!MScore::noGui) {
mscore->addRecentScore(score);
mscore->writeSessionFile(false);
}
2012-05-26 14:49:10 +02:00
}
}
}
if (mscore->noScore())
currentScoreView = -1;
mscore->setCurrentView(0, currentScoreView);
mscore->setCurrentView(1, currentScoreView);
}
//---------------------------------------------------------
// doConvert
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
static bool doConvert(Score* cs, QString fn, QString plugin = "")
2012-05-26 14:49:10 +02:00
{
bool rv = true;
LayoutMode layoutMode = cs->layoutMode();
2016-05-18 15:43:47 +02:00
cs->setLayoutMode(LayoutMode::PAGE);
if (cs->layoutMode() != layoutMode) {
cs->setLayoutMode(LayoutMode::PAGE);
cs->doLayout();
}
if (!styleFile.isEmpty()) {
QFile f(styleFile);
if (f.open(QIODevice::ReadOnly))
2017-01-05 11:23:47 +01:00
cs->style().load(&f);
}
if (!plugin.isEmpty()) {
mscore->setCurrentScore(cs);
LayoutMode layoutMode = cs->layoutMode();
2017-05-06 15:33:04 +02:00
if (layoutMode != LayoutMode::PAGE) {
cs->setLayoutMode(LayoutMode::PAGE);
cs->doLayout();
}
if (mscore->loadPlugin(plugin))
mscore->pluginTriggered(0);
mscore->unloadPlugins();
2017-05-06 15:33:04 +02:00
if (layoutMode != cs->layoutMode()) {
cs->setLayoutMode(layoutMode);
cs->doLayout();
}
}
if (fn.endsWith(".mscx")) {
QFileInfo fi(fn);
2016-09-14 12:08:14 +02:00
if (!cs->saveFile(fi))
return false;
return true;
}
else if (fn.endsWith(".mscz")) {
QFileInfo fi(fn);
2016-09-14 12:08:14 +02:00
if (!cs->saveCompressedFile(fi, false))
return false;
return true;
}
else if (fn.endsWith(".xml") || fn.endsWith(".musicxml")) {
rv = saveXml(cs, fn);
}
else if (fn.endsWith(".mxl")) {
rv = saveMxl(cs, fn);
}
else if (fn.endsWith(".mid"))
return mscore->saveMidi(cs, fn);
else if (fn.endsWith(".pdf")) {
if (!exportScoreParts) {
2016-09-03 17:45:59 +02:00
rv = mscore->savePdf(cs, fn);
}
else {
if (cs->excerpts().size() == 0) {
auto excerpts = Excerpt::createAllExcerpt(cs->masterScore());
for (Excerpt* e : excerpts) {
Score* nscore = new Score(e->oscore());
e->setPartScore(nscore);
2018-03-27 15:36:00 +02:00
nscore->style().set(Sid::createMultiMeasureRests, true);
Excerpt::createExcerpt(e);
2016-10-10 18:37:28 +02:00
cs->startCmd();
cs->undo(new AddExcerpt(e));
cs->endCmd();
}
}
QList<Score*> scores;
scores.append(cs);
for (Excerpt* e : cs->excerpts())
scores.append(e->partScore());
return mscore->savePdf(scores, fn);
}
}
else if (fn.endsWith(".png")) {
if (!exportScoreParts)
return mscore->savePng(cs, fn);
else {
if (cs->excerpts().size() == 0) {
auto excerpts = Excerpt::createAllExcerpt(cs->masterScore());
for (Excerpt* e: excerpts) {
Score* nscore = new Score(e->oscore());
e->setPartScore(nscore);
2016-07-31 15:23:11 +02:00
nscore->setExcerpt(e);
2016-10-10 18:37:28 +02:00
// nscore->setName(e->title()); // needed before AddExcerpt
2018-03-27 15:36:00 +02:00
nscore->style().set(Sid::createMultiMeasureRests, true);
Excerpt::createExcerpt(e);
2016-10-10 18:37:28 +02:00
cs->startCmd();
cs->undo(new AddExcerpt(e));
cs->endCmd();
}
}
if (!mscore->savePng(cs, fn))
return false;
int idx = 0;
int padding = QString("%1").arg(cs->excerpts().size()).size();
for (Excerpt* e: cs->excerpts()) {
QString suffix = QString("__excerpt__%1.png").arg(idx, padding, 10, QLatin1Char('0'));
QString excerptFn = fn.left(fn.size() - 4) + suffix;
if (!mscore->savePng(e->partScore(), excerptFn))
return false;
idx++;
}
return true;
}
}
else if (fn.endsWith(".svg")) {
rv = mscore->saveSvg(cs, fn);
}
2012-05-26 14:49:10 +02:00
#ifdef HAS_AUDIOFILE
else if (fn.endsWith(".wav") || fn.endsWith(".ogg") || fn.endsWith(".flac"))
2014-11-14 11:44:48 +01:00
return mscore->saveAudio(cs, fn);
2012-05-26 14:49:10 +02:00
#endif
#ifdef USE_LAME
else if (fn.endsWith(".mp3"))
2012-05-26 14:49:10 +02:00
return mscore->saveMp3(cs, fn);
#endif
else if (fn.endsWith(".spos")) {
rv = savePositions(cs, fn, true);
}
else if (fn.endsWith(".mpos")) {
rv = savePositions(cs, fn, false);
}
else if (fn.endsWith(".mlog"))
return cs->sanityCheck(fn);
else if (plugin.isEmpty()) {
qDebug("don't know how to convert to %s", qPrintable(outFileName));
return false;
}
2016-05-18 15:43:47 +02:00
if (layoutMode != cs->layoutMode()) {
cs->setLayoutMode(layoutMode);
cs->doLayout();
}
return rv;
}
//---------------------------------------------------------
// convert
//---------------------------------------------------------
static bool convert(const QString& inFile, const QString& outFile, const QString& plugin = "")
{
if (inFile.isEmpty() || (outFile.isEmpty() && plugin.isEmpty())) {
fprintf(stderr, "cannot convert <%s> to <%s>\n", qPrintable(inFile), qPrintable(outFile));
return false;
}
fprintf(stderr, "convert <%s> to <%s>\n", qPrintable(inFile), qPrintable(outFile));
MasterScore* score = mscore->readScore(inFile);
if (!score)
return false;
if (!doConvert(score, outFile, plugin)) {
delete score;
return false;
}
delete score;
return true;
}
//---------------------------------------------------------
// doProcessJob
//---------------------------------------------------------
static bool doProcessJob(QString jsonFile)
{
QFile f(jsonFile);
if (!f.open(QIODevice::ReadOnly)) {
fprintf(stderr, "cannon open json file <%s>\n", qPrintable(jsonFile));
return false;
}
QJsonParseError pe;
QJsonDocument doc = QJsonDocument::fromJson(f.readAll(), &pe);
if (pe.error != QJsonParseError::NoError) {
fprintf(stderr, "error reading json file <%s> at %d: %s\n",
qPrintable(jsonFile), pe.offset, qPrintable(pe.errorString()));
return false;
}
if (!doc.isArray()) {
fprintf(stderr, "json file <%s> is not an array\n", qPrintable(jsonFile));
return false;
}
QJsonArray a = doc.array();
for (const auto i : a) {
QString inFile;
QString outFile;
QString plugin;
if (!i.isObject()) {
fprintf(stderr, "array value is not an object\n");
return false;
}
QJsonObject obj = i.toObject();
for (const auto& key : obj.keys()) {
QString val = obj.value(key).toString();
if (key == "in")
inFile = val;
else if (key == "out")
outFile = val;
else if (key == "plugin")
plugin = val;
else {
fprintf(stderr, "unknown key <%s>\n", qPrintable(key));
return false;
}
}
if (!convert(inFile, outFile, plugin))
2012-05-26 14:49:10 +02:00
return false;
}
return true;
}
//---------------------------------------------------------
// processNonGui
//---------------------------------------------------------
static bool processNonGui(const QStringList& argv)
{
if (pluginMode) {
loadScores(argv);
QString pn(pluginName);
bool res = false;
if (mscore->loadPlugin(pn)){
Score* cs = mscore->currentScore();
if (!styleFile.isEmpty()) {
QFile f(styleFile);
if (f.open(QIODevice::ReadOnly))
2017-01-05 11:23:47 +01:00
cs->style().load(&f);
}
2016-05-18 15:43:47 +02:00
LayoutMode layoutMode = cs->layoutMode();
if (layoutMode != LayoutMode::PAGE) {
cs->setLayoutMode(LayoutMode::PAGE);
cs->doLayout();
}
mscore->pluginTriggered(0);
2016-05-18 15:43:47 +02:00
if (layoutMode != cs->layoutMode()) {
cs->setLayoutMode(layoutMode);
cs->doLayout();
}
res = true;
2012-05-26 14:49:10 +02:00
}
if (!converterMode)
return res;
}
if (converterMode) {
if (processJob)
return doProcessJob(jsonFileName);
else
return convert(argv[0], outFileName);
2012-05-26 14:49:10 +02:00
}
if (!extensionName.isEmpty()) {
QFileInfo fi(extensionName);
QString suffix = fi.suffix().toLower();
if (suffix == "muxt")
return mscore->importExtension(extensionName);
else {
fprintf(stderr, "cannot install extension: <%s>\n", qPrintable(extensionName));
return false;
}
}
return true;
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// Message handler
//---------------------------------------------------------
#if defined(QT_DEBUG) && defined(Q_OS_WIN)
static void mscoreMessageHandler(QtMsgType type, const QMessageLogContext &context,const QString &msg)
2012-05-26 14:49:10 +02:00
{
QTextStream err(stderr);
QByteArray localMsg = msg.toLocal8Bit();
2014-02-28 14:25:47 +01:00
2012-05-26 14:49:10 +02:00
switch (type) {
case QtInfoMsg:
err << "Info: " << localMsg.constData() << " (" << context.file << ":" << context.line << ", " << context.function << ")" << endl;
break;
2012-05-26 14:49:10 +02:00
case QtDebugMsg:
err << "Debug: " << localMsg.constData() << " (" << context.file << ":" << context.line << ", " << context.function << ")" << endl;
2012-05-26 14:49:10 +02:00
break;
case QtWarningMsg:
err << "Warning: " << localMsg.constData() << " (" << context.file << ":" << context.line << ", " << context.function << ")" << endl;
2012-05-26 14:49:10 +02:00
break;
case QtCriticalMsg: // same as QtSystemMsg
err << "Critical: " << localMsg.constData() << " (" << context.file << ":" << context.line << ", " << context.function << ")" << endl;
2012-05-26 14:49:10 +02:00
break;
case QtFatalMsg: // set your breakpoint here, if you want to catch the abort
err << "Fatal: " << localMsg.constData() << " (" << context.file << ":" << context.line << ", " << context.function << ")" << endl;
2012-05-26 14:49:10 +02:00
abort();
}
}
#endif
//---------------------------------------------------------
// synthesizerFactory
// create and initialize the master synthesizer
//---------------------------------------------------------
MasterSynthesizer* synthesizerFactory()
{
MasterSynthesizer* ms = new MasterSynthesizer();
2013-04-03 12:13:23 +02:00
FluidS::Fluid* fluid = new FluidS::Fluid();
ms->registerSynthesizer(fluid);
2013-04-03 20:20:36 +02:00
#ifdef AEOLUS
2013-05-13 18:49:17 +02:00
ms->registerSynthesizer(::createAeolus());
2013-04-03 20:20:36 +02:00
#endif
#ifdef ZERBERUS
ms->registerSynthesizer(createZerberus());
2013-04-03 20:20:36 +02:00
#endif
ms->registerEffect(0, new NoEffect);
ms->registerEffect(0, new ZitaReverb);
2015-09-08 15:39:04 +02:00
ms->registerEffect(0, new Compressor);
2013-05-13 18:49:17 +02:00
// ms->registerEffect(0, new Freeverb);
ms->registerEffect(1, new NoEffect);
ms->registerEffect(1, new ZitaReverb);
2015-09-08 15:39:04 +02:00
ms->registerEffect(1, new Compressor);
2013-05-13 18:49:17 +02:00
// ms->registerEffect(1, new Freeverb);
ms->setEffect(0, 1);
ms->setEffect(1, 0);
return ms;
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// unstable
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
bool MuseScore::unstable()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
#ifdef MSCORE_UNSTABLE
return true;
2012-05-26 14:49:10 +02:00
#else
2013-05-13 18:49:17 +02:00
return false;
2012-05-26 14:49:10 +02:00
#endif
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// MuseScoreApplication::event (mac only)
//---------------------------------------------------------
2013-04-17 14:00:26 +02:00
2015-02-19 19:55:17 +01:00
bool MuseScoreApplication::event(QEvent* event)
2013-05-13 18:49:17 +02:00
{
switch(event->type()) {
case QEvent::FileOpen:
2013-06-24 22:07:44 +02:00
// store names of files requested to be loaded by OS X to be handled later
// this event is generated when a file is dragged onto the MuseScore icon
// in the dock and MuseScore is not running yet
2013-05-13 18:49:17 +02:00
paths.append(static_cast<QFileOpenEvent *>(event)->file());
return true;
default:
return QtSingleApplication::event(event);
2013-05-08 15:27:43 +02:00
}
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// eventFilter
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
bool MuseScore::eventFilter(QObject *obj, QEvent *event)
{
switch(event->type()) {
#ifdef Q_OS_MAC
2013-05-13 18:49:17 +02:00
case QEvent::FileOpen:
2013-06-24 22:07:44 +02:00
// open files requested to be loaded by OS X
// this event is generated when a file is dragged onto the MuseScore icon
// in the dock when MuseScore is already running
scoresOnCommandline = true;
2013-05-13 18:49:17 +02:00
handleMessage(static_cast<QFileOpenEvent *>(event)->file());
return true;
#endif
2015-02-19 19:55:17 +01:00
case QEvent::MouseMove: {
QMouseEvent* me = static_cast<QMouseEvent*>(event);
globalX = me->globalX();
globalY = me->globalY();
return QMainWindow::eventFilter(obj, event);
}
case QEvent::StatusTip:
return true; // prevent updates to the status bar
case QEvent::KeyPress:
{
QKeyEvent* e = static_cast<QKeyEvent*>(event);
if(obj->isWidgetType() && e->key() == Qt::Key_Escape && e->modifiers() == Qt::NoModifier) {
// Close the search dialog when Escape is pressed:
if(_searchDialog != 0)
endSearch();
if (isActiveWindow()) {
obj->event(e);
if(currentScoreView())
currentScoreView()->setFocus();
else
mscore->setFocus();
return true;
}
QWidget* w = static_cast<QWidget*>(obj);
if(getPaletteBox()->isAncestorOf(w) ||
2015-02-03 01:55:16 +01:00
inspector()->isAncestorOf(w) ||
(selectionWindow && selectionWindow->isAncestorOf(w))) {
activateWindow();
if(currentScoreView())
currentScoreView()->setFocus();
else
mscore->setFocus();
return true;
}
}
break;
}
2013-05-13 18:49:17 +02:00
default:
return QMainWindow::eventFilter(obj, event);
2012-05-26 14:49:10 +02:00
}
return QMainWindow::eventFilter(obj, event);
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// hasToCheckForUpdate
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
bool MuseScore::hasToCheckForUpdate()
{
if (ucheck)
return ucheck->hasToCheck();
else
return false;
}
//---------------------------------------------------------
// hasToCheckForExtensionsUpdate
//---------------------------------------------------------
bool MuseScore::hasToCheckForExtensionsUpdate()
{
return packUChecker ? packUChecker->hasToCheck() : false;
2013-05-13 18:49:17 +02:00
}
2013-03-01 17:14:21 +01:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// checkForUpdate
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::checkForUpdate()
{
if (ucheck)
2015-03-09 17:20:23 +01:00
ucheck->check(version(), sender() != 0);
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// checkForExtensionsUpdate
//---------------------------------------------------------
void MuseScore::checkForExtensionsUpdate()
{
if (packUChecker)
packUChecker->check();
}
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// readLanguages
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
bool MuseScore::readLanguages(const QString& path)
{
2014-02-24 22:42:07 +01:00
//: The default language of the operating system. NOT a music system.
_languages.append(LanguageItem("system", tr("System")));
2013-05-13 18:49:17 +02:00
QFile qf(path);
if (qf.exists()){
QDomDocument doc;
int line, column;
QString err;
if (!doc.setContent(&qf, false, &err, &line, &column)) {
QString error;
error.sprintf(qPrintable(tr("Error reading language file %s at line %d column %d: %s\n")),
2013-05-13 18:49:17 +02:00
qPrintable(qf.fileName()), line, column, qPrintable(err));
QMessageBox::warning(0,
QWidget::tr("Load Languages Failed:"),
2013-05-13 18:49:17 +02:00
error,
QString::null, QWidget::tr("Quit"), QString::null, 0, 1);
return false;
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
for (QDomElement e = doc.documentElement(); !e.isNull(); e = e.nextSiblingElement()) {
if(e.tagName() == "languages") {
for (e = e.firstChildElement(); !e.isNull(); e = e.nextSiblingElement()) {
if (e.tagName() == "language") {
QString code = e.attribute(QString("code"));
QString name = e.attribute(QString("name"));
QString handbook = e.attribute(QString("handbook"));
_languages.append(LanguageItem(code, name, handbook));
}
}
}
}
return true;
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
return false;
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// clipboardChanged
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::clipboardChanged()
{
#if 0
const QMimeData* ms = QApplication::clipboard()->mimeData();
QStringList formats = ms->formats();
2012-06-11 14:38:33 +02:00
2013-05-13 18:49:17 +02:00
bool flag = ms->hasFormat(mimeSymbolFormat)
|| ms->hasFormat(mimeStaffListFormat)
|| ms->hasFormat(mimeSymbolListFormat)
|| ms->hasText();
// TODO: depends on selection state
#endif
2012-06-11 14:38:33 +02:00
2013-05-13 18:49:17 +02:00
bool flag = true;
getAction("paste")->setEnabled(flag);
getAction("swap")->setEnabled(flag);
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// inputMethodLocaleChanged
//---------------------------------------------------------
void MuseScore::inputMethodLocaleChanged()
{
qDebug("Input method QLocale::Script enum now #%d.", QGuiApplication::inputMethod()->locale().script());
}
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// showModeText
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::showModeText(const QString& s)
{
_modeText->setText(s);
_modeText->show();
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// changeState
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::changeState(ScoreState val)
{
if (MScore::debugMode)
qDebug("MuseScore::changeState: %s", stateName(val));
// disallow change to edit modes if currently in play mode
if (_sstate == STATE_PLAY && (val == STATE_EDIT || val & STATE_ALLTEXTUAL_EDIT || val & STATE_NOTE_ENTRY))
return;
2013-05-13 18:49:17 +02:00
static const char* stdNames[] = {
"note-longa", "note-breve", "pad-note-1", "pad-note-2", "pad-note-4",
"pad-note-8", "pad-note-16", "pad-note-32", "pad-note-64", "pad-note-128", "pad-rest", "rest"};
2013-05-13 18:49:17 +02:00
static const char* tabNames[] = {
"note-longa-TAB", "note-breve-TAB", "pad-note-1-TAB", "pad-note-2-TAB", "pad-note-4-TAB",
"pad-note-8-TAB", "pad-note-16-TAB", "pad-note-32-TAB", "pad-note-64-TAB", "pad-note-128-TAB", "pad-rest-TAB", "rest-TAB"};
2016-08-08 17:45:45 +02:00
bool intoTAB = (_sstate != STATE_NOTE_ENTRY_STAFF_TAB) && (val == STATE_NOTE_ENTRY_STAFF_TAB);
bool fromTAB = (_sstate == STATE_NOTE_ENTRY_STAFF_TAB) && (val != STATE_NOTE_ENTRY_STAFF_TAB);
2013-05-13 18:49:17 +02:00
// if activating TAB note entry, swap "pad-note-...-TAB" shorctuts into "pad-note-..." actions
if (intoTAB) {
for (unsigned i = 0; i < sizeof(stdNames)/sizeof(char*); ++i) {
QAction* act = getAction(stdNames[i]);
2014-11-14 11:44:48 +01:00
const Shortcut* srt = Shortcut::getShortcut(tabNames[i]);
2013-05-13 18:49:17 +02:00
act->setShortcuts(srt->keys());
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
}
// if de-ativating TAB note entry, restore shortcuts for "pad-note-..." actions
else if (fromTAB) {
for (unsigned i = 0; i < sizeof(stdNames)/sizeof(char*); ++i) {
QAction* act = getAction(stdNames[i]);
2014-11-14 11:44:48 +01:00
const Shortcut* srt = Shortcut::getShortcut(stdNames[i]);
2013-05-13 18:49:17 +02:00
act->setShortcuts(srt->keys());
}
}
2013-10-05 14:03:34 +02:00
bool enable = (val != STATE_DISABLED) && (val != STATE_LOCK);
for (const Shortcut* s : Shortcut::shortcuts()) {
2013-05-13 18:49:17 +02:00
QAction* a = s->action();
if (!a)
continue;
2014-11-14 18:08:02 +01:00
if (enable && (s->key() == "undo"))
2016-03-10 10:41:31 +01:00
a->setEnabled((s->state() & val) && (cs ? cs->undoStack()->canUndo() : false));
2014-11-14 18:08:02 +01:00
else if (enable && (s->key() == "redo"))
2016-03-10 10:41:31 +01:00
a->setEnabled((s->state() & val) && (cs ? cs->undoStack()->canRedo() : false));
2014-11-14 18:08:02 +01:00
else if (enable && (s->key() == "cut"))
a->setEnabled(cs && cs->selection().state() != SelState::NONE);
2014-11-14 18:08:02 +01:00
else if (enable && (s->key() == "copy"))
a->setEnabled(cs && (cs->selection().state() != SelState::NONE || val == STATE_FOTO));
2014-11-14 18:08:02 +01:00
else if (enable && (s->key() == "select-similar-range"))
2014-07-14 13:24:47 +02:00
a->setEnabled(cs && cs->selection().state() == SelState::RANGE);
2014-11-14 18:08:02 +01:00
else if (enable && (s->key() == "synth-control")) {
2013-05-13 18:49:17 +02:00
Driver* driver = seq ? seq->driver() : 0;
// a->setEnabled(driver && driver->getSynth());
if (MScore::debugMode)
qDebug("disable synth control");
a->setEnabled(driver);
2013-04-12 10:21:13 +02:00
}
else {
a->setEnabled(s->state() & val);
2013-04-12 10:21:13 +02:00
}
2012-05-26 14:49:10 +02:00
}
2014-11-17 13:06:34 +01:00
if (getAction("file-part-export")->isEnabled())
2016-03-10 10:41:31 +01:00
getAction("file-part-export")->setEnabled(cs && cs->masterScore()->excerpts().size() > 0);
if (getAction("join-measures")->isEnabled())
getAction("join-measures")->setEnabled(cs && cs->masterScore()->excerpts().size() == 0);
if (getAction("split-measure")->isEnabled())
2016-03-10 10:41:31 +01:00
getAction("split-measure")->setEnabled(cs && cs->masterScore()->excerpts().size() == 0);
2015-04-03 18:23:51 +02:00
2016-03-10 10:41:31 +01:00
//getAction("split-measure")->setEnabled(cs->masterScore()->excerpts().size() == 0);
2013-05-13 18:49:17 +02:00
// disabling top level menu entries does not
// work for MAC
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
QList<QObject*> ol = menuBar()->children();
foreach(QObject* o, ol) {
QMenu* menu = qobject_cast<QMenu*>(o);
if (!menu)
continue;
QString s(menu->objectName());
if (s == "File" || s == "Help" || s == "Edit" || s == "Plugins" || s == "View")
2013-05-13 18:49:17 +02:00
continue;
menu->setEnabled(enable);
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
menuWorkspaces->setEnabled(enable);
2012-05-26 14:49:10 +02:00
transportTools->setEnabled(enable && !noSeq && seq && seq->isRunning());
2013-05-13 18:49:17 +02:00
cpitchTools->setEnabled(enable);
mag->setEnabled(enable);
entryTools->setEnabled(enable);
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (_sstate == STATE_FOTO)
updateInspector();
//always hide drumtools to support hiding on switching docs, switching node entry mode off, etc.
//it will be shown later if (_sstate == STATE_NOTE_ENTRY_STAFF_DRUM)
showDrumTools(0, 0);
2013-03-01 17:14:21 +01:00
2017-05-02 14:17:31 +02:00
switch (val) {
2013-05-13 18:49:17 +02:00
case STATE_DISABLED:
2015-01-26 21:12:29 +01:00
showModeText(tr("No score"));
2013-05-13 18:49:17 +02:00
if (debugger)
debugger->hide();
showPianoKeyboard(false);
break;
case STATE_NORMAL:
_modeText->hide();
break;
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY:
if (cv && !cv->noteEntryMode())
2017-07-19 17:53:45 +02:00
cv->cmd("note-input");
// fall through
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY_STAFF_PITCHED:
if (getAction("note-input-repitch")->isChecked()) {
showModeText(tr("Repitch input mode"));
cs->setNoteEntryMethod(NoteEntryMethod::REPITCH);
2016-08-08 17:45:45 +02:00
val = STATE_NOTE_ENTRY_METHOD_REPITCH;
}
else if (getAction("note-input-rhythm")->isChecked()) {
showModeText(tr("Rhythm input mode"));
cs->setNoteEntryMethod(NoteEntryMethod::RHYTHM);
2016-08-08 17:45:45 +02:00
val = STATE_NOTE_ENTRY_METHOD_RHYTHM;
}
else if (getAction("note-input-realtime-auto")->isChecked()) {
showModeText(tr("Realtime (automatic) note input mode"));
cs->setNoteEntryMethod(NoteEntryMethod::REALTIME_AUTO);
2016-08-08 17:45:45 +02:00
val = STATE_NOTE_ENTRY_METHOD_REALTIME_AUTO;
}
else if (getAction("note-input-realtime-manual")->isChecked()) {
showModeText(tr("Realtime (manual) note input mode"));
cs->setNoteEntryMethod(NoteEntryMethod::REALTIME_MANUAL);
2016-08-08 17:45:45 +02:00
val = STATE_NOTE_ENTRY_METHOD_REALTIME_MANUAL;
}
else {
showModeText(tr("Steptime note input mode"));
cs->setNoteEntryMethod(NoteEntryMethod::STEPTIME);
2016-08-08 17:45:45 +02:00
val = STATE_NOTE_ENTRY_METHOD_STEPTIME;
}
2013-05-13 18:49:17 +02:00
break;
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY_STAFF_DRUM:
2013-05-13 18:49:17 +02:00
{
if (getAction("note-input-repitch")->isChecked())
cs->setNoteEntryMethod(NoteEntryMethod::REPITCH);
showModeText(tr("Drum input mode"));
2013-05-13 18:49:17 +02:00
InputState& is = cs->inputState();
showDrumTools(is.drumset(), cs->staff(is.track() / VOICES));
if (_drumTools)
is.setDrumNote(_drumTools->selectedDrumNote());
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
break;
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY_STAFF_TAB:
showModeText(tr("TAB input mode"));
2013-05-13 18:49:17 +02:00
break;
case STATE_EDIT:
2015-01-26 21:12:29 +01:00
showModeText(tr("Edit mode"));
2013-05-13 18:49:17 +02:00
break;
case STATE_TEXT_EDIT:
2015-01-26 21:12:29 +01:00
showModeText(tr("Text edit mode"));
2013-05-13 18:49:17 +02:00
break;
case STATE_LYRICS_EDIT:
2015-01-26 21:12:29 +01:00
showModeText(tr("Lyrics edit mode"));
2013-05-13 18:49:17 +02:00
break;
case STATE_HARMONY_FIGBASS_EDIT:
2015-01-26 21:12:29 +01:00
showModeText(tr("Chord symbol/figured bass edit mode"));
2013-05-13 18:49:17 +02:00
break;
case STATE_PLAY:
2015-01-26 21:12:29 +01:00
showModeText(tr("Play"));
2013-05-13 18:49:17 +02:00
break;
case STATE_FOTO:
showModeText(tr("Image capture mode"));
2013-05-13 18:49:17 +02:00
updateInspector();
break;
case STATE_LOCK:
2015-01-26 21:12:29 +01:00
showModeText(tr("Score locked"));
2013-05-13 18:49:17 +02:00
break;
default:
qFatal("MuseScore::changeState: illegal state %d", val);
break;
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
if (paletteBox)
paletteBox->setDisabled(val == STATE_PLAY || val == STATE_DISABLED);
if (selectionWindow)
selectionWindow->setDisabled(val == STATE_PLAY || val == STATE_DISABLED);
2013-05-13 18:49:17 +02:00
QAction* a = getAction("note-input");
2016-08-08 17:45:45 +02:00
bool noteEntry = val & STATE_NOTE_ENTRY;
2013-05-13 18:49:17 +02:00
a->setChecked(noteEntry);
_sstate = val;
2012-05-26 14:49:10 +02:00
2017-05-02 14:17:31 +02:00
Element* e = cv && (_sstate & STATE_ALLTEXTUAL_EDIT || _sstate == STATE_EDIT) ? cv->getEditElement() : 0;
2013-05-13 18:49:17 +02:00
if (!e) {
textTools()->hide();
if (textTools()->kbAction()->isChecked())
textTools()->kbAction()->setChecked(false);
2012-05-26 14:49:10 +02:00
}
else {
if (e->isTextBase()) {
textTools()->updateTools(cv->getEditData());
if (!(e->isFiguredBass() || e->isHarmony())) { // do not show text tools for f.b.
if (timelineScrollArea() && timelineScrollArea()->isVisible()) {
if (dockWidgetArea(timelineScrollArea()) != dockWidgetArea(textTools()) || timelineScrollArea()->isFloating()) {
QSizePolicy policy(QSizePolicy::Maximum, QSizePolicy::Maximum);
textTools()->widget()->setSizePolicy(policy);
}
else {
QSizePolicy policy(QSizePolicy::Expanding, QSizePolicy::Expanding);
textTools()->widget()->setSizePolicy(policy);
}
}
else {
QSizePolicy policy(QSizePolicy::Maximum, QSizePolicy::Maximum);
textTools()->widget()->setSizePolicy(policy);
}
if (timelineScrollArea())
splitDockWidget(textTools(), timelineScrollArea(), Qt::Vertical);
2014-03-28 10:49:19 +01:00
textTools()->show();
}
}
2014-08-29 19:18:39 +02:00
if (_inspector)
2017-03-31 13:03:15 +02:00
_inspector->update(e->score());
2012-05-26 14:49:10 +02:00
}
}
2014-11-13 13:04:04 +01:00
//---------------------------------------------------------
// saveDialogState
//---------------------------------------------------------
void MuseScore::saveDialogState(const char* name, QFileDialog* d)
{
if (d) {
settings.beginGroup(name);
settings.setValue("geometry", d->saveGeometry());
2014-11-13 13:04:04 +01:00
settings.setValue("state", d->saveState());
settings.endGroup();
}
}
//---------------------------------------------------------
// restoreDialogState
//---------------------------------------------------------
void MuseScore::restoreDialogState(const char* name, QFileDialog* d)
{
settings.beginGroup(name);
d->restoreGeometry(settings.value("geometry").toByteArray());
2014-11-13 13:04:04 +01:00
d->restoreState(settings.value("state").toByteArray());
settings.endGroup();
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// writeSettings
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::writeSettings()
2012-05-26 14:49:10 +02:00
{
2014-11-13 13:04:04 +01:00
// save score list
for (int i = 0; i < RECENT_LIST_SIZE; ++i)
settings.setValue(QString("recent-%1").arg(i), _recentScores.value(i));
settings.setValue("scores", scoreList.size());
if (cs) {
int curScore = scoreList.indexOf(cs->masterScore());
if (curScore == -1) // cs removed if new created and not modified
curScore = 0;
settings.setValue("currentScore", curScore);
2014-11-13 13:04:04 +01:00
for (int idx = 0; idx < scoreList.size(); ++idx)
settings.setValue(QString("score-%1").arg(idx), scoreList[idx]->masterScore()->fileInfo()->absoluteFilePath());
}
2014-11-13 13:04:04 +01:00
settings.setValue("lastSaveCopyDirectory", lastSaveCopyDirectory);
settings.setValue("lastSaveCopyFormat", lastSaveCopyFormat);
2014-11-13 13:04:04 +01:00
settings.setValue("lastSaveDirectory", lastSaveDirectory);
MuseScore::saveGeometry(this);
2013-05-13 18:49:17 +02:00
settings.beginGroup("MainWindow");
settings.setValue("showPanel", paletteBox && paletteBox->isVisible());
2014-08-29 19:18:39 +02:00
settings.setValue("showInspector", _inspector && _inspector->isVisible());
2013-05-13 18:49:17 +02:00
settings.setValue("showPianoKeyboard", _pianoTools && _pianoTools->isVisible());
settings.setValue("showSelectionWindow", selectionWindow && selectionWindow->isVisible());
2013-05-13 18:49:17 +02:00
settings.setValue("state", saveState());
settings.setValue("splitScreen", _splitScreen);
settings.setValue("debuggerSplitter", mainWindow->saveState());
2014-11-13 13:04:04 +01:00
settings.setValue("split", _horizontalSplit);
settings.setValue("splitter", splitter->saveState());
2013-05-13 18:49:17 +02:00
settings.endGroup();
2014-11-13 13:04:04 +01:00
2013-05-13 18:49:17 +02:00
Workspace::currentWorkspace->save();
if (keyEditor && keyEditor->dirty())
keyEditor->save();
if (chordStyleEditor)
chordStyleEditor->save();
2014-11-13 13:04:04 +01:00
saveDialogState("loadScoreDialog", loadScoreDialog);
saveDialogState("saveScoreDialog", saveScoreDialog);
saveDialogState("loadStyleDialog", loadStyleDialog);
saveDialogState("saveStyleDialog", saveStyleDialog);
saveDialogState("saveImageDialog", saveImageDialog);
saveDialogState("loadChordStyleDialog", loadChordStyleDialog);
saveDialogState("saveChordStyleDialog", saveChordStyleDialog);
saveDialogState("loadScanDialog", loadScanDialog);
saveDialogState("loadAudioDialog", loadAudioDialog);
saveDialogState("loadDrumsetDialog", loadDrumsetDialog);
saveDialogState("saveDrumsetDialog", saveDrumsetDialog);
saveDialogState("loadPaletteDialog", loadPaletteDialog);
saveDialogState("savePaletteDialog", savePaletteDialog);
if (debugger)
debugger->writeSettings();
#ifdef SCRIPT_INTERFACE
if (_pluginCreator)
_pluginCreator->writeSettings();
#endif
if (synthControl)
synthControl->writeSettings();
2015-09-08 17:14:02 +02:00
settings.setValue("synthControlVisible", synthControl && synthControl->isVisible());
2014-11-13 13:04:04 +01:00
if (mixer)
mixer->writeSettings();
2015-09-08 17:14:02 +02:00
settings.setValue("mixerVisible", mixer && mixer->isVisible());
2014-11-13 13:04:04 +01:00
if (seq) {
seq->exit();
}
if (instrList)
instrList->writeSettings();
if (pianorollEditor)
pianorollEditor->writeSettings();
if (drumrollEditor)
drumrollEditor->writeSettings();
if (startcenter)
startcenter->writeSettings();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// readSettings
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::readSettings()
2012-05-26 14:49:10 +02:00
{
2017-03-25 12:37:24 +01:00
int margin = 100;
int offset = margin / 2;
int w = 1024;
int h = 768;
QScreen* screen = QGuiApplication::primaryScreen();
const QSize screenSize = screen->availableVirtualSize();
if (screenSize.width() - margin > w)
w = screenSize.width() - margin;
else
offset = 0;
if (screenSize.height() - margin > h)
h = screenSize.height() - margin;
resize(QSize(w, h)); //ensure default size if no geometry in settings
move(offset, 0);
2013-05-13 18:49:17 +02:00
if (useFactorySettings) {
QList<int> sizes;
sizes << 500 << 100;
mainWindow->setSizes(sizes);
mscore->showPalette(true);
2014-12-19 21:12:48 +01:00
mscore->showInspector(true);
2013-05-13 18:49:17 +02:00
return;
2012-05-26 14:49:10 +02:00
}
MuseScore::restoreGeometry(this);
2013-05-13 18:49:17 +02:00
settings.beginGroup("MainWindow");
mainWindow->restoreState(settings.value("debuggerSplitter").toByteArray());
mainWindow->setOpaqueResize(false);
scorePageLayoutChanged();
2012-05-26 14:49:10 +02:00
//for some reason when MuseScore starts maximized the screen-reader
//doesn't respond to QAccessibleEvents --> so force normal mode
if (isMaximized() && QAccessible::isActive()) {
showNormal();
}
2013-05-13 18:49:17 +02:00
mscore->showPalette(settings.value("showPanel", "1").toBool());
2014-12-19 21:12:48 +01:00
mscore->showInspector(settings.value("showInspector", "1").toBool());
2013-05-13 18:49:17 +02:00
mscore->showPianoKeyboard(settings.value("showPianoKeyboard", "0").toBool());
mscore->showSelectionWindow(settings.value("showSelectionWindow", "0").toBool());
2013-05-13 18:49:17 +02:00
restoreState(settings.value("state").toByteArray());
//if we were in full screen mode, go to maximized mode
if (isFullScreen()) {
showMaximized();
}
2013-05-13 18:49:17 +02:00
_horizontalSplit = settings.value("split", true).toBool();
bool splitScreen = settings.value("splitScreen", false).toBool();
if (splitScreen) {
splitWindow(_horizontalSplit);
QAction* a = getAction(_horizontalSplit ? "split-h" : "split-v");
a->setChecked(true);
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
splitter->restoreState(settings.value("splitter").toByteArray());
settings.endGroup();
QAction* a = getAction("toggle-fileoperations");
a->setChecked(!fileTools->isHidden());
a = getAction("toggle-transport");
a->setChecked(!transportTools->isHidden());
a = getAction("toggle-concertpitch");
a->setChecked(!cpitchTools->isHidden());
a = getAction("toggle-imagecapture");
a->setChecked(!fotoTools->isHidden());
2017-01-16 20:51:12 +01:00
a = getAction("toggle-noteinput");
a->setChecked(!entryTools->isHidden());
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// play
// play note for preferences.defaultPlayDuration
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::play(Element* e) const
2012-05-26 14:49:10 +02:00
{
if (noSeq || !(seq && seq->isRunning()) || !preferences.getBool(PREF_SCORE_NOTE_PLAYONCLICK))
2013-05-13 18:49:17 +02:00
return;
2012-05-26 14:49:10 +02:00
2017-01-20 11:05:52 +01:00
if (e->isNote()) {
Note* note = toNote(e);
2013-05-13 18:49:17 +02:00
play(e, note->ppitch());
}
2017-01-20 11:05:52 +01:00
else if (e->isChord()) {
2013-05-13 18:49:17 +02:00
seq->stopNotes();
2017-01-20 11:05:52 +01:00
Chord* c = toChord(e);
2013-05-13 18:49:17 +02:00
Part* part = c->staff()->part();
2017-01-20 11:05:52 +01:00
int tick = c->segment() ? c->segment()->tick() : 0;
2013-05-13 18:49:17 +02:00
seq->seek(tick);
Instrument* instr = part->instrument(tick);
2017-01-20 11:05:52 +01:00
for (Note* n : c->notes()) {
const Channel* channel = instr->channel(n->subchannel());
seq->startNote(channel->channel, n->ppitch(), 80, n->tuning());
2013-05-13 18:49:17 +02:00
}
seq->startNoteTimer(MScore::defaultPlayDuration);
2012-05-26 14:49:10 +02:00
}
}
2013-05-13 18:49:17 +02:00
void MuseScore::play(Element* e, int pitch) const
2012-05-26 14:49:10 +02:00
{
if (noSeq || !(seq && seq->isRunning()))
2013-10-29 09:54:53 +01:00
return;
if (preferences.getBool(PREF_SCORE_NOTE_PLAYONCLICK) && e->isNote()) {
2013-05-13 18:49:17 +02:00
Note* note = static_cast<Note*>(e);
int tick = note->chord()->tick();
if (tick < 0)
tick = 0;
Instrument* instr = note->part()->instrument(tick);
const Channel* channel = instr->channel(note->subchannel());
seq->startNote(channel->channel, pitch, 80, MScore::defaultPlayDuration, note->tuning());
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// reportBug
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::reportBug()
2012-05-26 14:49:10 +02:00
{
2015-07-01 14:17:22 +02:00
QString url = QString("https://musescore.org/redirect/post/bug-report?sha=%1&locale=%2").arg(revision()).arg(getLocaleISOCode());
QDesktopServices::openUrl(QUrl(url.trimmed()));
}
//---------------------------------------------------------
// askForHelp
//---------------------------------------------------------
void MuseScore::askForHelp()
{
2015-07-01 14:17:22 +02:00
QString url = QString("https://musescore.org/redirect/post/question?locale=%1").arg(getLocaleISOCode());
QDesktopServices::openUrl(QUrl(url.trimmed()));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// about
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::about()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
AboutBoxDialog ab;
ab.show();
ab.exec();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// AboutBoxDialog
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
AboutBoxDialog::AboutBoxDialog()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
setupUi(this);
museLogo->setPixmap(QPixmap(preferences.isThemeDark() ?
2015-02-27 12:59:27 +01:00
":/data/musescore-logo-transbg-m.png" : ":/data/musescore_logo_full.png"));
if (MuseScore::unstable())
versionLabel->setText(tr("Unstable Prerelease for Version: %1").arg(VERSION));
else
versionLabel->setText(tr("Version: %1").arg(VERSION));
2013-05-13 18:49:17 +02:00
revisionLabel->setText(tr("Revision: %1").arg(revision));
setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
copyrightLabel->setText(QString("<span style=\"font-size:10pt;\">%1</span>")
.arg(tr( "Visit %1www.musescore.org%2 for new versions and more information.\n"
"Support MuseScore with your %3donation%4.\n\n"
"Copyright &copy; 1999-2018 Werner Schweer and Others.\n"
"Published under the GNU General Public License.")
.arg("<a href=\"http://www.musescore.org/\">")
.arg("</a>")
.arg("<a href=\"http://www.musescore.org/donate\">")
.arg("</a>")
.replace("\n","<br/>")));
2013-05-13 18:49:17 +02:00
connect(copyRevisionButton, SIGNAL(clicked()), this, SLOT(copyRevisionToClipboard()));
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// copyRevisionToClipboard
//---------------------------------------------------------
void AboutBoxDialog::copyRevisionToClipboard()
{
QClipboard* cb = QApplication::clipboard();
2018-02-08 10:25:16 +01:00
QString sysinfo = "OS: ";
sysinfo += QSysInfo::prettyProductName();
sysinfo += ", Arch.: ";
sysinfo += QSysInfo::currentCpuArchitecture();
// endianness?
2018-02-08 10:25:16 +01:00
sysinfo += ", MuseScore version (";
sysinfo += QSysInfo::WordSize==32?"32":"64";
sysinfo += "-bit): " VERSION ", revision: ";
sysinfo += "github-musescore-musescore-";
sysinfo += revision;
cb->setText(sysinfo);
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// AboutBoxDialog
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
AboutMusicXMLBoxDialog::AboutMusicXMLBoxDialog()
{
2013-05-13 18:49:17 +02:00
setupUi(this);
setWindowFlags(this->windowFlags() & ~Qt::WindowContextHelpButtonHint);
label->setText(QString("<span style=\"font-size:10pt;\">%1<br/></span>")
.arg(tr( "MusicXML is an open file format for exchanging digital sheet music,\n"
2017-09-27 09:24:47 +02:00
"supported by many applications.\n"
"Copyright © 2004-2017 the Contributors to the MusicXML\n"
"Specification, published by the W3C Music Notation Community\n"
"Group under the W3C Community Contributor License Agreement\n"
"(CLA):\n%1\n"
"A human-readable summary is available:\n%2")
.arg( "\n&nbsp;&nbsp;&nbsp;&nbsp;<a href=\"https://www.w3.org/community/about/agreements/cla/\">https://www.w3.org/community/about/agreements/cla/</a>\n",
"\n&nbsp;&nbsp;&nbsp;&nbsp;<a href=\"https://www.w3.org/community/about/agreements/cla-deed/\">https://www.w3.org/community/about/agreements/cla-deed/</a>\n")
.replace("\n","<br/>")));
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// dirtyChanged
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::dirtyChanged(Score* s)
2012-05-26 14:49:10 +02:00
{
2016-03-11 12:18:46 +01:00
MasterScore* score = s->masterScore();
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
int idx = scoreList.indexOf(score);
if (idx == -1) {
qDebug("score not in list");
return;
}
2016-03-18 09:29:16 +01:00
QString label(score->fileInfo()->completeBaseName());
2013-05-13 18:49:17 +02:00
if (score->dirty())
label += "*";
tab1->setTabText(idx, label);
if (tab2)
tab2->setTabText(idx, label);
setWindowModified(score->dirty());
2013-05-13 18:49:17 +02:00
}
//---------------------------------------------------------
// magChanged
//---------------------------------------------------------
2015-11-23 16:02:33 +01:00
void MuseScore::magChanged(MagIdx idx)
2013-05-13 18:49:17 +02:00
{
if (cv)
2015-11-23 16:02:33 +01:00
cv->setMag(idx, mag->getLMag(cv));
2015-02-02 01:25:35 +01:00
}
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// incMag
//---------------------------------------------------------
void MuseScore::incMag()
{
if (cv) {
2015-11-23 16:02:33 +01:00
qreal _mag = cv->lmag() * SCALE_STEP;
if (_mag > SCALE_MAX)
_mag = SCALE_MAX;
cv->setMag(MagIdx::MAG_FREE, _mag);
2013-05-13 18:49:17 +02:00
setMag(_mag);
}
2013-05-13 18:49:17 +02:00
}
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// decMag
//---------------------------------------------------------
void MuseScore::decMag()
{
if (cv) {
2015-11-23 16:02:33 +01:00
qreal _mag = cv->lmag() / SCALE_STEP;
if (_mag < SCALE_MIN)
_mag = SCALE_MIN;
cv->setMag(MagIdx::MAG_FREE, _mag);
2013-05-13 18:49:17 +02:00
setMag(_mag);
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// getMag
2015-11-23 16:02:33 +01:00
// return physical scale
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
double MuseScore::getMag(ScoreView* canvas) const
{
return mag->getMag(canvas);
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// setMag
2015-11-23 16:02:33 +01:00
// set logical scale
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::setMag(double d)
{
mag->setMag(d);
mag->setMagIdx(MagIdx::MAG_FREE);
2013-05-13 18:49:17 +02:00
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// setPos
// set position label
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::setPos(int t)
{
if (cs == 0 || t < 0)
return;
2012-07-12 15:35:35 +02:00
2013-05-13 18:49:17 +02:00
TimeSigMap* s = cs->sigmap();
int bar, beat, tick;
s->tickValues(t, &bar, &beat, &tick);
_positionLabel->setText(QString("%1:%2:%3")
2013-05-13 18:49:17 +02:00
.arg(bar + 1, 3, 10, QLatin1Char(' '))
.arg(beat + 1, 2, 10, QLatin1Char('0'))
2013-05-13 18:49:17 +02:00
.arg(tick, 3, 10, QLatin1Char('0'))
);
}
2013-02-15 22:39:56 +01:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// undoRedo
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
void MuseScore::undoRedo(bool undo)
2012-05-26 14:49:10 +02:00
{
Q_ASSERT(cv);
Q_ASSERT(cs);
2017-06-30 12:27:54 +02:00
if (_sstate & (STATE_EDIT | STATE_HARMONY_FIGBASS_EDIT | STATE_LYRICS_EDIT))
2017-06-13 17:23:11 +02:00
cv->changeState(ViewState::NORMAL);
cv->startUndoRedo(undo);
updateInputState(cs);
2013-05-13 18:49:17 +02:00
endCmd();
if (_inspector)
2017-03-31 13:03:15 +02:00
_inspector->update();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// showProgressBar
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
QProgressBar* MuseScore::showProgressBar()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (_progressBar == 0)
_progressBar = new QProgressBar(this);
_statusBar->addWidget(_progressBar);
_progressBar->show();
return _progressBar;
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// hideProgressBar
//---------------------------------------------------------
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
void MuseScore::hideProgressBar()
{
if (_progressBar)
_statusBar->removeWidget(_progressBar);
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// handleMessage
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::handleMessage(const QString& message)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (message.isEmpty())
return;
if (startcenter)
showStartcenter(false);
2013-05-13 18:49:17 +02:00
((QtSingleApplication*)(qApp))->activateWindow();
2016-03-11 12:18:46 +01:00
MasterScore* score = readScore(message);
2013-05-13 18:49:17 +02:00
if (score) {
setCurrentScoreView(appendScore(score));
addRecentScore(score);
2013-05-13 18:49:17 +02:00
writeSessionFile(false);
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// editInPianoroll
//---------------------------------------------------------
2013-01-22 12:30:51 +01:00
2013-05-13 18:49:17 +02:00
void MuseScore::editInPianoroll(Staff* staff)
{
2013-05-13 18:49:17 +02:00
if (pianorollEditor == 0)
pianorollEditor = new PianorollEditor;
pianorollEditor->setScore(staff->score());
pianorollEditor->setStaff(staff);
pianorollEditor->show();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// editInDrumroll
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::editInDrumroll(Staff* staff)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (drumrollEditor == 0)
drumrollEditor = new DrumrollEditor;
drumrollEditor->setStaff(staff);
drumrollEditor->show();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// writeSessionFile
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::writeSessionFile(bool cleanExit)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
// qDebug("write session file");
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
QDir dir;
dir.mkpath(dataPath);
QFile f(dataPath + "/session");
if (!f.open(QIODevice::WriteOnly)) {
qDebug("cannot create session file <%s>", qPrintable(f.fileName()));
2012-05-26 14:49:10 +02:00
return;
}
2016-11-19 11:51:21 +01:00
XmlWriter xml(0, &f);
2013-05-13 18:49:17 +02:00
xml.header();
xml.stag("museScore version=\"" MSC_VERSION "\"");
xml.tagE(cleanExit ? "clean" : "dirty");
2016-03-11 12:18:46 +01:00
foreach(MasterScore* score, scoreList) {
2013-05-13 18:49:17 +02:00
xml.stag("Score");
xml.tag("created", score->created());
xml.tag("dirty", score->dirty());
2014-12-01 15:13:22 +01:00
QString path;
if (score->importedFilePath().isEmpty()) {
// score was not imported from another format, e.g. MIDI file
2016-03-10 10:41:31 +01:00
path = score->masterScore()->fileInfo()->absoluteFilePath();
2014-12-01 15:13:22 +01:00
}
2016-03-10 10:41:31 +01:00
else if (score->masterScore()->fileInfo()->exists()) { // score was saved
path = score->masterScore()->fileInfo()->absoluteFilePath();
2014-12-01 15:13:22 +01:00
}
else { // score was imported but not saved - store original file (e.g. MIDI) path
path = score->importedFilePath();
}
if (cleanExit || score->tmpName().isEmpty()) {
2014-12-01 15:13:22 +01:00
xml.tag("path", path);
2013-05-13 18:49:17 +02:00
}
else {
2014-12-01 15:13:22 +01:00
xml.tag("name", path);
2013-05-13 18:49:17 +02:00
xml.tag("path", score->tmpName());
}
xml.etag();
}
int tab = 0;
int idx = 0;
for (int i = 0; i < tab1->count(); ++i) {
ScoreView* v = tab1->view(i);
if (v) {
if (v == cv) {
tab = 0;
idx = i;
}
xml.stag("ScoreView");
xml.tag("tab", tab); // 0 instead of "tab" does not work
xml.tag("idx", i);
if (v->magIdx() == MagIdx::MAG_FREE)
2015-11-18 18:09:42 +01:00
xml.tag("mag", v->lmag());
2013-05-13 18:49:17 +02:00
else
xml.tag("magIdx", int(v->magIdx()));
2015-11-16 14:24:47 +01:00
xml.tag("x", v->xoffset() / DPMM);
xml.tag("y", v->yoffset() / DPMM);
2013-05-13 18:49:17 +02:00
xml.etag();
}
}
if (splitScreen()) {
for (int i = 0; i < tab2->count(); ++i) {
ScoreView* v = tab2->view(i);
if (v) {
if (v == cv) {
tab = 1;
idx = i;
}
xml.stag("ScoreView");
xml.tag("tab", 1);
xml.tag("idx", i);
if (v->magIdx() == MagIdx::MAG_FREE)
2015-11-18 18:09:42 +01:00
xml.tag("mag", v->lmag());
2013-05-13 18:49:17 +02:00
else
xml.tag("magIdx", int(v->magIdx()));
2015-11-16 14:24:47 +01:00
xml.tag("x", v->xoffset() / DPMM);
xml.tag("y", v->yoffset() / DPMM);
2013-05-13 18:49:17 +02:00
xml.etag();
}
}
}
xml.tag("tab", tab);
xml.tag("idx", idx);
xml.etag();
f.close();
if (cleanExit) {
// TODO: remove all temporary session backup files
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// removeSessionFile
// remove temp files and session file
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::removeSessionFile()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
QFile f(dataPath + "/session");
if (!f.exists())
return;
if (!f.remove()) {
qDebug("cannot remove session file <%s>", qPrintable(f.fileName()));
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// autoSaveTimerTimeout
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::autoSaveTimerTimeout()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
bool sessionChanged = false;
2018-07-23 16:50:36 +02:00
Score::isScoreLoaded() = true; //disable debug message "no active command"
2018-07-23 16:50:36 +02:00
2017-06-02 11:21:01 +02:00
for (MasterScore* s : scoreList) {
2013-05-13 18:49:17 +02:00
if (s->autosaveDirty()) {
2018-07-23 16:50:36 +02:00
qDebug("<%s>", qPrintable(s->fileInfo()->baseName()));
2013-05-13 18:49:17 +02:00
QString tmp = s->tmpName();
if (!tmp.isEmpty()) {
QFileInfo fi(tmp);
// TODO: cannot catch exception here:
2014-08-26 19:31:04 +02:00
s->saveCompressedFile(fi, false);
2013-05-13 18:49:17 +02:00
}
else {
QDir dir;
dir.mkpath(dataPath);
QTemporaryFile tf(dataPath + "/scXXXXXX.mscz");
tf.setAutoRemove(false);
if (!tf.open()) {
qDebug("autoSaveTimerTimeout(): create temporary file failed");
return;
}
s->setTmpName(tf.fileName());
QFileInfo info(tf.fileName());
2017-06-02 11:21:01 +02:00
s->saveCompressedFile(&tf, info, false, false); // no thumbnail
2013-05-13 18:49:17 +02:00
tf.close();
sessionChanged = true;
}
s->setAutosaveDirty(false);
}
}
Score::isScoreLoaded() = false;
2018-07-23 16:50:36 +02:00
2013-05-13 18:49:17 +02:00
if (sessionChanged)
writeSessionFile(false);
if (preferences.getBool(PREF_APP_AUTOSAVE_USEAUTOSAVE)) {
int t = preferences.getInt(PREF_APP_AUTOSAVE_AUTOSAVETIME) * 60 * 1000;
2013-05-13 18:49:17 +02:00
autoSaveTimer->start(t);
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// restoreSession
// Restore last session. If "always" is true, then restore
// last session even on a clean exit else only if last
// session ended abnormal.
// Return true if a session was found and restored.
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
bool MuseScore::restoreSession(bool always)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
QFile f(dataPath + "/session");
if (!f.exists())
return false;
if (!f.open(QIODevice::ReadOnly)) {
qDebug("Cannot open session file <%s>", qPrintable(f.fileName()));
2013-05-13 18:49:17 +02:00
return false;
}
XmlReader e(&f);
2013-05-13 18:49:17 +02:00
int tab = 0;
int idx = -1;
bool cleanExit = false;
while (e.readNextStartElement()) {
if (e.name() == "museScore") {
/* QString version = e.attribute(QString("version"));
QStringList sl = version.split('.');
int v = sl[0].toInt() * 100 + sl[1].toInt();
*/
while (e.readNextStartElement()) {
const QStringRef& tag(e.name());
if (tag == "clean") {
if (!always)
return false;
cleanExit = true;
e.readNext();
}
else if (tag == "dirty") {
QMessageBox::StandardButton b = QMessageBox::question(0,
tr("MuseScore"),
tr("The previous session quit unexpectedly.\n\nRestore session?"),
QMessageBox::Yes | QMessageBox::No,
QMessageBox::Yes
);
if (b != QMessageBox::Yes)
return false;
e.readNext();
}
else if (tag == "Score") {
QString name;
bool created = false;
while (e.readNextStartElement()) {
2018-08-17 15:06:15 +02:00
const QStringRef& t(e.name());
if (t == "name")
2013-05-13 18:49:17 +02:00
name = e.readElementText();
2018-08-17 15:06:15 +02:00
else if (t == "created")
2013-05-13 18:49:17 +02:00
created = e.readInt();
2018-08-17 15:06:15 +02:00
else if (t == "dirty")
2015-01-30 17:03:51 +01:00
/*int dirty =*/ e.readInt();
2018-08-17 15:06:15 +02:00
else if (t == "path") {
2016-03-11 12:18:46 +01:00
MasterScore* score = readScore(e.readElementText());
2013-05-13 18:49:17 +02:00
if (score) {
if (!name.isEmpty()) {
QFileInfo* fi = score->masterScore()->fileInfo();
fi->setFile(name);
// TODO: what if that path is no longer valid?
}
2013-05-13 18:49:17 +02:00
if (cleanExit) {
// override if last session did a clean exit
created = false;
}
appendScore(score);
score->setCreated(created);
}
}
else {
e.unknown();
return false;
}
}
}
else if (tag == "ScoreView") {
2015-11-23 16:02:33 +01:00
qreal x = .0;
qreal y = .0;
qreal vmag = 1.0;
MagIdx magIdx = MagIdx::MAG_FREE;
2015-11-23 16:02:33 +01:00
int tab = 0;
int idx = 0;
2013-05-13 18:49:17 +02:00
while (e.readNextStartElement()) {
2018-08-17 15:06:15 +02:00
const QStringRef& t(e.name());
if (t == "tab")
2013-05-13 18:49:17 +02:00
tab = e.readInt();
2018-08-17 15:06:15 +02:00
else if (t == "idx")
2013-05-13 18:49:17 +02:00
idx = e.readInt();
2018-08-17 15:06:15 +02:00
else if (t == "mag")
2013-05-13 18:49:17 +02:00
vmag = e.readDouble();
2018-08-17 15:06:15 +02:00
else if (t == "magIdx")
magIdx = MagIdx(e.readInt());
2018-08-17 15:06:15 +02:00
else if (t == "x")
2015-11-16 14:24:47 +01:00
x = e.readDouble() * DPMM;
2018-08-17 15:06:15 +02:00
else if (t == "y")
2015-11-16 14:24:47 +01:00
y = e.readDouble() * DPMM;
2013-05-13 18:49:17 +02:00
else {
e.unknown();
return false;
}
}
(tab == 0 ? tab1 : tab2)->initScoreView(idx, vmag, magIdx, x, y);
}
else if (tag == "tab")
tab = e.readInt();
else if (tag == "idx")
idx = e.readInt();
else {
e.unknown();
return false;
}
}
}
else {
e.unknown();
return false;
}
}
setCurrentView(tab, idx);
return true;
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// splitWindow
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::splitWindow(bool horizontal)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (!_splitScreen) {
if (tab2 == 0) {
tab2 = new ScoreTab(&scoreList, this);
2013-05-13 18:49:17 +02:00
tab2->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
connect(tab2, SIGNAL(currentScoreViewChanged(ScoreView*)), SLOT(setCurrentScoreView(ScoreView*)));
connect(tab2, SIGNAL(tabCloseRequested(int)), SLOT(removeTab(int)));
connect(tab2, SIGNAL(actionTriggered(QAction*)), SLOT(cmd(QAction*)));
2013-05-13 18:49:17 +02:00
splitter->addWidget(tab2);
}
tab2->setVisible(true);
_splitScreen = true;
_horizontalSplit = horizontal;
splitter->setOrientation(_horizontalSplit ? Qt::Horizontal : Qt::Vertical);
if (!scoreList.isEmpty()) {
tab2->setCurrentIndex(0);
2016-03-11 12:18:46 +01:00
MasterScore* s = scoreList[0];
2016-03-02 13:20:19 +01:00
s->setLayoutAll();
s->update();
2013-05-13 18:49:17 +02:00
setCurrentView(1, 0);
}
}
else {
if (_horizontalSplit == horizontal) {
_splitScreen = false;
tab2->setVisible(false);
setCurrentView(0, tab1->currentIndex());
2013-05-13 18:49:17 +02:00
}
else {
_horizontalSplit = horizontal;
QAction* a;
if (_horizontalSplit)
a = getAction("split-v");
else
a = getAction("split-h");
a->setChecked(false);
splitter->setOrientation(_horizontalSplit ? Qt::Horizontal : Qt::Vertical);
}
}
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// stateName
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
const char* stateName(ScoreState s)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
switch(s) {
case STATE_DISABLED: return "STATE_DISABLED";
case STATE_NORMAL: return "STATE_NORMAL";
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY_STAFF_PITCHED: return "STATE_NOTE_ENTRY_STAFF_PITCHED";
case STATE_NOTE_ENTRY_STAFF_DRUM: return "STATE_NOTE_ENTRY_STAFF_DRUM";
case STATE_NOTE_ENTRY_STAFF_TAB: return "STATE_NOTE_ENTRY_STAFF_TAB";
2013-05-13 18:49:17 +02:00
case STATE_NOTE_ENTRY: return "STATE_NOTE_ENTRY";
2016-08-08 17:45:45 +02:00
case STATE_NOTE_ENTRY_METHOD_STEPTIME: return "STATE_NOTE_ENTRY_METHOD_STEPTIME";
case STATE_NOTE_ENTRY_METHOD_REPITCH: return "STATE_NOTE_ENTRY_METHOD_REPITCH";
case STATE_NOTE_ENTRY_METHOD_RHYTHM: return "STATE_NOTE_ENTRY_METHOD_RHYTHM";
case STATE_NOTE_ENTRY_METHOD_REALTIME_AUTO: return "STATE_NOTE_ENTRY_METHOD_REALTIME_AUTO";
case STATE_NOTE_ENTRY_METHOD_REALTIME_MANUAL: return "STATE_NOTE_ENTRY_METHOD_REALTIME_MANUAL";
2013-05-13 18:49:17 +02:00
case STATE_EDIT: return "STATE_EDIT";
case STATE_TEXT_EDIT: return "STATE_TEXT_EDIT";
case STATE_LYRICS_EDIT: return "STATE_LYRICS_EDIT";
case STATE_HARMONY_FIGBASS_EDIT: return "STATE_HARMONY_FIGBASS_EDIT";
case STATE_PLAY: return "STATE_PLAY";
case STATE_FOTO: return "STATE_FOTO";
default: return "??";
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
// scorePageLayoutChanged
//---------------------------------------------------------
void MuseScore::scorePageLayoutChanged()
{
if (mainWindow) {
mainWindow->setOrientation(MScore::verticalOrientation() ? Qt::Horizontal : Qt::Vertical);
if (navigatorScrollArea())
navigatorScrollArea()->orientationChanged();
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// editRaster
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::editRaster()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (editRasterDialog == 0) {
editRasterDialog = new EditRaster(this);
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
if (editRasterDialog->exec()) {
qDebug("=====accept config raster");
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// showPianoKeyboard
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::showPianoKeyboard(bool on)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (_pianoTools == 0) {
2014-12-22 09:56:39 +01:00
QAction* a = getAction("toggle-piano");
2013-05-13 18:49:17 +02:00
_pianoTools = new PianoTools(this);
addDockWidget(Qt::BottomDockWidgetArea, _pianoTools);
connect(_pianoTools, SIGNAL(keyPressed(int, bool, int)), SLOT(midiNoteReceived(int, bool, int)));
connect(_pianoTools, SIGNAL(keyReleased(int, bool, int)), SLOT(midiNoteReceived(int, bool, int)));
2013-05-13 18:49:17 +02:00
connect(_pianoTools, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
if (on) {
_pianoTools->show();
if (currentScore())
_pianoTools->changeSelection(currentScore()->selection());
else
_pianoTools->clearSelection();
2013-05-13 18:49:17 +02:00
}
else {
if (_pianoTools)
_pianoTools->hide();
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// showPluginCreator
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::showPluginCreator(QAction* a)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
#ifdef SCRIPT_INTERFACE
bool on = a->isChecked();
if (on) {
2014-08-29 19:18:39 +02:00
if (_pluginCreator == 0) {
_pluginCreator = new PluginCreator(0);
connect(_pluginCreator, SIGNAL(closed(bool)), a, SLOT(setChecked(bool)));
2013-05-13 18:49:17 +02:00
}
2014-08-29 19:18:39 +02:00
_pluginCreator->show();
2013-05-13 18:49:17 +02:00
}
else {
2014-08-29 19:18:39 +02:00
if (_pluginCreator)
_pluginCreator->hide();
2013-05-13 18:49:17 +02:00
}
#endif
2012-05-26 14:49:10 +02:00
}
2013-10-05 12:03:30 +02:00
//---------------------------------------------------------
// showPluginManager
//---------------------------------------------------------
2014-09-01 22:35:04 +02:00
void MuseScore::showPluginManager()
2013-10-05 12:03:30 +02:00
{
#ifdef SCRIPT_INTERFACE
pluginManager->init();
pluginManager->show();
2013-10-05 12:03:30 +02:00
#endif
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// showMediaDialog
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::showMediaDialog()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (_mediaDialog == 0)
_mediaDialog = new MediaDialog(this);
_mediaDialog->setScore(cs);
_mediaDialog->exec();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// getPaletteBox
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
PaletteBox* MuseScore::getPaletteBox()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (paletteBox == 0) {
paletteBox = new PaletteBox(this);
QAction* a = getAction("toggle-palette");
connect(paletteBox, SIGNAL(visibilityChanged(bool)), a, SLOT(setChecked(bool)));
addDockWidget(Qt::LeftDockWidgetArea, paletteBox);
}
return paletteBox;
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// midiNoteReceived
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
void MuseScore::midiNoteReceived(int pitch, bool ctrl, int vel)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (cv)
cv->midiNoteReceived(pitch, ctrl, vel);
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// switchLayer
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::switchLayer(const QString& s)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (cs->switchLayer(s)) {
2016-03-02 13:20:19 +01:00
cs->setLayoutAll();
2013-05-13 18:49:17 +02:00
cs->update();
2012-05-26 14:49:10 +02:00
}
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// switchPlayMode
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::switchPlayMode(int mode)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (cs)
cs->setPlayMode(PlayMode(mode));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// networkFinished
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
void MuseScore::networkFinished()
2012-05-26 14:49:10 +02:00
{
QNetworkReply* reply = qobject_cast<QNetworkReply*>(QObject::sender());
2013-05-13 18:49:17 +02:00
if (reply->error() != QNetworkReply::NoError) {
qDebug("Error while checking update [%s]", qPrintable(reply->errorString()));
2012-05-26 14:49:10 +02:00
return;
}
2013-05-13 18:49:17 +02:00
QByteArray ha = reply->rawHeader("Content-Disposition");
QString s(ha);
QString name;
QRegExp re(".*filename=\"(.*)\"");
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (!s.isEmpty() && re.indexIn(s) != -1) {
name = re.cap(1);
}
else {
QUrl url = reply->url();
QString path = url.path();
qDebug("Path <%s>", qPrintable(path));
QFileInfo fi(path);
name = fi.fileName();
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
qDebug("header <%s>", qPrintable(s));
qDebug("name <%s>", qPrintable(name));
QByteArray data = reply->readAll();
QString tmpName = QDir::tempPath () + "/"+ name;
QFile f(tmpName);
f.open(QIODevice::WriteOnly);
f.write(data);
2012-05-26 14:49:10 +02:00
f.close();
2013-05-13 18:49:17 +02:00
reply->deleteLater();
2016-03-11 12:18:46 +01:00
MasterScore* score = readScore(tmpName);
2013-05-13 18:49:17 +02:00
if (!score) {
qDebug("readScore failed");
return;
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
score->setCreated(true);
setCurrentScoreView(appendScore(score));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// loadFile
// load file from url
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::loadFile(const QString& s)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
loadFile(QUrl(s));
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
void MuseScore::loadFile(const QUrl& url)
2012-05-26 14:49:10 +02:00
{
QEventLoop loop;
QNetworkReply* nr = mscore->networkManager()->get(QNetworkRequest(url));
connect(nr, SIGNAL(finished()), this,
SLOT(networkFinished()));
connect(nr, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();
}
//---------------------------------------------------------
// networkManager
//---------------------------------------------------------
QNetworkAccessManager* MuseScore::networkManager()
{
if (!_networkManager)
_networkManager = new QNetworkAccessManager(this);
return _networkManager;
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// selectSimilar
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::selectSimilar(Element* e, bool sameStaff)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
Score* score = e->score();
score->selectSimilar(e, sameStaff);
2012-05-26 14:49:10 +02:00
if (score->selectionChanged()) {
score->setSelectionChanged(false);
SelState ss = score->selection().state();
selectionChanged(ss);
}
2012-05-26 14:49:10 +02:00
}
2014-07-14 13:24:47 +02:00
void MuseScore::selectSimilarInRange(Element* e)
{
Score* score = e->score();
score->selectSimilarInRange(e);
2014-07-14 13:24:47 +02:00
if (score->selectionChanged()) {
score->setSelectionChanged(false);
SelState ss = score->selection().state();
selectionChanged(ss);
}
}
2014-11-22 12:03:22 +01:00
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// selectElementDialog
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::selectElementDialog(Element* e)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
Score* score = e->score();
if (e->isNote()) {
Note* n = toNote(e);
SelectNoteDialog sd(n, 0);
// hide same string option if it doesn't make sense for the instrument
if (n->staff()->part()->instrument()->stringData()->strings() == 0)
sd.setSameStringVisible(false);
if (sd.exec()) {
NotePattern pattern;
sd.setPattern(&pattern);
if (sd.isInSelection())
score->scanElementsInRange(&pattern, Score::collectNoteMatch);
else
score->scanElements(&pattern, Score::collectNoteMatch);
if (sd.doReplace()) {
score->select(0, SelectType::SINGLE, 0);
for (Note* ee : pattern.el)
score->select(ee, SelectType::ADD, 0);
}
else if (sd.doSubtract()) {
QList<Element*> sl(score->selection().elements());
for (Note* ee : pattern.el)
sl.removeOne(ee);
score->select(0, SelectType::SINGLE, 0);
for (Element* ee : sl)
score->select(ee, SelectType::ADD, 0);
2013-05-13 18:49:17 +02:00
}
else if (sd.doAdd()) {
QList<Element*> sl(score->selection().elements());
for (Note* ee : pattern.el) {
if(!sl.contains(ee))
score->select(ee, SelectType::ADD, 0);
}
}
2012-05-26 14:49:10 +02:00
}
}
else {
SelectDialog sd(e, 0);
if (sd.exec()) {
ElementPattern pattern;
sd.setPattern(&pattern);
if (sd.isInSelection())
score->scanElementsInRange(&pattern, Score::collectMatch);
else
score->scanElements(&pattern, Score::collectMatch);
if (sd.doReplace()) {
score->select(0, SelectType::SINGLE, 0);
for (Element* ee : pattern.el)
score->select(ee, SelectType::ADD, 0);
}
else if (sd.doSubtract()) {
QList<Element*> sl(score->selection().elements());
for (Element* ee : pattern.el)
sl.removeOne(ee);
score->select(0, SelectType::SINGLE, 0);
for (Element* ee : sl)
score->select(ee, SelectType::ADD, 0);
}
else if (sd.doAdd()) {
QList<Element*> sl(score->selection().elements());
for (Element* ee : pattern.el) {
if(!sl.contains(ee))
score->select(ee, SelectType::ADD, 0);
}
}
}
2012-05-26 14:49:10 +02:00
}
if (score->selectionChanged()) {
score->setSelectionChanged(false);
SelState ss = score->selection().state();
selectionChanged(ss);
}
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// RecordButton
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
RecordButton::RecordButton(QWidget* parent)
: SimpleButton(":/data/recordOn.svg", ":/data/recordOff.svg", parent)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
setCheckable(true);
defaultAction()->setCheckable(true);
2014-10-15 20:43:06 +02:00
setToolTip(qApp->translate("RecordButton", "Record"));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// GreendotButton
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
GreendotButton::GreendotButton(QWidget* parent)
: SimpleButton(":/data/greendot.svg", ":/data/darkgreendot.svg", parent)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
setCheckable(true);
2014-10-15 20:43:06 +02:00
setToolTip(qApp->translate("GreendotButton", "Record"));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// drawHandle
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
QRectF drawHandle(QPainter& p, const QPointF& pos, bool active)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
p.save();
2013-07-04 21:07:38 +02:00
p.setPen(QPen(QColor(MScore::selectColor[0]), 2.0/p.matrix().m11()));
2013-05-13 18:49:17 +02:00
if (active)
2013-07-04 21:07:38 +02:00
p.setBrush(MScore::selectColor[0]);
2013-05-13 18:49:17 +02:00
else
p.setBrush(Qt::NoBrush);
qreal w = 8.0 / p.matrix().m11();
qreal h = 8.0 / p.matrix().m22();
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
QRectF r(-w/2, -h/2, w, h);
r.translate(pos);
p.drawRect(r);
p.restore();
return r;
2012-05-26 14:49:10 +02:00
}
2012-07-02 18:05:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// transpose
2012-07-02 18:05:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::transpose()
2012-07-02 18:05:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (cs->last() == 0) // empty score?
return;
bool noSelection = cs->selection().isNone();
if (noSelection)
cs->cmdSelectAll();
2014-05-24 12:53:50 +02:00
bool rangeSelection = cs->selection().isRange();
2013-05-13 18:49:17 +02:00
TransposeDialog td;
2012-07-02 18:05:10 +02:00
2014-05-26 13:21:04 +02:00
// TRANSPOSE_BY_KEY and "transpose keys" is only possible if selection state is SelState::RANGE
2013-05-13 18:49:17 +02:00
td.enableTransposeKeys(rangeSelection);
td.enableTransposeByKey(rangeSelection);
td.enableTransposeChordNames(rangeSelection);
2012-07-02 18:05:10 +02:00
2013-05-13 18:49:17 +02:00
int startStaffIdx = 0;
int endStaffIdx = 0;
2013-05-13 18:49:17 +02:00
int startTick = 0;
if (rangeSelection) {
startStaffIdx = cs->selection().staffStart();
endStaffIdx = cs->selection().staffEnd();
2013-05-13 18:49:17 +02:00
startTick = cs->selection().tickStart();
2012-05-26 14:49:10 +02:00
}
// find the key of the first pitched staff
Key key = Key::C;
for (int i = startStaffIdx; i < endStaffIdx; ++i) {
Staff* staff = cs->staff(i);
2016-12-13 13:16:17 +01:00
if (staff->isPitchedStaff(startTick)) {
key = staff->key(startTick);
2018-03-27 15:36:00 +02:00
if (!cs->styleB(Sid::concertPitch)) {
int diff = staff->part()->instrument(startTick)->transpose().chromatic;
if (diff)
key = transposeKey(key, diff);
}
break;
}
}
td.setKey(key);
2013-05-13 18:49:17 +02:00
if (!td.exec())
return;
2013-05-13 18:49:17 +02:00
cs->transpose(td.mode(), td.direction(), td.transposeKey(), td.transposeInterval(),
td.getTransposeKeys(), td.getTransposeChordNames(), td.useDoubleSharpsFlats());
if (noSelection)
cs->deselectAll();
2012-07-02 18:05:10 +02:00
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// cmd
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::cmd(QAction* a)
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
if (inChordEditor) // HACK
return;
2016-11-28 18:25:27 +01:00
MScore::setError(MS_NO_ERROR);
2013-05-13 18:49:17 +02:00
QString cmdn(a->data().toString());
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (MScore::debugMode)
qDebug("MuseScore::cmd <%s>", qPrintable(cmdn));
2012-05-26 14:49:10 +02:00
2014-11-14 11:44:48 +01:00
const Shortcut* sc = Shortcut::getShortcut(cmdn.toLatin1().data());
2013-05-13 18:49:17 +02:00
if (sc == 0) {
qDebug("MuseScore::cmd(): unknown action <%s>", qPrintable(cmdn));
2012-05-26 14:49:10 +02:00
return;
}
2013-05-13 18:49:17 +02:00
if (cs && (sc->state() & _sstate) == 0) {
qDebug("invalid state %04x %04x", sc->state(), _sstate);
2013-05-13 18:49:17 +02:00
QMessageBox::warning(0,
tr("Invalid Command"),
tr("Command %1 not valid in current state").arg(cmdn));
2012-05-26 14:49:10 +02:00
return;
}
if (cmdn == "toggle-palette") {
showPalette(a->isChecked());
PaletteBox* pb = mscore->getPaletteBox();
QLineEdit* sb = pb->searchBox();
if (a->isChecked()) {
lastFocusWidget = QApplication::focusWidget();
sb->setFocus();
if (pb->noSelection())
pb->setKeyboardNavigation(false);
else
pb->setKeyboardNavigation(true);
}
else {
if (lastFocusWidget)
lastFocusWidget->setFocus();
}
return;
}
2017-07-30 10:20:17 +02:00
if (cmdn == "palette-search") {
PaletteBox* pb = getPaletteBox();
QLineEdit* sb = pb->searchBox();
sb->setFocus();
if (pb->noSelection())
pb->setKeyboardNavigation(false);
else
pb->setKeyboardNavigation(true);
return;
}
if (cmdn == "apply-current-palette-element") {
PaletteBox* pb = getPaletteBox();
for (Palette* p : pb->palettes()) {
if (p->getCurrentIdx() != -1) {
p->applyPaletteElement();
break;
}
}
return;
}
2013-05-13 18:49:17 +02:00
if (cmdn == "repeat-cmd") {
a = lastCmd;
sc = lastShortcut;
if (a == 0)
2012-05-26 14:49:10 +02:00
return;
2013-05-13 18:49:17 +02:00
cmdn = a->data().toString();
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
else {
lastCmd = a;
lastShortcut = sc;
2012-05-26 14:49:10 +02:00
}
2014-11-14 11:44:48 +01:00
if (sc->needsScore() && ! cs) {
2013-05-13 18:49:17 +02:00
qDebug("no score");
return;
}
2014-11-14 11:44:48 +01:00
if (sc->isCmd()) {
2013-05-13 18:49:17 +02:00
if (!cv->editMode())
cs->startCmd();
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
cmd(a, cmdn);
2014-11-14 11:44:48 +01:00
if (lastShortcut->isCmd())
2013-05-13 18:49:17 +02:00
cs->endCmd();
endCmd();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// endCmd
// Called after every command action (including every
// mouse action).
// Updates the UI after a possible score change.
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::endCmd()
2012-05-26 14:49:10 +02:00
{
if (timeline())
timeline()->updateGrid();
2016-11-28 18:25:27 +01:00
if (MScore::_error != MS_NO_ERROR)
showError();
2013-05-13 18:49:17 +02:00
if (cs) {
setPos(cs->inputState().tick());
updateInputState(cs);
updateUndoRedo();
dirtyChanged(cs);
Element* e = cs->selection().element();
// For multiple notes selected check if they all have same pitch and tuning
bool samePitch = true;
int pitch = -1;
float tuning = 0;
for (Element* e : cs->selection().elements()) {
if (!e->isNote()) {
samePitch = false;
break;
}
Note* note = toNote(e);
if (pitch == -1) {
pitch = note->ppitch();
tuning = note->tuning();
}
else if (note->ppitch() != pitch || fabs(tuning - note->tuning()) > 0.01) {
samePitch = false;
break;
}
}
if (samePitch && pitch >= 0)
e = cs->selection().elements()[0];
2016-08-08 17:46:37 +02:00
NoteEntryMethod entryMethod = cs->noteEntryMethod();
if (e && (cs->playNote() || cs->playChord())
&& entryMethod != NoteEntryMethod::REALTIME_AUTO
&& entryMethod != NoteEntryMethod::REALTIME_MANUAL) {
if (cs->playChord() && preferences.getBool(PREF_SCORE_CHORD_PLAYONADDNOTE) && e->type() == ElementType::NOTE)
2015-02-08 21:06:26 +01:00
play(static_cast<Note*>(e)->chord());
else
play(e);
2013-05-13 18:49:17 +02:00
cs->setPlayNote(false);
2015-02-08 21:06:26 +01:00
cs->setPlayChord(false);
2012-05-26 14:49:10 +02:00
}
MasterScore* ms = cs->masterScore();
if (ms->excerptsChanged()) {
if (tab2) {
// ScoreView* v = tab2->view();
// if (v && v->score() == ms) {
tab2->updateExcerpts();
// }
}
if (tab1) {
ScoreView* v = tab1->view();
if (v && v->score()->masterScore() == ms) {
tab1->updateExcerpts();
}
else if (v == 0) {
tab1->setExcerpt(0);
tab1->updateExcerpts();
}
}
ms->setExcerptsChanged(false);
2012-05-26 14:49:10 +02:00
}
if (ms->instrumentsChanged()) {
if (!noSeq && (seq && seq->isRunning()))
seq->initInstruments();
instrumentChanged(); // update mixer
ms->setInstrumentsChanged(false);
2012-05-26 14:49:10 +02:00
}
2013-05-13 18:49:17 +02:00
if (cs->selectionChanged()) {
cs->setSelectionChanged(false);
SelState ss = cs->selection().state();
selectionChanged(ss);
}
2018-03-27 15:36:00 +02:00
getAction("concert-pitch")->setChecked(cs->styleB(Sid::concertPitch));
2012-05-26 14:49:10 +02:00
2013-10-24 12:09:00 +02:00
if (e == 0 && cs->noteEntryMode())
2013-05-13 18:49:17 +02:00
e = cs->inputState().cr();
updateViewModeCombo();
ScoreAccessibility::instance()->updateAccessibilityInfo();
2012-05-26 14:49:10 +02:00
}
else {
2014-05-26 13:21:04 +02:00
selectionChanged(SelState::NONE);
2012-05-26 14:49:10 +02:00
}
2017-03-31 13:03:15 +02:00
updateInspector();
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// updateUndoRedo
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::updateUndoRedo()
2012-05-26 14:49:10 +02:00
{
2013-05-13 18:49:17 +02:00
QAction* a = getAction("undo");
2016-03-10 10:41:31 +01:00
a->setEnabled(cs ? cs->undoStack()->canUndo() : false);
2013-05-13 18:49:17 +02:00
a = getAction("redo");
2016-03-10 10:41:31 +01:00
a->setEnabled(cs ? cs->undoStack()->canRedo() : false);
2012-05-26 14:49:10 +02:00
}
void MuseScore::setPlayRepeats(bool repeat)
{
getAction("repeat")->setChecked(repeat);
preferences.setPreference(PREF_APP_PLAYBACK_PLAYREPEATS, repeat);
MScore::playRepeats = repeat;
if (cs) {
cs->updateRepeatList(repeat);
emit cs->playlistChanged();
}
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// cmd
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::cmd(QAction* a, const QString& cmd)
2012-05-26 14:49:10 +02:00
{
if (cmd == "instruments") {
editInstrList();
2013-04-03 12:13:23 +02:00
if (mixer)
2016-03-11 12:18:46 +01:00
mixer->updateAll(cs->masterScore());
2012-05-26 14:49:10 +02:00
}
else if (cmd == "rewind") {
if (cs) {
int tick = loop() ? cs->loopInTick() : 0;
seq->seek(tick);
if (cv) {
Measure* m = cs->tick2measureMM(tick);
if (m)
cv->gotoMeasure(m);
}
if (playPanel)
playPanel->heartBeat(0, 0, 0);
}
2012-05-26 14:49:10 +02:00
}
else if (cmd == "play-next-measure")
seq->nextMeasure();
else if (cmd == "play-next-chord")
seq->nextChord();
else if (cmd == "play-prev-measure")
seq->prevMeasure();
else if (cmd == "play-prev-chord")
seq->prevChord();
else if (cmd == "seek-begin")
seq->rewindStart();
else if (cmd == "seek-end")
seq->seekEnd();
else if (cmd == "keys")
showKeyEditor();
else if (cmd == "file-open")
loadFiles();
else if (cmd == "file-save")
saveFile();
else if (cmd == "file-save-online" && MuseScore::unstable() && enableExperimental)
showUploadScoreDialog();
2012-05-26 14:49:10 +02:00
else if (cmd == "file-export")
exportFile();
else if (cmd == "file-part-export")
exportParts();
2015-03-10 19:43:22 +01:00
else if (cmd == "file-import-pdf")
openExternalLink("https://musescore.com/import");
2012-05-26 14:49:10 +02:00
else if (cmd == "file-close")
closeScore(cs);
else if (cmd == "file-save-as")
2012-05-26 14:49:10 +02:00
saveAs(cs, false);
else if (cmd == "file-save-selection")
2012-05-26 14:49:10 +02:00
saveSelection(cs);
else if (cmd == "file-save-a-copy")
2012-05-26 14:49:10 +02:00
saveAs(cs, true);
else if (cmd == "file-new")
newFile();
else if (cmd == "quit")
2012-05-26 14:49:10 +02:00
close();
else if (cmd == "masterpalette")
showMasterPalette();
else if (cmd == "key-signatures")
showMasterPalette(qApp->translate("Palette", "Key Signatures"));
else if (cmd == "time-signatures")
showMasterPalette(qApp->translate("Palette", "Time Signatures"));
else if (cmd == "symbols")
showMasterPalette(qApp->translate("MasterPalette", "Symbols"));
2012-05-26 14:49:10 +02:00
else if (cmd == "toggle-statusbar") {
preferences.setPreference(PREF_UI_APP_SHOWSTATUSBAR, a->isChecked());
_statusBar->setVisible(a->isChecked());
2012-05-26 14:49:10 +02:00
}
else if (cmd == "append-measures")
cmdAppendMeasures();
else if (cmd == "insert-measures")
cmdInsertMeasures();
else if (cmd == "debugger")
startDebugger();
else if (cmd == "album")
showAlbumManager();
else if (cmd == "layer" && enableExperimental)
2012-05-26 14:49:10 +02:00
showLayerManager();
else if (cmd == "backspace") {
if (_sstate != STATE_NORMAL )
undoRedo(true);
#ifdef Q_OS_MAC
else if (cs) {
cs->startCmd();
cs->cmdDeleteSelection();
cs->endCmd();
}
#endif
}
2012-05-26 14:49:10 +02:00
else if (cmd == "zoomin")
incMag();
else if (cmd == "zoomout")
decMag();
else if (cmd == "zoom100") {
if (cv)
cv->setMag(MagIdx::MAG_100, 1.0);
setMag(1.0);
}
2012-05-26 14:49:10 +02:00
else if (cmd == "midi-on")
midiinToggled(a->isChecked());
else if (cmd == "undo")
undoRedo(true);
else if (cmd == "redo")
undoRedo(false);
else if (cmd == "startcenter")
showStartcenter(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "inspector")
showInspector(a->isChecked());
#ifdef OMR
else if (cmd == "omr")
showOmrPanel(a->isChecked());
#endif
else if (cmd == "toggle-playpanel")
showPlayPanel(a->isChecked());
else if (cmd == "toggle-navigator")
showNavigator(a->isChecked());
else if (cmd == "toggle-timeline")
showTimeline(a->isChecked());
2013-04-28 21:29:12 +02:00
else if (cmd == "toggle-midiimportpanel")
importmidiPanel->setVisible(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "toggle-mixer")
showMixer(a->isChecked());
else if (cmd == "synth-control")
2013-04-24 12:41:38 +02:00
showSynthControl(a->isChecked());
2014-06-02 12:12:53 +02:00
else if (cmd == "toggle-selection-window")
showSelectionWindow(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "show-keys")
;
else if (cmd == "toggle-fileoperations")
fileTools->setVisible(!fileTools->isVisible());
else if (cmd == "toggle-transport")
transportTools->setVisible(!transportTools->isVisible());
else if (cmd == "toggle-concertpitch")
cpitchTools->setVisible(!cpitchTools->isVisible());
else if (cmd == "toggle-imagecapture")
fotoTools->setVisible(!fotoTools->isVisible());
else if (cmd == "toggle-noteinput")
entryTools->setVisible(!entryTools->isVisible());
else if (cmd == "help")
2015-02-19 19:55:17 +01:00
showContextHelp();
2012-05-26 14:49:10 +02:00
else if (cmd == "follow")
preferences.setPreference(PREF_APP_PLAYBACK_FOLLOWSONG, a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "split-h")
splitWindow(true);
else if (cmd == "split-v")
splitWindow(false);
else if (cmd == "edit-harmony" && enableExperimental)
2012-05-26 14:49:10 +02:00
editChordStyle();
else if (cmd == "parts")
startExcerptsDialog();
else if (cmd == "fullscreen") {
_fullscreen = a->isChecked();
if (_fullscreen)
showFullScreen();
else
showNormal();
#ifdef Q_OS_MAC
2012-05-26 14:49:10 +02:00
// Qt Bug: Toolbar goes into unified mode
// after switching back from fullscreen
setUnifiedTitleAndToolBarOnMac(false);
#endif
}
else if (cmd == "config-raster")
editRaster();
else if (cmd == "hraster" || cmd == "vraster") // value in [hv]RasterAction already set
;
2014-12-22 09:56:39 +01:00
else if (cmd == "toggle-piano")
2012-05-26 14:49:10 +02:00
showPianoKeyboard(a->isChecked());
2012-07-02 18:05:10 +02:00
else if (cmd == "plugin-creator")
showPluginCreator(a);
2013-10-05 12:03:30 +02:00
else if (cmd == "plugin-manager")
2014-09-01 22:35:04 +02:00
showPluginManager();
else if(cmd == "resource-manager"){
ResourceManager r(0);
r.exec();
}
else if (cmd == "media" && enableExperimental)
2012-05-26 14:49:10 +02:00
showMediaDialog();
else if (cmd == "page-settings")
showPageSettings();
else if (cmd == "next-score")
changeScore(1);
2012-05-26 14:49:10 +02:00
else if (cmd == "previous-score")
changeScore(1);
2012-05-26 14:49:10 +02:00
else if (cmd == "transpose")
transpose();
else if (cmd == "save-style") {
QString name = getStyleFilename(false);
if (!name.isEmpty()) {
if (!cs->saveStyle(name)) {
QMessageBox::critical(this,
tr("Save Style"), MScore::lastError);
2012-05-26 14:49:10 +02:00
}
}
}
else if (cmd == "load-style") {
QString name = mscore->getStyleFilename(true);
if (!name.isEmpty()) {
cs->startCmd();
if (!cs->loadStyle(name)) {
QMessageBox::critical(this,
tr("Load Style"), MScore::lastError);
2012-05-26 14:49:10 +02:00
}
cs->endCmd();
}
}
else if (cmd == "edit-style") {
EditStyle es(cs, this);
es.exec();
}
else if (cmd == "edit-info") {
MetaEditDialog med(cs, 0);
med.exec();
}
else if (cmd == "print")
printFile();
else if (cmd == "repeat")
setPlayRepeats(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "pan")
MScore::panPlayback = !MScore::panPlayback;
else if (cmd == "show-invisible")
cs->setShowInvisible(a->isChecked());
else if (cmd == "show-unprintable")
cs->setShowUnprintable(a->isChecked());
else if (cmd == "show-frames")
2016-08-06 11:36:51 +02:00
cs->setShowFrames(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "show-pageborders")
2016-08-06 11:36:51 +02:00
cs->setShowPageborders(a->isChecked());
else if (cmd == "mark-irregular")
cs->setMarkIrregularMeasures(a->isChecked());
2012-05-26 14:49:10 +02:00
else if (cmd == "tempo")
addTempo();
else if (cmd == "loop") {
if (loop()) {
2014-05-24 12:53:50 +02:00
if (cs->selection().isRange())
seq->setLoopSelection();
2013-08-19 08:53:30 +02:00
}
}
else if (cmd == "loop-in") {
seq->setLoopIn();
2013-10-18 12:21:01 +02:00
loopAction->setChecked(true);
2013-08-19 08:53:30 +02:00
}
else if (cmd == "loop-out") {
seq->setLoopOut();
2013-10-18 12:21:01 +02:00
loopAction->setChecked(true);
2013-08-19 08:53:30 +02:00
}
2012-05-26 14:49:10 +02:00
else if (cmd == "metronome") // no action
;
else if (cmd == "countin") // no action
;
else if (cmd == "lock") {
if (_sstate == STATE_LOCK)
changeState(STATE_NORMAL);
else
changeState(STATE_LOCK);
}
else if (cmd == "find")
showSearchDialog();
else if (cmd == "text-b") {
if (_textTools)
_textTools->toggleBold();
}
else if (cmd == "text-i") {
if (_textTools)
_textTools->toggleItalic();
}
else if (cmd == "text-u") {
if (_textTools)
_textTools->toggleUnderline();
}
2016-05-31 17:44:39 +02:00
else if (cmd == "edit-toolbars")
showToolbarEditor();
2014-11-17 13:06:34 +01:00
else if (cmd == "viewmode") {
if (cs) {
if (cs->layoutMode() == LayoutMode::PAGE)
2016-05-18 15:43:47 +02:00
switchLayoutMode(LayoutMode::LINE);
2014-11-17 13:06:34 +01:00
else
2016-05-18 15:43:47 +02:00
switchLayoutMode(LayoutMode::PAGE);
2014-11-17 13:06:34 +01:00
}
}
#ifndef NDEBUG
else if (cmd == "no-horizontal-stretch") {
MScore::noHorizontalStretch = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "no-vertical-stretch") {
MScore::noVerticalStretch = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "show-segment-shapes") {
MScore::showSegmentShapes = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "show-skylines") {
MScore::showSkylines = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "show-bounding-rect") {
MScore::showBoundingRect = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "show-corrupted-measures") {
MScore::showCorruptedMeasures = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
else if (cmd == "autoplace-slurs") {
MScore::autoplaceSlurs = a->isChecked();
if (cs) {
cs->setLayoutAll();
cs->update();
}
}
#endif
2012-05-26 14:49:10 +02:00
else {
if (cv) {
//isAncestorOf is called to see if a widget from inspector has focus
//if so, the focus doesn't get shifted to the score, unless escape is
//pressed, or the user clicks in the score
if (!inspector()->isAncestorOf(qApp->focusWidget()) || cmd == "escape")
cv->setFocus();
2012-05-26 14:49:10 +02:00
cv->cmd(a);
}
else
qDebug("2:unknown cmd <%s>", qPrintable(cmd));
}
if (debugger)
debugger->reloadClicked();
}
//---------------------------------------------------------
// openExternalLink
//---------------------------------------------------------
void MuseScore::openExternalLink(const QString& url)
{
QDesktopServices::openUrl(url);
}
//---------------------------------------------------------
// navigator
//---------------------------------------------------------
Navigator* MuseScore::navigator() const
{
return _navigator ? static_cast<Navigator*>(_navigator->widget()) : 0;
}
//---------------------------------------------------------
// timeline
//---------------------------------------------------------
Timeline* MuseScore::timeline() const
{
if (_timeline) {
QSplitter* s = static_cast<QSplitter *>(_timeline->widget());
if (s && s->count() > 0)
return _timeline ? static_cast<Timeline*>(s->widget(1)) : 0;
return 0;
}
return 0;
}
//---------------------------------------------------------
// getSearchDialog
//---------------------------------------------------------
QWidget* MuseScore::searchDialog() const
{
return _searchDialog;
}
2012-05-26 14:49:10 +02:00
//---------------------------------------------------------
// updateLayer
//---------------------------------------------------------
void MuseScore::updateLayer()
{
layerSwitch->clear();
bool enable;
if (cs) {
enable = cs->layer().size() > 1;
if (enable) {
foreach(const Layer& l, cs->layer())
layerSwitch->addItem(l.name);
layerSwitch->setCurrentIndex(cs->currentLayer());
}
}
else
enable = false;
layerSwitch->setVisible(enable);
}
//---------------------------------------------------------
// updatePlayMode
//---------------------------------------------------------
void MuseScore::updatePlayMode()
{
bool enable = false;
if (cs) {
enable = cs->audio() != 0;
playMode->setCurrentIndex(int(cs->playMode()));
}
playMode->setVisible(enable);
}
//---------------------------------------------------------
// closeScore
//---------------------------------------------------------
void MuseScore::closeScore(Score* score)
{
// Let's compute maximum count of ports in remaining scores
if (seq)
seq->recomputeMaxMidiOutPort();
2016-03-10 10:41:31 +01:00
removeTab(scoreList.indexOf(score->masterScore()));
2012-05-26 14:49:10 +02:00
}
//---------------------------------------------------------
// noteTooShortForTupletDialog
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
void MuseScore::noteTooShortForTupletDialog()
{
QMessageBox::warning(this, tr("Warning"),
tr("Cannot create tuplet: Note value is too short")
2013-05-13 18:49:17 +02:00
);
}
//---------------------------------------------------------
// instrumentChanged
//---------------------------------------------------------
void MuseScore::instrumentChanged()
{
if (mixer)
2016-03-11 12:18:46 +01:00
mixer->updateAll(cs->masterScore());
2013-05-13 18:49:17 +02:00
}
//---------------------------------------------------------
// mixerPreferencesChanged
//---------------------------------------------------------
void MuseScore::mixerPreferencesChanged(bool showMidiControls)
{
if (mixer)
mixer->midiPrefsChanged(showMidiControls);
}
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
// changeScore
// switch current score
// step = 1 switch to next score
// step = -1 switch to previous score
//---------------------------------------------------------
void MuseScore::changeScore(int step)
{
int index = tab1->currentIndex();
int n = tab1->count();
if (n == 1)
return;
index += step;
if (index >= n)
index = 0;
if (index < 0)
index = n - 1;
setCurrentScoreView(index);
}
//---------------------------------------------------------
// switchLayoutMode
// val is index in QComboBox viewModeCombo
2013-05-13 18:49:17 +02:00
//---------------------------------------------------------
void MuseScore::switchLayoutMode(int val)
{
if (!cs)
return;
2016-05-18 15:43:47 +02:00
switchLayoutMode(static_cast<LayoutMode>(viewModeCombo->itemData(val).toInt()));
}
2016-05-18 15:43:47 +02:00
void MuseScore::switchLayoutMode(LayoutMode mode)
{
// find a measure to use as reference, if possible
QRectF view = cv->toLogical(QRect(0.0, 0.0, width(), height()));
Measure* m = cs->firstMeasureMM();
while (m && !view.intersects(m->canvasBoundingRect()))
m = m->nextMeasureMM();
cv->loopUpdate(getAction("loop")->isChecked());
2016-05-18 15:43:47 +02:00
if (mode != cs->layoutMode()) {
cs->setLayoutMode(mode);
cs->doLayout();
}
// adjustCanvasPosition often tries to preserve Y position
// but this doesn't make sense when switching modes
// also, better positioning is usually achieved if you start from the top
// and there is really no better place to position canvas if we were all the way off page previously
cv->pageTop();
if (m && m != cs->firstMeasureMM())
cv->adjustCanvasPosition(m, false);
2013-05-13 18:49:17 +02:00
}
//---------------------------------------------------------
// showDrumTools
//---------------------------------------------------------
void MuseScore::showDrumTools(const Drumset* drumset, Staff* staff)
2013-05-13 18:49:17 +02:00
{
if (drumset) {
if (!_drumTools) {
_drumTools = new DrumTools(this);
addDockWidget(Qt::BottomDockWidgetArea, _drumTools);
}
if (timelineScrollArea())
splitDockWidget(_drumTools, timelineScrollArea(), Qt::Vertical);
2013-05-13 18:49:17 +02:00
_drumTools->setDrumset(cs, staff, drumset);
_drumTools->show();
}
else {
if (_drumTools)
_drumTools->hide();
}
}
//---------------------------------------------------------
// updateDrumTools
//---------------------------------------------------------
2015-03-17 11:50:59 +01:00
void MuseScore::updateDrumTools(const Drumset* ds)
2013-05-13 18:49:17 +02:00
{
if (_drumTools)
2015-03-17 11:50:59 +01:00
_drumTools->updateDrumset(ds);
2013-05-13 18:49:17 +02:00
}
//---------------------------------------------------------
// endSearch
//---------------------------------------------------------
void MuseScore::endSearch()
{
_searchDialog->hide();
if (cv)
cv->setFocus();
}
//---------------------------------------------------------
2013-05-13 18:49:17 +02:00
// showSearchDialog
//---------------------------------------------------------
void MuseScore::showSearchDialog()
{
if (_searchDialog == 0) {
_searchDialog = new QWidget;
_searchDialog->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Maximum);
2013-05-13 18:49:17 +02:00
QHBoxLayout* searchDialogLayout = new QHBoxLayout;
_searchDialog->setLayout(searchDialogLayout);
layout->insertWidget(2, _searchDialog);
2013-05-13 18:49:17 +02:00
QToolButton* searchExit = new QToolButton;
2013-05-09 18:46:51 +02:00
searchExit->setAutoRaise(true);
searchExit->setIcon(*icons[int(Icons::close_ICON)]);
2013-05-13 18:49:17 +02:00
connect(searchExit, SIGNAL(clicked()), SLOT(endSearch()));
searchDialogLayout->addWidget(searchExit);
2013-05-09 18:46:51 +02:00
searchDialogLayout->addSpacing(10);
2013-05-13 18:49:17 +02:00
searchDialogLayout->addWidget(new QLabel(tr("Go To: ")));
2015-02-01 20:50:01 +01:00
searchCombo = new SearchComboBox;
2013-05-13 18:49:17 +02:00
searchDialogLayout->addWidget(searchCombo);
searchDialogLayout->addStretch(10);
_searchDialog->hide();
2013-05-13 18:49:17 +02:00
2014-03-04 16:48:06 +01:00
qDebug("Line edit %p", searchCombo->lineEdit());
2013-05-13 18:49:17 +02:00
connect(searchCombo->lineEdit(), SIGNAL(returnPressed()), SLOT(endSearch()));
2013-05-13 18:49:17 +02:00
}
searchCombo->clearEditText();
searchCombo->setFocus();
_searchDialog->show();
2013-05-13 18:49:17 +02:00
}
2013-05-13 18:49:17 +02:00
#ifndef SCRIPT_INTERFACE
void MuseScore::pluginTriggered(int) {}
void MuseScore::loadPlugins() {}
bool MuseScore::loadPlugin(const QString&) { return false;}
void MuseScore::unloadPlugins() {}
#endif
void MuseScore::saveGeometry(QWidget const*const qw)
{
QSettings settings;
QString objectName = qw->objectName();
Q_ASSERT(!objectName.isEmpty());
settings.beginGroup("Geometries");
settings.setValue(objectName, qw->saveGeometry());
settings.endGroup();
}
void MuseScore::restoreGeometry(QWidget *const qw)
{
if (!useFactorySettings) {
QSettings settings;
QString objectName = qw->objectName();
Q_ASSERT(!objectName.isEmpty());
settings.beginGroup("Geometries");
qw->restoreGeometry(settings.value(objectName).toByteArray());
settings.endGroup();
}
}
void MuseScore::updateWindowTitle(Score* score)
{
#ifdef Q_OS_MAC
2018-04-04 19:06:21 +02:00
if (!cs->isMaster())
setWindowTitle(cs->masterScore()->title() + "-" + cs->title());
else
setWindowTitle(cs->title());
if (score->masterScore()->created())
setWindowFilePath(QString());
2018-04-04 19:06:21 +02:00
else
setWindowFilePath(score->masterScore()->fileInfo()->absoluteFilePath());
#else
if (!cs->isMaster())
setWindowTitle(MUSESCORE_NAME_VERSION ": " + cs->masterScore()->title() + "-" + cs->title() + "[*]");
else
setWindowTitle(MUSESCORE_NAME_VERSION ": " + score->title() + "[*]");
#endif
}
//---------------------------------------------------------
// recentScores
// return a list of recent scores
// omit loaded scores
//---------------------------------------------------------
QFileInfoList MuseScore::recentScores() const
{
QFileInfoList fil;
for (const QString& s : _recentScores) {
if (s.isEmpty())
continue;
QString data(s);
QFileInfo fi(s);
bool alreadyLoaded = false;
QString fp = fi.canonicalFilePath();
2018-08-17 15:06:15 +02:00
for (Score* sc : mscore->scores()) {
if ((sc->masterScore()->fileInfo()->canonicalFilePath() == fp) || (sc->importedFilePath() == fp)) {
alreadyLoaded = true;
break;
}
}
if (!alreadyLoaded && fi.exists())
fil.append(fi);
}
return fil;
}
2014-11-14 13:17:56 +01:00
//---------------------------------------------------------
// createPopupMenu
//---------------------------------------------------------
QMenu* MuseScore::createPopupMenu()
{
QMenu* m = QMainWindow::createPopupMenu();
QList<QAction*> al = m->actions();
for (QAction* a : al) {
// textTool visibility is handled differentlyr
if (_textTools && a->text() == _textTools->windowTitle())
m->removeAction(a);
}
return m;
}
//---------------------------------------------------------
// synthesizerState
//---------------------------------------------------------
SynthesizerState MuseScore::synthesizerState()
{
SynthesizerState state;
return synti ? synti->state() : state;
}
2017-05-16 05:01:10 +02:00
//---------------------------------------------------------
// canSaveMp3
//---------------------------------------------------------
bool MuseScore::canSaveMp3()
{
#ifndef USE_LAME
return false;
#else
2017-05-16 05:01:10 +02:00
MP3Exporter exporter;
if (!exporter.loadLibrary(MP3Exporter::AskUser::NO)) {
qDebug("Could not open MP3 encoding library!");
return false;
}
if (!exporter.validLibraryLoaded()) {
qDebug("Not a valid or supported MP3 encoding library!");
return false;
}
return true;
#endif
2017-05-16 05:01:10 +02:00
}
//---------------------------------------------------------
// saveMp3
//---------------------------------------------------------
bool MuseScore::saveMp3(Score* score, const QString& name)
{
#ifndef USE_LAME
Q_UNUSED(score);
Q_UNUSED(name);
return false;
#else
2017-05-16 05:01:10 +02:00
EventMap events;
score->renderMidi(&events);
if(events.size() == 0)
return false;
MP3Exporter exporter;
if (!exporter.loadLibrary(MP3Exporter::AskUser::MAYBE)) {
QSettings settings;
settings.setValue("/Export/lameMP3LibPath", "");
if(!MScore::noGui)
QMessageBox::warning(0,
tr("Error Opening LAME library"),
tr("Could not open MP3 encoding library!"),
QString::null, QString::null);
qDebug("Could not open MP3 encoding library!");
return false;
}
if (!exporter.validLibraryLoaded()) {
QSettings settings;
settings.setValue("/Export/lameMP3LibPath", "");
if(!MScore::noGui)
QMessageBox::warning(0,
tr("Error Opening LAME library"),
tr("Not a valid or supported MP3 encoding library!"),
QString::null, QString::null);
qDebug("Not a valid or supported MP3 encoding library!");
return false;
}
// Retrieve preferences
// int highrate = 48000;
// int lowrate = 8000;
// int bitrate = 64;
// int brate = 128;
// int rmode = MODE_CBR;
// int vmode = ROUTINE_FAST;
// int cmode = CHANNEL_STEREO;
int channels = 2;
int oldSampleRate = MScore::sampleRate;
int sampleRate = preferences.getInt(PREF_EXPORT_AUDIO_SAMPLERATE);
exporter.setBitrate(preferences.getInt(PREF_EXPORT_MP3_BITRATE));
2017-05-16 05:01:10 +02:00
int inSamples = exporter.initializeStream(channels, sampleRate);
if (inSamples < 0) {
if (!MScore::noGui) {
QMessageBox::warning(0, tr("Encoding Error"),
tr("Unable to initialize MP3 stream"),
QString::null, QString::null);
}
qDebug("Unable to initialize MP3 stream");
MScore::sampleRate = oldSampleRate;
return false;
}
QFile file(name);
if (!file.open(QIODevice::WriteOnly)) {
if (!MScore::noGui) {
QMessageBox::warning(0,
tr("Encoding Error"),
tr("Unable to open target file for writing"),
QString::null, QString::null);
}
MScore::sampleRate = oldSampleRate;
return false;
}
int bufferSize = exporter.getOutBufferSize();
uchar* bufferOut = new uchar[bufferSize];
MasterSynthesizer* synth = synthesizerFactory();
synth->init();
synth->setSampleRate(sampleRate);
2017-05-16 05:01:10 +02:00
if (MScore::noGui) { // use score settings if possible
bool r = synth->setState(score->synthesizerState());
2017-05-16 05:01:10 +02:00
if (!r)
synth->init();
2017-05-16 05:01:10 +02:00
}
else { // use current synth settings
bool r = synth->setState(mscore->synthesizerState());
2017-05-16 05:01:10 +02:00
if (!r)
synth->init();
2017-05-16 05:01:10 +02:00
}
MScore::sampleRate = sampleRate;
QProgressDialog progress(this);
progress.setWindowFlags(Qt::WindowFlags(Qt::Dialog | Qt::FramelessWindowHint | Qt::WindowTitleHint));
progress.setWindowModality(Qt::ApplicationModal);
//progress.setCancelButton(0);
progress.setCancelButtonText(tr("Cancel"));
progress.setLabelText(tr("Exporting..."));
if (!MScore::noGui)
progress.show();
static const int FRAMES = 512;
float bufferL[FRAMES];
float bufferR[FRAMES];
float peak = 0.0;
double gain = 1.0;
EventMap::const_iterator endPos = events.cend();
--endPos;
const int et = (score->utick2utime(endPos->first) + 1) * MScore::sampleRate;
const int maxEndTime = (score->utick2utime(endPos->first) + 3) * MScore::sampleRate;
progress.setRange(0, et);
for (int pass = 0; pass < 2; ++pass) {
EventMap::const_iterator playPos;
playPos = events.cbegin();
synth->allSoundsOff(-1);
2017-05-16 05:01:10 +02:00
//
// init instruments
//
foreach(Part* part, score->parts()) {
const InstrumentList* il = part->instruments();
for(auto i = il->begin(); i!= il->end(); i++) {
for (const Channel* a : i->second->channel()) {
2017-05-16 05:01:10 +02:00
a->updateInitList();
for (MidiCoreEvent e : a->init) {
2017-05-16 05:01:10 +02:00
if (e.type() == ME_INVALID)
continue;
e.setChannel(a->channel);
int syntiIdx= synth->index(score->masterScore()->midiMapping(a->channel)->articulation->synti);
synth->play(e, syntiIdx);
2017-05-16 05:01:10 +02:00
}
}
}
}
int playTime = 0.0;
for (;;) {
unsigned frames = FRAMES;
float max = 0;
//
// collect events for one segment
//
memset(bufferL, 0, sizeof(float) * FRAMES);
memset(bufferR, 0, sizeof(float) * FRAMES);
double endTime = playTime + frames;
float* l = bufferL;
float* r = bufferR;
for (; playPos != events.cend(); ++playPos) {
double f = score->utick2utime(playPos->first) * MScore::sampleRate;
if (f >= endTime)
break;
int n = f - playTime;
if (n) {
Solved all compilation errors on MSVC This commit contains changes required for MuseScore to compile under MSVC with no errors. There are several general categories of problems that resulted in errors with the compilation. Main issues: - Variable Length Arrays (VLA). This is a non-standard extension to C++, supported by clang toolchains, but not by MSVC. The initial workaround is to use std::vector<> instead of VLAs, eventually (if needed) with the original variable pointing to the beginning of the allocated array. More efficient alternatives are possible if profiling shows any code using VLAs to be timing-critical. - Floating-point constants not suffixed with "f" are doubles by default; in some instances, this leads to narrowing conversion errors (in other instances, just warnings). - MSVC does not support "or"/"and"/"not" in lieu of "||"/"&&"/"!". Changed unconditionally to use standard C++ symbols. - MSVC does not support the "__builtin_unreachable()" compiler hint. A similar, albeit not exactly equal, alternative is "__assume(0)", which is MSVC-specific. - MSVC does not support ranges in case statements. Replaced with list of cases instead of range (non-conditionally) Detailed changes, with per-file comments: - all.h: opt-in to deprecated features, include <io.h> and <process.h> instead of POSIX <unistd.h>, undefine "STRING_NONE" and "small", which Microsoft defines as macros, which result in compilation errors. - effects/compressor/zita.cpp: eliminated narrowing conversion error by appending "f" to float constants. - fluid/voice.cpp: appended "f" to float constants to eliminate narrowing conversion errors/warnings - libmscore/beam.cpp: conditionally replaced VLA - libmscore/edit.cpp: conditionally replaced VLA - libmscore/element.cpp: appended "f" to float constant - libmscore/fret.cpp: changed or -> || - libmscore/layout.cpp: conditionally replaced VLA - libmscore/mscore.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - libmscore/scorefile.coo: use correct char representation conversion for MSVC - libmscore/stringdata.cpp: conditionally replaced VLA - libmscore/system.cpp: conditionally replaced VLA - libmscroe/text.cpp: replaced range in case statement. - manual/genManual.cpp: use getopt() replacement. - midi/midifile.cpp: conditionally replaced VLA. This does not use the default replacement. - mscore/bb.cpp: replaced range in case statement. - mscore/capella.cpp: conditionally replaced VLA. Changed and -> && - mscore/driver:cpp: preclude errors due to macro redefinitions. - mscore/editstyle.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - mscore/importgtp-gp6.cpp: conditionally replaced VLA. - mscore/instrwidget.cpp: conditionally replaced VLA. - mscore/jackaudio.cpp: conditionally replaced VLA. Preclude errors due to macro redefinitions. - mscore/jackweakapi.cpp: Preclude errors due to macro redefinitions. Replacement for __atribute__((constructor)) through static object construction for MSVC. Force use of LoadLibraryA instead of LoadLibrary. - mscore/mididriver.h: Changed not -> ! - mscore/musescore.cpp: Changed not -> !. Conditionally replaced VLA. - mscore/resourceManager.cpp: conditionally replaced VLA. - mscore/timeline.cpp: conditionally replaced VLA. - omr/omrpage.cpp: conditionally replaced VLA. - synthesizer/msynthesizer.cpp: replaced UNIX sleep(1) method with MSVC Sleep(1000) (equivalent, but in ms instead of seconds) - synthesizer/msynthsizer.h: appended "f" to float constant - thirdparty/poppler/config.h: set defines for MSVC to preclude the use of inexistent libraries. - thirdparty/poppler/poppler/poppler-config.h: set defines for MSVC to preclude the use of inexistent libraries. Eliminated #defines for fmin and fmax which where causing problems. - thirdparty/poppler/poppler/PSOutputDev.cc: added #include <algorithm> for std::min() and std::max(). Note this is required per-C++ standard. - thirdparty/portmidi/pm_win/pmwinmm.c: undefined UNICODE to use char-based library functions. - thirdparty/qzip/qzip.cpp: changed or -> || - thirdparty/rtf2html/rtf_keyword.h: file format changed from Apple to UNIX, as MSVC does not supported the former. (NOT error related, just improvement on previous commit; manual/getopt/README.md: added link to source article)
2018-05-01 19:14:44 +02:00
#if (!defined (_MSCVER) && !defined (_MSC_VER))
2017-05-16 05:01:10 +02:00
float bu[n * 2];
memset(bu, 0, sizeof(float) * 2 * n);
Solved all compilation errors on MSVC This commit contains changes required for MuseScore to compile under MSVC with no errors. There are several general categories of problems that resulted in errors with the compilation. Main issues: - Variable Length Arrays (VLA). This is a non-standard extension to C++, supported by clang toolchains, but not by MSVC. The initial workaround is to use std::vector<> instead of VLAs, eventually (if needed) with the original variable pointing to the beginning of the allocated array. More efficient alternatives are possible if profiling shows any code using VLAs to be timing-critical. - Floating-point constants not suffixed with "f" are doubles by default; in some instances, this leads to narrowing conversion errors (in other instances, just warnings). - MSVC does not support "or"/"and"/"not" in lieu of "||"/"&&"/"!". Changed unconditionally to use standard C++ symbols. - MSVC does not support the "__builtin_unreachable()" compiler hint. A similar, albeit not exactly equal, alternative is "__assume(0)", which is MSVC-specific. - MSVC does not support ranges in case statements. Replaced with list of cases instead of range (non-conditionally) Detailed changes, with per-file comments: - all.h: opt-in to deprecated features, include <io.h> and <process.h> instead of POSIX <unistd.h>, undefine "STRING_NONE" and "small", which Microsoft defines as macros, which result in compilation errors. - effects/compressor/zita.cpp: eliminated narrowing conversion error by appending "f" to float constants. - fluid/voice.cpp: appended "f" to float constants to eliminate narrowing conversion errors/warnings - libmscore/beam.cpp: conditionally replaced VLA - libmscore/edit.cpp: conditionally replaced VLA - libmscore/element.cpp: appended "f" to float constant - libmscore/fret.cpp: changed or -> || - libmscore/layout.cpp: conditionally replaced VLA - libmscore/mscore.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - libmscore/scorefile.coo: use correct char representation conversion for MSVC - libmscore/stringdata.cpp: conditionally replaced VLA - libmscore/system.cpp: conditionally replaced VLA - libmscroe/text.cpp: replaced range in case statement. - manual/genManual.cpp: use getopt() replacement. - midi/midifile.cpp: conditionally replaced VLA. This does not use the default replacement. - mscore/bb.cpp: replaced range in case statement. - mscore/capella.cpp: conditionally replaced VLA. Changed and -> && - mscore/driver:cpp: preclude errors due to macro redefinitions. - mscore/editstyle.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - mscore/importgtp-gp6.cpp: conditionally replaced VLA. - mscore/instrwidget.cpp: conditionally replaced VLA. - mscore/jackaudio.cpp: conditionally replaced VLA. Preclude errors due to macro redefinitions. - mscore/jackweakapi.cpp: Preclude errors due to macro redefinitions. Replacement for __atribute__((constructor)) through static object construction for MSVC. Force use of LoadLibraryA instead of LoadLibrary. - mscore/mididriver.h: Changed not -> ! - mscore/musescore.cpp: Changed not -> !. Conditionally replaced VLA. - mscore/resourceManager.cpp: conditionally replaced VLA. - mscore/timeline.cpp: conditionally replaced VLA. - omr/omrpage.cpp: conditionally replaced VLA. - synthesizer/msynthesizer.cpp: replaced UNIX sleep(1) method with MSVC Sleep(1000) (equivalent, but in ms instead of seconds) - synthesizer/msynthsizer.h: appended "f" to float constant - thirdparty/poppler/config.h: set defines for MSVC to preclude the use of inexistent libraries. - thirdparty/poppler/poppler/poppler-config.h: set defines for MSVC to preclude the use of inexistent libraries. Eliminated #defines for fmin and fmax which where causing problems. - thirdparty/poppler/poppler/PSOutputDev.cc: added #include <algorithm> for std::min() and std::max(). Note this is required per-C++ standard. - thirdparty/portmidi/pm_win/pmwinmm.c: undefined UNICODE to use char-based library functions. - thirdparty/qzip/qzip.cpp: changed or -> || - thirdparty/rtf2html/rtf_keyword.h: file format changed from Apple to UNIX, as MSVC does not supported the former. (NOT error related, just improvement on previous commit; manual/getopt/README.md: added link to source article)
2018-05-01 19:14:44 +02:00
#else
// MSVC does not support VLA. Replace with std::vector. If profiling determines that the
// heap allocation is slow, an optimization might be used.
std::vector<float> vBu(n * 2, 0); // Default initialized, memset() not required.
float* bu = vBu.data();
#endif
2017-05-16 05:01:10 +02:00
synth->process(n, bu);
2017-05-16 05:01:10 +02:00
float* sp = bu;
for (int i = 0; i < n; ++i) {
*l++ = *sp++;
*r++ = *sp++;
}
playTime += n;
frames -= n;
}
const NPlayEvent& e = playPos->second;
if (e.isChannelEvent()) {
int channelIdx = e.channel();
Channel* c = score->masterScore()->midiMapping(channelIdx)->articulation;
if (!c->mute) {
synth->play(e, synth->index(c->synti));
2017-05-16 05:01:10 +02:00
}
}
}
if (frames) {
Solved all compilation errors on MSVC This commit contains changes required for MuseScore to compile under MSVC with no errors. There are several general categories of problems that resulted in errors with the compilation. Main issues: - Variable Length Arrays (VLA). This is a non-standard extension to C++, supported by clang toolchains, but not by MSVC. The initial workaround is to use std::vector<> instead of VLAs, eventually (if needed) with the original variable pointing to the beginning of the allocated array. More efficient alternatives are possible if profiling shows any code using VLAs to be timing-critical. - Floating-point constants not suffixed with "f" are doubles by default; in some instances, this leads to narrowing conversion errors (in other instances, just warnings). - MSVC does not support "or"/"and"/"not" in lieu of "||"/"&&"/"!". Changed unconditionally to use standard C++ symbols. - MSVC does not support the "__builtin_unreachable()" compiler hint. A similar, albeit not exactly equal, alternative is "__assume(0)", which is MSVC-specific. - MSVC does not support ranges in case statements. Replaced with list of cases instead of range (non-conditionally) Detailed changes, with per-file comments: - all.h: opt-in to deprecated features, include <io.h> and <process.h> instead of POSIX <unistd.h>, undefine "STRING_NONE" and "small", which Microsoft defines as macros, which result in compilation errors. - effects/compressor/zita.cpp: eliminated narrowing conversion error by appending "f" to float constants. - fluid/voice.cpp: appended "f" to float constants to eliminate narrowing conversion errors/warnings - libmscore/beam.cpp: conditionally replaced VLA - libmscore/edit.cpp: conditionally replaced VLA - libmscore/element.cpp: appended "f" to float constant - libmscore/fret.cpp: changed or -> || - libmscore/layout.cpp: conditionally replaced VLA - libmscore/mscore.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - libmscore/scorefile.coo: use correct char representation conversion for MSVC - libmscore/stringdata.cpp: conditionally replaced VLA - libmscore/system.cpp: conditionally replaced VLA - libmscroe/text.cpp: replaced range in case statement. - manual/genManual.cpp: use getopt() replacement. - midi/midifile.cpp: conditionally replaced VLA. This does not use the default replacement. - mscore/bb.cpp: replaced range in case statement. - mscore/capella.cpp: conditionally replaced VLA. Changed and -> && - mscore/driver:cpp: preclude errors due to macro redefinitions. - mscore/editstyle.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - mscore/importgtp-gp6.cpp: conditionally replaced VLA. - mscore/instrwidget.cpp: conditionally replaced VLA. - mscore/jackaudio.cpp: conditionally replaced VLA. Preclude errors due to macro redefinitions. - mscore/jackweakapi.cpp: Preclude errors due to macro redefinitions. Replacement for __atribute__((constructor)) through static object construction for MSVC. Force use of LoadLibraryA instead of LoadLibrary. - mscore/mididriver.h: Changed not -> ! - mscore/musescore.cpp: Changed not -> !. Conditionally replaced VLA. - mscore/resourceManager.cpp: conditionally replaced VLA. - mscore/timeline.cpp: conditionally replaced VLA. - omr/omrpage.cpp: conditionally replaced VLA. - synthesizer/msynthesizer.cpp: replaced UNIX sleep(1) method with MSVC Sleep(1000) (equivalent, but in ms instead of seconds) - synthesizer/msynthsizer.h: appended "f" to float constant - thirdparty/poppler/config.h: set defines for MSVC to preclude the use of inexistent libraries. - thirdparty/poppler/poppler/poppler-config.h: set defines for MSVC to preclude the use of inexistent libraries. Eliminated #defines for fmin and fmax which where causing problems. - thirdparty/poppler/poppler/PSOutputDev.cc: added #include <algorithm> for std::min() and std::max(). Note this is required per-C++ standard. - thirdparty/portmidi/pm_win/pmwinmm.c: undefined UNICODE to use char-based library functions. - thirdparty/qzip/qzip.cpp: changed or -> || - thirdparty/rtf2html/rtf_keyword.h: file format changed from Apple to UNIX, as MSVC does not supported the former. (NOT error related, just improvement on previous commit; manual/getopt/README.md: added link to source article)
2018-05-01 19:14:44 +02:00
#if (!defined (_MSCVER) && !defined (_MSC_VER))
2017-05-16 05:01:10 +02:00
float bu[frames * 2];
memset(bu, 0, sizeof(float) * 2 * frames);
Solved all compilation errors on MSVC This commit contains changes required for MuseScore to compile under MSVC with no errors. There are several general categories of problems that resulted in errors with the compilation. Main issues: - Variable Length Arrays (VLA). This is a non-standard extension to C++, supported by clang toolchains, but not by MSVC. The initial workaround is to use std::vector<> instead of VLAs, eventually (if needed) with the original variable pointing to the beginning of the allocated array. More efficient alternatives are possible if profiling shows any code using VLAs to be timing-critical. - Floating-point constants not suffixed with "f" are doubles by default; in some instances, this leads to narrowing conversion errors (in other instances, just warnings). - MSVC does not support "or"/"and"/"not" in lieu of "||"/"&&"/"!". Changed unconditionally to use standard C++ symbols. - MSVC does not support the "__builtin_unreachable()" compiler hint. A similar, albeit not exactly equal, alternative is "__assume(0)", which is MSVC-specific. - MSVC does not support ranges in case statements. Replaced with list of cases instead of range (non-conditionally) Detailed changes, with per-file comments: - all.h: opt-in to deprecated features, include <io.h> and <process.h> instead of POSIX <unistd.h>, undefine "STRING_NONE" and "small", which Microsoft defines as macros, which result in compilation errors. - effects/compressor/zita.cpp: eliminated narrowing conversion error by appending "f" to float constants. - fluid/voice.cpp: appended "f" to float constants to eliminate narrowing conversion errors/warnings - libmscore/beam.cpp: conditionally replaced VLA - libmscore/edit.cpp: conditionally replaced VLA - libmscore/element.cpp: appended "f" to float constant - libmscore/fret.cpp: changed or -> || - libmscore/layout.cpp: conditionally replaced VLA - libmscore/mscore.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - libmscore/scorefile.coo: use correct char representation conversion for MSVC - libmscore/stringdata.cpp: conditionally replaced VLA - libmscore/system.cpp: conditionally replaced VLA - libmscroe/text.cpp: replaced range in case statement. - manual/genManual.cpp: use getopt() replacement. - midi/midifile.cpp: conditionally replaced VLA. This does not use the default replacement. - mscore/bb.cpp: replaced range in case statement. - mscore/capella.cpp: conditionally replaced VLA. Changed and -> && - mscore/driver:cpp: preclude errors due to macro redefinitions. - mscore/editstyle.cpp: conditionally replaced "__builtin_unreachable()" with "__assume(0)" for MSVC - mscore/importgtp-gp6.cpp: conditionally replaced VLA. - mscore/instrwidget.cpp: conditionally replaced VLA. - mscore/jackaudio.cpp: conditionally replaced VLA. Preclude errors due to macro redefinitions. - mscore/jackweakapi.cpp: Preclude errors due to macro redefinitions. Replacement for __atribute__((constructor)) through static object construction for MSVC. Force use of LoadLibraryA instead of LoadLibrary. - mscore/mididriver.h: Changed not -> ! - mscore/musescore.cpp: Changed not -> !. Conditionally replaced VLA. - mscore/resourceManager.cpp: conditionally replaced VLA. - mscore/timeline.cpp: conditionally replaced VLA. - omr/omrpage.cpp: conditionally replaced VLA. - synthesizer/msynthesizer.cpp: replaced UNIX sleep(1) method with MSVC Sleep(1000) (equivalent, but in ms instead of seconds) - synthesizer/msynthsizer.h: appended "f" to float constant - thirdparty/poppler/config.h: set defines for MSVC to preclude the use of inexistent libraries. - thirdparty/poppler/poppler/poppler-config.h: set defines for MSVC to preclude the use of inexistent libraries. Eliminated #defines for fmin and fmax which where causing problems. - thirdparty/poppler/poppler/PSOutputDev.cc: added #include <algorithm> for std::min() and std::max(). Note this is required per-C++ standard. - thirdparty/portmidi/pm_win/pmwinmm.c: undefined UNICODE to use char-based library functions. - thirdparty/qzip/qzip.cpp: changed or -> || - thirdparty/rtf2html/rtf_keyword.h: file format changed from Apple to UNIX, as MSVC does not supported the former. (NOT error related, just improvement on previous commit; manual/getopt/README.md: added link to source article)
2018-05-01 19:14:44 +02:00
#else
// MSVC does not support VLA. Replace with std::vector. If profiling determines that the
// heap allocation is slow, an optimization might be used.
std::vector<float> vBu(frames * 2, 0); // Default initialized, memset() not required.
float* bu = vBu.data();
#endif
synth->process(frames, bu);
2017-05-16 05:01:10 +02:00
float* sp = bu;
for (unsigned i = 0; i < frames; ++i) {
*l++ = *sp++;
*r++ = *sp++;
}
playTime += frames;
}
if (pass == 1) {
for (int i = 0; i < FRAMES; ++i) {
max = qMax(max, qAbs(bufferL[i]));
max = qMax(max, qAbs(bufferR[i]));
bufferL[i] *= gain;
bufferR[i] *= gain;
}
long bytes;
if (FRAMES < inSamples)
bytes = exporter.encodeRemainder(bufferL, bufferR, FRAMES , bufferOut);
else
bytes = exporter.encodeBuffer(bufferL, bufferR, bufferOut);
if (bytes < 0) {
if (MScore::noGui)
qDebug("exportmp3: error from encoder: %ld", bytes);
else
QMessageBox::warning(0,
tr("Encoding Error"),
tr("Error %1 returned from MP3 encoder").arg(bytes),
QString::null, QString::null);
break;
}
else
file.write((char*)bufferOut, bytes);
}
else {
for (int i = 0; i < FRAMES; ++i) {
max = qMax(max, qAbs(bufferL[i]));
max = qMax(max, qAbs(bufferR[i]));
peak = qMax(peak, qAbs(bufferL[i]));
peak = qMax(peak, qAbs(bufferR[i]));
}
}
playTime = endTime;
if (!MScore::noGui) {
if (progress.wasCanceled())
break;
progress.setValue((pass * et + playTime) / 2);
qApp->processEvents();
}
if (playTime >= et)
synti->allNotesOff(-1);
// create sound until the sound decays
if (playTime >= et && max * peak < 0.000001)
break;
// hard limit
if (playTime > maxEndTime)
break;
}
if (progress.wasCanceled())
break;
if (pass == 0 && peak == 0.0) {
qDebug("song is empty");
break;
}
gain = 0.99 / peak;
}
long bytes = exporter.finishStream(bufferOut);
if (bytes > 0L)
file.write((char*)bufferOut, bytes);
bool wasCanceled = progress.wasCanceled();
progress.close();
delete synth;
2017-05-16 05:01:10 +02:00
delete[] bufferOut;
file.close();
if (wasCanceled)
file.remove();
MScore::sampleRate = oldSampleRate;
return true;
#endif
2017-05-16 05:01:10 +02:00
}
void MuseScore::updateUiStyleAndTheme()
{
// set UI Theme
QApplication::setStyle(QStyleFactory::create("Fusion"));
QString wd = QString("%1/%2").arg(QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation)).arg(QCoreApplication::applicationName());
// set UI Color Palette
QPalette p(QApplication::palette());
QString jsonPaletteFilename = preferences.isThemeDark() ? "palette_dark_fusion.json" : "palette_light_fusion.json";;
QFile jsonPalette(QString(":/themes/%1").arg(jsonPaletteFilename));
// read from Documents TODO: remove this
if (QFile::exists(QString("%1/%2").arg(wd, "ms_palette.json")))
jsonPalette.setFileName(QString("%1/%2").arg(wd, "ms_palette.json"));
if (jsonPalette.open(QFile::ReadOnly | QFile::Text)) {
QJsonDocument d = QJsonDocument::fromJson(jsonPalette.readAll());
QJsonObject o = d.object();
QMetaEnum metaEnum = QMetaEnum::fromType<QPalette::ColorRole>();
for (int i = 0; i < metaEnum.keyCount(); ++i) {
QJsonValue v = o.value(metaEnum.valueToKey(i));
if (!v.isUndefined())
p.setColor(static_cast<QPalette::ColorRole>(metaEnum.value(i)), QColor(v.toString()));
}
}
QApplication::setPalette(p);
// set UI Style
QString css;
QString styleFilename = preferences.isThemeDark() ? "style_dark_fusion.css" : "style_light_fusion.css";
QFile fstyle(QString(":/themes/%1").arg(styleFilename));
// read from Documents TODO: remove this
if (QFile::exists(QString("%1/%2").arg(wd, "ms_style.css")))
fstyle.setFileName(QString("%1/%2").arg(wd, "ms_style.css"));
if (fstyle.open(QFile::ReadOnly | QFile::Text)) {
QTextStream in(&fstyle);
css = in.readAll();
}
css.replace("$voice1-bgcolor", MScore::selectColor[0].name(QColor::HexRgb));
css.replace("$voice2-bgcolor", MScore::selectColor[1].name(QColor::HexRgb));
css.replace("$voice3-bgcolor", MScore::selectColor[2].name(QColor::HexRgb));
css.replace("$voice4-bgcolor", MScore::selectColor[3].name(QColor::HexRgb));
qApp->setStyleSheet(css);
genIcons();
Shortcut::refreshIcons();
}
2013-05-13 18:49:17 +02:00
}
using namespace Ms;
//---------------------------------------------------------
// main
//---------------------------------------------------------
int main(int argc, char* av[])
{
2016-02-04 11:27:47 +01:00
#ifndef NDEBUG
qSetMessagePattern("%{file}:%{function}: %{message}");
2017-01-16 20:51:12 +01:00
Ms::checkStyles();
2016-02-04 11:27:47 +01:00
#endif
2015-01-07 17:16:58 +01:00
QApplication::setDesktopSettingsAware(true);
QApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
#if defined(QT_DEBUG) && defined(Q_OS_WIN)
qInstallMessageHandler(mscoreMessageHandler);
2013-05-13 18:49:17 +02:00
#endif
qRegisterMetaTypeStreamOperators<SessionStart>("SessionStart");
qRegisterMetaTypeStreamOperators<MusicxmlExportBreaks>("MusicxmlExportBreaks");
qRegisterMetaTypeStreamOperators<MuseScoreStyleType>("MuseScoreStyleType");
2013-05-13 18:49:17 +02:00
QFile f(":/revision.h");
f.open(QIODevice::ReadOnly);
revision = QString(f.readAll()).trimmed();
f.close();
2017-05-02 14:17:31 +02:00
const char* appName;
const char* appName2;
if (MuseScore::unstable()) {
2017-05-02 14:17:31 +02:00
appName2 = "mscore-dev3";
appName = "MuseScore3Development";
}
else {
2017-05-02 14:17:31 +02:00
appName2 = "mscore3";
appName = "MuseScore3";
}
2017-11-27 09:56:41 +01:00
2017-05-02 14:17:31 +02:00
MuseScoreApplication* app = new MuseScoreApplication(appName2, argc, av);
QCoreApplication::setApplicationName(appName);
2013-05-13 18:49:17 +02:00
QCoreApplication::setOrganizationName("MuseScore");
QCoreApplication::setOrganizationDomain("musescore.org");
2015-03-09 17:20:23 +01:00
QCoreApplication::setApplicationVersion(VERSION);
2017-05-02 14:17:31 +02:00
QAccessible::installFactory(AccessibleScoreView::ScoreViewFactory);
2015-02-01 20:50:01 +01:00
QAccessible::installFactory(AccessibleSearchBox::SearchBoxFactory);
QAccessible::installFactory(Awl::AccessibleAbstractSlider::AbstractSliderFactory);
2013-05-13 18:49:17 +02:00
Q_INIT_RESOURCE(zita);
#ifndef Q_OS_MAC
2013-05-13 18:49:17 +02:00
QSettings::setDefaultFormat(QSettings::IniFormat);
#endif
QCommandLineParser parser;
parser.addHelpOption(); // -?, -h, --help
parser.addVersionOption(); // -v, --version
//parser.addOption(QCommandLineOption({"v", "version"}, "Print version")); // see above
parser.addOption(QCommandLineOption( "long-version", "Print detailed version information"));
parser.addOption(QCommandLineOption({"d", "debug"}, "Debug mode"));
parser.addOption(QCommandLineOption({"L", "layout-debug"}, "Layout debug mode"));
parser.addOption(QCommandLineOption({"s", "no-synthesizer"}, "No internal synthesizer"));
parser.addOption(QCommandLineOption({"m", "no-midi"}, "No MIDI"));
parser.addOption(QCommandLineOption({"a", "use-audio"}, "Use audio driver: jack, alsa, pulse, or portaudio", "driver"));
parser.addOption(QCommandLineOption({"n", "new-score"}, "Start with new score"));
parser.addOption(QCommandLineOption({"I", "dump-midi-in"}, "Dump midi input"));
parser.addOption(QCommandLineOption({"O", "dump-midi-out"}, "Dump midi output"));
parser.addOption(QCommandLineOption({"o", "export-to"}, "Export to 'file'. Format depends on file's extension", "file"));
parser.addOption(QCommandLineOption({"r", "image-resolution"}, "Used with '-o <file>.png'. Set output resolution for image export", "DPI"));
parser.addOption(QCommandLineOption({"T", "trim-image"}, "Used with '-o <file>.png' and '-o <file.svg>'. Trim exported image with specified margin (in pixels)", "margin"));
parser.addOption(QCommandLineOption({"x", "gui-scaling"}, "Set scaling factor for GUI elements", "factor"));
parser.addOption(QCommandLineOption({"D", "monitor-resolution"}, "Specify monitor resolution", "DPI"));
parser.addOption(QCommandLineOption({"S", "style"}, "Load style file", "style"));
2017-01-18 20:56:41 +01:00
parser.addOption(QCommandLineOption({"p", "plugin"}, "Execute named plugin", "name"));
parser.addOption(QCommandLineOption( "template-mode", "Save template mode, no page size"));
parser.addOption(QCommandLineOption({"F", "factory-settings"}, "Use factory settings"));
parser.addOption(QCommandLineOption({"R", "revert-settings"}, "Revert to default preferences"));
parser.addOption(QCommandLineOption({"i", "load-icons"}, "Load icons from INSTALLPATH/icons"));
parser.addOption(QCommandLineOption({"j", "job"}, "Process a conversion job", "file"));
parser.addOption(QCommandLineOption({"e", "experimental"}, "Enable experimental features"));
parser.addOption(QCommandLineOption({"c", "config-folder"}, "Override configuration and settings folder", "dir"));
parser.addOption(QCommandLineOption({"t", "test-mode"}, "Set test mode flag for all files"));
parser.addOption(QCommandLineOption({"M", "midi-operations"}, "Specify MIDI import operations file", "file"));
parser.addOption(QCommandLineOption({"w", "no-webview"}, "No web view in start center"));
parser.addOption(QCommandLineOption({"P", "export-score-parts"}, "Used with '-o <file>.pdf', export score and parts"));
parser.addOption(QCommandLineOption( "no-fallback-font", "Don't use Bravura as fallback musical font"));
parser.addOption(QCommandLineOption({"f", "force"}, "Used with '-o <file>', ignore warnings reg. score being corrupted or from wrong version"));
parser.addOption(QCommandLineOption({"b", "bitrate"}, "Used with '-o <file>.mp3', sets bitrate, in kbps", "bitrate"));
parser.addOption(QCommandLineOption({"E", "install-extension"}, "Install an extension, load soundfont as default unless if -e is passed too", "extension file"));
parser.addPositionalArgument("scorefiles", "The files to open", "[scorefile...]");
parser.process(QCoreApplication::arguments());
//if (parser.isSet("v")) parser.showVersion(); // a) needs Qt >= 5.4 , b) instead we use addVersionOption()
if (parser.isSet("long-version")) {
printVersion("MuseScore");
return EXIT_SUCCESS;
}
MScore::debugMode = parser.isSet("d");
MScore::noHorizontalStretch = MScore::noVerticalStretch = parser.isSet("L");
noSeq = parser.isSet("s");
noMidi = parser.isSet("m");
if (parser.isSet("a")) {
audioDriver = parser.value("a");
if (audioDriver.isEmpty())
parser.showHelp(EXIT_FAILURE);
}
startWithNewScore = parser.isSet("n");
externalIcons = parser.isSet("i");
midiInputTrace = parser.isSet("I");
midiOutputTrace = parser.isSet("O");
MScore::useFallbackFont = !parser.isSet("no-fallback-font");
if ((converterMode = parser.isSet("o"))) {
MScore::noGui = true;
outFileName = parser.value("o");
if (outFileName.isEmpty())
parser.showHelp(EXIT_FAILURE);
}
if ((processJob = parser.isSet("j"))) {
MScore::noGui = true;
converterMode = true;
jsonFileName = parser.value("j");
if (jsonFileName.isEmpty()) {
fprintf(stderr, "json file name missing\n");
parser.showHelp(EXIT_FAILURE);
}
}
if ((pluginMode = parser.isSet("p"))) {
MScore::noGui = true;
pluginName = parser.value("p");
if (pluginName.isEmpty())
parser.showHelp(EXIT_FAILURE);
}
if (parser.isSet("E")) {
MScore::noGui = true;
extensionName = parser.value("E");
}
MScore::saveTemplateMode = parser.isSet("template-mode");
if (parser.isSet("r")) {
QString temp = parser.value("r");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
bool ok = false;
double res = temp.toDouble(&ok);
if (ok)
preferences.setTemporaryPreference(PREF_EXPORT_PNG_RESOLUTION, res);
else
fprintf(stderr, "PNG resolution value '%s' not recognized, using default setting from preferences instead.\n", qPrintable(temp));
}
if (parser.isSet("T")) {
QString temp = parser.value("T");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
bool ok = false;
trimMargin = temp.toInt(&ok);
if (!ok) {
fprintf(stderr, "Trim margin value '%s' not recognized, so no trimming will be done.\n", qPrintable(temp));
trimMargin = -1;
}
}
if (parser.isSet("x")) {
QString temp = parser.value("x");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
bool ok = false;
guiScaling = temp.toDouble(&ok);
if (!ok) {
fprintf(stderr, "GUI scaling value '%s' not recognized, so the values detected by Qt are taken.\n", qPrintable(temp));
guiScaling = 0.0;
}
}
if (parser.isSet("D")) {
QString temp = parser.value("D");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
bool ok = false;
userDPI = temp.toDouble(&ok);
if (!ok) {
fprintf(stderr, "DPI value '%s' not recognized, so the values detected by Qt are taken.\n", qPrintable(temp));
userDPI = 0.0;
}
}
if (parser.isSet("S")) {
styleFile = parser.value("S");
if (styleFile.isEmpty())
parser.showHelp(EXIT_FAILURE);
}
useFactorySettings = parser.isSet("F");
deletePreferences = (useFactorySettings || parser.isSet("R"));
enableExperimental = parser.isSet("e");
if (parser.isSet("c")) {
QString path = parser.value("c");
if (path.isEmpty())
parser.showHelp(EXIT_FAILURE);
QDir dir;
if (dir.exists(path)) {
QSettings::setPath(QSettings::IniFormat, QSettings::UserScope, path);
QSettings::setPath(QSettings::IniFormat, QSettings::SystemScope, path);
dataPath = path;
2013-05-13 18:49:17 +02:00
}
}
2017-03-31 13:03:15 +02:00
MScore::testMode = parser.isSet("t");
if (parser.isSet("M")) {
QString temp = parser.value("M");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
midiImportOperations.setOperationsFile(temp);
}
noWebView = parser.isSet("w");
exportScoreParts = parser.isSet("export-score-parts");
if (exportScoreParts && !converterMode)
parser.showHelp(EXIT_FAILURE);
ignoreWarnings = parser.isSet("f");
if (parser.isSet("b")) {
QString temp = parser.value("b");
if (temp.isEmpty())
parser.showHelp(EXIT_FAILURE);
bool ok = false;
int rate = temp.toInt(&ok);
if (ok)
preferences.setTemporaryPreference(PREF_EXPORT_MP3_BITRATE, rate);
else
fprintf(stderr, "MP3 bitrate value '%s' not recognized, using default setting from preferences instead.\n", qPrintable(temp));
}
QStringList argv = parser.positionalArguments();
2013-05-13 18:49:17 +02:00
mscoreGlobalShare = getSharePath();
iconPath = externalIcons ? mscoreGlobalShare + QString("icons/") : QString(":/data/icons/");
2013-05-13 18:49:17 +02:00
if (!converterMode && !pluginMode) {
2013-05-13 18:49:17 +02:00
if (!argv.isEmpty()) {
int ok = true;
for (const QString& message : argv) {
2013-05-13 18:49:17 +02:00
QFileInfo fi(message);
if (!app->sendMessage(fi.absoluteFilePath())) {
ok = false;
break;
}
2017-11-27 09:56:41 +01:00
2013-05-13 18:49:17 +02:00
}
if (ok)
return 0;
2012-05-26 14:49:10 +02:00
}
else
2015-01-07 13:47:03 +01:00
if (app->sendMessage(QString("")))
2013-05-13 18:49:17 +02:00
return 0;
2012-05-26 14:49:10 +02:00
}
2017-11-27 09:56:41 +01:00
2013-05-13 18:49:17 +02:00
if (dataPath.isEmpty())
dataPath = QStandardPaths::writableLocation(QStandardPaths::DataLocation);
2014-06-03 15:28:10 +02:00
if (deletePreferences) {
QDir(dataPath).removeRecursively();
QSettings settings;
QFile::remove(settings.fileName() + ".lock"); //forcibly remove lock
QFile::remove(settings.fileName());
settings.clear();
}
2014-06-03 15:28:10 +02:00
2013-05-13 18:49:17 +02:00
// create local plugin directory
// if not already there:
2015-01-07 13:47:03 +01:00
QDir().mkpath(dataPath + "/plugins");
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (MScore::debugMode)
qDebug("global share: <%s>", qPrintable(mscoreGlobalShare));
2012-05-26 14:49:10 +02:00
// set translator before Shortcut are initialized to get translations for all shortcuts
// can not use preferences to retrieve these values as it needs to be initialized after translator is set
2013-05-13 18:49:17 +02:00
if (useFactorySettings)
localeName = "system";
2012-05-26 14:49:10 +02:00
else {
2013-05-13 18:49:17 +02:00
QSettings s;
localeName = s.value(PREF_UI_APP_LANGUAGE, "system").toString();
2012-05-26 14:49:10 +02:00
}
2014-06-03 15:28:10 +02:00
2013-05-13 18:49:17 +02:00
setMscoreLocale(localeName);
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
Shortcut::init();
preferences.init();
2012-05-26 14:49:10 +02:00
QNetworkProxyFactory::setUseSystemConfiguration(true);
MScore::init(); // initialize libmscore
updateExternalValuesFromPreferences();
// initialize current page size from default printer
#ifndef QT_NO_PRINTER
2017-05-02 14:17:31 +02:00
if (!MScore::testMode) {
QPrinter p;
if (p.isValid()) {
2018-04-09 11:51:35 +02:00
// qDebug("set paper size from default printer");
QRectF psf = p.paperRect(QPrinter::Inch);
2018-03-27 15:36:00 +02:00
MScore::defaultStyle().set(Sid::pageWidth, psf.width());
MScore::defaultStyle().set(Sid::pageHeight, psf.height());
MScore::defaultStyle().set(Sid::pagePrintableWidth, psf.width()-20.0/INCH);
}
}
2017-05-02 14:17:31 +02:00
#endif
2012-05-26 14:49:10 +02:00
#ifdef SCRIPT_INTERFACE
2017-06-28 15:18:59 +02:00
if (-1 == qmlRegisterType<QmlPlugin> ("MuseScore", 3, 0, "MuseScore"))
qDebug("qmlRegisterType failed: MuseScore");
#endif
2013-10-30 09:42:08 +01:00
if (MScore::debugMode) {
2015-11-16 14:24:47 +01:00
qDebug("DPI %f", DPI);
2012-05-26 14:49:10 +02:00
2015-11-16 14:24:47 +01:00
QScreen* screen = QGuiApplication::primaryScreen();
2013-10-30 09:42:08 +01:00
qDebug() << "Information for screen:" << screen->name();
qDebug() << " Available geometry:" << screen->availableGeometry().x() << screen->availableGeometry().y() << screen->availableGeometry().width() << "x" << screen->availableGeometry().height();
qDebug() << " Available size:" << screen->availableSize().width() << "x" << screen->availableSize().height();
qDebug() << " Available virtual geometry:" << screen->availableVirtualGeometry().x() << screen->availableVirtualGeometry().y() << screen->availableVirtualGeometry().width() << "x" << screen->availableVirtualGeometry().height();
qDebug() << " Available virtual size:" << screen->availableVirtualSize().width() << "x" << screen->availableVirtualSize().height();
qDebug() << " Depth:" << screen->depth() << "bits";
qDebug() << " Geometry:" << screen->geometry().x() << screen->geometry().y() << screen->geometry().width() << "x" << screen->geometry().height();
qDebug() << " Logical DPI:" << screen->logicalDotsPerInch();
qDebug() << " Logical DPI X:" << screen->logicalDotsPerInchX();
qDebug() << " Logical DPI Y:" << screen->logicalDotsPerInchY();
qDebug() << " Physical DPI:" << screen->physicalDotsPerInch();
qDebug() << " Physical DPI X:" << screen->physicalDotsPerInchX();
qDebug() << " Physical DPI Y:" << screen->physicalDotsPerInchY();
qDebug() << " Physical size:" << screen->physicalSize().width() << "x" << screen->physicalSize().height() << "mm";
qDebug() << " Refresh rate:" << screen->refreshRate() << "Hz";
qDebug() << " Size:" << screen->size().width() << "x" << screen->size().height();
qDebug() << " Virtual geometry:" << screen->virtualGeometry().x() << screen->virtualGeometry().y() << screen->virtualGeometry().width() << "x" << screen->virtualGeometry().height();
qDebug() << " Virtual size:" << screen->virtualSize().width() << "x" << screen->virtualSize().height();
}
2013-10-08 20:03:34 +02:00
2017-03-31 13:03:15 +02:00
if (!MScore::testMode)
MScore::readDefaultStyle(preferences.getString(PREF_SCORE_STYLE_DEFAULTSTYLEFILE));
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
QSplashScreen* sc = 0;
if (!MScore::noGui && preferences.getBool(PREF_UI_APP_STARTUP_SHOWSPLASHSCREEN)) {
2014-12-18 21:02:04 +01:00
QPixmap pm(":/data/splash.png");
2013-05-13 18:49:17 +02:00
sc = new QSplashScreen(pm);
sc->setWindowTitle(QString("MuseScore Startup"));
#ifdef Q_OS_MAC // to have session dialog on top of splashscreen on mac
sc->setWindowFlags(Qt::FramelessWindowHint);
#endif
2013-05-13 18:49:17 +02:00
sc->show();
qApp->processEvents();
2012-05-26 14:49:10 +02:00
}
if (!MScore::noGui)
MuseScore::updateUiStyleAndTheme();
else
noSeq = true;
2013-07-16 18:16:16 +02:00
genIcons();
// Do not create sequencer and audio drivers if run with '-s'
if (!noSeq) {
2013-05-13 18:49:17 +02:00
seq = new Seq();
MScore::seq = seq;
Driver* driver = driverFactory(seq, audioDriver);
synti = synthesizerFactory();
2013-05-13 18:49:17 +02:00
if (driver) {
MScore::sampleRate = driver->sampleRate();
synti->setSampleRate(MScore::sampleRate);
synti->init();
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
seq->setDriver(driver);
}
else {
// Do not delete the sequencer If we can't load driver.
// Allow user to select the working driver later.
MScore::sampleRate = 44100; // Would be changed when user changes driver
synti->setSampleRate(MScore::sampleRate);
synti->init();
2013-05-13 18:49:17 +02:00
}
seq->setMasterSynthesizer(synti);
2013-05-13 18:49:17 +02:00
}
else {
seq = 0;
MScore::seq = 0;
}
2017-05-02 14:17:31 +02:00
//---
2013-05-13 18:49:17 +02:00
//
// avoid font problems by overriding the environment
// fall back to "C" locale
//
2012-05-26 14:49:10 +02:00
2013-10-08 19:25:04 +02:00
//#ifndef Q_OS_WIN
//setenv("LANG", "C", 1);
//#endif
//QLocale::setDefault(QLocale(QLocale::C));
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (MScore::debugMode) {
QStringList sl(QCoreApplication::libraryPaths());
foreach(const QString& s, sl)
qDebug("LibraryPath: <%s>", qPrintable(s));
}
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
// rastral size of font is 20pt = 20/72 inch = 20*DPI/72 dots
// staff has 5 lines = 4 * _spatium
2015-11-16 14:24:47 +01:00
// _spatium = SPATIUM20; // 20.0 / 72.0 * DPI / 4.0;
2012-05-26 14:49:10 +02:00
2014-11-18 16:17:28 +01:00
if (!MScore::noGui) {
2015-02-10 16:15:56 +01:00
#ifndef Q_OS_MAC
qApp->setWindowIcon(*icons[int(Icons::window_ICON)]);
2015-02-10 16:15:56 +01:00
#endif
2014-11-18 16:17:28 +01:00
Workspace::initWorkspace();
}
2013-05-13 18:49:17 +02:00
mscore = new MuseScore();
2014-07-23 17:08:48 +02:00
// create a score for internal use
2017-01-05 11:23:47 +01:00
gscore = new MasterScore();
gscore->setMovements(new Movements());
gscore->setStyle(MScore::baseStyle());
2018-03-27 15:36:00 +02:00
gscore->style().set(Sid::MusicalTextFont, QString("Bravura Text"));
ScoreFont* scoreFont = ScoreFont::fontFactory("Bravura");
gscore->setScoreFont(scoreFont);
2015-11-16 14:24:47 +01:00
gscore->setNoteHeadWidth(scoreFont->width(SymId::noteheadBlack, gscore->spatium()) / SPATIUM20);
2012-05-26 14:49:10 +02:00
2013-05-13 18:49:17 +02:00
if (!noSeq) {
if (!seq->init())
2013-05-13 18:49:17 +02:00
qDebug("sequencer init failed");
}
2013-05-13 18:49:17 +02:00
//read languages list
mscore->readLanguages(mscoreGlobalShare + "locale/languages.xml");
if (!MScore::noGui) {
if (preferences.getBool(PREF_APP_STARTUP_FIRSTSTART)) {
StartupWizard* sw = new StartupWizard;
sw->exec();
preferences.setPreference(PREF_APP_STARTUP_FIRSTSTART, false);
preferences.setPreference(PREF_APP_KEYBOARDLAYOUT, sw->keyboardLayout());
preferences.setPreference(PREF_UI_APP_LANGUAGE, sw->language());
setMscoreLocale(sw->language());
for (auto ws : Workspace::workspaces()) {
if (ws->name().compare(sw->workspace()) == 0) {
mscore->changeWorkspace(ws);
preferences.setPreference(PREF_APP_WORKSPACE, ws->name());
mscore->getPaletteBox()->updateWorkspaces();
2017-08-01 10:52:35 +02:00
}
}
delete sw;
// reinitialize preferences so some default values are calculated based on chosen language
preferences.init();
// store preferences with locale-dependent default values
// so that the values from first start will be used later
preferences.setToDefaultValue(PREF_APP_PATHS_MYSCORES);
preferences.setToDefaultValue(PREF_APP_PATHS_MYSTYLES);
preferences.setToDefaultValue(PREF_APP_PATHS_MYIMAGES);
preferences.setToDefaultValue(PREF_APP_PATHS_MYTEMPLATES);
preferences.setToDefaultValue(PREF_APP_PATHS_MYPLUGINS);
preferences.setToDefaultValue(PREF_APP_PATHS_MYSOUNDFONTS);
preferences.setToDefaultValue(PREF_APP_PATHS_MYEXTENSIONS);
updateExternalValuesFromPreferences();
}
QString keyboardLayout = preferences.getString(PREF_APP_KEYBOARDLAYOUT);
StartupWizard::autoSelectShortcuts(keyboardLayout);
}
2013-05-13 18:49:17 +02:00
QApplication::instance()->installEventFilter(mscore);
2013-05-13 18:49:17 +02:00
mscore->setRevision(revision);
int files = 0;
bool restoredSession = false;
2014-02-28 11:14:43 +01:00
if (MScore::noGui) {
#ifdef Q_OS_MAC
// see issue #28706: Hangup in converter mode with MusicXML source
qApp->processEvents();
#endif
exit(processNonGui(argv) ? 0 : EXIT_FAILURE);
2013-05-13 18:49:17 +02:00
}
else {
mscore->readSettings();
QObject::connect(qApp, SIGNAL(messageReceived(const QString&)),
mscore, SLOT(handleMessage(const QString&)));
2013-07-10 17:00:21 +02:00
static_cast<QtSingleApplication*>(qApp)->setActivationWindow(mscore, false);
2013-06-24 22:07:44 +02:00
// count filenames specified on the command line
// these are the non-empty strings remaining in argv
2013-05-13 18:49:17 +02:00
foreach(const QString& name, argv) {
if (!name.isEmpty())
++files;
}
#ifdef Q_OS_MAC
2013-06-24 22:07:44 +02:00
// app->paths contains files requested to be loaded by OS X
// append these to argv and update file count
foreach(const QString& name, app->paths) {
if (!name.isEmpty()) {
argv << name;
++files;
}
2013-05-13 18:49:17 +02:00
}
2013-06-24 22:07:44 +02:00
#endif
2013-05-13 18:49:17 +02:00
//
// TODO: delete old session backups
//
restoredSession = mscore->restoreSession((preferences.sessionStart() == SessionStart::LAST && (files == 0)));
if (!restoredSession || files)
2013-05-13 18:49:17 +02:00
loadScores(argv);
}
2017-05-02 14:17:31 +02:00
2013-07-17 14:36:34 +02:00
errorMessage = new QErrorMessage(mscore);
mscore->getPluginManager()->readPluginList();
2013-05-13 18:49:17 +02:00
mscore->loadPlugins();
mscore->writeSessionFile(false);
#ifdef Q_OS_MAC
2013-05-13 18:49:17 +02:00
// there's a bug in Qt showing the toolbar unified after switching showFullScreen(), showMaximized(),
// showNormal()...
mscore->setUnifiedTitleAndToolBarOnMac(false);
// don't let macOS add "Show Tab Bar" to "View" Menu
CocoaBridge::setAllowsAutomaticWindowTabbing(false);
2013-05-13 18:49:17 +02:00
#endif
2013-05-13 18:49:17 +02:00
mscore->changeState(mscore->noScore() ? STATE_DISABLED : STATE_NORMAL);
mscore->show();
#ifndef MSCORE_NO_UPDATE_CHECKER
2013-05-13 18:49:17 +02:00
if (mscore->hasToCheckForUpdate())
mscore->checkForUpdate();
#endif
2014-11-14 11:44:48 +01:00
if (mscore->hasToCheckForExtensionsUpdate())
mscore->checkForExtensionsUpdate();
if (!scoresOnCommandline && preferences.getBool(PREF_UI_APP_STARTUP_SHOWSTARTCENTER) && (!restoredSession || mscore->scores().size() == 0)) {
#ifdef Q_OS_MAC
// ugly, but on mac we get an event when a file is open.
// We can't get the event when the startcenter is shown.
// So we let the event loop run a bit before showing the start center.
2015-02-23 18:11:14 +01:00
QTimer *timer = new QTimer();
timer->setSingleShot(true);
QObject::connect(timer, &QTimer::timeout, [=]() {
if (!scoresOnCommandline) {
getAction("startcenter")->setChecked(true);
mscore->showStartcenter(true);
}
2015-02-23 18:11:14 +01:00
timer->deleteLater();
} );
timer->start(500);
#else
2014-11-10 12:30:53 +01:00
getAction("startcenter")->setChecked(true);
2014-11-10 10:05:25 +01:00
mscore->showStartcenter(true);
#endif
2014-11-10 12:30:53 +01:00
}
if (sc) {
sc->close();
qApp->processEvents();
}
mscore->showPlayPanel(preferences.getBool(PREF_UI_APP_STARTUP_SHOWPLAYPANEL));
2015-09-08 17:14:02 +02:00
QSettings settings;
if (settings.value("synthControlVisible", false).toBool())
mscore->showSynthControl(true);
if (settings.value("mixerVisible", false).toBool())
mscore->showMixer(true);
2013-05-13 18:49:17 +02:00
return qApp->exec();
}