comparison layer/TimeValueLayer.cpp @ 437:755243c67f59

* Add vertical zoom and pan to time-value layer. Still some defects, particularly in logarithmic mode. Now need to get this in note layer as well! * Some fixes to log colouring in segmentation mode of time-value layer.
author Chris Cannam
date Fri, 24 Oct 2008 14:52:40 +0000 (2008-10-24)
parents 6324461340b1
children 76cd1c89eb06
comparison
equal deleted inserted replaced
436:1e0744b2cebe 437:755243c67f59
17 17
18 #include "data/model/Model.h" 18 #include "data/model/Model.h"
19 #include "base/RealTime.h" 19 #include "base/RealTime.h"
20 #include "base/Profiler.h" 20 #include "base/Profiler.h"
21 #include "base/LogRange.h" 21 #include "base/LogRange.h"
22 #include "base/RangeMapper.h"
22 #include "ColourDatabase.h" 23 #include "ColourDatabase.h"
23 #include "view/View.h" 24 #include "view/View.h"
24 25
25 #include "data/model/SparseTimeValueModel.h" 26 #include "data/model/SparseTimeValueModel.h"
26 #include "data/model/Labeller.h" 27 #include "data/model/Labeller.h"
49 m_originalPoint(0, 0.0, tr("New Point")), 50 m_originalPoint(0, 0.0, tr("New Point")),
50 m_editingPoint(0, 0.0, tr("New Point")), 51 m_editingPoint(0, 0.0, tr("New Point")),
51 m_editingCommand(0), 52 m_editingCommand(0),
52 m_colourMap(0), 53 m_colourMap(0),
53 m_plotStyle(PlotConnectedPoints), 54 m_plotStyle(PlotConnectedPoints),
54 m_verticalScale(AutoAlignScale) 55 m_verticalScale(AutoAlignScale),
56 m_scaleMinimum(0),
57 m_scaleMaximum(0)
55 { 58 {
56 59
57 } 60 }
58 61
59 void 62 void
258 bool 261 bool
259 TimeValueLayer::getDisplayExtents(float &min, float &max) const 262 TimeValueLayer::getDisplayExtents(float &min, float &max) const
260 { 263 {
261 if (!m_model || shouldAutoAlign()) return false; 264 if (!m_model || shouldAutoAlign()) return false;
262 265
263 min = m_model->getValueMinimum(); 266 if (m_scaleMinimum == m_scaleMaximum) {
264 max = m_model->getValueMaximum(); 267 m_scaleMinimum = m_model->getValueMinimum();
268 m_scaleMaximum = m_model->getValueMaximum();
269 }
270
271 min = m_scaleMinimum;
272 max = m_scaleMaximum;
273
274 // std::cerr << "TimeValueLayer::getDisplayExtents: min = " << min << ", max = " << max << std::endl;
275
265 return true; 276 return true;
277 }
278
279 bool
280 TimeValueLayer::setDisplayExtents(float min, float max)
281 {
282 if (!m_model) return false;
283
284 if (min == max) {
285 if (min == 0.f) {
286 max = 1.f;
287 } else {
288 max = min * 1.0001;
289 }
290 }
291
292 m_scaleMinimum = min;
293 m_scaleMaximum = max;
294
295 std::cerr << "TimeValueLayer::setDisplayExtents: min = " << min << ", max = " << max << std::endl;
296
297 emit layerParametersChanged();
298 return true;
299 }
300
301 int
302 TimeValueLayer::getVerticalZoomSteps(int &defaultStep) const
303 {
304 if (shouldAutoAlign()) return 0;
305 if (!m_model) return 0;
306
307 defaultStep = 100;
308 return 100;
309 }
310
311 int
312 TimeValueLayer::getCurrentVerticalZoomStep() const
313 {
314 if (shouldAutoAlign()) return 0;
315 if (!m_model) return 0;
316
317 RangeMapper *mapper = getNewVerticalZoomRangeMapper();
318 if (!mapper) return 0;
319
320 float dmin, dmax;
321 getDisplayExtents(dmin, dmax);
322
323 int nr = mapper->getPositionForValue(dmax - dmin);
324 /*
325 int n0 = mapper->getPositionForValue(dmax);
326 int n1 = mapper->getPositionForValue(dmin);
327 int nr = n1 - n0;
328 if (nr < 0) nr = -nr;
329
330 std::cerr << "TimeValueLayer::getCurrentVerticalZoomStep: dmin = " << dmin << ", dmax = " << dmax << ", n0 = " << n0 << ", n1 = " << n1 << ", nr = " << nr << std::endl;
331 */
332 delete mapper;
333
334 return 100 - nr;
335 }
336
337 void
338 TimeValueLayer::setVerticalZoomStep(int step)
339 {
340 if (shouldAutoAlign()) return;
341 if (!m_model) return;
342
343 RangeMapper *mapper = getNewVerticalZoomRangeMapper();
344 if (!mapper) return;
345
346 float min, max;
347 bool logarithmic;
348 QString unit;
349 getValueExtents(min, max, logarithmic, unit);
350
351 float dmin, dmax;
352 getDisplayExtents(dmin, dmax);
353
354 float newdist = mapper->getValueForPosition(100 - step);
355
356 float newmin, newmax;
357
358 if (logarithmic) {
359
360 // see SpectrogramLayer::setVerticalZoomStep
361
362 newmax = (newdist + sqrtf(newdist*newdist + 4*dmin*dmax)) / 2;
363 newmin = newmax - newdist;
364
365 // std::cerr << "newmin = " << newmin << ", newmax = " << newmax << std::endl;
366
367 } else {
368 float dmid = (dmax + dmin) / 2;
369 newmin = dmid - newdist / 2;
370 newmax = dmid + newdist / 2;
371 }
372
373 if (newmin < min) {
374 newmax += (min - newmin);
375 newmin = min;
376 }
377 if (newmax > max) {
378 newmax = max;
379 }
380
381 std::cerr << "TimeValueLayer::setVerticalZoomStep: " << step << ": " << newmin << " -> " << newmax << " (range " << newdist << ")" << std::endl;
382
383 setDisplayExtents(newmin, newmax);
384 }
385
386 RangeMapper *
387 TimeValueLayer::getNewVerticalZoomRangeMapper() const
388 {
389 if (!m_model) return 0;
390
391 RangeMapper *mapper;
392
393 float min, max;
394 bool logarithmic;
395 QString unit;
396 getValueExtents(min, max, logarithmic, unit);
397
398 if (min == max) return 0;
399
400 if (logarithmic) {
401 mapper = new LogRangeMapper(0, 100, min, max, unit);
402 } else {
403 mapper = new LinearRangeMapper(0, 100, min, max, unit);
404 }
405
406 return mapper;
266 } 407 }
267 408
268 SparseTimeValueModel::PointList 409 SparseTimeValueModel::PointList
269 TimeValueLayer::getLocalPoints(View *v, int x) const 410 TimeValueLayer::getLocalPoints(View *v, int x) const
270 { 411 {
444 min = -1.0; 585 min = -1.0;
445 max = 1.0; 586 max = 1.0;
446 587
447 } else { 588 } else {
448 589
449 min = m_model->getValueMinimum(); 590 getDisplayExtents(min, max);
450 max = m_model->getValueMaximum();
451 591
452 if (m_verticalScale == LogScale) { 592 if (m_verticalScale == LogScale) {
453 LogRange::mapRange(min, max); 593 LogRange::mapRange(min, max);
454 log = true; 594 log = true;
455 } 595 }
487 getScaleExtents(v, min, max, logarithmic); 627 getScaleExtents(v, min, max, logarithmic);
488 628
489 float val = min + (float(h - y) * float(max - min)) / h; 629 float val = min + (float(h - y) * float(max - min)) / h;
490 630
491 if (logarithmic) { 631 if (logarithmic) {
492 val = powf(10.f, val); 632 val = LogRange::map(val);
493 } 633 }
494 634
495 return val; 635 return val;
496 } 636 }
497 637
512 652
513 if (min > max) std::swap(min, max); 653 if (min > max) std::swap(min, max);
514 if (max == min) max = min + 1; 654 if (max == min) max = min + 1;
515 655
516 if (log) { 656 if (log) {
517 LogRange::mapRange(min, max);
518 val = LogRange::map(val); 657 val = LogRange::map(val);
519 } 658 }
520 659
521 // std::cerr << "TimeValueLayer::getColourForValue: min " << min << ", max " 660 // std::cerr << "TimeValueLayer::getColourForValue: min " << min << ", max "
522 // << max << ", log " << log << ", value " << val << std::endl; 661 // << max << ", log " << log << ", value " << val << std::endl;
779 918
780 int h = v->height(); 919 int h = v->height();
781 920
782 int n = 10; 921 int n = 10;
783 922
784 float max = m_model->getValueMaximum(); 923 float min, max;
785 float min = m_model->getValueMinimum(); 924 bool logarithmic;
925 getScaleExtents(v, min, max, logarithmic);
926
927 if (m_plotStyle == PlotSegmentation) {
928 QString unit;
929 getValueExtents(min, max, logarithmic, unit);
930 if (logarithmic) {
931 LogRange::mapRange(min, max);
932 }
933 }
934
786 float val = min; 935 float val = min;
787 float inc = (max - val) / n; 936 float inc = (max - val) / n;
788 937
789 char buffer[40]; 938 char buffer[40];
790 939
805 954
806 if (m_plotStyle == PlotSegmentation) { 955 if (m_plotStyle == PlotSegmentation) {
807 paint.save(); 956 paint.save();
808 for (int y = 0; y < boxh; ++y) { 957 for (int y = 0; y < boxh; ++y) {
809 float val = ((boxh - y) * (max - min)) / boxh + min; 958 float val = ((boxh - y) * (max - min)) / boxh + min;
810 paint.setPen(getColourForValue(v, val)); 959 if (logarithmic) {
960 paint.setPen(getColourForValue(v, LogRange::unmap(val)));
961 } else {
962 paint.setPen(getColourForValue(v, val));
963 }
811 paint.drawLine(boxx + 1, y + boxy + 1, boxx + boxw, y + boxy + 1); 964 paint.drawLine(boxx + 1, y + boxy + 1, boxx + boxw, y + boxy + 1);
812 } 965 }
813 paint.restore(); 966 paint.restore();
814 } 967 }
815 968
834 987
835 if (m_plotStyle == PlotSegmentation) { 988 if (m_plotStyle == PlotSegmentation) {
836 y = boxy + int(boxh - ((val - min) * boxh) / (max - min)); 989 y = boxy + int(boxh - ((val - min) * boxh) / (max - min));
837 ty = y; 990 ty = y;
838 } else { 991 } else {
839 if (i == n-1) { 992 if (i == n-1 &&
993 v->height() < paint.fontMetrics().height() * (n*2)) {
840 if (m_model->getScaleUnits() != "") drawText = false; 994 if (m_model->getScaleUnits() != "") drawText = false;
841 } 995 }
842 dispval = lrintf(val / round) * round; 996 dispval = lrintf(val / round) * round;
843 // std::cerr << "val = " << val << ", dispval = " << dispval << std::endl; 997 // std::cerr << "val = " << val << ", dispval = " << dispval << std::endl;
844 y = getYForValue(v, dispval); 998 if (logarithmic) {
999 y = getYForValue(v, LogRange::unmap(dispval));
1000 } else {
1001 y = getYForValue(v, dispval);
1002 }
845 ty = y - paint.fontMetrics().height() + 1003 ty = y - paint.fontMetrics().height() +
846 paint.fontMetrics().ascent() + 2; 1004 paint.fontMetrics().ascent() + 2;
847 1005
848 if (prevy >= 0 && (prevy - y) < paint.fontMetrics().height()) { 1006 if (prevy >= 0 && (prevy - y) < paint.fontMetrics().height()) {
849 val += inc; 1007 val += inc;
850 continue; 1008 continue;
851 } 1009 }
852 } 1010 }
853 1011
854 sprintf(buffer, "%.*f", dp, dispval); 1012 if (logarithmic) {
1013 sprintf(buffer, "%.*g", dp < 2 ? 2 : dp, LogRange::unmap(dispval));
1014 } else {
1015 sprintf(buffer, "%.*f", dp, dispval);
1016 }
855 QString label = QString(buffer); 1017 QString label = QString(buffer);
856 1018
857 if (m_plotStyle != PlotSegmentation) { 1019 if (m_plotStyle != PlotSegmentation) {
858 paint.drawLine(w - 5, y, w, y); 1020 paint.drawLine(w - 5, y, w, y);
859 } else { 1021 } else {