summaryrefslogtreecommitdiffstats
path: root/src/document/RoseXmlHandler.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/document/RoseXmlHandler.cpp')
-rw-r--r--src/document/RoseXmlHandler.cpp2368
1 files changed, 2368 insertions, 0 deletions
diff --git a/src/document/RoseXmlHandler.cpp b/src/document/RoseXmlHandler.cpp
new file mode 100644
index 0000000..028c89a
--- /dev/null
+++ b/src/document/RoseXmlHandler.cpp
@@ -0,0 +1,2368 @@
+/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
+
+/*
+ Rosegarden
+ A MIDI and audio sequencer and musical notation editor.
+
+ This program is Copyright 2000-2008
+ Guillaume Laurent <glaurent@telegraph-road.org>,
+ Chris Cannam <cannam@all-day-breakfast.com>,
+ Richard Bown <richard.bown@ferventsoftware.com>
+
+ The moral rights of Guillaume Laurent, Chris Cannam, and Richard
+ Bown to claim authorship of this work have been asserted.
+
+ Other copyrights also apply to some parts of this work. Please
+ see the AUTHORS file and individual file headers for details.
+
+ This program is free software; you can redistribute it and/or
+ modify it under the terms of the GNU General Public License as
+ published by the Free Software Foundation; either version 2 of the
+ License, or (at your option) any later version. See the file
+ COPYING included with this distribution for more information.
+*/
+
+
+#include "RoseXmlHandler.h"
+
+#include "sound/Midi.h"
+#include <klocale.h>
+#include "misc/Debug.h"
+#include "misc/Strings.h"
+#include "base/AudioLevel.h"
+#include "base/AudioPluginInstance.h"
+#include "base/BaseProperties.h"
+#include "base/Colour.h"
+#include "base/ColourMap.h"
+#include "base/Composition.h"
+#include "base/ControlParameter.h"
+#include "base/Device.h"
+#include "base/Instrument.h"
+#include "base/Marker.h"
+#include "base/MidiDevice.h"
+#include "base/MidiProgram.h"
+#include "base/MidiTypes.h"
+#include "base/NotationTypes.h"
+#include "base/RealTime.h"
+#include "base/Segment.h"
+#include "base/Studio.h"
+#include "base/Track.h"
+#include "base/TriggerSegment.h"
+#include "gui/application/RosegardenGUIApp.h"
+#include "gui/application/RosegardenApplication.h"
+#include "gui/dialogs/FileLocateDialog.h"
+#include "gui/general/ProgressReporter.h"
+#include "gui/kdeext/KStartupLogo.h"
+#include "gui/studio/AudioPlugin.h"
+#include "gui/studio/AudioPluginManager.h"
+#include "gui/widgets/CurrentProgressDialog.h"
+#include "gui/widgets/ProgressDialog.h"
+#include "RosegardenGUIDoc.h"
+#include "sound/AudioFileManager.h"
+#include <kfiledialog.h>
+#include <kmessagebox.h>
+#include <qcstring.h>
+#include <qdatastream.h>
+#include <qdialog.h>
+#include <qfileinfo.h>
+#include <qstring.h>
+#include <qstringlist.h>
+#include "XmlStorableEvent.h"
+#include "XmlSubHandler.h"
+
+namespace Rosegarden
+{
+
+using namespace BaseProperties;
+
+class ConfigurationXmlSubHandler : public XmlSubHandler
+{
+public:
+ ConfigurationXmlSubHandler(const QString &elementName,
+ Rosegarden::Configuration *configuration);
+
+ virtual bool startElement(const QString& namespaceURI,
+ const QString& localName,
+ const QString& qName,
+ const QXmlAttributes& atts);
+
+ virtual bool endElement(const QString& namespaceURI,
+ const QString& localName,
+ const QString& qName,
+ bool& finished);
+
+ virtual bool characters(const QString& ch);
+
+ //--------------- Data members ---------------------------------
+
+ Rosegarden::Configuration *m_configuration;
+
+ QString m_elementName;
+ QString m_propertyName;
+ QString m_propertyType;
+};
+
+ConfigurationXmlSubHandler::ConfigurationXmlSubHandler(const QString &elementName,
+ Rosegarden::Configuration *configuration)
+ : m_configuration(configuration),
+ m_elementName(elementName)
+{
+}
+
+bool ConfigurationXmlSubHandler::startElement(const QString&, const QString&,
+ const QString& lcName,
+ const QXmlAttributes& atts)
+{
+ m_propertyName = lcName;
+ m_propertyType = atts.value("type");
+
+ if (m_propertyName == "property") {
+ // handle alternative encoding for properties with arbitrary names
+ m_propertyName = atts.value("name");
+ QString value = atts.value("value");
+ if (value) {
+ m_propertyType = "String";
+ m_configuration->set<String>(qstrtostr(m_propertyName),
+ qstrtostr(value));
+ }
+ }
+
+ return true;
+}
+
+bool ConfigurationXmlSubHandler::characters(const QString& chars)
+{
+ QString ch = chars.stripWhiteSpace();
+ // this method is also called on newlines - skip these cases
+ if (ch.isEmpty()) return true;
+
+
+ if (m_propertyType == "Int") {
+ long i = ch.toInt();
+ RG_DEBUG << "\"" << m_propertyName << "\" "
+ << "value = " << i << endl;
+ m_configuration->set<Int>(qstrtostr(m_propertyName), i);
+
+ return true;
+ }
+
+ if (m_propertyType == "RealTime") {
+ Rosegarden::RealTime rt;
+ int sepIdx = ch.find(',');
+
+ rt.sec = ch.left(sepIdx).toInt();
+ rt.nsec = ch.mid(sepIdx + 1).toInt();
+
+ RG_DEBUG << "\"" << m_propertyName << "\" "
+ << "sec = " << rt.sec << ", nsec = " << rt.nsec << endl;
+
+ m_configuration->set<Rosegarden::RealTimeT>(qstrtostr(m_propertyName), rt);
+
+ return true;
+ }
+
+ if (m_propertyType == "Bool") {
+ QString chLc = ch.lower();
+
+ bool b = (chLc == "true" ||
+ chLc == "1" ||
+ chLc == "on");
+
+ m_configuration->set<Rosegarden::Bool>(qstrtostr(m_propertyName), b);
+
+ return true;
+ }
+
+ if (!m_propertyType ||
+ m_propertyType == "String") {
+
+ m_configuration->set<Rosegarden::String>(qstrtostr(m_propertyName),
+ qstrtostr(ch));
+
+ return true;
+ }
+
+
+ return true;
+}
+
+bool
+ConfigurationXmlSubHandler::endElement(const QString&,
+ const QString&,
+ const QString& lcName,
+ bool& finished)
+{
+ m_propertyName = "";
+ m_propertyType = "";
+ finished = (lcName == m_elementName);
+ return true;
+}
+
+
+//----------------------------------------
+
+
+
+RoseXmlHandler::RoseXmlHandler(RosegardenGUIDoc *doc,
+ unsigned int elementCount,
+ bool createNewDevicesWhenNeeded)
+ : ProgressReporter(0),
+ m_doc(doc),
+ m_currentSegment(0),
+ m_currentEvent(0),
+ m_currentTime(0),
+ m_chordDuration(0),
+ m_segmentEndMarkerTime(0),
+ m_inChord(false),
+ m_inGroup(false),
+ m_inComposition(false),
+ m_groupId(0),
+ m_foundTempo(false),
+ m_section(NoSection),
+ m_device(0),
+ m_deviceRunningId(Device::NO_DEVICE),
+ m_msb(0),
+ m_lsb(0),
+ m_instrument(0),
+ m_plugin(0),
+ m_pluginInBuss(false),
+ m_colourMap(0),
+ m_keyMapping(0),
+ m_pluginId(0),
+ m_totalElements(elementCount),
+ m_elementsSoFar(0),
+ m_subHandler(0),
+ m_deprecation(false),
+ m_createDevices(createNewDevicesWhenNeeded),
+ m_haveControls(false),
+ m_cancelled(false),
+ m_skipAllAudio(false),
+ m_hasActiveAudio(false)
+{}
+
+RoseXmlHandler::~RoseXmlHandler()
+{
+ delete m_subHandler;
+}
+
+Composition &
+RoseXmlHandler::getComposition()
+{
+ return m_doc->getComposition();
+}
+
+Studio &
+RoseXmlHandler::getStudio()
+{
+ return m_doc->getStudio();
+}
+
+AudioFileManager &
+RoseXmlHandler::getAudioFileManager()
+{
+ return m_doc->getAudioFileManager();
+}
+
+AudioPluginManager *
+RoseXmlHandler::getAudioPluginManager()
+{
+ return m_doc->getPluginManager();
+}
+
+bool
+RoseXmlHandler::startDocument()
+{
+ // Clear tracks
+ //
+ getComposition().clearTracks();
+
+ // And the loop
+ //
+ getComposition().setLoopStart(0);
+ getComposition().setLoopEnd(0);
+
+ // All plugins
+ //
+ m_doc->clearAllPlugins();
+
+ // reset state
+ return true;
+}
+
+bool
+RoseXmlHandler::startElement(const QString& namespaceURI,
+ const QString& localName,
+ const QString& qName, const QXmlAttributes& atts)
+{
+ // First check if user pressed cancel button on the progress
+ // dialog
+ //
+ if (isOperationCancelled()) {
+ // Ideally, we'd throw here, but at this point Qt is in the stack
+ // and Qt is very often compiled without exception support.
+ //
+ m_cancelled = true;
+ return false;
+ }
+
+ QString lcName = qName.lower();
+
+ if (getSubHandler()) {
+ return getSubHandler()->startElement(namespaceURI, localName, lcName, atts);
+ }
+
+ if (lcName == "event") {
+
+ // RG_DEBUG << "RoseXmlHandler::startElement: found event, current time is " << m_currentTime << endl;
+
+ if (m_currentEvent) {
+ RG_DEBUG << "RoseXmlHandler::startElement: Warning: new event found at time " << m_currentTime << " before previous event has ended; previous event will be lost" << endl;
+ delete m_currentEvent;
+ }
+
+ m_currentEvent = new XmlStorableEvent(atts, m_currentTime);
+
+ if (m_currentEvent->has(BEAMED_GROUP_ID)) {
+
+ // remap -- we want to ensure that the segment's nextId
+ // is always used (and incremented) in preference to the
+ // stored id
+
+ if (!m_currentSegment) {
+ m_errorString = "Got grouped event outside of a segment";
+ return false;
+ }
+
+ long storedId = m_currentEvent->get
+ <Int>(BEAMED_GROUP_ID);
+
+ if (m_groupIdMap.find(storedId) == m_groupIdMap.end()) {
+ m_groupIdMap[storedId] = m_currentSegment->getNextId();
+ }
+
+ m_currentEvent->set
+ <Int>(BEAMED_GROUP_ID, m_groupIdMap[storedId]);
+
+ } else if (m_inGroup) {
+ m_currentEvent->set
+ <Int>(BEAMED_GROUP_ID, m_groupId);
+ m_currentEvent->set
+ <String>(BEAMED_GROUP_TYPE, m_groupType);
+ if (m_groupType == GROUP_TYPE_TUPLED) {
+ m_currentEvent->set
+ <Int>
+ (BEAMED_GROUP_TUPLET_BASE, m_groupTupletBase);
+ m_currentEvent->set
+ <Int>
+ (BEAMED_GROUP_TUPLED_COUNT, m_groupTupledCount);
+ m_currentEvent->set
+ <Int>
+ (BEAMED_GROUP_UNTUPLED_COUNT, m_groupUntupledCount);
+ }
+ }
+
+ timeT duration = m_currentEvent->getDuration();
+
+ if (!m_inChord) {
+
+ m_currentTime = m_currentEvent->getAbsoluteTime() + duration;
+
+ // RG_DEBUG << "RoseXmlHandler::startElement: (we're not in a chord) " << endl;
+
+ } else if (duration != 0) {
+
+ // set chord duration to the duration of the shortest
+ // element with a non-null duration (if no such elements,
+ // leave it as 0).
+
+ if (m_chordDuration == 0 || duration < m_chordDuration) {
+ m_chordDuration = duration;
+ }
+ }
+
+ } else if (lcName == "property") {
+
+ if (!m_currentEvent) {
+ RG_DEBUG << "RoseXmlHandler::startElement: Warning: Found property outside of event at time " << m_currentTime << ", ignoring" << endl;
+ } else {
+ m_currentEvent->setPropertyFromAttributes(atts, true);
+ }
+
+ } else if (lcName == "nproperty") {
+
+ if (!m_currentEvent) {
+ RG_DEBUG << "RoseXmlHandler::startElement: Warning: Found nproperty outside of event at time " << m_currentTime << ", ignoring" << endl;
+ } else {
+ m_currentEvent->setPropertyFromAttributes(atts, false);
+ }
+
+ } else if (lcName == "chord") {
+
+ m_inChord = true;
+
+ } else if (lcName == "group") {
+
+ if (!m_currentSegment) {
+ m_errorString = "Got group outside of a segment";
+ return false;
+ }
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"group\". We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ m_inGroup = true;
+ m_groupId = m_currentSegment->getNextId();
+ m_groupType = qstrtostr(atts.value("type"));
+
+ if (m_groupType == GROUP_TYPE_TUPLED) {
+ m_groupTupletBase = atts.value("base").toInt();
+ m_groupTupledCount = atts.value("tupled").toInt();
+ m_groupUntupledCount = atts.value("untupled").toInt();
+ }
+
+ } else if (lcName == "rosegarden-data") {
+
+ // FILE FORMAT VERSIONING -- see comments in
+ // rosegardenguidoc.cpp. We only care about major and minor
+ // here, not point.
+
+ QString version = atts.value("version");
+ QString smajor = atts.value("format-version-major");
+ QString sminor = atts.value("format-version-minor");
+
+// std::cerr << "\n\n\nRosegarden file version = \"" << version << "\"\n\n\n" << std::endl;
+
+ if (smajor) {
+
+ int major = smajor.toInt();
+ int minor = sminor.toInt();
+
+ if (major > RosegardenGUIDoc::FILE_FORMAT_VERSION_MAJOR) {
+ m_errorString = i18n("This file was written by Rosegarden %1, and it uses\na different file format that cannot be read by this version.").arg(version);
+ return false;
+ }
+
+ if (major == RosegardenGUIDoc::FILE_FORMAT_VERSION_MAJOR &&
+ minor > RosegardenGUIDoc::FILE_FORMAT_VERSION_MINOR) {
+
+ CurrentProgressDialog::freeze();
+ KStartupLogo::hideIfStillThere();
+
+ KMessageBox::information(0, i18n("This file was written by Rosegarden %1, which is more recent than this version.\nThere may be some incompatibilities with the file format.").arg(version));
+
+ CurrentProgressDialog::thaw();
+ }
+ }
+
+ } else if (lcName == "studio") {
+
+ if (m_section != NoSection) {
+ m_errorString = "Found Studio in another section";
+ return false;
+ }
+
+ // In the Studio we clear down everything apart from Devices and
+ // Instruments before we reload. Instruments are derived from
+ // the Sequencer, the bank/program information is loaded from
+ // the file we're currently examining.
+ //
+ getStudio().clearMidiBanksAndPrograms();
+ getStudio().clearBusses();
+ getStudio().clearRecordIns();
+
+ m_section = InStudio; // set top level section
+
+ // Get and set MIDI filters
+ //
+ QString thruStr = atts.value("thrufilter");
+
+ if (thruStr)
+ getStudio().setMIDIThruFilter(thruStr.toInt());
+
+ QString recordStr = atts.value("recordfilter");
+
+ if (recordStr)
+ getStudio().setMIDIRecordFilter(recordStr.toInt());
+
+ QString inputStr = atts.value("audioinputpairs");
+
+ if (inputStr) {
+ int inputs = inputStr.toInt();
+ if (inputs < 1)
+ inputs = 1; // we simply don't permit no inputs
+ while (int(getStudio().getRecordIns().size()) < inputs) {
+ getStudio().addRecordIn(new RecordIn());
+ }
+ }
+
+ QString mixerStr = atts.value("mixerdisplayoptions");
+
+ if (mixerStr) {
+ unsigned int mixer = mixerStr.toUInt();
+ getStudio().setMixerDisplayOptions(mixer);
+ }
+
+ QString metronomeStr = atts.value("metronomedevice");
+
+ if (metronomeStr) {
+ DeviceId metronome = metronomeStr.toUInt();
+ getStudio().setMetronomeDevice(metronome);
+ }
+
+ } else if (lcName == "timesignature") {
+
+ if (m_inComposition == false) {
+ m_errorString = "TimeSignature object found outside Composition";
+ return false;
+ }
+
+ timeT t = 0;
+ QString timeStr = atts.value("time");
+ if (timeStr)
+ t = timeStr.toInt();
+
+ int num = 4;
+ QString numStr = atts.value("numerator");
+ if (numStr)
+ num = numStr.toInt();
+
+ int denom = 4;
+ QString denomStr = atts.value("denominator");
+ if (denomStr)
+ denom = denomStr.toInt();
+
+ bool common = false;
+ QString commonStr = atts.value("common");
+ if (commonStr)
+ common = (commonStr == "true");
+
+ bool hidden = false;
+ QString hiddenStr = atts.value("hidden");
+ if (hiddenStr)
+ hidden = (hiddenStr == "true");
+
+ bool hiddenBars = false;
+ QString hiddenBarsStr = atts.value("hiddenbars");
+ if (hiddenBarsStr)
+ hiddenBars = (hiddenBarsStr == "true");
+
+ getComposition().addTimeSignature
+ (t, TimeSignature(num, denom, common, hidden, hiddenBars));
+
+ } else if (lcName == "tempo") {
+
+ timeT t = 0;
+ QString timeStr = atts.value("time");
+ if (timeStr)
+ t = timeStr.toInt();
+
+ tempoT tempo = Composition::getTempoForQpm(120.0);
+ QString tempoStr = atts.value("tempo");
+ QString targetStr = atts.value("target");
+ QString bphStr = atts.value("bph");
+ if (tempoStr) {
+ tempo = tempoStr.toInt();
+ } else if (bphStr) {
+ tempo = Composition::getTempoForQpm
+ (double(bphStr.toInt()) / 60.0);
+ }
+
+ if (targetStr) {
+ getComposition().addTempoAtTime(t, tempo, targetStr.toInt());
+ } else {
+ getComposition().addTempoAtTime(t, tempo);
+ }
+
+ } else if (lcName == "composition") {
+
+ if (m_section != NoSection) {
+ m_errorString = "Found Composition in another section";
+ return false;
+ }
+
+ // set Segment
+ m_section = InComposition;
+
+ // Get and set the record track
+ //
+ QString recordStr = atts.value("recordtrack");
+ if (recordStr) {
+ getComposition().setTrackRecording(recordStr.toInt(), true);
+ }
+
+ QString recordPlStr = atts.value("recordtracks");
+ if (recordPlStr) {
+ RG_DEBUG << "Record tracks: " << recordPlStr << endl;
+ QStringList recordList = QStringList::split(',', recordPlStr);
+ for (QStringList::iterator i = recordList.begin();
+ i != recordList.end(); ++i) {
+ RG_DEBUG << "Record track: " << (*i).toInt() << endl;
+ getComposition().setTrackRecording((*i).toInt(), true);
+ }
+ }
+
+ // Get and set the position pointer
+ //
+ int position = 0;
+ QString positionStr = atts.value("pointer");
+ if (positionStr) {
+ position = positionStr.toInt();
+ }
+
+ getComposition().setPosition(position);
+
+
+ // Get and (eventually) set the default tempo.
+ // We prefer the new compositionDefaultTempo over the
+ // older defaultTempo.
+ //
+ QString tempoStr = atts.value("compositionDefaultTempo");
+ if (tempoStr) {
+ tempoT tempo = tempoT(tempoStr.toInt());
+ getComposition().setCompositionDefaultTempo(tempo);
+ } else {
+ tempoStr = atts.value("defaultTempo");
+ if (tempoStr) {
+ double tempo = qstrtodouble(tempoStr);
+ getComposition().setCompositionDefaultTempo
+ (Composition::getTempoForQpm(tempo));
+ }
+ }
+
+ // set the composition flag
+ m_inComposition = true;
+
+
+ // Set the loop
+ //
+ QString loopStartStr = atts.value("loopstart");
+ QString loopEndStr = atts.value("loopend");
+
+ if (loopStartStr && loopEndStr) {
+ int loopStart = loopStartStr.toInt();
+ int loopEnd = loopEndStr.toInt();
+
+ getComposition().setLoopStart(loopStart);
+ getComposition().setLoopEnd(loopEnd);
+ }
+
+ QString selectedTrackStr = atts.value("selected");
+
+ if (selectedTrackStr) {
+ TrackId selectedTrack =
+ (TrackId)selectedTrackStr.toInt();
+
+ getComposition().setSelectedTrack(selectedTrack);
+ }
+
+ QString soloTrackStr = atts.value("solo");
+ if (soloTrackStr) {
+ if (soloTrackStr.toInt() == 1)
+ getComposition().setSolo(true);
+ else
+ getComposition().setSolo(false);
+ }
+
+
+ QString playMetStr = atts.value("playmetronome");
+ if (playMetStr) {
+ if (playMetStr.toInt())
+ getComposition().setPlayMetronome(true);
+ else
+ getComposition().setPlayMetronome(false);
+ }
+
+ QString recMetStr = atts.value("recordmetronome");
+ if (recMetStr) {
+ if (recMetStr.toInt())
+ getComposition().setRecordMetronome(true);
+ else
+ getComposition().setRecordMetronome(false);
+ }
+
+ QString nextTriggerIdStr = atts.value("nexttriggerid");
+ if (nextTriggerIdStr) {
+ getComposition().setNextTriggerSegmentId(nextTriggerIdStr.toInt());
+ }
+
+ QString copyrightStr = atts.value("copyright");
+ if (copyrightStr) {
+ getComposition().setCopyrightNote(qstrtostr(copyrightStr));
+ }
+
+ QString startMarkerStr = atts.value("startMarker");
+ QString endMarkerStr = atts.value("endMarker");
+
+ if (startMarkerStr) {
+ getComposition().setStartMarker(startMarkerStr.toInt());
+ }
+
+ if (endMarkerStr) {
+ getComposition().setEndMarker(endMarkerStr.toInt());
+ }
+
+ } else if (lcName == "track") {
+
+ if (m_section != InComposition) {
+ m_errorString = "Track object found outside Composition";
+ return false;
+ }
+
+ int id = -1;
+ int position = -1;
+ int instrument = -1;
+ std::string label;
+ bool muted = false;
+
+ QString trackNbStr = atts.value("id");
+ if (trackNbStr) {
+ id = trackNbStr.toInt();
+ }
+
+ QString labelStr = atts.value("label");
+ if (labelStr) {
+ label = qstrtostr(labelStr);
+ }
+
+ QString mutedStr = atts.value("muted");
+ if (mutedStr) {
+ if (mutedStr == "true")
+ muted = true;
+ else
+ muted = false;
+ }
+
+ QString positionStr = atts.value("position");
+ if (positionStr) {
+ position = positionStr.toInt();
+ }
+
+ QString instrumentStr = atts.value("instrument");
+ if (instrumentStr) {
+ instrument = instrumentStr.toInt();
+ }
+
+ Track *track = new Track(id,
+ instrument,
+ position,
+ label,
+ muted);
+
+ // track properties affecting newly created segments are initialized
+ // to default values in the ctor, so they don't need to be initialized
+ // here
+
+ QString presetLabelStr = atts.value("defaultLabel");
+ if (labelStr) {
+ track->setPresetLabel(presetLabelStr);
+ }
+
+ QString clefStr = atts.value("defaultClef");
+ if (clefStr) {
+ track->setClef(clefStr.toInt());
+ }
+
+ QString transposeStr = atts.value("defaultTranspose");
+ if (transposeStr) {
+ track->setTranspose(transposeStr.toInt());
+ }
+
+ QString colorStr = atts.value("defaultColour");
+ if (colorStr) {
+ track->setColor(colorStr.toInt());
+ }
+
+ QString highplayStr = atts.value("defaultHighestPlayable");
+ if (highplayStr) {
+ track->setHighestPlayable(highplayStr.toInt());
+ }
+
+ QString lowplayStr = atts.value("defaultLowestPlayable");
+ if (lowplayStr) {
+ track->setLowestPlayable(lowplayStr.toInt());
+ }
+
+ QString staffSizeStr = atts.value("staffSize");
+ if (staffSizeStr) {
+ track->setStaffSize(staffSizeStr.toInt());
+ }
+
+ QString staffBracketStr = atts.value("staffBracket");
+ if (staffBracketStr) {
+ track->setStaffBracket(staffBracketStr.toInt());
+ }
+
+ getComposition().addTrack(track);
+
+
+ } else if (lcName == "segment") {
+
+ if (m_section != NoSection) {
+ m_errorString = "Found Segment in another section";
+ return false;
+ }
+
+ // set Segment
+ m_section = InSegment;
+
+ int track = -1, startTime = 0;
+ unsigned int colourindex = 0;
+ QString trackNbStr = atts.value("track");
+ if (trackNbStr) {
+ track = trackNbStr.toInt();
+ }
+
+ QString startIdxStr = atts.value("start");
+ if (startIdxStr) {
+ startTime = startIdxStr.toInt();
+ }
+
+ QString segmentType = (atts.value("type")).lower();
+ if (segmentType) {
+ if (segmentType == "audio") {
+ int audioFileId = atts.value("file").toInt();
+
+ // check this file id exists on the AudioFileManager
+
+ if (getAudioFileManager().fileExists(audioFileId) == false) {
+ // We don't report an error as this audio file might've
+ // been excluded deliberately as we could't actually
+ // find the audio file itself.
+ //
+ return true;
+ }
+
+ // Create an Audio segment and add its reference
+ //
+ m_currentSegment = new Segment(Segment::Audio);
+ m_currentSegment->setAudioFileId(audioFileId);
+ m_currentSegment->setStartTime(startTime);
+ } else {
+ // Create a (normal) internal Segment
+ m_currentSegment = new Segment(Segment::Internal);
+ }
+
+ } else {
+ // for the moment we default
+ m_currentSegment = new Segment(Segment::Internal);
+ }
+
+ QString repeatStr = atts.value("repeat");
+ if (repeatStr.lower() == "true") {
+ m_currentSegment->setRepeating(true);
+ }
+
+ QString delayStr = atts.value("delay");
+ if (delayStr) {
+ RG_DEBUG << "Delay string is \"" << delayStr << "\"" << endl;
+ long delay = delayStr.toLong();
+ RG_DEBUG << "Delay is " << delay << endl;
+ m_currentSegment->setDelay(delay);
+ }
+
+ QString rtDelaynSec = atts.value("rtdelaynsec");
+ QString rtDelayuSec = atts.value("rtdelayusec");
+ QString rtDelaySec = atts.value("rtdelaysec");
+ if (rtDelaySec && (rtDelaynSec || rtDelayuSec)) {
+ if (rtDelaynSec) {
+ m_currentSegment->setRealTimeDelay
+ (RealTime(rtDelaySec.toInt(),
+ rtDelaynSec.toInt()));
+ } else {
+ m_currentSegment->setRealTimeDelay
+ (RealTime(rtDelaySec.toInt(),
+ rtDelayuSec.toInt() * 1000));
+ }
+ }
+
+ QString transposeStr = atts.value("transpose");
+ if (transposeStr)
+ m_currentSegment->setTranspose(transposeStr.toInt());
+
+ // fill in the label
+ QString labelStr = atts.value("label");
+ if (labelStr)
+ m_currentSegment->setLabel(qstrtostr(labelStr));
+
+ m_currentSegment->setTrack(track);
+ //m_currentSegment->setStartTime(startTime);
+
+ QString colourIndStr = atts.value("colourindex");
+ if (colourIndStr) {
+ colourindex = colourIndStr.toInt();
+ }
+
+ m_currentSegment->setColourIndex(colourindex);
+
+ QString snapGridSizeStr = atts.value("snapgridsize");
+ if (snapGridSizeStr) {
+ m_currentSegment->setSnapGridSize(snapGridSizeStr.toInt());
+ }
+
+ QString viewFeaturesStr = atts.value("viewfeatures");
+ if (viewFeaturesStr) {
+ m_currentSegment->setViewFeatures(viewFeaturesStr.toInt());
+ }
+
+ m_currentTime = startTime;
+
+ QString triggerIdStr = atts.value("triggerid");
+ QString triggerPitchStr = atts.value("triggerbasepitch");
+ QString triggerVelocityStr = atts.value("triggerbasevelocity");
+ QString triggerRetuneStr = atts.value("triggerretune");
+ QString triggerAdjustTimeStr = atts.value("triggeradjusttimes");
+
+ if (triggerIdStr) {
+ int pitch = -1;
+ if (triggerPitchStr)
+ pitch = triggerPitchStr.toInt();
+ int velocity = -1;
+ if (triggerVelocityStr)
+ velocity = triggerVelocityStr.toInt();
+ TriggerSegmentRec *rec =
+ getComposition().addTriggerSegment(m_currentSegment,
+ triggerIdStr.toInt(),
+ pitch, velocity);
+ if (rec) {
+ if (triggerRetuneStr)
+ rec->setDefaultRetune(triggerRetuneStr.lower() == "true");
+ if (triggerAdjustTimeStr)
+ rec->setDefaultTimeAdjust(qstrtostr(triggerAdjustTimeStr));
+ }
+ m_currentSegment->setStartTimeDataMember(startTime);
+ } else {
+ getComposition().addSegment(m_currentSegment);
+ getComposition().setSegmentStartTime(m_currentSegment, startTime);
+ }
+
+ QString endMarkerStr = atts.value("endmarker");
+ if (endMarkerStr) {
+ delete m_segmentEndMarkerTime;
+ m_segmentEndMarkerTime = new timeT(endMarkerStr.toInt());
+ }
+
+ m_groupIdMap.clear();
+
+ } else if (lcName == "gui") {
+
+ if (m_section != InSegment) {
+ m_errorString = "Found GUI element outside Segment";
+ return false;
+ }
+
+ } else if (lcName == "controller") {
+
+ if (m_section != InSegment) {
+ m_errorString = "Found Controller element outside Segment";
+ return false;
+ }
+
+ QString type = atts.value("type");
+ //RG_DEBUG << "RoseXmlHandler::startElement - controller type = " << type << endl;
+
+ if (type == strtoqstr(PitchBend::EventType))
+ m_currentSegment->addEventRuler(PitchBend::EventType);
+ else if (type == strtoqstr(Controller::EventType)) {
+ QString value = atts.value("value");
+
+ if (value != "")
+ m_currentSegment->addEventRuler(Controller::EventType, value.toInt());
+ }
+
+ } else if (lcName == "resync") {
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"resync\". We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ QString time(atts.value("time"));
+ bool isNumeric;
+ int numTime = time.toInt(&isNumeric);
+ if (isNumeric)
+ m_currentTime = numTime;
+
+ } else if (lcName == "audio") {
+
+ if (m_section != InAudioFiles) {
+ m_errorString = "Audio object found outside Audio section";
+ return false;
+ }
+
+ if (m_skipAllAudio) {
+ std::cout << "SKIPPING audio file" << std::endl;
+ return true;
+ }
+
+ QString id(atts.value("id"));
+ QString file(atts.value("file"));
+ QString label(atts.value("label"));
+
+ if (id.isEmpty() || file.isEmpty() || label.isEmpty()) {
+ m_errorString = "Audio object has empty parameters";
+ return false;
+ }
+
+ m_hasActiveAudio = true;
+
+ // attempt to insert file into AudioFileManager
+ // (this checks the integrity of the file at the
+ // same time)
+ //
+ if (getAudioFileManager().insertFile(qstrtostr(label),
+ qstrtostr(file),
+ id.toInt()) == false) {
+ // Ok, now attempt to use the KFileDialog saved default
+ // value for the AudioPath.
+ //
+ QString thing;
+ KURL url = KFileDialog::getStartURL(QString(":WAVS"), thing);
+ getAudioFileManager().setAudioPath(url.path().latin1());
+
+ /*
+ RG_DEBUG << "ATTEMPTING TO FIND IN PATH = "
+ << url.path() << endl;
+ */
+
+ if (getAudioFileManager().
+ insertFile(qstrtostr(label),
+ qstrtostr(file), id.toInt()) == false) {
+
+ // Freeze the progress dialog
+ CurrentProgressDialog::freeze();
+
+ // Hide splash screen if present on startup
+ KStartupLogo::hideIfStillThere();
+
+ // Create a locate file dialog - give it the file name
+ // and the AudioFileManager path that we've already
+ // tried. If we manually locate the file then we reset
+ // the audiofilepath to the new value and see if this
+ // helps us locate the rest of the files.
+ //
+
+ QString newFilename = "";
+ QString newPath = "";
+
+ do {
+
+ FileLocateDialog fL((RosegardenGUIApp *)m_doc->parent(),
+ file,
+ QString(getAudioFileManager().getAudioPath().c_str()));
+ int result = fL.exec();
+
+ if (result == QDialog::Accepted) {
+ newFilename = fL.getFilename();
+ newPath = fL.getDirectory();
+ } else if (result == QDialog::Rejected) {
+ // just skip the file
+ break;
+ } else {
+ // don't process any more audio files
+ m_skipAllAudio = true;
+ CurrentProgressDialog::thaw();
+ return true;
+ }
+
+
+ } while (getAudioFileManager().insertFile(qstrtostr(label),
+ qstrtostr(newFilename),
+ id.toInt()) == false);
+
+ if (newPath != "") {
+ getAudioFileManager().setAudioPath(qstrtostr(newPath));
+ // Set a document post-modify flag
+ //m_doc->setModified(true);
+ }
+
+ getAudioFileManager().print();
+
+ // Restore progress dialog's normal state
+ CurrentProgressDialog::thaw();
+ } else {
+ // AudioPath is modified so set a document post modify flag
+ //
+ //m_doc->setModified(true);
+ }
+
+ }
+
+ } else if (lcName == "audiopath") {
+
+ if (m_section != InAudioFiles) {
+ m_errorString = "Audiopath object found outside AudioFiles section";
+ return false;
+ }
+
+ QString search(atts.value("value"));
+
+ if (search.isEmpty()) {
+ m_errorString = "Audiopath has no value";
+ return false;
+ }
+
+ if (!search.startsWith("/") && !search.startsWith("~")) {
+ QString docPath = m_doc->getAbsFilePath();
+ QString dirPath = QFileInfo(docPath).dirPath();
+ if (QFileInfo(dirPath).exists()) {
+ search = dirPath + "/" + search;
+ }
+ }
+
+ getAudioFileManager().setAudioPath(qstrtostr(search));
+
+ } else if (lcName == "begin") {
+
+ double marker = qstrtodouble(atts.value("index"));
+
+ if (!m_currentSegment) {
+ // Don't fail - as this segment could be defunct if we
+ // skipped loading the audio file
+ //
+ return true;
+ }
+
+ if (m_currentSegment->getType() != Segment::Audio) {
+ m_errorString = "Found audio begin index in non audio segment";
+ return false;
+ }
+
+ // convert to RealTime from float
+ int sec = (int)marker;
+ int usec = (int)((marker - ((double)sec)) * 1000000.0);
+ m_currentSegment->setAudioStartTime(RealTime(sec, usec * 1000));
+
+
+ } else if (lcName == "end") {
+
+ double marker = qstrtodouble(atts.value("index"));
+
+ if (!m_currentSegment) {
+ // Don't fail - as this segment could be defunct if we
+ // skipped loading the audio file
+ //
+ return true;
+ }
+
+ if (m_currentSegment->getType() != Segment::Audio) {
+ m_errorString = "found audio end index in non audio segment";
+ return false;
+ }
+
+ int sec = (int)marker;
+ int usec = (int)((marker - ((double)sec)) * 1000000.0);
+ RealTime markerTime(sec, usec * 1000);
+
+ if (markerTime < m_currentSegment->getAudioStartTime()) {
+ m_errorString = "Audio end index before audio start marker";
+ return false;
+ }
+
+ m_currentSegment->setAudioEndTime(markerTime);
+
+ // Ensure we set end time according to correct RealTime end of Segment
+ //
+ RealTime realEndTime = getComposition().
+ getElapsedRealTime(m_currentSegment->getStartTime()) +
+ m_currentSegment->getAudioEndTime() -
+ m_currentSegment->getAudioStartTime();
+
+ timeT absEnd = getComposition().getElapsedTimeForRealTime(realEndTime);
+ m_currentSegment->setEndTime(absEnd);
+
+ } else if (lcName == "fadein") {
+
+ if (!m_currentSegment) {
+ // Don't fail - as this segment could be defunct if we
+ // skipped loading the audio file
+ //
+ return true;
+ }
+
+ if (m_currentSegment->getType() != Segment::Audio) {
+ m_errorString = "found fade in time in non audio segment";
+ return false;
+ }
+
+ double marker = qstrtodouble(atts.value("time"));
+ int sec = (int)marker;
+ int usec = (int)((marker - ((double)sec)) * 1000000.0);
+ RealTime markerTime(sec, usec * 1000);
+
+ m_currentSegment->setFadeInTime(markerTime);
+ m_currentSegment->setAutoFade(true);
+
+
+ } else if (lcName == "fadeout") {
+
+ if (!m_currentSegment) {
+ // Don't fail - as this segment could be defunct if we
+ // skipped loading the audio file
+ //
+ return true;
+ }
+
+ if (m_currentSegment->getType() != Segment::Audio) {
+ m_errorString = "found fade out time in non audio segment";
+ return false;
+ }
+
+ double marker = qstrtodouble(atts.value("time"));
+ int sec = (int)marker;
+ int usec = (int)((marker - ((double)sec)) * 1000000.0);
+ RealTime markerTime(sec, usec * 1000);
+
+ m_currentSegment->setFadeOutTime(markerTime);
+ m_currentSegment->setAutoFade(true);
+
+ } else if (lcName == "device") {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found Device outside Studio";
+ return false;
+ }
+
+ m_haveControls = false;
+
+ QString type = (atts.value("type")).lower();
+ QString idString = atts.value("id");
+ QString nameStr = atts.value("name");
+
+ if (idString.isNull()) {
+ m_errorString = "No ID on Device tag";
+ return false;
+ }
+ int id = idString.toInt();
+
+ if (type == "midi") {
+ QString direction = atts.value("direction").lower();
+
+ if (direction.isNull() ||
+ direction == "" ||
+ direction == "play") { // ignore inputs
+
+ // This will leave m_device set only if there is a
+ // valid play midi device to modify:
+ skipToNextPlayDevice();
+
+ if (m_device) {
+ if (nameStr && nameStr != "") {
+ m_device->setName(qstrtostr(nameStr));
+ }
+ } else if (nameStr && nameStr != "") {
+ addMIDIDevice(nameStr, m_createDevices); // also sets m_device
+ }
+ }
+
+ QString connection = atts.value("connection");
+ if (m_createDevices && m_device &&
+ !connection.isNull() && connection != "") {
+ setMIDIDeviceConnection(connection);
+ }
+
+ setMIDIDeviceName(nameStr);
+
+ QString vstr = atts.value("variation").lower();
+ MidiDevice::VariationType variation =
+ MidiDevice::NoVariations;
+ if (!vstr.isNull()) {
+ if (vstr == "lsb") {
+ variation = MidiDevice::VariationFromLSB;
+ } else if (vstr == "msb") {
+ variation = MidiDevice::VariationFromMSB;
+ } else if (vstr == "") {
+ variation = MidiDevice::NoVariations;
+ }
+ }
+ MidiDevice *md = dynamic_cast<MidiDevice *>
+ (m_device);
+ if (md) {
+ md->setVariationType(variation);
+ }
+ } else if (type == "softsynth") {
+ m_device = getStudio().getDevice(id);
+
+ if (m_device && m_device->getType() == Device::SoftSynth)
+ m_device->setName(qstrtostr(nameStr));
+ } else if (type == "audio") {
+ m_device = getStudio().getDevice(id);
+
+ if (m_device && m_device->getType() == Device::Audio)
+ m_device->setName(qstrtostr(nameStr));
+ } else {
+ m_errorString = "Found unknown Device type";
+ return false;
+ }
+
+ } else if (lcName == "librarian") {
+
+ // The contact details for the maintainer of the banks/programs
+ // information.
+ //
+ if (m_device && m_device->getType() == Device::Midi) {
+ QString name = atts.value("name");
+ QString email = atts.value("email");
+
+ dynamic_cast<MidiDevice*>(m_device)->
+ setLibrarian(qstrtostr(name), qstrtostr(email));
+ }
+
+ } else if (lcName == "bank") {
+
+ if (m_device) // only if we have a device
+ {
+ if (m_section != InStudio && m_section != InInstrument)
+ {
+ m_errorString = "Found Bank outside Studio or Instrument";
+ return false;
+ }
+
+ QString nameStr = atts.value("name");
+ m_percussion = (atts.value("percussion").lower() == "true");
+ m_msb = (atts.value("msb")).toInt();
+ m_lsb = (atts.value("lsb")).toInt();
+
+ // To actually create a bank
+ //
+ if (m_section == InStudio)
+ {
+ // Create a new bank
+ MidiBank bank(m_percussion,
+ m_msb,
+ m_lsb,
+ qstrtostr(nameStr));
+
+ if (m_device->getType() == Device::Midi) {
+ // Insert the bank
+ //
+ dynamic_cast<MidiDevice*>(m_device)->addBank(bank);
+ }
+ } else // otherwise we're referencing it in an instrument
+ if (m_section == InInstrument)
+ {
+ if (m_instrument) {
+ m_instrument->setPercussion(m_percussion);
+ m_instrument->setMSB(m_msb);
+ m_instrument->setLSB(m_lsb);
+ m_instrument->setSendBankSelect(true);
+ }
+ }
+ }
+
+ } else if (lcName == "program") {
+
+ if (m_device) // only if we have a device
+ {
+ if (m_section == InStudio)
+ {
+ QString nameStr = (atts.value("name"));
+ MidiByte pc = atts.value("id").toInt();
+ QString keyMappingStr = (atts.value("keymapping"));
+
+ // Create a new program
+ MidiProgram program
+ (MidiBank(m_percussion,
+ m_msb,
+ m_lsb),
+ pc,
+ qstrtostr(nameStr),
+ keyMappingStr ? qstrtostr(keyMappingStr) : "");
+
+ if (m_device->getType() == Device::Midi) {
+ // Insert the program
+ //
+ dynamic_cast<MidiDevice*>(m_device)->
+ addProgram(program);
+ }
+
+ } else if (m_section == InInstrument)
+ {
+ if (m_instrument) {
+ MidiByte id = atts.value("id").toInt();
+ m_instrument->setProgramChange(id);
+ m_instrument->setSendProgramChange(true);
+ }
+ } else
+ {
+ m_errorString = "Found Program outside Studio and Instrument";
+ return false;
+ }
+ }
+
+ } else if (lcName == "keymapping") {
+
+ if (m_section == InInstrument) {
+ RG_DEBUG << "Old-style keymapping in instrument found, ignoring" << endl;
+ } else {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found Keymapping outside Studio";
+ return false;
+ }
+
+ if (m_device && (m_device->getType() == Device::Midi)) {
+ QString name = atts.value("name");
+ m_keyMapping = new MidiKeyMapping(qstrtostr(name));
+ m_keyNameMap.clear();
+ }
+ }
+
+ } else if (lcName == "key") {
+
+ if (m_keyMapping) {
+ QString numStr = atts.value("number");
+ QString namStr = atts.value("name");
+ if (numStr && namStr) {
+ m_keyNameMap[numStr.toInt()] = qstrtostr(namStr);
+ }
+ }
+
+ } else if (lcName == "controls") {
+
+ // Only clear down the controllers list if we have found some controllers in the RG file
+ //
+ if (m_device) {
+ dynamic_cast<MidiDevice*>(m_device)->clearControlList();
+ }
+
+ m_haveControls = true;
+
+ } else if (lcName == "control") {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found ControlParameter outside Studio";
+ return false;
+ }
+
+ if (!m_device) {
+ //!!! ach no, we can't give this warning -- we might be in a <device> elt
+ // but have no sequencer support, for example. we need a separate m_inDevice
+ // flag
+ // m_deprecation = true;
+ // std::cerr << "WARNING: This Rosegarden file uses a deprecated control parameter structure. We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+
+ } else if (m_device->getType() == Device::Midi) {
+
+ if (!m_haveControls) {
+ m_errorString = "Found ControlParameter outside Controls block";
+ return false;
+ }
+
+ QString name = atts.value("name");
+ QString type = atts.value("type");
+ QString descr = atts.value("description");
+ QString min = atts.value("min");
+ QString max = atts.value("max");
+ QString def = atts.value("default");
+ QString conVal = atts.value("controllervalue");
+ QString colour = atts.value("colourindex");
+ QString ipbPosition = atts.value("ipbposition");
+
+ ControlParameter con(qstrtostr(name),
+ qstrtostr(type),
+ qstrtostr(descr),
+ min.toInt(),
+ max.toInt(),
+ def.toInt(),
+ MidiByte(conVal.toInt()),
+ colour.toInt(),
+ ipbPosition.toInt());
+
+ dynamic_cast<MidiDevice*>(m_device)->
+ addControlParameter(con);
+ }
+
+ } else if (lcName == "reverb") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"reverb\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Reverb outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_REVERB, value);
+
+
+ } else if (lcName == "chorus") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"chorus\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Chorus outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_CHORUS, value);
+
+ } else if (lcName == "filter") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"filter\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Filter outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_FILTER, value);
+
+
+ } else if (lcName == "resonance") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"resonance\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Resonance outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_RESONANCE, value);
+
+
+ } else if (lcName == "attack") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"attack\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Attack outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_ATTACK, value);
+
+ } else if (lcName == "release") { // deprecated but we still read 'em
+
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"release\" (now replaced by a control parameter). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Release outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument)
+ m_instrument->setControllerValue(MIDI_CONTROLLER_RELEASE, value);
+
+ } else if (lcName == "pan") {
+
+ if (m_section != InInstrument && m_section != InBuss) {
+ m_errorString = "Found Pan outside Instrument or Buss";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_section == InInstrument) {
+ if (m_instrument) {
+ m_instrument->setPan(value);
+ m_instrument->setSendPan(true);
+ }
+ } else if (m_section == InBuss) {
+ if (m_buss) {
+ m_buss->setPan(value);
+ }
+ }
+
+ // keep "velocity" so we're backwards compatible
+ } else if (lcName == "velocity" || lcName == "volume") {
+
+ if (lcName == "velocity") {
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"velocity\" for an overall MIDI instrument level (now replaced by \"volume\"). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+ }
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found Volume outside Instrument";
+ return false;
+ }
+
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument) {
+ if (m_instrument->getType() == Instrument::Audio ||
+ m_instrument->getType() == Instrument::SoftSynth) {
+ // Backward compatibility: "volume" was in a 0-127
+ // range and we now store "level" (float dB) instead.
+ // Note that we have no such compatibility for
+ // "recordLevel", whose range has changed silently.
+ if (!m_deprecation)
+ std::cerr << "WARNING: This Rosegarden file uses the deprecated element \"volume\" for an audio instrument (now replaced by \"level\"). We recommend re-saving the file from this version of Rosegarden to assure your ability to re-load it in future versions" << std::endl;
+ m_deprecation = true;
+ m_instrument->setLevel
+ (AudioLevel::multiplier_to_dB(float(value) / 100.0));
+ } else {
+ m_instrument->setVolume(value);
+ m_instrument->setSendVolume(true);
+ }
+ }
+
+ } else if (lcName == "level") {
+
+ if (m_section != InBuss &&
+ (m_section != InInstrument ||
+ (m_instrument &&
+ m_instrument->getType() != Instrument::Audio &&
+ m_instrument->getType() != Instrument::SoftSynth))) {
+ m_errorString = "Found Level outside (audio) Instrument or Buss";
+ return false;
+ }
+
+ double value = qstrtodouble(atts.value("value"));
+
+ if (m_section == InBuss) {
+ if (m_buss)
+ m_buss->setLevel(value);
+ } else {
+ if (m_instrument)
+ m_instrument->setLevel(value);
+ }
+
+ } else if (lcName == "controlchange") {
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found ControlChange outside Instrument";
+ return false;
+ }
+
+ MidiByte type = atts.value("type").toInt();
+ MidiByte value = atts.value("value").toInt();
+
+ if (m_instrument) {
+ m_instrument->setControllerValue(type, value);
+ }
+
+ } else if (lcName == "plugin" || lcName == "synth") {
+
+ PluginContainer *container = 0;
+
+ if (m_section == InInstrument) {
+// std::cerr << "Found plugin in instrument" << std::endl;
+ container = m_instrument;
+ m_pluginInBuss = false;
+ } else if (m_section == InBuss) {
+// std::cerr << "Found plugin in buss" << std::endl;
+ container = m_buss;
+ m_pluginInBuss = true;
+ } else {
+ m_errorString = "Found Plugin outside Instrument or Buss";
+ return false;
+ }
+
+ // Despite being InInstrument or InBuss we might not actually
+ // have a valid one.
+ //
+ if (container) {
+
+// std::cerr << "Have container" << std::endl;
+
+ emit setOperationName(i18n("Loading plugins..."));
+ ProgressDialog::processEvents();
+
+ // Get the details
+ int position;
+ if (lcName == "synth") {
+ position = Instrument::SYNTH_PLUGIN_POSITION;
+ } else {
+ position = atts.value("position").toInt();
+ }
+
+ bool bypassed = false;
+ QString bpStr = atts.value("bypassed");
+ if (bpStr.lower() == "true")
+ bypassed = true;
+
+ std::string program = "";
+ QString progStr = atts.value("program");
+ if (progStr) {
+ program = qstrtostr(progStr);
+ }
+
+ // Plugins are identified by a structured identifier
+ // string, but we will accept a LADSPA UniqueId if there's
+ // no identifier, for backward compatibility
+
+ QString identifier = atts.value("identifier");
+
+ AudioPlugin *plugin = 0;
+ AudioPluginManager *apm = getAudioPluginManager();
+
+ if (!identifier) {
+ if (atts.value("id")) {
+ unsigned long id = atts.value("id").toULong();
+ if (apm)
+ plugin = apm->getPluginByUniqueId(id);
+ }
+ } else {
+ if (apm)
+ plugin = apm->getPluginByIdentifier(identifier);
+ }
+
+// std::cerr << "Plugin identifier " << identifier << " -> plugin " << plugin << std::endl;
+
+ // If we find the plugin all is well and good but if
+ // we don't we just skip it.
+ //
+ if (plugin) {
+ m_plugin = container->getPlugin(position);
+ if (!m_plugin) {
+ RG_DEBUG << "WARNING: RoseXmlHandler: instrument/buss "
+ << container->getId() << " has no plugin position "
+ << position << endl;
+ } else {
+ m_plugin->setAssigned(true);
+ m_plugin->setBypass(bypassed);
+ m_plugin->setIdentifier(plugin->getIdentifier().data());
+// std::cerr << "set identifier to plugin at position " << position << std::endl;
+ if (program != "") {
+ m_plugin->setProgram(program);
+ }
+ }
+ } else {
+ // we shouldn't be halting import of the RG file just because
+ // we can't match a plugin
+ //
+ if (identifier) {
+ RG_DEBUG << "WARNING: RoseXmlHandler: plugin " << identifier << " not found" << endl;
+ m_pluginsNotFound.insert(identifier);
+ } else if (atts.value("id")) {
+ RG_DEBUG << "WARNING: RoseXmlHandler: plugin uid " << atts.value("id") << " not found" << endl;
+ } else {
+ m_errorString = "No plugin identifier or uid specified";
+ return false;
+ }
+ }
+ } else { // no instrument
+
+ if (lcName == "synth") {
+ QString identifier = atts.value("identifier");
+ if (identifier) {
+ RG_DEBUG << "WARNING: RoseXmlHandler: no instrument for plugin " << identifier << endl;
+ m_pluginsNotFound.insert(identifier);
+ }
+ }
+ }
+
+ m_section = InPlugin;
+
+ } else if (lcName == "port") {
+
+ if (m_section != InPlugin) {
+ m_errorString = "Found Port outside Plugin";
+ return false;
+ }
+ unsigned long portId = atts.value("id").toULong();
+ double value = qstrtodouble(atts.value("value"));
+
+ QString changed = atts.value("changed");
+ bool changedSinceProgram = (changed == "true");
+
+ if (m_plugin) {
+ m_plugin->addPort(portId, value);
+ if (changedSinceProgram) {
+ PluginPortInstance *ppi = m_plugin->getPort(portId);
+ if (ppi)
+ ppi->changedSinceProgramChange = true;
+ }
+ }
+
+ } else if (lcName == "configure") {
+
+ if (m_section != InPlugin) {
+ m_errorString = "Found Configure outside Plugin";
+ return false;
+ }
+
+ QString key = atts.value("key");
+ QString value = atts.value("value");
+
+ if (m_plugin) {
+ m_plugin->setConfigurationValue(qstrtostr(key), qstrtostr(value));
+ }
+
+ } else if (lcName == "metronome") {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found Metronome outside Studio";
+ return false;
+ }
+
+ // Only create if we have a device
+ //
+ if (m_device && m_device->getType() == Device::Midi) {
+ InstrumentId instrument =
+ atts.value("instrument").toInt();
+
+ MidiMetronome metronome(instrument);
+
+ if (atts.value("barpitch"))
+ metronome.setBarPitch(atts.value("barpitch").toInt());
+ if (atts.value("beatpitch"))
+ metronome.setBeatPitch(atts.value("beatpitch").toInt());
+ if (atts.value("subbeatpitch"))
+ metronome.setSubBeatPitch(atts.value("subbeatpitch").toInt());
+ if (atts.value("depth"))
+ metronome.setDepth(atts.value("depth").toInt());
+ if (atts.value("barvelocity"))
+ metronome.setBarVelocity(atts.value("barvelocity").toInt());
+ if (atts.value("beatvelocity"))
+ metronome.setBeatVelocity(atts.value("beatvelocity").toInt());
+ if (atts.value("subbeatvelocity"))
+ metronome.setSubBeatVelocity(atts.value("subbeatvelocity").toInt());
+
+ dynamic_cast<MidiDevice*>(m_device)->
+ setMetronome(metronome);
+ }
+
+ } else if (lcName == "instrument") {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found Instrument outside Studio";
+ return false;
+ }
+
+ m_section = InInstrument;
+
+ InstrumentId id = atts.value("id").toInt();
+ std::string stringType = qstrtostr(atts.value("type"));
+ Instrument::InstrumentType type;
+
+ if (stringType == "midi")
+ type = Instrument::Midi;
+ else if (stringType == "audio")
+ type = Instrument::Audio;
+ else if (stringType == "softsynth")
+ type = Instrument::SoftSynth;
+ else {
+ m_errorString = "Found unknown Instrument type";
+ return false;
+ }
+
+ // Try and match an Instrument in the file with one in
+ // our studio
+ //
+ Instrument *instrument = getStudio().getInstrumentById(id);
+
+ // If we've got an instrument and the types match then
+ // we use it from now on.
+ //
+ if (instrument && instrument->getType() == type) {
+ m_instrument = instrument;
+
+ // We can also get the channel from this tag
+ //
+ MidiByte channel =
+ (MidiByte)atts.value("channel").toInt();
+ m_instrument->setMidiChannel(channel);
+ }
+
+ } else if (lcName == "buss") {
+
+ if (m_section != InStudio) {
+ m_errorString = "Found Buss outside Studio";
+ return false;
+ }
+
+ m_section = InBuss;
+
+ BussId id = atts.value("id").toInt();
+ Buss *buss = getStudio().getBussById(id);
+
+ // If we've got a buss then we use it from now on.
+ //
+ if (buss) {
+ m_buss = buss;
+ } else {
+ m_buss = new Buss(id);
+ getStudio().addBuss(m_buss);
+ }
+
+ } else if (lcName == "audiofiles") {
+
+ if (m_section != NoSection) {
+ m_errorString = "Found AudioFiles inside another section";
+ return false;
+ }
+
+ m_section = InAudioFiles;
+
+ int rate = atts.value("expectedRate").toInt();
+ if (rate) {
+ getAudioFileManager().setExpectedSampleRate(rate);
+ }
+
+ } else if (lcName == "configuration") {
+
+ setSubHandler(new ConfigurationXmlSubHandler
+ (lcName, &m_doc->getConfiguration()));
+
+ } else if (lcName == "metadata") {
+
+ if (m_section != InComposition) {
+ m_errorString = "Found Metadata outside Composition";
+ return false;
+ }
+
+ setSubHandler(new ConfigurationXmlSubHandler
+ (lcName, &getComposition().getMetadata()));
+
+ } else if (lcName == "recordlevel") {
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found recordLevel outside Instrument";
+ return false;
+ }
+
+ double value = qstrtodouble(atts.value("value"));
+
+ // if the value retrieved is greater than (say) 15 then we
+ // must have an old-style 0-127 value instead of a shiny new
+ // dB value, so convert it
+ if (value > 15.0) {
+ value = AudioLevel::multiplier_to_dB(value / 100);
+ }
+
+ if (m_instrument)
+ m_instrument->setRecordLevel(value);
+
+ } else if (lcName == "audioinput") {
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found audioInput outside Instrument";
+ return false;
+ }
+
+ int value = atts.value("value").toInt();
+ int channel = atts.value("channel").toInt();
+
+ QString type = atts.value("type");
+ if (type) {
+ if (type.lower() == "buss") {
+ if (m_instrument)
+ m_instrument->setAudioInputToBuss(value, channel);
+ } else if (type.lower() == "record") {
+ if (m_instrument)
+ m_instrument->setAudioInputToRecord(value, channel);
+ }
+ }
+
+ } else if (lcName == "audiooutput") {
+
+ if (m_section != InInstrument) {
+ m_errorString = "Found audioOutput outside Instrument";
+ return false;
+ }
+
+ int value = atts.value("value").toInt();
+ if (m_instrument)
+ m_instrument->setAudioOutput(value);
+
+ } else if (lcName == "appearance") {
+
+ m_section = InAppearance;
+
+ } else if (lcName == "colourmap") {
+
+ if (m_section == InAppearance) {
+ QString mapName = atts.value("name");
+ m_inColourMap = true;
+ if (mapName == "segmentmap") {
+ m_colourMap = &m_doc->getComposition().getSegmentColourMap();
+ } else
+ if (mapName == "generalmap") {
+ m_colourMap = &m_doc->getComposition().getGeneralColourMap();
+ } else { // This will change later once we get more of the Appearance code sorted out
+ RG_DEBUG << "RoseXmlHandler::startElement : Found colourmap with unknown name\n";
+ }
+ } else {
+ m_errorString = "Found colourmap outside Appearance";
+ return false;
+ }
+
+ } else if (lcName == "colourpair") {
+
+ if (m_inColourMap && m_colourMap) {
+ unsigned int id = atts.value("id").toInt();
+ QString name = atts.value("name");
+ unsigned int red = atts.value("red").toInt();
+ unsigned int blue = atts.value("blue").toInt();
+ unsigned int green = atts.value("green").toInt();
+ Colour colour(red, green, blue);
+ m_colourMap->addItem(colour, qstrtostr(name), id);
+ } else {
+ m_errorString = "Found colourpair outside ColourMap";
+ return false;
+ }
+
+ } else if (lcName == "markers") {
+
+ if (!m_inComposition) {
+ m_errorString = "Found Markers outside Composition";
+ return false;
+ }
+
+ // clear down any markers
+ getComposition().clearMarkers();
+
+ } else if (lcName == "marker") {
+ if (!m_inComposition) {
+ m_errorString = "Found Marker outside Composition";
+ return false;
+ }
+ int time = atts.value("time").toInt();
+ QString name = atts.value("name");
+ QString descr = atts.value("description");
+
+ Marker *marker =
+ new Marker(time,
+ qstrtostr(name),
+ qstrtostr(descr));
+
+ getComposition().addMarker(marker);
+ } else {
+ RG_DEBUG << "RoseXmlHandler::startElement : Don't know how to parse this : " << qName << endl;
+ }
+
+ return true;
+}
+
+bool
+RoseXmlHandler::endElement(const QString& namespaceURI,
+ const QString& localName,
+ const QString& qName)
+{
+ if (getSubHandler()) {
+ bool finished;
+ bool res = getSubHandler()->endElement(namespaceURI, localName, qName.lower(), finished);
+ if (finished)
+ setSubHandler(0);
+ return res;
+ }
+
+ // Set percentage done
+ //
+ if ((m_totalElements > m_elementsSoFar) &&
+ (++m_elementsSoFar % 300 == 0)) {
+
+ emit setProgress(int(double(m_elementsSoFar) / double(m_totalElements) * 100.0));
+ ProgressDialog::processEvents();
+ }
+
+ QString lcName = qName.lower();
+
+ if (lcName == "rosegarden-data") {
+
+ getComposition().updateTriggerSegmentReferences();
+
+ } else if (lcName == "event") {
+
+ if (m_currentSegment && m_currentEvent) {
+ m_currentSegment->insert(m_currentEvent);
+ m_currentEvent = 0;
+ } else if (!m_currentSegment && m_currentEvent) {
+ m_errorString = "Got event outside of a Segment";
+ return false;
+ }
+
+ } else if (lcName == "chord") {
+
+ m_currentTime += m_chordDuration;
+ m_inChord = false;
+ m_chordDuration = 0;
+
+ } else if (lcName == "group") {
+
+ m_inGroup = false;
+
+ } else if (lcName == "segment") {
+
+ if (m_currentSegment && m_segmentEndMarkerTime) {
+ m_currentSegment->setEndMarkerTime(*m_segmentEndMarkerTime);
+ delete m_segmentEndMarkerTime;
+ m_segmentEndMarkerTime = 0;
+ }
+
+ m_currentSegment = 0;
+ m_section = NoSection;
+
+ } else if (lcName == "bar-segment" || lcName == "tempo-segment") {
+
+ m_currentSegment = 0;
+
+ } else if (lcName == "composition") {
+ m_inComposition = false;
+ m_section = NoSection;
+
+ } else if (lcName == "studio") {
+
+ m_section = NoSection;
+
+ } else if (lcName == "buss") {
+
+ m_section = InStudio;
+ m_buss = 0;
+
+ } else if (lcName == "instrument") {
+
+ m_section = InStudio;
+ m_instrument = 0;
+
+ } else if (lcName == "plugin") {
+
+ if (m_pluginInBuss) {
+ m_section = InBuss;
+ } else {
+ m_section = InInstrument;
+ }
+ m_plugin = 0;
+ m_pluginId = 0;
+
+ } else if (lcName == "device") {
+
+ m_device = 0;
+
+ } else if (lcName == "keymapping") {
+
+ if (m_section == InStudio) {
+ if (m_keyMapping) {
+ if (!m_keyNameMap.empty()) {
+ MidiDevice *md = dynamic_cast<MidiDevice *>
+ (m_device);
+ if (md) {
+ m_keyMapping->setMap(m_keyNameMap);
+ md->addKeyMapping(*m_keyMapping);
+ }
+ }
+ m_keyMapping = 0;
+ }
+ }
+
+ } else if (lcName == "audiofiles") {
+
+ m_section = NoSection;
+
+ } else if (lcName == "appearance") {
+
+ m_section = NoSection;
+
+ } else if (lcName == "colourmap") {
+ m_inColourMap = false;
+ m_colourMap = 0;
+ }
+
+ return true;
+}
+
+bool
+RoseXmlHandler::characters(const QString& s)
+{
+ if (m_subHandler)
+ return m_subHandler->characters(s);
+
+ return true;
+}
+
+QString
+RoseXmlHandler::errorString()
+{
+ return m_errorString;
+}
+
+bool
+RoseXmlHandler::error(const QXmlParseException& exception)
+{
+ m_errorString = QString("%1 at line %2, column %3")
+ .arg(exception.message())
+ .arg(exception.lineNumber())
+ .arg(exception.columnNumber());
+ return QXmlDefaultHandler::error( exception );
+}
+
+bool
+RoseXmlHandler::fatalError(const QXmlParseException& exception)
+{
+ m_errorString = QString("%1 at line %2, column %3")
+ .arg(exception.message())
+ .arg(exception.lineNumber())
+ .arg(exception.columnNumber());
+ return QXmlDefaultHandler::fatalError( exception );
+}
+
+bool
+RoseXmlHandler::endDocument()
+{
+ if (m_foundTempo == false) {
+ getComposition().setCompositionDefaultTempo
+ (Composition::getTempoForQpm(120.0));
+ }
+
+ return true;
+}
+
+void
+RoseXmlHandler::setSubHandler(XmlSubHandler* sh)
+{
+ delete m_subHandler;
+ m_subHandler = sh;
+}
+
+void
+RoseXmlHandler::addMIDIDevice(QString name, bool createAtSequencer)
+{
+ unsigned int deviceId = 0;
+
+ if (createAtSequencer) {
+
+ QByteArray data;
+ QByteArray replyData;
+ QCString replyType;
+ QDataStream arg(data, IO_WriteOnly);
+
+ arg << (int)Device::Midi;
+ arg << (unsigned int)MidiDevice::Play;
+
+ if (!rgapp->sequencerCall("addDevice(int, unsigned int)", replyType, replyData, data)) {
+ SEQMAN_DEBUG << "RoseXmlHandler::addMIDIDevice - "
+ << "can't call sequencer addDevice" << endl;
+ return ;
+ }
+
+ if (replyType == "unsigned int") {
+ QDataStream reply(replyData, IO_ReadOnly);
+ reply >> deviceId;
+ } else {
+ SEQMAN_DEBUG << "RoseXmlHandler::addMIDIDevice - "
+ << "got unknown returntype from addDevice()" << endl;
+ return ;
+ }
+
+ if (deviceId == Device::NO_DEVICE) {
+ SEQMAN_DEBUG << "RoseXmlHandler::addMIDIDevice - "
+ << "sequencer addDevice failed" << endl;
+ return ;
+ }
+
+ SEQMAN_DEBUG << "RoseXmlHandler::addMIDIDevice - "
+ << " added device " << deviceId << endl;
+
+ } else {
+ // Generate a new device id at the base Studio side only.
+ // This may not correspond to any given device id at the
+ // sequencer side. We should _never_ do this in a document
+ // that's actually intended to be retained for use, only
+ // in temporary documents for device import etc.
+ int tempId = -1;
+ for (DeviceListIterator i = getStudio().getDevices()->begin();
+ i != getStudio().getDevices()->end(); ++i) {
+ if (int((*i)->getId()) > tempId)
+ tempId = int((*i)->getId());
+ }
+ deviceId = tempId + 1;
+ }
+
+ // add the device, so we can name it and set our pointer to it --
+ // instruments will be sync'd later in the natural course of things
+ getStudio().addDevice(qstrtostr(name), deviceId, Device::Midi);
+ m_device = getStudio().getDevice(deviceId);
+ m_deviceRunningId = deviceId;
+}
+
+void
+RoseXmlHandler::skipToNextPlayDevice()
+{
+ SEQMAN_DEBUG << "RoseXmlHandler::skipToNextPlayDevice; m_deviceRunningId is " << m_deviceRunningId << endl;
+
+ for (DeviceList::iterator i = getStudio().getDevices()->begin();
+ i != getStudio().getDevices()->end(); ++i) {
+
+ MidiDevice *md =
+ dynamic_cast<MidiDevice *>(*i);
+
+ if (md && md->getDirection() == MidiDevice::Play) {
+ if (m_deviceRunningId == Device::NO_DEVICE ||
+ md->getId() > m_deviceRunningId) {
+
+ SEQMAN_DEBUG << "RoseXmlHandler::skipToNextPlayDevice: found next device: id " << md->getId() << endl;
+
+ m_device = md;
+ m_deviceRunningId = md->getId();
+ return ;
+ }
+ }
+ }
+
+ SEQMAN_DEBUG << "RoseXmlHandler::skipToNextPlayDevice: fresh out of devices" << endl;
+
+ m_device = 0;
+}
+
+void
+RoseXmlHandler::setMIDIDeviceConnection(QString connection)
+{
+ SEQMAN_DEBUG << "RoseXmlHandler::setMIDIDeviceConnection(" << connection << ")" << endl;
+
+ MidiDevice *md = dynamic_cast<MidiDevice *>(m_device);
+ if (!md)
+ return ;
+
+ QByteArray data;
+ QDataStream arg(data, IO_WriteOnly);
+
+ arg << (unsigned int)md->getId();
+ arg << connection;
+
+ rgapp->sequencerSend("setPlausibleConnection(unsigned int, QString)",
+ data);
+ // connection should be sync'd later in the natural course of things
+}
+
+void
+RoseXmlHandler::setMIDIDeviceName(QString name)
+{
+ SEQMAN_DEBUG << "RoseXmlHandler::setMIDIDeviceName(" << name << ")" << endl;
+
+ MidiDevice *md = dynamic_cast<MidiDevice *>(m_device);
+ if (!md)
+ return ;
+
+ QByteArray data;
+ QDataStream arg(data, IO_WriteOnly);
+
+ arg << (unsigned int)md->getId();
+ arg << name;
+
+ std::cerr << "Renaming device " << md->getId() << " to " << name << std::endl;
+
+ rgapp->sequencerSend("renameDevice(unsigned int, QString)",
+ data);
+}
+
+}