MuseScore/libmscore/staff.cpp

1491 lines
48 KiB
C++

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2002-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 "mscore.h"
#include "staff.h"
#include "part.h"
#include "clef.h"
#include "xml.h"
#include "score.h"
#include "bracket.h"
#include "keysig.h"
#include "segment.h"
#include "style.h"
#include "measure.h"
#include "stringdata.h"
#include "stafftype.h"
#include "undo.h"
#include "cleflist.h"
#include "timesig.h"
#include "instrtemplate.h"
#include "barline.h"
#include "ottava.h"
#include "harmony.h"
#include "bracketItem.h"
// #define DEBUG_CLEFS
#ifdef DEBUG_CLEFS
#define DUMP_CLEFS(s) dumpClefs(s)
#else
#define DUMP_CLEFS(s)
#endif
namespace Ms {
//---------------------------------------------------------
// Staff
//---------------------------------------------------------
Staff::Staff(Score* score)
: ScoreElement(score)
{
initFromStaffType(0);
}
//---------------------------------------------------------
// idx
//---------------------------------------------------------
int Staff::idx() const
{
return score()->staves().indexOf((Staff*)this, 0);
}
//---------------------------------------------------------
// fillBrackets
// make sure index idx is valid
//---------------------------------------------------------
void Staff::fillBrackets(int idx)
{
for (int i = _brackets.size(); i <= idx; ++i) {
BracketItem* bi = new BracketItem(score());
bi->setStaff(this);
bi->setColumn(i);
_brackets.append(bi);
}
}
//---------------------------------------------------------
// cleanBrackets
// remove NO_BRACKET entries from the end of list
//---------------------------------------------------------
void Staff::cleanBrackets()
{
while (!_brackets.empty() && (_brackets.last()->bracketType() == BracketType::NO_BRACKET)) {
BracketItem* bi = _brackets.takeLast();
delete bi;
}
}
//---------------------------------------------------------
// bracket
//---------------------------------------------------------
BracketType Staff::bracketType(int idx) const
{
if (idx < _brackets.size())
return _brackets[idx]->bracketType();
return BracketType::NO_BRACKET;
}
//---------------------------------------------------------
// bracketSpan
//---------------------------------------------------------
int Staff::bracketSpan(int idx) const
{
if (idx < _brackets.size())
return _brackets[idx]->bracketSpan();
return 0;
}
//---------------------------------------------------------
// setBracket
//---------------------------------------------------------
void Staff::setBracketType(int idx, BracketType val)
{
fillBrackets(idx);
_brackets[idx]->setBracketType(val);
cleanBrackets();
}
//---------------------------------------------------------
// swapBracket
//---------------------------------------------------------
void Staff::swapBracket(int oldIdx, int newIdx)
{
int idx = qMax(oldIdx, newIdx);
fillBrackets(idx);
_brackets[oldIdx]->setColumn(newIdx);
_brackets[newIdx]->setColumn(oldIdx);
_brackets.swap(oldIdx, newIdx);
cleanBrackets();
}
//---------------------------------------------------------
// changeBracketColumn
//---------------------------------------------------------
void Staff::changeBracketColumn(int oldColumn, int newColumn)
{
int idx = qMax(oldColumn, newColumn);
fillBrackets(idx);
int step = newColumn > oldColumn ? 1 : -1;
for (int i = oldColumn; i != newColumn; i += step) {
int oldIdx = i;
int newIdx = i + step;
_brackets[oldIdx]->setColumn(newIdx);
_brackets[newIdx]->setColumn(oldIdx);
_brackets.swap(oldIdx, newIdx);
}
cleanBrackets();
}
//---------------------------------------------------------
// setBracketSpan
//---------------------------------------------------------
void Staff::setBracketSpan(int idx, int val)
{
Q_ASSERT(idx >= 0);
Q_ASSERT(val >= 0);
fillBrackets(idx);
_brackets[idx]->setBracketSpan(val);
}
//---------------------------------------------------------
// addBracket
//---------------------------------------------------------
void Staff::addBracket(BracketItem* b)
{
b->setStaff(this);
if (!_brackets.empty() && _brackets[0]->bracketType() == BracketType::NO_BRACKET)
_brackets[0] = b;
else {
//
// create new bracket level
//
for (Staff* s : score()->staves()) {
if (s == this)
s->_brackets.append(b);
else {
BracketItem* bi = new BracketItem(score());
bi->setStaff(this);
s->_brackets.append(bi);
}
}
}
}
//---------------------------------------------------------
// innerBracket
// Return type inner bracket.
// The bracket type determines the staff distance.
//---------------------------------------------------------
BracketType Staff::innerBracket() const
{
int staffIdx = idx();
BracketType t = BracketType::NO_BRACKET;
int level = 1000;
for (int i = 0; i < score()->nstaves(); ++i) {
Staff* staff = score()->staff(i);
for (int k = 0; k < staff->brackets().size(); ++k) {
const BracketItem* bi = staff->brackets().at(k);
if (bi->bracketType() != BracketType::NO_BRACKET) {
if (i < staffIdx && ((i + bi->bracketSpan()) > staffIdx) && k < level) {
t = bi->bracketType();
level = k;
break;
}
}
}
}
return t;
}
//---------------------------------------------------------
// cleanupBrackets
//---------------------------------------------------------
void Staff::cleanupBrackets()
{
int index = idx();
int n = score()->nstaves();
for (int i = 0; i < _brackets.size(); ++i) {
if (_brackets[i]->bracketType() == BracketType::NO_BRACKET)
continue;
int span = _brackets[i]->bracketSpan();
if (span > (n - index)) {
span = n - index;
_brackets[i]->setBracketSpan(span);
}
}
for (int i = 0; i < _brackets.size(); ++i) {
if (_brackets[i]->bracketType() == BracketType::NO_BRACKET)
continue;
int span = _brackets[i]->bracketSpan();
if (span <= 1) {
_brackets[i] = new BracketItem(score());
_brackets[i]->setStaff(this);
}
else {
// delete all other brackets with same span
for (int k = i + 1; k < _brackets.size(); ++k) {
if (span == _brackets[k]->bracketSpan()) {
_brackets[k] = new BracketItem(score());
_brackets[k]->setStaff(this);
}
}
}
}
}
//---------------------------------------------------------
// bracketLevels
//---------------------------------------------------------
int Staff::bracketLevels() const
{
int columns = 0;
for (auto bi : _brackets)
columns = qMax(columns, bi->column());
return columns;
}
//---------------------------------------------------------
// partName
//---------------------------------------------------------
QString Staff::partName() const
{
return _part->partName();
}
//---------------------------------------------------------
// Staff::clefType
//---------------------------------------------------------
ClefTypeList Staff::clefType(const Fraction& tick) const
{
ClefTypeList ct = clefs.clef(tick.ticks());
if (ct._concertClef == ClefType::INVALID) {
switch (staffType(tick)->group()) {
case StaffGroup::TAB:
{
ClefType sct = ClefType(score()->styleI(Sid::tabClef));
ct = staffType(tick)->lines() <= 4 ? ClefTypeList(sct == ClefType::TAB ? ClefType::TAB4 : ClefType::TAB4_SERIF) : ClefTypeList(sct == ClefType::TAB ? ClefType::TAB : ClefType::TAB_SERIF);
}
break;
case StaffGroup::STANDARD:
ct = defaultClefType();
break;
case StaffGroup::PERCUSSION:
ct = ClefTypeList(ClefType::PERC);
break;
}
}
return ct;
}
//---------------------------------------------------------
// Staff::clef
//---------------------------------------------------------
ClefType Staff::clef(const Fraction& tick) const
{
ClefTypeList c = clefType(tick);
return score()->styleB(Sid::concertPitch) ? c._concertClef : c._transposingClef;
}
//---------------------------------------------------------
// Staff::nextClefTick
//
// return the tick of next clef after tick
// return last tick of score if not found
//---------------------------------------------------------
Fraction Staff::nextClefTick(const Fraction& tick) const
{
Fraction t = Fraction::fromTicks(clefs.nextClefTick(tick.ticks()));
return t != Fraction(-1,1) ? t : score()->endTick();
}
//---------------------------------------------------------
// Staff::currentClefTick
//
// return the tick position of the clef currently
// in effect at tick
// return 0, if no such clef
//---------------------------------------------------------
Fraction Staff::currentClefTick(const Fraction& tick) const
{
return Fraction::fromTicks(clefs.currentClefTick(tick.ticks()));
}
#ifndef NDEBUG
//---------------------------------------------------------
// dumpClef
//---------------------------------------------------------
void Staff::dumpClefs(const char* title) const
{
qDebug("(%zd): %s", clefs.size(), title);
for (auto& i : clefs) {
qDebug(" %d: %d %d", i.first, int(i.second._concertClef), int(i.second._transposingClef));
}
}
//---------------------------------------------------------
// dumpKeys
//---------------------------------------------------------
void Staff::dumpKeys(const char* title) const
{
qDebug("(%zd): %s", _keys.size(), title);
for (auto& i : _keys) {
qDebug(" %d: %d", i.first, int(i.second.key()));
}
}
//---------------------------------------------------------
// dumpTimeSigs
//---------------------------------------------------------
void Staff::dumpTimeSigs(const char* title) const
{
qDebug("size (%zd) staffIdx %d: %s", timesigs.size(), idx(), title);
for (auto& i : timesigs) {
qDebug(" %d: %d/%d", i.first, i.second->sig().numerator(), i.second->sig().denominator());
}
}
#endif
//---------------------------------------------------------
// setClef
//---------------------------------------------------------
void Staff::setClef(Clef* clef)
{
if (clef->generated())
return;
Fraction tick = clef->segment()->tick();
for (Segment* s = clef->segment()->next1(); s && s->tick() == tick; s = s->next1()) {
if ((s->segmentType() == SegmentType::Clef || s->segmentType() == SegmentType::HeaderClef)
&& s->element(clef->track())
&& !s->element(clef->track())->generated()) {
// adding this clef has no effect on the clefs list
return;
}
}
clefs.setClef(clef->segment()->tick().ticks(), clef->clefTypeList());
DUMP_CLEFS("setClef");
}
//---------------------------------------------------------
// removeClef
//---------------------------------------------------------
void Staff::removeClef(const Clef* clef)
{
if (clef->generated())
return;
Fraction tick = clef->segment()->tick();
for (Segment* s = clef->segment()->next1(); s && s->tick() == tick; s = s->next1()) {
if ((s->segmentType() == SegmentType::Clef || s->segmentType() == SegmentType::HeaderClef)
&& s->element(clef->track())
&& !s->element(clef->track())->generated()) {
// removal of this clef has no effect on the clefs list
return;
}
}
clefs.erase(clef->segment()->tick().ticks());
for (Segment* s = clef->segment()->prev1(); s && s->tick() == tick; s = s->prev1()) {
if ((s->segmentType() == SegmentType::Clef || s->segmentType() == SegmentType::HeaderClef)
&& s->element(clef->track())
&& !s->element(clef->track())->generated()) {
// a previous clef at the same tick position gets valid
clefs.setClef(tick.ticks(), toClef(s->element(clef->track()))->clefTypeList());
break;
}
}
DUMP_CLEFS("removeClef");
}
//---------------------------------------------------------
// timeStretch
//---------------------------------------------------------
Fraction Staff::timeStretch(const Fraction& tick) const
{
TimeSig* timesig = timeSig(tick);
return timesig ? timesig->stretch() : Fraction(1,1);
}
//---------------------------------------------------------
// timeSig
// lookup time signature before or at tick
//---------------------------------------------------------
TimeSig* Staff::timeSig(const Fraction& tick) const
{
auto i = timesigs.upper_bound(tick.ticks());
if (i != timesigs.begin())
--i;
if (i == timesigs.end())
return 0;
else if (tick < Fraction::fromTicks(i->first))
return 0;
return i->second;
}
//---------------------------------------------------------
// nextTimeSig
// lookup time signature at tick or after
//---------------------------------------------------------
TimeSig* Staff::nextTimeSig(const Fraction& tick) const
{
auto i = timesigs.lower_bound(tick.ticks());
return (i == timesigs.end()) ? 0 : i->second;
}
//---------------------------------------------------------
// currentTimeSigTick
//
// return the tick position of the time sig currently
// in effect at tick
//---------------------------------------------------------
Fraction Staff::currentTimeSigTick(const Fraction& tick) const
{
if (timesigs.empty())
return Fraction(0, 1);
auto i = timesigs.upper_bound(tick.ticks());
if (i == timesigs.begin())
return Fraction(0, 1);
--i;
return Fraction::fromTicks(i->first);
}
//---------------------------------------------------------
// group
//---------------------------------------------------------
const Groups& Staff::group(const Fraction& tick) const
{
TimeSig* ts = timeSig(tick);
if (ts) {
if (!ts->groups().empty())
return ts->groups();
return Groups::endings(ts->sig());
}
Measure* m = score()->tick2measure(tick);
return Groups::endings(m ? m->timesig() : Fraction(4,4));
}
//---------------------------------------------------------
// addTimeSig
//---------------------------------------------------------
void Staff::addTimeSig(TimeSig* timesig)
{
if (timesig->segment()->segmentType() == SegmentType::TimeSig)
timesigs[timesig->segment()->tick().ticks()] = timesig;
// dumpTimeSigs("after addTimeSig");
}
//---------------------------------------------------------
// removeTimeSig
//---------------------------------------------------------
void Staff::removeTimeSig(TimeSig* timesig)
{
if (timesig->segment()->segmentType() == SegmentType::TimeSig)
timesigs.erase(timesig->segment()->tick().ticks());
// dumpTimeSigs("after removeTimeSig");
}
//---------------------------------------------------------
// clearTimeSig
//---------------------------------------------------------
void Staff::clearTimeSig()
{
timesigs.clear();
}
//---------------------------------------------------------
// Staff::keySigEvent
//
// locates the key sig currently in effect at tick
//---------------------------------------------------------
KeySigEvent Staff::keySigEvent(const Fraction& tick) const
{
return _keys.key(tick.ticks());
}
//---------------------------------------------------------
// setKey
//---------------------------------------------------------
void Staff::setKey(const Fraction& tick, KeySigEvent k)
{
_keys.setKey(tick.ticks(), k);
}
//---------------------------------------------------------
// removeKey
//---------------------------------------------------------
void Staff::removeKey(const Fraction& tick)
{
_keys.erase(tick.ticks());
}
//---------------------------------------------------------
// prevkey
//---------------------------------------------------------
KeySigEvent Staff::prevKey(const Fraction& tick) const
{
return _keys.prevKey(tick.ticks());
}
//---------------------------------------------------------
// Staff::nextKeyTick
//
// return the tick at which the key sig after tick is located
// return 0, if no such a key sig
//---------------------------------------------------------
Fraction Staff::nextKeyTick(const Fraction& tick) const
{
Fraction t = Fraction::fromTicks(_keys.nextKeyTick(tick.ticks()));
return t != Fraction(-1,1) ? t : score()->endTick();
}
//---------------------------------------------------------
// Staff::currentKeyTick
//
// return the tick position of the key currently
// in effect at tick
// return 0, if no such a key sig
//---------------------------------------------------------
Fraction Staff::currentKeyTick(const Fraction& tick) const
{
return Fraction::fromTicks(_keys.currentKeyTick(tick.ticks()));
}
//---------------------------------------------------------
// write
//---------------------------------------------------------
void Staff::write(XmlWriter& xml) const
{
int idx = this->idx();
xml.stag(this, QString("id=\"%1\"").arg(idx + 1));
if (links()) {
Score* s = masterScore();
for (auto le : *links()) {
Staff* staff = toStaff(le);
if ((staff->score() == s) && (staff != this))
xml.tag("linkedTo", staff->idx() + 1);
}
}
// for copy/paste we need to know the actual transposition
if (xml.clipboardmode()) {
Interval v = part()->instrument()->transpose(); // TODO: tick?
if (v.diatonic)
xml.tag("transposeDiatonic", v.diatonic);
if (v.chromatic)
xml.tag("transposeChromatic", v.chromatic);
}
staffType(Fraction(0,1))->write(xml);
ClefTypeList ct = _defaultClefType;
if (ct._concertClef == ct._transposingClef) {
if (ct._concertClef != ClefType::G)
xml.tag("defaultClef", ClefInfo::tag(ct._concertClef));
}
else {
xml.tag("defaultConcertClef", ClefInfo::tag(ct._concertClef));
xml.tag("defaultTransposingClef", ClefInfo::tag(ct._transposingClef));
}
if (invisible())
xml.tag("invisible", invisible());
if (hideWhenEmpty() != HideMode::AUTO)
xml.tag("hideWhenEmpty", int(hideWhenEmpty()));
if (cutaway())
xml.tag("cutaway", cutaway());
if (showIfEmpty())
xml.tag("showIfSystemEmpty", showIfEmpty());
if (_hideSystemBarLine)
xml.tag("hideSystemBarLine", _hideSystemBarLine);
for (const BracketItem* i : _brackets) {
BracketType a = i->bracketType();
int b = i->bracketSpan();
int c = i->column();
if (a != BracketType::NO_BRACKET || b > 0)
xml.tagE(QString("bracket type=\"%1\" span=\"%2\" col=\"%3\"").arg((int)(a)).arg(b).arg(c));
}
writeProperty(xml, Pid::STAFF_BARLINE_SPAN);
writeProperty(xml, Pid::STAFF_BARLINE_SPAN_FROM);
writeProperty(xml, Pid::STAFF_BARLINE_SPAN_TO);
writeProperty(xml, Pid::STAFF_USERDIST);
writeProperty(xml, Pid::COLOR);
writeProperty(xml, Pid::PLAYBACK_VOICE1);
writeProperty(xml, Pid::PLAYBACK_VOICE2);
writeProperty(xml, Pid::PLAYBACK_VOICE3);
writeProperty(xml, Pid::PLAYBACK_VOICE4);
xml.etag();
}
//---------------------------------------------------------
// read
//---------------------------------------------------------
void Staff::read(XmlReader& e)
{
while (e.readNextStartElement()) {
if (!readProperties(e))
e.unknown();
}
}
//---------------------------------------------------------
// readProperties
//---------------------------------------------------------
bool Staff::readProperties(XmlReader& e)
{
const QStringRef& tag(e.name());
if (tag == "StaffType") {
StaffType st;
st.read(e);
setStaffType(Fraction(0,1), st);
}
else if (tag == "defaultClef") { // sets both default transposing and concert clef
QString val(e.readElementText());
ClefType ct = Clef::clefType(val);
setDefaultClefType(ClefTypeList(ct, ct));
}
else if (tag == "defaultConcertClef") {
QString val(e.readElementText());
setDefaultClefType(ClefTypeList(Clef::clefType(val), defaultClefType()._transposingClef));
}
else if (tag == "defaultTransposingClef") {
QString val(e.readElementText());
setDefaultClefType(ClefTypeList(defaultClefType()._concertClef, Clef::clefType(val)));
}
else if (tag == "small") // obsolete
setSmall(Fraction(0,1), e.readInt());
else if (tag == "invisible")
setInvisible(e.readInt());
else if (tag == "hideWhenEmpty")
setHideWhenEmpty(HideMode(e.readInt()));
else if (tag == "cutaway")
setCutaway(e.readInt());
else if (tag == "showIfSystemEmpty")
setShowIfEmpty(e.readInt());
else if (tag == "hideSystemBarLine")
_hideSystemBarLine = e.readInt();
else if (tag == "keylist")
_keys.read(e, score());
else if (tag == "bracket") {
int col = e.intAttribute("col", -1);
if (col == -1)
col = _brackets.size();
setBracketType(col, BracketType(e.intAttribute("type", -1)));
setBracketSpan(col, e.intAttribute("span", 0));
e.readNext();
}
else if (tag == "barLineSpan")
_barLineSpan = e.readInt();
else if (tag == "barLineSpanFrom")
_barLineFrom = e.readInt();
else if (tag == "barLineSpanTo")
_barLineTo = e.readInt();
else if (tag == "distOffset")
_userDist = e.readDouble() * score()->spatium();
else if (tag == "mag")
/*_userMag =*/ e.readDouble(0.1, 10.0);
else if (tag == "linkedTo") {
int v = e.readInt() - 1;
Staff* st = masterScore()->staff(v);
if (_links) {
qDebug("Staff::readProperties: multiple <linkedTo> tags");
if (!st || isLinked(st)) // maybe we don't need actually to relink...
return true;
// not using unlink() here as it may delete _links
// a pointer to which is stored also in XmlReader.
_links->removeOne(this);
_links = nullptr;
}
if (st && st != this)
linkTo(st);
else if (!score()->isMaster() && !st) {
// if it is a master score it is OK not to find
// a staff which is going after the current one.
qDebug("staff %d not found in parent", v);
}
}
else if (tag == "color")
_color = e.readColor();
else if (tag == "transposeDiatonic")
e.setTransposeDiatonic(e.readInt());
else if (tag == "transposeChromatic")
e.setTransposeChromatic(e.readInt());
else if (tag == "playbackVoice1")
setPlaybackVoice(0, e.readInt());
else if (tag == "playbackVoice2")
setPlaybackVoice(1, e.readInt());
else if (tag == "playbackVoice3")
setPlaybackVoice(2, e.readInt());
else if (tag == "playbackVoice4")
setPlaybackVoice(3, e.readInt());
else
return false;
return true;
}
//---------------------------------------------------------
// height
//---------------------------------------------------------
qreal Staff::height() const
{
Fraction tick = Fraction(0,1); // TODO
// return (lines(tick) == 1 ? 2 : lines(tick)-1) * spatium(tick) * staffType(tick)->lineDistance().val();
return (lines(tick)-1) * spatium(tick) * staffType(tick)->lineDistance().val();
}
//---------------------------------------------------------
// spatium
//---------------------------------------------------------
qreal Staff::spatium(const Fraction& tick) const
{
return score()->spatium() * mag(tick);
}
//---------------------------------------------------------
// mag
//---------------------------------------------------------
qreal Staff::mag(const Fraction& tick) const
{
return (small(tick) ? score()->styleD(Sid::smallStaffMag) : 1.0) * userMag(tick);
}
//---------------------------------------------------------
// userMag
//---------------------------------------------------------
qreal Staff::userMag(const Fraction& tick) const
{
return staffType(tick)->userMag();
}
//---------------------------------------------------------
// setUserMag
//---------------------------------------------------------
void Staff::setUserMag(const Fraction& tick, qreal m)
{
staffType(tick)->setUserMag(m);
}
//---------------------------------------------------------
// small
//---------------------------------------------------------
bool Staff::small(const Fraction& tick) const
{
return staffType(tick)->small();
}
//---------------------------------------------------------
// setSmall
//---------------------------------------------------------
void Staff::setSmall(const Fraction& tick, bool val)
{
staffType(tick)->setSmall(val);
}
//---------------------------------------------------------
// swing
//---------------------------------------------------------
SwingParameters Staff::swing(const Fraction& tick) const
{
SwingParameters sp;
int swingUnit = 0;
QString unit = score()->styleSt(Sid::swingUnit);
int swingRatio = score()->styleI(Sid::swingRatio);
if (unit == TDuration(TDuration::DurationType::V_EIGHTH).name()) {
swingUnit = MScore::division / 2;
}
else if (unit == TDuration(TDuration::DurationType::V_16TH).name())
swingUnit = MScore::division / 4;
else if (unit == TDuration(TDuration::DurationType::V_ZERO).name())
swingUnit = 0;
sp.swingRatio = swingRatio;
sp.swingUnit = swingUnit;
if (_swingList.empty())
return sp;
QMap<int, SwingParameters>::const_iterator i = _swingList.upperBound(tick.ticks());
if (i == _swingList.begin())
return sp;
--i;
return i.value();
}
//---------------------------------------------------------
// capo
//---------------------------------------------------------
int Staff::capo(const Fraction& tick) const
{
if (_capoList.empty())
return 0;
QMap<int, int>::const_iterator i = _capoList.upperBound(tick.ticks());
if (i == _capoList.begin())
return 0;
--i;
return i.value();
}
//---------------------------------------------------------
// channel
//---------------------------------------------------------
int Staff::channel(const Fraction& tick, int voice) const
{
if (_channelList[voice].empty())
return 0;
QMap<int, int>::const_iterator i = _channelList[voice].upperBound(tick.ticks());
if (i == _channelList[voice].begin())
return 0;
--i;
return i.value();
}
//---------------------------------------------------------
// middleLine
// returns logical line number of middle staff line
//---------------------------------------------------------
int Staff::middleLine(const Fraction& tick) const
{
return lines(tick) - 1;
}
//---------------------------------------------------------
// bottomLine
// returns logical line number of bottom staff line
//---------------------------------------------------------
int Staff::bottomLine(const Fraction& tick) const
{
return (lines(tick) - 1) * 2;
}
//---------------------------------------------------------
// slashStyle
//---------------------------------------------------------
bool Staff::slashStyle(const Fraction& tick) const
{
return staffType(tick)->slashStyle();
}
//---------------------------------------------------------
// setSlashStyle
//---------------------------------------------------------
void Staff::setSlashStyle(const Fraction& tick, bool val)
{
staffType(tick)->setSlashStyle(val);
}
//---------------------------------------------------------
// primaryStaff
/// if there are linked staves, the primary staff is
/// the one who is played back and it's not a tab staff
/// because we don't have enough information to play
/// e.g ornaments. NOTE: it's not necessarily the top staff!
//---------------------------------------------------------
bool Staff::primaryStaff() const
{
if (!_links)
return true;
QList<Staff*> s;
QList<Staff*> ss;
for (auto e : *_links) {
Staff* staff = toStaff(e);
if (staff->score() == score()) {
s.append(staff);
if (!staff->isTabStaff(Fraction(0,1)))
ss.append(staff);
}
}
if (s.size() == 1) // the linked staves are in different scores
return s.front() == this;
else // return a non tab linked staff in this score
return ss.front() == this;
}
//---------------------------------------------------------
// staffType
//---------------------------------------------------------
const StaffType* Staff::staffType(const Fraction& tick) const
{
return &_staffTypeList.staffType(tick);
}
const StaffType* Staff::constStaffType(const Fraction& tick) const
{
return &_staffTypeList.staffType(tick);
}
StaffType* Staff::staffType(const Fraction& tick)
{
return &_staffTypeList.staffType(tick);
}
//---------------------------------------------------------
// staffTypeListChanged
// Signal that the staffTypeList has changed at
// position tick. Update layout range.
//---------------------------------------------------------
void Staff::staffTypeListChanged(const Fraction& tick)
{
score()->setLayout(tick);
auto i = _staffTypeList.find(tick.ticks());
if (i == _staffTypeList.end()) {
score()->setLayoutAll();
}
else {
++i;
if (i != _staffTypeList.end())
score()->setLayout(Fraction::fromTicks(i->first));
else
score()->setLayout(score()->lastMeasure()->endTick());
}
}
//---------------------------------------------------------
// setStaffType
//---------------------------------------------------------
StaffType* Staff::setStaffType(const Fraction& tick, const StaffType& nst)
{
return _staffTypeList.setStaffType(tick, nst);
}
//---------------------------------------------------------
// setStaffType
//---------------------------------------------------------
void Staff::removeStaffType(const Fraction& tick)
{
auto i = _staffTypeList.find(tick.ticks());
if (i == _staffTypeList.end())
return;
qreal old = spatium(tick);
_staffTypeList.erase(i);
localSpatiumChanged(old, spatium(tick), tick);
staffTypeListChanged(tick);
}
//---------------------------------------------------------
// init
//---------------------------------------------------------
void Staff::init(const InstrumentTemplate* t, const StaffType* staffType, int cidx)
{
// set staff-type-independent parameters
const StaffType* pst = staffType ? staffType : t->staffTypePreset;
if (!pst)
pst = StaffType::getDefaultPreset(t->staffGroup);
setStaffType(Fraction(0,1), *pst);
if (cidx >= MAX_STAVES) {
setSmall(Fraction(0,1), false);
}
else {
setSmall(Fraction(0,1), t->smallStaff[cidx]);
setBracketType(0, t->bracket[cidx]);
setBracketSpan(0, t->bracketSpan[cidx]);
setBarLineSpan(t->barlineSpan[cidx]);
}
setDefaultClefType(t->clefType(cidx));
}
//---------------------------------------------------------
// init
//---------------------------------------------------------
void Staff::init(const Staff* s)
{
_staffTypeList = s->_staffTypeList;
setDefaultClefType(s->defaultClefType());
for (BracketItem* i : s->_brackets){
BracketItem* ni = new BracketItem(*i);
ni->setScore(score());
ni->setStaff(this);
_brackets.push_back(ni);
}
_barLineSpan = s->_barLineSpan;
_barLineFrom = s->_barLineFrom;
_barLineTo = s->_barLineTo;
_invisible = s->_invisible;
_hideWhenEmpty = s->_hideWhenEmpty;
_cutaway = s->_cutaway;
_showIfEmpty = s->_showIfEmpty;
_hideSystemBarLine = s->_hideSystemBarLine;
_color = s->_color;
_userDist = s->_userDist;
}
//---------------------------------------------------------
// initFromStaffType
//---------------------------------------------------------
void Staff::initFromStaffType(const StaffType* staffType)
{
// get staff type if given (if none, get default preset for default staff group)
if (!staffType)
staffType = StaffType::getDefaultPreset(StaffGroup::STANDARD);
// use selected staff type
setStaffType(Fraction(0,1), *staffType);
}
//---------------------------------------------------------
// spatiumChanged
//---------------------------------------------------------
void Staff::spatiumChanged(qreal oldValue, qreal newValue)
{
_userDist = (_userDist / oldValue) * newValue;
}
//---------------------------------------------------------
// show
//---------------------------------------------------------
bool Staff::show() const
{
return _part->show();
}
//---------------------------------------------------------
// genKeySig
//---------------------------------------------------------
bool Staff::genKeySig()
{
if (constStaffType(Fraction(0,1))->group() == StaffGroup::TAB)
return false;
else
return constStaffType(Fraction(0,1))->genKeysig();
}
//---------------------------------------------------------
// showLedgerLines
//---------------------------------------------------------
bool Staff::showLedgerLines(const Fraction& tick) const
{
return staffType(tick)->showLedgerLines();
}
//---------------------------------------------------------
// updateOttava
//---------------------------------------------------------
void Staff::updateOttava()
{
int staffIdx = idx();
_pitchOffsets.clear();
for (auto i : score()->spanner()) {
const Spanner* s = i.second;
if (s->type() == ElementType::OTTAVA && s->staffIdx() == staffIdx) {
const Ottava* o = static_cast<const Ottava*>(s);
_pitchOffsets.setPitchOffset(o->tick().ticks(), o->pitchShift());
_pitchOffsets.setPitchOffset(o->tick2().ticks(), 0);
}
}
}
//---------------------------------------------------------
// undoSetColor
//---------------------------------------------------------
void Staff::undoSetColor(const QColor& /*val*/)
{
// undoChangeProperty(Pid::COLOR, val);
}
//---------------------------------------------------------
// insertTime
//---------------------------------------------------------
void Staff::insertTime(const Fraction& tick, const Fraction& len)
{
if (len.isZero())
return;
// move all keys and clefs >= tick
if (len < Fraction(0,1)) {
// remove entries between tickpos >= tick and tickpos < (tick+len)
_keys.erase(_keys.lower_bound(tick.ticks()), _keys.lower_bound((tick - len).ticks()));
clefs.erase(clefs.lower_bound(tick.ticks()), clefs.lower_bound((tick - len).ticks()));
}
KeyList kl2;
for (auto i = _keys.lower_bound(tick.ticks()); i != _keys.end();) {
KeySigEvent kse = i->second;
Fraction t = Fraction::fromTicks(i->first);
_keys.erase(i++);
kl2[(t + len).ticks()] = kse;
}
_keys.insert(kl2.begin(), kl2.end());
// check if there is a clef at the end of measure
// before tick
Clef* clef = 0;
Measure* m = score()->tick2measure(tick);
if (m && (m->tick() == tick) && (m->prevMeasure())) {
m = m->prevMeasure();
Segment* s = m->findSegment(SegmentType::Clef, tick);
if (s) {
int track = idx() * VOICES;
clef = toClef(s->element(track));
}
}
ClefList cl2;
for (auto i = clefs.lower_bound(tick.ticks()); i != clefs.end();) {
ClefTypeList ctl = i->second;
Fraction t = Fraction::fromTicks(i->first);
if (clef && tick == t) {
++i;
continue;
}
clefs.erase(i++);
cl2.setClef((t + len).ticks(), ctl);
}
clefs.insert(cl2.begin(), cl2.end());
// check if there is a clef at the end of measure
// before tick: do not remove from clefs list
if (clef)
setClef(clef);
updateOttava();
DUMP_CLEFS(" insertTime");
}
//---------------------------------------------------------
// staffList
// return list of linked staves
//---------------------------------------------------------
QList<Staff*> Staff::staffList() const
{
QList<Staff*> staffList;
if (_links) {
for (ScoreElement* e : *_links)
staffList.append(toStaff(e));
// staffList = _linkedStaves->staves();
}
else
staffList.append(const_cast<Staff*>(this));
return staffList;
}
//---------------------------------------------------------
// rstaff
//---------------------------------------------------------
int Staff::rstaff() const
{
return _part->staves()->indexOf((Staff*)this, 0);
}
//---------------------------------------------------------
// isTop
//---------------------------------------------------------
bool Staff::isTop() const
{
return _part->staves()->front() == this;
}
//---------------------------------------------------------
// getProperty
//---------------------------------------------------------
QVariant Staff::getProperty(Pid id) const
{
switch (id) {
case Pid::SMALL:
return small(Fraction(0,1));
case Pid::MAG:
return userMag(Fraction(0,1));
case Pid::COLOR:
return color();
case Pid::PLAYBACK_VOICE1:
return playbackVoice(0);
case Pid::PLAYBACK_VOICE2:
return playbackVoice(1);
case Pid::PLAYBACK_VOICE3:
return playbackVoice(2);
case Pid::PLAYBACK_VOICE4:
return playbackVoice(3);
case Pid::STAFF_BARLINE_SPAN:
return barLineSpan();
case Pid::STAFF_BARLINE_SPAN_FROM:
return barLineFrom();
case Pid::STAFF_BARLINE_SPAN_TO:
return barLineTo();
case Pid::STAFF_USERDIST:
return userDist();
case Pid::GENERATED:
return false;
default:
qDebug("unhandled id <%s>", propertyName(id));
return QVariant();
}
}
//---------------------------------------------------------
// setProperty
//---------------------------------------------------------
bool Staff::setProperty(Pid id, const QVariant& v)
{
switch (id) {
case Pid::SMALL: {
qreal _spatium = spatium(Fraction(0,1));
setSmall(Fraction(0,1), v.toBool());
localSpatiumChanged(_spatium, spatium(Fraction(0,1)), Fraction(0, 1));
break;
}
case Pid::MAG: {
qreal _spatium = spatium(Fraction(0,1));
setUserMag(Fraction(0,1), v.toReal());
localSpatiumChanged(_spatium, spatium(Fraction(0,1)), Fraction(0, 1));
}
break;
case Pid::COLOR:
setColor(v.value<QColor>());
break;
case Pid::PLAYBACK_VOICE1:
setPlaybackVoice(0, v.toBool());
break;
case Pid::PLAYBACK_VOICE2:
setPlaybackVoice(1, v.toBool());
break;
case Pid::PLAYBACK_VOICE3:
setPlaybackVoice(2, v.toBool());
break;
case Pid::PLAYBACK_VOICE4:
setPlaybackVoice(3, v.toBool());
break;
case Pid::STAFF_BARLINE_SPAN: {
setBarLineSpan(v.toInt());
// update non-generated barlines
int track = idx() * VOICES;
std::vector<Element*> blList;
for (Measure* m = score()->firstMeasure(); m; m = m->nextMeasure()) {
Segment* s = m->getSegmentR(SegmentType::EndBarLine, m->ticks());
if (s && s->element(track))
blList.push_back(s->element(track));
if (Measure* mm = m->mmRest()) {
Segment* ss = mm->getSegmentR(SegmentType::EndBarLine, mm->ticks());
if (ss && ss->element(track))
blList.push_back(ss->element(track));
}
}
for (Element* e : blList) {
if (e && e->isBarLine() && !e->generated())
toBarLine(e)->setSpanStaff(v.toInt());
}
}
break;
case Pid::STAFF_BARLINE_SPAN_FROM:
setBarLineFrom(v.toInt());
break;
case Pid::STAFF_BARLINE_SPAN_TO:
setBarLineTo(v.toInt());
break;
case Pid::STAFF_USERDIST:
setUserDist(v.toReal());
break;
default:
qDebug("unhandled id <%s>", propertyName(id));
break;
}
score()->setLayoutAll();
return true;
}
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
QVariant Staff::propertyDefault(Pid id) const
{
switch (id) {
case Pid::SMALL:
return false;
case Pid::MAG:
return 1.0;
case Pid::COLOR:
return QColor(Qt::black);
case Pid::PLAYBACK_VOICE1:
case Pid::PLAYBACK_VOICE2:
case Pid::PLAYBACK_VOICE3:
case Pid::PLAYBACK_VOICE4:
return true;
case Pid::STAFF_BARLINE_SPAN:
return false;
case Pid::STAFF_BARLINE_SPAN_FROM:
case Pid::STAFF_BARLINE_SPAN_TO:
return 0;
case Pid::STAFF_USERDIST:
return qreal(0.0);
default:
qDebug("unhandled id <%s>", propertyName(id));
return QVariant();
}
}
//---------------------------------------------------------
// localSpatiumChanged
//---------------------------------------------------------
void Staff::localSpatiumChanged(double oldVal, double newVal, Fraction tick)
{
Fraction etick;
auto i = _staffTypeList.find(tick.ticks());
++i;
if (i == _staffTypeList.end())
etick = score()->lastSegment()->tick();
else
etick = Fraction::fromTicks(i->first);
int staffIdx = idx();
int startTrack = staffIdx * VOICES;
int endTrack = startTrack + VOICES;
for (Segment* s = score()->tick2rightSegment(tick); s && s->tick() < etick; s = s->next1()) {
for (Element* e : s->annotations()) {
if (e->track() >= startTrack && e->track() < endTrack)
e->localSpatiumChanged(oldVal, newVal);
}
for (int track = startTrack; track < endTrack; ++track) {
if (s->element(track))
s->element(track)->localSpatiumChanged(oldVal, newVal);
}
}
auto spanners = score()->spannerMap().findContained(tick.ticks(), etick.ticks());
for (auto interval : spanners) {
Spanner* spanner = interval.value;
if (spanner->staffIdx() == staffIdx) {
for (auto k : spanner->spannerSegments())
k->localSpatiumChanged(oldVal, newVal);
}
}
}
//---------------------------------------------------------
// isPitchedStaff
//---------------------------------------------------------
bool Staff::isPitchedStaff(const Fraction& tick) const
{
return staffType(tick)->group() == StaffGroup::STANDARD;
}
//---------------------------------------------------------
// isTabStaff
//---------------------------------------------------------
bool Staff::isTabStaff(const Fraction& tick) const
{
return staffType(tick)->group() == StaffGroup::TAB;
}
//---------------------------------------------------------
// isDrumStaff
//---------------------------------------------------------
bool Staff::isDrumStaff(const Fraction& tick) const
{
return staffType(tick)->group() == StaffGroup::PERCUSSION;
}
//---------------------------------------------------------
// lines
//---------------------------------------------------------
int Staff::lines(const Fraction& tick) const
{
return staffType(tick)->lines();
}
//---------------------------------------------------------
// setLines
//---------------------------------------------------------
void Staff::setLines(const Fraction& tick, int val)
{
staffType(tick)->setLines(val);
}
//---------------------------------------------------------
// lineDistance
// distance between staff lines
//---------------------------------------------------------
qreal Staff::lineDistance(const Fraction& tick) const
{
return staffType(tick)->lineDistance().val();
}
}