annotate data/model/Dense3DModelPeakCache.cpp @ 546:95391b480e17

* Make use of peak cache in spectrogram
author Chris Cannam
date Wed, 04 Feb 2009 15:03:42 +0000
parents c603d9439b37
children 1469caaa8e67
rev   line source
Chris@545 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@545 2
Chris@545 3 /*
Chris@545 4 Sonic Visualiser
Chris@545 5 An audio file viewer and annotation editor.
Chris@545 6 Centre for Digital Music, Queen Mary, University of London.
Chris@545 7 This file copyright 2009 QMUL.
Chris@545 8
Chris@545 9 This program is free software; you can redistribute it and/or
Chris@545 10 modify it under the terms of the GNU General Public License as
Chris@545 11 published by the Free Software Foundation; either version 2 of the
Chris@545 12 License, or (at your option) any later version. See the file
Chris@545 13 COPYING included with this distribution for more information.
Chris@545 14 */
Chris@545 15
Chris@545 16 #include "Dense3DModelPeakCache.h"
Chris@545 17
Chris@545 18 Dense3DModelPeakCache::Dense3DModelPeakCache(DenseThreeDimensionalModel *source,
Chris@545 19 size_t columnsPerPeak) :
Chris@545 20 m_source(source),
Chris@545 21 m_resolution(columnsPerPeak)
Chris@545 22 {
Chris@546 23 m_coverage.resize(1); // otherwise it is simply invalid
Chris@546 24
Chris@545 25 m_cache = new EditableDenseThreeDimensionalModel
Chris@545 26 (source->getSampleRate(),
Chris@545 27 getResolution(),
Chris@545 28 source->getHeight(),
Chris@545 29 EditableDenseThreeDimensionalModel::NoCompression,
Chris@545 30 false);
Chris@545 31
Chris@545 32 connect(source, SIGNAL(modelChanged()),
Chris@545 33 this, SLOT(sourceModelChanged()));
Chris@546 34 connect(source, SIGNAL(aboutToBeDeleted()),
Chris@545 35 this, SLOT(sourceModelAboutToBeDeleted()));
Chris@545 36
Chris@545 37 }
Chris@545 38
Chris@545 39 Dense3DModelPeakCache::~Dense3DModelPeakCache()
Chris@545 40 {
Chris@545 41 delete m_cache;
Chris@545 42 }
Chris@545 43
Chris@545 44 bool
Chris@545 45 Dense3DModelPeakCache::isColumnAvailable(size_t column) const
Chris@545 46 {
Chris@545 47 if (!m_source) return false;
Chris@545 48 if (haveColumn(column)) return true;
Chris@545 49 for (int i = 0; i < m_resolution; ++i) {
Chris@545 50 if (!m_source->isColumnAvailable(column * m_resolution + i)) {
Chris@545 51 return false;
Chris@545 52 }
Chris@545 53 }
Chris@545 54 return true;
Chris@545 55 }
Chris@545 56
Chris@545 57 Dense3DModelPeakCache::Column
Chris@545 58 Dense3DModelPeakCache::getColumn(size_t column) const
Chris@545 59 {
Chris@545 60 if (!m_source) return Column();
Chris@545 61 if (!haveColumn(column)) fillColumn(column);
Chris@545 62 return m_cache->getColumn(column);
Chris@545 63 }
Chris@545 64
Chris@545 65 float
Chris@545 66 Dense3DModelPeakCache::getValueAt(size_t column, size_t n) const
Chris@545 67 {
Chris@545 68 if (!m_source) return 0.f;
Chris@545 69 if (!haveColumn(column)) fillColumn(column);
Chris@545 70 return m_cache->getValueAt(column, n);
Chris@545 71 }
Chris@545 72
Chris@545 73 void
Chris@545 74 Dense3DModelPeakCache::sourceModelChanged()
Chris@545 75 {
Chris@545 76 if (!m_source) return;
Chris@546 77 if (m_coverage.size() > 0) {
Chris@546 78 // The last peak may have come from an incomplete read, which
Chris@546 79 // may since have been filled, so reset it
Chris@546 80 m_coverage.reset(m_coverage.size()-1);
Chris@546 81 }
Chris@546 82 if (getWidth() > m_coverage.size()) {
Chris@546 83 m_coverage.resize(getWidth()); // clears all bits, which is OK with us
Chris@546 84 }
Chris@545 85 }
Chris@545 86
Chris@545 87 void
Chris@545 88 Dense3DModelPeakCache::sourceModelAboutToBeDeleted()
Chris@545 89 {
Chris@545 90 m_source = 0;
Chris@545 91 }
Chris@545 92
Chris@545 93 bool
Chris@545 94 Dense3DModelPeakCache::haveColumn(size_t column) const
Chris@545 95 {
Chris@546 96 return column < m_coverage.size() && m_coverage.get(column);
Chris@545 97 }
Chris@545 98
Chris@545 99 void
Chris@545 100 Dense3DModelPeakCache::fillColumn(size_t column) const
Chris@545 101 {
Chris@546 102 if (column >= m_coverage.size()) m_coverage.resize(column + 1);
Chris@546 103
Chris@545 104 Column peak;
Chris@545 105 for (int i = 0; i < m_resolution; ++i) {
Chris@545 106 Column here = m_source->getColumn(column * m_resolution + i);
Chris@545 107 if (i == 0) {
Chris@545 108 peak = here;
Chris@545 109 } else {
Chris@545 110 for (int j = 0; j < peak.size() && j < here.size(); ++j) {
Chris@545 111 if (here[j] > peak[j]) peak[j] = here[j];
Chris@545 112 }
Chris@545 113 }
Chris@545 114 }
Chris@546 115
Chris@545 116 m_cache->setColumn(column, peak);
Chris@545 117 m_coverage.set(column);
Chris@545 118 }
Chris@545 119
Chris@545 120