annotate layer/TimeRulerLayer.cpp @ 1:ab83c415a6cd

* Backed out partially complete changes to make the spectrogram only store results up to the requested max frequency. The speed improvement was minimal at the expense of annoyance when changing frequency limit, and although it did save memory, it wasn't yet reliable and fixing it is not a high enough priority.
author Chris Cannam
date Tue, 10 Jan 2006 17:04:02 +0000
parents 2a4f26e85b4c
children 37b110168acf
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@0 5 Chris Cannam, Queen Mary University of London, 2005
Chris@0 6
Chris@0 7 This is experimental software. Not for distribution.
Chris@0 8 */
Chris@0 9
Chris@0 10 #include "TimeRulerLayer.h"
Chris@0 11
Chris@0 12 #include "base/Model.h"
Chris@0 13 #include "base/RealTime.h"
Chris@0 14 #include "base/View.h"
Chris@0 15
Chris@0 16 #include <QPainter>
Chris@0 17
Chris@0 18 #include <iostream>
Chris@0 19
Chris@0 20 using std::cerr;
Chris@0 21 using std::endl;
Chris@0 22
Chris@0 23 TimeRulerLayer::TimeRulerLayer(View *w) :
Chris@0 24 Layer(w),
Chris@0 25 m_model(0),
Chris@0 26 m_colour(Qt::black),
Chris@0 27 m_labelHeight(LabelTop)
Chris@0 28 {
Chris@0 29 m_view->addLayer(this);
Chris@0 30 }
Chris@0 31
Chris@0 32 void
Chris@0 33 TimeRulerLayer::setModel(Model *model)
Chris@0 34 {
Chris@0 35 if (m_model == model) return;
Chris@0 36 m_model = model;
Chris@0 37 emit modelReplaced();
Chris@0 38 }
Chris@0 39
Chris@0 40 void
Chris@0 41 TimeRulerLayer::setBaseColour(QColor colour)
Chris@0 42 {
Chris@0 43 if (m_colour == colour) return;
Chris@0 44 m_colour = colour;
Chris@0 45 emit layerParametersChanged();
Chris@0 46 }
Chris@0 47
Chris@0 48 Layer::PropertyList
Chris@0 49 TimeRulerLayer::getProperties() const
Chris@0 50 {
Chris@0 51 PropertyList list;
Chris@0 52 list.push_back(tr("Colour"));
Chris@0 53 return list;
Chris@0 54 }
Chris@0 55
Chris@0 56 Layer::PropertyType
Chris@0 57 TimeRulerLayer::getPropertyType(const PropertyName &name) const
Chris@0 58 {
Chris@0 59 return ValueProperty;
Chris@0 60 }
Chris@0 61
Chris@0 62 int
Chris@0 63 TimeRulerLayer::getPropertyRangeAndValue(const PropertyName &name,
Chris@0 64 int *min, int *max) const
Chris@0 65 {
Chris@0 66 int deft = 0;
Chris@0 67
Chris@0 68 if (name == tr("Colour")) {
Chris@0 69
Chris@0 70 *min = 0;
Chris@0 71 *max = 5;
Chris@0 72
Chris@0 73 if (m_colour == Qt::black) deft = 0;
Chris@0 74 else if (m_colour == Qt::darkRed) deft = 1;
Chris@0 75 else if (m_colour == Qt::darkBlue) deft = 2;
Chris@0 76 else if (m_colour == Qt::darkGreen) deft = 3;
Chris@0 77 else if (m_colour == QColor(200, 50, 255)) deft = 4;
Chris@0 78 else if (m_colour == QColor(255, 150, 50)) deft = 5;
Chris@0 79
Chris@0 80 } else {
Chris@0 81
Chris@0 82 deft = Layer::getPropertyRangeAndValue(name, min, max);
Chris@0 83 }
Chris@0 84
Chris@0 85 return deft;
Chris@0 86 }
Chris@0 87
Chris@0 88 QString
Chris@0 89 TimeRulerLayer::getPropertyValueLabel(const PropertyName &name,
Chris@0 90 int value) const
Chris@0 91 {
Chris@0 92 if (name == tr("Colour")) {
Chris@0 93 switch (value) {
Chris@0 94 default:
Chris@0 95 case 0: return tr("Black");
Chris@0 96 case 1: return tr("Red");
Chris@0 97 case 2: return tr("Blue");
Chris@0 98 case 3: return tr("Green");
Chris@0 99 case 4: return tr("Purple");
Chris@0 100 case 5: return tr("Orange");
Chris@0 101 }
Chris@0 102 }
Chris@0 103 return tr("<unknown>");
Chris@0 104 }
Chris@0 105
Chris@0 106 void
Chris@0 107 TimeRulerLayer::setProperty(const PropertyName &name, int value)
Chris@0 108 {
Chris@0 109 if (name == tr("Colour")) {
Chris@0 110 switch (value) {
Chris@0 111 default:
Chris@0 112 case 0: setBaseColour(Qt::black); break;
Chris@0 113 case 1: setBaseColour(Qt::darkRed); break;
Chris@0 114 case 2: setBaseColour(Qt::darkBlue); break;
Chris@0 115 case 3: setBaseColour(Qt::darkGreen); break;
Chris@0 116 case 4: setBaseColour(QColor(200, 50, 255)); break;
Chris@0 117 case 5: setBaseColour(QColor(255, 150, 50)); break;
Chris@0 118 }
Chris@0 119 }
Chris@0 120 }
Chris@0 121
Chris@0 122 void
Chris@0 123 TimeRulerLayer::paint(QPainter &paint, QRect rect) const
Chris@0 124 {
Chris@0 125 // std::cerr << "TimeRulerLayer::paint (" << rect.x() << "," << rect.y()
Chris@0 126 // << ") [" << rect.width() << "x" << rect.height() << "]" << std::endl;
Chris@0 127
Chris@0 128 if (!m_model || !m_model->isOK()) return;
Chris@0 129
Chris@0 130 int sampleRate = m_model->getSampleRate();
Chris@0 131 if (!sampleRate) return;
Chris@0 132
Chris@0 133 long startFrame = m_view->getStartFrame();
Chris@0 134 long endFrame = m_view->getEndFrame();
Chris@0 135
Chris@0 136 int zoomLevel = m_view->getZoomLevel();
Chris@0 137
Chris@0 138 long rectStart = startFrame + (rect.x() - 100) * zoomLevel;
Chris@0 139 long rectEnd = startFrame + (rect.x() + rect.width() + 100) * zoomLevel;
Chris@0 140 if (rectStart < startFrame) rectStart = startFrame;
Chris@0 141 if (rectEnd > endFrame) rectEnd = endFrame;
Chris@0 142
Chris@0 143 // std::cerr << "TimeRulerLayer::paint: calling paint.save()" << std::endl;
Chris@0 144 paint.save();
Chris@0 145 //!!! paint.setClipRect(m_view->rect());
Chris@0 146
Chris@0 147 int minPixelSpacing = 50;
Chris@0 148
Chris@0 149 RealTime rtStart = RealTime::frame2RealTime(startFrame, sampleRate);
Chris@0 150 RealTime rtEnd = RealTime::frame2RealTime(endFrame, sampleRate);
Chris@0 151 // cerr << "startFrame " << startFrame << ", endFrame " << m_view->getEndFrame() << ", rtStart " << rtStart << ", rtEnd " << rtEnd << endl;
Chris@0 152 int count = m_view->width() / minPixelSpacing;
Chris@0 153 if (count < 1) count = 1;
Chris@0 154 RealTime rtGap = (rtEnd - rtStart) / count;
Chris@0 155 // cerr << "rtGap is " << rtGap << endl;
Chris@0 156
Chris@0 157 int incms;
Chris@0 158 bool quarter = false;
Chris@0 159
Chris@0 160 if (rtGap.sec > 0) {
Chris@0 161 incms = 1000;
Chris@0 162 int s = rtGap.sec;
Chris@0 163 if (s > 0) { incms *= 5; s /= 5; }
Chris@0 164 if (s > 0) { incms *= 2; s /= 2; }
Chris@0 165 if (s > 0) { incms *= 6; s /= 6; quarter = true; }
Chris@0 166 if (s > 0) { incms *= 5; s /= 5; quarter = false; }
Chris@0 167 if (s > 0) { incms *= 2; s /= 2; }
Chris@0 168 if (s > 0) { incms *= 6; s /= 6; quarter = true; }
Chris@0 169 while (s > 0) {
Chris@0 170 incms *= 10;
Chris@0 171 s /= 10;
Chris@0 172 quarter = false;
Chris@0 173 }
Chris@0 174 } else {
Chris@0 175 incms = 1;
Chris@0 176 int ms = rtGap.msec();
Chris@0 177 if (ms > 0) { incms *= 10; ms /= 10; }
Chris@0 178 if (ms > 0) { incms *= 10; ms /= 10; }
Chris@0 179 if (ms > 0) { incms *= 5; ms /= 5; }
Chris@0 180 if (ms > 0) { incms *= 2; ms /= 2; }
Chris@0 181 }
Chris@0 182 // cerr << "incms is " << incms << endl;
Chris@0 183
Chris@0 184 RealTime rt = RealTime::frame2RealTime(rectStart, sampleRate);
Chris@0 185 long ms = rt.sec * 1000 + rt.msec();
Chris@0 186 ms = (ms / incms) * incms - incms;
Chris@0 187
Chris@0 188 RealTime incRt = RealTime(incms / 1000, (incms % 1000) * 1000000);
Chris@0 189 long incFrame = RealTime::realTime2Frame(incRt, sampleRate);
Chris@0 190 int incX = incFrame / zoomLevel;
Chris@0 191 int ticks = 10;
Chris@0 192 if (incX < minPixelSpacing * 2) {
Chris@0 193 ticks = quarter ? 4 : 5;
Chris@0 194 }
Chris@0 195
Chris@0 196 QRect oldClipRect = rect;
Chris@0 197 QRect newClipRect(oldClipRect.x() - 25, oldClipRect.y(),
Chris@0 198 oldClipRect.width() + 50, oldClipRect.height());
Chris@0 199 paint.setClipRect(newClipRect);
Chris@0 200
Chris@0 201 QColor greyColour(m_colour);
Chris@0 202 if (m_colour == Qt::black) {
Chris@0 203 greyColour = QColor(200,200,200);
Chris@0 204 } else {
Chris@0 205 greyColour = m_colour.light(150);
Chris@0 206 }
Chris@0 207
Chris@0 208 while (1) {
Chris@0 209
Chris@0 210 rt = RealTime(ms / 1000, (ms % 1000) * 1000000);
Chris@0 211 ms += incms;
Chris@0 212
Chris@0 213 long frame = RealTime::realTime2Frame(rt, sampleRate);
Chris@0 214 if (frame >= rectEnd) break;
Chris@0 215
Chris@0 216 int x = (frame - startFrame) / zoomLevel;
Chris@0 217 if (x < rect.x() || x >= rect.x() + rect.width()) continue;
Chris@0 218
Chris@0 219 paint.setPen(greyColour);
Chris@0 220 paint.drawLine(x, 0, x, m_view->height());
Chris@0 221
Chris@0 222 paint.setPen(m_colour);
Chris@0 223 paint.drawLine(x, 0, x, 5);
Chris@0 224 paint.drawLine(x, m_view->height() - 6, x, m_view->height() - 1);
Chris@0 225
Chris@0 226 QString text(QString::fromStdString(rt.toText()));
Chris@0 227
Chris@0 228 int y;
Chris@0 229 QFontMetrics metrics = paint.fontMetrics();
Chris@0 230 switch (m_labelHeight) {
Chris@0 231 default:
Chris@0 232 case LabelTop:
Chris@0 233 y = 6 + metrics.ascent();
Chris@0 234 break;
Chris@0 235 case LabelMiddle:
Chris@0 236 y = m_view->height() / 2 - metrics.height() / 2 + metrics.ascent();
Chris@0 237 break;
Chris@0 238 case LabelBottom:
Chris@0 239 y = m_view->height() - metrics.height() + metrics.ascent() - 6;
Chris@0 240 }
Chris@0 241
Chris@0 242 int tw = metrics.width(text);
Chris@0 243
Chris@0 244 paint.setPen(m_view->palette().background().color());
Chris@0 245
Chris@0 246 //!!! simple drawing function for this please
Chris@0 247 //!!! and need getContrastingColour() in widget, or use the
Chris@0 248 //palette properly -- get the base class able to draw text
Chris@0 249 //using the proper colour (or this technique) automatically
Chris@0 250 for (int dx = -1; dx <= 1; ++dx) {
Chris@0 251 for (int dy = -1; dy <= 1; ++dy) {
Chris@0 252 if ((dx && dy) || !(dx || dy)) continue;
Chris@0 253 paint.drawText(x + 2 - tw / 2 + dx, y + dy, text);
Chris@0 254 }
Chris@0 255 }
Chris@0 256
Chris@0 257 paint.setPen(m_colour);
Chris@0 258 paint.drawText(x + 2 - tw / 2, y, text);
Chris@0 259
Chris@0 260 paint.setPen(greyColour);
Chris@0 261
Chris@0 262 for (int i = 1; i < ticks; ++i) {
Chris@0 263 rt = rt + (incRt / ticks);
Chris@0 264 frame = RealTime::realTime2Frame(rt, sampleRate);
Chris@0 265 x = (frame - startFrame) / zoomLevel;
Chris@0 266 int sz = 5;
Chris@0 267 if (ticks == 10) {
Chris@0 268 if ((i % 2) == 1) {
Chris@0 269 if (i == 5) {
Chris@0 270 paint.drawLine(x, 0, x, m_view->height());
Chris@0 271 } else sz = 3;
Chris@0 272 } else {
Chris@0 273 sz = 7;
Chris@0 274 }
Chris@0 275 }
Chris@0 276 paint.drawLine(x, 0, x, sz);
Chris@0 277 paint.drawLine(x, m_view->height() - sz - 1, x, m_view->height() - 1);
Chris@0 278 }
Chris@0 279 }
Chris@0 280
Chris@0 281 paint.restore();
Chris@0 282 }
Chris@0 283
Chris@0 284
Chris@0 285 #ifdef INCLUDE_MOCFILES
Chris@0 286 #include "TimeRulerLayer.moc.cpp"
Chris@0 287 #endif
Chris@0 288