13
0
livetrax/gtk2_ardour/shuttle_control.cc
Robin Gareus 417f3647cc
Keep shuttle ctrl in sync with actual speed
When using "VS" the numeric control shows the "default speed"
when not rolling. Then when starting to roll default speed will
be the actual speed.

Previously the shuttle was not updated when `last_speed_displayed`
was the current speed. This dated back to before "VS" showed
default speed.
2023-10-10 04:18:53 +02:00

813 lines
22 KiB
C++

/*
* Copyright (C) 2011-2016 Paul Davis <paul@linuxaudiosystems.com>
* Copyright (C) 2011 Carl Hetherington <carl@carlh.net>
* Copyright (C) 2011 David Robillard <d@drobilla.net>
* Copyright (C) 2012-2019 Robin Gareus <robin@gareus.org>
* Copyright (C) 2015 Tim Mayberry <mojofunk@gmail.com>
*
* 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.
*/
#define BASELINESTRETCH (1.25)
#define MARKER_SIZE (0.5) // * height
#include <algorithm>
#include <cairo.h>
#include "pbd/unwind.h"
#include "ardour/ardour.h"
#include "ardour/audioengine.h"
#include "ardour/rc_configuration.h"
#include "ardour/port.h"
#include "ardour/session.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/tooltips.h"
#include "actions.h"
#include "rgb_macros.h"
#include "shuttle_control.h"
#include "timers.h"
#include "pbd/i18n.h"
using namespace Gtk;
using namespace Gtkmm2ext;
using namespace ARDOUR;
using namespace ArdourWidgets;
using std::max;
using std::min;
gboolean
qt (gboolean, gint, gint, gboolean, Gtk::Tooltip*, gpointer)
{
return FALSE;
}
ShuttleInfoButton::~ShuttleInfoButton ()
{
delete disp_context_menu;
}
ShuttleInfoButton::ShuttleInfoButton ()
: disp_context_menu (0)
, _ignore_change (false)
{
set_layout_font (UIConfiguration::instance ().get_NormalFont ());
set_tooltip (*this, _("Speed Display (Context-click for options)"));
set_visual_state (Gtkmm2ext::NoVisualState);
set_elements (ArdourButton::Text);
parameter_changed ("shuttle-units");
Config->ParameterChanged.connect (parameter_connection, MISSING_INVALIDATOR, boost::bind (&ShuttleInfoButton::parameter_changed, this, _1), gui_context ());
add_events (Gdk::ENTER_NOTIFY_MASK | Gdk::LEAVE_NOTIFY_MASK | Gdk::BUTTON_RELEASE_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::POINTER_MOTION_MASK | Gdk::SCROLL_MASK);
}
void
ShuttleInfoButton::set_shuttle_units (ShuttleUnits s)
{
if (_ignore_change) {
return;
}
Config->set_shuttle_units (s);
}
void
ShuttleInfoButton::parameter_changed (std::string p)
{
/* units changed; recreate the menu when it is next opened to show the current selection*/
if (p == "shuttle-units") {
delete disp_context_menu;
disp_context_menu = 0;
if (Config->get_shuttle_units() == Percentage) {
set_sizing_text (S_("LogestShuttle|> 888.9%"));
} else {
set_sizing_text (S_("LogestShuttle|> +00 st"));
}
}
}
bool
ShuttleInfoButton::on_button_press_event (GdkEventButton* ev)
{
if (Keyboard::is_context_menu_event (ev)) {
if (disp_context_menu == 0) {
build_disp_context_menu ();
}
disp_context_menu->popup (ev->button, ev->time);
return true;
}
return true;
}
void
ShuttleInfoButton::build_disp_context_menu ()
{
PBD::Unwinder<bool> uw (_ignore_change, true);
using namespace Menu_Helpers;
disp_context_menu = new Gtk::Menu ();
MenuList& items = disp_context_menu->items ();
RadioMenuItem::Group units_group;
items.push_back (RadioMenuElem (units_group, _("Percent"), sigc::bind (sigc::mem_fun (*this, &ShuttleInfoButton::set_shuttle_units), Percentage)));
if (Config->get_shuttle_units () == Percentage) {
static_cast<RadioMenuItem*> (&items.back ())->set_active ();
}
items.push_back (RadioMenuElem (units_group, _("Semitones"), sigc::bind (sigc::mem_fun (*this, &ShuttleInfoButton::set_shuttle_units), Semitones)));
if (Config->get_shuttle_units () == Semitones) {
static_cast<RadioMenuItem*> (&items.back ())->set_active ();
}
}
/* ****************************************************************************/
ShuttleControl::ShuttleControl ()
: _controllable (new ShuttleControllable (*this))
, binding_proxy (_controllable)
{
set_tooltip (*this, _("Shuttle speed control (Context-click for options)"));
pattern = 0;
shine_pattern = 0;
last_shuttle_request = 0;
last_speed_displayed = -99999999;
last_shuttle_fract = -99999999;
shuttle_grabbed = false;
shuttle_speed_on_grab = 0;
shuttle_fract = 0.0;
shuttle_max_speed = Config->get_max_transport_speed ();
shuttle_context_menu = 0;
_hovering = false;
_ignore_change = false;
set_can_focus ();
add_events (Gdk::ENTER_NOTIFY_MASK | Gdk::LEAVE_NOTIFY_MASK | Gdk::BUTTON_RELEASE_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::POINTER_MOTION_MASK | Gdk::SCROLL_MASK);
set_name (X_("ShuttleControl"));
ensure_style ();
shuttle_max_speed = Config->get_shuttle_max_speed ();
if (shuttle_max_speed >= Config->get_max_transport_speed ()) {
shuttle_max_speed = Config->get_max_transport_speed ();
} else if (shuttle_max_speed >= 6.f) {
shuttle_max_speed = 6.0f;
} else if (shuttle_max_speed >= 4.f) {
shuttle_max_speed = 4.0f;
} else if (shuttle_max_speed >= 3.f) {
shuttle_max_speed = 3.0f;
} else if (shuttle_max_speed >= 2.f) {
shuttle_max_speed = 2.0f;
} else {
shuttle_max_speed = 1.5f;
}
Config->ParameterChanged.connect (parameter_connection, MISSING_INVALIDATOR, boost::bind (&ShuttleControl::parameter_changed, this, _1), gui_context ());
Port::ResamplerQualityChanged.connect (port_connection, MISSING_INVALIDATOR, boost::bind (&ShuttleControl::parameter_changed, this, "external-sync"), gui_context ());
UIConfiguration::instance ().ColorsChanged.connect (sigc::mem_fun (*this, &ShuttleControl::set_colors));
Timers::blink_connect (sigc::mem_fun (*this, &ShuttleControl::do_blink));
set_tooltip (_vari_button, _("Varispeed: change the default playback and recording speed"));
_vari_button.set_name ("vari button");
_vari_button.set_text (S_("VariSpeed|VS"));
_vari_button.signal_clicked.connect (sigc::mem_fun (*this, &ShuttleControl::varispeed_button_clicked));
_vari_button.signal_scroll_event ().connect (sigc::mem_fun (*this, &ShuttleControl::varispeed_button_scroll_event), false);
/* gtkmm 2.4: the C++ wrapper doesn't work */
g_signal_connect ((GObject*)gobj (), "query-tooltip", G_CALLBACK (qt), NULL);
// signal_query_tooltip().connect (sigc::mem_fun (*this, &ShuttleControl::on_query_tooltip));
}
ShuttleControl::~ShuttleControl ()
{
cairo_pattern_destroy (pattern);
cairo_pattern_destroy (shine_pattern);
delete shuttle_context_menu;
}
void
ShuttleControl::varispeed_button_clicked ()
{
if (_session->default_play_speed () == 1.0 && !_vari_dialog.get_visible ()) {
_vari_dialog.present ();
} else {
_vari_dialog.hide ();
}
}
bool
ShuttleControl::varispeed_button_scroll_event (GdkEventScroll* ev)
{
double semi = 1.0;
if (ev->state & Gtkmm2ext::Keyboard::GainFineScaleModifier) {
if (ev->state & Gtkmm2ext::Keyboard::GainExtraFineScaleModifier) {
semi = 0.1;
} else {
semi = 0.5;
}
}
switch (ev->direction) {
case GDK_SCROLL_UP:
case GDK_SCROLL_RIGHT:
_vari_dialog.present ();
_vari_dialog.adj_semi (semi);
break;
case GDK_SCROLL_DOWN:
case GDK_SCROLL_LEFT:
_vari_dialog.present ();
_vari_dialog.adj_semi (-semi);
break;
}
return true;
}
void
ShuttleControl::do_blink (bool onoff)
{
if (!shuttle_grabbed && _session && _session->default_play_speed () != 1.0) {
_vari_button.set_active (onoff);
if (_session->actual_speed () == 0) {
/* update info button text */
queue_draw ();
}
} else {
_vari_button.set_active (false);
}
}
void
ShuttleControl::set_session (Session* s)
{
SessionHandlePtr::set_session (s);
_vari_dialog.set_session (_session);
if (_session) {
_session->add_controllable (_controllable);
_info_button.set_session (s);
_session->config.ParameterChanged.connect (_session_connections, MISSING_INVALIDATOR, boost::bind (&ShuttleControl::parameter_changed, this, _1), gui_context());
/* set sensitivity */
parameter_changed ("external-sync");
} else {
_vari_dialog.hide ();
_vari_button.set_sensitive (false);
set_sensitive (false);
}
}
void
ShuttleControl::on_size_allocate (Gtk::Allocation& alloc)
{
if (pattern) {
cairo_pattern_destroy (pattern);
pattern = 0;
cairo_pattern_destroy (shine_pattern);
shine_pattern = 0;
}
CairoWidget::on_size_allocate (alloc);
/* background */
pattern = cairo_pattern_create_linear (0, 0, 0, alloc.get_height ());
uint32_t col = UIConfiguration::instance ().color ("shuttle");
int r, b, g, a;
UINT_TO_RGBA (col, &r, &g, &b, &a);
cairo_pattern_add_color_stop_rgb (pattern, 0.0, r / 400.0, g / 400.0, b / 400.0);
cairo_pattern_add_color_stop_rgb (pattern, 0.4, r / 255.0, g / 255.0, b / 255.0);
cairo_pattern_add_color_stop_rgb (pattern, 1.0, r / 512.0, g / 512.0, b / 512.0);
/* reflection */
shine_pattern = cairo_pattern_create_linear (0.0, 0.0, 0.0, 10);
cairo_pattern_add_color_stop_rgba (shine_pattern, 0, 1, 1, 1, 0.0);
cairo_pattern_add_color_stop_rgba (shine_pattern, 0.2, 1, 1, 1, 0.4);
cairo_pattern_add_color_stop_rgba (shine_pattern, 1, 1, 1, 1, 0.1);
}
void
ShuttleControl::map_transport_state ()
{
float speed = 0.0;
if (_session) {
speed = _session->actual_speed ();
}
if (fabs (speed) <= (2 * DBL_EPSILON)) {
shuttle_fract = 0;
} else {
if (Config->get_shuttle_units () == Semitones) {
shuttle_fract = speed / shuttle_max_speed;
} else {
shuttle_fract = speed_as_fract (speed);
}
}
if ((fabsf (shuttle_fract - last_shuttle_fract) < 0.005f)) {
/* dead-zone */
return;
}
queue_draw ();
}
void
ShuttleControl::build_shuttle_context_menu ()
{
PBD::Unwinder<bool> uw (_ignore_change, true);
using namespace Menu_Helpers;
shuttle_context_menu = new Menu ();
MenuList& items = shuttle_context_menu->items ();
float max_transport_speed = Config->get_max_transport_speed ();
RadioMenuItem::Group speed_group;
Menu* speed_menu = manage (new Menu ());
MenuList& speed_items = speed_menu->items ();
if (max_transport_speed >= 8) {
speed_items.push_back (RadioMenuElem (speed_group, "8", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 8.0f)));
if (shuttle_max_speed == 8.0) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
if (max_transport_speed >= 6) {
speed_items.push_back (RadioMenuElem (speed_group, "6", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 6.0f)));
if (shuttle_max_speed == 6.0) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
if (max_transport_speed >= 4) {
speed_items.push_back (RadioMenuElem (speed_group, "4", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 4.0f)));
if (shuttle_max_speed == 4.0) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
if (max_transport_speed >= 3) {
speed_items.push_back (RadioMenuElem (speed_group, "3", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 3.0f)));
if (shuttle_max_speed == 3.0) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
if (max_transport_speed >= 2) {
speed_items.push_back (RadioMenuElem (speed_group, "2", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 2.0f)));
if (shuttle_max_speed == 2.0) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
if (max_transport_speed >= 1.5) {
speed_items.push_back (RadioMenuElem (speed_group, "1.5", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 1.5f)));
if (shuttle_max_speed == 1.5) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
} else {
assert (max_transport_speed >= 1);
speed_items.push_back (RadioMenuElem (speed_group, "1", sigc::bind (sigc::mem_fun (*this, &ShuttleControl::set_shuttle_max_speed), 1.f)));
if (shuttle_max_speed == 1) {
static_cast<RadioMenuItem*> (&speed_items.back ())->set_active ();
}
}
items.push_back (MenuElem (_("Maximum speed"), *speed_menu));
}
void
ShuttleControl::set_shuttle_max_speed (float speed)
{
if (_ignore_change) {
return;
}
assert (speed <= Config->get_max_transport_speed ());
Config->set_shuttle_max_speed (speed);
}
bool
ShuttleControl::on_button_press_event (GdkEventButton* ev)
{
if (!_session) {
return true;
}
if (binding_proxy.button_press_handler (ev)) {
return true;
}
if (Keyboard::is_context_menu_event (ev)) {
if (shuttle_context_menu == 0) {
build_shuttle_context_menu ();
}
shuttle_context_menu->popup (ev->button, ev->time);
return true;
}
switch (ev->button) {
case 1:
if (Keyboard::modifier_state_equals (ev->state, Keyboard::TertiaryModifier)) {
_session->reset_transport_speed ();
} else {
add_modal_grab ();
shuttle_grabbed = true;
shuttle_speed_on_grab = _session->actual_speed ();
requested_speed = shuttle_speed_on_grab;
mouse_shuttle (ev->x, true);
gdk_pointer_grab (ev->window, false,
GdkEventMask (Gdk::POINTER_MOTION_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::BUTTON_RELEASE_MASK),
NULL, NULL, ev->time);
}
break;
case 2:
case 3:
default:
return true;
break;
}
return true;
}
bool
ShuttleControl::on_button_release_event (GdkEventButton* ev)
{
if (!_session) {
return true;
}
switch (ev->button) {
case 1:
if (shuttle_grabbed) {
shuttle_grabbed = false;
remove_modal_grab ();
gdk_pointer_ungrab (GDK_CURRENT_TIME);
if (shuttle_speed_on_grab == 0) {
_session->request_stop ();
} else {
_session->request_transport_speed (shuttle_speed_on_grab);
}
}
return true;
case 2:
case 3:
default:
return true;
}
return true;
}
bool
ShuttleControl::on_query_tooltip (int, int, bool, const Glib::RefPtr<Gtk::Tooltip>&)
{
return false;
}
bool
ShuttleControl::on_motion_notify_event (GdkEventMotion* ev)
{
if (!_session || !shuttle_grabbed) {
return true;
}
return mouse_shuttle (ev->x, false);
}
gint
ShuttleControl::mouse_shuttle (double x, bool force)
{
double const center = get_width () / 2.0;
double distance_from_center = x - center;
if (distance_from_center > 0) {
distance_from_center = min (distance_from_center, center);
} else {
distance_from_center = max (distance_from_center, -center);
}
/* compute shuttle fract as expressing how far between the center
and the edge we are. positive values indicate we are right of
center, negative values indicate left of center
*/
float marker_size = round (get_height () * MARKER_SIZE);
float avail_width = get_width () - marker_size;
shuttle_fract = 2 * distance_from_center / avail_width;
use_shuttle_fract (force);
return true;
}
void
ShuttleControl::set_shuttle_fract (double f, bool zero_ok)
{
shuttle_fract = f;
use_shuttle_fract (false, zero_ok);
}
int
ShuttleControl::speed_as_semitones (float speed, bool& reverse)
{
assert (speed != 0.0);
if (speed < 0.0) {
reverse = true;
return (int)round (12.0 * fast_log2 (-speed));
} else {
reverse = false;
return (int)round (12.0 * fast_log2 (speed));
}
}
int
ShuttleControl::speed_as_cents (float speed, bool& reverse)
{
assert (speed != 0.0);
if (speed < 0.0) {
reverse = true;
return (int)ceilf (1200.0 * fast_log2 (-speed));
} else {
reverse = false;
return (int)ceilf (1200.0 * fast_log2 (speed));
}
}
float
ShuttleControl::speed_as_fract (float speed) const
{
float fract = speed / shuttle_max_speed;
if (fract < 0) {
return -sqrtf (-fract);
} else {
return sqrtf (fract);
}
}
float
ShuttleControl::fract_as_speed (float fract) const
{
if (fract < 0) {
return -fract * fract * shuttle_max_speed;
} else {
return fract * fract * shuttle_max_speed;
}
}
void
ShuttleControl::use_shuttle_fract (bool force, bool zero_ok)
{
PBD::microseconds_t now = PBD::get_microseconds ();
shuttle_fract = max (-1.0f, shuttle_fract);
shuttle_fract = min (1.0f, shuttle_fract);
/* do not attempt to submit a motion-driven transport speed request
more than once per process cycle.
*/
if (!force && (now - last_shuttle_request) < (PBD::microseconds_t)AudioEngine::instance ()->usecs_per_cycle ()) {
return;
}
last_shuttle_request = now;
double speed = 0;
if (Config->get_shuttle_units () == Semitones) {
speed = shuttle_fract * shuttle_max_speed;
} else {
speed = fract_as_speed (shuttle_fract);
}
requested_speed = speed;
if (_session) {
if (zero_ok) {
_session->request_transport_speed (speed);
} else {
_session->request_transport_speed_nonzero (speed);
}
if (speed != 0 && !_session->transport_state_rolling ()) {
_session->request_roll ();
} else if (speed == 0 && zero_ok && _session->transport_state_rolling ()) {
_session->request_stop ();
}
}
}
void
ShuttleControl::set_colors ()
{
int r, g, b, a;
uint32_t bg_color = UIConfiguration::instance ().color (X_("shuttle bg"));
UINT_TO_RGBA (bg_color, &r, &g, &b, &a);
bg_r = r / 255.0;
bg_g = g / 255.0;
bg_b = b / 255.0;
}
void
ShuttleControl::render (Cairo::RefPtr<Cairo::Context> const& ctx, cairo_rectangle_t*)
{
cairo_t* cr = ctx->cobj ();
// center slider line
float yc = get_height () / 2;
float lw = 3;
cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
cairo_set_line_width (cr, 3);
cairo_move_to (cr, lw, yc);
cairo_line_to (cr, get_width () - lw, yc);
cairo_set_source_rgb (cr, bg_r, bg_g, bg_b);
if (UIConfiguration::instance ().get_widget_prelight () && _hovering) {
cairo_stroke_preserve (cr);
cairo_set_source_rgba (cr, 1, 1, 1, 0.15);
}
cairo_stroke (cr);
float speed = 0.0;
float actual_speed = 0.0;
float default_speed = 1.0;
char buf[32];
if (_session) {
speed = _session->actual_speed ();
actual_speed = speed;
default_speed = _session->default_play_speed ();
if (shuttle_grabbed) {
speed = requested_speed;
}
}
/* marker */
float visual_fraction = std::max (-1.0f, std::min (1.0f, shuttle_fract));
float marker_size = round (get_height () * MARKER_SIZE);
float avail_width = get_width () - marker_size;
float x = 0.5 * (get_width () + visual_fraction * avail_width - marker_size);
rounded_rectangle (cr, x, 0, marker_size, get_height (), 5);
cairo_set_source_rgba (cr, 0, 0, 0, 1);
cairo_fill (cr);
rounded_rectangle (cr, x + 1, 1, marker_size - 2, get_height () - 2, 3.5);
if (flat_buttons ()) {
uint32_t col = UIConfiguration::instance ().color ("shuttle");
Gtkmm2ext::set_source_rgba (cr, col);
} else {
cairo_set_source (cr, pattern);
}
if (UIConfiguration::instance ().get_widget_prelight () && _hovering) {
cairo_fill_preserve (cr);
cairo_set_source_rgba (cr, 1, 1, 1, 0.15);
}
cairo_fill (cr);
/* text */
if (actual_speed == 1.0) {
snprintf (buf, sizeof (buf), "%s", _("Play"));
} else if (actual_speed == 0 && default_speed == 1.0) {
snprintf (buf, sizeof (buf), "%s", _("Stop"));
} else {
if (actual_speed == 0) {
/*default_play_speed (varispeed) is always forward */
actual_speed = default_speed;
}
if (Config->get_shuttle_units() == Percentage) {
if (actual_speed < 0.0) {
snprintf (buf, sizeof (buf), "< %.1f%%", -actual_speed * 100.f);
} else {
snprintf (buf, sizeof (buf), "> %.1f%%", actual_speed * 100.f);
}
} else {
bool reversed;
int semi = speed_as_semitones (actual_speed, reversed);
if (reversed) {
snprintf (buf, sizeof (buf), _("< %+2d st"), semi);
} else {
snprintf (buf, sizeof (buf), _("> %+2d st"), semi);
}
}
}
last_speed_displayed = actual_speed;
last_shuttle_fract = shuttle_fract;
_info_button.set_text (buf);
#if 0
if (UIConfiguration::instance().get_widget_prelight()) {
if (_hovering) {
rounded_rectangle (cr, 0, 0, get_width(), get_height(), 3.5);
cairo_set_source_rgba (cr, 1, 1, 1, 0.15);
cairo_fill (cr);
}
}
#endif
}
ShuttleControl::ShuttleControllable::ShuttleControllable (ShuttleControl& s)
: PBD::Controllable (X_("Shuttle"))
, sc (s)
{
}
void
ShuttleControl::ShuttleControllable::set_value (double val, PBD::Controllable::GroupControlDisposition /*group_override*/)
{
sc.set_shuttle_fract ((val - lower ()) / (upper () - lower ()), true);
}
double
ShuttleControl::ShuttleControllable::get_value () const
{
return lower () + (sc.get_shuttle_fract () * (upper () - lower ()));
}
void
ShuttleControl::parameter_changed (std::string p)
{
if (p == "shuttle-units") {
map_transport_state ();
} else if (p == "shuttle-max-speed") {
shuttle_max_speed = Config->get_shuttle_max_speed ();
last_speed_displayed = -99999999;
last_shuttle_fract = -99999999;
map_transport_state ();
use_shuttle_fract (true);
delete shuttle_context_menu;
shuttle_context_menu = 0;
} else if (p == "external-sync") {
if (!_session || _session->config.get_external_sync() || !Port::can_varispeed ()) {
_vari_dialog.hide ();
_vari_button.set_sensitive (false);
if (shuttle_grabbed) {
shuttle_grabbed = false;
remove_modal_grab ();
gdk_pointer_ungrab (GDK_CURRENT_TIME);
}
set_sensitive (false);
} else {
_vari_button.set_sensitive (true);
set_sensitive (true);
}
}
}
bool
ShuttleControl::on_enter_notify_event (GdkEventCrossing* ev)
{
_hovering = true;
if (UIConfiguration::instance ().get_widget_prelight ()) {
queue_draw ();
}
return CairoWidget::on_enter_notify_event (ev);
}
bool
ShuttleControl::on_leave_notify_event (GdkEventCrossing* ev)
{
_hovering = false;
if (UIConfiguration::instance ().get_widget_prelight ()) {
queue_draw ();
}
return CairoWidget::on_leave_notify_event (ev);
}