annotate layer/TimeRulerLayer.cpp @ 282:d9319859a4cf tip

(none)
author benoitrigolleau
date Fri, 31 Oct 2008 11:00:24 +0000
parents d8e6709e9075
children
rev   line source
lbajardsilogic@0 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
lbajardsilogic@0 2
lbajardsilogic@0 3 /*
lbajardsilogic@0 4 Sonic Visualiser
lbajardsilogic@0 5 An audio file viewer and annotation editor.
lbajardsilogic@0 6 Centre for Digital Music, Queen Mary, University of London.
lbajardsilogic@0 7 This file copyright 2006 Chris Cannam.
lbajardsilogic@0 8
lbajardsilogic@0 9 This program is free software; you can redistribute it and/or
lbajardsilogic@0 10 modify it under the terms of the GNU General Public License as
lbajardsilogic@0 11 published by the Free Software Foundation; either version 2 of the
lbajardsilogic@0 12 License, or (at your option) any later version. See the file
lbajardsilogic@0 13 COPYING included with this distribution for more information.
lbajardsilogic@0 14 */
lbajardsilogic@0 15
lbajardsilogic@0 16 #include "TimeRulerLayer.h"
lbajardsilogic@0 17
lbajardsilogic@0 18 #include "data/model/Model.h"
lbajardsilogic@0 19 #include "base/RealTime.h"
lbajardsilogic@0 20 #include "view/View.h"
lbajardsilogic@0 21
lbajardsilogic@0 22 #include <QPainter>
lbajardsilogic@0 23
lbajardsilogic@0 24 #include <iostream>
lbajardsilogic@0 25
lbajardsilogic@0 26 using std::cerr;
lbajardsilogic@0 27 using std::endl;
lbajardsilogic@0 28
lbajardsilogic@0 29 TimeRulerLayer::TimeRulerLayer() :
lbajardsilogic@0 30 Layer(),
lbajardsilogic@0 31 m_model(0),
lbajardsilogic@0 32 m_colour(Qt::black),
lbajardsilogic@0 33 m_labelHeight(LabelTop)
lbajardsilogic@0 34 {
lbajardsilogic@0 35
lbajardsilogic@0 36 }
lbajardsilogic@0 37
lbajardsilogic@0 38 void
lbajardsilogic@0 39 TimeRulerLayer::setModel(Model *model)
lbajardsilogic@0 40 {
lbajardsilogic@0 41 if (m_model == model) return;
lbajardsilogic@0 42 m_model = model;
lbajardsilogic@0 43 emit modelReplaced();
lbajardsilogic@0 44 }
lbajardsilogic@0 45
lbajardsilogic@0 46 void
lbajardsilogic@0 47 TimeRulerLayer::setBaseColour(QColor colour)
lbajardsilogic@0 48 {
lbajardsilogic@0 49 if (m_colour == colour) return;
lbajardsilogic@0 50 m_colour = colour;
lbajardsilogic@0 51 emit layerParametersChanged();
lbajardsilogic@0 52 }
lbajardsilogic@0 53
lbajardsilogic@0 54 Layer::PropertyList
lbajardsilogic@0 55 TimeRulerLayer::getProperties() const
lbajardsilogic@0 56 {
lbajardsilogic@0 57 PropertyList list;
lbajardsilogic@0 58 list.push_back("Colour");
lbajardsilogic@0 59 return list;
lbajardsilogic@0 60 }
lbajardsilogic@0 61
lbajardsilogic@0 62 QString
lbajardsilogic@0 63 TimeRulerLayer::getPropertyLabel(const PropertyName &name) const
lbajardsilogic@0 64 {
lbajardsilogic@0 65 if (name == "Colour") return tr("Colour");
lbajardsilogic@0 66 return "";
lbajardsilogic@0 67 }
lbajardsilogic@0 68
lbajardsilogic@0 69 Layer::PropertyType
lbajardsilogic@0 70 TimeRulerLayer::getPropertyType(const PropertyName &) const
lbajardsilogic@0 71 {
lbajardsilogic@0 72 return ValueProperty;
lbajardsilogic@0 73 }
lbajardsilogic@0 74
lbajardsilogic@0 75 int
lbajardsilogic@0 76 TimeRulerLayer::getPropertyRangeAndValue(const PropertyName &name,
lbajardsilogic@0 77 int *min, int *max, int *deflt) const
lbajardsilogic@0 78 {
lbajardsilogic@0 79 int val = 0;
lbajardsilogic@0 80
lbajardsilogic@0 81 if (name == "Colour") {
lbajardsilogic@0 82
lbajardsilogic@0 83 if (min) *min = 0;
lbajardsilogic@0 84 if (max) *max = 5;
lbajardsilogic@0 85 if (deflt) *deflt = 0;
lbajardsilogic@0 86
lbajardsilogic@0 87 if (m_colour == Qt::black) val = 0;
lbajardsilogic@0 88 else if (m_colour == Qt::darkRed) val = 1;
lbajardsilogic@0 89 else if (m_colour == Qt::darkBlue) val = 2;
lbajardsilogic@0 90 else if (m_colour == Qt::darkGreen) val = 3;
lbajardsilogic@0 91 else if (m_colour == QColor(200, 50, 255)) val = 4;
lbajardsilogic@0 92 else if (m_colour == QColor(255, 150, 50)) val = 5;
lbajardsilogic@0 93
lbajardsilogic@0 94 } else {
lbajardsilogic@0 95
lbajardsilogic@0 96 val = Layer::getPropertyRangeAndValue(name, min, max, deflt);
lbajardsilogic@0 97 }
lbajardsilogic@0 98
lbajardsilogic@0 99 return val;
lbajardsilogic@0 100 }
lbajardsilogic@0 101
lbajardsilogic@0 102 QString
lbajardsilogic@0 103 TimeRulerLayer::getPropertyValueLabel(const PropertyName &name,
lbajardsilogic@0 104 int value) const
lbajardsilogic@0 105 {
lbajardsilogic@0 106 if (name == "Colour") {
lbajardsilogic@0 107 switch (value) {
lbajardsilogic@0 108 default:
lbajardsilogic@0 109 case 0: return tr("Black");
lbajardsilogic@0 110 case 1: return tr("Red");
lbajardsilogic@0 111 case 2: return tr("Blue");
lbajardsilogic@0 112 case 3: return tr("Green");
lbajardsilogic@0 113 case 4: return tr("Purple");
lbajardsilogic@0 114 case 5: return tr("Orange");
lbajardsilogic@0 115 }
lbajardsilogic@0 116 }
lbajardsilogic@0 117 return tr("<unknown>");
lbajardsilogic@0 118 }
lbajardsilogic@0 119
lbajardsilogic@0 120 void
lbajardsilogic@0 121 TimeRulerLayer::setProperty(const PropertyName &name, int value)
lbajardsilogic@0 122 {
lbajardsilogic@0 123 if (name == "Colour") {
lbajardsilogic@0 124 switch (value) {
lbajardsilogic@0 125 default:
lbajardsilogic@0 126 case 0: setBaseColour(Qt::black); break;
lbajardsilogic@0 127 case 1: setBaseColour(Qt::darkRed); break;
lbajardsilogic@0 128 case 2: setBaseColour(Qt::darkBlue); break;
lbajardsilogic@0 129 case 3: setBaseColour(Qt::darkGreen); break;
lbajardsilogic@0 130 case 4: setBaseColour(QColor(200, 50, 255)); break;
lbajardsilogic@0 131 case 5: setBaseColour(QColor(255, 150, 50)); break;
lbajardsilogic@0 132 }
lbajardsilogic@0 133 }
lbajardsilogic@0 134 }
lbajardsilogic@0 135
lbajardsilogic@0 136 void
lbajardsilogic@0 137 TimeRulerLayer::paint(View *v, QPainter &paint, QRect rect) const
lbajardsilogic@0 138 {
lbajardsilogic@0 139 // std::cerr << "TimeRulerLayer::paint (" << rect.x() << "," << rect.y()
lbajardsilogic@0 140 // << ") [" << rect.width() << "x" << rect.height() << "]" << std::endl;
lbajardsilogic@0 141
lbajardsilogic@0 142 if (!m_model || !m_model->isOK()) return;
lbajardsilogic@0 143
lbajardsilogic@0 144 int sampleRate = m_model->getSampleRate();
lbajardsilogic@0 145 if (!sampleRate) return;
lbajardsilogic@0 146
lbajardsilogic@0 147 long startFrame = v->getStartFrame();
lbajardsilogic@0 148 long endFrame = v->getEndFrame();
lbajardsilogic@0 149
lbajardsilogic@0 150 int zoomLevel = v->getZoomLevel();
lbajardsilogic@0 151
lbajardsilogic@0 152 long rectStart = startFrame + (rect.x() - 100) * zoomLevel;
lbajardsilogic@0 153 long rectEnd = startFrame + (rect.x() + rect.width() + 100) * zoomLevel;
lbajardsilogic@0 154 // if (rectStart < startFrame) rectStart = startFrame;
lbajardsilogic@0 155 // if (rectEnd > endFrame) rectEnd = endFrame;
lbajardsilogic@0 156
lbajardsilogic@0 157 // std::cerr << "TimeRulerLayer::paint: calling paint.save()" << std::endl;
lbajardsilogic@0 158 paint.save();
lbajardsilogic@0 159 //!!! paint.setClipRect(v->rect());
lbajardsilogic@0 160
lbajardsilogic@0 161 int minPixelSpacing = 50;
lbajardsilogic@0 162
lbajardsilogic@0 163 RealTime rtStart = RealTime::frame2RealTime(startFrame, sampleRate);
lbajardsilogic@0 164 RealTime rtEnd = RealTime::frame2RealTime(endFrame, sampleRate);
lbajardsilogic@0 165 // cerr << "startFrame " << startFrame << ", endFrame " << v->getEndFrame() << ", rtStart " << rtStart << ", rtEnd " << rtEnd << endl;
lbajardsilogic@0 166 int count = v->width() / minPixelSpacing;
lbajardsilogic@0 167 if (count < 1) count = 1;
lbajardsilogic@0 168 RealTime rtGap = (rtEnd - rtStart) / count;
lbajardsilogic@0 169 // cerr << "rtGap is " << rtGap << endl;
lbajardsilogic@0 170
lbajardsilogic@0 171 int incms;
lbajardsilogic@0 172 bool quarter = false;
lbajardsilogic@0 173
lbajardsilogic@0 174 if (rtGap.sec > 0) {
lbajardsilogic@0 175 incms = 1000;
lbajardsilogic@0 176 int s = rtGap.sec;
lbajardsilogic@0 177 if (s > 0) { incms *= 5; s /= 5; }
lbajardsilogic@0 178 if (s > 0) { incms *= 2; s /= 2; }
lbajardsilogic@0 179 if (s > 0) { incms *= 6; s /= 6; quarter = true; }
lbajardsilogic@0 180 if (s > 0) { incms *= 5; s /= 5; quarter = false; }
lbajardsilogic@0 181 if (s > 0) { incms *= 2; s /= 2; }
lbajardsilogic@0 182 if (s > 0) { incms *= 6; s /= 6; quarter = true; }
lbajardsilogic@0 183 while (s > 0) {
lbajardsilogic@0 184 incms *= 10;
lbajardsilogic@0 185 s /= 10;
lbajardsilogic@0 186 quarter = false;
lbajardsilogic@0 187 }
lbajardsilogic@0 188 } else {
lbajardsilogic@0 189 incms = 1;
lbajardsilogic@0 190 int ms = rtGap.msec();
lbajardsilogic@0 191 if (ms > 0) { incms *= 10; ms /= 10; }
lbajardsilogic@0 192 if (ms > 0) { incms *= 10; ms /= 10; }
lbajardsilogic@0 193 if (ms > 0) { incms *= 5; ms /= 5; }
lbajardsilogic@0 194 if (ms > 0) { incms *= 2; ms /= 2; }
lbajardsilogic@0 195 }
lbajardsilogic@0 196 // cerr << "incms is " << incms << endl;
lbajardsilogic@0 197
lbajardsilogic@0 198 RealTime rt = RealTime::frame2RealTime(rectStart, sampleRate);
lbajardsilogic@0 199 long ms = rt.sec * 1000 + rt.msec();
lbajardsilogic@0 200 ms = (ms / incms) * incms - incms;
lbajardsilogic@0 201
lbajardsilogic@0 202 RealTime incRt = RealTime::fromMilliseconds(incms);
lbajardsilogic@0 203 long incFrame = RealTime::realTime2Frame(incRt, sampleRate);
lbajardsilogic@0 204 int incX = incFrame / zoomLevel;
lbajardsilogic@0 205 int ticks = 10;
lbajardsilogic@0 206 if (incX < minPixelSpacing * 2) {
lbajardsilogic@0 207 ticks = quarter ? 4 : 5;
lbajardsilogic@0 208 }
lbajardsilogic@0 209
lbajardsilogic@0 210 QRect oldClipRect = rect;
lbajardsilogic@0 211 QRect newClipRect(oldClipRect.x() - 25, oldClipRect.y(),
lbajardsilogic@0 212 oldClipRect.width() + 50, oldClipRect.height());
lbajardsilogic@0 213 paint.setClipRect(newClipRect);
lbajardsilogic@0 214 paint.setClipRect(rect);
lbajardsilogic@0 215
lbajardsilogic@0 216 QColor greyColour(m_colour);
lbajardsilogic@0 217 if (m_colour == Qt::black) {
lbajardsilogic@0 218 greyColour = QColor(200,200,200);
lbajardsilogic@0 219 } else {
lbajardsilogic@0 220 greyColour = m_colour.light(150);
lbajardsilogic@0 221 }
lbajardsilogic@0 222
lbajardsilogic@0 223 while (1) {
lbajardsilogic@0 224
lbajardsilogic@0 225 rt = RealTime::fromMilliseconds(ms);
lbajardsilogic@0 226 ms += incms;
lbajardsilogic@0 227
lbajardsilogic@0 228 long frame = RealTime::realTime2Frame(rt, sampleRate);
lbajardsilogic@0 229 if (frame >= rectEnd) break;
lbajardsilogic@0 230
lbajardsilogic@0 231 int x = (frame - startFrame) / zoomLevel;
lbajardsilogic@0 232 if (x < rect.x() || x >= rect.x() + rect.width()) continue;
lbajardsilogic@0 233
lbajardsilogic@0 234 paint.setPen(greyColour);
lbajardsilogic@0 235 paint.drawLine(x, 0, x, v->height());
lbajardsilogic@0 236
lbajardsilogic@0 237 paint.setPen(m_colour);
lbajardsilogic@0 238 paint.drawLine(x, 0, x, 5);
lbajardsilogic@0 239 paint.drawLine(x, v->height() - 6, x, v->height() - 1);
lbajardsilogic@0 240
lbajardsilogic@0 241 QString text(QString::fromStdString(rt.toText()));
lbajardsilogic@0 242
lbajardsilogic@0 243 int y;
lbajardsilogic@0 244 QFontMetrics metrics = paint.fontMetrics();
lbajardsilogic@0 245 switch (m_labelHeight) {
lbajardsilogic@0 246 default:
lbajardsilogic@0 247 case LabelTop:
lbajardsilogic@0 248 y = 6 + metrics.ascent();
lbajardsilogic@0 249 break;
lbajardsilogic@0 250 case LabelMiddle:
lbajardsilogic@0 251 y = v->height() / 2 - metrics.height() / 2 + metrics.ascent();
lbajardsilogic@0 252 break;
lbajardsilogic@0 253 case LabelBottom:
lbajardsilogic@0 254 y = v->height() - metrics.height() + metrics.ascent() - 6;
lbajardsilogic@0 255 }
lbajardsilogic@0 256
lbajardsilogic@0 257 int tw = metrics.width(text);
lbajardsilogic@0 258
lbajardsilogic@0 259 if (v->getViewManager() && v->getViewManager()->getOverlayMode() !=
lbajardsilogic@0 260 ViewManager::NoOverlays) {
lbajardsilogic@0 261
lbajardsilogic@0 262 if (v->getLayer(0) == this) {
lbajardsilogic@0 263 // backmost layer, don't worry about outlining the text
lbajardsilogic@0 264 paint.drawText(x+2 - tw/2, y, text);
lbajardsilogic@0 265 } else {
lbajardsilogic@0 266 v->drawVisibleText(paint, x+2 - tw/2, y, text, View::OutlinedText);
lbajardsilogic@0 267 }
lbajardsilogic@0 268 }
lbajardsilogic@0 269
lbajardsilogic@0 270 paint.setPen(greyColour);
lbajardsilogic@0 271
lbajardsilogic@0 272 for (int i = 1; i < ticks; ++i) {
lbajardsilogic@0 273 rt = rt + (incRt / ticks);
lbajardsilogic@0 274 frame = RealTime::realTime2Frame(rt, sampleRate);
lbajardsilogic@0 275 x = (frame - startFrame) / zoomLevel;
lbajardsilogic@0 276 int sz = 5;
lbajardsilogic@0 277 if (ticks == 10) {
lbajardsilogic@0 278 if ((i % 2) == 1) {
lbajardsilogic@0 279 if (i == 5) {
lbajardsilogic@0 280 paint.drawLine(x, 0, x, v->height());
lbajardsilogic@0 281 } else sz = 3;
lbajardsilogic@0 282 } else {
lbajardsilogic@0 283 sz = 7;
lbajardsilogic@0 284 }
lbajardsilogic@0 285 }
lbajardsilogic@0 286 paint.drawLine(x, 0, x, sz);
lbajardsilogic@0 287 paint.drawLine(x, v->height() - sz - 1, x, v->height() - 1);
lbajardsilogic@0 288 }
lbajardsilogic@0 289 }
lbajardsilogic@0 290
lbajardsilogic@0 291 paint.restore();
lbajardsilogic@0 292 }
lbajardsilogic@0 293
lbajardsilogic@0 294 QString
lbajardsilogic@0 295 TimeRulerLayer::toXmlString(QString indent, QString extraAttributes) const
lbajardsilogic@0 296 {
lbajardsilogic@0 297 return Layer::toXmlString(indent, extraAttributes +
lbajardsilogic@0 298 QString(" colour=\"%1\"").arg(encodeColour(m_colour)));
lbajardsilogic@0 299 }
lbajardsilogic@0 300
lbajardsilogic@18 301 QString TimeRulerLayer::toEasaierXmlString(QString indent, QString extraAttributes) const
lbajardsilogic@18 302 {
lbajardsilogic@18 303 return Layer::toEasaierXmlString(indent, extraAttributes +
lbajardsilogic@18 304 QString(" colour=\"%1\"").arg(encodeColour(m_colour)));
lbajardsilogic@18 305 }
lbajardsilogic@18 306
lbajardsilogic@0 307 void
lbajardsilogic@0 308 TimeRulerLayer::setProperties(const QXmlAttributes &attributes)
lbajardsilogic@0 309 {
lbajardsilogic@0 310 QString colourSpec = attributes.value("colour");
lbajardsilogic@0 311 if (colourSpec != "") {
lbajardsilogic@0 312 QColor colour(colourSpec);
lbajardsilogic@0 313 if (colour.isValid()) {
lbajardsilogic@0 314 setBaseColour(QColor(colourSpec));
lbajardsilogic@0 315 }
lbajardsilogic@0 316 }
lbajardsilogic@0 317 }
lbajardsilogic@0 318