annotate layer/SpectrogramLayer.h @ 36:c28ebb4ba4de

* Improvements to text layer editing, and implement file I/O for it * Start some fixes to spectrogram frequency computation
author Chris Cannam
date Mon, 20 Feb 2006 17:23:40 +0000
parents 10ba9276a315
children 21d061e66177
rev   line source
Chris@0 1 /* -*- c-basic-offset: 4 -*- vi:set ts=8 sts=4 sw=4: */
Chris@0 2
Chris@0 3 /*
Chris@0 4 A waveform viewer and audio annotation editor.
Chris@5 5 Chris Cannam, Queen Mary University of London, 2005-2006
Chris@0 6
Chris@0 7 This is experimental software. Not for distribution.
Chris@0 8 */
Chris@0 9
Chris@30 10 #ifndef _SPECTROGRAM_LAYER_H_
Chris@30 11 #define _SPECTROGRAM_LAYER_H_
Chris@0 12
Chris@0 13 #include "base/Layer.h"
Chris@0 14 #include "base/Window.h"
Chris@0 15 #include "model/PowerOfSqrtTwoZoomConstraint.h"
Chris@0 16 #include "model/DenseTimeValueModel.h"
Chris@0 17
Chris@0 18 #include <QThread>
Chris@0 19 #include <QMutex>
Chris@0 20 #include <QWaitCondition>
Chris@0 21
Chris@0 22 #include <fftw3.h>
Chris@0 23
Chris@0 24 class View;
Chris@0 25 class QPainter;
Chris@0 26 class QImage;
Chris@0 27 class QPixmap;
Chris@0 28 class QTimer;
Chris@0 29 class RealTime;
Chris@0 30
Chris@0 31 /**
Chris@0 32 * SpectrogramLayer represents waveform data (obtained from a
Chris@0 33 * DenseTimeValueModel) in spectrogram form.
Chris@0 34 */
Chris@0 35
Chris@0 36 class SpectrogramLayer : public Layer,
Chris@31 37 public PowerOfSqrtTwoZoomConstraint
Chris@0 38 {
Chris@0 39 Q_OBJECT
Chris@0 40
Chris@0 41 public:
Chris@0 42 enum Configuration { FullRangeDb, MelodicRange };
Chris@0 43
Chris@0 44 SpectrogramLayer(View *w, Configuration = FullRangeDb);
Chris@0 45 ~SpectrogramLayer();
Chris@0 46
Chris@0 47 virtual const ZoomConstraint *getZoomConstraint() const { return this; }
Chris@0 48 virtual const Model *getModel() const { return m_model; }
Chris@0 49 virtual void paint(QPainter &paint, QRect rect) const;
Chris@0 50
Chris@0 51 virtual int getVerticalScaleWidth(QPainter &) const;
Chris@0 52 virtual void paintVerticalScale(QPainter &paint, QRect rect) const;
Chris@0 53
Chris@25 54 virtual QString getFeatureDescription(QPoint &) const;
Chris@0 55
Chris@28 56 virtual bool snapToFeatureFrame(int &frame,
Chris@28 57 size_t &resolution,
Chris@28 58 SnapType snap) const;
Chris@13 59
Chris@0 60 void setModel(const DenseTimeValueModel *model);
Chris@0 61
Chris@0 62 virtual PropertyList getProperties() const;
Chris@0 63 virtual PropertyType getPropertyType(const PropertyName &) const;
Chris@0 64 virtual QString getPropertyGroupName(const PropertyName &) const;
Chris@0 65 virtual int getPropertyRangeAndValue(const PropertyName &,
Chris@0 66 int *min, int *max) const;
Chris@0 67 virtual QString getPropertyValueLabel(const PropertyName &,
Chris@0 68 int value) const;
Chris@0 69 virtual void setProperty(const PropertyName &, int value);
Chris@0 70
Chris@0 71 /**
Chris@0 72 * Specify the channel to use from the source model.
Chris@0 73 * A value of -1 means to mix all available channels.
Chris@0 74 * The default is channel 0.
Chris@0 75 */
Chris@0 76 void setChannel(int);
Chris@0 77 int getChannel() const;
Chris@0 78
Chris@0 79 void setWindowSize(size_t);
Chris@0 80 size_t getWindowSize() const;
Chris@0 81
Chris@0 82 void setWindowOverlap(size_t percent);
Chris@0 83 size_t getWindowOverlap() const;
Chris@0 84
Chris@0 85 void setWindowType(WindowType type);
Chris@0 86 WindowType getWindowType() const;
Chris@0 87
Chris@0 88 /**
Chris@0 89 * Set the gain multiplier for sample values in this view prior to
Chris@0 90 * FFT calculation.
Chris@0 91 *
Chris@0 92 * The default is 1.0.
Chris@0 93 */
Chris@0 94 void setGain(float gain);
Chris@0 95 float getGain() const;
Chris@0 96
Chris@0 97 void setMaxFrequency(size_t); // 0 -> no maximum
Chris@0 98 size_t getMaxFrequency() const;
Chris@0 99
Chris@0 100 enum ColourScale { LinearColourScale, MeterColourScale, dBColourScale,
Chris@0 101 PhaseColourScale };
Chris@0 102
Chris@0 103 /**
Chris@0 104 * Specify the scale for sample levels. See WaveformLayer for
Chris@0 105 * details of meter and dB scaling. The default is dBColourScale.
Chris@0 106 */
Chris@0 107 void setColourScale(ColourScale);
Chris@0 108 ColourScale getColourScale() const;
Chris@0 109
Chris@35 110 enum FrequencyScale {
Chris@35 111 LinearFrequencyScale,
Chris@35 112 LogFrequencyScale
Chris@35 113 };
Chris@0 114
Chris@0 115 /**
Chris@0 116 * Specify the scale for the y axis.
Chris@0 117 */
Chris@0 118 void setFrequencyScale(FrequencyScale);
Chris@0 119 FrequencyScale getFrequencyScale() const;
Chris@0 120
Chris@35 121 enum FrequencyAdjustment {
Chris@35 122 RawFrequency,
Chris@35 123 PhaseAdjustedFrequency,
Chris@35 124 PhaseAdjustedPeaks
Chris@35 125 };
Chris@35 126
Chris@35 127 /**
Chris@35 128 * Specify the processing of frequency bins for the y axis.
Chris@35 129 */
Chris@35 130 void setFrequencyAdjustment(FrequencyAdjustment);
Chris@35 131 FrequencyAdjustment getFrequencyAdjustment() const;
Chris@35 132
Chris@36 133 void setNormalizeColumns(bool n);
Chris@36 134 bool getNormalizeColumns() const;
Chris@36 135
Chris@0 136 enum ColourScheme { DefaultColours, WhiteOnBlack, BlackOnWhite,
Chris@0 137 RedOnBlue, YellowOnBlack, RedOnBlack };
Chris@0 138
Chris@0 139 void setColourScheme(ColourScheme scheme);
Chris@0 140 ColourScheme getColourScheme() const;
Chris@0 141
Chris@9 142 /**
Chris@9 143 * Specify the colourmap rotation for the colour scale.
Chris@9 144 */
Chris@9 145 void setColourRotation(int);
Chris@9 146 int getColourRotation() const;
Chris@9 147
Chris@0 148 virtual VerticalPosition getPreferredFrameCountPosition() const {
Chris@0 149 return PositionTop;
Chris@0 150 }
Chris@0 151
Chris@15 152 virtual bool isLayerOpaque() const { return true; }
Chris@15 153
Chris@0 154 virtual int getCompletion() const;
Chris@0 155
Chris@6 156 virtual QString toXmlString(QString indent = "",
Chris@6 157 QString extraAttributes = "") const;
Chris@6 158
Chris@11 159 void setProperties(const QXmlAttributes &attributes);
Chris@11 160
Chris@33 161 virtual void setLayerDormant(bool dormant);
Chris@29 162
Chris@0 163 protected slots:
Chris@0 164 void cacheInvalid();
Chris@0 165 void cacheInvalid(size_t startFrame, size_t endFrame);
Chris@0 166
Chris@0 167 void fillTimerTimedOut();
Chris@0 168
Chris@0 169 protected:
Chris@0 170 const DenseTimeValueModel *m_model; // I do not own this
Chris@0 171
Chris@35 172 int m_channel;
Chris@35 173 size_t m_windowSize;
Chris@35 174 WindowType m_windowType;
Chris@35 175 size_t m_windowOverlap;
Chris@35 176 float m_gain;
Chris@35 177 int m_colourRotation;
Chris@35 178 size_t m_maxFrequency;
Chris@35 179 ColourScale m_colourScale;
Chris@35 180 ColourScheme m_colourScheme;
Chris@35 181 FrequencyScale m_frequencyScale;
Chris@35 182 FrequencyAdjustment m_frequencyAdjustment;
Chris@36 183 bool m_normalizeColumns;
Chris@0 184
Chris@31 185 // A QImage would do just as well here, and we originally used
Chris@31 186 // one: the problem is that we want to munlock() the memory it
Chris@31 187 // uses, and it's much easier to do that if we control it. This
Chris@31 188 // cache is hardwired to an effective 8-bit colour mapped layout.
Chris@31 189 class Cache {
Chris@31 190 public:
Chris@31 191 Cache(size_t width, size_t height);
Chris@31 192 ~Cache();
Chris@31 193
Chris@31 194 size_t getWidth() const;
Chris@31 195 size_t getHeight() const;
Chris@35 196
Chris@35 197 void resize(size_t width, size_t height);
Chris@31 198
Chris@31 199 unsigned char getValueAt(size_t x, size_t y) const;
Chris@31 200 void setValueAt(size_t x, size_t y, unsigned char value);
Chris@31 201
Chris@31 202 QColor getColour(unsigned char index) const;
Chris@31 203 void setColour(unsigned char index, QColor colour);
Chris@31 204
Chris@31 205 void fill(unsigned char value);
Chris@31 206
Chris@31 207 protected:
Chris@31 208 size_t m_width;
Chris@31 209 size_t m_height;
Chris@31 210 unsigned char *m_values;
Chris@31 211 QColor m_colours[256];
Chris@31 212 };
Chris@31 213
Chris@31 214 Cache *m_cache;
Chris@35 215 Cache *m_phaseAdjustCache;
Chris@31 216 bool m_cacheInvalid;
Chris@31 217
Chris@0 218 class CacheFillThread : public QThread
Chris@0 219 {
Chris@0 220 public:
Chris@0 221 CacheFillThread(SpectrogramLayer &layer) :
Chris@0 222 m_layer(layer), m_fillExtent(0) { }
Chris@0 223
Chris@0 224 size_t getFillExtent() const { return m_fillExtent; }
Chris@0 225 size_t getFillCompletion() const { return m_fillCompletion; }
Chris@0 226 virtual void run();
Chris@0 227
Chris@0 228 protected:
Chris@0 229 SpectrogramLayer &m_layer;
Chris@0 230 size_t m_fillExtent;
Chris@0 231 size_t m_fillCompletion;
Chris@0 232 };
Chris@0 233
Chris@0 234 void fillCache();
Chris@0 235
Chris@0 236 mutable QPixmap *m_pixmapCache;
Chris@0 237 mutable bool m_pixmapCacheInvalid;
Chris@0 238 mutable long m_pixmapCacheStartFrame;
Chris@0 239 mutable size_t m_pixmapCacheZoomLevel;
Chris@0 240
Chris@0 241 QWaitCondition m_condition;
Chris@0 242 mutable QMutex m_mutex;
Chris@0 243
Chris@0 244 CacheFillThread *m_fillThread;
Chris@0 245 QTimer *m_updateTimer;
Chris@0 246 size_t m_lastFillExtent;
Chris@0 247 bool m_cachedInitialVisibleArea;
Chris@0 248 bool m_exiting;
Chris@0 249
Chris@0 250 void setCacheColourmap();
Chris@9 251 void rotateCacheColourmap(int distance);
Chris@0 252
Chris@0 253 bool fillCacheColumn(int column,
Chris@0 254 double *inputBuffer,
Chris@0 255 fftw_complex *outputBuffer,
Chris@0 256 fftw_plan plan,
Chris@9 257 size_t windowSize,
Chris@9 258 size_t windowIncrement,
Chris@0 259 const Window<double> &windower,
Chris@35 260 bool resetStoredPhase)
Chris@0 261 const;
Chris@0 262
Chris@0 263 bool getYBinRange(int y, float &freqBinMin, float &freqBinMax) const;
Chris@0 264
Chris@0 265 struct LayerRange {
Chris@0 266 long startFrame;
Chris@0 267 int zoomLevel;
Chris@0 268 size_t modelStart;
Chris@0 269 size_t modelEnd;
Chris@0 270 };
Chris@20 271 bool getXBinRange(int x, float &windowMin, float &windowMax) const;
Chris@0 272
Chris@0 273 bool getYBinSourceRange(int y, float &freqMin, float &freqMax) const;
Chris@35 274 bool getAdjustedYBinSourceRange(int x, int y,
Chris@35 275 float &freqMin, float &freqMax,
Chris@35 276 float &adjFreqMin, float &adjFreqMax) const;
Chris@0 277 bool getXBinSourceRange(int x, RealTime &timeMin, RealTime &timeMax) const;
Chris@0 278 bool getXYBinSourceRange(int x, int y, float &dbMin, float &dbMax) const;
Chris@0 279
Chris@0 280 size_t getWindowIncrement() const {
Chris@0 281 return m_windowSize - m_windowSize * m_windowOverlap / 100;
Chris@0 282 }
Chris@0 283 };
Chris@0 284
Chris@0 285 #endif