//============================================================================= // MuseScore // Linux Music Score Editor // // Copyright (C) 2015 Werner Schweer and others // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License version 2. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. //============================================================================= #include "libmscore/arpeggio.h" #include "libmscore/accidental.h" #include "libmscore/breath.h" #include "libmscore/chord.h" #include "libmscore/chordline.h" #include "libmscore/chordlist.h" #include "libmscore/chordrest.h" #include "libmscore/drumset.h" #include "libmscore/dynamic.h" #include "libmscore/figuredbass.h" #include "libmscore/fingering.h" #include "libmscore/fret.h" #include "libmscore/glissando.h" #include "libmscore/hairpin.h" #include "libmscore/harmony.h" #include "libmscore/instrchange.h" #include "libmscore/instrtemplate.h" #include "libmscore/interval.h" #include "libmscore/jump.h" #include "libmscore/keysig.h" #include "libmscore/lyrics.h" #include "libmscore/marker.h" #include "libmscore/measure.h" #include "libmscore/mscore.h" #include "libmscore/note.h" #include "libmscore/part.h" #include "libmscore/pedal.h" #include "libmscore/rest.h" #include "libmscore/slur.h" #include "libmscore/staff.h" #include "libmscore/stafftext.h" #include "libmscore/sym.h" #include "libmscore/tempotext.h" #include "libmscore/tie.h" #include "libmscore/timesig.h" #include "libmscore/tremolo.h" #include "libmscore/trill.h" #include "libmscore/utils.h" #include "libmscore/volta.h" #include "libmscore/textline.h" #include "libmscore/barline.h" #include "libmscore/articulation.h" #include "libmscore/ottava.h" #include "libmscore/rehearsalmark.h" #include "libmscore/fermata.h" #include "importmxmllogger.h" #include "importmxmlnoteduration.h" #include "importmxmlnotepitch.h" #include "importmxmlpass2.h" #include "musicxmlfonthandler.h" #include "musicxmlsupport.h" #include "preferences.h" namespace Ms { //--------------------------------------------------------- // local defines for debug output //--------------------------------------------------------- //#define DEBUG_VOICE_MAPPER true //--------------------------------------------------------- // support enums / structs / classes //--------------------------------------------------------- //--------------------------------------------------------- // MusicXmlTupletDesc //--------------------------------------------------------- MusicXmlTupletDesc::MusicXmlTupletDesc() : type(MxmlStartStop::NONE), placement(Placement::BELOW), bracket(TupletBracketType::AUTO_BRACKET), shownumber(TupletNumberType::SHOW_NUMBER) { // nothing } //--------------------------------------------------------- // MusicXmlLyricsExtend //--------------------------------------------------------- //--------------------------------------------------------- // init //--------------------------------------------------------- void MusicXmlLyricsExtend::init() { _lyrics.clear(); } //--------------------------------------------------------- // addLyric //--------------------------------------------------------- // add a single lyric to be extended later // called when lyric with "extend" or "extend type=start" is found void MusicXmlLyricsExtend::addLyric(Lyrics* const lyric) { _lyrics.insert(lyric); } //--------------------------------------------------------- // lastChordTicks //--------------------------------------------------------- // find the duration of the chord starting at or after s in track and ending at tick static int lastChordTicks(const Segment* s, const int track, const int tick) { while (s && s->tick() < tick) { Element* el = s->element(track); if (el && el->isChordRest()) { ChordRest* cr = static_cast(el); if (cr->tick() + cr->actualTicks() == tick) return cr->actualTicks(); } s = s->nextCR(track, true); } return 0; } //--------------------------------------------------------- // setExtend //--------------------------------------------------------- // set extend for lyric no in track to end at tick // called when lyric (with or without "extend") or note with "extend type=stop" is found // note that no == -1 means all lyrics in this track void MusicXmlLyricsExtend::setExtend(const int no, const int track, const int tick) { QList list; foreach(Lyrics* l, _lyrics) { Element* const el = l->parent(); if (el->type() == ElementType::CHORD) { // TODO: rest also possible ? ChordRest* const par = static_cast(el); if (par->track() == track && (no == -1 || l->no() == no)) { int lct = lastChordTicks(l->segment(), track, tick); if (lct > 0) { // set lyric tick to the total length fron the lyric note // plus all notes covered by the melisma minus the last note length l->setTicks(tick - par->tick() - lct); } list.append(l); } } } // cleanup foreach(Lyrics* l, list) { _lyrics.remove(l); } } //--------------------------------------------------------- // MusicXMLStepAltOct2Pitch //--------------------------------------------------------- /** Convert MusicXML \a step (0=C, 1=D, etc.) / \a alter / \a octave to midi pitch. Note: same code is in pass 1 and in pass 2. TODO: combine */ static int MusicXMLStepAltOct2Pitch(int step, int alter, int octave) { // c d e f g a b static int table[7] = { 0, 2, 4, 5, 7, 9, 11 }; if (step < 0 || step > 6) { qDebug("MusicXMLStepAltOct2Pitch: illegal step %d", step); return -1; } int pitch = table[step] + alter + (octave+1) * 12; if (pitch < 0) pitch = -1; if (pitch > 127) pitch = -1; return pitch; } //--------------------------------------------------------- // xmlSetPitch //--------------------------------------------------------- /** Convert MusicXML \a step / \a alter / \a octave to midi pitch, set pitch and tpc. Note that n's staff and track have not been set yet */ static void xmlSetPitch(Note* n, int step, int alter, int octave, const int octaveShift, const Instrument* instr) { //qDebug("xmlSetPitch(n=%p, step=%d, alter=%d, octave=%d, octaveShift=%d)", // n, step, alter, octave, octaveShift); //const Staff* staff = n->score()->staff(track / VOICES); //const Instrument* instr = staff->part()->instr(); const Interval intval = instr->transpose(); // TODO: tick //qDebug(" staff=%p instr=%p dia=%d chro=%d", // staff, instr, (int) intval.diatonic, (int) intval.chromatic); int pitch = MusicXMLStepAltOct2Pitch(step, alter, octave); pitch += intval.chromatic; // assume not in concert pitch pitch += 12 * octaveShift; // correct for octave shift // ensure sane values pitch = limit(pitch, 0, 127); int tpc2 = step2tpc(step, AccidentalVal(alter)); int tpc1 = Ms::transposeTpc(tpc2, intval, true); n->setPitch(pitch, tpc1, tpc2); //qDebug(" pitch=%d tpc1=%d tpc2=%d", n->pitch(), n->tpc1(), n->tpc2()); } //--------------------------------------------------------- // fillGap //--------------------------------------------------------- /** Fill one gap (tstart - tend) in this track in this measure with rest(s). */ static void fillGap(Measure* measure, int track, int tstart, int tend) { int ctick = tstart; int restLen = tend - tstart; // qDebug("\nfillGIFV fillGap(measure %p track %d tstart %d tend %d) restLen %d len", // measure, track, tstart, tend, restLen); // note: as MScore::division (#ticks in a quarter note) equals 480 // MScore::division / 64 (#ticks in a 256th note) uequals 7.5 but is rounded down to 7 while (restLen > MScore::division / 64) { int len = restLen; TDuration d(TDuration::DurationType::V_INVALID); if (measure->ticks() == restLen) d.setType(TDuration::DurationType::V_MEASURE); else d.setVal(len); Rest* rest = new Rest(measure->score(), d); rest->setDuration(Fraction::fromTicks(len)); rest->setTrack(track); rest->setVisible(false); Segment* s = measure->getSegment(SegmentType::ChordRest, tstart); s->add(rest); len = rest->globalDuration().ticks(); // qDebug(" %d", len); ctick += len; restLen -= len; } } //--------------------------------------------------------- // fillGapsInFirstVoices //--------------------------------------------------------- /** Fill gaps in first voice of every staff in this measure for this part with rest(s). */ static void fillGapsInFirstVoices(Measure* measure, Part* part) { Q_ASSERT(measure); Q_ASSERT(part); int measTick = measure->tick(); int measLen = measure->ticks(); int nextMeasTick = measTick + measLen; int staffIdx = part->score()->staffIdx(part); /* qDebug("fillGIFV measure %p part %p idx %d nstaves %d tick %d - %d (len %d)", measure, part, staffIdx, part->nstaves(), measTick, nextMeasTick, measLen); */ for (int st = 0; st < part->nstaves(); ++st) { int track = (staffIdx + st) * VOICES; int endOfLastCR = measTick; for (Segment* s = measure->first(); s; s = s->next()) { // qDebug("fillGIFV segment %p tp %s", s, s->subTypeName()); Element* el = s->element(track); if (el) { // qDebug(" el[%d] %p", track, el); if (s->isChordRestType()) { ChordRest* cr = static_cast(el); int crTick = cr->tick(); int crLen = cr->globalDuration().ticks(); int nextCrTick = crTick + crLen; /* qDebug(" chord/rest tick %d - %d (len %d)", crTick, nextCrTick, crLen); */ if (crTick > endOfLastCR) { /* qDebug(" GAP: track %d tick %d - %d", track, endOfLastCR, crTick); */ fillGap(measure, track, endOfLastCR, crTick); } endOfLastCR = nextCrTick; } } } if (nextMeasTick > endOfLastCR) { /* qDebug("fillGIFV measure end GAP: track %d tick %d - %d", track, endOfLastCR, nextMeasTick); */ fillGap(measure, track, endOfLastCR, nextMeasTick); } } } //--------------------------------------------------------- // hasDrumset //--------------------------------------------------------- /** Determine if \a mxmlDrumset contains a valid drumset. This is the case if any instrument has a midi-unpitched element, (which stored in the MusicXMLDrumInstrument pitch field). */ static bool hasDrumset(const MusicXMLDrumset& mxmlDrumset) { bool res = false; MusicXMLDrumsetIterator ii(mxmlDrumset); while (ii.hasNext()) { ii.next(); // debug: dump the drumset //qDebug("hasDrumset: instrument: %s %s", qPrintable(ii.key()), qPrintable(ii.value().toString())); int pitch = ii.value().pitch; if (0 <= pitch && pitch <= 127) { res = true; } } /* for (const auto& instr : mxmlDrumset) { // MusicXML elements instrument-name, midi-program, instrument-sound, virtual-library, virtual-name // in a shell script use "mscore ... 2>&1 | grep GREP_ME | cut -d' ' -f3-" to extract qDebug("GREP_ME '%s',%d,'%s','%s','%s'", qPrintable(instr.name), instr.midiProgram + 1, qPrintable(instr.sound), qPrintable(instr.virtLib), qPrintable(instr.virtName) ); } */ return res; } //--------------------------------------------------------- // initDrumset //--------------------------------------------------------- /** Initialize drumset \a drumset. */ // determine if the part contains a drumset // this is the case if any instrument has a midi-unpitched element, // (which stored in the MusicXMLDrumInstrument pitch field) // if the part contains a drumset, Drumset drumset is initialized static void initDrumset(Drumset* drumset, const MusicXMLDrumset& mxmlDrumset) { drumset->clear(); MusicXMLDrumsetIterator ii(mxmlDrumset); while (ii.hasNext()) { ii.next(); // debug: also dump the drumset for this part //qDebug("initDrumset: instrument: %s %s", qPrintable(ii.key()), qPrintable(ii.value().toString())); int pitch = ii.value().pitch; if (0 <= pitch && pitch <= 127) { drumset->drum(ii.value().pitch) = DrumInstrument(ii.value().name.toLatin1().constData(), ii.value().notehead, ii.value().line, ii.value().stemDirection); } } } //--------------------------------------------------------- // createInstrument //--------------------------------------------------------- /** Create an Instrument based on the information in \a mxmlInstr. */ static Instrument createInstrument(const MusicXMLDrumInstrument& mxmlInstr) { Instrument instr; InstrumentTemplate* it {}; if (!mxmlInstr.sound.isEmpty()) { it = Ms::searchTemplateForMusicXmlId(mxmlInstr.sound); } /* qDebug("sound '%s' it %p trackname '%s' program %d", qPrintable(mxmlInstr.sound), it, it ? qPrintable(it->trackName) : "", mxmlInstr.midiProgram); */ if (it) { // initialize from template with matching MusicXmlId instr = Instrument::fromTemplate(it); // reset transpose, as it is determined later from MusicXML data instr.setTranspose(Interval()); } else { // set articulations to default (global articulations) instr.setArticulation(articulation); // set default program instr.channel(0)->program = mxmlInstr.midiProgram >= 0 ? mxmlInstr.midiProgram : 0; } // add / overrule with values read from MusicXML instr.channel(0)->pan = mxmlInstr.midiPan; instr.channel(0)->volume = mxmlInstr.midiVolume; instr.setTrackName(mxmlInstr.name); return instr; } //--------------------------------------------------------- // setFirstInstrument //--------------------------------------------------------- /** Set first instrument for Part \a part */ static void setFirstInstrument(MxmlLogger* logger, const QXmlStreamReader* const xmlreader, Part* part, const QString& partId, const QString& instrId, const MusicXMLDrumset& mxmlDrumset) { if (mxmlDrumset.size() > 0) { //qDebug("setFirstInstrument: initial instrument '%s'", qPrintable(instrId)); MusicXMLDrumInstrument mxmlInstr; if (instrId == "") mxmlInstr = mxmlDrumset.first(); else if (mxmlDrumset.contains(instrId)) mxmlInstr = mxmlDrumset.value(instrId); else { logger->logError(QString("initial instrument '%1' not found in part '%2'") .arg(instrId).arg(partId), xmlreader); mxmlInstr = mxmlDrumset.first(); } Instrument instr = createInstrument(mxmlInstr); part->setInstrument(instr); if (mxmlInstr.midiChannel >= 0) part->setMidiChannel(mxmlInstr.midiChannel, mxmlInstr.midiPort); // note: setMidiProgram() does more than simply setting the MIDI program if (mxmlInstr.midiProgram >= 0) part->setMidiProgram(mxmlInstr.midiProgram); } else logger->logError(QString("no instrument found for part '%1'") .arg(partId), xmlreader); } //--------------------------------------------------------- // setStaffTypePercussion //--------------------------------------------------------- /** Set staff type to percussion */ static void setStaffTypePercussion(Part* part, Drumset* drumset) { for (int j = 0; j < part->nstaves(); ++j) if (part->staff(j)->lines(0) == 5 && !part->staff(j)->isDrumStaff(0)) part->staff(j)->setStaffType(0, StaffType::preset(StaffTypes::PERC_DEFAULT)); // set drumset for instrument part->instrument()->setDrumset(drumset); part->instrument()->channel(0)->bank = 128; part->instrument()->channel(0)->updateInitList(); } //--------------------------------------------------------- // findDeleteStaffText //--------------------------------------------------------- /** Find a non-empty staff text in \a s at \a track (which originates as MusicXML ). If found, delete it and return its text. */ static QString findDeleteStaffText(Segment* s, int track) { //qDebug("findDeleteWords(s %p track %d)", s, track); foreach (Element* e, s->annotations()) { //qDebug("findDeleteWords e %p type %hhd track %d", e, e->type(), e->track()); if (e->type() != ElementType::STAFF_TEXT || e->track() < track || e->track() >= track+VOICES) continue; Text* t = static_cast(e); //qDebug("findDeleteWords t %p text '%s'", t, qPrintable(t->text())); QString res = t->xmlText(); if (res != "") { s->remove(t); return res; } } return ""; } //--------------------------------------------------------- // setPartInstruments //--------------------------------------------------------- static void setPartInstruments(MxmlLogger* logger, const QXmlStreamReader* const xmlreader, Part* part, const QString& partId, Score* score, const MusicXmlInstrList& il, const MusicXMLDrumset& mxmlDrumset) { QString prevInstrId; for (auto it = il.cbegin(); it != il.cend(); ++it) { Fraction f = (*it).first; if (f == Fraction(0, 1)) prevInstrId = (*it).second; // instrument id at t = 0 else if (f > Fraction(0, 1)) { auto instrId = (*it).second; bool mustInsert = instrId != prevInstrId; /* qDebug("f %s previd %s id %s mustInsert %d", qPrintable(f.print()), qPrintable(prevInstrId), qPrintable(instrId), mustInsert); */ if (mustInsert) { const int staff = score->staffIdx(part); const int track = staff * VOICES; const int tick = f.ticks(); //qDebug("instrument change: tick %s (%d) track %d instr '%s'", // qPrintable(f.print()), tick, track, qPrintable(instrId)); auto segment = score->tick2segment(tick, true, SegmentType::ChordRest, true); if (!segment) logger->logError(QString("segment for instrument change at tick %1 not found") .arg(tick), xmlreader); else if (!mxmlDrumset.contains(instrId)) logger->logError(QString("changed instrument '%1' at tick %2 not found in part '%3'") .arg(instrId).arg(tick).arg(partId), xmlreader); else { MusicXMLDrumInstrument mxmlInstr = mxmlDrumset.value(instrId); Instrument instr = createInstrument(mxmlInstr); //qDebug("instr %p", &instr); InstrumentChange* ic = new InstrumentChange(instr, score); ic->setTrack(track); // if there is already a staff text at this tick / track, // delete it and use its text here instead of "Instrument change" QString text = findDeleteStaffText(segment, track); ic->setXmlText(text.isEmpty() ? "Instrument change" : text); segment->add(ic); // note: includes part::setInstrument(instr); // setMidiChannel() depends on setInstrument() already been done if (mxmlInstr.midiChannel >= 0) part->setMidiChannel(mxmlInstr.midiChannel, mxmlInstr.midiPort, tick); } } prevInstrId = instrId; } } } //--------------------------------------------------------- // text2syms //--------------------------------------------------------- /** Convert SMuFL code points to MuseScore ... */ static QString text2syms(const QString& t) { //QTime time; //time.start(); // first create a map from symbol (Unicode) text to symId // note that this takes about 1 msec on a Core i5, // caching does not gain much ScoreFont* sf = ScoreFont::fallbackFont(); QMap map; int maxStringSize = 0; // maximum string size found for (int i = int(SymId::noSym); i < int(SymId::lastSym); ++i) { SymId id((SymId(i))); QString string(sf->toString(id)); // insert all syms except space to prevent matching all regular spaces if (id != SymId::space) map.insert(string, id); if (string.size() > maxStringSize) maxStringSize = string.size(); } //qDebug("text2syms map count %d maxsz %d filling time elapsed: %d ms", // map.size(), maxStringSize, time.elapsed()); // then look for matches QString in = t; QString res; while (in != "") { // try to find the largest match possible int maxMatch = qMin(in.size(), maxStringSize); QString sym; while (maxMatch > 0) { QString toBeMatched = in.left(maxMatch); if (map.contains(toBeMatched)) { sym = Sym::id2name(map.value(toBeMatched)); break; } maxMatch--; } if (maxMatch > 0) { // found a match, add sym to res and remove match from string in res += ""; res += sym; res += ""; in.remove(0, maxMatch); } else { // not found, move one char from res to in res += in.left(1); in.remove(0, 1); } } //qDebug("text2syms total time elapsed: %d ms, res '%s'", time.elapsed(), qPrintable(res)); return res; } //--------------------------------------------------------- // decodeEntities //--------------------------------------------------------- /** Decode &#...; in string \a src into UNICODE (utf8) character. */ static QString decodeEntities( const QString& src ) { QString ret(src); QRegExp re("&#([0-9]+);"); re.setMinimal(true); int pos = 0; while ( (pos = re.indexIn(src, pos)) != -1 ) { ret = ret.replace(re.cap(0), QChar(re.cap(1).toInt(0,10))); pos += re.matchedLength(); } return ret; } //--------------------------------------------------------- // nextPartOfFormattedString //--------------------------------------------------------- // TODO: probably should be shared between pass 1 and 2 /** Read the next part of a MusicXML formatted string and convert to MuseScore internal encoding. */ static QString nextPartOfFormattedString(QXmlStreamReader& e) { //QString lang = e.attribute(QString("xml:lang"), "it"); QString fontWeight = e.attributes().value("font-weight").toString(); QString fontSize = e.attributes().value("font-size").toString(); QString fontStyle = e.attributes().value("font-style").toString(); QString underline = e.attributes().value("underline").toString(); QString fontFamily = e.attributes().value("font-family").toString(); // TODO: color, enclosure, yoffset in only part of the text, ... QString txt = e.readElementText(); // replace HTML entities txt = decodeEntities(txt); QString syms = text2syms(txt); QString importedtext; if (!fontSize.isEmpty()) { bool ok = true; float size = fontSize.toFloat(&ok); if (ok) importedtext += QString("").arg(size); } if (!fontFamily.isEmpty() && txt == syms) { // add font family only if no replacement made importedtext += QString("").arg(fontFamily); } if (fontWeight == "bold") importedtext += ""; if (fontStyle == "italic") importedtext += ""; if (!underline.isEmpty()) { bool ok = true; int lines = underline.toInt(&ok); if (ok && (lines > 0)) // 1,2, or 3 underlines are imported as single underline importedtext += ""; else underline = ""; } if (txt == syms) { txt.replace(QString("\r"), QString("")); // convert Windows line break \r\n -> \n importedtext += txt.toHtmlEscaped(); } else { // replacement made, should be no need for line break or other conversions importedtext += syms; } if (underline != "") importedtext += ""; if (fontStyle == "italic") importedtext += ""; if (fontWeight == "bold") importedtext += ""; //qDebug("importedtext '%s'", qPrintable(importedtext)); return importedtext; } //--------------------------------------------------------- // addLyric //--------------------------------------------------------- /** Add a single lyric to the score or delete it (if number too high) */ static void addLyric(MxmlLogger* logger, const QXmlStreamReader* const xmlreader, ChordRest* cr, Lyrics* l, int lyricNo, MusicXmlLyricsExtend& extendedLyrics) { if (lyricNo > MAX_LYRICS) { logger->logError(QString("too much lyrics (>%1)") .arg(MAX_LYRICS), xmlreader); delete l; } else { l->setNo(lyricNo); cr->add(l); extendedLyrics.setExtend(lyricNo, cr->track(), cr->tick()); } } //--------------------------------------------------------- // addLyrics //--------------------------------------------------------- /** Add a notes lyrics to the score */ static void addLyrics(MxmlLogger* logger, const QXmlStreamReader* const xmlreader, ChordRest* cr, QMap& numbrdLyrics, QSet& extLyrics, MusicXmlLyricsExtend& extendedLyrics) { int lyricNo = -1; for (QMap::const_iterator i = numbrdLyrics.constBegin(); i != numbrdLyrics.constEnd(); ++i) { lyricNo = i.key(); Lyrics* l = i.value(); addLyric(logger, xmlreader, cr, l, lyricNo, extendedLyrics); if (extLyrics.contains(l)) extendedLyrics.addLyric(l); } } //--------------------------------------------------------- // addElemOffset //--------------------------------------------------------- static void addElemOffset(Element* el, int track, const QString& placement, Measure* measure, int tick) { /* qDebug("addElem el %p track %d placement %s tick %d", el, track, qPrintable(placement), tick); */ #if 0 // ws: use placement for symbols // move to correct position // TODO: handle rx, ry if (el->isSymbol()) { qreal y = 0; // calc y offset assuming five line staff and default style // note that required y offset is element type dependent const qreal stafflines = 5; // assume five line staff, but works OK-ish for other sizes too qreal offsAbove = 0; qreal offsBelow = 0; offsAbove = -2; offsBelow = 4 + (stafflines - 1); if (placement == "above") y += offsAbove; if (placement == "below") y += offsBelow; //qDebug(" y = %g", y); y *= el->score()->spatium(); el->setUserOff(QPoint(0, y)); } else { el->setPlacement(placement == "above" ? Placement::ABOVE : Placement::BELOW); } #endif el->setPlacement(placement == "above" ? Placement::ABOVE : Placement::BELOW); el->setTrack(track); Segment* s = measure->getSegment(SegmentType::ChordRest, tick); s->add(el); } //--------------------------------------------------------- // tupletAssert -- check assertions for tuplet handling //--------------------------------------------------------- /** Check assertions for tuplet handling. If this fails, MusicXML import will almost certainly break in non-obvious ways. Should never happen, thus it is OK to quit the application. */ #if 0 static void tupletAssert() { if (!(int(TDuration::DurationType::V_BREVE) == int(TDuration::DurationType::V_LONG) + 1 && int(TDuration::DurationType::V_WHOLE) == int(TDuration::DurationType::V_BREVE) + 1 && int(TDuration::DurationType::V_HALF) == int(TDuration::DurationType::V_WHOLE) + 1 && int(TDuration::DurationType::V_QUARTER) == int(TDuration::DurationType::V_HALF) + 1 && int(TDuration::DurationType::V_EIGHTH) == int(TDuration::DurationType::V_QUARTER) + 1 && int(TDuration::DurationType::V_16TH) == int(TDuration::DurationType::V_EIGHTH) + 1 && int(TDuration::DurationType::V_32ND) == int(TDuration::DurationType::V_16TH) + 1 && int(TDuration::DurationType::V_64TH) == int(TDuration::DurationType::V_32ND) + 1 && int(TDuration::DurationType::V_128TH) == int(TDuration::DurationType::V_64TH) + 1 && int(TDuration::DurationType::V_256TH) == int(TDuration::DurationType::V_128TH) + 1 )) { qFatal("tupletAssert() failed"); } } #endif //--------------------------------------------------------- // smallestTypeAndCount //--------------------------------------------------------- /** Determine the smallest note type and the number of those present in a ChordRest. For a note without dots the type equals the note type and count is one. For a single dotted note the type equals half the note type and count is three. A double dotted note is similar. Note: code assumes when duration().type() is incremented, the note length is divided by two, checked by tupletAssert(). */ static void smallestTypeAndCount(ChordRest const* const cr, int& type, int& count) { type = int(cr->durationType().type()); count = 1; switch (cr->durationType().dots()) { case 0: // nothing to do break; case 1: type += 1; // next-smaller type count = 3; break; case 2: type += 2; // next-next-smaller type count = 7; break; default: qDebug("smallestTypeAndCount() does not support more than 2 dots"); } } //--------------------------------------------------------- // matchTypeAndCount //--------------------------------------------------------- /** Given two note types and counts, if the types are not equal, make them equal by successively doubling the count of the largest type. */ static void matchTypeAndCount(int& type1, int& count1, int& type2, int& count2) { while (type1 < type2) { type1++; count1 *= 2; } while (type2 < type1) { type2++; count2 *= 2; } } //--------------------------------------------------------- // determineTupletTypeAndCount //--------------------------------------------------------- /** Determine type and number of smallest notes in the tuplet */ static void determineTupletTypeAndCount(Tuplet* t, int& tupletType, int& tupletCount) { int elemCount = 0; // number of tuplet elements handled foreach (DurationElement* de, t->elements()) { if (de->type() == ElementType::CHORD || de->type() == ElementType::REST) { ChordRest* cr = static_cast(de); if (elemCount == 0) { // first note: init variables smallestTypeAndCount(cr, tupletType, tupletCount); } else { int noteType = 0; int noteCount = 0; smallestTypeAndCount(cr, noteType, noteCount); // match the types matchTypeAndCount(tupletType, tupletCount, noteType, noteCount); tupletCount += noteCount; } } elemCount++; } } //--------------------------------------------------------- // determineTupletBaseLen //--------------------------------------------------------- /** Determine tuplet baseLen as determined by the tuplet ratio, and type and number of smallest notes in the tuplet. Example: baselen of a 3:2 tuplet with 1/16, 1/8, 1/8 and 1/16 is 1/8. For this tuplet smallest note is 1/16, count is 6. */ // TODO: this is defined twice, remove one static TDuration determineTupletBaseLen(Tuplet* t) { int tupletType = 0; // smallest note type in the tuplet int tupletCount = 0; // number of smallest notes in the tuplet // first determine type and number of smallest notes in the tuplet determineTupletTypeAndCount(t, tupletType, tupletCount); // sanity check: // for a 3:2 tuplet, count must be a multiple of 3 if (tupletCount % t->ratio().numerator()) { qDebug("determineTupletBaseLen(%p) cannot divide count %d by %d", t, tupletCount, t->ratio().numerator()); return TDuration(); } // calculate baselen in smallest notes tupletCount /= t->ratio().numerator(); // normalize while (tupletCount > 1 && (tupletCount % 2) == 0) { tupletCount /= 2; tupletType -= 1; } return TDuration(TDuration::DurationType(tupletType)); } //--------------------------------------------------------- // isTupletFilled //--------------------------------------------------------- /** Determine if the tuplet contains the required number of notes, either (1) of the specified normal type or (2) the amount of the smallest notes in the tuplet equals actual notes. Example (1): a 3:2 tuplet with a 1/4 and a 1/8 note is filled if normal type is 1/8, it is not filled if normal type is 1/4. Example (2): a 3:2 tuplet with a 1/4 and a 1/8 note is filled. Use note types instead of duration to prevent errors due to rounding. */ // TODO: this is defined twice, remove one static bool isTupletFilled(Tuplet* t, TDuration normalType) { if (!t) return false; int tupletType = 0; // smallest note type in the tuplet int tupletCount = 0; // number of smallest notes in the tuplet // first determine type and number of smallest notes in the tuplet determineTupletTypeAndCount(t, tupletType, tupletCount); // then compare ... if (normalType.isValid()) { int matchedNormalType = int(normalType.type()); int matchedNormalCount = t->ratio().numerator(); // match the types matchTypeAndCount(tupletType, tupletCount, matchedNormalType, matchedNormalCount); // ... result scenario (1) return tupletCount >= matchedNormalCount; } else { // ... result scenario (2) return tupletCount >= t->ratio().numerator(); } } //--------------------------------------------------------- // addTupletToChord //--------------------------------------------------------- /** Handle tuplet(s) using parse result tupletDesc Tuplets with and but without are handled correctly. TODO Nested tuplets are not (yet) supported. Note that cr must be initialized: fields measure, score, tick and track are used. */ void addTupletToChord(ChordRest* cr, Tuplet*& tuplet, bool& tuplImpl, const Fraction& timeMod, const MusicXmlTupletDesc& tupletDesc, const TDuration normalType) { int actualNotes = timeMod.denominator(); int normalNotes = timeMod.numerator(); // check for obvious errors if (tupletDesc.type == MxmlStartStop::START && tuplet) { qDebug("tuplet already started"); // TODO // TODO: how to recover ? } if (tupletDesc.type == MxmlStartStop::STOP && !tuplet) { qDebug("tuplet stop but no tuplet started"); // TODO // TODO: how to recover ? } // Tuplet are either started by the tuplet start // or when the time modification is first found. if (!tuplet) { if (tupletDesc.type == MxmlStartStop::START || (!tuplet && (actualNotes != 1 || normalNotes != 1))) { if (tupletDesc.type != MxmlStartStop::START) { tuplImpl = true; // report missing start qDebug("implicit tuplet start cr %p tick %d track %d", cr, cr->tick(), cr->track()); // TODO } else tuplImpl = false; // create a new tuplet tuplet = new Tuplet(cr->score()); tuplet->setTrack(cr->track()); tuplet->setRatio(Fraction(actualNotes, normalNotes)); tuplet->setTick(cr->tick()); tuplet->setBracketType(tupletDesc.bracket); tuplet->setNumberType(tupletDesc.shownumber); // TODO type, placement, bracket tuplet->setParent(cr->measure()); } } // Add chord to the current tuplet. // Must also check for actual/normal notes to prevent // adding one chord too much if tuplet stop is missing. if (tuplet && !(actualNotes == 1 && normalNotes == 1)) { cr->setTuplet(tuplet); tuplet->add(cr); } // Tuplets are stopped by the tuplet stop // or when the tuplet is filled completely // (either with knowledge of the normal type // or as a last resort calculated based on // actual and normal notes plus total duration) // or when the time-modification is not found. if (tuplet) { if (tupletDesc.type == MxmlStartStop::STOP || (tuplImpl && isTupletFilled(tuplet, normalType)) || (actualNotes == 1 && normalNotes == 1)) { // set baselen TDuration td = determineTupletBaseLen(tuplet); // qDebug("stop tuplet %p basetype %d", tuplet, tupletType); tuplet->setBaseLen(td); Fraction f(normalNotes, td.fraction().denominator()); f.reduce(); tuplet->setDuration(f); // TODO determine usefulness of following check int totalDuration = 0; foreach (DurationElement* de, tuplet->elements()) { if (de->type() == ElementType::CHORD || de->type() == ElementType::REST) { totalDuration+=de->globalDuration().ticks(); } } if (!(totalDuration && normalNotes)) { qDebug("MusicXML::import: tuplet stop but bad duration"); // TODO } tuplet = 0; } } } //--------------------------------------------------------- // addArticulationToChord //--------------------------------------------------------- static void addArticulationToChord(ChordRest* cr, SymId articSym, QString dir) { Articulation* na = new Articulation(articSym, cr->score()); if (dir == "up") { na->setUp(true); na->setAnchor(ArticulationAnchor::TOP_STAFF); } else if (dir == "down") { na->setUp(false); na->setAnchor(ArticulationAnchor::BOTTOM_STAFF); } cr->add(na); } //--------------------------------------------------------- // addFermataToChord //--------------------------------------------------------- static void addFermataToChord(ChordRest* cr, SymId articSym, bool up) { Fermata* na = new Fermata(articSym, cr->score()); na->setTrack(cr->track()); na->setPlacement(up ? Placement::ABOVE : Placement::BELOW); cr->segment()->add(na); } //--------------------------------------------------------- // addMordentToChord //--------------------------------------------------------- /** Add Mordent to Chord. */ static void addMordentToChord(ChordRest* cr, QString name, QString attrLong, QString attrAppr, QString attrDep) { SymId articSym = SymId::noSym; // legal but impossible ArticulationType value here indicating "not found" if (name == "inverted-mordent") { if ((attrLong == "" || attrLong == "no") && attrAppr == "" && attrDep == "") articSym = SymId::ornamentMordent; else if (attrLong == "yes" && attrAppr == "" && attrDep == "") articSym = SymId::ornamentTremblement; else if (attrLong == "yes" && attrAppr == "below" && attrDep == "") articSym = SymId::ornamentUpPrall; else if (attrLong == "yes" && attrAppr == "above" && attrDep == "") articSym = SymId::ornamentPrecompMordentUpperPrefix; else if (attrLong == "yes" && attrAppr == "" && attrDep == "below") articSym = SymId::ornamentPrallDown; else if (attrLong == "yes" && attrAppr == "" && attrDep == "above") articSym = SymId::ornamentPrallUp; } else if (name == "mordent") { if ((attrLong == "" || attrLong == "no") && attrAppr == "" && attrDep == "") articSym = SymId::ornamentMordentInverted; else if (attrLong == "yes" && attrAppr == "" && attrDep == "") articSym = SymId::ornamentPrallMordent; else if (attrLong == "yes" && attrAppr == "below" && attrDep == "") articSym = SymId::ornamentUpMordent; else if (attrLong == "yes" && attrAppr == "above" && attrDep == "") articSym = SymId::ornamentDownMordent; } if (articSym != SymId::noSym) { Articulation* na = new Articulation(cr->score()); na->setSymId(articSym); cr->add(na); } else qDebug("unknown ornament: name '%s' long '%s' approach '%s' departure '%s'", qPrintable(name), qPrintable(attrLong), qPrintable(attrAppr), qPrintable(attrDep)); // TODO } //--------------------------------------------------------- // addMxmlArticulationToChord //--------------------------------------------------------- /** Add a MusicXML articulation to a chord as a "simple" MuseScore articulation. These are the articulations that can be - represented by an enum ArticulationType - added to a ChordRest Return true (articulation recognized and handled) or false (articulation not recognized). Note simple implementation: MusicXML syntax is not strictly checked, the articulations parent element does not matter. */ static bool addMxmlArticulationToChord(ChordRest* cr, QString mxmlName) { QMap map; // map MusicXML articulation name to MuseScore symbol map["accent"] = SymId::articAccentAbove; map["staccatissimo"] = SymId::articStaccatissimoAbove; map["staccato"] = SymId::articStaccatoAbove; map["tenuto"] = SymId::articTenutoAbove; map["turn"] = SymId::ornamentTurn; map["inverted-turn"] = SymId::ornamentTurnInverted; map["stopped"] = SymId::brassMuteClosed; // TODO map["harmonic"] = SymId::stringsHarmonic; map["up-bow"] = SymId::stringsUpBow; map["down-bow"] = SymId::stringsDownBow; map["detached-legato"] = SymId::articTenutoStaccatoAbove; map["spiccato"] = SymId::articStaccatissimoAbove; map["snap-pizzicato"] = SymId::pluckedSnapPizzicatoAbove; map["schleifer"] = SymId::ornamentPrecompSlide; map["open-string"] = SymId::brassMuteOpen; map["thumb-position"] = SymId::stringsThumbPosition; if (map.contains(mxmlName)) { addArticulationToChord(cr, map.value(mxmlName), ""); return true; } else return false; } //--------------------------------------------------------- // convertNotehead //--------------------------------------------------------- /** Convert a MusicXML notehead name to a MuseScore headgroup. */ static NoteHead::Group convertNotehead(QString mxmlName) { QMap map; // map MusicXML notehead name to a MuseScore headgroup map["slash"] = int(NoteHead::Group::HEAD_SLASH); map["triangle"] = int(NoteHead::Group::HEAD_TRIANGLE_UP); map["diamond"] = int(NoteHead::Group::HEAD_DIAMOND); map["cross"] = int(NoteHead::Group::HEAD_PLUS); map["x"] = int(NoteHead::Group::HEAD_CROSS); map["circle-x"] = int(NoteHead::Group::HEAD_XCIRCLE); map["inverted triangle"] = int(NoteHead::Group::HEAD_TRIANGLE_DOWN); map["slashed"] = int(NoteHead::Group::HEAD_SLASHED1); map["back slashed"] = int(NoteHead::Group::HEAD_SLASHED2); map["normal"] = int(NoteHead::Group::HEAD_NORMAL); map["do"] = int(NoteHead::Group::HEAD_DO); map["re"] = int(NoteHead::Group::HEAD_RE); map["mi"] = int(NoteHead::Group::HEAD_MI); map["fa"] = int(NoteHead::Group::HEAD_FA); map["fa up"] = int(NoteHead::Group::HEAD_FA); map["so"] = int(NoteHead::Group::HEAD_SOL); map["la"] = int(NoteHead::Group::HEAD_LA); map["ti"] = int(NoteHead::Group::HEAD_TI); if (map.contains(mxmlName)) return NoteHead::Group(map.value(mxmlName)); else qDebug("unknown notehead %s", qPrintable(mxmlName)); // TODO // default: return 0 return NoteHead::Group::HEAD_NORMAL; } //--------------------------------------------------------- // addTextToNote //--------------------------------------------------------- /** Add Text to Note. */ static void addTextToNote(int l, int c, QString txt, Tid style, Score* score, Note* note) { if (note) { if (!txt.isEmpty()) { TextBase* t = new Fingering(score, style); t->setPlainText(txt); note->add(t); } } else qDebug("%s", qPrintable(QString("Error at line %1 col %2: no note for text").arg(l).arg(c))); // TODO } //--------------------------------------------------------- // addFermata //--------------------------------------------------------- /** Add a MusicXML fermata. Note: MusicXML common.mod: "The fermata type is upright if not specified." */ static void addFermata(ChordRest* cr, const QString type, const SymId articSym) { if (type == "upright" || type == "") addFermataToChord(cr, articSym, true); else if (type == "inverted") addFermataToChord(cr, articSym, false); else qDebug("unknown fermata type '%s'", qPrintable(type)); } //--------------------------------------------------------- // setSLinePlacement //--------------------------------------------------------- /** Helper for direction(). SLine placement is modified by changing the first segments user offset As the SLine has just been created, it does not have any segment yet */ static void setSLinePlacement(SLine* sli, const QString placement) { /* qDebug("setSLinePlacement sli %p type %d s=%g pl='%s'", sli, sli->type(), sli->score()->spatium(), qPrintable(placement)); */ #if 0 // calc y offset assuming five line staff and default style // note that required y offset is element type dependent if (sli->type() == ElementType::HAIRPIN) { if (placement == "above") { const qreal stafflines = 5; // assume five line staff, but works OK-ish for other sizes too qreal offsAbove = -6 - (stafflines - 1); qreal y = 0; y += offsAbove; // add linesegment containing the user offset LineSegment* tls= sli->createLineSegment(); //qDebug(" y = %g", y); tls->setAutoplace(false); y *= sli->score()->spatium(); tls->setUserOff(QPointF(0, y)); sli->add(tls); } } else { sli->setPlacement(placement == "above" ? Placement::ABOVE : Placement::BELOW); } #endif sli->setPlacement(placement == "above" ? Placement::ABOVE : Placement::BELOW); } //--------------------------------------------------------- // handleSpannerStart //--------------------------------------------------------- // note that in case of overlapping spanners, handleSpannerStart is called for every spanner // as spanners QMap allows only one value per key, this does not hurt at all static void handleSpannerStart(SLine* new_sp, int track, QString& placement, int tick, MusicXmlSpannerMap& spanners) { //qDebug("handleSpannerStart(sp %p, track %d, tick %d)", new_sp, track, tick); new_sp->setTrack(track); setSLinePlacement(new_sp, placement); spanners[new_sp] = QPair(tick, -1); } //--------------------------------------------------------- // handleSpannerStop //--------------------------------------------------------- static void handleSpannerStop(SLine* cur_sp, int track2, int tick, MusicXmlSpannerMap& spanners) { //qDebug("handleSpannerStop(sp %p, track2 %d, tick %d)", cur_sp, track2, tick); if (!cur_sp) return; cur_sp->setTrack2(track2); spanners[cur_sp].second = tick; } //--------------------------------------------------------- // The MusicXML parser, pass 2 //--------------------------------------------------------- //--------------------------------------------------------- // MusicXMLParserPass2 //--------------------------------------------------------- MusicXMLParserPass2::MusicXMLParserPass2(Score* score, MusicXMLParserPass1& pass1, MxmlLogger* logger) : _divs(0), _score(score), _pass1(pass1), _logger(logger) { // nothing } //--------------------------------------------------------- // initPartState //--------------------------------------------------------- /** Initialize members as required for reading the MusicXML part element. TODO: factor out part reading into a separate class TODO: preferably use automatically initialized variables Note that Qt automatically initializes new elements in QVector (tuplets). */ void MusicXMLParserPass2::initPartState(const QString& partId) { _timeSigDura = Fraction(0, 0); // invalid int nstaves = _pass1.getPart(partId)->nstaves(); _tuplets.resize(nstaves * VOICES); _tuplImpls.resize(nstaves * VOICES); _tie = 0; _lastVolta = 0; _hasDrumset = false; for (int i = 0; i < MAX_NUMBER_LEVEL; ++i) _slurs[i] = SlurDesc(); for (int i = 0; i < MAX_BRACKETS; ++i) _brackets[i] = 0; for (int i = 0; i < MAX_DASHES; ++i) _dashes[i] = 0; for (int i = 0; i < MAX_NUMBER_LEVEL; ++i) _ottavas[i] = 0; for (int i = 0; i < MAX_NUMBER_LEVEL; ++i) _hairpins[i] = 0; for (int i = 0; i < MAX_NUMBER_LEVEL; ++i) _trills[i] = 0; for (int i = 0; i < MAX_NUMBER_LEVEL; ++i) _glissandi[i][0] = _glissandi[i][1] = 0; _pedal = 0; _pedalContinue = 0; _harmony = 0; _tremStart = 0; _figBass = 0; // glissandoText = ""; // glissandoColor = ""; _multiMeasureRestCount = -1; _extendedLyrics.init(); } //--------------------------------------------------------- // multi-measure rest state handling //--------------------------------------------------------- // If any multi-measure rest is found, the "create multi-measure rest" style setting is enabled. // First measure in a multi-measure rest gets setBreakMultiMeasureRest(true), then count down // the remaining number of measures. // The first measure after a multi-measure rest gets setBreakMultiMeasureRest(true). // For all other measures breakMultiMeasureRest is unchanged (stays default (false)). //--------------------------------------------------------- // setMultiMeasureRestCount //--------------------------------------------------------- /** Set the multi-measure rest counter. */ void MusicXMLParserPass2::setMultiMeasureRestCount(int count) { _multiMeasureRestCount = count; } //--------------------------------------------------------- // getAndDecMultiMeasureRestCount //--------------------------------------------------------- /** Return current multi-measure rest counter. Decrement counter if possible (not beyond -1). */ int MusicXMLParserPass2::getAndDecMultiMeasureRestCount() { int res = _multiMeasureRestCount; if (_multiMeasureRestCount >= 0) _multiMeasureRestCount--; return res; } //--------------------------------------------------------- // skipLogCurrElem //--------------------------------------------------------- /** Skip the current element, log debug as info. */ void MusicXMLParserDirection::skipLogCurrElem() { //_logger->logDebugInfo(QString("skipping '%1'").arg(_e.name().toString()), &_e); _e.skipCurrentElement(); } //--------------------------------------------------------- // skipLogCurrElem //--------------------------------------------------------- /** Skip the current element, log debug as info. */ void MusicXMLParserPass2::skipLogCurrElem() { //_logger->logDebugInfo(QString("skipping '%1'").arg(_e.name().toString()), &_e); _e.skipCurrentElement(); } //--------------------------------------------------------- // parse //--------------------------------------------------------- /** Parse MusicXML in \a device and extract pass 2 data. */ Score::FileError MusicXMLParserPass2::parse(QIODevice* device) { //qDebug("MusicXMLParserPass2::parse()"); _e.setDevice(device); Score::FileError res = parse(); //qDebug("MusicXMLParserPass2::parse() res %d", int(res)); return res; } //--------------------------------------------------------- // parse //--------------------------------------------------------- /** Start the parsing process, after verifying the top-level node is score-partwise */ Score::FileError MusicXMLParserPass2::parse() { bool found = false; while (_e.readNextStartElement()) { if (_e.name() == "score-partwise") { found = true; scorePartwise(); } else { _logger->logError("this is not a MusicXML score-partwise file", &_e); _e.skipCurrentElement(); return Score::FileError::FILE_BAD_FORMAT; } } if (!found) { _logger->logError("this is not a MusicXML score-partwise file", &_e); return Score::FileError::FILE_BAD_FORMAT; } return Score::FileError::FILE_NO_ERROR; } //--------------------------------------------------------- // scorePartwise //--------------------------------------------------------- /** Parse the MusicXML top-level (XPath /score-partwise) node. */ void MusicXMLParserPass2::scorePartwise() { Q_ASSERT(_e.isStartElement() && _e.name() == "score-partwise"); while (_e.readNextStartElement()) { if (_e.name() == "part") { part(); } else if (_e.name() == "part-list") partList(); else skipLogCurrElem(); } // set last measure barline to normal or MuseScore will generate light-heavy EndBarline // TODO, handle other tracks? if (_score->lastMeasure()->endBarLineType() == BarLineType::NORMAL) _score->lastMeasure()->setEndBarLineType(BarLineType::NORMAL, 0); } //--------------------------------------------------------- // partList //--------------------------------------------------------- /** Parse the /score-partwise/part-list node. */ void MusicXMLParserPass2::partList() { Q_ASSERT(_e.isStartElement() && _e.name() == "part-list"); while (_e.readNextStartElement()) { if (_e.name() == "score-part") scorePart(); else skipLogCurrElem(); } } //--------------------------------------------------------- // scorePart //--------------------------------------------------------- // Parse the /score-partwise/part-list/score-part node. // TODO: nothing required for pass 2 ? void MusicXMLParserPass2::scorePart() { Q_ASSERT(_e.isStartElement() && _e.name() == "score-part"); while (_e.readNextStartElement()) { if (_e.name() == "midi-instrument") _e.skipCurrentElement(); // skip but don't log else if (_e.name() == "score-instrument") _e.skipCurrentElement(); // skip but don't log else if (_e.name() == "part-name") _e.skipCurrentElement(); // skip but don't log else skipLogCurrElem(); } } //--------------------------------------------------------- // part //--------------------------------------------------------- /** Parse the /score-partwise/part node. */ void MusicXMLParserPass2::part() { Q_ASSERT(_e.isStartElement() && _e.name() == "part"); const QString id = _e.attributes().value("id").toString(); if (!_pass1.hasPart(id)) { _logger->logError(QString("MusicXMLParserPass2::part cannot find part '%1'").arg(id), &_e); skipLogCurrElem(); } initPartState(id); const MusicXMLDrumset& mxmlDrumset = _pass1.getDrumset(id); _hasDrumset = hasDrumset(mxmlDrumset); // set the parts first instrument QString instrId = _pass1.getInstrList(id).instrument(Fraction(0, 1)); setFirstInstrument(_logger, &_e, _pass1.getPart(id), id, instrId, mxmlDrumset); // set the part name auto mxmlPart = _pass1.getMusicXmlPart(id); _pass1.getPart(id)->setPartName(mxmlPart.getName()); if (mxmlPart.getPrintName()) _pass1.getPart(id)->setLongName(mxmlPart.getName()); if (mxmlPart.getPrintAbbr()) _pass1.getPart(id)->setPlainShortName(mxmlPart.getAbbr()); // try to prevent an empty track name if (_pass1.getPart(id)->partName() == "") _pass1.getPart(id)->setPartName(mxmlDrumset[instrId].name); #ifdef DEBUG_VOICE_MAPPER VoiceList voicelist = _pass1.getVoiceList(id); // debug: print voice mapper contents qDebug("voiceMapperStats: part '%s'", qPrintable(id)); for (QMap::const_iterator i = voicelist.constBegin(); i != voicelist.constEnd(); ++i) { qDebug("voiceMapperStats: voice %s staff data %s", qPrintable(i.key()), qPrintable(i.value().toString())); } #endif // read the measures int nr = 0; // current measure sequence number while (_e.readNextStartElement()) { if (_e.name() == "measure") { Fraction t = _pass1.getMeasureStart(nr); if (t.isValid()) measure(id, t); else { _logger->logError(QString("no valid start time for measure %1").arg(nr + 1), &_e); _e.skipCurrentElement(); } ++nr; } else skipLogCurrElem(); } // stop all remaining extends for this part Measure* lm = _pass1.getPart(id)->score()->lastMeasure(); if (lm) { int strack = _pass1.trackForPart(id); int etrack = strack + _pass1.getPart(id)->nstaves() * VOICES; int lastTick = lm->tick() + lm->ticks(); for (int trk = strack; trk < etrack; trk++) _extendedLyrics.setExtend(-1, trk, lastTick); } //qDebug("spanner list:"); auto i = _spanners.constBegin(); while (i != _spanners.constEnd()) { Spanner* sp = i.key(); int tick1 = i.value().first; int tick2 = i.value().second; //qDebug("spanner %p tp %hhd tick1 %d tick2 %d track %d track2 %d", // sp, sp->type(), tick1, tick2, sp->track(), sp->track2()); sp->setTick(tick1); sp->setTick2(tick2); sp->score()->addElement(sp); ++i; } _spanners.clear(); // determine if the part contains a drumset // this is the case if any instrument has a midi-unpitched element, // (which stored in the MusicXMLDrumInstrument pitch field) // if the part contains a drumset, Drumset drumset is initialized Drumset* drumset = new Drumset; const MusicXMLDrumset& mxmlDrumsetAfterPass2 = _pass1.getDrumset(id); initDrumset(drumset, mxmlDrumsetAfterPass2); // debug: dump the instrument map /* { qDebug("instrlist"); auto il = _pass1.getInstrList(id); for (auto it = il.cbegin(); it != il.cend(); ++it) { Fraction f = (*it).first; qDebug("pass2: instrument map: tick %s (%d) instr '%s'", qPrintable(f.print()), f.ticks(), qPrintable((*it).second)); } } */ if (_hasDrumset) { // set staff type to percussion if incorrectly imported as pitched staff // Note: part has been read, staff type already set based on clef type and staff-details // but may be incorrect for a percussion staff that does not use a percussion clef setStaffTypePercussion(_pass1.getPart(id), drumset); } else { // drumset is not needed delete drumset; // set the instruments for this part setPartInstruments(_logger, &_e, _pass1.getPart(id), id, _score, _pass1.getInstrList(id), mxmlDrumset); } } //--------------------------------------------------------- // findMeasure //--------------------------------------------------------- /** In Score \a score find the measure starting at \a tick. */ static Measure* findMeasure(Score* score, const int tick) { for (Measure* m = score->firstMeasure(); m; m = m->nextMeasure()) { if (m->tick() == tick) return m; } return 0; } //--------------------------------------------------------- // removeBeam //--------------------------------------------------------- /** Set beam mode for all elements and remove the beam */ static void removeBeam(Beam*& beam) { for (int i = 0; i < beam->elements().size(); ++i) beam->elements().at(i)->setBeamMode(Beam::Mode::NONE); delete beam; beam = 0; } //--------------------------------------------------------- // handleBeamAndStemDir //--------------------------------------------------------- static void handleBeamAndStemDir(ChordRest* cr, const Beam::Mode bm, const Direction sd, Beam*& beam) { if (!cr) return; // create a new beam if (bm == Beam::Mode::BEGIN) { // if currently in a beam, delete it if (beam) { qDebug("handleBeamAndStemDir() new beam, removing previous incomplete beam %p", beam); removeBeam(beam); } // create a new beam beam = new Beam(cr->score()); beam->setTrack(cr->track()); beam->setBeamDirection(sd); } // add ChordRest to beam if (beam) { // verify still in the same track (switching voices in the middle of a beam is not supported) // and in a beam ... // (note no check is done on correct order of beam begin/continue/end) if (cr->track() != beam->track()) { qDebug("handleBeamAndStemDir() from track %d to track %d -> abort beam", beam->track(), cr->track()); // reset beam mode for all elements and remove the beam removeBeam(beam); } else if (bm == Beam::Mode::NONE) { qDebug("handleBeamAndStemDir() in beam, bm Beam::Mode::NONE -> abort beam"); // reset beam mode for all elements and remove the beam removeBeam(beam); } else if (!(bm == Beam::Mode::BEGIN || bm == Beam::Mode::MID || bm == Beam::Mode::END)) { qDebug("handleBeamAndStemDir() in beam, bm %d -> abort beam", static_cast(bm)); // reset beam mode for all elements and remove the beam removeBeam(beam); } else { // actually add cr to the beam beam->add(cr); } } // if no beam, set stem direction on chord itself and set beam to auto if (!beam) { static_cast(cr)->setStemDirection(sd); cr->setBeamMode(Beam::Mode::AUTO); } // terminate the currect beam and add to the score if (beam && bm == Beam::Mode::END) beam = 0; } //--------------------------------------------------------- // markUserAccidentals //--------------------------------------------------------- /** Check for "superfluous" accidentals to mark them as USER accidentals. The candidate map alterMap is ordered on note address. Check it here segment after segment. */ static void markUserAccidentals(const int firstStaff, const int staves, const Key key, const Measure* measure, const QMap& alterMap ) { QMap accTmp; AccidentalState currAcc; currAcc.init(key); SegmentType st = SegmentType::ChordRest; for (Ms::Segment* segment = measure->first(st); segment; segment = segment->next(st)) { for (int track = 0; track < staves * VOICES; ++track) { Element* e = segment->element(firstStaff * VOICES + track); if (!e || e->type() != Ms::ElementType::CHORD) continue; Chord* chord = static_cast(e); foreach (Note* nt, chord->notes()) { if (alterMap.contains(nt)) { int alter = alterMap.value(nt); int ln = absStep(nt->tpc(), nt->pitch()); bool error = false; AccidentalVal currAccVal = currAcc.accidentalVal(ln, error); if (error) continue; if ((alter == -1 && currAccVal == AccidentalVal::FLAT && nt->accidental()->accidentalType() == AccidentalType::FLAT && !accTmp.value(ln, false)) || (alter == 0 && currAccVal == AccidentalVal::NATURAL && nt->accidental()->accidentalType() == AccidentalType::NATURAL && !accTmp.value(ln, false)) || (alter == 1 && currAccVal == AccidentalVal::SHARP && nt->accidental()->accidentalType() == AccidentalType::SHARP && !accTmp.value(ln, false))) { nt->accidental()->setRole(AccidentalRole::USER); } else if (Accidental::isMicrotonal(nt->accidental()->accidentalType()) && nt->accidental()->accidentalType() < AccidentalType::END) { // microtonal accidental nt->accidental()->setRole(AccidentalRole::USER); accTmp.insert(ln, false); } else { accTmp.insert(ln, true); } } } } } } //--------------------------------------------------------- // addGraceChordsAfter //--------------------------------------------------------- /** Move \a gac grace chords from grace chord list \a gcl to the chord \a c grace note after list */ static void addGraceChordsAfter(Chord* c, GraceChordList& gcl, int& gac) { if (!c) return; while (gac > 0) { if (gcl.size() > 0) { Chord* graceChord = gcl.first(); gcl.removeFirst(); graceChord->toGraceAfter(); c->add(graceChord); // TODO check if same voice ? qDebug("addGraceChordsAfter chord %p grace after chord %p", c, graceChord); } gac--; } } //--------------------------------------------------------- // addGraceChordsBefore //--------------------------------------------------------- /** Move grace chords from grace chord list \a gcl to the chord \a c grace note before list */ static void addGraceChordsBefore(Chord* c, GraceChordList& gcl) { for (int i = gcl.size() - 1; i >= 0; i--) c->add(gcl.at(i)); // TODO check if same voice ? gcl.clear(); } //--------------------------------------------------------- // measure //--------------------------------------------------------- /** Parse the /score-partwise/part/measure node. */ void MusicXMLParserPass2::measure(const QString& partId, const Fraction time) { Q_ASSERT(_e.isStartElement() && _e.name() == "measure"); QString number = _e.attributes().value("number").toString(); //qDebug("measure %s start", qPrintable(number)); Measure* measure = findMeasure(_score, time.ticks()); if (!measure) { _logger->logError(QString("measure at tick %1 not found!").arg(time.ticks()), &_e); skipLogCurrElem(); } // handle implicit measure if (_e.attributes().value("implicit") == "yes") measure->setIrregular(true); // set measure's RepeatFlag to none because musicXML is allowing single measure repeat and no ordering in repeat start and end barlines measure->setRepeatStart(false); measure->setRepeatEnd(false); Fraction mTime; // current time stamp within measure Fraction prevTime; // time stamp within measure previous chord Chord* prevChord = 0; // previous chord Fraction mDura; // current total measure duration GraceChordList gcl; // grace chords collected sofar int gac = 0; // grace after count in the grace chord list Beam* beam = 0; // current beam QString cv = "1"; // current voice for chords, default is 1 FiguredBassList fbl; // List of figured bass elements under a single note // collect candidates for courtesy accidentals to work out at measure end QMap alterMap; while (_e.readNextStartElement()) { if (_e.name() == "attributes") attributes(partId, measure, (time + mTime).ticks()); else if (_e.name() == "direction") { MusicXMLParserDirection dir(_e, _score, _pass1, *this, _logger); dir.direction(partId, measure, (time + mTime).ticks(), _spanners); } else if (_e.name() == "figured-bass") { FiguredBass* fb = figuredBass(); if (fb) fbl.append(fb); } else if (_e.name() == "harmony") harmony(partId, measure, time + mTime); else if (_e.name() == "note") { Fraction dura; int alt = -10; // any number outside range of xml-tag "alter" // note: chord and grace note handling done in note() // dura > 0 iff valid rest or first note of chord found Note* n = note(partId, measure, time + mTime, time + prevTime, dura, cv, gcl, gac, beam, fbl, alt); if (n && !n->chord()->isGrace()) prevChord = n->chord(); // remember last non-grace chord if (n && n->accidental() && n->accidental()->accidentalType() != AccidentalType::NONE) alterMap.insert(n, alt); if (dura.isValid() && dura > Fraction(0, 1)) { prevTime = mTime; // save time stamp last chord created mTime += dura; if (mTime > mDura) mDura = mTime; } //qDebug("added note %p chord %p gac %d", n, n ? n->chord() : 0, gac); } else if (_e.name() == "forward") { Fraction dura; forward(dura); if (dura.isValid()) { mTime += dura; if (mTime > mDura) mDura = mTime; } } else if (_e.name() == "backup") { Fraction dura; backup(dura); if (dura.isValid()) { if (dura <= mTime) mTime -= dura; else { _logger->logError("backup beyond measure start", &_e); mTime.set(0, 1); } } } else if (_e.name() == "sound") { QString tempo = _e.attributes().value("tempo").toString(); if (!tempo.isEmpty()) { double tpo = tempo.toDouble() / 60; int tick = (time + mTime).ticks(); TempoText* t = new TempoText(_score); t->setXmlText(QString("%1 = %2").arg(TempoText::duration2tempoTextString(TDuration(TDuration::DurationType::V_QUARTER))).arg(tempo)); t->setTempo(tpo); t->setFollowText(true); _score->setTempo(tick, tpo); addElemOffset(t, _pass1.trackForPart(partId), "above", measure, tick); } _e.skipCurrentElement(); } else if (_e.name() == "barline") barline(partId, measure); else if (_e.name() == "print") print(measure); else skipLogCurrElem(); /* qDebug("mTime %s (%s) mDura %s (%s)", qPrintable(mTime.print()), qPrintable(mTime.reduced().print()), qPrintable(mDura.print()), qPrintable(mDura.reduced().print())); */ mDura.reduce(); mTime.reduce(); } // convert remaining grace chords to grace after gac = gcl.size(); addGraceChordsAfter(prevChord, gcl, gac); // fill possible gaps in voice 1 Part* part = _pass1.getPart(partId); // should not fail, we only get here if the part exists fillGapsInFirstVoices(measure, part); // can't have beams extending into the next measure if (beam) removeBeam(beam); // TODO: // - how to handle _timeSigDura.isZero (shouldn't happen ?) // - how to handle unmetered music if (_timeSigDura.isValid() && !_timeSigDura.isZero()) measure->setTimesig(_timeSigDura); // mark superfluous accidentals as user accidentals const int scoreRelStaff = _score->staffIdx(part); const Key key = _score->staff(scoreRelStaff)->keySigEvent(time.ticks()).key(); markUserAccidentals(scoreRelStaff, part->nstaves(), key, measure, alterMap); // multi-measure rest handling if (getAndDecMultiMeasureRestCount() == 0) { // measure is first measure after a multi-measure rest measure->setBreakMultiMeasureRest(true); } Q_ASSERT(_e.isEndElement() && _e.name() == "measure"); } //--------------------------------------------------------- // attributes //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes node. */ /* Notes: * Number of staves has already been set in pass 1 * MusicXML order is key, time, clef * -> check if it is necessary to insert them in order */ void MusicXMLParserPass2::attributes(const QString& partId, Measure* measure, const int tick) { Q_ASSERT(_e.isStartElement() && _e.name() == "attributes"); while (_e.readNextStartElement()) { if (_e.name() == "clef") clef(partId, measure, tick); else if (_e.name() == "divisions") divisions(); else if (_e.name() == "key") key(partId, measure, tick); else if (_e.name() == "measure-style") measureStyle(measure); else if (_e.name() == "staff-details") staffDetails(partId); else if (_e.name() == "time") time(partId, measure, tick); else if (_e.name() == "transpose") transpose(partId); else skipLogCurrElem(); } } //--------------------------------------------------------- // setStaffLines //--------------------------------------------------------- /** Set stafflines and barline span for a single staff */ static void setStaffLines(Score* score, int staffIdx, int stafflines) { score->staff(staffIdx)->setLines(0, stafflines); score->staff(staffIdx)->setBarLineTo(0); // default } //--------------------------------------------------------- // staffDetails //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/staff-details node. */ void MusicXMLParserPass2::staffDetails(const QString& partId) { Q_ASSERT(_e.isStartElement() && _e.name() == "staff-details"); //logDebugTrace("MusicXMLParserPass2::staffDetails"); Part* part = _pass1.getPart(partId); Q_ASSERT(part); int staves = part->nstaves(); QString number = _e.attributes().value("number").toString(); int n = 1; // default if (number != "") { n = number.toInt(); if (n <= 0 || n > staves) { _logger->logError(QString("invalid staff-details number %1").arg(number), &_e); n = 1; } } n--; // make zero-based int staffIdx = _score->staffIdx(part) + n; StringData* t = 0; if (_score->staff(staffIdx)->isTabStaff(0)) { t = new StringData; t->setFrets(25); // sensible default } int staffLines = 0; while (_e.readNextStartElement()) { if (_e.name() == "staff-lines") { // save staff lines for later staffLines = _e.readElementText().toInt(); // for a TAB staff also resize the string table and init with zeroes if (t) { if (0 < staffLines) t->stringList() = QVector(staffLines).toList(); else _logger->logError(QString("illegal staff-lines %1").arg(staffLines), &_e); } } else if (_e.name() == "staff-tuning") staffTuning(t); else skipLogCurrElem(); } if (staffLines > 0) { setStaffLines(_score, staffIdx, staffLines); } if (t) { Instrument* i = part->instrument(); i->setStringData(*t); } } //--------------------------------------------------------- // staffTuning //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/staff-details/staff-tuning node. */ void MusicXMLParserPass2::staffTuning(StringData* t) { Q_ASSERT(_e.isStartElement() && _e.name() == "staff-tuning"); //logDebugTrace("MusicXMLParserPass2::staffTuning"); // ignore if not a TAB staff if (!t) { _logger->logError(" on non-TAB staff", &_e); skipLogCurrElem(); return; } int line = _e.attributes().value("line").toInt(); int step = 0; int alter = 0; int octave = 0; while (_e.readNextStartElement()) { if (_e.name() == "tuning-alter") alter = _e.readElementText().toInt(); else if (_e.name() == "tuning-octave") octave = _e.readElementText().toInt(); else if (_e.name() == "tuning-step") { QString strStep = _e.readElementText(); int pos = QString("CDEFGAB").indexOf(strStep); if (strStep.size() == 1 && pos >=0 && pos < 7) step = pos; else _logger->logError(QString("invalid step '%1'").arg(strStep), &_e); } else skipLogCurrElem(); } if (0 < line && line <= t->stringList().size()) { int pitch = MusicXMLStepAltOct2Pitch(step, alter, octave); if (pitch >= 0) t->stringList()[line - 1].pitch = pitch; else _logger->logError(QString("invalid string %1 tuning step/alter/oct %2/%3/%4") .arg(line).arg(step).arg(alter).arg(octave), &_e); } } //--------------------------------------------------------- // measureStyle //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/measure-style node. Initializes the "in multi-measure rest" state */ void MusicXMLParserPass2::measureStyle(Measure* measure) { Q_ASSERT(_e.isStartElement() && _e.name() == "measure-style"); while (_e.readNextStartElement()) { if (_e.name() == "multiple-rest") { int multipleRest = _e.readElementText().toInt(); if (multipleRest > 1) { _multiMeasureRestCount = multipleRest; _score->style().set(Sid::createMultiMeasureRests, true); measure->setBreakMultiMeasureRest(true); } else _logger->logError(QString("multiple-rest %1 not supported").arg(multipleRest), &_e); } else skipLogCurrElem(); } } //--------------------------------------------------------- // print //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/print node. */ void MusicXMLParserPass2::print(Measure* measure) { Q_ASSERT(_e.isStartElement() && _e.name() == "print"); bool newSystem = _e.attributes().value("new-system") == "yes"; bool newPage = _e.attributes().value("new-page") == "yes"; int blankPage = _e.attributes().value("blank-page").toInt(); // // in MScore the break happens _after_ the marked measure: // MeasureBase* pm = measure->prevMeasure(); // We insert VBox only for title, no HBox for the moment if (pm == 0) { _logger->logDebugInfo("break on first measure", &_e); if (blankPage == 1) { // blank title page, insert a VBOX if needed pm = measure->prev(); if (pm == 0) { _score->insertMeasure(ElementType::VBOX, measure); pm = measure->prev(); } } } if (pm) { if (preferences.getBool(PREF_IMPORT_MUSICXML_IMPORTBREAKS) && (newSystem || newPage)) { if (!pm->lineBreak() && !pm->pageBreak()) { LayoutBreak* lb = new LayoutBreak(_score); lb->setLayoutBreakType(newSystem ? LayoutBreak::Type::LINE : LayoutBreak::Type::PAGE); pm->add(lb); } } } while (_e.readNextStartElement()) { skipLogCurrElem(); } } //--------------------------------------------------------- // direction //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction node. */ void MusicXMLParserDirection::direction(const QString& partId, Measure* measure, const int tick, MusicXmlSpannerMap& spanners) { Q_ASSERT(_e.isStartElement() && _e.name() == "direction"); //qDebug("direction tick %d", tick); QString placement = _e.attributes().value("placement").toString(); int track = _pass1.trackForPart(partId); //qDebug("direction track %d", track); QList starts; QList stops; // note: file order is direction-type first, then staff // this means staff is still unknown when direction-type is handled // easiest solution is to put spanners on a stop and start list // and handle these after the while loop // note that placement is a attribute (which is currently supported by MS) // but the children also have formatting attributes // (currently NOT supported by MS, at least not for spanners when children) while (_e.readNextStartElement()) { if (_e.name() == "direction-type") directionType(starts, stops); else if (_e.name() == "staff") { int nstaves = _pass1.getPart(partId)->nstaves(); QString strStaff = _e.readElementText(); int staff = strStaff.toInt(); if (0 < staff && staff <= nstaves) track += (staff - 1) * VOICES; else _logger->logError(QString("invalid staff %1").arg(strStaff), &_e); } else if (_e.name() == "sound") sound(); else skipLogCurrElem(); } handleRepeats(measure, track); // fix for Sibelius 7.1.3 (direct export) which creates metronomes without : // if necessary, use the value calculated by metronome() // note: no floating point comparisons with 0 ... if (_tpoSound < 0.1 && _tpoMetro > 0.1) _tpoSound = _tpoMetro; //qDebug("words '%s' rehearsal '%s' metro '%s' tpo %g", // qPrintable(_wordsText), qPrintable(_rehearsalText), qPrintable(_metroText), _tpoSound); // create text if any text was found if (_wordsText != "" || _rehearsalText != "" || _metroText != "") { TextBase* t = 0; if (_tpoSound > 0.1) { _tpoSound /= 60; t = new TempoText(_score); t->setXmlText(_wordsText + _metroText); ((TempoText*) t)->setTempo(_tpoSound); ((TempoText*) t)->setFollowText(true); _score->setTempo(tick, _tpoSound); } else { if (_wordsText != "" || _metroText != "") { t = new StaffText(_score); t->setXmlText(_wordsText + _metroText); } else { t = new RehearsalMark(_score); if (!_rehearsalText.contains("")) _rehearsalText = "" + _rehearsalText; // explicitly turn bold off t->setXmlText(_rehearsalText); if (!_hasDefaultY) t->setPlacement(Placement::ABOVE); // crude way to force placement TODO improve ? } } if (_enclosure == "circle") { t->setFrameType(FrameType::CIRCLE); } else if (_enclosure == "none") { t->setFrameType(FrameType::NO_FRAME); } else if (_enclosure == "rectangle") { t->setFrameType(FrameType::SQUARE); t->setFrameRound(0); } //TODO:ws if (_hasDefaultY) t->textStyle().setYoff(_defaultY); addElemOffset(t, track, placement, measure, tick); } else if (_tpoSound > 0) { double tpo = _tpoSound / 60; TempoText* t = new TempoText(_score); t->setXmlText(QString("%1 = %2").arg(TempoText::duration2tempoTextString(TDuration(TDuration::DurationType::V_QUARTER))).arg(_tpoSound)); t->setTempo(tpo); t->setFollowText(true); _score->setTempo(tick, tpo); addElemOffset(t, track, placement, measure, tick); } // do dynamics // LVIFIX: check import/export of unknown_text for (QStringList::Iterator it = _dynamicsList.begin(); it != _dynamicsList.end(); ++it ) { Dynamic* dyn = new Dynamic(_score); dyn->setDynamicType(*it); if (!_dynaVelocity.isEmpty()) { int dynaValue = round(_dynaVelocity.toDouble() * 0.9); if (dynaValue > 127) dynaValue = 127; else if (dynaValue < 0) dynaValue = 0; dyn->setVelocity( dynaValue ); } //TODO:ws if (_hasDefaultY) dyn->textStyle().setYoff(_defaultY); addElemOffset(dyn, track, placement, measure, tick); } // handle the elems foreach( auto elem, _elems) { // TODO (?) if (_hasDefaultY) elem->setYoff(_defaultY); addElemOffset(elem, track, placement, measure, tick); } // handle the spanner stops first foreach (auto desc, stops) { SLine* sp = _pass2.getSpanner(desc); if (sp) { handleSpannerStop(sp, track, tick, spanners); _pass2.clearSpanner(desc); } else _logger->logError("spanner stop without spanner start", &_e); } // then handle the spanner starts foreach (auto desc, starts) { SLine* sp = _pass2.getSpanner(desc); if (!sp) { _pass2.addSpanner(desc); handleSpannerStart(desc.sp, track, placement, tick, spanners); } else _logger->logError("spanner already started", &_e); } Q_ASSERT(_e.isEndElement() && _e.name() == "direction"); } //--------------------------------------------------------- // directionType //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type node. */ void MusicXMLParserDirection::directionType(QList& starts, QList& stops) { Q_ASSERT(_e.isStartElement() && _e.name() == "direction-type"); while (_e.readNextStartElement()) { _defaultY = _e.attributes().value("default-y").toDouble(&_hasDefaultY) * -0.1; QString number = _e.attributes().value("number").toString(); int n = 0; if (number != "") { n = number.toInt(); if (n <= 0) _logger->logError(QString("invalid number %1").arg(number), &_e); else n--; // make zero-based } QString type = _e.attributes().value("type").toString(); if (_e.name() == "metronome") _metroText = metronome(_tpoMetro); else if (_e.name() == "words") { _enclosure = _e.attributes().value("enclosure").toString(); _wordsText += nextPartOfFormattedString(_e); } else if (_e.name() == "rehearsal") { _enclosure = _e.attributes().value("enclosure").toString(); if (_enclosure == "") _enclosure = "square"; // note different default _rehearsalText += nextPartOfFormattedString(_e); } else if (_e.name() == "pedal") pedal(type, n, starts, stops); else if (_e.name() == "octave-shift") octaveShift(type, n, starts, stops); else if (_e.name() == "dynamics") dynamics(); else if (_e.name() == "bracket") bracket(type, n, starts, stops); else if (_e.name() == "dashes") dashes(type, n, starts, stops); else if (_e.name() == "wedge") wedge(type, n, starts, stops); else if (_e.name() == "coda") { _coda = true; _e.skipCurrentElement(); } else if (_e.name() == "segno") { _segno = true; _e.skipCurrentElement(); } else skipLogCurrElem(); } Q_ASSERT(_e.isEndElement() && _e.name() == "direction-type"); } //--------------------------------------------------------- // sound //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/sound node. */ void MusicXMLParserDirection::sound() { Q_ASSERT(_e.isStartElement() && _e.name() == "sound"); _sndCapo = _e.attributes().value("capo").toString(); _sndCoda = _e.attributes().value("coda").toString(); _sndDacapo = _e.attributes().value("dacapo").toString(); _sndDalsegno = _e.attributes().value("dalsegno").toString(); _sndFine = _e.attributes().value("fine").toString(); _sndSegno = _e.attributes().value("segno").toString(); _tpoSound = _e.attributes().value("tempo").toDouble(); _dynaVelocity = _e.attributes().value("dynamics").toString(); _e.skipCurrentElement(); } //--------------------------------------------------------- // dynamics //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/dynamics node. */ void MusicXMLParserDirection::dynamics() { Q_ASSERT(_e.isStartElement() && _e.name() == "dynamics"); while (_e.readNextStartElement()) { if (_e.name() == "other-dynamics") _dynamicsList.push_back(_e.readElementText()); else { _dynamicsList.push_back(_e.name().toString()); _e.skipCurrentElement(); } } } //--------------------------------------------------------- // matchRepeat //--------------------------------------------------------- /** Do a wild-card match with known repeat texts. */ static QString matchRepeat(const QString& lowerTxt) { QString repeat; QRegExp daCapo("d\\.? *c\\.?|da *capo"); QRegExp daCapoAlFine("d\\.? *c\\.? *al *fine|da *capo *al *fine"); QRegExp daCapoAlCoda("d\\.? *c\\.? *al *coda|da *capo *al *coda"); QRegExp dalSegno("d\\.? *s\\.?|d[ae]l *segno"); QRegExp dalSegnoAlFine("d\\.? *s\\.? *al *fine|d[ae]l *segno *al *fine"); QRegExp dalSegnoAlCoda("d\\.? *s\\.? *al *coda|d[ae]l *segno *al *coda"); QRegExp fine("fine"); QRegExp toCoda("to *coda"); if (daCapo.exactMatch(lowerTxt)) repeat = "daCapo"; if (daCapoAlFine.exactMatch(lowerTxt)) repeat = "daCapoAlFine"; if (daCapoAlCoda.exactMatch(lowerTxt)) repeat = "daCapoAlCoda"; if (dalSegno.exactMatch(lowerTxt)) repeat = "dalSegno"; if (dalSegnoAlFine.exactMatch(lowerTxt)) repeat = "dalSegnoAlFine"; if (dalSegnoAlCoda.exactMatch(lowerTxt)) repeat = "dalSegnoAlCoda"; if (fine.exactMatch(lowerTxt)) repeat = "fine"; if (toCoda.exactMatch(lowerTxt)) repeat = "toCoda"; return repeat; } //--------------------------------------------------------- // findJump //--------------------------------------------------------- /** Try to find a Jump in \a repeat. */ static Jump* findJump(const QString& repeat, Score* score) { Jump* jp = 0; if (repeat == "daCapo") { jp = new Jump(score); jp->setJumpType(Jump::Type::DC); } else if (repeat == "daCapoAlCoda") { jp = new Jump(score); jp->setJumpType(Jump::Type::DC_AL_CODA); } else if (repeat == "daCapoAlFine") { jp = new Jump(score); jp->setJumpType(Jump::Type::DC_AL_FINE); } else if (repeat == "dalSegno") { jp = new Jump(score); jp->setJumpType(Jump::Type::DS); } else if (repeat == "dalSegnoAlCoda") { jp = new Jump(score); jp->setJumpType(Jump::Type::DS_AL_CODA); } else if (repeat == "dalSegnoAlFine") { jp = new Jump(score); jp->setJumpType(Jump::Type::DS_AL_FINE); } return jp; } //--------------------------------------------------------- // findMarker //--------------------------------------------------------- /** Try to find a Marker in \a repeat. */ static Marker* findMarker(const QString& repeat, Score* score) { Marker* m = 0; if (repeat == "segno") { m = new Marker(score); // note: Marker::read() also contains code to set text style based on type // avoid duplicated code // apparently this MUST be after setTextStyle m->setMarkerType(Marker::Type::SEGNO); } else if (repeat == "coda") { m = new Marker(score); m->setMarkerType(Marker::Type::CODA); } else if (repeat == "fine") { m = new Marker(score, Tid::REPEAT_RIGHT); m->setMarkerType(Marker::Type::FINE); } else if (repeat == "toCoda") { m = new Marker(score, Tid::REPEAT_RIGHT); m->setMarkerType(Marker::Type::TOCODA); } return m; } //--------------------------------------------------------- // handleRepeats //--------------------------------------------------------- void MusicXMLParserDirection::handleRepeats(Measure* measure, const int track) { // Try to recognize the various repeats QString repeat = ""; // Easy cases first if (_coda) repeat = "coda"; if (_segno) repeat = "segno"; // As sound may be missing, next do a wild-card match with known repeat texts QString txt = MScoreTextToMXML::toPlainText(_wordsText.toLower()); if (repeat == "") repeat = matchRepeat(txt.toLower()); // If that did not work, try to recognize a sound attribute if (repeat == "" && _sndCoda != "") repeat = "coda"; if (repeat == "" && _sndDacapo != "") repeat = "daCapo"; if (repeat == "" && _sndDalsegno != "") repeat = "dalSegno"; if (repeat == "" && _sndFine != "") repeat = "fine"; if (repeat == "" && _sndSegno != "") repeat = "segno"; // If a repeat was found, assume words is no longer needed if (repeat != "") _wordsText = ""; /* qDebug(" txt=%s repeat=%s", qPrintable(txt), qPrintable(repeat) ); */ if (repeat != "") { if (Jump* jp = findJump(repeat, _score)) { jp->setTrack(track); qDebug("jumpsMarkers adding jm %p meas %p",jp, measure); // TODO jumpsMarkers.append(JumpMarkerDesc(jp, measure)); measure->add(jp); } if (Marker* m = findMarker(repeat, _score)) { m->setTrack(track); qDebug("jumpsMarkers adding jm %p meas %p",m, measure); // TODO jumpsMarkers.append(JumpMarkerDesc(m, measure)); measure->add(m); } } } //--------------------------------------------------------- // bracket //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/bracket node. */ void MusicXMLParserDirection::bracket(const QString& type, const int number, QList& starts, QList& stops) { QStringRef lineEnd = _e.attributes().value("line-end"); QStringRef lineType = _e.attributes().value("line-type"); if (type == "start") { TextLine* b = new TextLine(_score); // if (placement == "") placement = "above"; // TODO ? set default b->setBeginHookType(lineEnd != "none" ? HookType::HOOK_90 : HookType::NONE); if (lineEnd == "up") b->setBeginHookHeight(-1 * b->beginHookHeight()); // hack: combine with a previous words element if (!_wordsText.isEmpty()) { // TextLine supports only limited formatting, remove all (compatible with 1.3) b->setBeginText(MScoreTextToMXML::toPlainText(_wordsText)); _wordsText = ""; } if (lineType == "solid") b->setLineStyle(Qt::SolidLine); else if (lineType == "dashed") b->setLineStyle(Qt::DashLine); else if (lineType == "dotted") b->setLineStyle(Qt::DotLine); else _logger->logError(QString("unsupported line-type: %1").arg(lineType.toString()), &_e); starts.append(MusicXmlSpannerDesc(b, ElementType::TEXTLINE, number)); } else if (type == "stop") { TextLine* b = static_cast(_pass2.getSpanner(MusicXmlSpannerDesc(ElementType::TEXTLINE, number))); if (b) { b->setEndHookType(lineEnd != "none" ? HookType::HOOK_90 : HookType::NONE); if (lineEnd == "up") b->setEndHookHeight(-1 * b->endHookHeight()); } stops.append(MusicXmlSpannerDesc(ElementType::TEXTLINE, number)); } _e.skipCurrentElement(); } //--------------------------------------------------------- // dashes //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/dashes node. */ void MusicXMLParserDirection::dashes(const QString& type, const int number, QList& starts, QList& stops) { if (type == "start") { TextLine* b = new TextLine(_score); // if (placement == "") placement = "above"; // TODO ? set default // hack: combine with a previous words element if (!_wordsText.isEmpty()) { // TextLine supports only limited formatting, remove all (compatible with 1.3) b->setBeginText(MScoreTextToMXML::toPlainText(_wordsText)); _wordsText = ""; } b->setBeginHookType(HookType::NONE); b->setEndHookType(HookType::NONE); b->setLineStyle(Qt::DashLine); // TODO brackets and dashes now share the same storage // because they both use ElementType::TEXTLINE // use mxml specific type instead starts.append(MusicXmlSpannerDesc(b, ElementType::TEXTLINE, number)); } else if (type == "stop") stops.append(MusicXmlSpannerDesc(ElementType::TEXTLINE, number)); _e.skipCurrentElement(); } //--------------------------------------------------------- // octaveShift //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/octave-shift node. */ void MusicXMLParserDirection::octaveShift(const QString& type, const int number, QList& starts, QList& stops) { if (type == "up" || type == "down") { int ottavasize = _e.attributes().value("size").toInt(); if (!(ottavasize == 8 || ottavasize == 15)) { _logger->logError(QString("unknown octave-shift size %1").arg(ottavasize), &_e); } else { Ottava* o = new Ottava(_score); // if (placement == "") placement = "above"; // TODO ? set default if (type == "down" && ottavasize == 8) o->setOttavaType(OttavaType::OTTAVA_8VA); if (type == "down" && ottavasize == 15) o->setOttavaType(OttavaType::OTTAVA_15MA); if (type == "up" && ottavasize == 8) o->setOttavaType(OttavaType::OTTAVA_8VB); if (type == "up" && ottavasize == 15) o->setOttavaType(OttavaType::OTTAVA_15MB); starts.append(MusicXmlSpannerDesc(o, ElementType::OTTAVA, number)); } } else if (type == "stop") stops.append(MusicXmlSpannerDesc(ElementType::OTTAVA, number)); _e.skipCurrentElement(); } //--------------------------------------------------------- // pedal //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/pedal node. */ void MusicXMLParserDirection::pedal(const QString& type, const int /* number */, QList& starts, QList& stops) { QStringRef line = _e.attributes().value("line"); QString sign = _e.attributes().value("sign").toString(); if (line != "yes" && sign == "") sign = "yes"; // MusicXML 2.0 compatibility if (line == "yes" && sign == "") sign = "no"; // MusicXML 2.0 compatibility if (line == "yes") { if (type == "start") { Pedal* p = new Pedal(_score); if (sign == "yes") p->setBeginText("keyboardPedalPed"); else p->setBeginHookType(HookType::HOOK_90); p->setEndHookType(HookType::HOOK_90); // if (placement == "") placement = "below"; // TODO ? set default starts.append(MusicXmlSpannerDesc(p, ElementType::PEDAL, 0)); } else if (type == "stop") stops.append(MusicXmlSpannerDesc(ElementType::PEDAL, 0)); else if (type == "change") { #if 0 TODO // pedal change is implemented as two separate pedals // first stop the first one if (pedal) { pedal->setEndHookType(HookType::HOOK_45); handleSpannerStop(pedal, "pedal", track, tick, spanners); pedalContinue = pedal; // mark for later fixup pedal = 0; } // then start a new one pedal = static_cast(checkSpannerOverlap(pedal, new Pedal(score), "pedal")); pedal->setBeginHookType(HookType::HOOK_45); pedal->setEndHookType(HookType::HOOK_90); if (placement == "") placement = "below"; handleSpannerStart(pedal, "pedal", track, placement, tick, spanners); #endif } else if (type == "continue") { // ignore } else qDebug("unknown pedal type %s", qPrintable(type)); } else { // TBD: what happens when an unknown pedal type is found ? Symbol* s = new Symbol(_score); s->setAlign(Align::LEFT | Align::BASELINE); //s->setOffsetType(OffsetType::SPATIUM); if (type == "start") s->setSym(SymId::keyboardPedalPed); else if (type == "stop") s->setSym(SymId::keyboardPedalUp); else _logger->logError(QString("unknown pedal type %1").arg(type), &_e); _elems.append(s); } _e.skipCurrentElement(); } //--------------------------------------------------------- // wedge //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/wedge node. */ void MusicXMLParserDirection::wedge(const QString& type, const int number, QList& starts, QList& stops) { QStringRef niente = _e.attributes().value("niente"); if (type == "crescendo" || type == "diminuendo") { Hairpin* h = new Hairpin(_score); h->setHairpinType(type == "crescendo" ? HairpinType::CRESC_HAIRPIN : HairpinType::DECRESC_HAIRPIN); if (niente == "yes") h->setHairpinCircledTip(true); starts.append(MusicXmlSpannerDesc(h, ElementType::HAIRPIN, number)); } else if (type == "stop") { Hairpin* h = static_cast(_pass2.getSpanner(MusicXmlSpannerDesc(ElementType::HAIRPIN, number))); if (niente == "yes") h->setHairpinCircledTip(true); stops.append(MusicXmlSpannerDesc(ElementType::HAIRPIN, number)); } _e.skipCurrentElement(); } //--------------------------------------------------------- // addSpanner //--------------------------------------------------------- void MusicXMLParserPass2::addSpanner(const MusicXmlSpannerDesc& d) { if (d.tp == ElementType::HAIRPIN && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) _hairpins[d.nr] = d.sp; else if (d.tp == ElementType::OTTAVA && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) _ottavas[d.nr] = d.sp; else if (d.tp == ElementType::PEDAL && 0 == d.nr) _pedal = d.sp; // TODO: check MAX_BRACKETS vs MAX_NUMBER_LEVEL else if (d.tp == ElementType::TEXTLINE && 0 <= d.nr && d.nr < MAX_BRACKETS) _brackets[d.nr] = d.sp; } //--------------------------------------------------------- // getSpanner //--------------------------------------------------------- SLine* MusicXMLParserPass2::getSpanner(const MusicXmlSpannerDesc& d) { if (d.tp == ElementType::HAIRPIN && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) return _hairpins[d.nr]; else if (d.tp == ElementType::OTTAVA && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) return _ottavas[d.nr]; else if (d.tp == ElementType::PEDAL && 0 == d.nr) return _pedal; // TODO: check MAX_BRACKETS vs MAX_NUMBER_LEVEL else if (d.tp == ElementType::TEXTLINE && 0 <= d.nr && d.nr < MAX_BRACKETS) return _brackets[d.nr]; return 0; } //--------------------------------------------------------- // clearSpanner //--------------------------------------------------------- void MusicXMLParserPass2::clearSpanner(const MusicXmlSpannerDesc& d) { if (d.tp == ElementType::HAIRPIN && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) _hairpins[d.nr] = 0; else if (d.tp == ElementType::OTTAVA && 0 <= d.nr && d.nr < MAX_NUMBER_LEVEL) _ottavas[d.nr] = 0; else if (d.tp == ElementType::PEDAL && 0 == d.nr) _pedal = 0; // TODO: check MAX_BRACKETS vs MAX_NUMBER_LEVEL else if (d.tp == ElementType::TEXTLINE && 0 <= d.nr && d.nr < MAX_BRACKETS) _brackets[d.nr] = 0; } //--------------------------------------------------------- // metronome //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/direction/direction-type/metronome node. Convert to text and set r to calculated tempo. */ QString MusicXMLParserDirection::metronome(double& r) { Q_ASSERT(_e.isStartElement() && _e.name() == "metronome"); r = 0; QString tempoText; QString perMinute; bool parenth = _e.attributes().value("parentheses") == "yes"; if (parenth) tempoText += "("; TDuration dur1; TDuration dur2; while (_e.readNextStartElement()) { if (_e.name() == "metronome-note" || _e.name() == "metronome-relation") { skipLogCurrElem(); continue; } QString txt = _e.readElementText(); if (_e.name() == "beat-unit") { // set first dur that is still invalid if (!dur1.isValid()) dur1.setType(txt); else if (!dur2.isValid()) dur2.setType(txt); } else if (_e.name() == "beat-unit-dot") { if (dur2.isValid()) dur2.setDots(1); else if (dur1.isValid()) dur1.setDots(1); } else if (_e.name() == "per-minute") perMinute = txt; else skipLogCurrElem(); } if (dur1.isValid()) tempoText += TempoText::duration2tempoTextString(dur1); if (dur2.isValid()) { tempoText += " = "; tempoText += TempoText::duration2tempoTextString(dur2); } else if (perMinute != "") { tempoText += " = "; tempoText += perMinute; } if (dur1.isValid() && !dur2.isValid() && perMinute != "") { bool ok; double d = perMinute.toDouble(&ok); if (ok) { // convert fraction to beats per minute r = 4 * dur1.fraction().numerator() * d / dur1.fraction().denominator(); } } if (parenth) tempoText += ")"; return tempoText; } //--------------------------------------------------------- // determineBarLineType //--------------------------------------------------------- static bool determineBarLineType(const QString& barStyle, const QString& repeat, BarLineType& type, bool& visible) { // set defaults type = BarLineType::NORMAL; visible = true; if (barStyle == "light-heavy" && repeat == "backward") type = BarLineType::END_REPEAT; else if (barStyle == "heavy-light" && repeat == "forward") type = BarLineType::START_REPEAT; else if (barStyle == "light-heavy" && repeat.isEmpty()) type = BarLineType::END; else if (barStyle == "regular") type = BarLineType::NORMAL; else if (barStyle == "dashed") type = BarLineType::BROKEN; else if (barStyle == "dotted") type = BarLineType::DOTTED; else if (barStyle == "light-light") type = BarLineType::DOUBLE; /* else if (barStyle == "heavy-light") ; else if (barStyle == "heavy-heavy") ; */ else if (barStyle == "none") { type = BarLineType::NORMAL; visible = false; } else if (barStyle == "") { if (repeat == "backward") type = BarLineType::END_REPEAT; else if (repeat == "forward") type = BarLineType::START_REPEAT; else { qDebug("empty bar type"); // TODO return false; } } else if (barStyle == "tick") { } else if (barStyle == "short") { } else { qDebug("unsupported bar type <%s>", qPrintable(barStyle)); // TODO return false; } return true; } //--------------------------------------------------------- // barline //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/barline node. */ void MusicXMLParserPass2::barline(const QString& partId, Measure* measure) { Q_ASSERT(_e.isStartElement() && _e.name() == "barline"); QString loc = _e.attributes().value("location").toString(); if (loc == "") loc = "right"; QString barStyle; QString endingNumber; QString endingType; QString endingText; QString repeat; QString count; while (_e.readNextStartElement()) { if (_e.name() == "bar-style") barStyle = _e.readElementText(); else if (_e.name() == "ending") { endingNumber = _e.attributes().value("number").toString(); endingType = _e.attributes().value("type").toString(); endingText = _e.readElementText(); } else if (_e.name() == "repeat") { repeat = _e.attributes().value("direction").toString(); count = _e.attributes().value("times").toString(); if (count.isEmpty()) { count = "2"; } measure->setRepeatCount(count.toInt()); _e.skipCurrentElement(); } else skipLogCurrElem(); } BarLineType type = BarLineType::NORMAL; bool visible = true; if (determineBarLineType(barStyle, repeat, type, visible)) { if (type == BarLineType::START_REPEAT) { // combine start_repeat flag with current state initialized during measure parsing measure->setRepeatStart(true); } else if (type == BarLineType::END_REPEAT) { // combine end_repeat flag with current state initialized during measure parsing measure->setRepeatEnd(true); } else { int track = _pass1.trackForPart(partId); if (barStyle == "tick") { BarLine* b = new BarLine(measure->score()); b->setTrack(track); b->setBarLineType(BarLineType::NORMAL); b->setSpanStaff(false); b->setSpanFrom(BARLINE_SPAN_TICK1_FROM); b->setSpanTo(BARLINE_SPAN_TICK1_TO); Segment* segment = measure->getSegment(SegmentType::EndBarLine, measure->endTick()); segment->add(b); } else if (barStyle == "short") { BarLine* b = new BarLine(measure->score()); b->setTrack(track); b->setBarLineType(BarLineType::NORMAL); b->setSpanStaff(0); b->setSpanFrom(BARLINE_SPAN_SHORT1_FROM); b->setSpanTo(BARLINE_SPAN_SHORT1_TO); Segment* segment = measure->getSegment(SegmentType::EndBarLine, measure->endTick()); segment->add(b); } else if (loc == "right") measure->setEndBarLineType(type, track, visible); else if (measure->prevMeasure()) measure->prevMeasure()->setEndBarLineType(type, track, visible); } } doEnding(partId, measure, endingNumber, endingType, endingText); } //--------------------------------------------------------- // doEnding //--------------------------------------------------------- void MusicXMLParserPass2::doEnding(const QString& partId, Measure* measure, const QString& number, const QString& type, const QString& text) { if (!(number.isEmpty() && type.isEmpty())) { if (number.isEmpty()) _logger->logError("empty ending number", &_e); else if (type.isEmpty()) _logger->logError("empty ending type", &_e); else { QStringList sl = number.split(",", QString::SkipEmptyParts); QList iEndingNumbers; bool unsupported = false; foreach(const QString &s, sl) { int iEndingNumber = s.toInt(); if (iEndingNumber <= 0) { unsupported = true; break; } iEndingNumbers.append(iEndingNumber); } if (unsupported) _logger->logError(QString("unsupported ending number '%1'").arg(number), &_e); else { if (type == "start") { Volta* volta = new Volta(_score); volta->setTrack(_pass1.trackForPart(partId)); volta->setText(text.isEmpty() ? number : text); // LVIFIX TODO also support endings "1 - 3" volta->endings().clear(); volta->endings().append(iEndingNumbers); volta->setTick(measure->tick()); _score->addElement(volta); _lastVolta = volta; } else if (type == "stop") { if (_lastVolta) { _lastVolta->setVoltaType(Volta::Type::CLOSED); _lastVolta->setTick2(measure->tick() + measure->ticks()); _lastVolta = 0; } else _logger->logError("ending stop without start", &_e); } else if (type == "discontinue") { if (_lastVolta) { _lastVolta->setVoltaType(Volta::Type::OPEN); _lastVolta->setTick2(measure->tick() + measure->ticks()); _lastVolta = 0; } else _logger->logError("ending discontinue without start", &_e); } else _logger->logError(QString("unsupported ending type '%1'").arg(type), &_e); } } } } //--------------------------------------------------------- // addSymToSig //--------------------------------------------------------- /** Add a symbol defined as key-step \a step , -alter \a alter and -accidental \a accid to \a sig. */ static void addSymToSig(KeySigEvent& sig, const QString& step, const QString& alter, const QString& accid) { //qDebug("addSymToSig(step '%s' alt '%s' acc '%s')", // qPrintable(step), qPrintable(alter), qPrintable(accid)); SymId id = mxmlString2accSymId(accid); if (id == SymId::noSym) { bool ok; double d; d = alter.toDouble(&ok); AccidentalType accTpAlter = ok ? microtonalGuess(d) : AccidentalType::NONE; id = mxmlString2accSymId(accidentalType2MxmlString(accTpAlter)); } if (step.size() == 1 && id != SymId::noSym) { const QString table = "FEDCBAG"; const int line = table.indexOf(step); // no auto layout for custom keysig, calculate xpos // TODO: use symbol width ? const qreal spread = 1.4; // assumed glyph width in space const qreal x = sig.keySymbols().size() * spread; if (line >= 0) { KeySym ks; ks.sym = id; ks.spos = QPointF(x, qreal(line) * 0.5); sig.keySymbols().append(ks); sig.setCustom(true); } } } //--------------------------------------------------------- // addKey //--------------------------------------------------------- /** Add a KeySigEvent to the score. */ static void addKey(const KeySigEvent key, const bool printObj, Score* score, Measure* measure, const int staffIdx, const int tick) { Key oldkey = score->staff(staffIdx)->key(tick); // TODO only if different custom key ? if (oldkey != key.key() || key.custom() || key.isAtonal()) { // new key differs from key in effect at this tick KeySig* keysig = new KeySig(score); keysig->setTrack((staffIdx) * VOICES); keysig->setKeySigEvent(key); keysig->setVisible(printObj); Segment* s = measure->getSegment(SegmentType::KeySig, tick); s->add(keysig); //currKeySig->setKeySigEvent(key); } } //--------------------------------------------------------- // flushAlteredTone //--------------------------------------------------------- /** If a valid key-step, -alter, -accidental combination has been read, convert it to a key symbol and add to the key. Clear key-step, -alter, -accidental. */ static void flushAlteredTone(KeySigEvent& kse, QString& step, QString& alt, QString& acc) { //qDebug("flushAlteredTone(step '%s' alt '%s' acc '%s')", // qPrintable(step), qPrintable(alt), qPrintable(acc)); if (step == "" && alt == "" && acc == "") return; // nothing to do // step and alt are required, but also accept step and acc if (step != "" && (alt != "" || acc != "")) { addSymToSig(kse, step, alt, acc); } else { qDebug("flushAlteredTone invalid combination of step '%s' alt '%s' acc '%s')", qPrintable(step), qPrintable(alt), qPrintable(acc)); // TODO } // clean up step = ""; alt = ""; acc = ""; } //--------------------------------------------------------- // key //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/key node. */ // TODO: check currKeySig handling void MusicXMLParserPass2::key(const QString& partId, Measure* measure, const int tick) { Q_ASSERT(_e.isStartElement() && _e.name() == "key"); QString strKeyno = _e.attributes().value("number").toString(); int keyno = -1; // assume no number (see below) if (strKeyno != "") { keyno = strKeyno.toInt(); if (keyno == 0) { // conversion error (0), assume staff 1 _logger->logError(QString("invalid key number '%1'").arg(strKeyno), &_e); keyno = 1; } // convert to 0-based keyno--; } bool printObject = _e.attributes().value("print-object") != "no"; // for custom keys, a single altered tone is described by // key-step (required), key-alter (required) and key-accidental (optional) // none, one or more altered tone may be present // a simple state machine is required to detect them KeySigEvent key; QString keyStep; QString keyAlter; QString keyAccidental; while (_e.readNextStartElement()) { if (_e.name() == "fifths") key.setKey(Key(_e.readElementText().toInt())); else if (_e.name() == "mode") { QString m = _e.readElementText(); if (m == "none") { key.setCustom(true); key.setMode(KeyMode::NONE); } else if (m == "major") { key.setMode(KeyMode::MAJOR); } else if (m == "minor") { key.setMode(KeyMode::MINOR); } else { _logger->logError(QString("Unsupported mode '%1'").arg(m), &_e); } } else if (_e.name() == "cancel") skipLogCurrElem(); // TODO ?? else if (_e.name() == "key-step") { flushAlteredTone(key, keyStep, keyAlter, keyAccidental); keyStep = _e.readElementText(); } else if (_e.name() == "key-alter") keyAlter = _e.readElementText(); else if (_e.name() == "key-accidental") keyAccidental = _e.readElementText(); else skipLogCurrElem(); } flushAlteredTone(key, keyStep, keyAlter, keyAccidental); int nstaves = _pass1.getPart(partId)->nstaves(); int staffIdx = _pass1.trackForPart(partId) / VOICES; if (keyno == -1) { // apply key to all staves in the part for (int i = 0; i < nstaves; ++i) { addKey(key, printObject, _score, measure, staffIdx + i, tick); } } else if (keyno < nstaves) addKey(key, printObject, _score, measure, staffIdx + keyno, tick); } //--------------------------------------------------------- // clef //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/clef node. */ void MusicXMLParserPass2::clef(const QString& partId, Measure* measure, const int tick) { Q_ASSERT(_e.isStartElement() && _e.name() == "clef"); Part* part = _pass1.getPart(partId); Q_ASSERT(part); // TODO: check error handling for // - single staff // - multi-staff with same clef QString strClefno = _e.attributes().value("number").toString(); int clefno = 1; // default if (strClefno != "") clefno = strClefno.toInt(); if (clefno <= 0 || clefno > part->nstaves()) { // conversion error (0) or other issue, assume staff 1 // Also for Cubase 6.5.5 which generates clef number="2" in a single staff part // Same fix is required in pass 1 and pass 2 _logger->logError(QString("invalid clef number '%1'").arg(strClefno), &_e); clefno = 1; } // convert to 0-based clefno--; ClefType clef = ClefType::G; StaffTypes st = StaffTypes::STANDARD; QString c; int i = 0; int line = -1; while (_e.readNextStartElement()) { if (_e.name() == "sign") c = _e.readElementText(); else if (_e.name() == "line") line = _e.readElementText().toInt(); else if (_e.name() == "clef-octave-change") { i = _e.readElementText().toInt(); if (i && !(c == "F" || c == "G")) qDebug("clef-octave-change only implemented for F and G key"); // TODO } else skipLogCurrElem(); } //some software (Primus) don't include line and assume some default // it's permitted by MusicXML 2.0 XSD if (line == -1) { if (c == "G") line = 2; else if (c == "F") line = 4; else if (c == "C") line = 3; } if (c == "G" && i == 0 && line == 2) clef = ClefType::G; else if (c == "G" && i == 1 && line == 2) clef = ClefType::G8_VA; else if (c == "G" && i == 2 && line == 2) clef = ClefType::G15_MA; else if (c == "G" && i == -1 && line == 2) clef = ClefType::G8_VB; else if (c == "G" && i == 0 && line == 1) clef = ClefType::G_1; else if (c == "F" && i == 0 && line == 3) clef = ClefType::F_B; else if (c == "F" && i == 0 && line == 4) clef = ClefType::F; else if (c == "F" && i == 1 && line == 4) clef = ClefType::F_8VA; else if (c == "F" && i == 2 && line == 4) clef = ClefType::F_15MA; else if (c == "F" && i == -1 && line == 4) clef = ClefType::F8_VB; else if (c == "F" && i == -2 && line == 4) clef = ClefType::F15_MB; else if (c == "F" && i == 0 && line == 5) clef = ClefType::F_C; else if (c == "C") { if (line == 5) clef = ClefType::C5; else if (line == 4) clef = ClefType::C4; else if (line == 3) clef = ClefType::C3; else if (line == 2) clef = ClefType::C2; else if (line == 1) clef = ClefType::C1; } else if (c == "percussion") { clef = ClefType::PERC; st = StaffTypes::PERC_DEFAULT; } else if (c == "TAB") { clef = ClefType::TAB; st= StaffTypes::TAB_DEFAULT; } else qDebug("clef: unknown clef ", qPrintable(c), line, i); // TODO Clef* clefs = new Clef(_score); clefs->setClefType(clef); int track = _pass1.trackForPart(partId) + clefno * VOICES; clefs->setTrack(track); Segment* s = measure->getSegment(tick ? SegmentType::Clef : SegmentType::HeaderClef, tick); s->add(clefs); // set the correct staff type // note that this overwrites the staff lines value set in pass 1 // also note that clef handling should probably done in pass1 int staffIdx = _score->staffIdx(part) + clefno; int lines = _score->staff(staffIdx)->lines(0); if (st == StaffTypes::TAB_DEFAULT || (_hasDrumset && st == StaffTypes::PERC_DEFAULT)) { _score->staff(staffIdx)->setStaffType(0, StaffType::preset(st)); _score->staff(staffIdx)->setLines(0, lines); // preserve previously set staff lines _score->staff(staffIdx)->setBarLineTo(0); // default } } //--------------------------------------------------------- // determineTimeSig //--------------------------------------------------------- /** Determine the time signature based on \a beats, \a beatType and \a timeSymbol. Sets return parameters \a st, \a bts, \a btp. Return true if OK, false on error. */ // TODO: share between pass 1 and pass 2 static bool determineTimeSig(const QString beats, const QString beatType, const QString timeSymbol, TimeSigType& st, int& bts, int& btp) { // initialize st = TimeSigType::NORMAL; bts = 0; // the beats (max 4 separated by "+") as integer btp = 0; // beat-type as integer // determine if timesig is valid if (beats == "2" && beatType == "2" && timeSymbol == "cut") { st = TimeSigType::ALLA_BREVE; bts = 2; btp = 2; return true; } else if (beats == "4" && beatType == "4" && timeSymbol == "common") { st = TimeSigType::FOUR_FOUR; bts = 4; btp = 4; return true; } else { if (!timeSymbol.isEmpty() && timeSymbol != "normal") { qDebug("determineTimeSig: time symbol <%s> not recognized with beats=%s and beat-type=%s", qPrintable(timeSymbol), qPrintable(beats), qPrintable(beatType)); // TODO return false; } btp = beatType.toInt(); QStringList list = beats.split("+"); for (int i = 0; i < list.size(); i++) bts += list.at(i).toInt(); } // determine if bts and btp are valid if (bts <= 0 || btp <=0) { qDebug("determineTimeSig: beats=%s and/or beat-type=%s not recognized", qPrintable(beats), qPrintable(beatType)); // TODO return false; } return true; } //--------------------------------------------------------- // time //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/time node. */ void MusicXMLParserPass2::time(const QString& partId, Measure* measure, const int tick) { Q_ASSERT(_e.isStartElement() && _e.name() == "time"); QString beats; QString beatType; QString timeSymbol = _e.attributes().value("symbol").toString(); bool printObject = _e.attributes().value("print-object") != "no"; while (_e.readNextStartElement()) { if (_e.name() == "beats") beats = _e.readElementText(); else if (_e.name() == "beat-type") beatType = _e.readElementText(); else skipLogCurrElem(); } if (beats != "" && beatType != "") { // determine if timesig is valid TimeSigType st = TimeSigType::NORMAL; int bts = 0; // total beats as integer (beats may contain multiple numbers, separated by "+") int btp = 0; // beat-type as integer if (determineTimeSig(beats, beatType, timeSymbol, st, bts, btp)) { _timeSigDura = Fraction(bts, btp); Fraction fractionTSig = Fraction(bts, btp); for (int i = 0; i < _pass1.getPart(partId)->nstaves(); ++i) { TimeSig* timesig = new TimeSig(_score); timesig->setVisible(printObject); int track = _pass1.trackForPart(partId) + i * VOICES; timesig->setTrack(track); timesig->setSig(fractionTSig, st); // handle simple compound time signature if (beats.contains(QChar('+'))) { timesig->setNumeratorString(beats); timesig->setDenominatorString(beatType); } Segment* s = measure->getSegment(SegmentType::TimeSig, tick); s->add(timesig); } } } } //--------------------------------------------------------- // transpose //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/transpose node. */ void MusicXMLParserPass2::transpose(const QString& partId) { Q_ASSERT(_e.isStartElement() && _e.name() == "transpose"); Interval interval; bool diatonic = false; bool chromatic = false; while (_e.readNextStartElement()) { int i = _e.readElementText().toInt(); if (_e.name() == "diatonic") { interval.diatonic = i; diatonic = true; } else if (_e.name() == "chromatic") { interval.chromatic = i; chromatic = true; } else if (_e.name() == "octave-change") { interval.diatonic += i * 7; interval.chromatic += i * 12; } else skipLogCurrElem(); } if (chromatic && !diatonic) interval.diatonic += chromatic2diatonic(interval.chromatic); _pass1.getPart(partId)->instrument()->setTranspose(interval); } //--------------------------------------------------------- // divisions //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/attributes/divisions node. */ void MusicXMLParserPass2::divisions() { Q_ASSERT(_e.isStartElement() && _e.name() == "divisions"); _divs = _e.readElementText().toInt(); if (!(_divs > 0)) _logger->logError("illegal divisions", &_e); } //--------------------------------------------------------- // isWholeMeasureRest //--------------------------------------------------------- /** * Determine whole measure rest. */ // By convention, whole measure rests do not have a "type" element // As of MusicXML 3.0, this can be indicated by an attribute "measure", // but for backwards compatibility the "old" convention still has to be supported. // Also verify the rest fits exactly in the measure, as some programs // (e.g. Cakewalk SONAR X2 Studio [Version: 19.0.0.306]) leave out // the type for all rests. // Sibelius calls all whole-measure rests "whole", even if the duration != 4/4 static bool isWholeMeasureRest(const bool rest, const QString& type, const Fraction dura, const Fraction mDura) { if (!rest) return false; if (!dura.isValid()) return false; if (!mDura.isValid()) return false; return ((type == "" && dura == mDura) || (type == "whole" && dura == mDura && dura != Fraction(1, 1))); } //--------------------------------------------------------- // determineDuration //--------------------------------------------------------- /** * Determine duration for a note or rest. * This includes whole measure rest detection. */ static TDuration determineDuration(const bool rest, const QString& type, const int dots, const Fraction dura, const Fraction mDura) { //qDebug("determineDuration rest %d type '%s' dots %d dura %s mDura %s", // rest, qPrintable(type), dots, qPrintable(dura.print()), qPrintable(mDura.print())); TDuration res; if (rest) { if (isWholeMeasureRest(rest, type, dura, mDura)) res.setType(TDuration::DurationType::V_MEASURE); else if (type == "") { // If no type, set duration type based on duration. // Note that sometimes unusual duration (e.g. 261/256) are found. res.setVal(dura.ticks()); } else { res.setType(type); res.setDots(dots); } } else { res.setType(type); res.setDots(dots); if (res.type() == TDuration::DurationType::V_INVALID) res.setType(TDuration::DurationType::V_QUARTER); // default, TODO: use dura ? } //qDebug("-> dur %hhd (%s) dots %d ticks %s", // res.type(), qPrintable(res.name()), res.dots(), qPrintable(dura.print())); return res; } //--------------------------------------------------------- // setChordRestDuration //--------------------------------------------------------- /** * Set \a cr duration */ static void setChordRestDuration(ChordRest* cr, TDuration duration, const Fraction dura) { if (duration.type() == TDuration::DurationType::V_MEASURE) { cr->setDurationType(duration); cr->setDuration(dura); } else { cr->setDurationType(duration); cr->setDuration(cr->durationType().fraction()); } } //--------------------------------------------------------- // addRest //--------------------------------------------------------- /** * Add a rest to the score * TODO: beam handling * TODO: display step handling * TODO: visible handling * TODO: whole measure rest handling */ static Rest* addRest(Score* score, Measure* m, const int tick, const int track, const int move, const TDuration duration, const Fraction dura) { Segment* s = m->getSegment(SegmentType::ChordRest, tick); // Sibelius might export two rests at the same place, ignore the 2nd one // if (s->element(track)) { qDebug("cannot add rest at tick %d track %d: element already present", tick, track); // TODO return 0; } Rest* cr = new Rest(score); setChordRestDuration(cr, duration, dura); cr->setTrack(track); cr->setStaffMove(move); s->add(cr); return cr; } //--------------------------------------------------------- // findOrCreateChord //--------------------------------------------------------- /** * Find (or create if not found) the chord at \a tick and \a track. * Note: staff move is a note property in MusicXML, but chord property in MuseScore * This is simply ignored here, effectively using the last chords value. */ static Chord* findOrCreateChord(Score* score, Measure* m, const int tick, const int track, const int move, const TDuration duration, const Fraction dura, Beam::Mode bm) { //qDebug("findOrCreateChord tick %d track %d dur ticks %d ticks %s bm %hhd", // tick, track, duration.ticks(), qPrintable(dura.print()), bm); Chord* c = m->findChord(tick, track); if (c == 0) { c = new Chord(score); // better not to force beam end, as the beam palette does not support it if (bm == Beam::Mode::END) c->setBeamMode(Beam::Mode::AUTO); else c->setBeamMode(bm); c->setTrack(track); setChordRestDuration(c, duration, dura); Segment* s = m->getSegment(SegmentType::ChordRest, tick); s->add(c); } c->setStaffMove(move); return c; } //--------------------------------------------------------- // graceNoteType //--------------------------------------------------------- /** * convert duration and slash to grace note type */ NoteType graceNoteType(const TDuration duration, const bool slash) { NoteType nt = NoteType::APPOGGIATURA; if (slash) nt = NoteType::ACCIACCATURA; if (duration.type() == TDuration::DurationType::V_QUARTER) { nt = NoteType::GRACE4; } else if (duration.type() == TDuration::DurationType::V_16TH) { nt = NoteType::GRACE16; } else if (duration.type() == TDuration::DurationType::V_32ND) { nt = NoteType::GRACE32; } return nt; } //--------------------------------------------------------- // createGraceChord //--------------------------------------------------------- /** * Create a grace chord. */ static Chord* createGraceChord(Score* score, const int track, const TDuration duration, const bool slash) { Chord* c = new Chord(score); c->setNoteType(graceNoteType(duration, slash)); c->setTrack(track); // note grace notes have no durations, use default fraction 0/1 setChordRestDuration(c, duration, Fraction()); return c; } //--------------------------------------------------------- // elementMustBePostponed //--------------------------------------------------------- /** Check if handling the current element must be postponed until after allocating the note. */ static bool elementMustBePostponed(const QXmlStreamReader& e) { return e.name() == "notations" || e.name() == "lyric" || e.name() == "play"; } //--------------------------------------------------------- // handleDisplayStep //--------------------------------------------------------- /** * convert display-step and display-octave to staff line */ static void handleDisplayStep(ChordRest* cr, int step, int octave, int tick, qreal spatium) { if (0 <= step && step <= 6 && 0 <= octave && octave <= 9) { //qDebug("rest step=%d oct=%d", step, octave); ClefType clef = cr->staff()->clef(tick); int po = ClefInfo::pitchOffset(clef); //qDebug(" clef=%hhd po=%d step=%d", clef, po, step); int dp = 7 * (octave + 2) + step; //qDebug(" dp=%d po-dp=%d", dp, po-dp); cr->ryoffset() = (po - dp + 3) * spatium / 2; } } //--------------------------------------------------------- // setNoteHead //--------------------------------------------------------- /** Set the notehead parameters. */ static void setNoteHead(Note* note, const QColor noteheadColor, const bool noteheadParentheses, const QString& noteheadFilled) { const auto score = note->score(); if (noteheadColor != QColor::Invalid) note->setColor(noteheadColor); if (noteheadParentheses) { auto s = new Symbol(score); s->setSym(SymId::noteheadParenthesisLeft); s->setParent(note); score->addElement(s); s = new Symbol(score); s->setSym(SymId::noteheadParenthesisRight); s->setParent(note); score->addElement(s); } if (noteheadFilled == "no") note->setHeadType(NoteHead::Type::HEAD_HALF); else if (noteheadFilled == "yes") note->setHeadType(NoteHead::Type::HEAD_QUARTER); } //--------------------------------------------------------- // addFiguredBassElemens //--------------------------------------------------------- /** Add the figured bass elements. */ static void addFiguredBassElemens(FiguredBassList& fbl, const Fraction noteStartTime, const int msTrack, const Fraction dura, Measure* measure) { if (!fbl.isEmpty()) { auto sTick = noteStartTime.ticks(); // starting tick foreach (FiguredBass* fb, fbl) { fb->setTrack(msTrack); // No duration tag defaults ticks() to 0; set to note value if (fb->ticks() == 0) fb->setTicks(dura.ticks()); // TODO: set correct onNote value Segment* s = measure->getSegment(SegmentType::ChordRest, sTick); s->add(fb); sTick += fb->ticks(); } fbl.clear(); } } //--------------------------------------------------------- // note //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note node. */ Note* MusicXMLParserPass2::note(const QString& partId, Measure* measure, const Fraction sTime, const Fraction prevSTime, Fraction& dura, QString& currentVoice, GraceChordList& gcl, int& gac, Beam*& currBeam, FiguredBassList& fbl, int& alt ) { Q_ASSERT(_e.isStartElement() && _e.name() == "note"); if (_e.attributes().value("print-spacing") == "no") { notePrintSpacingNo(dura); return 0; } bool chord = false; bool cue = false; bool small = false; bool grace = false; bool rest = false; int staff = 1; QString type; QString voice; Direction stemDir = Direction::AUTO; bool noStem = false; NoteHead::Group headGroup = NoteHead::Group::HEAD_NORMAL; QColor noteColor = QColor::Invalid; noteColor.setNamedColor(_e.attributes().value("color").toString()); QColor noteheadColor = QColor::Invalid; bool noteheadParentheses = false; QString noteheadFilled; int velocity = round(_e.attributes().value("dynamics").toDouble() * 0.9); bool graceSlash = false; bool printObject = _e.attributes().value("print-object") != "no"; Beam::Mode bm = Beam::Mode::AUTO; QString instrumentId; mxmlNoteDuration mnd(_divs, _logger); mxmlNotePitch mnp(_logger); while (_e.readNextStartElement() && !elementMustBePostponed(_e)) { if (mnp.readProperties(_e, _score)) { // element handled } else if (mnd.readProperties(_e)) { // element handled } else if (_e.name() == "beam") beam(bm); else if (_e.name() == "chord") { chord = true; _e.readNext(); } else if (_e.name() == "cue") { cue = true; _e.readNext(); } else if (_e.name() == "grace") { grace = true; graceSlash = _e.attributes().value("slash") == "yes"; _e.readNext(); } else if (_e.name() == "instrument") { instrumentId = _e.attributes().value("id").toString(); _e.readNext(); } else if (_e.name() == "notehead") { noteheadColor.setNamedColor(_e.attributes().value("color").toString()); noteheadParentheses = _e.attributes().value("parentheses") == "yes"; noteheadFilled = _e.attributes().value("filled").toString(); headGroup = convertNotehead(_e.readElementText()); } else if (_e.name() == "rest") { rest = true; mnp.displayStepOctave(_e); } else if (_e.name() == "staff") { auto ok = false; auto strStaff = _e.readElementText(); staff = strStaff.toInt(&ok); if (!ok) { // error already reported in pass 1 staff = 1; } } else if (_e.name() == "stem") stem(stemDir, noStem); else if (_e.name() == "type") { small = _e.attributes().value("size") == "cue"; type = _e.readElementText(); } else if (_e.name() == "voice") voice = _e.readElementText(); else skipLogCurrElem(); } // convert staff to zero-based (in case of error, staff will be -1) staff--; // Bug fix for Sibelius 7.1.3 which does not write for notes with if (!chord) // remember voice currentVoice = voice; else if (voice == "") // use voice from last note w/o voice = currentVoice; // Assume voice 1 if voice is empty (legal in a single voice part) if (voice == "") voice = "1"; // check for timing error(s) and set dura // keep in this order as checkTiming() might change dura auto errorStr = mnd.checkTiming(type, rest, grace); dura = mnd.dura(); if (errorStr != "") _logger->logError(errorStr, &_e); // At this point all checks have been done, the note should be added // note: in case of error exit from here, the postponed children // must still be skipped int msMove = 0; int msTrack = 0; int msVoice = 0; if (!_pass1.determineStaffMoveVoice(partId, staff, voice, msMove, msTrack, msVoice)) { _logger->logDebugInfo(QString("could not map staff %1 voice '%2'").arg(staff + 1).arg(voice), &_e); // begin experimental fix for postponed children // TODO test /repair #if 0 while (_e.tokenType() == QXmlStreamReader::StartElement) { //qDebug("in second loop element '%s'", qPrintable(_e.name().toString())); skipLogCurrElem(); // skip to either start of next child or end of // currently at end of last child handled //qDebug("::note before skip tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); do _e.readNext(); while (!(_e.tokenType() == QXmlStreamReader::StartElement) && !(_e.tokenType() == QXmlStreamReader::EndElement && _e.name() == "note")); //qDebug("::note after skip tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); } #endif // end experimental fix for testVoiceMapper* Q_ASSERT(_e.isEndElement() && _e.name() == "note"); return 0; } else { } TDuration duration = determineDuration(rest, type, mnd.dots(), dura, Fraction::fromTicks(measure->ticks())); ChordRest* cr = 0; Note* note = 0; // start time for note: // - sTime for non-chord / first chord note // - prevTime for others const Fraction noteStartTime = chord ? prevSTime : sTime; if (rest) { int track = msTrack + msVoice; cr = addRest(_score, measure, noteStartTime.ticks(), track, msMove, duration, dura); if (cr) { if (currBeam) { if (currBeam->track() == track) { cr->setBeamMode(Beam::Mode::MID); currBeam->add(cr); } else removeBeam(currBeam); } else cr->setBeamMode(Beam::Mode::NONE); cr->setSmall(small); if (noteColor != QColor::Invalid) cr->setColor(noteColor); cr->setVisible(printObject); handleDisplayStep(cr, mnp.displayStep(), mnp.displayOctave(), noteStartTime.ticks(), _score->spatium()); } } else { Chord* c; if (!grace) { // regular note // if there is already a chord just add to it // else create a new one // this basically ignores errors c = findOrCreateChord(_score, measure, noteStartTime.ticks(), msTrack + msVoice, msMove, duration, dura, bm); // handle beam if (!chord) handleBeamAndStemDir(c, bm, stemDir, currBeam); // append any grace chord after chord to the previous chord Chord* prevChord = measure->findChord(prevSTime.ticks(), msTrack + msVoice); if (prevChord && prevChord != c) addGraceChordsAfter(prevChord, gcl, gac); // append any grace chord addGraceChordsBefore(c, gcl); } else { // grace note // TODO: check if explicit stem direction should also be set for grace notes // (the DOM parser does that, but seems to have no effect on the autotester) if (!chord || gcl.isEmpty()) { c = createGraceChord(_score, msTrack + msVoice, duration, graceSlash); // TODO FIX // the setStaffMove() below results in identical behaviour as 2.0: // grace note will be at the wrong staff with the wrong pitch, // seems to use the line value calculated for the right staff // leaving it places the note at the wrong staff with the right pitch // this affects only grace notes where staff move differs from // the main note, e.g. DebuMandSample.xml first grace in part 2 // c->setStaffMove(msMove); // END TODO gcl.append(c); } else c = gcl.last(); } note = new Note(_score); note->setSmall(small); note->setHeadGroup(headGroup); if (noteColor != QColor::Invalid) note->setColor(noteColor); setNoteHead(note, noteheadColor, noteheadParentheses, noteheadFilled); note->setVisible(printObject); // TODO also set the stem to invisible if (velocity > 0) { note->setVeloType(Note::ValueType::USER_VAL); note->setVeloOffset(velocity); } const MusicXMLDrumset& mxmlDrumset = _pass1.getDrumset(partId); if (mnp.unpitched()) { //&& drumsets.contains(partId) if (_hasDrumset && mxmlDrumset.contains(instrumentId)) { // step and oct are display-step and ...-oct // get pitch from instrument definition in drumset instead int pitch = mxmlDrumset[instrumentId].pitch; note->setPitch(limit(pitch, 0, 127)); // TODO - does this need to be key-aware? note->setTpc(pitch2tpc(pitch, Key::C, Prefer::NEAREST)); // TODO: necessary ? } else { //qDebug("disp step %d oct %d", displayStep, displayOctave); xmlSetPitch(note, mnp.displayStep(), 0, mnp.displayOctave(), 0, _pass1.getPart(partId)->instrument()); } } else { int ottavaStaff = (msTrack - _pass1.trackForPart(partId)) / VOICES; int octaveShift = _pass1.octaveShift(partId, ottavaStaff, noteStartTime); xmlSetPitch(note, mnp.step(), mnp.alter(), mnp.octave(), octaveShift, _pass1.getPart(partId)->instrument()); } // set drumset information // note that in MuseScore, the drumset contains defaults for notehead, // line and stem direction, while a MusicXML file contains actuals. // the MusicXML values for each note are simply copied to the defaults if (mnp.unpitched()) { // determine staff line based on display-step / -octave and clef type ClefType clef = c->staff()->clef(noteStartTime.ticks()); int po = ClefInfo::pitchOffset(clef); int pitch = MusicXMLStepAltOct2Pitch(mnp.displayStep(), 0, mnp.displayOctave()); int line = po - absStep(pitch); // correct for number of staff lines // see ExportMusicXml::unpitch2xml for explanation // TODO handle other # staff lines ? int staffLines = c->staff()->lines(0); if (staffLines == 1) line -= 8; if (staffLines == 3) line -= 2; // the drum palette cannot handle stem direction AUTO, // overrule if necessary if (stemDir == Direction::AUTO) { if (line > 4) stemDir = Direction::DOWN; else stemDir = Direction::UP; } /* if (drumsets.contains(partId) && mxmlDrumset.contains(instrId)) { mxmlDrumset[instrId].notehead = headGroup; mxmlDrumset[instrId].line = line; mxmlDrumset[instrId].stemDirection = sd; } */ // this should be done in pass 1, would make _pass1 const here _pass1.setDrumsetDefault(partId, instrumentId, headGroup, line, stemDir); } // accidental handling //qDebug("note acc %p type %hhd acctype %hhd", // acc, acc ? acc->accidentalType() : static_cast(0), accType); Accidental* acc = mnp.acc(); if (!acc && mnp.accType() != AccidentalType::NONE) { acc = new Accidental(_score); acc->setAccidentalType(mnp.accType()); } if (acc) { note->add(acc); // save alter value for user accidental if (acc->accidentalType() != AccidentalType::NONE) alt = mnp.alter(); } c->add(note); //c->setStemDirection(stemDir); // already done in handleBeamAndStemDir() c->setNoStem(noStem); cr = c; } // cr can be 0 here (if a rest cannot be added) // TODO: complete and cleanup handling this case if (cr) { cr->setVisible(printObject); if (cue) cr->setSmall(cue); // only once per chord } // handle the postponed children of // if one of these was found, the first while loop was terminated // at a StartElement instead of the usual EndElement QMap numberedLyrics; // lyrics with valid number QSet extendedLyrics; // lyrics with the extend flag set MusicXmlTupletDesc tupletDesc; bool lastGraceAFter = false; // set by notations() if end of grace after sequence found while (_e.tokenType() == QXmlStreamReader::StartElement) { //qDebug("in second loop element '%s'", qPrintable(_e.name().toString())); if (_e.name() == "lyric") { // lyrics on grace notes not (yet) supported by MuseScore if (!grace) lyric(partId, numberedLyrics, extendedLyrics); // TODO: move track handling to addlyric else { _logger->logDebugInfo("ignoring lyrics on grace notes", &_e); skipLogCurrElem(); } } else if (_e.name() == "notations") notations(note, cr, noteStartTime.ticks(), tupletDesc, lastGraceAFter); else skipLogCurrElem(); // skip to either start of next child or end of // currently at end of last child handled //qDebug("::note before skip tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); do _e.readNext(); while (!(_e.tokenType() == QXmlStreamReader::StartElement) && !(_e.tokenType() == QXmlStreamReader::EndElement && _e.name() == "note")); //qDebug("::note after skip tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); } //qDebug("::note after second loop tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); // handle grace after state: remember current grace list size if (grace && lastGraceAFter) gac = gcl.size(); if (cr) { if (!chord && !grace) { // do tuplet if valid time-modification is not 1/1 and is not 1/2 (tremolo) auto timeMod = mnd.timeMod(); if (timeMod.isValid() && timeMod != Fraction(1, 1) && timeMod != Fraction(1, 2)) { // find part-relative track Part* part = _pass1.getPart(partId); Q_ASSERT(part); int scoreRelStaff = _score->staffIdx(part); // zero-based number of parts first staff in the score int partRelTrack = msTrack + msVoice - scoreRelStaff * VOICES; addTupletToChord(cr, _tuplets[partRelTrack], _tuplImpls[partRelTrack], timeMod, tupletDesc, mnd.normalType()); } } } // add lyrics found by lyric if (cr) { // add lyrics and stop corresponding extends addLyrics(_logger, &_e, cr, numberedLyrics, extendedLyrics, _extendedLyrics); if (rest) { // stop all extends _extendedLyrics.setExtend(-1, cr->track(), cr->tick()); } } // add figured bass element addFiguredBassElemens(fbl, noteStartTime, msTrack, dura, measure); // don't count chord or grace note duration // note that this does not check the MusicXML requirement that notes in a chord // cannot have a duration longer than the first note in the chord if (chord || grace) dura.set(0, 1); if (!(_e.isEndElement() && _e.name() == "note")) qDebug("name %s line %lld", qPrintable(_e.name().toString()), _e.lineNumber()); Q_ASSERT(_e.isEndElement() && _e.name() == "note"); return note; } //--------------------------------------------------------- // notePrintSpacingNo //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note node for a note with print-spacing="no". These are handled like a forward: only moving the time forward. */ void MusicXMLParserPass2::notePrintSpacingNo(Fraction& dura) { Q_ASSERT(_e.isStartElement() && _e.name() == "note"); //_logger->logDebugTrace("MusicXMLParserPass1::notePrintSpacingNo", &_e); bool chord = false; bool grace = false; while (_e.readNextStartElement()) { if (_e.name() == "chord") { chord = true; _e.readNext(); } else if (_e.name() == "duration") duration(dura); else if (_e.name() == "grace") { grace = true; _e.readNext(); } else _e.skipCurrentElement(); // skip but don't log } // don't count chord or grace note duration // note that this does not check the MusicXML requirement that notes in a chord // cannot have a duration longer than the first note in the chord if (chord || grace) dura.set(0, 1); Q_ASSERT(_e.isEndElement() && _e.name() == "note"); } //--------------------------------------------------------- // calcTicks //--------------------------------------------------------- static Fraction calcTicks(const QString& text, int divs) { Fraction dura(0, 0); // invalid unless set correctly int intDura = text.toInt(); if (divs > 0) dura.set(intDura, 4 * divs); else qDebug("illegal or uninitialized divisions (%d)", divs); // TODO return dura; } //--------------------------------------------------------- // duration //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/duration node. */ void MusicXMLParserPass2::duration(Fraction& dura) { Q_ASSERT(_e.isStartElement() && _e.name() == "duration"); dura.set(0, 0); // invalid unless set correctly int intDura = _e.readElementText().toInt(); if (intDura > 0) { if (_divs > 0) { dura.set(intDura, 4 * _divs); dura.reduce(); // prevent overflow in later Fraction operations } else _logger->logError(QString("illegal or uninitialized divisions (%1)").arg(_divs), &_e); } else _logger->logError(QString("illegal duration %1").arg(dura.print()), &_e); //qDebug("duration %s valid %d", qPrintable(dura.print()), dura.isValid()); } //--------------------------------------------------------- // figure //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/harmony/figured-bass/figure node. Return the result as a FiguredBassItem. */ FiguredBassItem* MusicXMLParserPass2::figure(const int idx, const bool paren) { Q_ASSERT(_e.isStartElement() && _e.name() == "figure"); FiguredBassItem* fgi = new FiguredBassItem(_score, idx); // read the figure while (_e.readNextStartElement()) { if (_e.name() == "extend") { QStringRef type = _e.attributes().value("type"); if (type == "start") fgi->setContLine(FiguredBassItem::ContLine::EXTENDED); else if (type == "continue") fgi->setContLine(FiguredBassItem::ContLine::EXTENDED); else if (type == "stop") fgi->setContLine(FiguredBassItem::ContLine::SIMPLE); _e.skipCurrentElement(); } else if (_e.name() == "figure-number") { QString val = _e.readElementText(); int iVal = val.toInt(); // MusicXML spec states figure-number is a number // MuseScore can only handle single digit if (1 <= iVal && iVal <= 9) fgi->setDigit(iVal); else _logger->logError(QString("incorrect figure-number '%1'").arg(val), &_e); } else if (_e.name() == "prefix") fgi->setPrefix(fgi->MusicXML2Modifier(_e.readElementText())); else if (_e.name() == "suffix") fgi->setSuffix(fgi->MusicXML2Modifier(_e.readElementText())); else skipLogCurrElem(); } // set parentheses if (paren) { // parenthesis open if (fgi->prefix() != FiguredBassItem::Modifier::NONE) fgi->setParenth1(FiguredBassItem::Parenthesis::ROUNDOPEN); // before prefix else if (fgi->digit() != FBIDigitNone) fgi->setParenth2(FiguredBassItem::Parenthesis::ROUNDOPEN); // before digit else if (fgi->suffix() != FiguredBassItem::Modifier::NONE) fgi->setParenth3(FiguredBassItem::Parenthesis::ROUNDOPEN); // before suffix // parenthesis close if (fgi->suffix() != FiguredBassItem::Modifier::NONE) fgi->setParenth4(FiguredBassItem::Parenthesis::ROUNDCLOSED); // after suffix else if (fgi->digit() != FBIDigitNone) fgi->setParenth3(FiguredBassItem::Parenthesis::ROUNDCLOSED); // after digit else if (fgi->prefix() != FiguredBassItem::Modifier::NONE) fgi->setParenth2(FiguredBassItem::Parenthesis::ROUNDCLOSED); // after prefix } return fgi; } //--------------------------------------------------------- // figuredBass //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/harmony/figured-bass node. TODO check description: // Set the FiguredBass state based on the MusicXML node de. // Note that onNote and ticks must be set by the MusicXML importer, // as the required context is not present in the items DOM tree. // Exception: if a element is present, tick can be set. Return the result as a FiguredBass if valid, non-empty figure(s) are found. Return 0 in case of error. */ FiguredBass* MusicXMLParserPass2::figuredBass() { Q_ASSERT(_e.isStartElement() && _e.name() == "figured-bass"); FiguredBass* fb = new FiguredBass(_score); bool parentheses = _e.attributes().value("parentheses") == "yes"; QString normalizedText; int idx = 0; while (_e.readNextStartElement()) { if (_e.name() == "duration") { Fraction dura; duration(dura); if (dura.isValid() && dura > Fraction(0, 1)) fb->setTicks(dura.ticks()); } else if (_e.name() == "figure") { FiguredBassItem* pItem = figure(idx++, parentheses); pItem->setTrack(0 /* TODO fb->track() */); pItem->setParent(fb); fb->appendItem(pItem); // add item normalized text if (!normalizedText.isEmpty()) normalizedText.append('\n'); normalizedText.append(pItem->normalizedText()); } else { skipLogCurrElem(); delete fb; return 0; } } fb->setXmlText(normalizedText); // this is the text to show while editing if (normalizedText.isEmpty()) { delete fb; return 0; } return fb; } //--------------------------------------------------------- // frame //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/harmony/frame node. Return the result as a FretDiagram. */ FretDiagram* MusicXMLParserPass2::frame() { Q_ASSERT(_e.isStartElement() && _e.name() == "frame"); FretDiagram* fd = new FretDiagram(_score); while (_e.readNextStartElement()) { if (_e.name() == "frame-frets") { int val = _e.readElementText().toInt(); if (val > 0) fd->setFrets(val); else _logger->logError(QString("FretDiagram::readMusicXML: illegal frame-fret %1").arg(val), &_e); } else if (_e.name() == "frame-note") { int fret = -1; int string = -1; while (_e.readNextStartElement()) { if (_e.name() == "fret") fret = _e.readElementText().toInt(); else if (_e.name() == "string") string = _e.readElementText().toInt(); else skipLogCurrElem(); } _logger->logDebugInfo(QString("FretDiagram::readMusicXML string %1 fret %2").arg(string).arg(fret), &_e); if (string > 0) { if (fret == 0) fd->setMarker(fd->strings() - string, 79 /* ??? */); else if (fret > 0) fd->setDot(fd->strings() - string, fret); } } else if (_e.name() == "frame-strings") { int val = _e.readElementText().toInt(); if (val > 0) { fd->setStrings(val); for (int i = 0; i < val; ++i) fd->setMarker(i, 88 /* ??? */); } else _logger->logError(QString("FretDiagram::readMusicXML: illegal frame-strings %1").arg(val), &_e); } else skipLogCurrElem(); } return fd; } //--------------------------------------------------------- // harmony //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/harmony node. */ void MusicXMLParserPass2::harmony(const QString& partId, Measure* measure, const Fraction sTime) { Q_ASSERT(_e.isStartElement() && _e.name() == "harmony"); int track = _pass1.trackForPart(partId); // placement: // in order to work correctly, this should probably be adjusted to account for spatium // but in any case, we don't support import relative-x/y for other elements // no reason to do so for chord symbols #if 0 // TODO:ws double rx = 0.0; // 0.1 * e.attribute("relative-x", "0").toDouble(); double ry = 0.0; // -0.1 * e.attribute("relative-y", "0").toDouble(); double styleYOff = _score->textStyle(Tid::HARMONY).offset().y(); OffsetType offsetType = _score->textStyle(Tid::HARMONY).offsetType(); if (offsetType == OffsetType::ABS) { styleYOff = styleYOff * DPMM / _score->spatium(); } // TODO: check correct dy handling // previous code: double dy = -0.1 * e.attribute("default-y", QString::number(styleYOff* -10)).toDouble(); double dy = -0.1 * _e.attributes().value("default-y").toDouble(); #endif bool printObject = _e.attributes().value("print-object") != "no"; QString printFrame = _e.attributes().value("print-frame").toString(); QString printStyle = _e.attributes().value("print-style").toString(); QString kind, kindText, symbols, parens; QList degreeList; /* TODO ? if (harmony) { qDebug("MusicXML::import: more than one harmony"); return; } */ FretDiagram* fd = 0; Harmony* ha = new Harmony(_score); //TODO:ws ha->setUserOff(QPointF(rx, ry + dy - styleYOff)); Fraction offset; while (_e.readNextStartElement()) { if (_e.name() == "root") { QString step; int alter = 0; bool invalidRoot = false; while (_e.readNextStartElement()) { if (_e.name() == "root-step") { // attributes: print-style step = _e.readElementText(); /* TODO: check if this is required if (ee.hasAttribute("text")) { QString rtext = ee.attribute("text"); if (rtext == "") { invalidRoot = true; } } */ } else if (_e.name() == "root-alter") { // attributes: print-object, print-style // location (left-right) alter = _e.readElementText().toInt(); } else skipLogCurrElem(); } if (invalidRoot) ha->setRootTpc(Tpc::TPC_INVALID); else ha->setRootTpc(step2tpc(step, AccidentalVal(alter))); } else if (_e.name() == "function") { // attributes: print-style skipLogCurrElem(); } else if (_e.name() == "kind") { // attributes: use-symbols yes-no // text, stack-degrees, parentheses-degree, bracket-degrees, // print-style, halign, valign kindText = _e.attributes().value("text").toString(); symbols = _e.attributes().value("use-symbols").toString(); parens = _e.attributes().value("parentheses-degrees").toString(); kind = _e.readElementText(); } else if (_e.name() == "inversion") { // attributes: print-style skipLogCurrElem(); } else if (_e.name() == "bass") { QString step; int alter = 0; while (_e.readNextStartElement()) { if (_e.name() == "bass-step") { // attributes: print-style step = _e.readElementText(); } else if (_e.name() == "bass-alter") { // attributes: print-object, print-style // location (left-right) alter = _e.readElementText().toInt(); } else skipLogCurrElem(); } ha->setBaseTpc(step2tpc(step, AccidentalVal(alter))); } else if (_e.name() == "degree") { int degreeValue = 0; int degreeAlter = 0; QString degreeType = ""; while (_e.readNextStartElement()) { if (_e.name() == "degree-value") { degreeValue = _e.readElementText().toInt(); } else if (_e.name() == "degree-alter") { degreeAlter = _e.readElementText().toInt(); } else if (_e.name() == "degree-type") { degreeType = _e.readElementText(); } else skipLogCurrElem(); } if (degreeValue <= 0 || degreeValue > 13 || degreeAlter < -2 || degreeAlter > 2 || (degreeType != "add" && degreeType != "alter" && degreeType != "subtract")) { _logger->logError(QString("incorrect degree: degreeValue=%1 degreeAlter=%2 degreeType=%3") .arg(degreeValue).arg(degreeAlter).arg(degreeType), &_e); } else { if (degreeType == "add") degreeList << HDegree(degreeValue, degreeAlter, HDegreeType::ADD); else if (degreeType == "alter") degreeList << HDegree(degreeValue, degreeAlter, HDegreeType::ALTER); else if (degreeType == "subtract") degreeList << HDegree(degreeValue, degreeAlter, HDegreeType::SUBTRACT); } } else if (_e.name() == "frame") fd = frame(); else if (_e.name() == "level") skipLogCurrElem(); else if (_e.name() == "offset") offset = calcTicks(_e.readElementText(), _divs); else if (_e.name() == "staff") { int nstaves = _pass1.getPart(partId)->nstaves(); QString strStaff = _e.readElementText(); int staff = strStaff.toInt(); if (0 < staff && staff <= nstaves) track += (staff - 1) * VOICES; else _logger->logError(QString("invalid staff %1").arg(strStaff), &_e); } else skipLogCurrElem(); } if (fd) { fd->setTrack(track); Segment* s = measure->getSegment(SegmentType::ChordRest, (sTime + offset).ticks()); s->add(fd); } const ChordDescription* d = 0; if (ha->rootTpc() != Tpc::TPC_INVALID) d = ha->fromXml(kind, kindText, symbols, parens, degreeList); if (d) { ha->setId(d->id); ha->setTextName(d->names.front()); } else { ha->setId(-1); ha->setTextName(kindText); } ha->render(); ha->setVisible(printObject); // TODO-LV: do this only if ha points to a valid harmony // harmony = ha; ha->setTrack(track); Segment* s = measure->getSegment(SegmentType::ChordRest, (sTime + offset).ticks()); s->add(ha); } //--------------------------------------------------------- // beam //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/beam node. Sets beamMode in case of begin, continue or end beam number 1. */ void MusicXMLParserPass2::beam(Beam::Mode& beamMode) { Q_ASSERT(_e.isStartElement() && _e.name() == "beam"); int beamNo = _e.attributes().value("number").toInt(); if (beamNo == 1) { QString s = _e.readElementText(); if (s == "begin") beamMode = Beam::Mode::BEGIN; else if (s == "end") beamMode = Beam::Mode::END; else if (s == "continue") beamMode = Beam::Mode::MID; else if (s == "backward hook") ; else if (s == "forward hook") ; else _logger->logError(QString("unknown beam keyword '%1'").arg(s), &_e); } else _e.skipCurrentElement(); } //--------------------------------------------------------- // forward //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/forward node. */ void MusicXMLParserPass2::forward(Fraction& dura) { Q_ASSERT(_e.isStartElement() && _e.name() == "forward"); while (_e.readNextStartElement()) { if (_e.name() == "duration") duration(dura); else if (_e.name() == "staff") _e.skipCurrentElement(); // skip but don't log else if (_e.name() == "voice") _e.skipCurrentElement(); // skip but don't log else skipLogCurrElem(); } } //--------------------------------------------------------- // backup //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/backup node. */ void MusicXMLParserPass2::backup(Fraction& dura) { Q_ASSERT(_e.isStartElement() && _e.name() == "backup"); while (_e.readNextStartElement()) { if (_e.name() == "duration") duration(dura); else skipLogCurrElem(); } } //--------------------------------------------------------- // lyric -- parse a MusicXML lyric element //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/lyric node. */ void MusicXMLParserPass2::lyric(const QString& partId, QMap& numbrdLyrics, QSet& extLyrics) { Q_ASSERT(_e.isStartElement() && _e.name() == "lyric"); Lyrics* l = new Lyrics(_score); // TODO in addlyrics: l->setTrack(trk); bool hasExtend = false; QString lyricNumber = _e.attributes().value("number").toString(); QColor lyricColor = QColor::Invalid; lyricColor.setNamedColor(_e.attributes().value("color").toString()); QString extendType; QString formattedText; while (_e.readNextStartElement()) { if (_e.name() == "elision") { // TODO verify elision handling /* QString text = _e.readElementText(); if (text.isEmpty()) formattedText += " "; else */ formattedText += nextPartOfFormattedString(_e); } else if (_e.name() == "extend") { hasExtend = true; extendType = _e.attributes().value("type").toString(); _e.readNext(); } else if (_e.name() == "syllabic") { QString syll = _e.readElementText(); if (syll == "single") l->setSyllabic(Lyrics::Syllabic::SINGLE); else if (syll == "begin") l->setSyllabic(Lyrics::Syllabic::BEGIN); else if (syll == "end") l->setSyllabic(Lyrics::Syllabic::END); else if (syll == "middle") l->setSyllabic(Lyrics::Syllabic::MIDDLE); else qDebug("unknown syllabic %s", qPrintable(syll)); // TODO } else if (_e.name() == "text") formattedText += nextPartOfFormattedString(_e); else skipLogCurrElem(); } // if no lyric read (e.g. only 'extend "type=stop"'), no further action required if (formattedText == "") { delete l; return; } auto mxmlPart = _pass1.getMusicXmlPart(partId); auto lyricNo = mxmlPart.lyricNumberHandler().getLyricNo(lyricNumber); if (lyricNo < 0) { _logger->logError("invalid lyrics number (<0)", &_e); delete l; return; } else if (lyricNo > MAX_LYRICS) { _logger->logError(QString("too much lyrics (>%1)").arg(MAX_LYRICS), &_e); delete l; return; } else if (numbrdLyrics.contains(lyricNo)) { _logger->logError(QString("duplicate lyrics number (%1)").arg(lyricNumber), &_e); delete l; return; } numbrdLyrics[lyricNo] = l; if (hasExtend && (extendType == "" || extendType == "start")) extLyrics.insert(l); //qDebug("formatted lyric '%s'", qPrintable(formattedText)); l->setXmlText(formattedText); if (lyricColor != QColor::Invalid) l->setColor(lyricColor); } //--------------------------------------------------------- // slur //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/slur node. */ void MusicXMLParserPass2::slur(ChordRest* cr, const int tick, const int track, bool& lastGraceAFter) { Q_ASSERT(_e.isStartElement() && _e.name() == "slur"); int slurNo = _e.attributes().value("number").toString().toInt(); if (slurNo > 0) slurNo--; QString slurType = _e.attributes().value("type").toString(); QString lineType = _e.attributes().value("line-type").toString(); if (lineType == "") lineType = "solid"; // PriMus Music-Notation by Columbussoft (build 10093) generates overlapping // slurs that do not have a number attribute to distinguish them. // The duplicates must be ignored, to prevent memory allocation issues, // which caused a MuseScore crash // Similar issues happen with Sibelius 7.1.3 (direct export) if (slurType == "start") { if (_slurs[slurNo].isStart()) // slur start when slur already started: report error _logger->logError(QString("ignoring duplicate slur start"), &_e); else if (_slurs[slurNo].isStop()) { // slur start when slur already stopped: wrap up Slur* newSlur = _slurs[slurNo].slur(); newSlur->setTick(tick); newSlur->setStartElement(cr); _slurs[slurNo] = SlurDesc(); } else { // slur start for new slur: init Slur* newSlur = new Slur(_score); if (cr->isGrace()) newSlur->setAnchor(Spanner::Anchor::CHORD); if (lineType == "dotted") newSlur->setLineType(1); else if (lineType == "dashed") newSlur->setLineType(2); newSlur->setTick(tick); newSlur->setStartElement(cr); QString pl = _e.attributes().value("placement").toString(); if (pl == "above") newSlur->setSlurDirection(Direction::UP); else if (pl == "below") newSlur->setSlurDirection(Direction::DOWN); newSlur->setTrack(track); newSlur->setTrack2(track); _slurs[slurNo].start(newSlur); _score->addElement(newSlur); } } else if (slurType == "stop") { if (_slurs[slurNo].isStart()) { // slur stop when slur already started: wrap up Slur* newSlur = _slurs[slurNo].slur(); if (!(cr->isGrace())) { newSlur->setTick2(tick); newSlur->setTrack2(track); } newSlur->setEndElement(cr); _slurs[slurNo] = SlurDesc(); } else if (_slurs[slurNo].isStop()) // slur stop when slur already stopped: report error _logger->logError(QString("ignoring duplicate slur stop"), &_e); else { // slur stop for new slur: init Slur* newSlur = new Slur(_score); if (!(cr->isGrace())) { newSlur->setTick2(tick); newSlur->setTrack2(track); } newSlur->setEndElement(cr); _slurs[slurNo].stop(newSlur); } // any grace note containing a slur stop means // last note of a grace after set has been found if (cr->isGrace()) lastGraceAFter = true; } else if (slurType == "continue") ; // ignore else _logger->logError(QString("unknown slur type %1").arg(slurType), &_e); _e.readNext(); } //--------------------------------------------------------- // tied //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/tied node. */ void MusicXMLParserPass2::tied(Note* note, const int track) { Q_ASSERT(_e.isStartElement() && _e.name() == "tied"); QString tiedType = _e.attributes().value("type").toString(); if (tiedType == "start") { if (_tie) { _logger->logError(QString("Tie already active"), &_e); } else if (note) { _tie = new Tie(_score); note->setTieFor(_tie); _tie->setStartNote(note); _tie->setTrack(track); QString tiedOrientation = _e.attributes().value("orientation").toString(); if (tiedOrientation == "over") _tie->setSlurDirection(Direction::UP); else if (tiedOrientation == "under") _tie->setSlurDirection(Direction::DOWN); else if (tiedOrientation == "auto") ; // ignore else if (tiedOrientation == "") ; // ignore else _logger->logError(QString("unknown tied orientation: %1").arg(tiedOrientation), &_e); QString lineType = _e.attributes().value("line-type").toString(); if (lineType == "dotted") _tie->setLineType(1); else if (lineType == "dashed") _tie->setLineType(2); _tie = 0; } } else if (tiedType == "stop") ; // ignore else _logger->logError(QString("unknown tied type %").arg(tiedType), &_e); _e.readNext(); } //--------------------------------------------------------- // dynamics //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/dynamics node. */ void MusicXMLParserPass2::dynamics(QString& placement, QStringList& dynamicslist) { Q_ASSERT(_e.isStartElement() && _e.name() == "dynamics"); placement = _e.attributes().value("placement").toString(); if (preferences.getBool(PREF_IMPORT_MUSICXML_IMPORTLAYOUT)) { // ry = ee.attribute(QString("relative-y"), "0").toDouble() * -.1; // rx = ee.attribute(QString("relative-x"), "0").toDouble() * .1; // yoffset = _e.attributes().value("default-y").toDouble(&hasYoffset) * -0.1; // xoffset = ee.attribute("default-x", "0.0").toDouble() * 0.1; } while (_e.readNextStartElement()) { if (_e.name() == "other-dynamics") dynamicslist.push_back(_e.readElementText()); else { dynamicslist.push_back(_e.name().toString()); _e.readNext(); } } } //--------------------------------------------------------- // articulations //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/articulations node. Note that some notations attach to notes only in MuseScore, which means trying to attach them to a rest will crash, as in that case note is 0. */ void MusicXMLParserPass2::articulations(ChordRest* cr, SymId& breath, QString& chordLineType) { Q_ASSERT(_e.isStartElement() && _e.name() == "articulations"); while (_e.readNextStartElement()) { if (addMxmlArticulationToChord(cr, _e.name().toString())) { _e.readNext(); continue; } else if (_e.name() == "breath-mark") { breath = SymId::breathMarkComma; _e.readElementText(); // TODO: handle value read (note: encoding unknown, only "comma" found) } else if (_e.name() == "caesura") { breath = SymId::caesura; _e.readNext(); } else if (_e.name() == "doit" || _e.name() == "falloff" || _e.name() == "plop" || _e.name() == "scoop") { chordLineType = _e.name().toString(); _e.readNext(); } else if (_e.name() == "strong-accent") { QString strongAccentType = _e.attributes().value("type").toString(); if (strongAccentType == "up" || strongAccentType == "") addArticulationToChord(cr, SymId::articMarcatoAbove, "up"); else if (strongAccentType == "down") addArticulationToChord(cr, SymId::articMarcatoAbove, "down"); else _logger->logError(QString("unknown mercato type %1").arg(strongAccentType), &_e); _e.readNext(); } else skipLogCurrElem(); } //qDebug("::notations tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); } //--------------------------------------------------------- // ornaments //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/ornaments node. */ void MusicXMLParserPass2::ornaments(ChordRest* cr, QString& wavyLineType, int& wavyLineNo, QString& tremoloType, int& tremoloNr, bool& lastGraceAFter) { Q_ASSERT(_e.isStartElement() && _e.name() == "ornaments"); bool trillMark = false; // while (_e.readNextStartElement()) { if (addMxmlArticulationToChord(cr, _e.name().toString())) { _e.readNext(); continue; } else if (_e.name() == "trill-mark") { trillMark = true; _e.readNext(); } else if (_e.name() == "wavy-line") { wavyLineType = _e.attributes().value("type").toString(); wavyLineNo = _e.attributes().value("number").toString().toInt(); if (wavyLineNo > 0) wavyLineNo--; // any grace note containing a wavy-line stop means // last note of a grace after set has been found if (wavyLineType == "stop" && cr->isGrace()) lastGraceAFter = true; _e.readNext(); } else if (_e.name() == "tremolo") { tremoloType = _e.attributes().value("type").toString(); tremoloNr = _e.readElementText().toInt(); } else if (_e.name() == "accidental-mark") skipLogCurrElem(); else if (_e.name() == "delayed-turn") { // TODO: actually this should be offset a bit to the right addArticulationToChord(cr, SymId::ornamentTurn, ""); _e.readNext(); } else if (_e.name() == "inverted-mordent" || _e.name() == "mordent") { addMordentToChord(cr, _e.name().toString(), _e.attributes().value("long").toString(), _e.attributes().value("approach").toString(), _e.attributes().value("departure").toString()); _e.readNext(); } else skipLogCurrElem(); } //qDebug("::notations tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); // note that mscore wavy line already implicitly includes a trillsym // so don't add an additional one if (trillMark && wavyLineType != "start") addArticulationToChord(cr, SymId::ornamentTrill, ""); } //--------------------------------------------------------- // technical //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/technical node. */ void MusicXMLParserPass2::technical(Note* note, ChordRest* cr) { Q_ASSERT(_e.isStartElement() && _e.name() == "technical"); while (_e.readNextStartElement()) { if (addMxmlArticulationToChord(cr, _e.name().toString())) { _e.readNext(); continue; } else if (_e.name() == "fingering") // TODO: distinguish between keyboards (style Tid::FINGERING) // and (plucked) strings (style Tid::LH_GUITAR_FINGERING) addTextToNote(_e.lineNumber(), _e.columnNumber(), _e.readElementText(), Tid::FINGERING, _score, note); else if (_e.name() == "fret") { int fret = _e.readElementText().toInt(); if (note) { if (note->staff()->isTabStaff(0)) note->setFret(fret); } else _logger->logError("no note for fret", &_e); } else if (_e.name() == "pluck") addTextToNote(_e.lineNumber(), _e.columnNumber(), _e.readElementText(), Tid::RH_GUITAR_FINGERING, _score, note); else if (_e.name() == "string") { QString txt = _e.readElementText(); if (note) { if (note->staff()->isTabStaff(0)) note->setString(txt.toInt() - 1); else addTextToNote(_e.lineNumber(), _e.columnNumber(), txt, Tid::STRING_NUMBER, _score, note); } else _logger->logError("no note for string", &_e); } else if (_e.name() == "pull-off") skipLogCurrElem(); else skipLogCurrElem(); } //qDebug("::notations tokenString '%s' name '%s'", qPrintable(_e.tokenString()), qPrintable(_e.name().toString())); } //--------------------------------------------------------- // glissando //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/glissando and /score-partwise/part/measure/note/notations/slide nodes. */ void MusicXMLParserPass2::glissando(Note* note, const int tick, const int ticks, const int track) { Q_ASSERT(_e.isStartElement() && (_e.name() == "glissando" || _e.name() == "slide")); int n = _e.attributes().value("number").toString().toInt(); if (n > 0) n--; QString spannerType = _e.attributes().value("type").toString(); int tag = _e.name() == "slide" ? 0 : 1; // QString lineType = ee.attribute(QString("line-type"), "solid"); Glissando*& gliss = _glissandi[n][tag]; if (spannerType == "start") { QColor color(_e.attributes().value("color").toString()); QString glissText = _e.readElementText(); if (gliss) { _logger->logError(QString("overlapping glissando/slide number %1").arg(n+1), &_e); } else if (!note) { _logger->logError(QString("no note for glissando/slide number %1 start").arg(n+1), &_e); } else { gliss = new Glissando(_score); gliss->setAnchor(Spanner::Anchor::NOTE); gliss->setStartElement(note); gliss->setTick(tick); gliss->setTrack(track); gliss->setParent(note); if (color.isValid()) gliss->setColor(color); gliss->setText(glissText); gliss->setGlissandoType(tag == 0 ? GlissandoType::STRAIGHT : GlissandoType::WAVY); _spanners[gliss] = QPair(tick, -1); // qDebug("glissando/slide=%p inserted at first tick %d", gliss, tick); } } else if (spannerType == "stop") { if (!gliss) { _logger->logError(QString("glissando/slide number %1 stop without start").arg(n+1), &_e); } else if (!note) { _logger->logError(QString("no note for glissando/slide number %1 stop").arg(n+1), &_e); } else { _spanners[gliss].second = tick + ticks; gliss->setEndElement(note); gliss->setTick2(tick); gliss->setTrack2(track); // qDebug("glissando/slide=%p second tick %d", gliss, tick); gliss = 0; } } else _logger->logError(QString("unknown glissando/slide type %1").arg(spannerType), &_e); _e.readNext(); } //--------------------------------------------------------- // addArpeggio //--------------------------------------------------------- static void addArpeggio(ChordRest* cr, const QString& arpeggioType, MxmlLogger* logger, const QXmlStreamReader* const xmlreader) { // no support for arpeggio on rest if (!arpeggioType.isEmpty() && cr->type() == ElementType::CHORD) { Arpeggio* a = new Arpeggio(cr->score()); if (arpeggioType == "none") a->setArpeggioType(ArpeggioType::NORMAL); else if (arpeggioType == "up") a->setArpeggioType(ArpeggioType::UP); else if (arpeggioType == "down") a->setArpeggioType(ArpeggioType::DOWN); else if (arpeggioType == "non-arpeggiate") a->setArpeggioType(ArpeggioType::BRACKET); else { logger->logError(QString("unknown arpeggio type %1").arg(arpeggioType), xmlreader); delete a; a = 0; } if ((static_cast(cr))->arpeggio()) { // there can be only one delete a; a = 0; } else cr->add(a); } } //--------------------------------------------------------- // addTremolo //--------------------------------------------------------- static void addTremolo(ChordRest* cr, const int tremoloNr, const QString& tremoloType, const int ticks, Chord*& tremStart, MxmlLogger* logger, const QXmlStreamReader* const xmlreader) { if (!cr->isChord()) return; if (tremoloNr) { //qDebug("tremolo %d type '%s' ticks %d tremStart %p", tremoloNr, qPrintable(tremoloType), ticks, _tremStart); if (tremoloNr == 1 || tremoloNr == 2 || tremoloNr == 3 || tremoloNr == 4) { if (tremoloType == "" || tremoloType == "single") { Tremolo* t = new Tremolo(cr->score()); switch (tremoloNr) { case 1: t->setTremoloType(TremoloType::R8); break; case 2: t->setTremoloType(TremoloType::R16); break; case 3: t->setTremoloType(TremoloType::R32); break; case 4: t->setTremoloType(TremoloType::R64); break; } cr->add(t); } else if (tremoloType == "start") { if (tremStart) logger->logError("MusicXML::import: double tremolo start", xmlreader); tremStart = static_cast(cr); } else if (tremoloType == "stop") { if (tremStart) { Tremolo* t = new Tremolo(cr->score()); switch (tremoloNr) { case 1: t->setTremoloType(TremoloType::C8); break; case 2: t->setTremoloType(TremoloType::C16); break; case 3: t->setTremoloType(TremoloType::C32); break; case 4: t->setTremoloType(TremoloType::C64); break; } t->setChords(tremStart, static_cast(cr)); // fixup chord duration and type const int tremDur = ticks / 2; t->chord1()->setDurationType(tremDur); t->chord1()->setDuration(Fraction::fromTicks(tremDur)); t->chord2()->setDurationType(tremDur); t->chord2()->setDuration(Fraction::fromTicks(tremDur)); // add tremolo to first chord (only) tremStart->add(t); } else logger->logError("MusicXML::import: double tremolo stop w/o start", xmlreader); tremStart = 0; } } else logger->logError(QString("unknown tremolo type %1").arg(tremoloNr), xmlreader); } } //--------------------------------------------------------- // addWavyLine //--------------------------------------------------------- static void addWavyLine(ChordRest* cr, const int tick, const int wavyLineNo, const QString& wavyLineType, MusicXmlSpannerMap& spanners, TrillStack& trills, MxmlLogger* logger, const QXmlStreamReader* const xmlreader) { if (!wavyLineType.isEmpty()) { const auto ticks = cr->duration().ticks(); const auto track = cr->track(); const auto trk = (track / VOICES) * VOICES; // first track of staff Trill*& t = trills[wavyLineNo]; if (wavyLineType == "start") { if (t) { logger->logError(QString("overlapping wavy-line number %1").arg(wavyLineNo+1), xmlreader); } else { t = new Trill(cr->score()); t->setTrack(trk); spanners[t] = QPair(tick, -1); // qDebug("wedge trill=%p inserted at first tick %d", trill, tick); } } else if (wavyLineType == "stop") { if (!t) { logger->logError(QString("wavy-line number %1 stop without start").arg(wavyLineNo+1), xmlreader); } else { spanners[t].second = tick + ticks; // qDebug("wedge trill=%p second tick %d", trill, tick); t = 0; } } else logger->logError(QString("unknown wavy-line type %1").arg(wavyLineType), xmlreader); } } //--------------------------------------------------------- // addBreath //--------------------------------------------------------- static void addBreath(ChordRest* cr, const int tick, SymId breath) { if (breath != SymId::noSym && !cr->isGrace()) { Breath* b = new Breath(cr->score()); // b->setTrack(trk + voice); TODO check next line b->setTrack(cr->track()); b->setSymId(breath); const auto ticks = cr->duration().ticks(); auto seg = cr->measure()->getSegment(SegmentType::Breath, tick + ticks); seg->add(b); } } //--------------------------------------------------------- // addChordLine //--------------------------------------------------------- static void addChordLine(Note* note, const QString& chordLineType, MxmlLogger* logger, const QXmlStreamReader* const xmlreader) { if (chordLineType != "") { if (note) { ChordLine* cl = new ChordLine(note->score()); if (chordLineType == "falloff") cl->setChordLineType(ChordLineType::FALL); if (chordLineType == "doit") cl->setChordLineType(ChordLineType::DOIT); if (chordLineType == "plop") cl->setChordLineType(ChordLineType::PLOP); if (chordLineType == "scoop") cl->setChordLineType(ChordLineType::SCOOP); note->chord()->add(cl); } else logger->logError(QString("no note for %1").arg(chordLineType), xmlreader); } } //--------------------------------------------------------- // notations //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations node. Note that some notations attach to notes only in MuseScore, which means trying to attach them to a rest will crash, as in that case note is a nullptr. */ void MusicXMLParserPass2::notations(Note* note, ChordRest* cr, const int tick, MusicXmlTupletDesc& tupletDesc, bool& lastGraceAFter) { Q_ASSERT(_e.isStartElement() && _e.name() == "notations"); if (cr) { lastGraceAFter = false; // ensure default Measure* measure = cr->measure(); int ticks = cr->duration().ticks(); int track = cr->track(); QString wavyLineType; int wavyLineNo = 0; QString arpeggioType; SymId breath = SymId::noSym; int tremoloNr = 0; QString tremoloType; QString placement; QStringList dynamicslist; // qreal rx = 0.0; // qreal ry = 0.0; // qreal yoffset = 0.0; // actually this is default-y // qreal xoffset = 0.0; // not used // bool hasYoffset = false; QString chordLineType; while (_e.readNextStartElement()) { if (_e.name() == "slur") { slur(cr, tick, track, lastGraceAFter); } else if (_e.name() == "tied") { tied(note, track); } else if (_e.name() == "tuplet") { tuplet(tupletDesc); } else if (_e.name() == "dynamics") { placement = _e.attributes().value("placement").toString(); if (preferences.getBool(PREF_IMPORT_MUSICXML_IMPORTLAYOUT)) { // ry = ee.attribute(QString("relative-y"), "0").toDouble() * -.1; // rx = ee.attribute(QString("relative-x"), "0").toDouble() * .1; // yoffset = _e.attributes().value("default-y").toDouble(&hasYoffset) * -0.1; // xoffset = ee.attribute("default-x", "0.0").toDouble() * 0.1; } dynamics(placement, dynamicslist); } else if (_e.name() == "articulations") { articulations(cr, breath, chordLineType); } else if (_e.name() == "fermata") fermata(cr); else if (_e.name() == "ornaments") { ornaments(cr, wavyLineType, wavyLineNo, tremoloType, tremoloNr, lastGraceAFter); } else if (_e.name() == "technical") { technical(note, cr); } else if (_e.name() == "arpeggiate") { arpeggioType = _e.attributes().value("direction").toString(); if (arpeggioType == "") arpeggioType = "none"; _e.readNext(); } else if (_e.name() == "non-arpeggiate") { arpeggioType = "non-arpeggiate"; _e.readNext(); } else if (_e.name() == "glissando" || _e.name() == "slide") { glissando(note, tick, ticks, track); } else skipLogCurrElem(); } addArpeggio(cr, arpeggioType, _logger, &_e); addWavyLine(cr, tick, wavyLineNo, wavyLineType, _spanners, _trills, _logger, &_e); addBreath(cr, tick, breath); addTremolo(cr, tremoloNr, tremoloType, ticks, _tremStart, _logger, &_e); addChordLine(note, chordLineType, _logger, &_e); // more than one dynamic ??? // LVIFIX: check import/export of unknown_text // TODO remove duplicate code (see MusicXml::direction) for (QStringList::Iterator it = dynamicslist.begin(); it != dynamicslist.end(); ++it ) { Dynamic* dyn = new Dynamic(_score); dyn->setDynamicType(*it); //TODO:ws if (hasYoffset) dyn->textStyle().setYoff(yoffset); addElemOffset(dyn, track, placement, measure, tick); } } else { _logger->logDebugInfo("no note to attach to, skipping notations", &_e); _e.skipCurrentElement(); } Q_ASSERT(_e.isEndElement() && _e.name() == "notations"); } //--------------------------------------------------------- // stem //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/stem node. */ void MusicXMLParserPass2::stem(Direction& sd, bool& nost) { Q_ASSERT(_e.isStartElement() && _e.name() == "stem"); // defaults sd = Direction::AUTO; nost = false; QString s = _e.readElementText(); if (s == "up") sd = Direction::UP; else if (s == "down") sd = Direction::DOWN; else if (s == "none") nost = true; else if (s == "double") ; else _logger->logError(QString("unknown stem direction %1").arg(s), &_e); } //--------------------------------------------------------- // fermata //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/fermata node. Note: MusicXML common.mod: "An empty fermata element represents a normal fermata." */ void MusicXMLParserPass2::fermata(ChordRest* cr) { Q_ASSERT(_e.isStartElement() && _e.name() == "fermata"); QString fermataType = _e.attributes().value("type").toString(); QString fermata = _e.readElementText(); if (fermata == "normal" || fermata == "") addFermata(cr, fermataType, SymId::fermataAbove); else if (fermata == "angled") addFermata(cr, fermataType, SymId::fermataShortAbove); else if (fermata == "square") addFermata(cr, fermataType, SymId::fermataLongAbove); else _logger->logError(QString("unknown fermata '%1'").arg(fermata), &_e); } //--------------------------------------------------------- // tuplet //--------------------------------------------------------- /** Parse the /score-partwise/part/measure/note/notations/tuplet node. */ void MusicXMLParserPass2::tuplet(MusicXmlTupletDesc& tupletDesc) { Q_ASSERT(_e.isStartElement() && _e.name() == "tuplet"); QString tupletType = _e.attributes().value("type").toString(); // QString tupletPlacement = _e.attributes().value("placement").toString(); not used (TODO) QString tupletBracket = _e.attributes().value("bracket").toString(); QString tupletShowNumber = _e.attributes().value("show-number").toString(); // ignore possible children (currently not supported) _e.skipCurrentElement(); if (tupletType == "start") tupletDesc.type = MxmlStartStop::START; else if (tupletType == "stop") tupletDesc.type = MxmlStartStop::STOP; else if (tupletType != "" && tupletType != "start" && tupletType != "stop") { _logger->logError(QString("unknown tuplet type '%1'").arg(tupletType), &_e); } // set bracket, leave at default if unspecified if (tupletBracket == "yes") tupletDesc.bracket = TupletBracketType::SHOW_BRACKET; else if (tupletBracket == "no") tupletDesc.bracket = TupletBracketType::SHOW_NO_BRACKET; // set number, default is "actual" (=NumberType::SHOW_NUMBER) if (tupletShowNumber == "both") tupletDesc.shownumber = TupletNumberType::SHOW_RELATION; else if (tupletShowNumber == "none") tupletDesc.shownumber = TupletNumberType::NO_TEXT; else tupletDesc.shownumber = TupletNumberType::SHOW_NUMBER; } //--------------------------------------------------------- // MusicXMLParserDirection //--------------------------------------------------------- /** MusicXMLParserDirection constructor. */ MusicXMLParserDirection::MusicXMLParserDirection(QXmlStreamReader& e, Score* score, const MusicXMLParserPass1& pass1, MusicXMLParserPass2& pass2, MxmlLogger* logger) : _e(e), _score(score), _pass1(pass1), _pass2(pass2), _logger(logger), _hasDefaultY(false), _defaultY(0.0), _coda(false), _segno(false), _tpoMetro(0), _tpoSound(0) { // nothing } }