| Chris@58 | 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*-  vi:set ts=8 sts=4 sw=4: */ | 
| Chris@0 | 2 | 
| Chris@0 | 3 /* | 
| Chris@59 | 4     Sonic Visualiser | 
| Chris@59 | 5     An audio file viewer and annotation editor. | 
| Chris@59 | 6     Centre for Digital Music, Queen Mary, University of London. | 
| Chris@182 | 7     This file copyright 2006 Chris Cannam and QMUL. | 
| Chris@0 | 8 | 
| Chris@59 | 9     This program is free software; you can redistribute it and/or | 
| Chris@59 | 10     modify it under the terms of the GNU General Public License as | 
| Chris@59 | 11     published by the Free Software Foundation; either version 2 of the | 
| Chris@59 | 12     License, or (at your option) any later version.  See the file | 
| Chris@59 | 13     COPYING included with this distribution for more information. | 
| Chris@0 | 14 */ | 
| Chris@0 | 15 | 
| Chris@30 | 16 #ifndef _WAVEFORM_LAYER_H_ | 
| Chris@30 | 17 #define _WAVEFORM_LAYER_H_ | 
| Chris@0 | 18 | 
| Chris@0 | 19 #include <QRect> | 
| Chris@0 | 20 | 
| Chris@287 | 21 #include "SingleColourLayer.h" | 
| Chris@0 | 22 | 
| Chris@128 | 23 #include "data/model/RangeSummarisableTimeValueModel.h" | 
| Chris@0 | 24 | 
| Chris@0 | 25 class View; | 
| Chris@0 | 26 class QPainter; | 
| Chris@0 | 27 class QPixmap; | 
| Chris@0 | 28 | 
| Chris@287 | 29 class WaveformLayer : public SingleColourLayer | 
| Chris@0 | 30 { | 
| Chris@0 | 31     Q_OBJECT | 
| Chris@0 | 32 | 
| Chris@0 | 33 public: | 
| Chris@44 | 34     WaveformLayer(); | 
| Chris@0 | 35     ~WaveformLayer(); | 
| Chris@0 | 36 | 
| Chris@156 | 37     virtual const ZoomConstraint *getZoomConstraint() const { | 
| Chris@156 | 38         return m_model ? m_model->getZoomConstraint() : 0; | 
| Chris@156 | 39     } | 
| Chris@0 | 40     virtual const Model *getModel() const { return m_model; } | 
| Chris@44 | 41     virtual void paint(View *v, QPainter &paint, QRect rect) const; | 
| Chris@0 | 42 | 
| Chris@44 | 43     virtual QString getFeatureDescription(View *v, QPoint &) const; | 
| Chris@25 | 44 | 
| Chris@287 | 45     virtual ColourSignificance getLayerColourSignificance() const { | 
| Chris@287 | 46         return ColourAndBackgroundSignificant; | 
| Chris@287 | 47     } | 
| Chris@287 | 48 | 
| Chris@44 | 49     virtual int getVerticalScaleWidth(View *v, QPainter &) const; | 
| Chris@44 | 50     virtual void paintVerticalScale(View *v, QPainter &paint, QRect rect) const; | 
| Chris@0 | 51 | 
| Chris@0 | 52     void setModel(const RangeSummarisableTimeValueModel *model); | 
| Chris@0 | 53 | 
| Chris@0 | 54     virtual PropertyList getProperties() const; | 
| Chris@87 | 55     virtual QString getPropertyLabel(const PropertyName &) const; | 
| Chris@335 | 56     virtual QString getPropertyIconName(const PropertyName &) const; | 
| Chris@0 | 57     virtual PropertyType getPropertyType(const PropertyName &) const; | 
| Chris@0 | 58     virtual QString getPropertyGroupName(const PropertyName &) const; | 
| Chris@0 | 59     virtual int getPropertyRangeAndValue(const PropertyName &, | 
| Chris@216 | 60                                          int *min, int *max, int *deflt) const; | 
| Chris@0 | 61     virtual QString getPropertyValueLabel(const PropertyName &, | 
| Chris@0 | 62 					  int value) const; | 
| Chris@167 | 63     virtual RangeMapper *getNewPropertyRangeMapper(const PropertyName &) const; | 
| Chris@0 | 64     virtual void setProperty(const PropertyName &, int value); | 
| Chris@0 | 65 | 
| Chris@0 | 66     /** | 
| Chris@0 | 67      * Set the gain multiplier for sample values in this view. | 
| Chris@0 | 68      * | 
| Chris@0 | 69      * The default is 1.0. | 
| Chris@0 | 70      */ | 
| Chris@0 | 71     void setGain(float gain); | 
| Chris@0 | 72     float getGain() const { return m_gain; } | 
| Chris@0 | 73 | 
| Chris@0 | 74     /** | 
| Chris@67 | 75      * Toggle automatic normalization of the currently visible waveform. | 
| Chris@67 | 76      */ | 
| Chris@67 | 77     void setAutoNormalize(bool); | 
| Chris@67 | 78     bool getAutoNormalize() const { return m_autoNormalize; } | 
| Chris@67 | 79 | 
| Chris@67 | 80     /** | 
| Chris@0 | 81      * Set whether to display mean values as a lighter-coloured area | 
| Chris@0 | 82      * beneath the peaks.  Rendering will be slightly faster without | 
| Chris@0 | 83      * but arguably prettier with. | 
| Chris@0 | 84      * | 
| Chris@0 | 85      * The default is to display means. | 
| Chris@0 | 86      */ | 
| Chris@0 | 87     void setShowMeans(bool); | 
| Chris@0 | 88     bool getShowMeans() const { return m_showMeans; } | 
| Chris@0 | 89 | 
| Chris@0 | 90     /** | 
| Chris@0 | 91      * Set whether to use shades of grey (or of the base colour) to | 
| Chris@0 | 92      * provide additional perceived vertical resolution (i.e. using | 
| Chris@0 | 93      * half-filled pixels to represent levels that only just meet the | 
| Chris@0 | 94      * pixel unit boundary).  This provides a small improvement in | 
| Chris@0 | 95      * waveform quality at a small cost in rendering speed. | 
| Chris@0 | 96      * | 
| Chris@0 | 97      * The default is to use greyscale. | 
| Chris@0 | 98      */ | 
| Chris@0 | 99     void setUseGreyscale(bool); | 
| Chris@0 | 100     bool getUseGreyscale() const { return m_greyscale; } | 
| Chris@0 | 101 | 
| Chris@0 | 102 | 
| Chris@67 | 103     enum ChannelMode { SeparateChannels, MixChannels, MergeChannels }; | 
| Chris@0 | 104 | 
| Chris@0 | 105     /** | 
| Chris@0 | 106      * Specify whether multi-channel audio data should be displayed | 
| Chris@67 | 107      * with a separate axis per channel (SeparateChannels), with a | 
| Chris@0 | 108      * single synthetic axis showing channel 0 above the axis and | 
| Chris@67 | 109      * channel 1 below (MergeChannels), or with a single axis showing | 
| Chris@67 | 110      * the average of the channels (MixChannels). | 
| Chris@0 | 111      * | 
| Chris@0 | 112      * MergeChannels does not work for files with more than 2 | 
| Chris@0 | 113      * channels. | 
| Chris@0 | 114      * | 
| Chris@0 | 115      * The default is SeparateChannels. | 
| Chris@0 | 116      */ | 
| Chris@0 | 117     void setChannelMode(ChannelMode); | 
| Chris@0 | 118     ChannelMode getChannelMode() const { return m_channelMode; } | 
| Chris@0 | 119 | 
| Chris@0 | 120 | 
| Chris@0 | 121     /** | 
| Chris@0 | 122      * Specify the channel to use from the source model.  A value of | 
| Chris@0 | 123      * -1 means to show all available channels (laid out to the | 
| Chris@0 | 124      * channel mode). The default is -1. | 
| Chris@0 | 125      */ | 
| Chris@0 | 126     void setChannel(int); | 
| Chris@0 | 127     int getChannel() const { return m_channel; } | 
| Chris@0 | 128 | 
| Chris@0 | 129 | 
| Chris@0 | 130     enum Scale { LinearScale, MeterScale, dBScale }; | 
| Chris@0 | 131 | 
| Chris@0 | 132     /** | 
| Chris@0 | 133      * Specify the vertical scale for sample levels.  With LinearScale, | 
| Chris@0 | 134      * the scale is directly proportional to the raw [-1, +1) | 
| Chris@0 | 135      * floating-point audio sample values.  With dBScale the | 
| Chris@0 | 136      * vertical scale is proportional to dB level (truncated at | 
| Chris@0 | 137      * -50dB).  MeterScale provides a hybrid variable scale based on | 
| Chris@0 | 138      * IEC meter scale, intended to provide a clear overview at | 
| Chris@0 | 139      * relatively small heights. | 
| Chris@0 | 140      * | 
| Chris@0 | 141      * Note that the effective gain (see setGain()) is applied before | 
| Chris@0 | 142      * vertical scaling. | 
| Chris@0 | 143      * | 
| Chris@0 | 144      * The default is LinearScale. | 
| Chris@0 | 145      */ | 
| Chris@0 | 146     void setScale(Scale); | 
| Chris@0 | 147     Scale getScale() const { return m_scale; } | 
| Chris@0 | 148 | 
| Chris@0 | 149     /** | 
| Chris@0 | 150      * Enable or disable aggressive pixmap cacheing.  If enabled, | 
| Chris@0 | 151      * waveforms will be rendered to an off-screen pixmap and | 
| Chris@0 | 152      * refreshed from there instead of being redrawn from the peak | 
| Chris@0 | 153      * data each time.  This may be faster if the data and zoom level | 
| Chris@0 | 154      * do not change often, but it may be slower for frequently zoomed | 
| Chris@0 | 155      * data and it will only work if the waveform is the "bottom" | 
| Chris@0 | 156      * layer on the displayed widget, as each refresh will erase | 
| Chris@0 | 157      * anything beneath the waveform. | 
| Chris@0 | 158      * | 
| Chris@0 | 159      * This is intended specifically for a panner widget display in | 
| Chris@0 | 160      * which the waveform never moves, zooms, or changes, but some | 
| Chris@0 | 161      * graphic such as a panner outline is frequently redrawn over the | 
| Chris@0 | 162      * waveform.  This situation would necessitate a lot of waveform | 
| Chris@0 | 163      * refresh if the default cacheing strategy was used. | 
| Chris@0 | 164      * | 
| Chris@0 | 165      * The default is not to use aggressive cacheing. | 
| Chris@0 | 166      */ | 
| Chris@0 | 167     void setAggressiveCacheing(bool); | 
| Chris@0 | 168     bool getAggressiveCacheing() const { return m_aggressive; } | 
| Chris@0 | 169 | 
| Chris@67 | 170     virtual bool isLayerScrollable(const View *) const; | 
| Chris@67 | 171 | 
| Chris@115 | 172     virtual int getCompletion(View *) const; | 
| Chris@0 | 173 | 
| Chris@101 | 174     virtual bool getValueExtents(float &min, float &max, | 
| Chris@101 | 175                                  bool &log, QString &unit) const; | 
| Chris@79 | 176 | 
| Chris@267 | 177     virtual bool getYScaleValue(const View *v, int y, | 
| Chris@261 | 178                                 float &value, QString &unit) const; | 
| Chris@274 | 179 | 
| Chris@274 | 180     virtual bool getYScaleDifference(const View *v, int y0, int y1, | 
| Chris@274 | 181                                      float &diff, QString &unit) const; | 
| Chris@261 | 182 | 
| Chris@316 | 183     virtual void toXml(QTextStream &stream, QString indent = "", | 
| Chris@316 | 184                        QString extraAttributes = "") const; | 
| Chris@6 | 185 | 
| Chris@287 | 186     virtual void setProperties(const QXmlAttributes &attributes); | 
| Chris@11 | 187 | 
| Chris@133 | 188     virtual int getVerticalZoomSteps(int &defaultStep) const; | 
| Chris@133 | 189     virtual int getCurrentVerticalZoomStep() const; | 
| Chris@133 | 190     virtual void setVerticalZoomStep(int); | 
| Chris@133 | 191 | 
| Chris@0 | 192 protected: | 
| Chris@0 | 193     int dBscale(float sample, int m) const; | 
| Chris@0 | 194 | 
| Chris@0 | 195     const RangeSummarisableTimeValueModel *m_model; // I do not own this | 
| Chris@0 | 196 | 
| Chris@0 | 197     /// Return value is number of channels displayed | 
| Chris@67 | 198     size_t getChannelArrangement(size_t &min, size_t &max, | 
| Chris@67 | 199                                  bool &merging, bool &mixing) const; | 
| Chris@0 | 200 | 
| Chris@274 | 201     int getYForValue(const View *v, float value, size_t channel) const; | 
| Chris@68 | 202 | 
| Chris@274 | 203     float getValueForY(const View *v, int y, size_t &channel) const; | 
| Chris@261 | 204 | 
| Chris@365 | 205     bool getSourceFramesForX(View *v, int x, size_t modelZoomLevel, | 
| Chris@365 | 206                              size_t &f0, size_t &f1) const; | 
| Chris@365 | 207 | 
| Chris@365 | 208     float getNormalizeGain(View *v, int channel) const; | 
| Chris@365 | 209 | 
| Chris@287 | 210     virtual void flagBaseColourChanged() { m_cacheValid = false; } | 
| Chris@287 | 211 | 
| Chris@6 | 212     float        m_gain; | 
| Chris@67 | 213     bool         m_autoNormalize; | 
| Chris@6 | 214     bool         m_showMeans; | 
| Chris@6 | 215     bool         m_greyscale; | 
| Chris@6 | 216     ChannelMode  m_channelMode; | 
| Chris@6 | 217     int          m_channel; | 
| Chris@6 | 218     Scale        m_scale; | 
| Chris@6 | 219     bool         m_aggressive; | 
| Chris@0 | 220 | 
| Chris@67 | 221     mutable std::vector<float> m_effectiveGains; | 
| Chris@67 | 222 | 
| Chris@0 | 223     mutable QPixmap *m_cache; | 
| Chris@0 | 224     mutable bool m_cacheValid; | 
| Chris@0 | 225     mutable int m_cacheZoomLevel; | 
| Chris@0 | 226 }; | 
| Chris@0 | 227 | 
| Chris@0 | 228 #endif |