MuseScore/libmscore/scoreElement.cpp

652 lines
29 KiB
C++
Raw Normal View History

//=============================================================================
// MuseScore
// Music Composition & Notation
//
// Copyright (C) 2015 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 "scoreElement.h"
#include "score.h"
#include "undo.h"
#include "xml.h"
2017-03-31 13:03:15 +02:00
#include "bracket.h"
#include "bracketItem.h"
namespace Ms {
2017-01-18 14:16:33 +01:00
//
// list has to be synchronized with ElementType enum
//
static const ElementName elementNames[] = {
{ ElementType::INVALID, "invalid", QT_TRANSLATE_NOOP("elementName", "invalid") },
2017-03-31 13:03:15 +02:00
{ ElementType::BRACKET_ITEM, "BracketItem", QT_TRANSLATE_NOOP("elementName", "BracketItem") },
2017-01-18 14:16:33 +01:00
{ ElementType::PART, "Part", QT_TRANSLATE_NOOP("elementName", "Part") },
{ ElementType::STAFF, "Staff", QT_TRANSLATE_NOOP("elementName", "Staff") },
{ ElementType::SCORE, "Score", QT_TRANSLATE_NOOP("elementName", "Score") },
{ ElementType::SYMBOL, "Symbol", QT_TRANSLATE_NOOP("elementName", "Symbol") },
{ ElementType::TEXT, "Text", QT_TRANSLATE_NOOP("elementName", "Text") },
{ ElementType::INSTRUMENT_NAME, "InstrumentName", QT_TRANSLATE_NOOP("elementName", "Instrument Name") },
{ ElementType::SLUR_SEGMENT, "SlurSegment", QT_TRANSLATE_NOOP("elementName", "Slur Segment") },
{ ElementType::TIE_SEGMENT, "TieSegment", QT_TRANSLATE_NOOP("elementName", "Tie Segment") },
{ ElementType::BAR_LINE, "BarLine", QT_TRANSLATE_NOOP("elementName", "Barline") },
2018-01-04 12:41:42 +01:00
{ ElementType::STAFF_LINES, "StaffLines", QT_TRANSLATE_NOOP("elementName", "Staff Lines") },
2017-01-18 14:16:33 +01:00
{ ElementType::SYSTEM_DIVIDER, "SystemDivider", QT_TRANSLATE_NOOP("elementName", "System Divider") },
{ ElementType::STEM_SLASH, "StemSlash", QT_TRANSLATE_NOOP("elementName", "Stem Slash") },
{ ElementType::ARPEGGIO, "Arpeggio", QT_TRANSLATE_NOOP("elementName", "Arpeggio") },
{ ElementType::ACCIDENTAL, "Accidental", QT_TRANSLATE_NOOP("elementName", "Accidental") },
{ ElementType::LEDGER_LINE, "LedgerLine", QT_TRANSLATE_NOOP("elementName", "Ledger Line") },
{ ElementType::STEM, "Stem", QT_TRANSLATE_NOOP("elementName", "Stem") },
{ ElementType::NOTE, "Note", QT_TRANSLATE_NOOP("elementName", "Note") },
{ ElementType::CLEF, "Clef", QT_TRANSLATE_NOOP("elementName", "Clef") },
{ ElementType::KEYSIG, "KeySig", QT_TRANSLATE_NOOP("elementName", "Key Signature") },
{ ElementType::AMBITUS, "Ambitus", QT_TRANSLATE_NOOP("elementName", "Ambitus") },
{ ElementType::TIMESIG, "TimeSig", QT_TRANSLATE_NOOP("elementName", "Time Signature") },
{ ElementType::REST, "Rest", QT_TRANSLATE_NOOP("elementName", "Rest") },
{ ElementType::BREATH, "Breath", QT_TRANSLATE_NOOP("elementName", "Breath") },
{ ElementType::REPEAT_MEASURE, "RepeatMeasure", QT_TRANSLATE_NOOP("elementName", "Repeat Measure") },
{ ElementType::TIE, "Tie", QT_TRANSLATE_NOOP("elementName", "Tie") },
{ ElementType::ARTICULATION, "Articulation", QT_TRANSLATE_NOOP("elementName", "Articulation") },
2018-01-16 13:38:17 +01:00
{ ElementType::FERMATA, "Fermata", QT_TRANSLATE_NOOP("elementName", "Fermata") },
2017-01-18 14:16:33 +01:00
{ ElementType::CHORDLINE, "ChordLine", QT_TRANSLATE_NOOP("elementName", "Chord Line") },
{ ElementType::DYNAMIC, "Dynamic", QT_TRANSLATE_NOOP("elementName", "Dynamic") },
{ ElementType::BEAM, "Beam", QT_TRANSLATE_NOOP("elementName", "Beam") },
{ ElementType::HOOK, "Hook", QT_TRANSLATE_NOOP("elementName", "Hook") },
{ ElementType::LYRICS, "Lyrics", QT_TRANSLATE_NOOP("elementName", "Lyrics") },
{ ElementType::FIGURED_BASS, "FiguredBass", QT_TRANSLATE_NOOP("elementName", "Figured Bass") },
{ ElementType::MARKER, "Marker", QT_TRANSLATE_NOOP("elementName", "Marker") },
{ ElementType::JUMP, "Jump", QT_TRANSLATE_NOOP("elementName", "Jump") },
{ ElementType::FINGERING, "Fingering", QT_TRANSLATE_NOOP("elementName", "Fingering") },
{ ElementType::TUPLET, "Tuplet", QT_TRANSLATE_NOOP("elementName", "Tuplet") },
{ ElementType::TEMPO_TEXT, "Tempo", QT_TRANSLATE_NOOP("elementName", "Tempo") },
{ ElementType::STAFF_TEXT, "StaffText", QT_TRANSLATE_NOOP("elementName", "Staff Text") },
2017-01-20 11:05:52 +01:00
{ ElementType::SYSTEM_TEXT, "SystemText", QT_TRANSLATE_NOOP("elementName", "System Text") },
2017-01-18 14:16:33 +01:00
{ ElementType::REHEARSAL_MARK, "RehearsalMark", QT_TRANSLATE_NOOP("elementName", "Rehearsal Mark") },
{ ElementType::INSTRUMENT_CHANGE, "InstrumentChange", QT_TRANSLATE_NOOP("elementName", "Instrument Change") },
2017-03-16 00:58:46 +01:00
{ ElementType::STAFFTYPE_CHANGE, "StaffTypeChange", QT_TRANSLATE_NOOP("elementName", "Staff Type Change") },
2017-01-18 14:16:33 +01:00
{ ElementType::HARMONY, "Harmony", QT_TRANSLATE_NOOP("elementName", "Chord Symbol") },
{ ElementType::FRET_DIAGRAM, "FretDiagram", QT_TRANSLATE_NOOP("elementName", "Fretboard Diagram") },
{ ElementType::BEND, "Bend", QT_TRANSLATE_NOOP("elementName", "Bend") },
{ ElementType::TREMOLOBAR, "TremoloBar", QT_TRANSLATE_NOOP("elementName", "Tremolo Bar") },
{ ElementType::VOLTA, "Volta", QT_TRANSLATE_NOOP("elementName", "Volta") },
{ ElementType::HAIRPIN_SEGMENT, "HairpinSegment", QT_TRANSLATE_NOOP("elementName", "Hairpin Segment") },
{ ElementType::OTTAVA_SEGMENT, "OttavaSegment", QT_TRANSLATE_NOOP("elementName", "Ottava Segment") },
{ ElementType::TRILL_SEGMENT, "TrillSegment", QT_TRANSLATE_NOOP("elementName", "Trill Segment") },
2017-11-27 09:56:41 +01:00
{ ElementType::LET_RING_SEGMENT, "LetRingSegment", QT_TRANSLATE_NOOP("elementName", "Let Ring Segment") },
{ ElementType::VIBRATO_SEGMENT, "VibratoSegment", QT_TRANSLATE_NOOP("elementName", "Vibrato Segment") },
2017-11-27 16:55:52 +01:00
{ ElementType::PALM_MUTE_SEGMENT, "PalmMuteSegment", QT_TRANSLATE_NOOP("elementName", "Palm Mute Segment") },
2017-01-18 14:16:33 +01:00
{ ElementType::TEXTLINE_SEGMENT, "TextLineSegment", QT_TRANSLATE_NOOP("elementName", "Text Line Segment") },
{ ElementType::VOLTA_SEGMENT, "VoltaSegment", QT_TRANSLATE_NOOP("elementName", "Volta Segment") },
{ ElementType::PEDAL_SEGMENT, "PedalSegment", QT_TRANSLATE_NOOP("elementName", "Pedal Segment") },
{ ElementType::LYRICSLINE_SEGMENT, "LyricsLineSegment", QT_TRANSLATE_NOOP("elementName", "Melisma Line Segment") },
{ ElementType::GLISSANDO_SEGMENT, "GlissandoSegment", QT_TRANSLATE_NOOP("elementName", "Glissando Segment") },
{ ElementType::LAYOUT_BREAK, "LayoutBreak", QT_TRANSLATE_NOOP("elementName", "Layout Break") },
{ ElementType::SPACER, "Spacer", QT_TRANSLATE_NOOP("elementName", "Spacer") },
{ ElementType::STAFF_STATE, "StaffState", QT_TRANSLATE_NOOP("elementName", "Staff State") },
{ ElementType::NOTEHEAD, "NoteHead", QT_TRANSLATE_NOOP("elementName", "Notehead") },
{ ElementType::NOTEDOT, "NoteDot", QT_TRANSLATE_NOOP("elementName", "Note Dot") },
{ ElementType::TREMOLO, "Tremolo", QT_TRANSLATE_NOOP("elementName", "Tremolo") },
{ ElementType::IMAGE, "Image", QT_TRANSLATE_NOOP("elementName", "Image") },
{ ElementType::MEASURE, "Measure", QT_TRANSLATE_NOOP("elementName", "Measure") },
{ ElementType::SELECTION, "Selection", QT_TRANSLATE_NOOP("elementName", "Selection") },
{ ElementType::LASSO, "Lasso", QT_TRANSLATE_NOOP("elementName", "Lasso") },
{ ElementType::SHADOW_NOTE, "ShadowNote", QT_TRANSLATE_NOOP("elementName", "Shadow Note") },
{ ElementType::TAB_DURATION_SYMBOL, "TabDurationSymbol", QT_TRANSLATE_NOOP("elementName", "Tab Duration Symbol") },
{ ElementType::FSYMBOL, "FSymbol", QT_TRANSLATE_NOOP("elementName", "Font Symbol") },
{ ElementType::PAGE, "Page", QT_TRANSLATE_NOOP("elementName", "Page") },
{ ElementType::HAIRPIN, "HairPin", QT_TRANSLATE_NOOP("elementName", "Hairpin") },
{ ElementType::OTTAVA, "Ottava", QT_TRANSLATE_NOOP("elementName", "Ottava") },
{ ElementType::PEDAL, "Pedal", QT_TRANSLATE_NOOP("elementName", "Pedal") },
{ ElementType::TRILL, "Trill", QT_TRANSLATE_NOOP("elementName", "Trill") },
2017-11-27 09:56:41 +01:00
{ ElementType::LET_RING, "LetRing", QT_TRANSLATE_NOOP("elementName", "Let Ring") },
{ ElementType::VIBRATO, "Vibrato", QT_TRANSLATE_NOOP("elementName", "Vibrato") },
2017-11-27 16:55:52 +01:00
{ ElementType::PALM_MUTE, "PalmMute", QT_TRANSLATE_NOOP("elementName", "Palm Mute") },
2017-01-18 14:16:33 +01:00
{ ElementType::TEXTLINE, "TextLine", QT_TRANSLATE_NOOP("elementName", "Text Line") },
{ ElementType::TEXTLINE_BASE, "TextLineBase", QT_TRANSLATE_NOOP("elementName", "Text Line Base") }, // remove
{ ElementType::NOTELINE, "NoteLine", QT_TRANSLATE_NOOP("elementName", "Note Line") },
{ ElementType::LYRICSLINE, "LyricsLine", QT_TRANSLATE_NOOP("elementName", "Melisma Line") },
{ ElementType::GLISSANDO, "Glissando", QT_TRANSLATE_NOOP("elementName", "Glissando") },
{ ElementType::BRACKET, "Bracket", QT_TRANSLATE_NOOP("elementName", "Bracket") },
{ ElementType::SEGMENT, "Segment", QT_TRANSLATE_NOOP("elementName", "Segment") },
{ ElementType::SYSTEM, "System", QT_TRANSLATE_NOOP("elementName", "System") },
{ ElementType::COMPOUND, "Compound", QT_TRANSLATE_NOOP("elementName", "Compound") },
{ ElementType::CHORD, "Chord", QT_TRANSLATE_NOOP("elementName", "Chord") },
{ ElementType::SLUR, "Slur", QT_TRANSLATE_NOOP("elementName", "Slur") },
{ ElementType::ELEMENT, "Element", QT_TRANSLATE_NOOP("elementName", "Element") },
{ ElementType::ELEMENT_LIST, "ElementList", QT_TRANSLATE_NOOP("elementName", "Element List") },
{ ElementType::STAFF_LIST, "StaffList", QT_TRANSLATE_NOOP("elementName", "Staff List") },
{ ElementType::MEASURE_LIST, "MeasureList", QT_TRANSLATE_NOOP("elementName", "Measure List") },
{ ElementType::HBOX, "HBox", QT_TRANSLATE_NOOP("elementName", "Horizontal Frame") },
{ ElementType::VBOX, "VBox", QT_TRANSLATE_NOOP("elementName", "Vertical Frame") },
{ ElementType::TBOX, "TBox", QT_TRANSLATE_NOOP("elementName", "Text Frame") },
{ ElementType::FBOX, "FBox", QT_TRANSLATE_NOOP("elementName", "Fretboard Diagram Frame") },
{ ElementType::ICON, "Icon", QT_TRANSLATE_NOOP("elementName", "Icon") },
{ ElementType::OSSIA, "Ossia", QT_TRANSLATE_NOOP("elementName", "Ossia") },
{ ElementType::BAGPIPE_EMBELLISHMENT,"BagpipeEmbellishment", QT_TRANSLATE_NOOP("elementName", "Bagpipe Embellishment") }
};
2016-03-30 22:33:04 +02:00
//---------------------------------------------------------
// ScoreElement
//---------------------------------------------------------
ScoreElement::ScoreElement(const ScoreElement& se)
{
2018-03-21 14:05:33 +01:00
_score = se._score;
_subStyleId = se._subStyleId;
2018-03-22 09:51:50 +01:00
int n = subStyle(_subStyleId).size() - 1; // don't count end of list marker
2018-03-21 14:05:33 +01:00
_propertyFlagsList = new PropertyFlags[n];
for (int i = 0; i < n; ++i)
_propertyFlagsList[i] = se._propertyFlagsList[i];
2016-03-30 22:33:04 +02:00
_links = 0;
}
//---------------------------------------------------------
2017-03-31 13:03:15 +02:00
// ~Element
2016-03-30 22:33:04 +02:00
//---------------------------------------------------------
ScoreElement::~ScoreElement()
{
if (_links) {
_links->removeOne(this);
if (_links->empty()) {
delete _links;
_links = 0;
}
}
2018-03-27 14:40:34 +02:00
delete[] _propertyFlagsList;
2018-03-21 14:05:33 +01:00
}
//---------------------------------------------------------
// setSubStyleId
//---------------------------------------------------------
void ScoreElement::setSubStyleId(SubStyleId ssid)
{
_subStyleId = ssid;
2018-03-27 14:40:34 +02:00
delete[] _propertyFlagsList;
2018-03-21 14:05:33 +01:00
int n = subStyle(_subStyleId).size() - 1; // don't count end of list marker
_propertyFlagsList = new PropertyFlags[n];
for (int i = 0; i < n; ++i)
_propertyFlagsList[i] = PropertyFlags::STYLED;
}
2018-03-27 14:40:34 +02:00
//---------------------------------------------------------
// propertyDefault
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
QVariant ScoreElement::propertyDefault(Pid id) const
2018-03-27 14:40:34 +02:00
{
2018-03-28 10:43:28 +02:00
if (id == Pid::SUB_STYLE)
2018-03-27 14:40:34 +02:00
return int(SubStyleId::DEFAULT);
2018-03-28 17:49:08 +02:00
#if 1 // this is wrong, styled properties should be considered first
for (const StyledProperty& p : subStyle(subStyleId())) {
if (p.pid == id)
return score()->styleV(p.sid);
}
#endif
qDebug("<%s> not found in <%s> style <%s>", propertyName(id), name(), subStyleName(subStyleId()));
return QVariant();
}
//---------------------------------------------------------
// styledPropertyDefault
//---------------------------------------------------------
QVariant ScoreElement::styledPropertyDefault(Pid id) const
{
2018-03-27 14:40:34 +02:00
for (const StyledProperty& p : subStyle(subStyleId())) {
2018-03-28 10:43:28 +02:00
if (p.pid == id)
return score()->styleV(p.sid);
2018-03-27 14:40:34 +02:00
}
return QVariant();
}
2018-03-21 14:05:33 +01:00
//---------------------------------------------------------
// initSubStyle
//---------------------------------------------------------
void ScoreElement::initSubStyle(SubStyleId ssid)
{
setSubStyleId(ssid);
int i = 0;
2018-03-28 10:43:28 +02:00
for (const StyledProperty* spp = styledProperties(); spp->sid != Sid::NOSTYLE; ++spp) {
Pid pid = spp->pid;
2018-03-21 14:05:33 +01:00
QVariant v = propertyDefault(pid);
if (v.isValid()) { // should always be true?
setProperty(pid, v);
PropertyFlags& p = propertyFlagsList()[i];
p = PropertyFlags::STYLED;
}
++i;
}
2016-03-30 22:33:04 +02:00
}
//---------------------------------------------------------
// resetProperty
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::resetProperty(Pid id)
{
QVariant v = propertyDefault(id);
if (v.isValid()) {
setProperty(id, v);
PropertyFlags& p = propertyFlags(id);
if (p != PropertyFlags::NOSTYLE)
p = PropertyFlags::STYLED;
}
}
2017-01-18 14:16:33 +01:00
//---------------------------------------------------------
// undoResetProperty
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::undoResetProperty(Pid id)
2017-01-18 14:16:33 +01:00
{
PropertyFlags f = propertyFlags(id);
if (f == PropertyFlags::UNSTYLED)
2017-01-18 14:16:33 +01:00
undoChangeProperty(id, propertyDefault(id), PropertyFlags::STYLED);
else
undoChangeProperty(id, propertyDefault(id), f);
}
2018-01-04 12:41:42 +01:00
//---------------------------------------------------------
// changeProperties
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
static void changeProperties(ScoreElement* e, Pid t, const QVariant& st, PropertyFlags ps)
2018-01-04 12:41:42 +01:00
{
if (propertyLink(t)) {
for (ScoreElement* ee : e->linkList()) {
if (ee->getProperty(t) != st || ee->propertyFlags(t) != ps)
ee->score()->undo(new ChangeProperty(ee, t, st, ps));
}
}
else {
if (e->getProperty(t) != st || e->propertyFlags(t) != ps)
e->score()->undo(new ChangeProperty(e, t, st, ps));
}
}
//---------------------------------------------------------
// undoChangeProperty
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::undoChangeProperty(Pid id, const QVariant& v)
{
2017-01-18 14:16:33 +01:00
undoChangeProperty(id, v, propertyFlags(id));
}
2018-03-27 15:36:00 +02:00
void ScoreElement::undoChangeProperty(Pid id, const QVariant& v, PropertyFlags ps)
2017-01-18 14:16:33 +01:00
{
2017-03-31 13:03:15 +02:00
if (isBracket()) {
// brackets do not survive layout() and therefore cannot be on
// the undo stack; delegate to BracketItem:
BracketItem* bi = toBracket(this)->bracketItem();
bi->undoChangeProperty(id, v, ps);
return;
}
2018-03-27 15:36:00 +02:00
if (id == Pid::AUTOPLACE && v.toBool() && !getProperty(id).toBool()) {
2017-01-18 14:16:33 +01:00
// special case: if we switch to autoplace, we must save
// user offset values
2018-03-27 15:36:00 +02:00
undoResetProperty(Pid::USER_OFF);
2017-01-18 14:16:33 +01:00
if (isSlurSegment()) {
2018-03-27 15:36:00 +02:00
undoResetProperty(Pid::SLUR_UOFF1);
undoResetProperty(Pid::SLUR_UOFF2);
undoResetProperty(Pid::SLUR_UOFF3);
undoResetProperty(Pid::SLUR_UOFF4);
2017-01-18 14:16:33 +01:00
}
}
2018-03-27 15:36:00 +02:00
else if (id == Pid::SUB_STYLE) {
2017-01-18 14:16:33 +01:00
//
// change a list of properties
//
2018-03-21 14:05:33 +01:00
auto l = subStyle(SubStyleId(v.toInt()));
2017-01-18 14:16:33 +01:00
// Change to SubStyle defaults
2018-03-21 14:05:33 +01:00
for (const StyledProperty& p : l) {
2018-03-28 10:43:28 +02:00
if (p.sid == Sid::NOSTYLE)
2018-03-21 14:05:33 +01:00
break;
2018-03-28 10:43:28 +02:00
changeProperties(this, p.pid, score()->styleV(p.sid), PropertyFlags::STYLED);
2018-03-21 14:05:33 +01:00
}
2017-01-18 14:16:33 +01:00
}
2018-01-04 12:41:42 +01:00
changeProperties(this, id, v, ps);
}
//---------------------------------------------------------
// undoPushProperty
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::undoPushProperty(Pid id)
{
QVariant val = getProperty(id);
2016-03-10 10:41:31 +01:00
score()->undoStack()->push1(new ChangeProperty(this, id, val));
}
2018-03-28 17:49:08 +02:00
//---------------------------------------------------------
// readProperty
//---------------------------------------------------------
bool ScoreElement::readProperty(const QStringRef& s, XmlReader& e, Pid id)
{
if (s == propertyName(id)) {
if (id == Pid::SUB_STYLE)
initSubStyle(SubStyleId(Ms::getProperty(id, e).toInt()));
else {
setProperty(id, Ms::getProperty(id, e));
setPropertyFlags(id, PropertyFlags::UNSTYLED);
}
return true;
}
return false;
}
//---------------------------------------------------------
// writeProperty
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::writeProperty(XmlWriter& xml, Pid id) const
{
2016-12-28 16:23:10 +01:00
if (propertyType(id) == P_TYPE::SP_REAL) {
qreal _spatium = score()->spatium();
2017-02-23 11:41:53 +01:00
xml.tag(id, QVariant(getProperty(id).toReal()/_spatium),
QVariant(propertyDefault(id).toReal()/_spatium));
2016-12-28 16:23:10 +01:00
}
else {
2018-03-22 13:17:22 +01:00
if (getProperty(id).isValid())
2018-03-15 11:52:17 +01:00
xml.tag(id, getProperty(id), propertyDefault(id));
2018-03-21 14:05:33 +01:00
else
2018-03-28 10:43:28 +02:00
qDebug("%s invalid property <%s>", name(), propertyName(id));
}
}
2018-03-28 17:49:08 +02:00
//---------------------------------------------------------
// readStyledProperty
//---------------------------------------------------------
bool ScoreElement::readStyledProperty(XmlReader& e, const QStringRef& tag)
{
for (const StyledProperty* spp = styledProperties(); spp->sid != Sid::NOSTYLE; ++spp) {
if (readProperty(tag, e, spp->pid))
return true;
}
return false;
}
2018-03-21 16:00:19 +01:00
//---------------------------------------------------------
// writeStyledProperties
//---------------------------------------------------------
void ScoreElement::writeStyledProperties(XmlWriter& xml) const
{
2018-03-28 10:43:28 +02:00
for (const StyledProperty* spp = styledProperties(); spp->sid != Sid::NOSTYLE; ++spp)
writeProperty(xml, spp->pid);
2018-03-21 16:00:19 +01:00
}
//---------------------------------------------------------
2018-03-28 17:49:08 +02:00
// reset
2018-03-21 16:00:19 +01:00
//---------------------------------------------------------
2018-03-28 17:49:08 +02:00
void ScoreElement::reset()
2018-03-21 16:00:19 +01:00
{
2018-03-28 10:43:28 +02:00
for (const StyledProperty* spp = styledProperties(); spp->sid != Sid::NOSTYLE; ++spp)
2018-03-28 17:49:08 +02:00
undoResetProperty(spp->pid);
2018-03-21 16:00:19 +01:00
}
//---------------------------------------------------------
// linkTo
//---------------------------------------------------------
void ScoreElement::linkTo(ScoreElement* element)
{
Q_ASSERT(element != this);
if (!_links) {
if (element->links()) {
_links = element->_links;
Q_ASSERT(_links->contains(element));
}
else {
_links = new LinkedElements(score());
_links->append(element);
element->_links = _links;
}
Q_ASSERT(!_links->contains(this));
_links->append(this);
}
else {
_links->append(element);
element->_links = _links;
}
}
//---------------------------------------------------------
// unlink
//---------------------------------------------------------
void ScoreElement::unlink()
{
if (_links) {
2018-04-18 14:31:36 +02:00
if (!_links->contains(this)) {
qDebug("%s: links size %d, this %p score %p", name(), _links->size(), this, score());
for (auto e : *_links)
printf(" %s %p score %p\n", e->name(), e, e->score());
qFatal("list does not contain 'this'");
}
_links->removeOne(this);
// if link list is empty, remove list
if (_links->size() <= 1) {
2018-04-18 14:31:36 +02:00
if (!_links->empty()) { // abnormal case: only "this" is in list
_links->front()->_links = 0;
2018-04-18 14:31:36 +02:00
qDebug("one element left in list");
}
delete _links;
}
2018-04-18 14:31:36 +02:00
_links = 0; // this element is not linked anymore
}
}
//---------------------------------------------------------
// undoUnlink
//---------------------------------------------------------
void ScoreElement::undoUnlink()
{
2016-12-02 10:47:09 +01:00
if (_links)
_score->undo(new Unlink(this));
}
//---------------------------------------------------------
// linkList
//---------------------------------------------------------
QList<ScoreElement*> ScoreElement::linkList() const
{
QList<ScoreElement*> el;
2018-01-04 12:41:42 +01:00
if (_links)
el.append(*_links);
else
2018-01-04 12:41:42 +01:00
el.append(const_cast<ScoreElement*>(this));
return el;
}
//---------------------------------------------------------
// LinkedElements
//---------------------------------------------------------
LinkedElements::LinkedElements(Score* score)
{
_lid = score->linkId(); // create new unique id
}
LinkedElements::LinkedElements(Score* score, int id)
{
_lid = id;
score->linkId(id); // remember used id
}
//---------------------------------------------------------
// setLid
//---------------------------------------------------------
void LinkedElements::setLid(Score* score, int id)
{
_lid = id;
score->linkId(id);
}
2016-03-10 10:41:31 +01:00
//---------------------------------------------------------
// masterScore
//---------------------------------------------------------
MasterScore* ScoreElement::masterScore() const
{
return _score->masterScore();
}
//---------------------------------------------------------
// propertyFlags
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
PropertyFlags& ScoreElement::propertyFlags(Pid id)
{
static PropertyFlags f = PropertyFlags::NOSTYLE;
2018-03-21 14:05:33 +01:00
2018-03-15 11:52:17 +01:00
const StyledProperty* spl = styledProperties();
for (int i = 0;;++i) {
const StyledProperty& k = spl[i];
2018-03-28 10:43:28 +02:00
if (k.sid == Sid::NOSTYLE)
2018-03-15 11:52:17 +01:00
break;
2018-03-28 10:43:28 +02:00
if (k.pid == id)
2018-03-15 11:52:17 +01:00
return propertyFlagsList()[i];
}
return f;
}
//---------------------------------------------------------
// setPropertyFlags
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
void ScoreElement::setPropertyFlags(Pid id, PropertyFlags f)
{
PropertyFlags& p = propertyFlags(id);
if (p != PropertyFlags::NOSTYLE)
p = f;
}
//---------------------------------------------------------
// getPropertyStyle
//---------------------------------------------------------
2018-03-27 15:36:00 +02:00
Sid ScoreElement::getPropertyStyle(Pid id) const
{
2018-03-15 11:52:17 +01:00
const StyledProperty* spl = styledProperties();
for (int i = 0;;++i) {
const StyledProperty& k = spl[i];
2018-03-28 10:43:28 +02:00
if (k.sid == Sid::NOSTYLE)
2018-03-15 11:52:17 +01:00
break;
2018-03-28 10:43:28 +02:00
if (k.pid == id)
return k.sid;
2018-03-15 11:52:17 +01:00
}
2018-03-27 15:36:00 +02:00
return Sid::NOSTYLE;
}
2017-01-18 14:16:33 +01:00
//---------------------------------------------------------
// styleChanged
//---------------------------------------------------------
void ScoreElement::styleChanged()
{
2018-03-28 10:43:28 +02:00
for (const StyledProperty* spp = styledProperties(); spp->sid != Sid::NOSTYLE; ++spp) {
PropertyFlags& f = propertyFlags(spp->pid);
if (f == PropertyFlags::STYLED) {
if (propertyType(spp->pid) == P_TYPE::SP_REAL) {
qreal val = score()->styleP(spp->sid);
setProperty(spp->pid, val);
}
else {
setProperty(spp->pid, score()->styleV(spp->sid));
}
}
}
}
2017-01-18 14:16:33 +01:00
//---------------------------------------------------------
// name
//---------------------------------------------------------
const char* ScoreElement::name() const
{
return name(type());
}
//---------------------------------------------------------
// name
//---------------------------------------------------------
const char* ScoreElement::name(ElementType type)
{
return elementNames[int(type)].name;
}
//---------------------------------------------------------
// userName
//---------------------------------------------------------
QString ScoreElement::userName() const
{
return qApp->translate("elementName", elementNames[int(type())].userName);
}
//---------------------------------------------------------
// name2type
//---------------------------------------------------------
ElementType ScoreElement::name2type(const QStringRef& s)
{
for (int i = 0; i < int(ElementType::MAXTYPE); ++i) {
if (s == elementNames[i].name)
return ElementType(i);
}
2017-03-31 13:03:15 +02:00
qDebug("unknown type <%s>", qPrintable(s.toString()));
2017-01-18 14:16:33 +01:00
return ElementType::INVALID;
}
//---------------------------------------------------------
// isSLineSegment
//---------------------------------------------------------
bool ScoreElement::isSLineSegment() const
{
return isHairpinSegment() || isOttavaSegment() || isPedalSegment()
|| isTrillSegment() || isVoltaSegment() || isTextLineSegment()
2017-11-27 16:55:52 +01:00
|| isGlissandoSegment() || isLetRingSegment() || isVibratoSegment() || isPalmMuteSegment();
2017-01-18 14:16:33 +01:00
}
//---------------------------------------------------------
// isText
//---------------------------------------------------------
2018-04-18 14:31:36 +02:00
bool ScoreElement::isTextBase() const
2017-01-18 14:16:33 +01:00
{
return type() == ElementType::TEXT
|| type() == ElementType::LYRICS
|| type() == ElementType::DYNAMIC
|| type() == ElementType::FINGERING
|| type() == ElementType::HARMONY
|| type() == ElementType::MARKER
|| type() == ElementType::JUMP
|| type() == ElementType::STAFF_TEXT
|| type() == ElementType::REHEARSAL_MARK
|| type() == ElementType::INSTRUMENT_CHANGE
|| type() == ElementType::FIGURED_BASS
|| type() == ElementType::TEMPO_TEXT
|| type() == ElementType::INSTRUMENT_NAME
;
}
}