summaryrefslogtreecommitdiff
path: root/gtk2_ardour/virtual_keyboard_window.cc
diff options
context:
space:
mode:
authorRobin Gareus <robin@gareus.org>2019-12-16 15:02:39 +0100
committerRobin Gareus <robin@gareus.org>2019-12-16 15:02:39 +0100
commit322e6e08c436d98402c430fff7af5196137fe292 (patch)
treef44f55f85a44424b29099fb0b0fad85d4346748c /gtk2_ardour/virtual_keyboard_window.cc
parent2add7302637ba645e946ee127fe87c929b84212e (diff)
Vkeybd: grab all key-events and use timer instead of key-repeat
This fixes an issue with arrow-keys (up/down, left/right). Those were previously only handled when the Virtual Keyboard window itself had focus. Also key-repeat for pitch-bend is now ignored and a dedicated timer is used to queue events. This fixes an issue with the first repeat taking longer than successive ones, and makes this feature independent of any desktop user settings.
Diffstat (limited to 'gtk2_ardour/virtual_keyboard_window.cc')
-rw-r--r--gtk2_ardour/virtual_keyboard_window.cc102
1 files changed, 47 insertions, 55 deletions
diff --git a/gtk2_ardour/virtual_keyboard_window.cc b/gtk2_ardour/virtual_keyboard_window.cc
index 5ce4b28b75..f277693386 100644
--- a/gtk2_ardour/virtual_keyboard_window.cc
+++ b/gtk2_ardour/virtual_keyboard_window.cc
@@ -243,6 +243,8 @@ VirtualKeyboardWindow::VirtualKeyboardWindow ()
_piano.NoteOn.connect (sigc::mem_fun (*this, &VirtualKeyboardWindow::note_on_event_handler));
_piano.NoteOff.connect (sigc::mem_fun (*this, &VirtualKeyboardWindow::note_off_event_handler));
+ _piano.SwitchOctave.connect (sigc::mem_fun (*this, &VirtualKeyboardWindow::octave_key_event_handler));
+ _piano.PitchBend.connect (sigc::mem_fun (*this, &VirtualKeyboardWindow::pitch_bend_key_event_handler));
update_velocity_settings (0);
update_octave_range ();
@@ -379,41 +381,6 @@ VirtualKeyboardWindow::on_key_press_event (GdkEventKey* ev)
_piano.grab_focus ();
- /* handle up/down */
- // XXX consider to handle these in APianoKeyboard::on_key_press_event
- // and use signals. -- also subscribe SustainChanged, indicate sustain.
- // TODO: pitch-bend shortcuts
- if (ev->type == GDK_KEY_PRESS) {
- switch (ev->keyval) {
- case GDK_KEY_Left:
- _piano_octave_key.set_value (_piano_octave_key.get_value_as_int () - 1);
- return true;
- case GDK_KEY_Right:
- _piano_octave_key.set_value (_piano_octave_key.get_value_as_int () + 1);
- return true;
- case GDK_KEY_F1:
- _pitch_adjustment.set_value (0);
- return true;
- case GDK_KEY_F2:
- _pitch_adjustment.set_value (4096);
- return true;
- case GDK_KEY_F3:
- _pitch_adjustment.set_value (12288);
- return true;
- case GDK_KEY_F4:
- _pitch_adjustment.set_value (16383);
- return true;
- case GDK_KEY_Down:
- _pitch_adjustment.set_value (std::max(0., _pitch_adjustment.get_value() - 1024));
- return true;
- case GDK_KEY_Up:
- _pitch_adjustment.set_value (std::min(16383., _pitch_adjustment.get_value() + 1024));
- return true;
- default:
- break;
- }
- }
-
return ARDOUR_UI_UTILS::relay_key_press (ev, this);
}
@@ -429,26 +396,6 @@ VirtualKeyboardWindow::on_key_release_event (GdkEventKey* ev)
_piano.grab_focus ();
- if (ev->type == GDK_KEY_RELEASE) {
- switch (ev->keyval) {
- case GDK_KEY_F1:
- /* fallthrough */
- case GDK_KEY_F2:
- /* fallthrough */
- case GDK_KEY_F3:
- /* fallthrough */
- case GDK_KEY_F4:
- /* fallthrough */
- case GDK_KEY_Up:
- /* fallthrough */
- case GDK_KEY_Down:
- _pitch_adjustment.set_value (8192);
- return true;
- default:
- break;
- }
- }
-
return ArdourWindow::on_key_release_event (ev);
}
@@ -605,6 +552,51 @@ VirtualKeyboardWindow::update_sensitivity ()
}
void
+VirtualKeyboardWindow::octave_key_event_handler (bool up)
+{
+ if (up) {
+ _piano_octave_key.set_value (_piano_octave_key.get_value_as_int () - 1);
+ } else {
+ _piano_octave_key.set_value (_piano_octave_key.get_value_as_int () + 1);
+ }
+}
+
+void
+VirtualKeyboardWindow::pitch_bend_key_event_handler (int target, bool interpolate)
+{
+ if (_pitch_adjustment.get_value() == target) {
+ return;
+ }
+ if (interpolate) {
+ _pitch_bend_target = target;
+ if (!_bender_connection.connected ()){
+ _bender_connection = Glib::signal_timeout().connect (sigc::mem_fun(*this, &VirtualKeyboardWindow::pitch_bend_timeout), 50 /*ms*/);
+ } else {
+ }
+ return;
+ }
+ _bender_connection.disconnect ();
+ _pitch_adjustment.set_value (target);
+ _pitch_bend_target = target;
+}
+
+bool
+VirtualKeyboardWindow::pitch_bend_timeout ()
+{
+ int cur = _pitch_adjustment.get_value();
+ int target;
+ if (cur < _pitch_bend_target) {
+ target = std::min (_pitch_bend_target, cur + 1024);
+ } else if (cur > _pitch_bend_target) {
+ target = std::max (_pitch_bend_target, cur - 1024);
+ } else {
+ target = _pitch_bend_target;
+ }
+ _pitch_adjustment.set_value (target);
+ return _pitch_bend_target != target;
+}
+
+void
VirtualKeyboardWindow::pitch_slider_adjusted ()
{
_pitchbend->set_value (_pitch_adjustment.get_value (), PBD::Controllable::NoGroup);