总览
每当你按下或放开一个按键,就会发出一个事件。你可以连接到信号处理函数来处理此类事件。
事件处理函数接受的参数取决于事件的类型。对于键盘事件,它接受一个GdkEventKey*参数。如附录中所述,事件处理函数返回一个bool值,以表示信号已经被完全处理(返回true)或允许事件传播(false)。
要确定哪个键被按下或放开,你需要读取GdkEventKey::keyval的值并将其与头文件<gdk/gdkkeysyms.h>中的常量进行比较。GdkEventKey::state的位标志可以用作表示修饰键(shift、ctrl等)的状态。
这是一个简单的例子:
bool on_key_press_or_release_event(GdkEventKey* event) { if (event->type == GDK_KEY_PRESS && event->keyval == GDK_KEY_1 && (event->state & (GDK_SHIFT_MASK | GDK_CONTROL_MASK | GDK_MOD1_MASK)) == GDK_MOD1_MASK) { handle_alt_1_press(); // GDK_MOD1_MASK is normally the Alt key return true; } return false; } Gtk::Entry m_entry; // in a class definition // in the class constructor m_entry.signal_key_press_event().connect( sigc::ptr_fun(&on_key_press_or_release_event) ); m_entry.signal_key_release_event().connect( sigc::ptr_fun(&on_key_press_or_release_event) );
- 21.1.1. 示例
21.1.1. 示例
在此示例中,有三个键盘快捷键:Alt+1选择第一个单选按钮、Alt+2选择第二个单选按钮、Esc隐藏(关闭)窗口。默认信号处理程序被覆盖如附录中覆盖默认信号处理程序小节所述。
File: examplewindow.h (For use with gtkmm 4)
#ifndef GTKMM_EXAMPLEWINDOW_H #define GTKMM_EXAMPLEWINDOW_H #include <gtkmm.h> class ExampleWindow : public Gtk::Window { public: ExampleWindow(); virtual ~ExampleWindow(); private: // Signal handler: bool on_window_key_pressed(guint keyval, guint keycode, Gdk::ModifierType state); Gtk::Box m_container; Gtk::CheckButton m_first; Gtk::CheckButton m_second; }; #endif //GTKMM_EXAMPLEWINDOW_H
File: main.cc (For use with gtkmm 4)
#include "examplewindow.h" #include <gtkmm/application.h> int main(int argc, char *argv[]) { auto app = Gtk::Application::create("org.gtkmm.example"); //Shows the window and returns when it is closed. return app->make_window_and_run<ExampleWindow>(argc, argv); }
File: examplewindow.cc (For use with gtkmm 4)
#include "examplewindow.h" ExampleWindow::ExampleWindow() { set_title("Keyboard Events"); m_container.set_margin(10); set_child(m_container); // Radio buttons: m_first.set_label("First"); m_second.set_label("Second"); m_second.set_group(m_first); m_first.set_active(); // Main Container: m_container.set_orientation(Gtk::Orientation::HORIZONTAL); m_container.append(m_first); m_container.append(m_second); // Events. auto controller = Gtk::EventControllerKey::create(); controller->signal_key_pressed().connect( sigc::mem_fun(*this, &ExampleWindow::on_window_key_pressed), false); add_controller(controller); } bool ExampleWindow::on_window_key_pressed(guint keyval, guint, Gdk::ModifierType state) { //Gdk::ModifierType::ALT_MASK -> the 'Alt' key(mask) //GDK_KEY_1 -> the '1' key //GDK_KEY_2 -> the '2' key //select the first radio button, when we press alt + 1 if((keyval == GDK_KEY_1) && (state & (Gdk::ModifierType::SHIFT_MASK | Gdk::ModifierType::CONTROL_MASK | Gdk::ModifierType::ALT_MASK)) == Gdk::ModifierType::ALT_MASK) { m_first.set_active(); //returning true, cancels the propagation of the event return true; } else if((keyval == GDK_KEY_2) && (state & (Gdk::ModifierType::SHIFT_MASK | Gdk::ModifierType::CONTROL_MASK | Gdk::ModifierType::ALT_MASK)) == Gdk::ModifierType::ALT_MASK) { //and the second radio button, when we press alt + 2 m_second.set_active(); return true; } else if(keyval == GDK_KEY_Escape) { //close the window, when the 'esc' key is pressed hide(); return true; } //the event has not been handled return false; } ExampleWindow::~ExampleWindow() { }