diff --git a/importexport/musicxml/exportxml.cpp b/importexport/musicxml/exportxml.cpp index 93b095232cced..e709d477e5327 100644 --- a/importexport/musicxml/exportxml.cpp +++ b/importexport/musicxml/exportxml.cpp @@ -3092,6 +3092,9 @@ static QString symIdToTechn(const SymId sid) return "half-muted"; break; case SymId::brassHarmonMuteClosed: + case SymId::brassHarmonMuteStemHalfLeft: + case SymId::brassHarmonMuteStemHalfRight: + case SymId::brassHarmonMuteStemOpen: return "harmon-mute"; break; case SymId::guitarGolpe: @@ -3301,7 +3304,6 @@ void ExportMusicXml::chordAttributes(Chord* chord, Notations& notations, Technic QString placement; QString direction; - QString attr; if (!a->isStyled(Pid::ARTICULATION_ANCHOR) && a->anchor() != ArticulationAnchor::CHORD) { placement = (a->anchor() == ArticulationAnchor::BOTTOM_STAFF || a->anchor() == ArticulationAnchor::BOTTOM_CHORD) ? "below" : "above"; } @@ -3323,18 +3325,35 @@ void ExportMusicXml::chordAttributes(Chord* chord, Notations& notations, Technic technical.tag(_xml); mxmlTechn += color2xml(a); mxmlTechn += positioningAttributes(a); + if (!placement.isEmpty()) + mxmlTechn += QString(" placement=\"%1\"").arg(placement); if (sid == SymId::stringsHarmonic) { - if (!placement.isEmpty()) - attr += QString(" placement=\"%1\"").arg(placement); - _xml.stag(mxmlTechn + attr); + _xml.stag(mxmlTechn); _xml.tagE("natural"); _xml.etag(); } - else { - if (!placement.isEmpty()) - attr += QString(" placement=\"%1\"").arg(placement); - _xml.tagE(mxmlTechn + attr); + else if (mxmlTechn.startsWith("harmon")) { + _xml.stag(mxmlTechn); + QString location = {}; + QString harmonClosedValue; + switch (sid) + { + case SymId::brassHarmonMuteClosed: + harmonClosedValue = "yes"; + break; + case SymId::brassHarmonMuteStemOpen: + harmonClosedValue = "no"; + break; + default: + harmonClosedValue = "half"; + location = QString(" location=\"%1\"").arg(sid == SymId::brassHarmonMuteStemHalfLeft ? "left" : "right"); + break; + } + _xml.tag("harmon-closed" + location, harmonClosedValue); + _xml.etag(); } + else + _xml.tagE(mxmlTechn); } } diff --git a/importexport/musicxml/importmxmlpass2.cpp b/importexport/musicxml/importmxmlpass2.cpp index 547b0f795144b..a39a10247a47f 100644 --- a/importexport/musicxml/importmxmlpass2.cpp +++ b/importexport/musicxml/importmxmlpass2.cpp @@ -7694,6 +7694,8 @@ void MusicXMLParserNotations::technical() } else if (_e.name() == "harmonic") harmonic(); + else if (_e.name() == "harmon-mute") + harmonMute(); else if (_e.name() == "other-technical") otherTechnical(); else @@ -7743,6 +7745,42 @@ void MusicXMLParserNotations::harmonic() } } +//--------------------------------------------------------- +// harmonMute +//--------------------------------------------------------- + +/** + Parse the /score-partwise/part/measure/note/notations/technical/harmon-mute node. + */ + +void MusicXMLParserNotations::harmonMute() + { + SymId mute = SymId::brassHarmonMuteClosed; + while (_e.readNextStartElement()) { + QString name = _e.name().toString(); + if (name == "harmon-closed") { + const QString location = _e.attributes().value("location").toString(); + QString value = _e.readElementText(); + if (value == "yes") + mute = SymId::brassHarmonMuteClosed; + else if (value == "no") + mute = SymId::brassHarmonMuteStemOpen; + else if (value == "half") { + if (location == "left") + mute = SymId::brassHarmonMuteStemHalfLeft; + else if (location == "right") + mute = SymId::brassHarmonMuteStemHalfRight; + else { + _logger->logError(QString("unsupported harmon-closed location '%1'").arg(location), &_e); + mute = SymId::brassHarmonMuteStemHalfLeft; + } + } + } else + _e.skipCurrentElement(); + } + _notations.push_back(Notation::notationWithAttributes("harmon-closed", _e.attributes(), "technical", mute)); + } + //--------------------------------------------------------- // addTechnical //--------------------------------------------------------- diff --git a/importexport/musicxml/importmxmlpass2.h b/importexport/musicxml/importmxmlpass2.h index 80a5b11ec55e7..2bc7fd9670a24 100644 --- a/importexport/musicxml/importmxmlpass2.h +++ b/importexport/musicxml/importmxmlpass2.h @@ -242,6 +242,7 @@ class MusicXMLParserNotations { void addNotation(const Notation& notation, ChordRest* const cr, Note* const note); void addTechnical(const Notation& notation, Note* note); void harmonic(); + void harmonMute(); void articulations(); void dynamics(); void fermata(); diff --git a/mtest/musicxml/io/testHarmonMutes.xml b/mtest/musicxml/io/testHarmonMutes.xml new file mode 100644 index 0000000000000..67568f3319a70 --- /dev/null +++ b/mtest/musicxml/io/testHarmonMutes.xml @@ -0,0 +1,126 @@ + + + + + harmon-mute test + + + Klaus Rettinghaus + + MuseScore 0.7.0 + 2007-09-10 + + + + + + + + + + Posaune + Pos. + + Posaune + + + + 1 + 58 + 78.7402 + 0 + + + + + + + 1 + + 0 + + + + F + 4 + + + + + C + 3 + + 2 + 1 + half + up + + + + yes + + + + + + + C + 3 + + 2 + 1 + half + up + + + + no + + + + + + + + + C + 3 + + 2 + 1 + half + up + + + + half + + + + + + + C + 3 + + 2 + 1 + half + up + + + + half + + + + + + light-heavy + + + + diff --git a/mtest/musicxml/io/tst_mxml_io.cpp b/mtest/musicxml/io/tst_mxml_io.cpp index b08b51d6715df..a5de697cdc573 100644 --- a/mtest/musicxml/io/tst_mxml_io.cpp +++ b/mtest/musicxml/io/tst_mxml_io.cpp @@ -160,6 +160,7 @@ private slots: void harmony7() { mxmlMscxExportTestRef("testHarmony7"); } void harmony8() { mxmlIoTest("testHarmony8"); } void harmony9() { mxmlIoTest("testHarmony9"); } + void harmonMutes() { mxmlIoTest("testHarmonMutes"); } void hello() { mxmlIoTest("testHello"); } void helloReadCompr() { mxmlReadTestCompr("testHello"); } void helloReadWriteCompr() { mxmlReadWriteTestCompr("testHello"); }