MuseScore/libmscore/instrchange.cpp
2020-05-29 21:15:25 +02:00

247 lines
8.4 KiB
C++

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2008-2011 Werner Schweer
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2
// as published by the Free Software Foundation and appearing in
// the file LICENCE.GPL
//=============================================================================
#include "instrchange.h"
#include "score.h"
#include "segment.h"
#include "staff.h"
#include "part.h"
#include "undo.h"
#include "mscore.h"
#include "xml.h"
#include "measure.h"
#include "system.h"
#include "chord.h"
#include "keysig.h"
namespace Ms {
//---------------------------------------------------------
// instrumentChangeStyle
//---------------------------------------------------------
static const ElementStyle instrumentChangeStyle {
{ Sid::instrumentChangePlacement, Pid::PLACEMENT },
{ Sid::instrumentChangeMinDistance, Pid::MIN_DISTANCE },
};
//---------------------------------------------------------
// InstrumentChange
//---------------------------------------------------------
InstrumentChange::InstrumentChange(Score* s)
: TextBase(s, Tid::INSTRUMENT_CHANGE, ElementFlag::MOVABLE | ElementFlag::ON_STAFF)
{
initElementStyle(&instrumentChangeStyle);
_instrument = new Instrument();
}
InstrumentChange::InstrumentChange(const Instrument& i, Score* s)
: TextBase(s, Tid::INSTRUMENT_CHANGE, ElementFlag::MOVABLE | ElementFlag::ON_STAFF)
{
initElementStyle(&instrumentChangeStyle);
_instrument = new Instrument(i);
}
InstrumentChange::InstrumentChange(const InstrumentChange& is)
: TextBase(is)
{
_instrument = new Instrument(*is._instrument);
_init = is._init;
}
InstrumentChange::~InstrumentChange()
{
delete _instrument;
}
void InstrumentChange::setInstrument(const Instrument& i)
{
*_instrument = i;
//delete _instrument;
//_instrument = new Instrument(i);
}
void InstrumentChange::setupInstrument(const Instrument* instrument)
{
if (_init) {
Fraction tickStart = segment()->tick();
Part* part = staff()->part();
Interval oldV = part->instrument(tickStart)->transpose();
// change the clef for each staff
for (int i = 0; i < part->nstaves(); i++) {
if (part->instrument(tickStart)->clefType(i) != instrument->clefType(i)) {
ClefType clefType
= score()->styleB(Sid::concertPitch) ? instrument->clefType(i)._concertClef : instrument->clefType(i)
._transposingClef;
// If instrument change is at the start of a measure, use the measure as the element, as this will place the instrument change before the barline.
Element* element = rtick().isZero() ? toElement(findMeasure()) : toElement(this);
score()->undoChangeClef(part->staff(i), element, clefType, true);
}
}
// Change key signature if necessary
if (instrument->transpose() != oldV) {
for (int i = 0; i < part->nstaves(); i++) {
if (!part->staff(i)->keySigEvent(tickStart).isAtonal()) {
KeySigEvent ks;
ks.setForInstrumentChange(true);
Key key = part->staff(i)->key(tickStart);
if (!score()->styleB(Sid::concertPitch)) {
key = transposeKey(key, oldV);
}
ks.setKey(key);
score()->undoChangeKeySig(part->staff(i), tickStart, ks);
}
}
}
// change instrument in all linked scores
for (ScoreElement* se : linkList()) {
InstrumentChange* lic = static_cast<InstrumentChange*>(se);
Instrument* newInstrument = new Instrument(*instrument);
lic->score()->undo(new ChangeInstrument(lic, newInstrument));
}
// transpose for current score only
// this automatically propagates to linked scores
if (part->instrument(tickStart)->transpose() != oldV) {
auto i = part->instruments()->upper_bound(tickStart.ticks()); // find(), ++i
Fraction tickEnd;
if (i == part->instruments()->end()) {
tickEnd = Fraction(-1, 1);
} else {
tickEnd = Fraction::fromTicks(i->first);
}
score()->transpositionChanged(part, oldV, tickStart, tickEnd);
}
const QString newInstrChangeText = tr("To %1").arg(instrument->trackName());
undoChangeProperty(Pid::TEXT, TextBase::plainToXmlText(newInstrChangeText));
}
}
//---------------------------------------------------------
// keySigs
//---------------------------------------------------------
std::vector<KeySig*> InstrumentChange::keySigs() const
{
std::vector<KeySig*> keysigs;
Segment* seg = segment()->prev1(SegmentType::KeySig);
if (seg) {
int startVoice = part()->staff(0)->idx() * VOICES;
int endVoice = part()->staff(part()->nstaves() - 1)->idx() * VOICES;
Fraction t = tick();
for (int i = startVoice; i <= endVoice; i += VOICES) {
KeySig* ks = toKeySig(seg->element(i));
if (ks && ks->forInstrumentChange() && ks->tick() == t) {
keysigs.push_back(ks);
}
}
}
return keysigs;
}
//---------------------------------------------------------
// clefs
//---------------------------------------------------------
std::vector<Clef*> InstrumentChange::clefs() const
{
std::vector<Clef*> clefs;
Segment* seg = segment()->prev1(SegmentType::Clef);
if (seg) {
int startVoice = part()->staff(0)->idx() * VOICES;
int endVoice = part()->staff(part()->nstaves() - 1)->idx() * VOICES;
Fraction t = tick();
for (int i = startVoice; i <= endVoice; i += VOICES) {
Clef* clef = toClef(seg->element(i));
if (clef && clef->forInstrumentChange() && clef->tick() == t) {
clefs.push_back(clef);
}
}
}
return clefs;
}
//---------------------------------------------------------
// write
//---------------------------------------------------------
void InstrumentChange::write(XmlWriter& xml) const
{
xml.stag(this);
_instrument->write(xml, part());
if (_init) {
xml.tag("init", _init);
}
TextBase::writeProperties(xml);
xml.etag();
}
//---------------------------------------------------------
// read
//---------------------------------------------------------
void InstrumentChange::read(XmlReader& e)
{
while (e.readNextStartElement()) {
const QStringRef& tag(e.name());
if (tag == "Instrument") {
_instrument->read(e, part());
} else if (tag == "init") {
_init = e.readBool();
} else if (!TextBase::readProperties(e)) {
e.unknown();
}
}
if (score()->mscVersion() < 206) {
// previous versions did not honor transposition of instrument change
// except in ways that it should not have
// notes entered before the instrument change was added would not be altered,
// so original transposition remained in effect
// notes added afterwards would be transposed by both intervals, resulting in tpc corruption
// here we set the instrument change to inherit the staff transposition to emulate previous versions
// in Note::read(), we attempt to fix the tpc corruption
// There is also code in read206 to try to deal with this, but it is out of date and therefore disabled
// What this means is, scores created in 2.1 or later should be fine, scores created in 2.0 maybe not so much
Interval v = staff() ? staff()->part()->instrument()->transpose() : 0;
_instrument->setTranspose(v);
}
}
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
QVariant InstrumentChange::propertyDefault(Pid propertyId) const
{
switch (propertyId) {
case Pid::SUB_STYLE:
return int(Tid::INSTRUMENT_CHANGE);
default:
return TextBase::propertyDefault(propertyId);
}
}
//---------------------------------------------------------
// layout
//---------------------------------------------------------
void InstrumentChange::layout()
{
TextBase::layout();
autoplaceSegmentElement();
}
}