annotate layer/SliceLayer.cpp @ 333:e74b56f07c73

* Some work on correct alignment when moving panes during playback * Overhaul alignment for playback frame values (view manager now always refers to reference-timeline values, only the play source deals in playback model timeline values) * When making a selection, ensure the selection regions shown in other panes (and used for playback constraints if appropriate) are aligned correctly. This may be the coolest feature ever implemented in any program ever.
author Chris Cannam
date Thu, 22 Nov 2007 14:17:19 +0000
parents 984c1975f1ff
children 2f83b6e3b8ca
rev   line source
Chris@133 1
Chris@133 2 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@133 3
Chris@133 4 /*
Chris@133 5 Sonic Visualiser
Chris@133 6 An audio file viewer and annotation editor.
Chris@133 7 Centre for Digital Music, Queen Mary, University of London.
Chris@195 8 This file copyright 2006-2007 QMUL.
Chris@133 9
Chris@133 10 This program is free software; you can redistribute it and/or
Chris@133 11 modify it under the terms of the GNU General Public License as
Chris@133 12 published by the Free Software Foundation; either version 2 of the
Chris@133 13 License, or (at your option) any later version. See the file
Chris@133 14 COPYING included with this distribution for more information.
Chris@133 15 */
Chris@133 16
Chris@193 17 #include "SliceLayer.h"
Chris@133 18
Chris@133 19 #include "view/View.h"
Chris@153 20 #include "base/AudioLevel.h"
Chris@167 21 #include "base/RangeMapper.h"
Chris@195 22 #include "base/RealTime.h"
Chris@285 23 #include "base/ColourMapper.h"
Chris@287 24 #include "base/ColourDatabase.h"
Chris@195 25
Chris@195 26 #include "PaintAssistant.h"
Chris@133 27
Chris@133 28 #include <QPainter>
Chris@133 29 #include <QPainterPath>
Chris@316 30 #include <QTextStream>
Chris@316 31
Chris@133 32
Chris@193 33 SliceLayer::SliceLayer() :
Chris@193 34 m_sliceableModel(0),
Chris@197 35 m_colourMap(0),
Chris@153 36 m_energyScale(dBScale),
Chris@198 37 m_samplingMode(SampleMean),
Chris@195 38 m_plotStyle(PlotSteps),
Chris@193 39 m_binScale(LinearBins),
Chris@153 40 m_normalize(false),
Chris@284 41 m_threshold(0.0),
Chris@284 42 m_initialThreshold(0.0),
Chris@198 43 m_gain(1.0),
Chris@198 44 m_currentf0(0),
Chris@198 45 m_currentf1(0)
Chris@133 46 {
Chris@133 47 }
Chris@133 48
Chris@193 49 SliceLayer::~SliceLayer()
Chris@133 50 {
Chris@193 51
Chris@133 52 }
Chris@133 53
Chris@133 54 void
Chris@193 55 SliceLayer::setSliceableModel(const Model *model)
Chris@133 56 {
Chris@193 57 const DenseThreeDimensionalModel *sliceable =
Chris@193 58 dynamic_cast<const DenseThreeDimensionalModel *>(model);
Chris@193 59
Chris@193 60 if (model && !sliceable) {
Chris@193 61 std::cerr << "WARNING: SliceLayer::setSliceableModel(" << model
Chris@193 62 << "): model is not a DenseThreeDimensionalModel" << std::endl;
Chris@193 63 }
Chris@193 64
Chris@193 65 if (m_sliceableModel == sliceable) return;
Chris@193 66
Chris@193 67 m_sliceableModel = sliceable;
Chris@193 68
Chris@320 69 connectSignals(m_sliceableModel);
Chris@193 70
Chris@193 71 emit modelReplaced();
Chris@153 72 }
Chris@153 73
Chris@153 74 void
Chris@193 75 SliceLayer::sliceableModelReplaced(const Model *orig, const Model *replacement)
Chris@153 76 {
Chris@193 77 std::cerr << "SliceLayer::sliceableModelReplaced(" << orig << ", " << replacement << ")" << std::endl;
Chris@153 78
Chris@193 79 if (orig == m_sliceableModel) {
Chris@193 80 setSliceableModel
Chris@193 81 (dynamic_cast<const DenseThreeDimensionalModel *>(replacement));
Chris@153 82 }
Chris@153 83 }
Chris@153 84
Chris@153 85 void
Chris@193 86 SliceLayer::modelAboutToBeDeleted(Model *m)
Chris@153 87 {
Chris@193 88 std::cerr << "SliceLayer::modelAboutToBeDeleted(" << m << ")" << std::endl;
Chris@153 89
Chris@193 90 if (m == m_sliceableModel) {
Chris@193 91 setSliceableModel(0);
Chris@153 92 }
Chris@133 93 }
Chris@133 94
Chris@198 95 QString
Chris@198 96 SliceLayer::getFeatureDescription(View *v, QPoint &p) const
Chris@198 97 {
Chris@199 98 int minbin, maxbin, range;
Chris@199 99 return getFeatureDescription(v, p, true, minbin, maxbin, range);
Chris@199 100 }
Chris@199 101
Chris@199 102 QString
Chris@199 103 SliceLayer::getFeatureDescription(View *v, QPoint &p,
Chris@199 104 bool includeBinDescription,
Chris@199 105 int &minbin, int &maxbin, int &range) const
Chris@199 106 {
Chris@199 107 minbin = 0;
Chris@199 108 maxbin = 0;
Chris@198 109 if (!m_sliceableModel) return "";
Chris@198 110
Chris@198 111 int xorigin = m_xorigins[v];
Chris@198 112 int w = v->width() - xorigin - 1;
Chris@198 113
Chris@198 114 int mh = m_sliceableModel->getHeight();
Chris@199 115 minbin = getBinForX(p.x() - xorigin, mh, w);
Chris@199 116 maxbin = getBinForX(p.x() - xorigin + 1, mh, w);
Chris@199 117
Chris@199 118 if (minbin >= mh) minbin = mh - 1;
Chris@199 119 if (maxbin >= mh) maxbin = mh - 1;
Chris@199 120 if (minbin < 0) minbin = 0;
Chris@199 121 if (maxbin < 0) maxbin = 0;
Chris@198 122
Chris@198 123 int sampleRate = m_sliceableModel->getSampleRate();
Chris@198 124
Chris@198 125 size_t f0 = m_currentf0;
Chris@198 126 size_t f1 = m_currentf1;
Chris@198 127
Chris@198 128 RealTime rt0 = RealTime::frame2RealTime(f0, sampleRate);
Chris@198 129 RealTime rt1 = RealTime::frame2RealTime(f1, sampleRate);
Chris@198 130
Chris@199 131 range = f1 - f0 + 1;
Chris@198 132
Chris@280 133 QString rtrangestr = QString("%1 s").arg((rt1 - rt0).toText().c_str());
Chris@280 134
Chris@199 135 if (includeBinDescription) {
Chris@198 136
Chris@199 137 float minvalue = 0.f;
Chris@248 138 if (minbin < int(m_values.size())) minvalue = m_values[minbin];
Chris@198 139
Chris@199 140 float maxvalue = minvalue;
Chris@248 141 if (maxbin < int(m_values.size())) maxvalue = m_values[maxbin];
Chris@199 142
Chris@199 143 if (minvalue > maxvalue) std::swap(minvalue, maxvalue);
Chris@199 144
Chris@199 145 QString binstr;
Chris@199 146 if (maxbin != minbin) {
Chris@199 147 binstr = tr("%1 - %2").arg(minbin+1).arg(maxbin+1);
Chris@199 148 } else {
Chris@199 149 binstr = QString("%1").arg(minbin+1);
Chris@199 150 }
Chris@199 151
Chris@199 152 QString valuestr;
Chris@199 153 if (maxvalue != minvalue) {
Chris@199 154 valuestr = tr("%1 - %2").arg(minvalue).arg(maxvalue);
Chris@199 155 } else {
Chris@199 156 valuestr = QString("%1").arg(minvalue);
Chris@199 157 }
Chris@199 158
Chris@280 159 QString description = tr("Time:\t%1 - %2\nRange:\t%3 samples (%4)\nBin:\t%5\n%6 value:\t%7")
Chris@199 160 .arg(QString::fromStdString(rt0.toText(true)))
Chris@199 161 .arg(QString::fromStdString(rt1.toText(true)))
Chris@199 162 .arg(range)
Chris@280 163 .arg(rtrangestr)
Chris@199 164 .arg(binstr)
Chris@199 165 .arg(m_samplingMode == NearestSample ? tr("First") :
Chris@199 166 m_samplingMode == SampleMean ? tr("Mean") : tr("Peak"))
Chris@199 167 .arg(valuestr);
Chris@199 168
Chris@199 169 return description;
Chris@199 170
Chris@199 171 } else {
Chris@199 172
Chris@280 173 QString description = tr("Time:\t%1 - %2\nRange:\t%3 samples (%4)")
Chris@199 174 .arg(QString::fromStdString(rt0.toText(true)))
Chris@199 175 .arg(QString::fromStdString(rt1.toText(true)))
Chris@280 176 .arg(range)
Chris@280 177 .arg(rtrangestr);
Chris@199 178
Chris@199 179 return description;
Chris@199 180 }
Chris@198 181 }
Chris@198 182
Chris@198 183 float
Chris@198 184 SliceLayer::getXForBin(int bin, int count, float w) const
Chris@198 185 {
Chris@198 186 float x = 0;
Chris@198 187
Chris@198 188 switch (m_binScale) {
Chris@198 189
Chris@198 190 case LinearBins:
Chris@198 191 x = (float(w) * bin) / count;
Chris@198 192 break;
Chris@198 193
Chris@198 194 case LogBins:
Chris@198 195 x = (float(w) * log10f(bin + 1)) / log10f(count + 1);
Chris@198 196 break;
Chris@198 197
Chris@198 198 case InvertedLogBins:
Chris@198 199 x = w - (float(w) * log10f(count - bin - 1)) / log10f(count);
Chris@198 200 break;
Chris@198 201 }
Chris@198 202
Chris@198 203 return x;
Chris@198 204 }
Chris@198 205
Chris@198 206 int
Chris@198 207 SliceLayer::getBinForX(float x, int count, float w) const
Chris@198 208 {
Chris@198 209 int bin = 0;
Chris@198 210
Chris@198 211 switch (m_binScale) {
Chris@198 212
Chris@198 213 case LinearBins:
Chris@198 214 bin = int((x * count) / w + 0.0001);
Chris@198 215 break;
Chris@198 216
Chris@198 217 case LogBins:
Chris@198 218 bin = int(powf(10.f, (x * log10f(count + 1)) / w) - 1 + 0.0001);
Chris@198 219 break;
Chris@198 220
Chris@198 221 case InvertedLogBins:
Chris@198 222 bin = count + 1 - int(powf(10.f, (log10f(count) * (w - x)) / float(w)) + 0.0001);
Chris@198 223 break;
Chris@198 224 }
Chris@198 225
Chris@198 226 return bin;
Chris@198 227 }
Chris@198 228
Chris@274 229 float
Chris@274 230 SliceLayer::getYForValue(float value, const View *v, float &norm) const
Chris@274 231 {
Chris@274 232 norm = 0.f;
Chris@274 233
Chris@274 234 if (m_yorigins.find(v) == m_yorigins.end()) return 0;
Chris@274 235
Chris@274 236 value *= m_gain;
Chris@274 237
Chris@274 238 int yorigin = m_yorigins[v];
Chris@274 239 int h = m_heights[v];
Chris@284 240 float thresh = getThresholdDb();
Chris@274 241
Chris@274 242 float y = 0.f;
Chris@274 243
Chris@274 244 if (h <= 0) return y;
Chris@274 245
Chris@274 246 switch (m_energyScale) {
Chris@274 247
Chris@274 248 case dBScale:
Chris@274 249 {
Chris@274 250 float db = thresh;
Chris@274 251 if (value > 0.f) db = 10.f * log10f(value);
Chris@274 252 if (db < thresh) db = thresh;
Chris@274 253 norm = (db - thresh) / -thresh;
Chris@274 254 y = yorigin - (float(h) * norm);
Chris@274 255 break;
Chris@274 256 }
Chris@274 257
Chris@274 258 case MeterScale:
Chris@274 259 y = AudioLevel::multiplier_to_preview(value, h);
Chris@274 260 norm = float(y) / float(h);
Chris@274 261 y = yorigin - y;
Chris@274 262 break;
Chris@274 263
Chris@274 264 default:
Chris@290 265 // std::cerr << "thresh = " << m_threshold << std::endl;
Chris@284 266 norm = (fabsf(value) - m_threshold);
Chris@284 267 if (norm < 0) norm = 0;
Chris@284 268 y = yorigin - (float(h) * norm);
Chris@274 269 break;
Chris@274 270 }
Chris@274 271
Chris@274 272 return y;
Chris@274 273 }
Chris@274 274
Chris@274 275 float
Chris@274 276 SliceLayer::getValueForY(float y, const View *v) const
Chris@274 277 {
Chris@274 278 float value = 0.f;
Chris@274 279
Chris@274 280 if (m_yorigins.find(v) == m_yorigins.end()) return value;
Chris@274 281
Chris@274 282 int yorigin = m_yorigins[v];
Chris@274 283 int h = m_heights[v];
Chris@284 284 float thresh = getThresholdDb();
Chris@274 285
Chris@274 286 if (h <= 0) return value;
Chris@274 287
Chris@274 288 y = yorigin - y;
Chris@274 289
Chris@274 290 switch (m_energyScale) {
Chris@274 291
Chris@274 292 case dBScale:
Chris@274 293 {
Chris@274 294 float db = ((y / h) * -thresh) + thresh;
Chris@274 295 value = powf(10.f, db/10.f);
Chris@274 296 break;
Chris@274 297 }
Chris@274 298
Chris@274 299 case MeterScale:
Chris@274 300 value = AudioLevel::preview_to_multiplier(lrintf(y), h);
Chris@274 301 break;
Chris@274 302
Chris@274 303 default:
Chris@284 304 value = y / h + m_threshold;
Chris@274 305 }
Chris@274 306
Chris@274 307 return value / m_gain;
Chris@274 308 }
Chris@274 309
Chris@133 310 void
Chris@193 311 SliceLayer::paint(View *v, QPainter &paint, QRect rect) const
Chris@133 312 {
Chris@254 313 if (!m_sliceableModel || !m_sliceableModel->isOK() ||
Chris@254 314 !m_sliceableModel->isReady()) return;
Chris@133 315
Chris@195 316 paint.save();
Chris@195 317 paint.setRenderHint(QPainter::Antialiasing, false);
Chris@195 318
Chris@195 319 if (v->getViewManager() && v->getViewManager()->shouldShowScaleGuides()) {
Chris@195 320 if (!m_scalePoints.empty()) {
Chris@195 321 paint.setPen(QColor(240, 240, 240)); //!!! and dark background?
Chris@195 322 for (size_t i = 0; i < m_scalePoints.size(); ++i) {
Chris@195 323 paint.drawLine(0, m_scalePoints[i], rect.width(), m_scalePoints[i]);
Chris@195 324 }
Chris@195 325 }
Chris@195 326 }
Chris@195 327
Chris@287 328 paint.setPen(getBaseQColor());
Chris@195 329
Chris@274 330 int xorigin = getVerticalScaleWidth(v, paint) + 1;
Chris@195 331 int w = v->width() - xorigin - 1;
Chris@198 332
Chris@198 333 m_xorigins[v] = xorigin; // for use in getFeatureDescription
Chris@133 334
Chris@195 335 int yorigin = v->height() - 20 - paint.fontMetrics().height() - 7;
Chris@195 336 int h = yorigin - paint.fontMetrics().height() - 8;
Chris@133 337
Chris@274 338 m_yorigins[v] = yorigin; // for getYForValue etc
Chris@274 339 m_heights[v] = h;
Chris@274 340
Chris@274 341 if (h <= 0) return;
Chris@274 342
Chris@133 343 QPainterPath path;
Chris@133 344
Chris@248 345 size_t mh = m_sliceableModel->getHeight();
Chris@133 346
Chris@193 347 int divisor = 0;
Chris@193 348
Chris@198 349 m_values.clear();
Chris@193 350 for (size_t bin = 0; bin < mh; ++bin) {
Chris@198 351 m_values.push_back(0.f);
Chris@193 352 }
Chris@193 353
Chris@193 354 size_t f0 = v->getCentreFrame();
Chris@193 355 int f0x = v->getXForFrame(f0);
Chris@195 356 f0 = v->getFrameForX(f0x);
Chris@193 357 size_t f1 = v->getFrameForX(f0x + 1);
Chris@195 358 if (f1 > f0) --f1;
Chris@193 359
Chris@284 360 // std::cerr << "centre frame " << v->getCentreFrame() << ", x " << f0x << ", f0 " << f0 << ", f1 " << f1 << std::endl;
Chris@274 361
Chris@274 362 size_t res = m_sliceableModel->getResolution();
Chris@274 363 size_t col0 = f0 / res;
Chris@193 364 size_t col1 = col0;
Chris@274 365 if (m_samplingMode != NearestSample) col1 = f1 / res;
Chris@274 366 f0 = col0 * res;
Chris@274 367 f1 = (col1 + 1) * res - 1;
Chris@274 368
Chris@284 369 // std::cerr << "resolution " << res << ", col0 " << col0 << ", col1 " << col1 << ", f0 " << f0 << ", f1 " << f1 << std::endl;
Chris@193 370
Chris@198 371 m_currentf0 = f0;
Chris@198 372 m_currentf1 = f1;
Chris@198 373
Chris@254 374 BiasCurve curve;
Chris@254 375 getBiasCurve(curve);
Chris@254 376 size_t cs = curve.size();
Chris@254 377
Chris@195 378 for (size_t col = col0; col <= col1; ++col) {
Chris@193 379 for (size_t bin = 0; bin < mh; ++bin) {
Chris@193 380 float value = m_sliceableModel->getValueAt(col, bin);
Chris@254 381 if (bin < cs) value *= curve[bin];
Chris@193 382 if (m_samplingMode == SamplePeak) {
Chris@198 383 if (value > m_values[bin]) m_values[bin] = value;
Chris@193 384 } else {
Chris@198 385 m_values[bin] += value;
Chris@193 386 }
Chris@153 387 }
Chris@193 388 ++divisor;
Chris@193 389 }
Chris@193 390
Chris@193 391 float max = 0.f;
Chris@193 392 for (size_t bin = 0; bin < mh; ++bin) {
Chris@198 393 if (m_samplingMode == SampleMean) m_values[bin] /= divisor;
Chris@198 394 if (m_values[bin] > max) max = m_values[bin];
Chris@193 395 }
Chris@193 396 if (max != 0.f && m_normalize) {
Chris@193 397 for (size_t bin = 0; bin < mh; ++bin) {
Chris@198 398 m_values[bin] /= max;
Chris@193 399 }
Chris@193 400 }
Chris@193 401
Chris@193 402 float py = 0;
Chris@193 403 float nx = xorigin;
Chris@193 404
Chris@197 405 ColourMapper mapper(m_colourMap, 0, 1);
Chris@197 406
Chris@193 407 for (size_t bin = 0; bin < mh; ++bin) {
Chris@193 408
Chris@198 409 float x = nx;
Chris@198 410 nx = xorigin + getXForBin(bin + 1, mh, w);
Chris@193 411
Chris@198 412 float value = m_values[bin];
Chris@197 413 float norm = 0.f;
Chris@274 414 float y = getYForValue(value, v, norm);
Chris@133 415
Chris@193 416 if (m_plotStyle == PlotLines) {
Chris@193 417
Chris@193 418 if (bin == 0) {
Chris@193 419 path.moveTo(x, y);
Chris@193 420 } else {
Chris@193 421 path.lineTo(x, y);
Chris@193 422 }
Chris@193 423
Chris@193 424 } else if (m_plotStyle == PlotSteps) {
Chris@193 425
Chris@193 426 if (bin == 0) {
Chris@193 427 path.moveTo(x, y);
Chris@193 428 } else {
Chris@193 429 path.lineTo(x, y);
Chris@193 430 }
Chris@193 431 path.lineTo(nx, y);
Chris@193 432
Chris@193 433 } else if (m_plotStyle == PlotBlocks) {
Chris@193 434
Chris@193 435 path.moveTo(x, yorigin);
Chris@133 436 path.lineTo(x, y);
Chris@193 437 path.lineTo(nx, y);
Chris@193 438 path.lineTo(nx, yorigin);
Chris@193 439 path.lineTo(x, yorigin);
Chris@197 440
Chris@197 441 } else if (m_plotStyle == PlotFilledBlocks) {
Chris@197 442
Chris@197 443 paint.fillRect(QRectF(x, y, nx - x, yorigin - y), mapper.map(norm));
Chris@133 444 }
Chris@193 445
Chris@193 446 py = y;
Chris@133 447 }
Chris@133 448
Chris@197 449 if (m_plotStyle != PlotFilledBlocks) {
Chris@197 450 paint.drawPath(path);
Chris@197 451 }
Chris@133 452 paint.restore();
Chris@198 453 /*
Chris@197 454 QPoint discard;
Chris@197 455
Chris@197 456 if (v->getViewManager() && v->getViewManager()->shouldShowFrameCount() &&
Chris@197 457 v->shouldIlluminateLocalFeatures(this, discard)) {
Chris@195 458
Chris@195 459 int sampleRate = m_sliceableModel->getSampleRate();
Chris@195 460
Chris@195 461 QString startText = QString("%1 / %2")
Chris@195 462 .arg(QString::fromStdString
Chris@195 463 (RealTime::frame2RealTime
Chris@195 464 (f0, sampleRate).toText(true)))
Chris@195 465 .arg(f0);
Chris@195 466
Chris@195 467 QString endText = QString(" %1 / %2")
Chris@195 468 .arg(QString::fromStdString
Chris@195 469 (RealTime::frame2RealTime
Chris@195 470 (f1, sampleRate).toText(true)))
Chris@195 471 .arg(f1);
Chris@195 472
Chris@195 473 QString durationText = QString("(%1 / %2) ")
Chris@195 474 .arg(QString::fromStdString
Chris@195 475 (RealTime::frame2RealTime
Chris@195 476 (f1 - f0 + 1, sampleRate).toText(true)))
Chris@195 477 .arg(f1 - f0 + 1);
Chris@195 478
Chris@195 479 v->drawVisibleText
Chris@195 480 (paint, xorigin + 5,
Chris@195 481 paint.fontMetrics().ascent() + 5,
Chris@195 482 startText, View::OutlinedText);
Chris@195 483
Chris@195 484 v->drawVisibleText
Chris@195 485 (paint, xorigin + 5,
Chris@195 486 paint.fontMetrics().ascent() + paint.fontMetrics().height() + 10,
Chris@195 487 endText, View::OutlinedText);
Chris@195 488
Chris@195 489 v->drawVisibleText
Chris@195 490 (paint, xorigin + 5,
Chris@195 491 paint.fontMetrics().ascent() + 2*paint.fontMetrics().height() + 15,
Chris@195 492 durationText, View::OutlinedText);
Chris@195 493 }
Chris@195 494 */
Chris@195 495 }
Chris@195 496
Chris@195 497 int
Chris@248 498 SliceLayer::getVerticalScaleWidth(View *, QPainter &paint) const
Chris@195 499 {
Chris@195 500 if (m_energyScale == LinearScale) {
Chris@220 501 return std::max(paint.fontMetrics().width("0.0") + 13,
Chris@220 502 paint.fontMetrics().width("x10-10"));
Chris@195 503 } else {
Chris@195 504 return std::max(paint.fontMetrics().width(tr("0dB")),
Chris@195 505 paint.fontMetrics().width(tr("-Inf"))) + 13;
Chris@195 506 }
Chris@195 507 }
Chris@195 508
Chris@195 509 void
Chris@195 510 SliceLayer::paintVerticalScale(View *v, QPainter &paint, QRect rect) const
Chris@195 511 {
Chris@284 512 float thresh = m_threshold;
Chris@195 513 if (m_energyScale != LinearScale) {
Chris@284 514 thresh = AudioLevel::dB_to_multiplier(getThresholdDb());
Chris@195 515 }
Chris@195 516
Chris@195 517 // int h = (rect.height() * 3) / 4;
Chris@195 518 // int y = (rect.height() / 2) - (h / 2);
Chris@195 519
Chris@195 520 int yorigin = v->height() - 20 - paint.fontMetrics().height() - 6;
Chris@195 521 int h = yorigin - paint.fontMetrics().height() - 8;
Chris@195 522 if (h < 0) return;
Chris@195 523
Chris@195 524 QRect actual(rect.x(), rect.y() + yorigin - h, rect.width(), h);
Chris@195 525
Chris@220 526 int mult = 1;
Chris@220 527
Chris@195 528 PaintAssistant::paintVerticalLevelScale
Chris@195 529 (paint, actual, thresh, 1.0 / m_gain,
Chris@195 530 PaintAssistant::Scale(m_energyScale),
Chris@220 531 mult,
Chris@195 532 const_cast<std::vector<int> *>(&m_scalePoints));
Chris@220 533
Chris@220 534 if (mult != 1 && mult != 0) {
Chris@220 535 int log = lrintf(log10f(mult));
Chris@220 536 QString a = tr("x10");
Chris@220 537 QString b = QString("%1").arg(-log);
Chris@220 538 paint.drawText(3, 8 + paint.fontMetrics().ascent(), a);
Chris@220 539 paint.drawText(3 + paint.fontMetrics().width(a),
Chris@220 540 3 + paint.fontMetrics().ascent(), b);
Chris@220 541 }
Chris@133 542 }
Chris@133 543
Chris@153 544 Layer::PropertyList
Chris@193 545 SliceLayer::getProperties() const
Chris@153 546 {
Chris@287 547 PropertyList list = SingleColourLayer::getProperties();
Chris@193 548 list.push_back("Plot Type");
Chris@198 549 // list.push_back("Sampling Mode");
Chris@153 550 list.push_back("Scale");
Chris@153 551 list.push_back("Normalize");
Chris@284 552 list.push_back("Threshold");
Chris@153 553 list.push_back("Gain");
Chris@193 554 list.push_back("Bin Scale");
Chris@153 555
Chris@153 556 return list;
Chris@153 557 }
Chris@153 558
Chris@153 559 QString
Chris@193 560 SliceLayer::getPropertyLabel(const PropertyName &name) const
Chris@153 561 {
Chris@193 562 if (name == "Plot Type") return tr("Plot Type");
Chris@290 563 if (name == "Scale") return tr("Scale");
Chris@153 564 if (name == "Normalize") return tr("Normalize");
Chris@284 565 if (name == "Threshold") return tr("Threshold");
Chris@153 566 if (name == "Gain") return tr("Gain");
Chris@193 567 if (name == "Sampling Mode") return tr("Sampling Mode");
Chris@193 568 if (name == "Bin Scale") return tr("Plot X Scale");
Chris@287 569 return SingleColourLayer::getPropertyLabel(name);
Chris@153 570 }
Chris@153 571
Chris@153 572 Layer::PropertyType
Chris@193 573 SliceLayer::getPropertyType(const PropertyName &name) const
Chris@153 574 {
Chris@153 575 if (name == "Gain") return RangeProperty;
Chris@153 576 if (name == "Normalize") return ToggleProperty;
Chris@284 577 if (name == "Threshold") return RangeProperty;
Chris@287 578 if (name == "Plot Type") return ValueProperty;
Chris@290 579 if (name == "Scale") return ValueProperty;
Chris@287 580 if (name == "Sampling Mode") return ValueProperty;
Chris@287 581 if (name == "Bin Scale") return ValueProperty;
Chris@287 582 if (name == "Colour" && m_plotStyle == PlotFilledBlocks) return ValueProperty;
Chris@287 583 return SingleColourLayer::getPropertyType(name);
Chris@153 584 }
Chris@153 585
Chris@153 586 QString
Chris@193 587 SliceLayer::getPropertyGroupName(const PropertyName &name) const
Chris@153 588 {
Chris@153 589 if (name == "Scale" ||
Chris@153 590 name == "Normalize" ||
Chris@193 591 name == "Sampling Mode" ||
Chris@284 592 name == "Threshold" ||
Chris@193 593 name == "Gain") return tr("Scale");
Chris@193 594 if (name == "Plot Type" ||
Chris@193 595 name == "Bin Scale") return tr("Plot Type");
Chris@287 596 return SingleColourLayer::getPropertyGroupName(name);
Chris@153 597 }
Chris@153 598
Chris@153 599 int
Chris@193 600 SliceLayer::getPropertyRangeAndValue(const PropertyName &name,
Chris@216 601 int *min, int *max, int *deflt) const
Chris@153 602 {
Chris@216 603 int val = 0;
Chris@153 604
Chris@216 605 int garbage0, garbage1, garbage2;
Chris@153 606 if (!min) min = &garbage0;
Chris@153 607 if (!max) max = &garbage1;
Chris@248 608 if (!deflt) deflt = &garbage2;
Chris@153 609
Chris@153 610 if (name == "Gain") {
Chris@153 611
Chris@153 612 *min = -50;
Chris@153 613 *max = 50;
Chris@216 614 *deflt = 0;
Chris@153 615
Chris@193 616 std::cerr << "gain is " << m_gain << ", mode is " << m_samplingMode << std::endl;
Chris@193 617
Chris@216 618 val = lrint(log10(m_gain) * 20.0);
Chris@216 619 if (val < *min) val = *min;
Chris@216 620 if (val > *max) val = *max;
Chris@153 621
Chris@284 622 } else if (name == "Threshold") {
Chris@284 623
Chris@284 624 *min = -80;
Chris@284 625 *max = 0;
Chris@284 626
Chris@284 627 *deflt = lrintf(AudioLevel::multiplier_to_dB(m_initialThreshold));
Chris@284 628 if (*deflt < *min) *deflt = *min;
Chris@284 629 if (*deflt > *max) *deflt = *max;
Chris@284 630
Chris@284 631 val = lrintf(AudioLevel::multiplier_to_dB(m_threshold));
Chris@284 632 if (val < *min) val = *min;
Chris@284 633 if (val > *max) val = *max;
Chris@284 634
Chris@153 635 } else if (name == "Normalize") {
Chris@153 636
Chris@216 637 val = (m_normalize ? 1 : 0);
Chris@216 638 *deflt = 0;
Chris@153 639
Chris@287 640 } else if (name == "Colour" && m_plotStyle == PlotFilledBlocks) {
Chris@197 641
Chris@287 642 *min = 0;
Chris@287 643 *max = ColourMapper::getColourMapCount() - 1;
Chris@287 644 *deflt = 0;
Chris@287 645
Chris@287 646 val = m_colourMap;
Chris@153 647
Chris@153 648 } else if (name == "Scale") {
Chris@153 649
Chris@153 650 *min = 0;
Chris@153 651 *max = 2;
Chris@216 652 *deflt = (int)dBScale;
Chris@153 653
Chris@216 654 val = (int)m_energyScale;
Chris@153 655
Chris@193 656 } else if (name == "Sampling Mode") {
Chris@153 657
Chris@153 658 *min = 0;
Chris@193 659 *max = 2;
Chris@216 660 *deflt = (int)SampleMean;
Chris@193 661
Chris@216 662 val = (int)m_samplingMode;
Chris@153 663
Chris@193 664 } else if (name == "Plot Type") {
Chris@193 665
Chris@193 666 *min = 0;
Chris@197 667 *max = 3;
Chris@216 668 *deflt = (int)PlotSteps;
Chris@193 669
Chris@216 670 val = (int)m_plotStyle;
Chris@193 671
Chris@193 672 } else if (name == "Bin Scale") {
Chris@193 673
Chris@193 674 *min = 0;
Chris@198 675 *max = 2;
Chris@216 676 *deflt = (int)LinearBins;
Chris@198 677 // *max = 1; // I don't think we really do want to offer inverted log
Chris@193 678
Chris@216 679 val = (int)m_binScale;
Chris@193 680
Chris@153 681 } else {
Chris@287 682 val = SingleColourLayer::getPropertyRangeAndValue(name, min, max, deflt);
Chris@153 683 }
Chris@153 684
Chris@216 685 return val;
Chris@153 686 }
Chris@153 687
Chris@153 688 QString
Chris@193 689 SliceLayer::getPropertyValueLabel(const PropertyName &name,
Chris@153 690 int value) const
Chris@153 691 {
Chris@287 692 if (name == "Colour" && m_plotStyle == PlotFilledBlocks) {
Chris@287 693 return ColourMapper::getColourMapName(value);
Chris@153 694 }
Chris@153 695 if (name == "Scale") {
Chris@153 696 switch (value) {
Chris@153 697 default:
Chris@153 698 case 0: return tr("Linear");
Chris@153 699 case 1: return tr("Meter");
Chris@153 700 case 2: return tr("dB");
Chris@153 701 }
Chris@153 702 }
Chris@193 703 if (name == "Sampling Mode") {
Chris@153 704 switch (value) {
Chris@153 705 default:
Chris@193 706 case 0: return tr("Any");
Chris@193 707 case 1: return tr("Mean");
Chris@193 708 case 2: return tr("Peak");
Chris@193 709 }
Chris@193 710 }
Chris@193 711 if (name == "Plot Type") {
Chris@193 712 switch (value) {
Chris@193 713 default:
Chris@193 714 case 0: return tr("Lines");
Chris@193 715 case 1: return tr("Steps");
Chris@193 716 case 2: return tr("Blocks");
Chris@197 717 case 3: return tr("Colours");
Chris@193 718 }
Chris@193 719 }
Chris@193 720 if (name == "Bin Scale") {
Chris@193 721 switch (value) {
Chris@193 722 default:
Chris@198 723 case 0: return tr("Linear Bins");
Chris@198 724 case 1: return tr("Log Bins");
Chris@198 725 case 2: return tr("Rev Log Bins");
Chris@153 726 }
Chris@153 727 }
Chris@287 728 return SingleColourLayer::getPropertyValueLabel(name, value);
Chris@153 729 }
Chris@153 730
Chris@167 731 RangeMapper *
Chris@193 732 SliceLayer::getNewPropertyRangeMapper(const PropertyName &name) const
Chris@167 733 {
Chris@167 734 if (name == "Gain") {
Chris@167 735 return new LinearRangeMapper(-50, 50, -25, 25, tr("dB"));
Chris@167 736 }
Chris@284 737 if (name == "Threshold") {
Chris@284 738 return new LinearRangeMapper(-80, 0, -80, 0, tr("dB"));
Chris@284 739 }
Chris@287 740 return SingleColourLayer::getNewPropertyRangeMapper(name);
Chris@167 741 }
Chris@167 742
Chris@133 743 void
Chris@193 744 SliceLayer::setProperty(const PropertyName &name, int value)
Chris@133 745 {
Chris@153 746 if (name == "Gain") {
Chris@153 747 setGain(pow(10, float(value)/20.0));
Chris@284 748 } else if (name == "Threshold") {
Chris@284 749 if (value == -80) setThreshold(0.0);
Chris@284 750 else setThreshold(AudioLevel::dB_to_multiplier(value));
Chris@287 751 } else if (name == "Colour" && m_plotStyle == PlotFilledBlocks) {
Chris@287 752 setFillColourMap(value);
Chris@153 753 } else if (name == "Scale") {
Chris@153 754 switch (value) {
Chris@153 755 default:
Chris@153 756 case 0: setEnergyScale(LinearScale); break;
Chris@153 757 case 1: setEnergyScale(MeterScale); break;
Chris@153 758 case 2: setEnergyScale(dBScale); break;
Chris@153 759 }
Chris@193 760 } else if (name == "Plot Type") {
Chris@193 761 setPlotStyle(PlotStyle(value));
Chris@193 762 } else if (name == "Sampling Mode") {
Chris@193 763 switch (value) {
Chris@193 764 default:
Chris@193 765 case 0: setSamplingMode(NearestSample); break;
Chris@193 766 case 1: setSamplingMode(SampleMean); break;
Chris@193 767 case 2: setSamplingMode(SamplePeak); break;
Chris@193 768 }
Chris@193 769 } else if (name == "Bin Scale") {
Chris@193 770 switch (value) {
Chris@193 771 default:
Chris@193 772 case 0: setBinScale(LinearBins); break;
Chris@193 773 case 1: setBinScale(LogBins); break;
Chris@193 774 case 2: setBinScale(InvertedLogBins); break;
Chris@193 775 }
Chris@153 776 } else if (name == "Normalize") {
Chris@153 777 setNormalize(value ? true : false);
Chris@287 778 } else {
Chris@287 779 SingleColourLayer::setProperty(name, value);
Chris@153 780 }
Chris@153 781 }
Chris@153 782
Chris@153 783 void
Chris@197 784 SliceLayer::setFillColourMap(int map)
Chris@197 785 {
Chris@197 786 if (m_colourMap == map) return;
Chris@197 787 m_colourMap = map;
Chris@197 788 emit layerParametersChanged();
Chris@197 789 }
Chris@197 790
Chris@197 791 void
Chris@193 792 SliceLayer::setEnergyScale(EnergyScale scale)
Chris@153 793 {
Chris@153 794 if (m_energyScale == scale) return;
Chris@153 795 m_energyScale = scale;
Chris@153 796 emit layerParametersChanged();
Chris@153 797 }
Chris@153 798
Chris@153 799 void
Chris@193 800 SliceLayer::setSamplingMode(SamplingMode mode)
Chris@153 801 {
Chris@193 802 if (m_samplingMode == mode) return;
Chris@193 803 m_samplingMode = mode;
Chris@153 804 emit layerParametersChanged();
Chris@153 805 }
Chris@153 806
Chris@153 807 void
Chris@193 808 SliceLayer::setPlotStyle(PlotStyle style)
Chris@153 809 {
Chris@193 810 if (m_plotStyle == style) return;
Chris@197 811 bool colourTypeChanged = (style == PlotFilledBlocks ||
Chris@197 812 m_plotStyle == PlotFilledBlocks);
Chris@193 813 m_plotStyle = style;
Chris@197 814 if (colourTypeChanged) {
Chris@197 815 emit layerParameterRangesChanged();
Chris@197 816 }
Chris@153 817 emit layerParametersChanged();
Chris@153 818 }
Chris@153 819
Chris@153 820 void
Chris@193 821 SliceLayer::setBinScale(BinScale scale)
Chris@153 822 {
Chris@193 823 if (m_binScale == scale) return;
Chris@193 824 m_binScale = scale;
Chris@153 825 emit layerParametersChanged();
Chris@153 826 }
Chris@153 827
Chris@153 828 void
Chris@193 829 SliceLayer::setNormalize(bool n)
Chris@153 830 {
Chris@153 831 if (m_normalize == n) return;
Chris@153 832 m_normalize = n;
Chris@153 833 emit layerParametersChanged();
Chris@153 834 }
Chris@153 835
Chris@153 836 void
Chris@284 837 SliceLayer::setThreshold(float thresh)
Chris@284 838 {
Chris@284 839 if (m_threshold == thresh) return;
Chris@284 840 m_threshold = thresh;
Chris@284 841 emit layerParametersChanged();
Chris@284 842 }
Chris@284 843
Chris@284 844 void
Chris@193 845 SliceLayer::setGain(float gain)
Chris@153 846 {
Chris@153 847 if (m_gain == gain) return;
Chris@153 848 m_gain = gain;
Chris@153 849 emit layerParametersChanged();
Chris@153 850 }
Chris@153 851
Chris@284 852 float
Chris@284 853 SliceLayer::getThresholdDb() const
Chris@284 854 {
Chris@284 855 if (m_threshold == 0.0) return -80.f;
Chris@284 856 float db = AudioLevel::multiplier_to_dB(m_threshold);
Chris@284 857 return db;
Chris@284 858 }
Chris@284 859
Chris@287 860 int
Chris@287 861 SliceLayer::getDefaultColourHint(bool darkbg, bool &impose)
Chris@287 862 {
Chris@287 863 impose = false;
Chris@287 864 return ColourDatabase::getInstance()->getColourIndex
Chris@287 865 (QString(darkbg ? "Bright Blue" : "Blue"));
Chris@287 866 }
Chris@287 867
Chris@316 868 void
Chris@316 869 SliceLayer::toXml(QTextStream &stream,
Chris@316 870 QString indent, QString extraAttributes) const
Chris@153 871 {
Chris@153 872 QString s;
Chris@153 873
Chris@287 874 s += QString("colourScheme=\"%1\" "
Chris@287 875 "energyScale=\"%2\" "
Chris@287 876 "samplingMode=\"%3\" "
Chris@287 877 "gain=\"%4\" "
Chris@287 878 "normalize=\"%5\"")
Chris@197 879 .arg(m_colourMap)
Chris@153 880 .arg(m_energyScale)
Chris@193 881 .arg(m_samplingMode)
Chris@153 882 .arg(m_gain)
Chris@153 883 .arg(m_normalize ? "true" : "false");
Chris@153 884
Chris@316 885 SingleColourLayer::toXml(stream, indent, extraAttributes + " " + s);
Chris@153 886 }
Chris@153 887
Chris@153 888 void
Chris@193 889 SliceLayer::setProperties(const QXmlAttributes &attributes)
Chris@153 890 {
Chris@153 891 bool ok = false;
Chris@153 892
Chris@287 893 SingleColourLayer::setProperties(attributes);
Chris@153 894
Chris@153 895 EnergyScale scale = (EnergyScale)
Chris@153 896 attributes.value("energyScale").toInt(&ok);
Chris@153 897 if (ok) setEnergyScale(scale);
Chris@153 898
Chris@193 899 SamplingMode mode = (SamplingMode)
Chris@193 900 attributes.value("samplingMode").toInt(&ok);
Chris@193 901 if (ok) setSamplingMode(mode);
Chris@153 902
Chris@197 903 int colourMap = attributes.value("colourScheme").toInt(&ok);
Chris@197 904 if (ok) setFillColourMap(colourMap);
Chris@197 905
Chris@153 906 float gain = attributes.value("gain").toFloat(&ok);
Chris@153 907 if (ok) setGain(gain);
Chris@153 908
Chris@153 909 bool normalize = (attributes.value("normalize").trimmed() == "true");
Chris@153 910 setNormalize(normalize);
Chris@133 911 }
Chris@133 912
Chris@133 913 bool
Chris@248 914 SliceLayer::getValueExtents(float &, float &, bool &, QString &) const
Chris@133 915 {
Chris@133 916 return false;
Chris@133 917 }
Chris@133 918