annotate layer/TimeRulerLayer.cpp @ 35:10ba9276a315

* Add TextModel and TextLayer types * Make View refresh work better when editing a model (previously edits might not be refreshed if their visible changed area extended beyond the strict frame range that was being modified in the model) * Add phase-adjusted instantaneous frequency display to spectrogram layer (still a work in progress) * Pull maths aliases out into a separate header in dsp/maths so MathUtilities can be included without introducing them
author Chris Cannam
date Mon, 20 Feb 2006 13:33:36 +0000
parents ea6fe8cfcdd5
children ad214997dddb
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@5 5 Chris Cannam, Queen Mary University of London, 2005-2006
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@10 70 if (min) *min = 0;
Chris@10 71 if (max) *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@30 188 RealTime incRt = RealTime::fromMilliseconds(incms);
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@30 210 rt = RealTime::fromMilliseconds(ms);
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@6 284 QString
Chris@6 285 TimeRulerLayer::toXmlString(QString indent, QString extraAttributes) const
Chris@6 286 {
Chris@6 287 return Layer::toXmlString(indent, extraAttributes +
Chris@6 288 QString(" colour=\"%1\"").arg(encodeColour(m_colour)));
Chris@6 289 }
Chris@6 290
Chris@11 291 void
Chris@11 292 TimeRulerLayer::setProperties(const QXmlAttributes &attributes)
Chris@11 293 {
Chris@11 294 QString colourSpec = attributes.value("colour");
Chris@11 295 if (colourSpec != "") {
Chris@11 296 QColor colour(colourSpec);
Chris@11 297 if (colour.isValid()) {
Chris@11 298 setBaseColour(QColor(colourSpec));
Chris@11 299 }
Chris@11 300 }
Chris@11 301 }
Chris@11 302
Chris@0 303
Chris@0 304 #ifdef INCLUDE_MOCFILES
Chris@0 305 #include "TimeRulerLayer.moc.cpp"
Chris@0 306 #endif
Chris@0 307