#include "FractoriumPch.h" #include "DoubleSpinBox.h" QTimer DoubleSpinBox::s_Timer; /// /// Constructor that passes parent to the base and sets up height and step. /// Specific focus policy is used to allow the user to hover over the control /// and change its value using the mouse wheel without explicitly having to click /// inside of it. /// /// The parent widget. Default: nullptr. /// The height of the spin box. Default: 16. /// The step used to increment/decrement the spin box when using the mouse wheel. Default: 0.05. DoubleSpinBox::DoubleSpinBox(QWidget* p, int h, double step, bool clearSel) : QDoubleSpinBox(p) { m_DoubleClick = false; m_ClearSel = clearSel; m_DoubleClickLowVal = 0; m_DoubleClickNonZero = 0; m_DoubleClickZero = 1; m_Step = step; m_SmallStep = step / 10.0; m_Settings = FractoriumSettings::DefInstance(); setSingleStep(step); setFrame(false); setButtonSymbols(QAbstractSpinBox::NoButtons); setFocusPolicy(Qt::StrongFocus); setMinimumHeight(h);//setGeometry() has no effect, so must set both of these instead. setMaximumHeight(h); setContextMenuPolicy(Qt::PreventContextMenu); lineEdit()->installEventFilter(this); lineEdit()->setAlignment(Qt::AlignLeft | Qt::AlignVCenter); connect(this, SIGNAL(valueChanged(double)), this, SLOT(OnSpinBoxValueChanged(double)), Qt::QueuedConnection); } /// /// Set the value of the control without triggering signals. /// /// The value to set it to void DoubleSpinBox::SetValueStealth(double d) { blockSignals(true); setValue(d); blockSignals(false); } /// /// Set whether to respond to double click events. /// /// True if this should respond to double click events, else false. void DoubleSpinBox::DoubleClick(bool b) { m_DoubleClick = b; } /// /// Set the value to be used instead of zero to represent the lower value /// used when responding to a double click. /// /// The value to be used for the lower value instead of zero void DoubleSpinBox::DoubleClickLowVal(double val) { m_DoubleClickLowVal = val; } /// /// Set the value to be used when the user double clicks the spinner while /// it contains zero. /// /// The value to be used void DoubleSpinBox::DoubleClickZero(double val) { m_DoubleClickZero = val; } /// /// Set the value to be used when the user double clicks the spinner while /// it contains a non-zero value. /// /// The value to be used void DoubleSpinBox::DoubleClickNonZero(double val) { m_DoubleClickNonZero = val; } /// /// Get the default step used when the user scrolls. /// /// The default step as a double. double DoubleSpinBox::Step() { return m_Step; } /// /// Set the default step to be used when the user scrolls. /// /// The step to use for scrolling void DoubleSpinBox::Step(double step) { m_Step = step; } /// /// Get the small step to be used when the user holds down shift while scrolling. /// /// The small step as a double. double DoubleSpinBox::SmallStep() { return m_SmallStep; } /// /// Set the small step to be used when the user holds down shift while scrolling. /// The default is step / 10, so use this if something else is needed. /// /// The small step to use for scrolling while the shift key is down void DoubleSpinBox::SmallStep(double step) { m_SmallStep = step; } /// /// Expose the underlying QLineEdit control to the caller. /// /// A pointer to the QLineEdit QLineEdit* DoubleSpinBox::lineEdit() { return QDoubleSpinBox::lineEdit(); } /// /// Another workaround for the persistent text selection bug in Qt. /// void DoubleSpinBox::OnSpinBoxValueChanged(double) { if (m_ClearSel) lineEdit()->deselect();//Gets rid of nasty "feature" that always has text selected. } /// /// Called while the timer is activated due to the right mouse button being held down. /// void DoubleSpinBox::OnTimeout() { int xdistance = m_MouseMovePoint.x() - m_MouseDownPoint.x(); int ydistance = m_MouseMovePoint.y() - m_MouseDownPoint.y(); int distance = abs(xdistance) > abs(ydistance) ? xdistance : ydistance; distance = Sqr(distance) * (distance < 0 ? -1 : 1); double scale, val; double d = value(); bool shift = QGuiApplication::keyboardModifiers().testFlag(Qt::ShiftModifier); bool ctrl = QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier); double amount = (m_SmallStep + m_Step) * 0.5; if (shift) scale = 0.0001; else if (ctrl) scale = 0.01; else scale = 0.001; val = d + (distance * amount * scale); setValue(val); } /// /// Event filter for taking special action on double click events. /// /// The object /// The eevent /// false bool DoubleSpinBox::eventFilter(QObject* o, QEvent* e) { auto me = dynamic_cast(e); if (isEnabled() && me) { bool isRight = me->button() == Qt::RightButton; if (!m_Settings->ToggleType() &&//Ensure double click toggles, not right click. me->type() == QMouseEvent::MouseButtonPress && isRight) { m_MouseDownPoint = m_MouseMovePoint = me->pos(); StartTimer(); e->accept(); return true; } else if (!m_Settings->ToggleType() && me->type() == QMouseEvent::MouseButtonRelease && isRight) { StopTimer(); m_MouseDownPoint = m_MouseMovePoint = me->pos(); e->accept(); return true; } else if (!m_Settings->ToggleType() && me->type() == QMouseEvent::MouseMove && QGuiApplication::mouseButtons() & Qt::RightButton) { m_MouseMovePoint = me->pos(); e->accept(); return true; } else if (m_DoubleClick && ((!m_Settings->ToggleType() && e->type() == QMouseEvent::MouseButtonDblClick && me->button() == Qt::LeftButton) || (m_Settings->ToggleType() && me->type() == QMouseEvent::MouseButtonRelease && isRight))) { if (IsClose(m_DoubleClickLowVal, value())) setValue(m_DoubleClickZero); else setValue(m_DoubleClickNonZero); e->accept(); return true; } } else if (isEnabled()) { if (e->type() == QEvent::Wheel) { if (QWheelEvent* we = dynamic_cast(e)) { bool shift = QGuiApplication::keyboardModifiers().testFlag(Qt::ShiftModifier); bool ctrl = QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier); if (we->angleDelta().ry() > 0) { if (shift) { setSingleStep(m_SmallStep); setValue(value() + m_SmallStep); } else { setSingleStep(m_Step); setValue(value() + (ctrl ? m_Step * 10 : m_Step)); } } else { if (shift) { setSingleStep(m_SmallStep); setValue(value() - m_SmallStep); } else { setSingleStep(m_Step); setValue(value() - (ctrl ? m_Step * 10 : m_Step)); } } e->accept(); return true; } } else if (dynamic_cast(e)) { e->accept(); return true; } } return QDoubleSpinBox::eventFilter(o, e); } /// /// Override which is for handling specific key presses while this control is focused. /// In particular, + = and up arrow increase the value, equivalent to scrolling the mouse wheel up, while also observing shift/ctrl modifiers. /// Values are decreased in the same way by pressing - _ or down arrow. /// /// The key event void DoubleSpinBox::keyPressEvent(QKeyEvent* ke) { bool shift = QGuiApplication::keyboardModifiers().testFlag(Qt::ShiftModifier); bool ctrl = QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier); if (ke->key() == Qt::Key_Up) { if (shift) { setSingleStep(m_SmallStep); setValue(value() + m_SmallStep); } else { setSingleStep(m_Step); setValue(value() + (ctrl ? m_Step * 10 : m_Step)); } ke->accept(); } else if (ke->key() == Qt::Key_Down) { if (shift) { setSingleStep(m_SmallStep); setValue(value() - m_SmallStep); } else { setSingleStep(m_Step); setValue(value() - (ctrl ? m_Step * 10 : m_Step)); } ke->accept(); } else if (ke->key() == Qt::Key_Space) { if (IsClose(m_DoubleClickLowVal, value())) setValue(m_DoubleClickZero); else setValue(m_DoubleClickNonZero); } else QDoubleSpinBox::keyPressEvent(ke); } /// /// Called when focus enters the spinner. /// /// The event void DoubleSpinBox::focusInEvent(QFocusEvent* e) { StopTimer(); QDoubleSpinBox::focusInEvent(e); } /// /// Called when focus leaves the spinner. /// Qt has a nasty "feature" that leaves the text in a spinner selected /// and the cursor visible, regardless of whether it has the focus. /// Manually clear both here. /// /// The event void DoubleSpinBox::focusOutEvent(QFocusEvent* e) { StopTimer(); QDoubleSpinBox::focusOutEvent(e); } /// /// Called when focus enters the spinner. /// Must set the focus to make sure key down messages don't erroneously go to the GLWidget. /// /// The event void DoubleSpinBox::enterEvent(QEvent* e) { StopTimer(); QDoubleSpinBox::enterEvent(e); } /// /// Called when focus leaves the spinner. /// Must clear the focus to make sure key down messages don't erroneously go to the GLWidget. /// /// The event void DoubleSpinBox::leaveEvent(QEvent* e) { StopTimer(); QDoubleSpinBox::leaveEvent(e); } /// /// Start the timer in response to the right mouse button being pressed. /// void DoubleSpinBox::StartTimer() { s_Timer.stop(); connect(&s_Timer, SIGNAL(timeout()), this, SLOT(OnTimeout())); s_Timer.start(300); } /// /// Stop the timer in response to the left mouse button being pressed. /// void DoubleSpinBox::StopTimer() { s_Timer.stop(); disconnect(&s_Timer, SIGNAL(timeout()), this, SLOT(OnTimeout())); } /// /// Constructor that does nothing but pass arguments to the base. /// /// The parent widget /// The height of the spin box. Default: 16. /// The step used to increment/decrement the spin box when using the mouse wheel. Default: 0.05. SpecialDoubleSpinBox::SpecialDoubleSpinBox(QWidget* p, int h, double step) : DoubleSpinBox(p, h, step) { } /// /// Called when focus enters the spinner. /// When leaving the spinner, the context menu was disabled so it doesn't pop up on /// distant loctions on the screen when dragging with the right mouse button then releasing. /// So re-enable it here just to ensure whenever they focus this control, the menu works. /// /// The event void SpecialDoubleSpinBox::enterEvent(QEvent* e) { this->setContextMenuPolicy(Qt::ActionsContextMenu); DoubleSpinBox::enterEvent(e); } /// /// Called when focus leaves the spinner. /// When leaving the spinner, disable the context menu so it doesn't pop up on distant loctions on the screen /// when dragging with the right mouse button then releasing. /// /// The event void SpecialDoubleSpinBox::leaveEvent(QEvent* e) { this->setContextMenuPolicy(Qt::PreventContextMenu); DoubleSpinBox::leaveEvent(e); } /// /// Event filter for taking special action on right click events. /// /// The object /// The eevent /// True to stop processing the event, else false./ bool SpecialDoubleSpinBox::eventFilter(QObject* o, QEvent* e) { if (isEnabled()) { auto me = dynamic_cast(e); auto cme = dynamic_cast(e); if (m_DoubleClick && m_Settings->ToggleType())//If they use right click to toggle... { if (me) { if (me->type() == QMouseEvent::MouseButtonRelease && me->button() == Qt::RightButton) { if (me->modifiers().testFlag(Qt::ShiftModifier))//...then do not take the action if shift was pressed. return false;//Shift was pressed, so continue normal event processing to show the menu, but do not call the base to toggle the value. } } else if (cme)//Context menu. { if (!cme->modifiers().testFlag(Qt::ShiftModifier))//If they are not holding shift, call the base to toggle, and do not process further which suppresses showing the menu. { DoubleSpinBox::eventFilter(o, e); return true; } } } } return DoubleSpinBox::eventFilter(o, e); } /// /// Constructor that passes agruments to the base and assigns the m_Param and m_Variation members. /// It also sets up the context menu for special numerical values. /// /// The parent widget /// The widget item this spinner is contained in /// The variation this spinner is for /// The name of the parameter this is for /// The height of the spin box. Default: 16. /// The step used to increment/decrement the spin box when using the mouse wheel. Default: 0.05. VariationTreeDoubleSpinBox::VariationTreeDoubleSpinBox(QWidget* p, VariationTreeWidgetItem* widgetItem, eVariationId id, const string& param, int h, double step) : SpecialDoubleSpinBox(p, h, step) { m_WidgetItem = widgetItem; m_Param = param; m_Id = id; setDecimals(7); //PI auto piAction = new QAction("PI", this); connect(piAction, SIGNAL(triggered(bool)), this, SLOT(PiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(piAction); //PI * 2 auto twoPiAction = new QAction("2 PI", this); connect(twoPiAction, SIGNAL(triggered(bool)), this, SLOT(TwoPiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(twoPiAction); //PI / 2 auto piOver2Action = new QAction("PI / 2", this); connect(piOver2Action, SIGNAL(triggered(bool)), this, SLOT(PiOver2ActionTriggered(bool)), Qt::QueuedConnection); this->addAction(piOver2Action); //PI / 3 auto piOver3Action = new QAction("PI / 3", this); connect(piOver3Action, SIGNAL(triggered(bool)), this, SLOT(PiOver3ActionTriggered(bool)), Qt::QueuedConnection); this->addAction(piOver3Action); //PI / 4 auto piOver4Action = new QAction("PI / 4", this); connect(piOver4Action, SIGNAL(triggered(bool)), this, SLOT(PiOver4ActionTriggered(bool)), Qt::QueuedConnection); this->addAction(piOver4Action); //PI / 6 auto piOver6Action = new QAction("PI / 6", this); connect(piOver6Action, SIGNAL(triggered(bool)), this, SLOT(PiOver6ActionTriggered(bool)), Qt::QueuedConnection); this->addAction(piOver6Action); //1 / PI auto oneOverPiAction = new QAction("1 / PI", this); connect(oneOverPiAction, SIGNAL(triggered(bool)), this, SLOT(OneOverPiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(oneOverPiAction); //2 / PI auto twoOverPiAction = new QAction("2 / PI", this); connect(twoOverPiAction, SIGNAL(triggered(bool)), this, SLOT(TwoOverPiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(twoOverPiAction); //3 / PI auto threeOverPiAction = new QAction("3 / PI", this); connect(threeOverPiAction, SIGNAL(triggered(bool)), this, SLOT(ThreeOverPiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(threeOverPiAction); //4 / PI auto fourOverPiAction = new QAction("4 / PI", this); connect(fourOverPiAction, SIGNAL(triggered(bool)), this, SLOT(FourOverPiActionTriggered(bool)), Qt::QueuedConnection); this->addAction(fourOverPiAction); //Sqrt(2) auto sqrtTwoAction = new QAction("Sqrt(2)", this); connect(sqrtTwoAction, SIGNAL(triggered(bool)), this, SLOT(SqrtTwoActionTriggered(bool)), Qt::QueuedConnection); this->addAction(sqrtTwoAction); //Sqrt(2) auto sqrtThreeAction = new QAction("Sqrt(3)", this); connect(sqrtThreeAction, SIGNAL(triggered(bool)), this, SLOT(SqrtThreeActionTriggered(bool)), Qt::QueuedConnection); this->addAction(sqrtThreeAction); //Need this for it to show up properly. this->setContextMenuPolicy(Qt::ActionsContextMenu); lineEdit()->setValidator(new QDoubleValidator(this)); } void VariationTreeDoubleSpinBox::PiActionTriggered(bool checked) { setValue(M_PI); } void VariationTreeDoubleSpinBox::TwoPiActionTriggered(bool checked) { setValue(M_PI * 2); } void VariationTreeDoubleSpinBox::PiOver2ActionTriggered(bool checked) { setValue(M_PI_2); } void VariationTreeDoubleSpinBox::PiOver3ActionTriggered(bool checked) { setValue(M_PI / 3); } void VariationTreeDoubleSpinBox::PiOver4ActionTriggered(bool checked) { setValue(M_PI / 4); } void VariationTreeDoubleSpinBox::PiOver6ActionTriggered(bool checked) { setValue(M_PI / 6); } void VariationTreeDoubleSpinBox::OneOverPiActionTriggered(bool checked) { setValue(1 / M_PI); } void VariationTreeDoubleSpinBox::TwoOverPiActionTriggered(bool checked) { setValue(2 / M_PI); } void VariationTreeDoubleSpinBox::ThreeOverPiActionTriggered(bool checked) { setValue(3 / M_PI); } void VariationTreeDoubleSpinBox::FourOverPiActionTriggered(bool checked) { setValue(4 / M_PI); } void VariationTreeDoubleSpinBox::SqrtTwoActionTriggered(bool checked) { setValue(M_SQRT2); } void VariationTreeDoubleSpinBox::SqrtThreeActionTriggered(bool checked) { setValue(std::sqrt(3.0)); } /// /// Override which converts the passed in double to text. /// /// Text showing decimals() decimal places, or sometimes scientific notation. QString VariationTreeDoubleSpinBox::textFromValue(double value) const { return QWidget::locale().toString(value, 'g', decimals()); } /// /// Override which converts the passed in text to a double /// /// The converted double double VariationTreeDoubleSpinBox::valueFromText(const QString& text) const { return QWidget::locale().toDouble(text); } /// /// Constructor that sets up the context menu for special numerical values specific to affine spinners. /// /// The parent widget /// The height of the spin box. Default: 20. /// The step used to increment/decrement the spin box when using the mouse wheel. Default: 0.01. AffineDoubleSpinBox::AffineDoubleSpinBox(QWidget* p, int h, double step) : SpecialDoubleSpinBox(p, h, step) { //-1 auto neg1Action = new QAction("-1", this); connect(neg1Action, SIGNAL(triggered(bool)), this, SLOT(NegOneActionTriggered(bool)), Qt::QueuedConnection); this->addAction(neg1Action); //0 auto zeroAction = new QAction("0", this); connect(zeroAction, SIGNAL(triggered(bool)), this, SLOT(ZeroActionTriggered(bool)), Qt::QueuedConnection); this->addAction(zeroAction); //1 auto oneAction = new QAction("1", this); connect(oneAction, SIGNAL(triggered(bool)), this, SLOT(OneActionTriggered(bool)), Qt::QueuedConnection); this->addAction(oneAction); //45 auto fortyFiveAction = new QAction("45", this); connect(fortyFiveAction, SIGNAL(triggered(bool)), this, SLOT(FortyFiveActionTriggered(bool)), Qt::QueuedConnection); this->addAction(fortyFiveAction); //-45 auto negFortyFiveAction = new QAction("-45", this); connect(negFortyFiveAction, SIGNAL(triggered(bool)), this, SLOT(NegFortyFiveActionTriggered(bool)), Qt::QueuedConnection); this->addAction(negFortyFiveAction); //Need this for it to show up properly. this->setContextMenuPolicy(Qt::ActionsContextMenu); } void AffineDoubleSpinBox::NegOneActionTriggered(bool checked) { setValue(-1); } void AffineDoubleSpinBox::ZeroActionTriggered(bool checked) { setValue(0); } void AffineDoubleSpinBox::OneActionTriggered(bool checked) { setValue(1); } void AffineDoubleSpinBox::FortyFiveActionTriggered(bool checked) { setValue(0.707107); } void AffineDoubleSpinBox::NegFortyFiveActionTriggered(bool checked) { setValue(-0.707107); }