# HG changeset patch # User Chris Cannam # Date 1147261432 0 # Node ID a0e7edf9703a54283ed9928020537250abe9faaa # Parent 095916d7ed4d6c5d198a43aa831cd8446e2fa05b * Use fractional window overlaps in the spectrogram, instead of percentages (90% is kind of meaningless when none of your window sizes are divisible by 10!) * ResizeableBitmap -> ResizeableBitset and the odd other tidy up diff -r 095916d7ed4d -r a0e7edf9703a layer/SpectrogramLayer.cpp --- a/layer/SpectrogramLayer.cpp Tue May 09 20:39:23 2006 +0000 +++ b/layer/SpectrogramLayer.cpp Wed May 10 11:43:52 2006 +0000 @@ -34,7 +34,7 @@ #include #include -//#define DEBUG_SPECTROGRAM_REPAINT 1 +#define DEBUG_SPECTROGRAM_REPAINT 1 static double mod(double x, double y) { @@ -55,7 +55,7 @@ m_channel(0), m_windowSize(1024), m_windowType(HanningWindow), - m_windowOverlap(50), + m_windowHopLevel(2), m_gain(1.0), m_threshold(0.0), m_colourRotation(0), @@ -77,13 +77,13 @@ { if (config == MelodicRange) { setWindowSize(8192); - setWindowOverlap(90); + setWindowHopLevel(4); setWindowType(ParzenWindow); setMaxFrequency(1000); setColourScale(LinearColourScale); } else if (config == MelodicPeaks) { setWindowSize(4096); - setWindowOverlap(90); + setWindowHopLevel(5); setWindowType(BlackmanWindow); setMaxFrequency(2000); setMinFrequency(40); @@ -143,7 +143,7 @@ list.push_back("Colour Scale"); list.push_back("Window Type"); list.push_back("Window Size"); - list.push_back("Window Overlap"); + list.push_back("Window Increment"); list.push_back("Normalize Columns"); list.push_back("Bin Display"); list.push_back("Threshold"); @@ -162,7 +162,7 @@ if (name == "Colour Scale") return tr("Colour Scale"); if (name == "Window Type") return tr("Window Type"); if (name == "Window Size") return tr("Window Size"); - if (name == "Window Overlap") return tr("Window Overlap"); + if (name == "Window Increment") return tr("Window Increment"); if (name == "Normalize Columns") return tr("Normalize Columns"); if (name == "Bin Display") return tr("Bin Display"); if (name == "Threshold") return tr("Threshold"); @@ -189,7 +189,7 @@ { if (name == "Window Size" || name == "Window Type" || - name == "Window Overlap") return tr("Window"); + name == "Window Increment") return tr("Window"); if (name == "Colour" || name == "Gain" || name == "Threshold" || @@ -269,13 +269,12 @@ int ws = m_windowSize; while (ws > 32) { ws >>= 1; deft ++; } - } else if (name == "Window Overlap") { + } else if (name == "Window Increment") { *min = 0; - *max = 4; + *max = 5; - deft = m_windowOverlap / 25; - if (m_windowOverlap == 90) deft = 4; + deft = m_windowHopLevel; } else if (name == "Min Frequency") { @@ -376,14 +375,15 @@ if (name == "Window Size") { return QString("%1").arg(32 << value); } - if (name == "Window Overlap") { + if (name == "Window Increment") { switch (value) { default: - case 0: return tr("0%"); - case 1: return tr("25%"); - case 2: return tr("50%"); - case 3: return tr("75%"); - case 4: return tr("90%"); + case 0: return tr("1/1"); + case 1: return tr("3/4"); + case 2: return tr("1/2"); + case 3: return tr("1/4"); + case 4: return tr("1/8"); + case 5: return tr("1/16"); } } if (name == "Min Frequency") { @@ -459,9 +459,8 @@ setWindowType(WindowType(value)); } else if (name == "Window Size") { setWindowSize(32 << value); - } else if (name == "Window Overlap") { - if (value == 4) setWindowOverlap(90); - else setWindowOverlap(25 * value); + } else if (name == "Window Increment") { + setWindowHopLevel(value); } else if (name == "Min Frequency") { switch (value) { default: @@ -588,15 +587,15 @@ } void -SpectrogramLayer::setWindowOverlap(size_t wi) +SpectrogramLayer::setWindowHopLevel(size_t v) { - if (m_windowOverlap == wi) return; + if (m_windowHopLevel == v) return; m_mutex.lock(); m_cacheInvalid = true; invalidatePixmapCaches(); - m_windowOverlap = wi; + m_windowHopLevel = v; m_mutex.unlock(); @@ -606,9 +605,9 @@ } size_t -SpectrogramLayer::getWindowOverlap() const +SpectrogramLayer::getWindowHopLevel() const { - return m_windowOverlap; + return m_windowHopLevel; } void @@ -1155,12 +1154,6 @@ double factor = 0.0; - // Calculate magnitude and phase from real and imaginary in - // output[i][0] and output[i][1] respectively, and store the phase - // straight into cache and the magnitude back into output[i][0] - // (because we'll need to know the normalization factor, - // i.e. maximum magnitude in this column, before we can store it) - for (size_t i = 0; i < windowSize/2; ++i) { double mag = sqrt(output[i][0] * output[i][0] + @@ -1310,6 +1303,8 @@ size_t windowSize = m_layer.m_windowSize; size_t windowIncrement = m_layer.getWindowIncrement(); + std::cerr << "\nWINDOW INCREMENT: " << windowIncrement << " (for hop level " << m_layer.m_windowHopLevel << ")\n" << std::endl; + size_t visibleStart = m_layer.m_candidateFillStartFrame; visibleStart = (visibleStart / windowIncrement) * windowIncrement; @@ -1675,7 +1670,7 @@ for (int s = s0i; s <= s1i; ++s) { if (s >= 0 && q >= 0 && s < cw && q < ch) { - if (!m_cache->haveColumnAt(s)) continue; + if (!m_cache->haveSetColumnAt(s)) continue; float value; @@ -1957,9 +1952,7 @@ m_drawBuffer = QImage(w, h, QImage::Format_RGB32); } -// if (m_binDisplay == PeakFrequencies) { - m_drawBuffer.fill(m_colourMap.getColour(0).rgb()); -// } + m_drawBuffer.fill(m_colourMap.getColour(0).rgb()); int sr = m_model->getSampleRate(); @@ -2031,7 +2024,7 @@ for (int s = s0i; s <= s1i; ++s) { - if (!m_cache->haveColumnAt(s)) continue; + if (!m_cache->haveSetColumnAt(s)) continue; for (size_t q = minbin; q < bins; ++q) { @@ -2579,13 +2572,13 @@ s += QString("channel=\"%1\" " "windowSize=\"%2\" " "windowType=\"%3\" " - "windowOverlap=\"%4\" " + "windowHopLevel=\"%4\" " "gain=\"%5\" " "threshold=\"%6\" ") .arg(m_channel) .arg(m_windowSize) .arg(m_windowType) - .arg(m_windowOverlap) + .arg(m_windowHopLevel) .arg(m_gain) .arg(m_threshold); @@ -2624,8 +2617,19 @@ attributes.value("windowType").toInt(&ok); if (ok) setWindowType(windowType); - size_t windowOverlap = attributes.value("windowOverlap").toUInt(&ok); - if (ok) setWindowOverlap(windowOverlap); + size_t windowHopLevel = attributes.value("windowHopLevel").toUInt(&ok); + if (ok) setWindowHopLevel(windowHopLevel); + else { + size_t windowOverlap = attributes.value("windowOverlap").toUInt(&ok); + // a percentage value + if (ok) { + if (windowOverlap == 0) setWindowHopLevel(0); + else if (windowOverlap == 25) setWindowHopLevel(1); + else if (windowOverlap == 50) setWindowHopLevel(2); + else if (windowOverlap == 75) setWindowHopLevel(3); + else if (windowOverlap == 90) setWindowHopLevel(4); + } + } float gain = attributes.value("gain").toFloat(&ok); if (ok) setGain(gain); diff -r 095916d7ed4d -r a0e7edf9703a layer/SpectrogramLayer.h --- a/layer/SpectrogramLayer.h Tue May 09 20:39:23 2006 +0000 +++ b/layer/SpectrogramLayer.h Wed May 10 11:43:52 2006 +0000 @@ -20,7 +20,6 @@ #include "base/Window.h" #include "base/RealTime.h" #include "base/Thread.h" -#include "base/ResizeableBitmap.h" #include "model/PowerOfSqrtTwoZoomConstraint.h" #include "model/DenseTimeValueModel.h" @@ -94,8 +93,8 @@ void setWindowSize(size_t); size_t getWindowSize() const; - void setWindowOverlap(size_t percent); - size_t getWindowOverlap() const; + void setWindowHopLevel(size_t percent); + size_t getWindowHopLevel() const; void setWindowType(WindowType type); WindowType getWindowType() const; @@ -210,7 +209,7 @@ int m_channel; size_t m_windowSize; WindowType m_windowType; - size_t m_windowOverlap; + size_t m_windowHopLevel; float m_gain; float m_threshold; int m_colourRotation; @@ -332,7 +331,9 @@ float &phaseMin, float &phaseMax) const; size_t getWindowIncrement() const { - return m_windowSize - m_windowSize * m_windowOverlap / 100; + if (m_windowHopLevel == 0) return m_windowSize; + else if (m_windowHopLevel == 1) return (m_windowSize * 3) / 4; + else return m_windowSize / (1 << (m_windowHopLevel - 1)); } };