Mercurial > hg > svgui
diff layer/SpectrogramLayer.cpp @ 332:6440e280122e spectrogram-cache-rejig
* Some bits and bobs of cache mangling -- closer to working, but still not
author | Chris Cannam |
---|---|
date | Mon, 19 Nov 2007 15:50:37 +0000 |
parents | 846746e4e865 |
children | 64e84e5efb76 |
line wrap: on
line diff
--- a/layer/SpectrogramLayer.cpp Fri Nov 16 17:15:30 2007 +0000 +++ b/layer/SpectrogramLayer.cpp Mon Nov 19 15:50:37 2007 +0000 @@ -1756,13 +1756,16 @@ rect = v->rect(); //!!! + float ratio = float(v->getZoomLevel()) / float(getWindowIncrement()); + int imageWidth = lrintf(v->width() * ratio); + PixmapCache *cache = m_pixmapCaches[v]; if (!cache) { cache = new PixmapCache; cache->pixmap = 0; m_pixmapCaches[v] = cache; cache->mutex.lock(); - cache->pixmap = new QImage(v->width(), v->height(), + cache->pixmap = new QImage(imageWidth, v->height(), QImage::Format_RGB32); cache->mutex.unlock(); #ifdef DEBUG_SPECTROGRAM_REPAINT @@ -1773,23 +1776,41 @@ return; //!!! prod paint thread } + long myStartFrame = v->getStartFrame(); + long cacheStartFrame = cache->startFrame; + long xoff = (myStartFrame - cacheStartFrame) / getWindowIncrement(); + + QRect sourceRect = QRect(lrintf(rect.x() * ratio) + xoff, + rect.y(), + lrintf(rect.width() * ratio), + rect.height()); + if (!cache->mutex.tryLock()) { std::cerr << "lock unavailable" << std::endl; - paint.drawImage(rect,// & cache->validArea, - *cache->pixmap, - rect);// & cache->validArea); + + paint.drawImage(rect, + cache->pixmap->copy(sourceRect) + .scaled(rect.size(), + Qt::IgnoreAspectRatio, + Qt::SmoothTransformation)); + +// paint.drawImage(rect,// & cache->validArea, +// *cache->pixmap, +// rect);// & cache->validArea); // v->update(rect); return; } - if (cache->pixmap->width() != v->width() || + if (cache->pixmap->width() != imageWidth || cache->pixmap->height() != v->height()) { QImage *newImage = new QImage - (cache->pixmap->scaled(v->width(), v->height())); + (cache->pixmap->scaled(imageWidth, v->height(), + Qt::IgnoreAspectRatio, + Qt::SmoothTransformation)); #ifdef DEBUG_SPECTROGRAM_REPAINT std::cerr << "SpectrogramLayer::paint: Rescaling cache to size " - << v->width() << "x" << v->height() << " (" - << v->width() * v->height() * 4 << " bytes)" << std::endl; + << imageWidth << "x" << v->height() << " (" + << imageWidth * v->height() * 4 << " bytes)" << std::endl; #endif delete cache->pixmap; cache->pixmap = newImage; @@ -1805,19 +1826,8 @@ //source or target rect will have edges at non-integral //coordinates - float ratio = float(v->getZoomLevel()) / float(getWindowIncrement()); - - long myStartFrame = v->getStartFrame(); - long cacheStartFrame = cache->startFrame; - long xoff = (myStartFrame - cacheStartFrame) / getWindowIncrement(); - std::cerr << "my start frame = " << myStartFrame << ", cache start frame = " << cacheStartFrame << std::endl; - QRect sourceRect = QRect(lrintf(rect.x() * ratio) + xoff, - rect.y(), - lrintf(rect.width() * ratio), - rect.height()); - std::cerr << "ratio = " << ratio << " (zoom " << v->getZoomLevel() << ", incr " << getWindowIncrement() << "); source rect " << sourceRect.x() << "," << sourceRect.y() @@ -1830,8 +1840,12 @@ << std::endl; paint.drawImage(rect,// & cache->validArea, - *cache->pixmap, - sourceRect);// & cache->validArea); + cache->pixmap->copy(sourceRect) + .scaled(rect.size(), + Qt::IgnoreAspectRatio, + Qt::SmoothTransformation)); +// *cache->pixmap, +// sourceRect);// & cache->validArea); if (cache->startFrame != v->getStartFrame()) { #ifdef DEBUG_SPECTROGRAM_REPAINT @@ -1893,7 +1907,7 @@ if (!m_exiting) { //!!! wait on condition - if (workToDo) usleep(100); + if (workToDo) ; // usleep(100); else sleep(1); } } @@ -2007,134 +2021,6 @@ int x0 = rect.left(); int x1 = rect.right() + 1; - if (cache.validArea.width() > 0) { - - //!!! no, the gui thread will handle resizes -- we just draw - //the area that's available in the cache. If the image is - //enlarged, our valid area should remain the same and we'll - //pick up on the discrepancy naturally (we hope) -// if (cache.pixmap->width() == v->width() && -// cache.pixmap->height() == v->height()) { - - if (cache.startFrame / increment == startFrame / increment && - cache.validArea.x() <= x0 && - cache.validArea.x() + cache.validArea.width() >= x1) { - -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "SpectrogramLayer: pixmap cache good" << std::endl; -#endif - - //!!! set a flag to that effect? - - return true; - - } else { - -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "SpectrogramLayer: pixmap cache partially OK" << std::endl; -#endif - - recreateWholeCache = false; - - int dx = cache.startFrame / increment - startFrame / increment; - -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "SpectrogramLayer: dx = " << dx << " (pixmap cache " << cache.pixmap->width() << "x" << cache.pixmap->height() << ")" << std::endl; -#endif - - if (dx != 0 && dx > -w && dx < w) { - -#if defined(Q_WS_WIN32) || defined(Q_WS_MAC) - // Copying a pixmap to itself doesn't work - // properly on Windows or Mac (it only works when - // moving in one direction). - - //!!! Need a utility function for this - - //!!! test again on Win and OS/X for images (above - //comment applies to pixmaps) - - //!!! indeed, test again on linux for images too - //-- I bet it doesn't work - - static QImage *tmpPixmap = 0; - if (!tmpPixmap || - tmpPixmap->width() != cache.pixmap->width() || - tmpPixmap->height() != cache.pixmap->height()) { - delete tmpPixmap; - tmpPixmap = new QImage(cache.pixmap->width(), - cache.pixmap->height(), - QImage::Format_RGB32); - } - QPainter tmpPainter; - tmpPainter.begin(tmpPixmap); - tmpPainter.drawImage(0, 0, image); - tmpPainter.end(); - paint.drawImage(dx, 0, *tmpPixmap); -#else - paint.drawImage(dx, 0, image); //!!! see above -#endif - - int px = cache.validArea.x(); - int pw = cache.validArea.width(); - - if (dx < 0) { - x0 = w + dx; - x1 = w; - px += dx; - if (px < 0) { - pw += px; - px = 0; - if (pw < 0) pw = 0; - } - } else { - x0 = 0; - x1 = dx; - px += dx; - if (px + pw > w) { - pw = int(w) - px; - if (pw < 0) pw = 0; - } - } - - cache.validArea = - QRect(px, cache.validArea.y(), - pw, cache.validArea.height()); - } - } -// } else { -//#ifdef DEBUG_SPECTROGRAM_REPAINT -// std::cerr << "SpectrogramLayer: pixmap cache useless" << std::endl; -// if (cache.pixmap->width() != v->width()) { -// std::cerr << "(cache width " << cache.pixmap->width() -// << " != " << v->width(); -// } -// if (cache.pixmap->height() != v->height()) { -// std::cerr << "(cache height " << cache.pixmap->height() -// << " != " << v->height(); -// } -//#endif -// cache.validArea = QRect(); -// } -} - -/*!!! - if (updateViewMagnitudes(v)) { -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "SpectrogramLayer: magnitude range changed to [" << m_viewMags[v].getMin() << "->" << m_viewMags[v].getMax() << "]" << std::endl; -#endif - recreateWholeCache = true; - } else { -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "No change in magnitude range [" << m_viewMags[v].getMin() << "->" << m_viewMags[v].getMax() << "]" << std::endl; -#endif - } -*/ - if (recreateWholeCache) { - x0 = 0; - x1 = w; - } - struct timeval tv; (void)gettimeofday(&tv, 0); RealTime mainPaintStart = RealTime::fromTimeval(tv); @@ -2163,6 +2049,12 @@ std::cerr << "[" << this << "]: last paint width: " << m_lastPaintBlockWidth << ", last paint time: " << m_lastPaintTime << ", new paint width: " << paintBlockWidth << std::endl; #endif + recreateWholeCache = checkAndScrollCache(cache, paint, + startFrame, increment, + x0, x1); + + selectPaintStrip(cache, x0, x1, paintBlockWidth); + // We always paint the full height when refreshing the cache. // Smaller heights can be used when painting from cache, but we // want to ensure the cache is coherent without having to worry @@ -2245,6 +2137,8 @@ bool runOutOfData = false; + std::cerr << "painting from " << x0 << " to " << x1 << std::endl; + for (int x = 0; x < (x1 - x0); ++x) { if (runOutOfData) break; @@ -2456,55 +2350,22 @@ } */ if (cache.validArea.width() > 0) { - + int vx0 = 0, vx1 = 0; vx0 = cache.validArea.x(); vx1 = cache.validArea.x() + cache.validArea.width(); -#ifdef DEBUG_SPECTROGRAM_REPAINT - std::cerr << "x0 " << x0 << ", x1 " << x1 << ", vx0 " << vx0 << ", vx1 " << vx1 << ", paintBlockWidth " << paintBlockWidth << std::endl; -#endif - if (x0 < vx0) { - if (x0 + paintBlockWidth < vx0) { - x0 = vx0 - paintBlockWidth; - } else { - x0 = 0; - } - } else if (x0 > vx1) { - x0 = vx1; - } - - if (x1 < vx0) { - x1 = vx0; - } else if (x1 > vx1) { - if (vx1 + paintBlockWidth < x1) { - x1 = vx1 + paintBlockWidth; - } else { - x1 = v->width(); - } - } - cache.validArea = QRect (std::min(vx0, x0), cache.validArea.y(), std::max(vx1 - std::min(vx0, x0), x1 - std::min(vx0, x0)), cache.validArea.height()); - + } else { - if (x1 > x0 + paintBlockWidth) { - int sfx = x1; - if (startFrame < 0) sfx = v->getXForFrame(0); //!!! - if (sfx >= x0 && sfx + paintBlockWidth <= x1) { - x0 = sfx; - x1 = x0 + paintBlockWidth; - } else { - int mid = (x1 + x0) / 2; - x0 = mid - paintBlockWidth/2; - x1 = x0 + paintBlockWidth; - } - } + cache.validArea = QRect(x0, 0, x1 - x0, h); } + /* Profiler profiler2("SpectrogramLayer::paintCache: draw image", true); @@ -2589,6 +2450,214 @@ // if (fftSuspended) fft->resume(); } +bool +SpectrogramLayer::checkAndScrollCache(PixmapCache &cache, QPainter &paint, + long startFrame, int increment, + int &x0, int &x1) const +{ + int w = cache.pixmap->width(); + + if (cache.validArea.width() == 0) { + x0 = 0; + x1 = w; + return true; // redraw everything + } + + //!!! no, the gui thread will handle resizes -- we just draw + //the area that's available in the cache. If the image is + //enlarged, our valid area should remain the same and we'll + //pick up on the discrepancy naturally (we hope) +// if (cache.pixmap->width() == v->width() && +// cache.pixmap->height() == v->height()) { + + if (cache.startFrame / increment == startFrame / increment && + cache.validArea.x() <= x0 && + cache.validArea.x() + cache.validArea.width() >= x1) { + +#ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "SpectrogramLayer: pixmap cache good" << std::endl; +#endif + + //!!! set a flag to that effect? + + x1 = x0; + return false; + + } else { + +#ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "SpectrogramLayer: pixmap cache partially OK" << std::endl; +#endif + + int dx = cache.startFrame / increment - startFrame / increment; + +#ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "SpectrogramLayer: dx = " << dx << " (pixmap cache " << cache.pixmap->width() << "x" << cache.pixmap->height() << ")" << std::endl; +#endif + + if (dx != 0 && dx > -w && dx < w) { + +//#if defined(Q_WS_WIN32) || defined(Q_WS_MAC) + // Copying a pixmap to itself doesn't work + // properly on Windows or Mac (it only works when + // moving in one direction). + + //!!! Need a utility function for this + + //!!! test again on Win and OS/X for images (above + //comment applies to pixmaps) + + //!!! indeed, test again on linux for images too + //-- I bet it doesn't work + + std::cerr << "SpectrogramLayer: copying from " << 0 + << " to " << dx << std::endl; + + static QImage *tmp = 0; //!!! would need to be thread-local + if (!tmp || + tmp->width() != w || + tmp->height() != cache.pixmap->height()) { + delete tmp; + tmp = new QImage(w, + cache.pixmap->height(), + QImage::Format_RGB32); + } + QPainter tmpPainter; + tmpPainter.begin(tmp); + tmpPainter.drawImage(0, 0, *cache.pixmap); + tmpPainter.end(); + paint.drawImage(dx, 0, *tmp); +//#else +// paint.drawImage(dx, 0, image); //!!! see above +//#endif + + int px = cache.validArea.x(); + int pw = cache.validArea.width(); + + if (dx < 0) { + x0 = w + dx; + x1 = w; + px += dx; + if (px < 0) { + pw += px; + px = 0; + if (pw < 0) pw = 0; + } + } else { + x0 = 0; + x1 = dx; + px += dx; + if (px + pw > w) { + pw = int(w) - px; + if (pw < 0) pw = 0; + } + } + + cache.validArea = + QRect(px, cache.validArea.y(), + pw, cache.validArea.height()); + } + } +// } else { +//#ifdef DEBUG_SPECTROGRAM_REPAINT +// std::cerr << "SpectrogramLayer: pixmap cache useless" << std::endl; +// if (cache.pixmap->width() != v->width()) { +// std::cerr << "(cache width " << cache.pixmap->width() +// << " != " << v->width(); +// } +// if (cache.pixmap->height() != v->height()) { +// std::cerr << "(cache height " << cache.pixmap->height() +// << " != " << v->height(); +// } +//#endif +// cache.validArea = QRect(); +// } + +/*!!! + if (updateViewMagnitudes(v)) { + #ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "SpectrogramLayer: magnitude range changed to [" << m_viewMags[v].getMin() << "->" << m_viewMags[v].getMax() << "]" << std::endl; + #endif + recreateWholeCache = true; + } else { + #ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "No change in magnitude range [" << m_viewMags[v].getMin() << "->" << m_viewMags[v].getMax() << "]" << std::endl; + #endif + } +*/ + return false; +} + +void +SpectrogramLayer::selectPaintStrip(PixmapCache &cache, int &x0, int &x1, + int paintBlockWidth) const +{ + // This function narrows down the region [x0, x1) where x1 >= x0 + // to a strip of approximately paintBlockWidth width. + + if (x1 - x0 < paintBlockWidth * 3 / 2) return; // it'll do + + if (cache.validArea.width() > 0) { + + // If part of the cache is known to be valid, select a strip + // immediately to left or right of the valid part + + int vx0 = 0, vx1 = 0; // valid extents + vx0 = cache.validArea.x(); + vx1 = cache.validArea.x() + cache.validArea.width(); + +#ifdef DEBUG_SPECTROGRAM_REPAINT + std::cerr << "x0 " << x0 << ", x1 " << x1 << ", vx0 " << vx0 << ", vx1 " << vx1 << ", paintBlockWidth " << paintBlockWidth << std::endl; +#endif + if (x0 < vx0) { + if (x0 + paintBlockWidth < vx0) { + x0 = vx0 - paintBlockWidth; +// } else { +// x0 = 0; + } + x1 = vx0; + } else if (x0 >= vx1) { + x0 = vx1; + if (x1 > x0 + paintBlockWidth) { + x1 = x0 + paintBlockWidth; + } + } else { + // x0 is within the valid area + if (x1 > vx1) { + x0 = vx1; + if (x0 + paintBlockWidth < x1) { + x1 = x0 + paintBlockWidth; + } + } else { + x1 = x0; // it's all valid, paint nothing + } + } + + } else { + + // No existing valid area; start at the middle, unless x0 is + // already to the left of frame 0 + + //!!! no, for now just start at the middle always + + if (x1 > x0 + paintBlockWidth) { + +// int sfx = x1; // x for frame zero +// if (startFrame < 0) sfx = v->getXForFrame(0); //!!! + +// if (startFrame < 0) sfx = +// if (sfx >= x0 && sfx + paintBlockWidth <= x1) { +// x0 = sfx; +// x1 = x0 + paintBlockWidth; +// } else { + int mid = (x1 + x0) / 2; + x0 = mid - paintBlockWidth/2; + x1 = x0 + paintBlockWidth; +// } + } + } +} + void SpectrogramLayer::illuminateLocalFeatures(View *v, QPainter &paint) const {