Separate ArdourKnob core functionality into abstract base class
This commit is contained in:
parent
5834728e2f
commit
52f12cac8b
395
libs/widgets/ardour_ctrl_base.cc
Normal file
395
libs/widgets/ardour_ctrl_base.cc
Normal file
@ -0,0 +1,395 @@
|
||||
/*
|
||||
* Copyright (C) 2010 Paul Davis <paul@linuxaudiosystems.com>
|
||||
* Copyright (C) 2017-2021 Robin Gareus <robin@gareus.org>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*/
|
||||
|
||||
#include <iostream>
|
||||
#include <cmath>
|
||||
#include <algorithm>
|
||||
|
||||
#include <pangomm/layout.h>
|
||||
|
||||
#include "pbd/compose.h"
|
||||
#include "pbd/controllable.h"
|
||||
#include "pbd/error.h"
|
||||
|
||||
#include "gtkmm2ext/colors.h"
|
||||
#include "gtkmm2ext/gui_thread.h"
|
||||
#include "gtkmm2ext/keyboard.h"
|
||||
#include "gtkmm2ext/rgb_macros.h"
|
||||
#include "gtkmm2ext/utils.h"
|
||||
|
||||
#include "widgets/ardour_ctrl_base.h"
|
||||
#include "widgets/ui_config.h"
|
||||
|
||||
#include "pbd/i18n.h"
|
||||
|
||||
using namespace Gtkmm2ext;
|
||||
using namespace ArdourWidgets;
|
||||
using namespace Gtk;
|
||||
using namespace Glib;
|
||||
using namespace PBD;
|
||||
using std::max;
|
||||
using std::min;
|
||||
using namespace std;
|
||||
|
||||
ArdourCtrlBase::ArdourCtrlBase (Flags flags)
|
||||
: _req_width (0)
|
||||
, _req_height (0)
|
||||
, _hovering (false)
|
||||
, _val (0)
|
||||
, _normal (0)
|
||||
, _flags (flags)
|
||||
, _tooltip (this)
|
||||
, _grabbed_x (0)
|
||||
, _grabbed_y (0)
|
||||
, _dead_zone_delta (0)
|
||||
{
|
||||
UIConfigurationBase::instance().ColorsChanged.connect (sigc::mem_fun (*this, &ArdourCtrlBase::color_handler));
|
||||
|
||||
#ifdef VBM
|
||||
_flags = (Flags)(static_cast <int>(_flags) | (int)NoHorizontal);
|
||||
#endif
|
||||
}
|
||||
|
||||
ArdourCtrlBase::~ArdourCtrlBase()
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::set_size_request (int w, int h)
|
||||
{
|
||||
if (_req_width == w && _req_height == h) {
|
||||
return;
|
||||
}
|
||||
_req_width = w;
|
||||
_req_height = h;
|
||||
queue_resize ();
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::on_size_request (Gtk::Requisition* req)
|
||||
{
|
||||
req->width = _req_width;
|
||||
req->height = _req_height;
|
||||
if (req->width < 1) { req->width = 13; }
|
||||
if (req->height < 1) { req->height = 13; }
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_scroll_event (GdkEventScroll* ev)
|
||||
{
|
||||
/* mouse wheel */
|
||||
|
||||
float scale = 0.05; //by default, we step in 1/20ths of the knob travel
|
||||
if (ev->state & Keyboard::GainFineScaleModifier) {
|
||||
if (ev->state & Keyboard::GainExtraFineScaleModifier) {
|
||||
scale *= 0.01;
|
||||
} else {
|
||||
scale *= 0.10;
|
||||
}
|
||||
}
|
||||
if (_flags & Reverse) {
|
||||
scale *= -1;
|
||||
}
|
||||
|
||||
boost::shared_ptr<PBD::Controllable> c = binding_proxy.get_controllable();
|
||||
if (c) {
|
||||
float val = c->get_interface (true);
|
||||
|
||||
if ( ev->direction == GDK_SCROLL_UP )
|
||||
val += scale;
|
||||
else
|
||||
val -= scale;
|
||||
|
||||
c->set_interface (val, true);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_motion_notify_event (GdkEventMotion *ev)
|
||||
{
|
||||
if (!(ev->state & Gdk::BUTTON1_MASK)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
boost::shared_ptr<PBD::Controllable> c = binding_proxy.get_controllable();
|
||||
if (!c) {
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
/* scale the adjustment based on keyboard modifiers & GUI size */
|
||||
const float ui_scale = max (1.f, UIConfigurationBase::instance().get_ui_scale());
|
||||
float scale = 0.0025 / ui_scale;
|
||||
|
||||
if (ev->state & Keyboard::GainFineScaleModifier) {
|
||||
if (ev->state & Keyboard::GainExtraFineScaleModifier) {
|
||||
scale *= 0.01;
|
||||
} else {
|
||||
scale *= 0.10;
|
||||
}
|
||||
}
|
||||
|
||||
/* calculate the travel of the mouse */
|
||||
int delta = 0;
|
||||
if ((_flags & NoVertical) == 0) {
|
||||
delta += (_grabbed_y - ev->y);
|
||||
}
|
||||
if ((_flags & NoHorizontal) == 0) {
|
||||
delta -= (_grabbed_x - ev->x);
|
||||
}
|
||||
if (delta == 0) {
|
||||
return true;
|
||||
}
|
||||
if (_flags & Reverse) {
|
||||
delta *= -1;
|
||||
}
|
||||
|
||||
_grabbed_x = ev->x;
|
||||
_grabbed_y = ev->y;
|
||||
float val = c->get_interface (true);
|
||||
|
||||
if (_flags & Detent) {
|
||||
const float px_deadzone = 42.f * ui_scale;
|
||||
|
||||
if ((val - _normal) * (val - _normal + delta * scale) < 0) {
|
||||
/* detent */
|
||||
const int tozero = (_normal - val) * scale;
|
||||
int remain = delta - tozero;
|
||||
if (abs (remain) > px_deadzone) {
|
||||
/* slow down passing the default value */
|
||||
remain += (remain > 0) ? px_deadzone * -.5 : px_deadzone * .5;
|
||||
delta = tozero + remain;
|
||||
_dead_zone_delta = 0;
|
||||
} else {
|
||||
c->set_value (c->normal(), Controllable::NoGroup);
|
||||
_dead_zone_delta = remain / px_deadzone;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
if (fabsf (rintf((val - _normal) / scale) + _dead_zone_delta) < 1) {
|
||||
c->set_value (c->normal(), Controllable::NoGroup);
|
||||
_dead_zone_delta += delta / px_deadzone;
|
||||
return true;
|
||||
}
|
||||
|
||||
_dead_zone_delta = 0;
|
||||
}
|
||||
|
||||
val += delta * scale;
|
||||
c->set_interface (val, true);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_button_press_event (GdkEventButton *ev)
|
||||
{
|
||||
_grabbed_x = ev->x;
|
||||
_grabbed_y = ev->y;
|
||||
_dead_zone_delta = 0;
|
||||
|
||||
if (ev->type != GDK_BUTTON_PRESS) {
|
||||
if (_grabbed) {
|
||||
remove_modal_grab();
|
||||
_grabbed = false;
|
||||
StopGesture ();
|
||||
gdk_pointer_ungrab (GDK_CURRENT_TIME);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
if (binding_proxy.button_press_handler (ev)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (ev->button != 1 && ev->button != 2) {
|
||||
return false;
|
||||
}
|
||||
|
||||
set_active_state (Gtkmm2ext::ExplicitActive);
|
||||
_tooltip.start_drag();
|
||||
add_modal_grab();
|
||||
_grabbed = true;
|
||||
StartGesture ();
|
||||
gdk_pointer_grab(ev->window,false,
|
||||
GdkEventMask( Gdk::POINTER_MOTION_MASK | Gdk::BUTTON_PRESS_MASK |Gdk::BUTTON_RELEASE_MASK),
|
||||
NULL,NULL,ev->time);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_button_release_event (GdkEventButton *ev)
|
||||
{
|
||||
_tooltip.stop_drag();
|
||||
_grabbed = false;
|
||||
StopGesture ();
|
||||
remove_modal_grab();
|
||||
gdk_pointer_ungrab (GDK_CURRENT_TIME);
|
||||
|
||||
if ( (_grabbed_y == ev->y && _grabbed_x == ev->x) && Keyboard::modifier_state_equals (ev->state, Keyboard::TertiaryModifier)) { //no move, shift-click sets to default
|
||||
boost::shared_ptr<PBD::Controllable> c = binding_proxy.get_controllable();
|
||||
if (!c) return false;
|
||||
c->set_value (c->normal(), Controllable::NoGroup);
|
||||
return true;
|
||||
}
|
||||
|
||||
unset_active_state ();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::color_handler ()
|
||||
{
|
||||
set_dirty ();
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::set_controllable (boost::shared_ptr<Controllable> c)
|
||||
{
|
||||
watch_connection.disconnect (); //stop watching the old controllable
|
||||
|
||||
if (!c) return;
|
||||
|
||||
binding_proxy.set_controllable (c);
|
||||
|
||||
c->Changed.connect (watch_connection, invalidator(*this), boost::bind (&ArdourCtrlBase::controllable_changed, this, false), gui_context());
|
||||
|
||||
_normal = c->internal_to_interface(c->normal());
|
||||
|
||||
controllable_changed();
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::controllable_changed (bool force_update)
|
||||
{
|
||||
boost::shared_ptr<PBD::Controllable> c = binding_proxy.get_controllable();
|
||||
if (!c) return;
|
||||
|
||||
float val = c->get_interface (true);
|
||||
val = min( max(0.0f, val), 1.0f); // clamp
|
||||
|
||||
if (val == _val && !force_update) {
|
||||
return;
|
||||
}
|
||||
|
||||
_val = val;
|
||||
if (!_tooltip_prefix.empty()) {
|
||||
_tooltip.set_tip (_tooltip_prefix + c->get_user_string());
|
||||
}
|
||||
set_dirty();
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::on_style_changed (const RefPtr<Gtk::Style>&)
|
||||
{
|
||||
set_dirty ();
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::on_name_changed ()
|
||||
{
|
||||
set_dirty ();
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
ArdourCtrlBase::set_active_state (Gtkmm2ext::ActiveState s)
|
||||
{
|
||||
if (_active_state != s)
|
||||
CairoWidget::set_active_state (s);
|
||||
}
|
||||
|
||||
void
|
||||
ArdourCtrlBase::set_visual_state (Gtkmm2ext::VisualState s)
|
||||
{
|
||||
if (_visual_state != s)
|
||||
CairoWidget::set_visual_state (s);
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_focus_in_event (GdkEventFocus* ev)
|
||||
{
|
||||
set_dirty ();
|
||||
return CairoWidget::on_focus_in_event (ev);
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_focus_out_event (GdkEventFocus* ev)
|
||||
{
|
||||
set_dirty ();
|
||||
return CairoWidget::on_focus_out_event (ev);
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_enter_notify_event (GdkEventCrossing* ev)
|
||||
{
|
||||
_hovering = true;
|
||||
|
||||
set_dirty ();
|
||||
|
||||
boost::shared_ptr<PBD::Controllable> c (binding_proxy.get_controllable ());
|
||||
if (c) {
|
||||
PBD::Controllable::GUIFocusChanged (boost::weak_ptr<PBD::Controllable> (c));
|
||||
}
|
||||
|
||||
return CairoWidget::on_enter_notify_event (ev);
|
||||
}
|
||||
|
||||
bool
|
||||
ArdourCtrlBase::on_leave_notify_event (GdkEventCrossing* ev)
|
||||
{
|
||||
_hovering = false;
|
||||
|
||||
set_dirty ();
|
||||
|
||||
if (binding_proxy.get_controllable()) {
|
||||
PBD::Controllable::GUIFocusChanged (boost::weak_ptr<PBD::Controllable> ());
|
||||
}
|
||||
|
||||
return CairoWidget::on_leave_notify_event (ev);
|
||||
}
|
||||
|
||||
CtrlPersistentTooltip::CtrlPersistentTooltip (Gtk::Widget* w)
|
||||
: PersistentTooltip (w, true, 3)
|
||||
, _dragging (false)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
CtrlPersistentTooltip::start_drag ()
|
||||
{
|
||||
_dragging = true;
|
||||
}
|
||||
|
||||
void
|
||||
CtrlPersistentTooltip::stop_drag ()
|
||||
{
|
||||
_dragging = false;
|
||||
}
|
||||
|
||||
bool
|
||||
CtrlPersistentTooltip::dragging () const
|
||||
{
|
||||
return _dragging;
|
||||
}
|
129
libs/widgets/widgets/ardour_ctrl_base.h
Normal file
129
libs/widgets/widgets/ardour_ctrl_base.h
Normal file
@ -0,0 +1,129 @@
|
||||
/*
|
||||
* Copyright (C) 2014 Paul Davis <paul@linuxaudiosystems.com>
|
||||
* Copyright (C) 2017-2019 Robin Gareus <robin@gareus.org>
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*/
|
||||
|
||||
#ifndef _WIDGETS_ARDOUR_CTRL_BASE_H_
|
||||
#define _WIDGETS_ARDOUR_CTRL_BASE_H_
|
||||
|
||||
#include <list>
|
||||
#include <stdint.h>
|
||||
|
||||
#include <gtkmm/action.h>
|
||||
|
||||
#include "pbd/signals.h"
|
||||
|
||||
#include "gtkmm2ext/activatable.h"
|
||||
#include "gtkmm2ext/cairo_widget.h"
|
||||
#include "gtkmm2ext/persistent_tooltip.h"
|
||||
|
||||
#include "widgets/binding_proxy.h"
|
||||
#include "widgets/visibility.h"
|
||||
|
||||
namespace ArdourWidgets {
|
||||
|
||||
class LIBWIDGETS_API CtrlPersistentTooltip : public Gtkmm2ext::PersistentTooltip
|
||||
{
|
||||
public:
|
||||
CtrlPersistentTooltip (Gtk::Widget* w);
|
||||
|
||||
void start_drag ();
|
||||
void stop_drag ();
|
||||
bool dragging () const;
|
||||
|
||||
private:
|
||||
bool _dragging;
|
||||
};
|
||||
|
||||
|
||||
class LIBWIDGETS_API ArdourCtrlBase : public CairoWidget , public Gtkmm2ext::Activatable
|
||||
{
|
||||
public:
|
||||
enum Flags {
|
||||
NoFlags = 0x0,
|
||||
Detent = 0x1,
|
||||
ArcToZero = 0x2,
|
||||
NoHorizontal = 0x4,
|
||||
NoVertical = 0x8,
|
||||
Reverse = 0x10,
|
||||
};
|
||||
|
||||
ArdourCtrlBase (Flags flags = NoFlags);
|
||||
virtual ~ArdourCtrlBase ();
|
||||
|
||||
void set_active_state (Gtkmm2ext::ActiveState);
|
||||
void set_visual_state (Gtkmm2ext::VisualState);
|
||||
|
||||
void set_tooltip_prefix (std::string pfx) { _tooltip_prefix = pfx; controllable_changed (true); }
|
||||
|
||||
boost::shared_ptr<PBD::Controllable> get_controllable() { return binding_proxy.get_controllable(); }
|
||||
void set_controllable (boost::shared_ptr<PBD::Controllable> c);
|
||||
|
||||
bool on_button_press_event (GdkEventButton*);
|
||||
bool on_button_release_event (GdkEventButton*);
|
||||
bool on_scroll_event (GdkEventScroll* ev);
|
||||
bool on_motion_notify_event (GdkEventMotion *ev) ;
|
||||
|
||||
void color_handler ();
|
||||
|
||||
sigc::signal<void> StartGesture;
|
||||
sigc::signal<void> StopGesture;
|
||||
|
||||
void set_size_request (int, int);
|
||||
|
||||
protected:
|
||||
virtual void render (Cairo::RefPtr<Cairo::Context> const&, cairo_rectangle_t*) = 0;
|
||||
|
||||
void on_size_request (Gtk::Requisition* req);
|
||||
void on_style_changed (const Glib::RefPtr<Gtk::Style>&);
|
||||
void on_name_changed ();
|
||||
bool on_enter_notify_event (GdkEventCrossing*);
|
||||
bool on_leave_notify_event (GdkEventCrossing*);
|
||||
bool on_focus_in_event (GdkEventFocus*);
|
||||
bool on_focus_out_event (GdkEventFocus*);
|
||||
|
||||
int _req_width;
|
||||
int _req_height;
|
||||
|
||||
bool _hovering;
|
||||
|
||||
float _val; // current value [0..1]
|
||||
float _normal; // default value, arc
|
||||
|
||||
Flags _flags;
|
||||
CtrlPersistentTooltip _tooltip;
|
||||
|
||||
private:
|
||||
void controllable_changed (bool force_update = false);
|
||||
void action_sensitivity_changed ();
|
||||
void action_visibility_changed ();
|
||||
void action_tooltip_changed ();
|
||||
|
||||
PBD::ScopedConnection watch_connection;
|
||||
|
||||
BindingProxy binding_proxy;
|
||||
|
||||
float _grabbed_x;
|
||||
float _grabbed_y;
|
||||
float _dead_zone_delta;
|
||||
|
||||
std::string _tooltip_prefix;
|
||||
};
|
||||
|
||||
} /* namespace */
|
||||
|
||||
#endif /* __gtk2_ardour_ardour_knob_h__ */
|
@ -26,6 +26,7 @@ top = '.'
|
||||
out = 'build'
|
||||
|
||||
widgets_sources = [
|
||||
'ardour_ctrl_base.cc',
|
||||
'ardour_button.cc',
|
||||
'ardour_display.cc',
|
||||
'ardour_dropdown.cc',
|
||||
|
Loading…
Reference in New Issue
Block a user