annotate layer/LayerFactory.cpp @ 0:fc9323a41f5a

start base : Sonic Visualiser sv1-1.0rc1
author lbajardsilogic
date Fri, 11 May 2007 09:08:14 +0000
parents
children d8e6709e9075
rev   line source
lbajardsilogic@0 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
lbajardsilogic@0 2
lbajardsilogic@0 3 /*
lbajardsilogic@0 4 Sonic Visualiser
lbajardsilogic@0 5 An audio file viewer and annotation editor.
lbajardsilogic@0 6 Centre for Digital Music, Queen Mary, University of London.
lbajardsilogic@0 7 This file copyright 2006 Chris Cannam.
lbajardsilogic@0 8
lbajardsilogic@0 9 This program is free software; you can redistribute it and/or
lbajardsilogic@0 10 modify it under the terms of the GNU General Public License as
lbajardsilogic@0 11 published by the Free Software Foundation; either version 2 of the
lbajardsilogic@0 12 License, or (at your option) any later version. See the file
lbajardsilogic@0 13 COPYING included with this distribution for more information.
lbajardsilogic@0 14 */
lbajardsilogic@0 15
lbajardsilogic@0 16 #include "LayerFactory.h"
lbajardsilogic@0 17
lbajardsilogic@0 18 #include "WaveformLayer.h"
lbajardsilogic@0 19 #include "SpectrogramLayer.h"
lbajardsilogic@0 20 #include "TimeRulerLayer.h"
lbajardsilogic@0 21 #include "TimeInstantLayer.h"
lbajardsilogic@0 22 #include "TimeValueLayer.h"
lbajardsilogic@0 23 #include "NoteLayer.h"
lbajardsilogic@0 24 #include "TextLayer.h"
lbajardsilogic@0 25 #include "Colour3DPlotLayer.h"
lbajardsilogic@0 26 #include "SpectrumLayer.h"
lbajardsilogic@0 27 #include "SliceLayer.h"
lbajardsilogic@0 28 #include "SliceableLayer.h"
lbajardsilogic@0 29
lbajardsilogic@0 30 #include "data/model/RangeSummarisableTimeValueModel.h"
lbajardsilogic@0 31 #include "data/model/DenseTimeValueModel.h"
lbajardsilogic@0 32 #include "data/model/SparseOneDimensionalModel.h"
lbajardsilogic@0 33 #include "data/model/SparseTimeValueModel.h"
lbajardsilogic@0 34 #include "data/model/NoteModel.h"
lbajardsilogic@0 35 #include "data/model/TextModel.h"
lbajardsilogic@0 36 #include "data/model/DenseThreeDimensionalModel.h"
lbajardsilogic@0 37 #include "data/model/WaveFileModel.h"
lbajardsilogic@0 38 #include "data/model/WritableWaveFileModel.h"
lbajardsilogic@0 39
lbajardsilogic@0 40 LayerFactory *
lbajardsilogic@0 41 LayerFactory::m_instance = new LayerFactory;
lbajardsilogic@0 42
lbajardsilogic@0 43 LayerFactory *
lbajardsilogic@0 44 LayerFactory::getInstance()
lbajardsilogic@0 45 {
lbajardsilogic@0 46 return m_instance;
lbajardsilogic@0 47 }
lbajardsilogic@0 48
lbajardsilogic@0 49 LayerFactory::~LayerFactory()
lbajardsilogic@0 50 {
lbajardsilogic@0 51 }
lbajardsilogic@0 52
lbajardsilogic@0 53 QString
lbajardsilogic@0 54 LayerFactory::getLayerPresentationName(LayerType type)
lbajardsilogic@0 55 {
lbajardsilogic@0 56 switch (type) {
lbajardsilogic@0 57 case Waveform: return Layer::tr("Waveform");
lbajardsilogic@0 58 case Spectrogram: return Layer::tr("Spectrogram");
lbajardsilogic@0 59 case TimeRuler: return Layer::tr("Ruler");
lbajardsilogic@0 60 case TimeInstants: return Layer::tr("Time Instants");
lbajardsilogic@0 61 case TimeValues: return Layer::tr("Time Values");
lbajardsilogic@0 62 case Notes: return Layer::tr("Notes");
lbajardsilogic@0 63 case Text: return Layer::tr("Text");
lbajardsilogic@0 64 case Colour3DPlot: return Layer::tr("Colour 3D Plot");
lbajardsilogic@0 65 case Spectrum: return Layer::tr("Spectrum");
lbajardsilogic@0 66 case Slice: return Layer::tr("Time Slice");
lbajardsilogic@0 67
lbajardsilogic@0 68 case MelodicRangeSpectrogram:
lbajardsilogic@0 69 // The user can change all the parameters of this after the
lbajardsilogic@0 70 // fact -- there's nothing permanently melodic-range about it
lbajardsilogic@0 71 // that should be encoded in its name
lbajardsilogic@0 72 return Layer::tr("Spectrogram");
lbajardsilogic@0 73
lbajardsilogic@0 74 case PeakFrequencySpectrogram:
lbajardsilogic@0 75 // likewise
lbajardsilogic@0 76 return Layer::tr("Spectrogram");
lbajardsilogic@0 77
lbajardsilogic@0 78 default: break;
lbajardsilogic@0 79 }
lbajardsilogic@0 80
lbajardsilogic@0 81 return Layer::tr("Layer");
lbajardsilogic@0 82 }
lbajardsilogic@0 83
lbajardsilogic@0 84 bool
lbajardsilogic@0 85 LayerFactory::isLayerSliceable(const Layer *layer)
lbajardsilogic@0 86 {
lbajardsilogic@0 87 if (dynamic_cast<const SliceableLayer *>(layer)) {
lbajardsilogic@0 88 if (dynamic_cast<const SpectrogramLayer *>(layer)) {
lbajardsilogic@0 89
lbajardsilogic@0 90 //!!! We can create slices of spectrograms, but there's a
lbajardsilogic@0 91 // problem managing the models. The source model for the
lbajardsilogic@0 92 // slice layer has to be one of the spectrogram's FFT
lbajardsilogic@0 93 // models -- that's fine, except that we can't store &
lbajardsilogic@0 94 // recall the slice layer with a reference to that model
lbajardsilogic@0 95 // because the model is internal to the spectrogram layer
lbajardsilogic@0 96 // and the document has no record of it. We would need
lbajardsilogic@0 97 // some other way of managing models that are used in this
lbajardsilogic@0 98 // way. For the moment we just don't allow slices of
lbajardsilogic@0 99 // spectrograms -- and provide a spectrum layer for this
lbajardsilogic@0 100 // instead.
lbajardsilogic@0 101 //
lbajardsilogic@0 102 // This business needs a bit more thought -- either come
lbajardsilogic@0 103 // up with a sensible way to deal with that stuff, or
lbajardsilogic@0 104 // simplify the existing slice layer logic so that it
lbajardsilogic@0 105 // doesn't have to deal with models disappearing on it at
lbajardsilogic@0 106 // all (and use the normal Document setModel mechanism to
lbajardsilogic@0 107 // set its sliceable model instead of the fancy pants
lbajardsilogic@0 108 // nonsense it's doing at the moment).
lbajardsilogic@0 109
lbajardsilogic@0 110 return false;
lbajardsilogic@0 111 }
lbajardsilogic@0 112 return true;
lbajardsilogic@0 113 }
lbajardsilogic@0 114 return false;
lbajardsilogic@0 115 }
lbajardsilogic@0 116
lbajardsilogic@0 117 LayerFactory::LayerTypeSet
lbajardsilogic@0 118 LayerFactory::getValidLayerTypes(Model *model)
lbajardsilogic@0 119 {
lbajardsilogic@0 120 LayerTypeSet types;
lbajardsilogic@0 121
lbajardsilogic@0 122 if (dynamic_cast<DenseThreeDimensionalModel *>(model)) {
lbajardsilogic@0 123 types.insert(Colour3DPlot);
lbajardsilogic@0 124 types.insert(Slice);
lbajardsilogic@0 125 }
lbajardsilogic@0 126
lbajardsilogic@0 127 if (dynamic_cast<RangeSummarisableTimeValueModel *>(model)) {
lbajardsilogic@0 128 types.insert(Waveform);
lbajardsilogic@0 129 }
lbajardsilogic@0 130
lbajardsilogic@0 131 if (dynamic_cast<DenseTimeValueModel *>(model)) {
lbajardsilogic@0 132 types.insert(Spectrogram);
lbajardsilogic@0 133 types.insert(MelodicRangeSpectrogram);
lbajardsilogic@0 134 types.insert(PeakFrequencySpectrogram);
lbajardsilogic@0 135 }
lbajardsilogic@0 136
lbajardsilogic@0 137 if (dynamic_cast<SparseOneDimensionalModel *>(model)) {
lbajardsilogic@0 138 types.insert(TimeInstants);
lbajardsilogic@0 139 }
lbajardsilogic@0 140
lbajardsilogic@0 141 if (dynamic_cast<SparseTimeValueModel *>(model)) {
lbajardsilogic@0 142 types.insert(TimeValues);
lbajardsilogic@0 143
lbajardsilogic@0 144 }
lbajardsilogic@0 145 if (dynamic_cast<NoteModel *>(model)) {
lbajardsilogic@0 146 types.insert(Notes);
lbajardsilogic@0 147 }
lbajardsilogic@0 148
lbajardsilogic@0 149 if (dynamic_cast<TextModel *>(model)) {
lbajardsilogic@0 150 types.insert(Text);
lbajardsilogic@0 151 }
lbajardsilogic@0 152
lbajardsilogic@0 153 if (dynamic_cast<DenseTimeValueModel *>(model)) {
lbajardsilogic@0 154 types.insert(Spectrum);
lbajardsilogic@0 155 }
lbajardsilogic@0 156
lbajardsilogic@0 157 // We don't count TimeRuler here as it doesn't actually display
lbajardsilogic@0 158 // the data, although it can be backed by any model
lbajardsilogic@0 159
lbajardsilogic@0 160 return types;
lbajardsilogic@0 161 }
lbajardsilogic@0 162
lbajardsilogic@0 163 LayerFactory::LayerTypeSet
lbajardsilogic@0 164 LayerFactory::getValidEmptyLayerTypes()
lbajardsilogic@0 165 {
lbajardsilogic@0 166 LayerTypeSet types;
lbajardsilogic@0 167 types.insert(TimeInstants);
lbajardsilogic@0 168 types.insert(TimeValues);
lbajardsilogic@0 169 types.insert(Notes);
lbajardsilogic@0 170 types.insert(Text);
lbajardsilogic@0 171 //!!! and in principle Colour3DPlot -- now that's a challenge
lbajardsilogic@0 172 return types;
lbajardsilogic@0 173 }
lbajardsilogic@0 174
lbajardsilogic@0 175 LayerFactory::LayerType
lbajardsilogic@0 176 LayerFactory::getLayerType(const Layer *layer)
lbajardsilogic@0 177 {
lbajardsilogic@0 178 if (dynamic_cast<const WaveformLayer *>(layer)) return Waveform;
lbajardsilogic@0 179 if (dynamic_cast<const SpectrogramLayer *>(layer)) return Spectrogram;
lbajardsilogic@0 180 if (dynamic_cast<const TimeRulerLayer *>(layer)) return TimeRuler;
lbajardsilogic@0 181 if (dynamic_cast<const TimeInstantLayer *>(layer)) return TimeInstants;
lbajardsilogic@0 182 if (dynamic_cast<const TimeValueLayer *>(layer)) return TimeValues;
lbajardsilogic@0 183 if (dynamic_cast<const NoteLayer *>(layer)) return Notes;
lbajardsilogic@0 184 if (dynamic_cast<const TextLayer *>(layer)) return Text;
lbajardsilogic@0 185 if (dynamic_cast<const Colour3DPlotLayer *>(layer)) return Colour3DPlot;
lbajardsilogic@0 186 if (dynamic_cast<const SpectrumLayer *>(layer)) return Spectrum;
lbajardsilogic@0 187 if (dynamic_cast<const SliceLayer *>(layer)) return Slice;
lbajardsilogic@0 188 return UnknownLayer;
lbajardsilogic@0 189 }
lbajardsilogic@0 190
lbajardsilogic@0 191 QString
lbajardsilogic@0 192 LayerFactory::getLayerIconName(LayerType type)
lbajardsilogic@0 193 {
lbajardsilogic@0 194 switch (type) {
lbajardsilogic@0 195 case Waveform: return "waveform";
lbajardsilogic@0 196 case Spectrogram: return "spectrogram";
lbajardsilogic@0 197 case TimeRuler: return "timeruler";
lbajardsilogic@0 198 case TimeInstants: return "instants";
lbajardsilogic@0 199 case TimeValues: return "values";
lbajardsilogic@0 200 case Notes: return "notes";
lbajardsilogic@0 201 case Text: return "text";
lbajardsilogic@0 202 case Colour3DPlot: return "colour3d";
lbajardsilogic@0 203 case Spectrum: return "spectrum";
lbajardsilogic@0 204 case Slice: return "spectrum";
lbajardsilogic@0 205 default: return "unknown";
lbajardsilogic@0 206 }
lbajardsilogic@0 207 }
lbajardsilogic@0 208
lbajardsilogic@0 209 QString
lbajardsilogic@0 210 LayerFactory::getLayerTypeName(LayerType type)
lbajardsilogic@0 211 {
lbajardsilogic@0 212 switch (type) {
lbajardsilogic@0 213 case Waveform: return "waveform";
lbajardsilogic@0 214 case Spectrogram: return "spectrogram";
lbajardsilogic@0 215 case TimeRuler: return "timeruler";
lbajardsilogic@0 216 case TimeInstants: return "timeinstants";
lbajardsilogic@0 217 case TimeValues: return "timevalues";
lbajardsilogic@0 218 case Notes: return "notes";
lbajardsilogic@0 219 case Text: return "text";
lbajardsilogic@0 220 case Colour3DPlot: return "colour3dplot";
lbajardsilogic@0 221 case Spectrum: return "spectrum";
lbajardsilogic@0 222 case Slice: return "slice";
lbajardsilogic@0 223 default: return "unknown";
lbajardsilogic@0 224 }
lbajardsilogic@0 225 }
lbajardsilogic@0 226
lbajardsilogic@0 227 LayerFactory::LayerType
lbajardsilogic@0 228 LayerFactory::getLayerTypeForName(QString name)
lbajardsilogic@0 229 {
lbajardsilogic@0 230 if (name == "waveform") return Waveform;
lbajardsilogic@0 231 if (name == "spectrogram") return Spectrogram;
lbajardsilogic@0 232 if (name == "timeruler") return TimeRuler;
lbajardsilogic@0 233 if (name == "timeinstants") return TimeInstants;
lbajardsilogic@0 234 if (name == "timevalues") return TimeValues;
lbajardsilogic@0 235 if (name == "notes") return Notes;
lbajardsilogic@0 236 if (name == "text") return Text;
lbajardsilogic@0 237 if (name == "colour3dplot") return Colour3DPlot;
lbajardsilogic@0 238 if (name == "spectrum") return Spectrum;
lbajardsilogic@0 239 if (name == "slice") return Slice;
lbajardsilogic@0 240 return UnknownLayer;
lbajardsilogic@0 241 }
lbajardsilogic@0 242
lbajardsilogic@0 243 void
lbajardsilogic@0 244 LayerFactory::setModel(Layer *layer, Model *model)
lbajardsilogic@0 245 {
lbajardsilogic@0 246 // if (trySetModel<WaveformLayer, RangeSummarisableTimeValueModel>(layer, model))
lbajardsilogic@0 247 // return;
lbajardsilogic@0 248
lbajardsilogic@0 249 if (trySetModel<WaveformLayer, WaveFileModel>(layer, model))
lbajardsilogic@0 250 return;
lbajardsilogic@0 251
lbajardsilogic@0 252 if (trySetModel<WaveformLayer, WritableWaveFileModel>(layer, model))
lbajardsilogic@0 253 return;
lbajardsilogic@0 254
lbajardsilogic@0 255 if (trySetModel<SpectrogramLayer, DenseTimeValueModel>(layer, model))
lbajardsilogic@0 256 return;
lbajardsilogic@0 257
lbajardsilogic@0 258 if (trySetModel<TimeRulerLayer, Model>(layer, model))
lbajardsilogic@0 259 return;
lbajardsilogic@0 260
lbajardsilogic@0 261 if (trySetModel<TimeInstantLayer, SparseOneDimensionalModel>(layer, model))
lbajardsilogic@0 262 return;
lbajardsilogic@0 263
lbajardsilogic@0 264 if (trySetModel<TimeValueLayer, SparseTimeValueModel>(layer, model))
lbajardsilogic@0 265 return;
lbajardsilogic@0 266
lbajardsilogic@0 267 if (trySetModel<NoteLayer, NoteModel>(layer, model))
lbajardsilogic@0 268 return;
lbajardsilogic@0 269
lbajardsilogic@0 270 if (trySetModel<TextLayer, TextModel>(layer, model))
lbajardsilogic@0 271 return;
lbajardsilogic@0 272
lbajardsilogic@0 273 if (trySetModel<Colour3DPlotLayer, DenseThreeDimensionalModel>(layer, model))
lbajardsilogic@0 274 return;
lbajardsilogic@0 275
lbajardsilogic@0 276 if (trySetModel<SpectrogramLayer, DenseTimeValueModel>(layer, model))
lbajardsilogic@0 277 return;
lbajardsilogic@0 278
lbajardsilogic@0 279 if (trySetModel<SpectrumLayer, DenseTimeValueModel>(layer, model))
lbajardsilogic@0 280 return;
lbajardsilogic@0 281
lbajardsilogic@0 282 // if (trySetModel<SliceLayer, DenseThreeDimensionalModel>(layer, model))
lbajardsilogic@0 283 // return;
lbajardsilogic@0 284 }
lbajardsilogic@0 285
lbajardsilogic@0 286 Model *
lbajardsilogic@0 287 LayerFactory::createEmptyModel(LayerType layerType, Model *baseModel)
lbajardsilogic@0 288 {
lbajardsilogic@0 289 if (layerType == TimeInstants) {
lbajardsilogic@0 290 return new SparseOneDimensionalModel(baseModel->getSampleRate(), 1);
lbajardsilogic@0 291 } else if (layerType == TimeValues) {
lbajardsilogic@0 292 return new SparseTimeValueModel(baseModel->getSampleRate(), 1, true);
lbajardsilogic@0 293 } else if (layerType == Notes) {
lbajardsilogic@0 294 return new NoteModel(baseModel->getSampleRate(), 1, true);
lbajardsilogic@0 295 } else if (layerType == Text) {
lbajardsilogic@0 296 return new TextModel(baseModel->getSampleRate(), 1, true);
lbajardsilogic@0 297 } else {
lbajardsilogic@0 298 return 0;
lbajardsilogic@0 299 }
lbajardsilogic@0 300 }
lbajardsilogic@0 301
lbajardsilogic@0 302 int
lbajardsilogic@0 303 LayerFactory::getChannel(Layer *layer)
lbajardsilogic@0 304 {
lbajardsilogic@0 305 if (dynamic_cast<WaveformLayer *>(layer)) {
lbajardsilogic@0 306 return dynamic_cast<WaveformLayer *>(layer)->getChannel();
lbajardsilogic@0 307 }
lbajardsilogic@0 308 if (dynamic_cast<SpectrogramLayer *>(layer)) {
lbajardsilogic@0 309 return dynamic_cast<SpectrogramLayer *>(layer)->getChannel();
lbajardsilogic@0 310 }
lbajardsilogic@0 311 return -1;
lbajardsilogic@0 312 }
lbajardsilogic@0 313
lbajardsilogic@0 314 void
lbajardsilogic@0 315 LayerFactory::setChannel(Layer *layer, int channel)
lbajardsilogic@0 316 {
lbajardsilogic@0 317 if (dynamic_cast<WaveformLayer *>(layer)) {
lbajardsilogic@0 318 dynamic_cast<WaveformLayer *>(layer)->setChannel(channel);
lbajardsilogic@0 319 return;
lbajardsilogic@0 320 }
lbajardsilogic@0 321 if (dynamic_cast<SpectrogramLayer *>(layer)) {
lbajardsilogic@0 322 dynamic_cast<SpectrogramLayer *>(layer)->setChannel(channel);
lbajardsilogic@0 323 return;
lbajardsilogic@0 324 }
lbajardsilogic@0 325 }
lbajardsilogic@0 326
lbajardsilogic@0 327 Layer *
lbajardsilogic@0 328 LayerFactory::createLayer(LayerType type)
lbajardsilogic@0 329 {
lbajardsilogic@0 330 Layer *layer = 0;
lbajardsilogic@0 331
lbajardsilogic@0 332 switch (type) {
lbajardsilogic@0 333
lbajardsilogic@0 334 case Waveform:
lbajardsilogic@0 335 layer = new WaveformLayer;
lbajardsilogic@0 336 break;
lbajardsilogic@0 337
lbajardsilogic@0 338 case Spectrogram:
lbajardsilogic@0 339 layer = new SpectrogramLayer;
lbajardsilogic@0 340 break;
lbajardsilogic@0 341
lbajardsilogic@0 342 case TimeRuler:
lbajardsilogic@0 343 layer = new TimeRulerLayer;
lbajardsilogic@0 344 break;
lbajardsilogic@0 345
lbajardsilogic@0 346 case TimeInstants:
lbajardsilogic@0 347 layer = new TimeInstantLayer;
lbajardsilogic@0 348 break;
lbajardsilogic@0 349
lbajardsilogic@0 350 case TimeValues:
lbajardsilogic@0 351 layer = new TimeValueLayer;
lbajardsilogic@0 352 break;
lbajardsilogic@0 353
lbajardsilogic@0 354 case Notes:
lbajardsilogic@0 355 layer = new NoteLayer;
lbajardsilogic@0 356 break;
lbajardsilogic@0 357
lbajardsilogic@0 358 case Text:
lbajardsilogic@0 359 layer = new TextLayer;
lbajardsilogic@0 360 break;
lbajardsilogic@0 361
lbajardsilogic@0 362 case Colour3DPlot:
lbajardsilogic@0 363 layer = new Colour3DPlotLayer;
lbajardsilogic@0 364 break;
lbajardsilogic@0 365
lbajardsilogic@0 366 case Spectrum:
lbajardsilogic@0 367 layer = new SpectrumLayer;
lbajardsilogic@0 368 break;
lbajardsilogic@0 369
lbajardsilogic@0 370 case Slice:
lbajardsilogic@0 371 layer = new SliceLayer;
lbajardsilogic@0 372 break;
lbajardsilogic@0 373
lbajardsilogic@0 374 case MelodicRangeSpectrogram:
lbajardsilogic@0 375 layer = new SpectrogramLayer(SpectrogramLayer::MelodicRange);
lbajardsilogic@0 376 break;
lbajardsilogic@0 377
lbajardsilogic@0 378 case PeakFrequencySpectrogram:
lbajardsilogic@0 379 layer = new SpectrogramLayer(SpectrogramLayer::MelodicPeaks);
lbajardsilogic@0 380 break;
lbajardsilogic@0 381
lbajardsilogic@0 382 default: break;
lbajardsilogic@0 383 }
lbajardsilogic@0 384
lbajardsilogic@0 385 if (!layer) {
lbajardsilogic@0 386 std::cerr << "LayerFactory::createLayer: Unknown layer type "
lbajardsilogic@0 387 << type << std::endl;
lbajardsilogic@0 388 } else {
lbajardsilogic@0 389 // std::cerr << "LayerFactory::createLayer: Setting object name "
lbajardsilogic@0 390 // << getLayerPresentationName(type).toStdString() << " on " << layer << std::endl;
lbajardsilogic@0 391 layer->setObjectName(getLayerPresentationName(type));
lbajardsilogic@0 392 }
lbajardsilogic@0 393
lbajardsilogic@0 394 return layer;
lbajardsilogic@0 395 }
lbajardsilogic@0 396