annotate widgets/AudioDial.cpp @ 187:e7cf6044c2a0

* better icon * support range mappers in thumbwheel * supply range mapper for vertical zoom from spectrogram * fix bug in fftmodel for scaled ffts * make the various widgets all respond to double-click for edit, middle-click for reset, ctrl-left-click for reset
author Chris Cannam
date Fri, 12 Jan 2007 14:49:18 +0000
parents 42118892f428
children 5b7472db612b
rev   line source
Chris@58 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@0 2
Chris@0 3 /*
Chris@59 4 Sonic Visualiser
Chris@59 5 An audio file viewer and annotation editor.
Chris@59 6 Centre for Digital Music, Queen Mary, University of London.
Chris@0 7
Chris@59 8 This program is free software; you can redistribute it and/or
Chris@59 9 modify it under the terms of the GNU General Public License as
Chris@59 10 published by the Free Software Foundation; either version 2 of the
Chris@59 11 License, or (at your option) any later version. See the file
Chris@59 12 COPYING included with this distribution for more information.
Chris@0 13 */
Chris@0 14
Chris@0 15 /**
Chris@0 16 * A rotary dial widget.
Chris@0 17 *
Chris@0 18 * Based on an original design by Thorsten Wilms.
Chris@0 19 *
Chris@0 20 * Implemented as a widget for the Rosegarden MIDI and audio sequencer
Chris@0 21 * and notation editor by Chris Cannam.
Chris@0 22 *
Chris@0 23 * Extracted into a standalone Qt3 widget by Pedro Lopez-Cabanillas
Chris@0 24 * and adapted for use in QSynth.
Chris@0 25 *
Chris@0 26 * Ported to Qt4 by Chris Cannam.
Chris@0 27 *
Chris@168 28 * This file copyright 2003-2006 Chris Cannam, copyright 2005 Pedro
Chris@182 29 * Lopez-Cabanillas, copyright 2006 Queen Mary, University of London.
Chris@0 30 *
Chris@0 31 * This program is free software; you can redistribute it and/or
Chris@0 32 * modify it under the terms of the GNU General Public License as
Chris@0 33 * published by the Free Software Foundation; either version 2 of the
Chris@0 34 * License, or (at your option) any later version. See the file
Chris@0 35 * COPYING included with this distribution for more information.
Chris@0 36 */
Chris@0 37
Chris@0 38 #include "AudioDial.h"
Chris@0 39
Chris@167 40 #include "base/RangeMapper.h"
Chris@167 41
Chris@0 42 #include <cmath>
Chris@0 43 #include <iostream>
Chris@0 44
Chris@0 45 #include <QTimer>
Chris@0 46 #include <QPainter>
Chris@0 47 #include <QPixmap>
Chris@0 48 #include <QColormap>
Chris@0 49 #include <QMouseEvent>
Chris@0 50 #include <QPaintEvent>
Chris@34 51 #include <QInputDialog>
Chris@0 52
Chris@0 53 using std::endl;
Chris@0 54 using std::cerr;
Chris@0 55
Chris@0 56
Chris@0 57 //!!! Pedro updated his version to use my up/down response code from RG -- need to grab that code in preference to this version from Rui
Chris@0 58
Chris@0 59
Chris@0 60 //-------------------------------------------------------------------------
Chris@0 61 // AudioDial - Instance knob widget class.
Chris@0 62 //
Chris@0 63
Chris@0 64 #define AUDIO_DIAL_MIN (0.25 * M_PI)
Chris@0 65 #define AUDIO_DIAL_MAX (1.75 * M_PI)
Chris@0 66 #define AUDIO_DIAL_RANGE (AUDIO_DIAL_MAX - AUDIO_DIAL_MIN)
Chris@0 67
Chris@0 68
Chris@0 69 // Constructor.
Chris@0 70 AudioDial::AudioDial(QWidget *parent) :
Chris@0 71 QDial(parent),
Chris@170 72 m_knobColor(Qt::black),
Chris@170 73 m_meterColor(Qt::white),
Chris@167 74 m_defaultValue(0),
Chris@168 75 m_mappedValue(0),
Chris@168 76 m_noMappedUpdate(false),
Chris@187 77 m_showTooltip(true),
Chris@167 78 m_rangeMapper(0)
Chris@0 79 {
Chris@0 80 m_mouseDial = false;
Chris@0 81 m_mousePressed = false;
Chris@0 82 }
Chris@0 83
Chris@0 84
Chris@0 85 // Destructor.
Chris@0 86 AudioDial::~AudioDial (void)
Chris@0 87 {
Chris@167 88 delete m_rangeMapper;
Chris@167 89 }
Chris@167 90
Chris@167 91
Chris@167 92 void AudioDial::setRangeMapper(RangeMapper *mapper)
Chris@167 93 {
Chris@187 94 if (m_rangeMapper == mapper) return;
Chris@187 95
Chris@170 96 if (!m_rangeMapper && mapper) {
Chris@168 97 connect(this, SIGNAL(valueChanged(int)),
Chris@168 98 this, SLOT(updateMappedValue(int)));
Chris@168 99 }
Chris@170 100
Chris@167 101 delete m_rangeMapper;
Chris@167 102 m_rangeMapper = mapper;
Chris@170 103
Chris@187 104 updateMappedValue(value());
Chris@0 105 }
Chris@0 106
Chris@0 107
Chris@0 108 void AudioDial::paintEvent(QPaintEvent *)
Chris@0 109 {
Chris@0 110 QPainter paint;
Chris@0 111
Chris@0 112 float angle = AUDIO_DIAL_MIN // offset
Chris@0 113 + (AUDIO_DIAL_RANGE *
Chris@0 114 (float(QDial::value() - QDial::minimum()) /
Chris@0 115 (float(QDial::maximum() - QDial::minimum()))));
Chris@0 116 int degrees = int(angle * 180.0 / M_PI);
Chris@0 117
Chris@0 118 int ns = notchSize();
Chris@0 119 int numTicks = 1 + (maximum() + ns - minimum()) / ns;
Chris@0 120
Chris@0 121 QColor knobColor(m_knobColor);
Chris@0 122 if (knobColor == Qt::black)
Chris@82 123 knobColor = palette().background().color();
Chris@0 124
Chris@0 125 QColor meterColor(m_meterColor);
Chris@0 126 if (!isEnabled())
Chris@0 127 meterColor = palette().mid().color();
Chris@0 128 else if (m_meterColor == Qt::white)
Chris@0 129 meterColor = palette().highlight().color();
Chris@0 130
Chris@0 131 int m_size = width() < height() ? width() : height();
Chris@0 132 int scale = 1;
Chris@0 133 int width = m_size - 2*scale, height = m_size - 2*scale;
Chris@0 134
Chris@0 135 paint.begin(this);
Chris@0 136 paint.setRenderHint(QPainter::Antialiasing, true);
Chris@0 137 paint.translate(1, 1);
Chris@0 138
Chris@0 139 QPen pen;
Chris@0 140 QColor c;
Chris@0 141
Chris@0 142 // Knob body and face...
Chris@0 143
Chris@0 144 c = knobColor;
Chris@0 145 pen.setColor(knobColor);
Chris@0 146 pen.setWidth(scale * 2);
Chris@0 147 pen.setCapStyle(Qt::FlatCap);
Chris@0 148
Chris@0 149 paint.setPen(pen);
Chris@0 150 paint.setBrush(c);
Chris@0 151
Chris@0 152 int indent = (int)(width * 0.15 + 1);
Chris@0 153
Chris@0 154 paint.drawEllipse(indent-1, indent-1, width-2*indent, width-2*indent);
Chris@0 155
Chris@0 156 pen.setWidth(3 * scale);
Chris@0 157 int pos = indent-1 + (width-2*indent) / 20;
Chris@0 158 int darkWidth = (width-2*indent) * 3 / 4;
Chris@0 159 while (darkWidth) {
Chris@0 160 c = c.light(102);
Chris@0 161 pen.setColor(c);
Chris@0 162 paint.setPen(pen);
Chris@0 163 paint.drawEllipse(pos, pos, darkWidth, darkWidth);
Chris@0 164 if (!--darkWidth) break;
Chris@0 165 paint.drawEllipse(pos, pos, darkWidth, darkWidth);
Chris@0 166 if (!--darkWidth) break;
Chris@0 167 paint.drawEllipse(pos, pos, darkWidth, darkWidth);
Chris@0 168 ++pos; --darkWidth;
Chris@0 169 }
Chris@0 170
Chris@0 171 // Tick notches...
Chris@0 172
Chris@34 173 if ( notchesVisible() ) {
Chris@0 174 // std::cerr << "Notches visible" << std::endl;
Chris@0 175 pen.setColor(palette().dark().color());
Chris@0 176 pen.setWidth(scale);
Chris@0 177 paint.setPen(pen);
Chris@0 178 for (int i = 0; i < numTicks; ++i) {
Chris@0 179 int div = numTicks;
Chris@0 180 if (div > 1) --div;
Chris@0 181 drawTick(paint, AUDIO_DIAL_MIN + (AUDIO_DIAL_MAX - AUDIO_DIAL_MIN) * i / div,
Chris@0 182 width, true);
Chris@0 183 }
Chris@0 184 }
Chris@0 185
Chris@0 186 // The bright metering bit...
Chris@0 187
Chris@0 188 c = meterColor;
Chris@0 189 pen.setColor(c);
Chris@0 190 pen.setWidth(indent);
Chris@0 191 paint.setPen(pen);
Chris@0 192
Chris@0 193 // std::cerr << "degrees " << degrees << ", gives us " << -(degrees - 45) * 16 << std::endl;
Chris@0 194
Chris@0 195 int arcLen = -(degrees - 45) * 16;
Chris@0 196 if (arcLen == 0) arcLen = -16;
Chris@0 197
Chris@0 198 paint.drawArc(indent/2, indent/2,
Chris@0 199 width-indent, width-indent, (180 + 45) * 16, arcLen);
Chris@0 200
Chris@0 201 paint.setBrush(Qt::NoBrush);
Chris@0 202
Chris@0 203 // Shadowing...
Chris@0 204
Chris@0 205 pen.setWidth(scale);
Chris@0 206 paint.setPen(pen);
Chris@0 207
Chris@0 208 // Knob shadow...
Chris@0 209
Chris@0 210 int shadowAngle = -720;
Chris@0 211 c = knobColor.dark();
Chris@0 212 for (int arc = 120; arc < 2880; arc += 240) {
Chris@0 213 pen.setColor(c);
Chris@0 214 paint.setPen(pen);
Chris@0 215 paint.drawArc(indent, indent,
Chris@0 216 width-2*indent, width-2*indent, shadowAngle + arc, 240);
Chris@0 217 paint.drawArc(indent, indent,
Chris@0 218 width-2*indent, width-2*indent, shadowAngle - arc, 240);
Chris@0 219 c = c.light(110);
Chris@0 220 }
Chris@0 221
Chris@0 222 // Scale shadow...
Chris@0 223
Chris@0 224 shadowAngle = 2160;
Chris@0 225 c = palette().dark().color();
Chris@0 226 for (int arc = 120; arc < 2880; arc += 240) {
Chris@0 227 pen.setColor(c);
Chris@0 228 paint.setPen(pen);
Chris@0 229 paint.drawArc(scale/2, scale/2,
Chris@0 230 width-scale, width-scale, shadowAngle + arc, 240);
Chris@0 231 paint.drawArc(scale/2, scale/2,
Chris@0 232 width-scale, width-scale, shadowAngle - arc, 240);
Chris@0 233 c = c.light(108);
Chris@0 234 }
Chris@0 235
Chris@0 236 // Undraw the bottom part...
Chris@0 237
Chris@0 238 pen.setColor(palette().background().color());
Chris@0 239 pen.setWidth(scale * 4);
Chris@0 240 paint.setPen(pen);
Chris@0 241 paint.drawArc(scale/2, scale/2,
Chris@0 242 width-scale, width-scale, -45 * 16, -92 * 16);
Chris@0 243
Chris@0 244 // Scale ends...
Chris@0 245
Chris@0 246 pen.setColor(palette().dark().color());
Chris@0 247 pen.setWidth(scale);
Chris@0 248 paint.setPen(pen);
Chris@0 249 for (int i = 0; i < numTicks; ++i) {
Chris@0 250 if (i != 0 && i != numTicks - 1) continue;
Chris@0 251 int div = numTicks;
Chris@0 252 if (div > 1) --div;
Chris@0 253 drawTick(paint, AUDIO_DIAL_MIN + (AUDIO_DIAL_MAX - AUDIO_DIAL_MIN) * i / div,
Chris@0 254 width, false);
Chris@0 255 }
Chris@0 256
Chris@0 257 // Pointer notch...
Chris@0 258
Chris@0 259 float hyp = float(width) / 2.0;
Chris@0 260 float len = hyp - indent;
Chris@0 261 --len;
Chris@0 262
Chris@0 263 float x0 = hyp;
Chris@0 264 float y0 = hyp;
Chris@0 265
Chris@0 266 float x = hyp - len * sin(angle);
Chris@0 267 float y = hyp + len * cos(angle);
Chris@0 268
Chris@0 269 c = palette().dark().color();
Chris@0 270 pen.setColor(isEnabled() ? c.dark(130) : c);
Chris@0 271 pen.setWidth(scale * 2);
Chris@0 272 paint.setPen(pen);
Chris@0 273 paint.drawLine(int(x0), int(y0), int(x), int(y));
Chris@0 274
Chris@0 275 paint.end();
Chris@0 276 }
Chris@0 277
Chris@0 278
Chris@0 279 void AudioDial::drawTick(QPainter &paint,
Chris@0 280 float angle, int size, bool internal)
Chris@0 281 {
Chris@0 282 float hyp = float(size) / 2.0;
Chris@0 283 float x0 = hyp - (hyp - 1) * sin(angle);
Chris@0 284 float y0 = hyp + (hyp - 1) * cos(angle);
Chris@0 285
Chris@0 286 // cerr << "drawTick: angle " << angle << ", size " << size << ", internal " << internal << endl;
Chris@0 287
Chris@0 288 if (internal) {
Chris@0 289
Chris@0 290 float len = hyp / 4;
Chris@0 291 float x1 = hyp - (hyp - len) * sin(angle);
Chris@0 292 float y1 = hyp + (hyp - len) * cos(angle);
Chris@0 293
Chris@0 294 paint.drawLine(int(x0), int(y0), int(x1), int(y1));
Chris@0 295
Chris@0 296 } else {
Chris@0 297
Chris@0 298 float len = hyp / 4;
Chris@0 299 float x1 = hyp - (hyp + len) * sin(angle);
Chris@0 300 float y1 = hyp + (hyp + len) * cos(angle);
Chris@0 301
Chris@0 302 paint.drawLine(int(x0), int(y0), int(x1), int(y1));
Chris@0 303 }
Chris@0 304 }
Chris@0 305
Chris@0 306
Chris@0 307 void AudioDial::setKnobColor(const QColor& color)
Chris@0 308 {
Chris@0 309 m_knobColor = color;
Chris@0 310 update();
Chris@0 311 }
Chris@0 312
Chris@0 313
Chris@0 314 void AudioDial::setMeterColor(const QColor& color)
Chris@0 315 {
Chris@0 316 m_meterColor = color;
Chris@0 317 update();
Chris@0 318 }
Chris@0 319
Chris@0 320
Chris@0 321 void AudioDial::setMouseDial(bool mouseDial)
Chris@0 322 {
Chris@0 323 m_mouseDial = mouseDial;
Chris@0 324 }
Chris@0 325
Chris@0 326
Chris@34 327 void AudioDial::setDefaultValue(int defaultValue)
Chris@34 328 {
Chris@34 329 m_defaultValue = defaultValue;
Chris@34 330 }
Chris@34 331
Chris@34 332
Chris@177 333 void AudioDial::setMappedValue(float mappedValue)
Chris@177 334 {
Chris@177 335 if (m_rangeMapper) {
Chris@177 336 int newPosition = m_rangeMapper->getPositionForValue(mappedValue);
Chris@177 337 m_mappedValue = mappedValue;
Chris@177 338 m_noMappedUpdate = true;
Chris@177 339 std::cerr << "AudioDial::setMappedValue(" << mappedValue << "): new position is " << newPosition << std::endl;
Chris@177 340 if (newPosition != value()) {
Chris@177 341 setValue(newPosition);
Chris@177 342 } else {
Chris@177 343 emit valueChanged(newPosition);
Chris@177 344 }
Chris@177 345 m_noMappedUpdate = false;
Chris@177 346 } else {
Chris@187 347 setValue(int(mappedValue));
Chris@177 348 }
Chris@177 349 }
Chris@177 350
Chris@177 351
Chris@168 352 void AudioDial::setShowToolTip(bool show)
Chris@168 353 {
Chris@168 354 m_showTooltip = show;
Chris@168 355 m_noMappedUpdate = true;
Chris@168 356 updateMappedValue(value());
Chris@168 357 m_noMappedUpdate = false;
Chris@168 358 }
Chris@168 359
Chris@168 360
Chris@167 361 float AudioDial::mappedValue() const
Chris@167 362 {
Chris@168 363 if (m_rangeMapper) {
Chris@168 364 std::cerr << "AudioDial::mappedValue(): value = " << value() << ", mappedValue = " << m_mappedValue << std::endl;
Chris@168 365 return m_mappedValue;
Chris@168 366 }
Chris@168 367 return value();
Chris@168 368 }
Chris@168 369
Chris@168 370
Chris@168 371 void AudioDial::updateMappedValue(int value)
Chris@168 372 {
Chris@170 373 if (!m_noMappedUpdate) {
Chris@170 374 if (m_rangeMapper) {
Chris@168 375 m_mappedValue = m_rangeMapper->getValueForPosition(value);
Chris@170 376 } else {
Chris@170 377 m_mappedValue = value;
Chris@168 378 }
Chris@168 379 }
Chris@168 380
Chris@168 381 if (m_showTooltip) {
Chris@168 382 QString name = objectName();
Chris@168 383 QString unit = "";
Chris@168 384 QString text;
Chris@168 385 if (m_rangeMapper) unit = m_rangeMapper->getUnit();
Chris@168 386 if (name != "") {
Chris@168 387 text = tr("%1: %2%3").arg(name).arg(m_mappedValue).arg(unit);
Chris@168 388 } else {
Chris@168 389 text = tr("%2%3").arg(m_mappedValue).arg(unit);
Chris@168 390 }
Chris@168 391 setToolTip(text);
Chris@168 392 }
Chris@167 393 }
Chris@167 394
Chris@167 395
Chris@0 396 // Alternate mouse behavior event handlers.
Chris@0 397 void AudioDial::mousePressEvent(QMouseEvent *mouseEvent)
Chris@0 398 {
Chris@0 399 if (m_mouseDial) {
Chris@0 400 QDial::mousePressEvent(mouseEvent);
Chris@187 401 } else if (mouseEvent->button() == Qt::MidButton ||
Chris@187 402 ((mouseEvent->button() == Qt::LeftButton) &&
Chris@187 403 (mouseEvent->modifiers() & Qt::ControlModifier))) {
Chris@34 404 int dv = m_defaultValue;
Chris@34 405 if (dv < minimum()) dv = minimum();
Chris@34 406 if (dv > maximum()) dv = maximum();
Chris@34 407 setValue(m_defaultValue);
Chris@187 408 } else if (mouseEvent->button() == Qt::LeftButton) {
Chris@187 409 m_mousePressed = true;
Chris@187 410 m_posMouse = mouseEvent->pos();
Chris@34 411 }
Chris@34 412 }
Chris@34 413
Chris@34 414
Chris@34 415 void AudioDial::mouseDoubleClickEvent(QMouseEvent *mouseEvent)
Chris@34 416 {
Chris@187 417 //!!! needs a common base class with Thumbwheel
Chris@187 418
Chris@34 419 if (m_mouseDial) {
Chris@34 420 QDial::mouseDoubleClickEvent(mouseEvent);
Chris@187 421 } else if (mouseEvent->button() != Qt::LeftButton) {
Chris@187 422 return;
Chris@187 423 }
Chris@167 424
Chris@187 425 bool ok = false;
Chris@167 426
Chris@187 427 if (m_rangeMapper) {
Chris@187 428
Chris@187 429 float min = m_rangeMapper->getValueForPosition(minimum());
Chris@187 430 float max = m_rangeMapper->getValueForPosition(maximum());
Chris@187 431
Chris@187 432 if (min > max) {
Chris@187 433 float tmp = min;
Chris@187 434 min = max;
Chris@187 435 max = tmp;
Chris@187 436 }
Chris@167 437
Chris@187 438 QString unit = m_rangeMapper->getUnit();
Chris@187 439
Chris@187 440 QString text;
Chris@187 441 if (objectName() != "") {
Chris@187 442 if (unit != "") {
Chris@187 443 text = tr("New value for %1, from %2 to %3 %4:")
Chris@187 444 .arg(objectName()).arg(min).arg(max).arg(unit);
Chris@167 445 } else {
Chris@187 446 text = tr("New value for %1, from %2 to %3:")
Chris@187 447 .arg(objectName()).arg(min).arg(max);
Chris@167 448 }
Chris@187 449 } else {
Chris@187 450 if (unit != "") {
Chris@187 451 text = tr("Enter a new value from %1 to %2 %3:")
Chris@187 452 .arg(min).arg(max).arg(unit);
Chris@187 453 } else {
Chris@187 454 text = tr("Enter a new value from %1 to %2:")
Chris@187 455 .arg(min).arg(max);
Chris@168 456 }
Chris@187 457 }
Chris@187 458
Chris@187 459 float newValue = QInputDialog::getDouble
Chris@187 460 (this,
Chris@187 461 tr("Enter new value"),
Chris@187 462 text,
Chris@187 463 m_mappedValue,
Chris@187 464 min,
Chris@187 465 max,
Chris@187 466 4,
Chris@187 467 &ok);
Chris@187 468
Chris@187 469 if (ok) {
Chris@187 470 setMappedValue(newValue);
Chris@187 471 }
Chris@187 472
Chris@187 473 } else {
Chris@187 474
Chris@187 475 int newPosition = QInputDialog::getInteger
Chris@187 476 (this,
Chris@187 477 tr("Enter new value"),
Chris@187 478 tr("Enter a new value from %1 to %2:")
Chris@187 479 .arg(minimum()).arg(maximum()),
Chris@187 480 value(), minimum(), maximum(), pageStep(), &ok);
Chris@187 481
Chris@187 482 if (ok) {
Chris@187 483 setValue(newPosition);
Chris@167 484 }
Chris@0 485 }
Chris@0 486 }
Chris@0 487
Chris@0 488
Chris@0 489 void AudioDial::mouseMoveEvent(QMouseEvent *mouseEvent)
Chris@0 490 {
Chris@0 491 if (m_mouseDial) {
Chris@0 492 QDial::mouseMoveEvent(mouseEvent);
Chris@0 493 } else if (m_mousePressed) {
Chris@0 494 const QPoint& posMouse = mouseEvent->pos();
Chris@0 495 int v = QDial::value()
Chris@0 496 + (posMouse.x() - m_posMouse.x())
Chris@0 497 + (m_posMouse.y() - posMouse.y());
Chris@0 498 if (v > QDial::maximum())
Chris@0 499 v = QDial::maximum();
Chris@0 500 else
Chris@0 501 if (v < QDial::minimum())
Chris@0 502 v = QDial::minimum();
Chris@0 503 m_posMouse = posMouse;
Chris@0 504 QDial::setValue(v);
Chris@0 505 }
Chris@0 506 }
Chris@0 507
Chris@0 508
Chris@0 509 void AudioDial::mouseReleaseEvent(QMouseEvent *mouseEvent)
Chris@0 510 {
Chris@0 511 if (m_mouseDial) {
Chris@0 512 QDial::mouseReleaseEvent(mouseEvent);
Chris@0 513 } else if (m_mousePressed) {
Chris@0 514 m_mousePressed = false;
Chris@0 515 }
Chris@0 516 }
Chris@0 517
Chris@0 518 #ifdef INCLUDE_MOCFILES
Chris@0 519 #include "AudioDial.moc.cpp"
Chris@0 520 #endif
Chris@0 521