lbajardsilogic@0: /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ lbajardsilogic@0: lbajardsilogic@0: /* lbajardsilogic@0: Sonic Visualiser lbajardsilogic@0: An audio file viewer and annotation editor. lbajardsilogic@0: Centre for Digital Music, Queen Mary, University of London. lbajardsilogic@0: This file copyright 2006 Chris Cannam. lbajardsilogic@0: lbajardsilogic@0: This program is free software; you can redistribute it and/or lbajardsilogic@0: modify it under the terms of the GNU General Public License as lbajardsilogic@0: published by the Free Software Foundation; either version 2 of the lbajardsilogic@0: License, or (at your option) any later version. See the file lbajardsilogic@0: COPYING included with this distribution for more information. lbajardsilogic@0: */ lbajardsilogic@0: lbajardsilogic@0: #ifndef _NOTE_MODEL_H_ lbajardsilogic@0: #define _NOTE_MODEL_H_ lbajardsilogic@0: lbajardsilogic@0: #include "SparseValueModel.h" lbajardsilogic@0: #include "base/PlayParameterRepository.h" lbajardsilogic@0: #include "base/RealTime.h" lbajardsilogic@0: lbajardsilogic@0: /** lbajardsilogic@0: * Note type for use in a SparseModel or SparseValueModel. All we lbajardsilogic@0: * mean by a "note" is something that has an onset time, a single lbajardsilogic@0: * value, and a duration. Like other points, it can also have a lbajardsilogic@0: * label. With this point type, the model can be thought of as lbajardsilogic@0: * representing a simple MIDI-type piano roll, except that the y lbajardsilogic@0: * coordinates (values) do not have to be discrete integers. lbajardsilogic@0: */ lbajardsilogic@0: lbajardsilogic@0: struct Note lbajardsilogic@0: { lbajardsilogic@0: public: lbajardsilogic@0: Note(long _frame) : frame(_frame), value(0.0f), duration(0) { } lbajardsilogic@0: Note(long _frame, float _value, size_t _duration, QString _label) : lbajardsilogic@0: frame(_frame), value(_value), duration(_duration), label(_label) { } lbajardsilogic@0: lbajardsilogic@0: int getDimensions() const { return 3; } lbajardsilogic@0: lbajardsilogic@0: long frame; lbajardsilogic@0: float value; lbajardsilogic@0: size_t duration; lbajardsilogic@0: QString label; lbajardsilogic@0: lbajardsilogic@0: QString toXmlString(QString indent = "", lbajardsilogic@0: QString extraAttributes = "") const lbajardsilogic@0: { lbajardsilogic@0: return QString("%1\n") lbajardsilogic@0: .arg(indent).arg(frame).arg(value).arg(duration).arg(label).arg(extraAttributes); lbajardsilogic@0: } lbajardsilogic@0: lbajardsilogic@0: QString toDelimitedDataString(QString delimiter, size_t sampleRate) const lbajardsilogic@0: { lbajardsilogic@0: QStringList list; lbajardsilogic@0: list << RealTime::frame2RealTime(frame, sampleRate).toString().c_str(); lbajardsilogic@0: list << QString("%1").arg(value); lbajardsilogic@0: list << QString("%1").arg(duration); lbajardsilogic@0: list << label; lbajardsilogic@0: return list.join(delimiter); lbajardsilogic@0: } lbajardsilogic@0: lbajardsilogic@0: struct Comparator { lbajardsilogic@0: bool operator()(const Note &p1, lbajardsilogic@0: const Note &p2) const { lbajardsilogic@0: if (p1.frame != p2.frame) return p1.frame < p2.frame; lbajardsilogic@0: if (p1.value != p2.value) return p1.value < p2.value; lbajardsilogic@0: if (p1.duration != p2.duration) return p1.duration < p2.duration; lbajardsilogic@0: return p1.label < p2.label; lbajardsilogic@0: } lbajardsilogic@0: }; lbajardsilogic@0: lbajardsilogic@0: struct OrderComparator { lbajardsilogic@0: bool operator()(const Note &p1, lbajardsilogic@0: const Note &p2) const { lbajardsilogic@0: return p1.frame < p2.frame; lbajardsilogic@0: } lbajardsilogic@0: }; lbajardsilogic@0: }; lbajardsilogic@0: lbajardsilogic@0: lbajardsilogic@0: class NoteModel : public SparseValueModel lbajardsilogic@0: { lbajardsilogic@0: public: lbajardsilogic@0: NoteModel(size_t sampleRate, size_t resolution, lbajardsilogic@0: bool notifyOnAdd = true) : lbajardsilogic@0: SparseValueModel(sampleRate, resolution, lbajardsilogic@0: notifyOnAdd), lbajardsilogic@0: m_valueQuantization(0) lbajardsilogic@0: { lbajardsilogic@0: PlayParameterRepository::getInstance()->addModel(this); lbajardsilogic@0: } lbajardsilogic@0: lbajardsilogic@0: NoteModel(size_t sampleRate, size_t resolution, lbajardsilogic@0: float valueMinimum, float valueMaximum, lbajardsilogic@0: bool notifyOnAdd = true) : lbajardsilogic@0: SparseValueModel(sampleRate, resolution, lbajardsilogic@0: valueMinimum, valueMaximum, lbajardsilogic@0: notifyOnAdd), lbajardsilogic@0: m_valueQuantization(0) lbajardsilogic@0: { lbajardsilogic@0: PlayParameterRepository::getInstance()->addModel(this); lbajardsilogic@0: } lbajardsilogic@0: lbajardsilogic@0: float getValueQuantization() const { return m_valueQuantization; } lbajardsilogic@0: void setValueQuantization(float q) { m_valueQuantization = q; } lbajardsilogic@0: lbajardsilogic@0: /** lbajardsilogic@0: * Notes have a duration, so this returns all points that span any lbajardsilogic@0: * of the given range (as well as the usual additional few before lbajardsilogic@0: * and after). Consequently this can be very slow (optimised data lbajardsilogic@0: * structures still to be done!). lbajardsilogic@0: */ lbajardsilogic@0: virtual PointList getPoints(long start, long end) const; lbajardsilogic@0: lbajardsilogic@0: /** lbajardsilogic@0: * Notes have a duration, so this returns all points that span the lbajardsilogic@0: * given frame. Consequently this can be very slow (optimised lbajardsilogic@0: * data structures still to be done!). lbajardsilogic@0: */ lbajardsilogic@0: virtual PointList getPoints(long frame) const; lbajardsilogic@0: lbajardsilogic@0: virtual QString toXmlString(QString indent = "", lbajardsilogic@0: QString extraAttributes = "") const lbajardsilogic@0: { lbajardsilogic@0: return SparseValueModel::toXmlString lbajardsilogic@0: (indent, lbajardsilogic@0: QString("%1 valueQuantization=\"%2\"") lbajardsilogic@0: .arg(extraAttributes).arg(m_valueQuantization)); lbajardsilogic@0: } lbajardsilogic@0: lbajardsilogic@0: protected: lbajardsilogic@0: float m_valueQuantization; lbajardsilogic@0: }; lbajardsilogic@0: lbajardsilogic@0: #endif