Chris@45: /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ Chris@45: Chris@45: /* Chris@45: Sonic Visualiser Chris@45: An audio file viewer and annotation editor. Chris@45: Centre for Digital Music, Queen Mary, University of London. Chris@45: This file copyright 2006 Chris Cannam and QMUL. Chris@45: Chris@45: This program is free software; you can redistribute it and/or Chris@45: modify it under the terms of the GNU General Public License as Chris@45: published by the Free Software Foundation; either version 2 of the Chris@45: License, or (at your option) any later version. See the file Chris@45: COPYING included with this distribution for more information. Chris@45: */ Chris@45: Chris@635: #ifndef SV_FILE_READER_H Chris@635: #define SV_FILE_READER_H Chris@45: Chris@45: #include "layer/LayerFactory.h" Chris@106: #include "transform/Transform.h" Chris@45: Chris@45: #include <QXmlDefaultHandler> Chris@45: Chris@45: #include <map> Chris@45: Chris@45: class Pane; Chris@45: class Model; Chris@685: class Path; Chris@45: class Document; Chris@45: class PlayParameters; Chris@45: Chris@45: class SVFileReaderPaneCallback Chris@45: { Chris@45: public: Chris@45: virtual ~SVFileReaderPaneCallback(); Chris@45: virtual Pane *addPane() = 0; Chris@45: virtual void setWindowSize(int width, int height) = 0; Chris@435: virtual void addSelection(sv_frame_t start, sv_frame_t end) = 0; Chris@45: }; Chris@45: Chris@45: /** Chris@45: SVFileReader loads Sonic Visualiser XML files. (The SV file Chris@45: format is bzipped XML.) Chris@45: Chris@45: Some notes about the SV XML format follow. We're very lazy with Chris@45: our XML: there's no schema or DTD, and we depend heavily on Chris@45: elements being in a particular order. Chris@45: Chris@45: \verbatim Chris@45: Chris@45: <sv> Chris@45: Chris@45: <data> Chris@45: Chris@45: <!-- The data section contains definitions of both models and Chris@45: visual layers. Layers are considered data in the document; Chris@45: the structure of views that displays the layers is not. --> Chris@45: Chris@45: <!-- id numbers are unique within the data type (i.e. no two Chris@45: models can have the same id, but a model can have the same Chris@45: id as a layer, etc). SV generates its id numbers just for Chris@45: the purpose of cross-referencing within the current file; Chris@45: they don't necessarily have any meaning once the file has Chris@45: been loaded. --> Chris@45: Chris@45: <model id="0" name="..." type="..." ... /> Chris@45: <model id="1" name="..." type="..." ... /> Chris@45: Chris@45: <!-- Models that have data associated with them store it Chris@45: in a neighbouring dataset element. The dataset must follow Chris@45: the model and precede any derivation or layer elements that Chris@45: refer to the model. --> Chris@45: Chris@45: <model id="2" name="..." type="..." dataset="0" ... /> Chris@45: Chris@45: <dataset id="0" type="..."> Chris@45: <point frame="..." value="..." ... /> Chris@45: </dataset> Chris@45: Chris@45: <!-- Where one model is derived from another via a transform, Chris@45: it has an associated derivation element. This must follow Chris@45: both the source and target model elements. The source and Chris@45: model attributes give the source model id and target model Chris@45: id respectively. A model can have both dataset and Chris@45: derivation elements; if it does, dataset must appear first. Chris@45: If the model's data are not stored, but instead the model Chris@45: is to be regenerated completely from the transform when Chris@45: the session is reloaded, then the model should have _only_ Chris@45: a derivation element, and no model element should appear Chris@45: for it at all. --> Chris@45: Chris@72: <derivation type="transform" source="0" model="2" channel="-1"> Chris@72: <transform id="vamp:soname:pluginid:output" ... /> Chris@72: </derivation> Chris@72: Chris@72: <!-- Note that the derivation element just described replaces Chris@72: this earlier formulation, which had more attributes in the Chris@72: derivation element and a plugin element describing plugin Chris@72: parameters and properties. What we actually read and Chris@72: write these days is a horrid composite of the two formats, Chris@72: for backward compatibility reasons. --> Chris@72: Chris@72: <derivation source="0" model="2" transform="vamp:soname:pluginid:output" ...> Chris@72: <plugin id="pluginid" ... /> Chris@45: </derivation> Chris@45: Chris@45: <!-- The playparameters element lists playback settings for Chris@45: a model. --> Chris@45: Chris@45: <playparameters mute="false" pan="0" gain="1" model="1" ... /> Chris@45: Chris@45: <!-- Layer elements. The models must have already been defined. Chris@45: The same model may appear in more than one layer (of more Chris@45: than one type). --> Chris@45: Chris@45: <layer id="1" type="..." name="..." model="0" ... /> Chris@45: <layer id="2" type="..." name="..." model="1" ... /> Chris@45: Chris@45: </data> Chris@45: Chris@45: Chris@45: <display> Chris@45: Chris@45: <!-- The display element contains visual structure for the Chris@45: layers. It's simpler than the data section. --> Chris@45: Chris@587: <!-- Overall preferred window size for this session. (Now Chris@587: deprecated, it wasn't a good idea to try to persist this) --> Chris@45: Chris@45: <window width="..." height="..."/> Chris@45: Chris@45: <!-- List of view elements to stack up. Each one contains Chris@45: a list of layers in stacking order, back to front. --> Chris@45: Chris@45: <view type="pane" ...> Chris@45: <layer id="1"/> Chris@45: <layer id="2"/> Chris@45: </view> Chris@45: Chris@45: <!-- The layer elements just refer to layers defined in the Chris@45: data section, so they don't have to have any attributes Chris@45: other than the id. For sort-of-historical reasons SV Chris@45: actually does repeat the other attributes here, but Chris@45: it doesn't need to. --> Chris@45: Chris@45: <view type="pane" ...> Chris@45: <layer id="2"/> Chris@45: <view> Chris@45: Chris@45: </display> Chris@45: Chris@45: Chris@45: <!-- List of selected regions by audio frame extents. --> Chris@45: Chris@45: <selections> Chris@45: <selection start="..." end="..."/> Chris@45: </selections> Chris@45: Chris@45: Chris@45: </sv> Chris@45: Chris@45: \endverbatim Chris@45: */ Chris@45: Chris@45: Chris@79: class SVFileReader : public QObject, QXmlDefaultHandler Chris@45: { Chris@79: Q_OBJECT Chris@79: Chris@45: public: Chris@45: SVFileReader(Document *document, Chris@595: SVFileReaderPaneCallback &callback, Chris@45: QString location = ""); // for audio file locate mechanism Chris@45: virtual ~SVFileReader(); Chris@45: Chris@45: void parse(const QString &xmlData); Chris@45: void parse(QXmlInputSource &source); Chris@45: Chris@45: bool isOK(); Chris@45: QString getErrorString() const { return m_errorString; } Chris@45: Chris@45: // For loading a single layer onto an existing pane Chris@45: void setCurrentPane(Pane *pane) { m_currentPane = pane; } Chris@45: Chris@634: bool startElement(const QString &namespaceURI, Chris@685: const QString &localName, Chris@685: const QString &qName, Chris@685: const QXmlAttributes& atts) override; Chris@685: Chris@634: bool characters(const QString &) override; Chris@45: Chris@634: bool endElement(const QString &namespaceURI, Chris@685: const QString &localName, Chris@685: const QString &qName) override; Chris@45: Chris@634: bool error(const QXmlParseException &exception) override; Chris@634: bool fatalError(const QXmlParseException &exception) override; Chris@45: Chris@140: enum FileType Chris@140: { Chris@140: SVSessionFile, Chris@140: SVLayerFile, Chris@140: UnknownFileType Chris@140: }; Chris@140: Chris@140: static FileType identifyXmlFile(QString path); Chris@140: Chris@79: signals: Chris@79: void modelRegenerationFailed(QString layerName, QString transformName, Chris@79: QString message); Chris@79: void modelRegenerationWarning(QString layerName, QString transformName, Chris@79: QString message); Chris@79: Chris@45: protected: Chris@45: bool readWindow(const QXmlAttributes &); Chris@45: bool readModel(const QXmlAttributes &); Chris@45: bool readView(const QXmlAttributes &); Chris@45: bool readLayer(const QXmlAttributes &); Chris@45: bool readDatasetStart(const QXmlAttributes &); Chris@45: bool addBinToDataset(const QXmlAttributes &); Chris@45: bool addPointToDataset(const QXmlAttributes &); Chris@45: bool addRowToDataset(const QXmlAttributes &); Chris@45: bool readRowData(const QString &); Chris@45: bool readDerivation(const QXmlAttributes &); Chris@45: bool readPlayParameters(const QXmlAttributes &); Chris@45: bool readPlugin(const QXmlAttributes &); Chris@308: bool readPluginForTransform(const QXmlAttributes &); Chris@308: bool readPluginForPlayback(const QXmlAttributes &); Chris@72: bool readTransform(const QXmlAttributes &); Chris@72: bool readParameter(const QXmlAttributes &); Chris@45: bool readSelection(const QXmlAttributes &); Chris@45: bool readMeasurement(const QXmlAttributes &); Chris@629: Chris@629: void makeAggregateModels(); Chris@45: void addUnaddedModels(); Chris@45: Chris@685: // We use the term "pending" of things that have been referred to Chris@685: // but not yet constructed because their definitions are Chris@685: // incomplete. They are just referred to with an ExportId. Models Chris@685: // that have been constructed are always added straight away to Chris@685: // ById and are referred to with a ModelId (everywhere where Chris@685: // previously we would have used a Model *). m_models maps from Chris@685: // ExportId (as read from the file) to complete Models, or to a Chris@685: // ModelId of None for any model that could not be constructed for Chris@685: // some reason. Chris@685: Chris@685: typedef XmlExportable::ExportId ExportId; Chris@685: Chris@685: bool haveModel(ExportId id) { Chris@685: return (m_models.find(id) != m_models.end()) && !m_models[id].isNone(); Chris@45: } Chris@685: Chris@629: struct PendingAggregateRec { Chris@629: QString name; Chris@629: sv_samplerate_t sampleRate; Chris@685: std::vector<ExportId> components; Chris@629: }; Chris@629: Chris@45: Document *m_document; Chris@45: SVFileReaderPaneCallback &m_paneCallback; Chris@45: QString m_location; Chris@45: Pane *m_currentPane; Chris@685: std::map<ExportId, Layer *> m_layers; Chris@685: std::map<ExportId, ModelId> m_models; Chris@685: std::map<ExportId, Path *> m_paths; Chris@685: std::set<ModelId> m_addedModels; // i.e. added to Document, not just ById Chris@685: std::map<ExportId, PendingAggregateRec> m_pendingAggregates; Chris@685: Chris@685: // A model element often contains a dataset id, and the dataset Chris@685: // then follows it. When the model is read, an entry in this map Chris@685: // is added, mapping from the dataset's export id (the actual Chris@685: // dataset has not been read yet) back to the export id of the Chris@685: // object that needs it. We map to export id rather than model id, Chris@685: // because the object might be a path rather than a model. Chris@685: std::map<ExportId, ExportId> m_awaitingDatasets; Chris@685: Chris@685: // And then this is the model or path that a dataset element is Chris@685: // currently being read into, i.e. the value looked up from Chris@685: // m_awaitingDatasets at the point where the dataset is found. Chris@685: ExportId m_currentDataset; Chris@685: Chris@45: Layer *m_currentLayer; Chris@685: ModelId m_currentDerivedModel; Chris@685: ExportId m_pendingDerivedModel; Chris@686: std::shared_ptr<PlayParameters> m_currentPlayParameters; Chris@72: Transform m_currentTransform; Chris@685: ModelId m_currentTransformSource; Chris@72: int m_currentTransformChannel; Chris@72: bool m_currentTransformIsNewStyle; Chris@45: QString m_datasetSeparator; Chris@45: bool m_inRow; Chris@45: bool m_inLayer; Chris@45: bool m_inView; Chris@45: bool m_inData; Chris@45: bool m_inSelections; Chris@45: int m_rowNumber; Chris@45: QString m_errorString; Chris@45: bool m_ok; Chris@45: }; Chris@45: Chris@45: #endif