Merge pull request #3877 from techee/ctags_sync6
[geany-mirror.git] / src / keybindings.c
blob5cf755ab5740f2c04e2e448d4cd2f0c766dfded5
1 /*
2 * keybindings.c - this file is part of Geany, a fast and lightweight IDE
4 * Copyright 2006 The Geany contributors
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 /**
22 * @file keybindings.h
23 * Configurable keyboard shortcuts.
24 * - keybindings_send_command() mimics a built-in keybinding action.
25 * - @ref GeanyKeyGroupID lists groups of built-in keybindings.
26 * @see plugin_set_key_group().
27 **/
29 #ifdef HAVE_CONFIG_H
30 # include "config.h"
31 #endif
33 #include "keybindings.h"
35 #include "app.h"
36 #include "build.h"
37 #include "callbacks.h"
38 #include "documentprivate.h"
39 #include "filetypes.h"
40 #include "geanyobject.h"
41 #include "keybindingsprivate.h"
42 #include "main.h"
43 #include "msgwindow.h"
44 #include "navqueue.h"
45 #include "notebook.h"
46 #include "prefs.h"
47 #include "sciwrappers.h"
48 #include "sidebar.h"
49 #include "support.h"
50 #include "symbols.h"
51 #include "toolbar.h"
52 #include "tools.h"
53 #include "ui_utils.h"
54 #include "utils.h"
55 #include "vte.h"
57 #include <gtk/gtk.h>
58 #include <gdk/gdkkeysyms.h>
59 #include <string.h>
62 GPtrArray *keybinding_groups; /* array of GeanyKeyGroup pointers, in visual order */
64 /* keyfile group name for non-plugin KB groups */
65 static const gchar keybindings_keyfile_group_name[] = "Bindings";
67 /* core keybindings */
68 static GeanyKeyBinding binding_ids[GEANY_KEYS_COUNT];
70 static GtkAccelGroup *kb_accel_group = NULL;
71 static const gboolean swap_alt_tab_order = FALSE;
74 /* central keypress event handler, almost all keypress events go to this function */
75 static gboolean on_key_press_event(GtkWidget *widget, GdkEventKey *event, gpointer user_data);
77 static gboolean check_current_word(GeanyDocument *doc, gboolean sci_word);
78 static gboolean read_current_word(GeanyDocument *doc, gboolean sci_word);
79 static gchar *get_current_word_or_sel(GeanyDocument *doc, gboolean sci_word);
81 static gboolean cb_func_file_action(guint key_id);
82 static gboolean cb_func_project_action(guint key_id);
83 static gboolean cb_func_editor_action(guint key_id);
84 static gboolean cb_func_select_action(guint key_id);
85 static gboolean cb_func_format_action(guint key_id);
86 static gboolean cb_func_insert_action(guint key_id);
87 static gboolean cb_func_search_action(guint key_id);
88 static gboolean cb_func_goto_action(guint key_id);
89 static gboolean cb_func_switch_action(guint key_id);
90 static gboolean cb_func_clipboard_action(guint key_id);
91 static gboolean cb_func_document_action(guint key_id);
92 static gboolean cb_func_view_action(guint key_id);
94 /* note: new keybindings should normally use per group callbacks */
95 static void cb_func_menu_help(guint key_id);
96 static void cb_func_menu_preferences(guint key_id);
98 static void cb_func_menu_fullscreen(guint key_id);
99 static void cb_func_menu_messagewindow(guint key_id);
101 static void cb_func_menu_opencolorchooser(guint key_id);
103 static void cb_func_switch_tableft(guint key_id);
104 static void cb_func_switch_tabright(guint key_id);
105 static void cb_func_switch_tablastused(guint key_id);
106 static void cb_func_move_tab(guint key_id);
108 static void add_popup_menu_accels(void);
111 /** Gets significant modifiers from a GdkModifierType mask. The set of
112 * significant modifiers corresponds to the default modifier mask as returned
113 * by @c gtk_accelerator_get_default_mod_mask(). In addition, it improves
114 * the Command key handling on OS X by adding @c GEANY_PRIMARY_MOD_MASK
115 * when needed. For this reason it is preferred to use this function
116 * instead of @c gtk_accelerator_set_default_mod_mask().
117 * @param mods GdkModifierType mask.
118 * @return Significant modifiers from the mask.
119 * @since 1.25. */
120 GEANY_API_SYMBOL
121 GdkModifierType keybindings_get_modifiers(GdkModifierType mods)
123 #ifdef __APPLE__
124 if (mods & GDK_MOD2_MASK)
126 mods |= GEANY_PRIMARY_MOD_MASK;
127 mods &= ~GDK_MOD2_MASK;
129 #endif
130 return mods & gtk_accelerator_get_default_mod_mask();
134 /** Looks up a keybinding item.
135 * @param group Group.
136 * @param key_id Keybinding index for the group.
137 * @return @transfer{none} The keybinding.
138 * @since 0.19. */
139 GEANY_API_SYMBOL
140 GeanyKeyBinding *keybindings_get_item(GeanyKeyGroup *group, gsize key_id)
142 if (group->plugin)
144 g_assert(key_id < group->plugin_key_count);
145 return &group->plugin_keys[key_id];
147 g_assert(key_id < GEANY_KEYS_COUNT);
148 return &binding_ids[key_id];
152 /* This is used to set default keybindings on startup.
153 * Menu accels are set in apply_kb_accel(). */
154 /** @girskip
155 * Fills a GeanyKeyBinding struct item.
156 * @note Always set @a key and @a mod to 0, otherwise you will likely
157 * cause conflicts with the user's custom, other plugin's keybindings or
158 * future default keybindings.
159 * @param group Group.
160 * @param key_id Keybinding index for the group.
161 * @param callback @nullable Function to call when activated, or @c NULL to use the group callback.
162 * Usually it's better to use the group callback instead - see plugin_set_key_group().
163 * @param key Default key, e.g. @c GDK_KEY_j (must be lower case), but usually 0 for unset.
164 * @param mod Default modifier, e.g. @c GDK_CONTROL_MASK, but usually 0 for unset.
165 * @param kf_name Key name used for this item in the keybindings configuration file, i.e. @c "menu_new".
166 * @param label Label used in the preferences dialog keybindings tab. May contain
167 * underscores - these won't be displayed.
168 * @param menu_item @nullable Optional widget to set an accelerator for, or @c NULL.
169 * @return The keybinding - normally this is ignored. */
170 GEANY_API_SYMBOL
171 GeanyKeyBinding *keybindings_set_item(GeanyKeyGroup *group, gsize key_id,
172 GeanyKeyCallback callback, guint key, GdkModifierType mod,
173 const gchar *kf_name, const gchar *label, GtkWidget *menu_item)
175 GeanyKeyBinding *kb;
177 g_assert(group->name);
178 kb = keybindings_get_item(group, key_id);
179 g_assert(!kb->name);
180 g_ptr_array_add(group->key_items, kb);
182 if (group->plugin)
184 /* some plugins e.g. GeanyLua need these fields duplicated */
185 SETPTR(kb->name, g_strdup(kf_name));
186 SETPTR(kb->label, g_strdup(label));
188 else
190 /* we don't touch these strings unless group->plugin is set, const cast is safe */
191 kb->name = (gchar *)kf_name;
192 kb->label = (gchar *)label;
194 kb->key = key;
195 kb->mods = mod;
196 kb->default_key = key;
197 kb->default_mods = mod;
198 kb->callback = callback;
199 kb->cb_func = NULL;
200 kb->cb_data = NULL;
201 kb->menu_item = menu_item;
202 kb->id = key_id;
203 return kb;
207 /** Creates a new keybinding using a GeanyKeyBindingFunc and attaches it to a keybinding group
209 * If given the callback should return @c TRUE if the keybinding was handled, otherwise @c FALSE
210 * to allow other callbacks to be run. This allows for multiplexing keybindings on the same keys,
211 * depending on the focused widget (or context). If the callback is NULL the group's callback will
212 * be invoked, but the same rule applies.
214 * @param group Group.
215 * @param key_id Keybinding index for the group.
216 * @param key Default key, e.g. @c GDK_KEY_j (must be lower case), but usually 0 for unset.
217 * @param mod Default modifier, e.g. @c GDK_CONTROL_MASK, but usually 0 for unset.
218 * @param kf_name Key name used for this item in the keybindings configuration file, i.e. @c "menu_new".
219 * @param label Label used in the preferences dialog keybindings tab. May contain
220 * underscores - these won't be displayed.
221 * @param menu_item @nullable Optional widget to set an accelerator for, or @c NULL.
222 * @param cb @nullable New-style callback to be called when activated, or @c NULL to use the group callback.
223 * @param pdata Plugin-specific data passed back to the callback @a cb.
224 * @param destroy_notify Function that is invoked to free the plugin data when not needed anymore.
225 * @return @transfer{none} The keybinding - normally this is ignored.
227 * @since 1.26 (API 226)
228 * @see See plugin_set_key_group_full
230 GEANY_API_SYMBOL
231 GeanyKeyBinding *keybindings_set_item_full(GeanyKeyGroup *group, gsize key_id,
232 guint key, GdkModifierType mod, const gchar *kf_name, const gchar *label,
233 GtkWidget *menu_item, GeanyKeyBindingFunc cb, gpointer pdata,
234 GDestroyNotify destroy_notify)
236 GeanyKeyBinding *kb;
238 /* For now, this is intended for plugins only */
239 g_assert(group->plugin);
241 kb = keybindings_set_item(group, key_id, NULL, key, mod, kf_name, label, menu_item);
242 kb->cb_func = cb;
243 kb->cb_data = pdata;
244 kb->cb_data_destroy = destroy_notify;
245 return kb;
249 static void free_key_binding(gpointer item)
251 GeanyKeyBinding *kb = item;
253 g_free(kb->name);
254 g_free(kb->label);
256 if (kb->cb_data_destroy)
257 kb->cb_data_destroy(kb->cb_data);
261 static void add_kb_group(GeanyKeyGroup *group,
262 const gchar *name, const gchar *label, GeanyKeyGroupCallback callback, gboolean plugin)
264 g_ptr_array_add(keybinding_groups, group);
266 /* as for items, we only require duplicated name and label for plugins */
267 group->name = plugin ? g_strdup(name) : name;
268 group->label = plugin ? g_strdup(label) : label;
269 group->callback = callback;
270 group->cb_func = NULL;
271 group->cb_data = NULL;
272 group->plugin = plugin;
273 /* Only plugins use the destroy notify thus far */
274 group->key_items = g_ptr_array_new_with_free_func(plugin ? free_key_binding : NULL);
278 GeanyKeyGroup *keybindings_get_core_group(guint id)
280 static GeanyKeyGroup groups[GEANY_KEY_GROUP_COUNT];
282 g_return_val_if_fail(id < GEANY_KEY_GROUP_COUNT, NULL);
284 return &groups[id];
288 static void add_kb(GeanyKeyGroup *group, gsize key_id,
289 GeanyKeyCallback callback, guint key, GdkModifierType mod,
290 const gchar *kf_name, const gchar *label, const gchar *widget_name)
292 GtkWidget *widget = widget_name ?
293 ui_lookup_widget(main_widgets.window, widget_name) : NULL;
295 keybindings_set_item(group, key_id, callback,
296 key, mod, kf_name, label, widget);
300 #define ADD_KB_GROUP(group_id, label, callback) \
301 add_kb_group(keybindings_get_core_group(group_id),\
302 keybindings_keyfile_group_name, label, callback, FALSE)
304 static void init_default_kb(void)
306 GeanyKeyGroup *group;
308 /* visual group order */
309 ADD_KB_GROUP(GEANY_KEY_GROUP_FILE, _("File"), cb_func_file_action);
310 ADD_KB_GROUP(GEANY_KEY_GROUP_EDITOR, _("Editor"), cb_func_editor_action);
311 ADD_KB_GROUP(GEANY_KEY_GROUP_CLIPBOARD, _("Clipboard"), cb_func_clipboard_action);
312 ADD_KB_GROUP(GEANY_KEY_GROUP_SELECT, _("Select"), cb_func_select_action);
313 ADD_KB_GROUP(GEANY_KEY_GROUP_FORMAT, _("Format"), cb_func_format_action);
314 ADD_KB_GROUP(GEANY_KEY_GROUP_INSERT, _("Insert"), cb_func_insert_action);
315 ADD_KB_GROUP(GEANY_KEY_GROUP_SETTINGS, _("Settings"), NULL);
316 ADD_KB_GROUP(GEANY_KEY_GROUP_SEARCH, _("Search"), cb_func_search_action);
317 ADD_KB_GROUP(GEANY_KEY_GROUP_GOTO, _("Go to"), cb_func_goto_action);
318 ADD_KB_GROUP(GEANY_KEY_GROUP_VIEW, _("View"), cb_func_view_action);
319 ADD_KB_GROUP(GEANY_KEY_GROUP_DOCUMENT, _("Document"), cb_func_document_action);
320 ADD_KB_GROUP(GEANY_KEY_GROUP_PROJECT, _("Project"), cb_func_project_action);
321 ADD_KB_GROUP(GEANY_KEY_GROUP_BUILD, _("Build"), build_keybinding);
322 ADD_KB_GROUP(GEANY_KEY_GROUP_TOOLS, _("Tools"), NULL);
323 ADD_KB_GROUP(GEANY_KEY_GROUP_HELP, _("Help"), NULL);
324 ADD_KB_GROUP(GEANY_KEY_GROUP_FOCUS, _("Focus"), cb_func_switch_action);
325 ADD_KB_GROUP(GEANY_KEY_GROUP_NOTEBOOK, _("Notebook tab"), NULL);
327 /* Init all fields of keys with default values.
328 * The menu_item field is always the main menu item, popup menu accelerators are
329 * set in add_popup_menu_accels(). */
331 group = keybindings_get_core_group(GEANY_KEY_GROUP_FILE);
333 add_kb(group, GEANY_KEYS_FILE_NEW, NULL,
334 GDK_KEY_n, GEANY_PRIMARY_MOD_MASK, "menu_new", _("New"), "menu_new1");
335 add_kb(group, GEANY_KEYS_FILE_OPEN, NULL,
336 GDK_KEY_o, GEANY_PRIMARY_MOD_MASK, "menu_open", _("Open"), "menu_open1");
337 add_kb(group, GEANY_KEYS_FILE_OPENSELECTED, NULL,
338 GDK_KEY_o, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "menu_open_selected",
339 _("Open selected file"), "menu_open_selected_file1");
340 add_kb(group, GEANY_KEYS_FILE_SAVE, NULL,
341 GDK_KEY_s, GEANY_PRIMARY_MOD_MASK, "menu_save", _("Save"), "menu_save1");
342 add_kb(group, GEANY_KEYS_FILE_SAVEAS, NULL,
343 0, 0, "menu_saveas", _("Save as"), "menu_save_as1");
344 add_kb(group, GEANY_KEYS_FILE_SAVEALL, NULL,
345 GDK_KEY_s, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "menu_saveall", _("Save all"),
346 "menu_save_all1");
347 add_kb(group, GEANY_KEYS_FILE_PROPERTIES, NULL,
348 GDK_KEY_v, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "file_properties", _("Properties"), "properties1");
349 add_kb(group, GEANY_KEYS_FILE_PRINT, NULL,
350 GDK_KEY_p, GEANY_PRIMARY_MOD_MASK, "menu_print", _("Print"), "print1");
351 add_kb(group, GEANY_KEYS_FILE_CLOSE, NULL,
352 GDK_KEY_w, GEANY_PRIMARY_MOD_MASK, "menu_close", _("Close"), "menu_close1");
353 add_kb(group, GEANY_KEYS_FILE_CLOSEALL, NULL,
354 GDK_KEY_w, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "menu_closeall", _("Close all"),
355 "menu_close_all1");
356 add_kb(group, GEANY_KEYS_FILE_RELOAD, NULL,
357 GDK_KEY_r, GEANY_PRIMARY_MOD_MASK, "menu_reloadfile", _("Reload file"), "menu_reload1");
358 add_kb(group, GEANY_KEYS_FILE_RELOAD_ALL, NULL,
359 0, 0, "menu_reloadall", _("Reload all files"), NULL);
360 add_kb(group, GEANY_KEYS_FILE_OPENLASTTAB, NULL,
361 0, 0, "file_openlasttab", _("Re-open last closed tab"), NULL);
362 add_kb(group, GEANY_KEYS_FILE_QUIT, NULL,
363 GDK_KEY_q, GEANY_PRIMARY_MOD_MASK, "menu_quit", _("Quit"), "menu_quit1");
365 group = keybindings_get_core_group(GEANY_KEY_GROUP_PROJECT);
367 add_kb(group, GEANY_KEYS_PROJECT_NEW, NULL,
368 0, 0, "project_new", _("New"), "project_new1");
369 add_kb(group, GEANY_KEYS_PROJECT_NEW_FROM_FOLDER, NULL,
370 0, 0, "project_new_from_folder", _("New from Folder"), "project_new_from_folder1");
371 add_kb(group, GEANY_KEYS_PROJECT_OPEN, NULL,
372 0, 0, "project_open", _("Open"), "project_open1");
373 add_kb(group, GEANY_KEYS_PROJECT_PROPERTIES, NULL,
374 0, 0, "project_properties",
375 ui_lookup_stock_label(GTK_STOCK_PROPERTIES), "project_properties1");
376 add_kb(group, GEANY_KEYS_PROJECT_CLOSE, NULL,
377 0, 0, "project_close", _("Close"), "project_close1");
379 group = keybindings_get_core_group(GEANY_KEY_GROUP_EDITOR);
381 add_kb(group, GEANY_KEYS_EDITOR_UNDO, NULL,
382 GDK_KEY_z, GEANY_PRIMARY_MOD_MASK, "menu_undo", _("Undo"), "menu_undo2");
383 add_kb(group, GEANY_KEYS_EDITOR_REDO, NULL,
384 GDK_KEY_y, GEANY_PRIMARY_MOD_MASK, "menu_redo", _("Redo"), "menu_redo2");
385 add_kb(group, GEANY_KEYS_EDITOR_DUPLICATELINE, NULL,
386 GDK_KEY_d, GEANY_PRIMARY_MOD_MASK, "edit_duplicateline", _("D_uplicate Line or Selection"),
387 "duplicate_line_or_selection1");
388 add_kb(group, GEANY_KEYS_EDITOR_DELETELINE, NULL,
389 GDK_KEY_k, GEANY_PRIMARY_MOD_MASK, "edit_deleteline", _("_Delete Current Line(s)"),
390 "delete_current_lines1");
391 add_kb(group, GEANY_KEYS_EDITOR_DELETELINETOEND, NULL,
392 GDK_KEY_Delete, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "edit_deletelinetoend",
393 _("Delete to line end"), NULL);
394 add_kb(group, GEANY_KEYS_EDITOR_DELETELINETOBEGINNING, NULL,
395 GDK_KEY_BackSpace, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "edit_deletelinetobegin",
396 _("Delete to beginning of line"), NULL);
397 /* Note: transpose may fit better in format group, but that would break the API */
398 add_kb(group, GEANY_KEYS_EDITOR_TRANSPOSELINE, NULL,
399 0, 0, "edit_transposeline", _("_Transpose Current Line"), NULL);
400 add_kb(group, GEANY_KEYS_EDITOR_SCROLLTOLINE, NULL,
401 GDK_KEY_l, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "edit_scrolltoline", _("Scroll to current line"), NULL);
402 add_kb(group, GEANY_KEYS_EDITOR_SCROLLLINEUP, NULL,
403 GDK_KEY_Up, GDK_MOD1_MASK, "edit_scrolllineup", _("Scroll up the view by one line"), NULL);
404 add_kb(group, GEANY_KEYS_EDITOR_SCROLLLINEDOWN, NULL,
405 GDK_KEY_Down, GDK_MOD1_MASK, "edit_scrolllinedown", _("Scroll down the view by one line"), NULL);
406 add_kb(group, GEANY_KEYS_EDITOR_COMPLETESNIPPET, NULL,
407 GDK_KEY_Tab, 0, "edit_completesnippet", _("Complete snippet"), NULL);
408 add_kb(group, GEANY_KEYS_EDITOR_SNIPPETNEXTCURSOR, NULL,
409 0, 0, "move_snippetnextcursor", _("Move cursor in snippet"), NULL);
410 add_kb(group, GEANY_KEYS_EDITOR_SUPPRESSSNIPPETCOMPLETION, NULL,
411 0, 0, "edit_suppresssnippetcompletion", _("Suppress snippet completion"), NULL);
412 add_kb(group, GEANY_KEYS_EDITOR_CONTEXTACTION, NULL,
413 0, 0, "popup_contextaction", _("Context Action"), NULL);
414 add_kb(group, GEANY_KEYS_EDITOR_AUTOCOMPLETE, NULL,
415 GDK_KEY_space, GEANY_PRIMARY_MOD_MASK, "edit_autocomplete", _("Complete word"), NULL);
416 add_kb(group, GEANY_KEYS_EDITOR_CALLTIP, NULL,
417 GDK_KEY_space, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "edit_calltip", _("Show calltip"), NULL);
418 add_kb(group, GEANY_KEYS_EDITOR_WORDPARTCOMPLETION, NULL,
419 GDK_KEY_Tab, 0, "edit_wordpartcompletion", _("Word part completion"), NULL);
420 add_kb(group, GEANY_KEYS_EDITOR_MOVELINEUP, NULL,
421 GDK_KEY_Page_Up, GDK_MOD1_MASK, "edit_movelineup",
422 _("Move line(s) up"), "move_lines_up1");
423 add_kb(group, GEANY_KEYS_EDITOR_MOVELINEDOWN, NULL,
424 GDK_KEY_Page_Down, GDK_MOD1_MASK, "edit_movelinedown",
425 _("Move line(s) down"), "move_lines_down1");
427 group = keybindings_get_core_group(GEANY_KEY_GROUP_CLIPBOARD);
429 add_kb(group, GEANY_KEYS_CLIPBOARD_CUT, NULL,
430 GDK_KEY_x, GEANY_PRIMARY_MOD_MASK, "menu_cut", _("Cut"), "menu_cut1");
431 add_kb(group, GEANY_KEYS_CLIPBOARD_COPY, NULL,
432 GDK_KEY_c, GEANY_PRIMARY_MOD_MASK, "menu_copy", _("Copy"), "menu_copy1");
433 add_kb(group, GEANY_KEYS_CLIPBOARD_PASTE, NULL,
434 GDK_KEY_v, GEANY_PRIMARY_MOD_MASK, "menu_paste", _("Paste"), "menu_paste1");
435 add_kb(group, GEANY_KEYS_CLIPBOARD_COPYLINE, NULL,
436 GDK_KEY_c, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "edit_copyline", _("_Copy Current Line(s)"),
437 "copy_current_lines1");
438 add_kb(group, GEANY_KEYS_CLIPBOARD_CUTLINE, NULL,
439 GDK_KEY_x, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "edit_cutline", _("Cu_t Current Line(s)"),
440 "cut_current_lines1");
442 group = keybindings_get_core_group(GEANY_KEY_GROUP_SELECT);
444 add_kb(group, GEANY_KEYS_SELECT_ALL, NULL,
445 GDK_KEY_a, GEANY_PRIMARY_MOD_MASK, "menu_selectall", _("Select All"), "menu_select_all1");
446 add_kb(group, GEANY_KEYS_SELECT_WORD, NULL,
447 GDK_KEY_w, GDK_SHIFT_MASK | GDK_MOD1_MASK, "edit_selectword", _("Select current word"), NULL);
448 add_kb(group, GEANY_KEYS_SELECT_LINE, NULL,
449 GDK_KEY_l, GDK_SHIFT_MASK | GDK_MOD1_MASK, "edit_selectline", _("S_elect Current Line(s)"),
450 "select_current_lines1");
451 add_kb(group, GEANY_KEYS_SELECT_PARAGRAPH, NULL,
452 GDK_KEY_p, GDK_SHIFT_MASK | GDK_MOD1_MASK, "edit_selectparagraph", _("Se_lect Current Paragraph"),
453 "select_current_paragraph1");
454 add_kb(group, GEANY_KEYS_SELECT_WORDPARTLEFT, NULL,
455 0, 0, "edit_selectwordpartleft", _("Select to previous word part"), NULL);
456 add_kb(group, GEANY_KEYS_SELECT_WORDPARTRIGHT, NULL,
457 0, 0, "edit_selectwordpartright", _("Select to next word part"), NULL);
459 group = keybindings_get_core_group(GEANY_KEY_GROUP_FORMAT);
461 add_kb(group, GEANY_KEYS_FORMAT_TOGGLECASE, NULL,
462 GDK_KEY_u, GEANY_PRIMARY_MOD_MASK | GDK_MOD1_MASK, "edit_togglecase",
463 _("T_oggle Case of Selection"), "menu_toggle_case2");
464 add_kb(group, GEANY_KEYS_FORMAT_COMMENTLINETOGGLE, NULL,
465 GDK_KEY_e, GEANY_PRIMARY_MOD_MASK, "edit_commentlinetoggle", _("Toggle line commentation"),
466 "menu_toggle_line_commentation1");
467 add_kb(group, GEANY_KEYS_FORMAT_COMMENTLINE, NULL,
468 0, 0, "edit_commentline", _("Comment line(s)"), "menu_comment_line1");
469 add_kb(group, GEANY_KEYS_FORMAT_UNCOMMENTLINE, NULL,
470 0, 0, "edit_uncommentline", _("Uncomment line(s)"), "menu_uncomment_line1");
471 add_kb(group, GEANY_KEYS_FORMAT_INCREASEINDENT, NULL,
472 GDK_KEY_i, GEANY_PRIMARY_MOD_MASK, "edit_increaseindent", _("Increase indent"),
473 "menu_increase_indent1");
474 add_kb(group, GEANY_KEYS_FORMAT_DECREASEINDENT, NULL,
475 GDK_KEY_u, GEANY_PRIMARY_MOD_MASK, "edit_decreaseindent", _("Decrease indent"),
476 "menu_decrease_indent1");
477 add_kb(group, GEANY_KEYS_FORMAT_INCREASEINDENTBYSPACE, NULL,
478 0, 0, "edit_increaseindentbyspace", _("Increase indent by one space"), NULL);
479 add_kb(group, GEANY_KEYS_FORMAT_DECREASEINDENTBYSPACE, NULL,
480 0, 0, "edit_decreaseindentbyspace", _("Decrease indent by one space"), NULL);
481 add_kb(group, GEANY_KEYS_FORMAT_AUTOINDENT, NULL,
482 0, 0, "edit_autoindent", _("S_mart Line Indent"), "smart_line_indent1");
483 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD1, NULL,
484 GDK_KEY_1, GEANY_PRIMARY_MOD_MASK, "edit_sendtocmd1", _("Send to Custom Command 1"), NULL);
485 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD2, NULL,
486 GDK_KEY_2, GEANY_PRIMARY_MOD_MASK, "edit_sendtocmd2", _("Send to Custom Command 2"), NULL);
487 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD3, NULL,
488 GDK_KEY_3, GEANY_PRIMARY_MOD_MASK, "edit_sendtocmd3", _("Send to Custom Command 3"), NULL);
489 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD4, NULL,
490 0, 0, "edit_sendtocmd4", _("Send to Custom Command 4"), NULL);
491 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD5, NULL,
492 0, 0, "edit_sendtocmd5", _("Send to Custom Command 5"), NULL);
493 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD6, NULL,
494 0, 0, "edit_sendtocmd6", _("Send to Custom Command 6"), NULL);
495 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD7, NULL,
496 0, 0, "edit_sendtocmd7", _("Send to Custom Command 7"), NULL);
497 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD8, NULL,
498 0, 0, "edit_sendtocmd8", _("Send to Custom Command 8"), NULL);
499 add_kb(group, GEANY_KEYS_FORMAT_SENDTOCMD9, NULL,
500 0, 0, "edit_sendtocmd9", _("Send to Custom Command 9"), NULL);
501 /* may fit better in editor group */
502 add_kb(group, GEANY_KEYS_FORMAT_SENDTOVTE, NULL,
503 0, 0, "edit_sendtovte", _("_Send Selection to Terminal"), "send_selection_to_vte1");
504 add_kb(group, GEANY_KEYS_FORMAT_REFLOWPARAGRAPH, NULL,
505 GDK_KEY_j, GEANY_PRIMARY_MOD_MASK, "format_reflowparagraph", _("_Reflow Lines/Block"),
506 "reflow_lines_block1");
507 add_kb(group, GEANY_KEYS_FORMAT_JOINLINES, NULL,
508 0, 0, "edit_joinlines", _("_Join Lines"), "join_lines1");
510 group = keybindings_get_core_group(GEANY_KEY_GROUP_INSERT);
512 add_kb(group, GEANY_KEYS_INSERT_DATE, NULL,
513 GDK_KEY_d, GDK_SHIFT_MASK | GDK_MOD1_MASK, "menu_insert_date", _("Insert date"),
514 "insert_date_custom1");
515 add_kb(group, GEANY_KEYS_INSERT_ALTWHITESPACE, NULL,
516 0, 0, "edit_insertwhitespace", _("Insert Alternative _White Space"),
517 "insert_alternative_white_space1");
518 add_kb(group, GEANY_KEYS_INSERT_LINEBEFORE, NULL,
519 0, 0, "edit_insertlinebefore", _("Insert New Line Before Current"), NULL);
520 add_kb(group, GEANY_KEYS_INSERT_LINEAFTER, NULL,
521 0, 0, "edit_insertlineafter", _("Insert New Line After Current"), NULL);
523 group = keybindings_get_core_group(GEANY_KEY_GROUP_SETTINGS);
525 add_kb(group, GEANY_KEYS_SETTINGS_PREFERENCES, cb_func_menu_preferences,
526 GDK_KEY_p, GEANY_PRIMARY_MOD_MASK | GDK_MOD1_MASK, "menu_preferences", _("Preferences"),
527 "preferences1");
528 add_kb(group, GEANY_KEYS_SETTINGS_PLUGINPREFERENCES, cb_func_menu_preferences,
529 0, 0, "menu_pluginpreferences", _("P_lugin Preferences"), "plugin_preferences1");
531 group = keybindings_get_core_group(GEANY_KEY_GROUP_SEARCH);
533 add_kb(group, GEANY_KEYS_SEARCH_FIND, NULL,
534 GDK_KEY_f, GEANY_PRIMARY_MOD_MASK, "menu_find", _("Find"), "find1");
535 add_kb(group, GEANY_KEYS_SEARCH_FINDNEXT, NULL,
536 GDK_KEY_g, GEANY_PRIMARY_MOD_MASK, "menu_findnext", _("Find Next"), "find_next1");
537 add_kb(group, GEANY_KEYS_SEARCH_FINDPREVIOUS, NULL,
538 GDK_KEY_g, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "menu_findprevious", _("Find Previous"),
539 "find_previous1");
540 add_kb(group, GEANY_KEYS_SEARCH_FINDNEXTSEL, NULL,
541 0, 0, "menu_findnextsel", _("Find Next _Selection"), "find_nextsel1");
542 add_kb(group, GEANY_KEYS_SEARCH_FINDPREVSEL, NULL,
543 0, 0, "menu_findprevsel", _("Find Pre_vious Selection"), "find_prevsel1");
544 add_kb(group, GEANY_KEYS_SEARCH_REPLACE, NULL,
545 GDK_KEY_h, GEANY_PRIMARY_MOD_MASK, "menu_replace", _("Replace"), "replace1");
546 add_kb(group, GEANY_KEYS_SEARCH_FINDINFILES, NULL, GDK_KEY_f,
547 GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "menu_findinfiles", _("Find in Files"),
548 "find_in_files1");
549 add_kb(group, GEANY_KEYS_SEARCH_NEXTMESSAGE, NULL,
550 0, 0, "menu_nextmessage", _("Next Message"), "next_message1");
551 add_kb(group, GEANY_KEYS_SEARCH_PREVIOUSMESSAGE, NULL,
552 0, 0, "menu_previousmessage", _("Previous Message"), "previous_message1");
553 add_kb(group, GEANY_KEYS_SEARCH_FINDUSAGE, NULL,
554 GDK_KEY_e, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "popup_findusage",
555 _("Find Usage"), "find_usage1");
556 add_kb(group, GEANY_KEYS_SEARCH_FINDDOCUMENTUSAGE, NULL,
557 GDK_KEY_d, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "popup_finddocumentusage",
558 _("Find Document Usage"), "find_document_usage1");
559 add_kb(group, GEANY_KEYS_SEARCH_MARKALL, NULL,
560 GDK_KEY_m, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "find_markall", _("_Mark All"), "mark_all1");
562 group = keybindings_get_core_group(GEANY_KEY_GROUP_GOTO);
564 add_kb(group, GEANY_KEYS_GOTO_BACK, NULL,
565 GDK_KEY_Left, GDK_MOD1_MASK, "nav_back", _("Navigate back a location"), NULL);
566 add_kb(group, GEANY_KEYS_GOTO_FORWARD, NULL,
567 GDK_KEY_Right, GDK_MOD1_MASK, "nav_forward", _("Navigate forward a location"), NULL);
568 add_kb(group, GEANY_KEYS_GOTO_LINE, NULL,
569 GDK_KEY_l, GEANY_PRIMARY_MOD_MASK, "menu_gotoline", _("Go to Line"), "go_to_line1");
570 add_kb(group, GEANY_KEYS_GOTO_MATCHINGBRACE, NULL,
571 GDK_KEY_b, GEANY_PRIMARY_MOD_MASK, "edit_gotomatchingbrace",
572 _("Go to matching brace"), NULL);
573 add_kb(group, GEANY_KEYS_GOTO_TOGGLEMARKER, NULL,
574 GDK_KEY_m, GEANY_PRIMARY_MOD_MASK, "edit_togglemarker",
575 _("Toggle marker"), NULL);
576 add_kb(group, GEANY_KEYS_GOTO_NEXTMARKER, NULL,
577 GDK_KEY_period, GEANY_PRIMARY_MOD_MASK, "edit_gotonextmarker",
578 _("Go to Ne_xt Marker"), "go_to_next_marker1");
579 add_kb(group, GEANY_KEYS_GOTO_PREVIOUSMARKER, NULL,
580 GDK_KEY_comma, GEANY_PRIMARY_MOD_MASK, "edit_gotopreviousmarker",
581 _("Go to Pre_vious Marker"), "go_to_previous_marker1");
582 add_kb(group, GEANY_KEYS_GOTO_TAGDEFINITION, NULL,
583 GDK_KEY_t, GEANY_PRIMARY_MOD_MASK, "popup_gototagdefinition",
584 _("Go to Symbol Definition"), "goto_tag_definition1");
585 add_kb(group, GEANY_KEYS_GOTO_TAGDECLARATION, NULL,
586 GDK_KEY_t, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "popup_gototagdeclaration",
587 _("Go to Symbol Declaration"), "goto_tag_declaration1");
588 add_kb(group, GEANY_KEYS_GOTO_LINESTART, NULL,
589 GDK_KEY_Home, 0, "edit_gotolinestart", _("Go to Start of Line"), NULL);
590 add_kb(group, GEANY_KEYS_GOTO_LINEEND, NULL,
591 GDK_KEY_End, 0, "edit_gotolineend", _("Go to End of Line"), NULL);
592 add_kb(group, GEANY_KEYS_GOTO_LINESTARTVISUAL, NULL,
593 GDK_KEY_Home, GDK_MOD1_MASK, "edit_gotolinestartvisual", _("Go to Start of Display Line"), NULL);
594 add_kb(group, GEANY_KEYS_GOTO_LINEENDVISUAL, NULL,
595 GDK_KEY_End, GDK_MOD1_MASK, "edit_gotolineendvisual", _("Go to End of Display Line"), NULL);
596 add_kb(group, GEANY_KEYS_GOTO_PREVWORDPART, NULL,
597 GDK_KEY_slash, GEANY_PRIMARY_MOD_MASK, "edit_prevwordstart", _("Go to Previous Word Part"), NULL);
598 add_kb(group, GEANY_KEYS_GOTO_NEXTWORDPART, NULL,
599 GDK_KEY_backslash, GEANY_PRIMARY_MOD_MASK, "edit_nextwordstart", _("Go to Next Word Part"), NULL);
601 group = keybindings_get_core_group(GEANY_KEY_GROUP_VIEW);
603 add_kb(group, GEANY_KEYS_VIEW_TOGGLEALL, NULL,
604 0, 0, "menu_toggleall", _("Toggle All Additional Widgets"),
605 "menu_toggle_all_additional_widgets1");
606 add_kb(group, GEANY_KEYS_VIEW_FULLSCREEN, cb_func_menu_fullscreen,
607 GDK_KEY_F11, 0, "menu_fullscreen", _("Fullscreen"), "menu_fullscreen1");
608 add_kb(group, GEANY_KEYS_VIEW_MESSAGEWINDOW, cb_func_menu_messagewindow,
609 0, 0, "menu_messagewindow", _("Toggle Messages Window"),
610 "menu_show_messages_window1");
611 add_kb(group, GEANY_KEYS_VIEW_SIDEBAR, NULL,
612 0, 0, "toggle_sidebar", _("Toggle Sidebar"), "menu_show_sidebar1");
613 add_kb(group, GEANY_KEYS_VIEW_ZOOMIN, NULL,
614 GDK_KEY_plus, GEANY_PRIMARY_MOD_MASK, "menu_zoomin", _("Zoom In"), "menu_zoom_in1");
615 add_kb(group, GEANY_KEYS_VIEW_ZOOMOUT, NULL,
616 GDK_KEY_minus, GEANY_PRIMARY_MOD_MASK, "menu_zoomout", _("Zoom Out"), "menu_zoom_out1");
617 add_kb(group, GEANY_KEYS_VIEW_ZOOMRESET, NULL,
618 GDK_KEY_0, GEANY_PRIMARY_MOD_MASK, "normal_size", _("Zoom Reset"), "normal_size1");
620 group = keybindings_get_core_group(GEANY_KEY_GROUP_FOCUS);
622 add_kb(group, GEANY_KEYS_FOCUS_EDITOR, NULL,
623 GDK_KEY_F2, 0, "switch_editor", _("Switch to Editor"), NULL);
624 add_kb(group, GEANY_KEYS_FOCUS_SEARCHBAR, NULL,
625 GDK_KEY_F7, 0, "switch_search_bar", _("Switch to Search Bar"), NULL);
626 add_kb(group, GEANY_KEYS_FOCUS_MESSAGE_WINDOW, NULL,
627 0, 0, "switch_message_window", _("Switch to Message Window"), NULL);
628 add_kb(group, GEANY_KEYS_FOCUS_COMPILER, NULL,
629 0, 0, "switch_compiler", _("Switch to Compiler"), NULL);
630 add_kb(group, GEANY_KEYS_FOCUS_MESSAGES, NULL,
631 0, 0, "switch_messages", _("Switch to Messages"), NULL);
632 add_kb(group, GEANY_KEYS_FOCUS_SCRIBBLE, NULL,
633 GDK_KEY_F6, 0, "switch_scribble", _("Switch to Scribble"), NULL);
634 add_kb(group, GEANY_KEYS_FOCUS_VTE, NULL,
635 GDK_KEY_F4, 0, "switch_vte", _("Switch to VTE"), NULL);
636 add_kb(group, GEANY_KEYS_FOCUS_SIDEBAR, NULL,
637 0, 0, "switch_sidebar", _("Switch to Sidebar"), NULL);
638 add_kb(group, GEANY_KEYS_FOCUS_SIDEBAR_SYMBOL_LIST, NULL,
639 0, 0, "switch_sidebar_symbol_list", _("Switch to Sidebar Symbol List"), NULL);
640 add_kb(group, GEANY_KEYS_FOCUS_SIDEBAR_DOCUMENT_LIST, NULL,
641 0, 0, "switch_sidebar_doc_list", _("Switch to Sidebar Document List"), NULL);
643 group = keybindings_get_core_group(GEANY_KEY_GROUP_NOTEBOOK);
645 add_kb(group, GEANY_KEYS_NOTEBOOK_SWITCHTABLEFT, cb_func_switch_tableft,
646 GDK_KEY_Page_Up, GEANY_PRIMARY_MOD_MASK, "switch_tableft", _("Switch to left document"), NULL);
647 add_kb(group, GEANY_KEYS_NOTEBOOK_SWITCHTABRIGHT, cb_func_switch_tabright,
648 GDK_KEY_Page_Down, GEANY_PRIMARY_MOD_MASK, "switch_tabright", _("Switch to right document"), NULL);
649 add_kb(group, GEANY_KEYS_NOTEBOOK_SWITCHTABLASTUSED, cb_func_switch_tablastused,
650 GDK_KEY_Tab, GEANY_PRIMARY_MOD_MASK, "switch_tablastused", _("Switch to last used document"), NULL);
651 add_kb(group, GEANY_KEYS_NOTEBOOK_MOVETABLEFT, cb_func_move_tab,
652 GDK_KEY_Page_Up, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "move_tableft",
653 _("Move document left"), NULL);
654 add_kb(group, GEANY_KEYS_NOTEBOOK_MOVETABRIGHT, cb_func_move_tab,
655 GDK_KEY_Page_Down, GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK, "move_tabright",
656 _("Move document right"), NULL);
657 add_kb(group, GEANY_KEYS_NOTEBOOK_MOVETABFIRST, cb_func_move_tab,
658 0, 0, "move_tabfirst", _("Move document first"), NULL);
659 add_kb(group, GEANY_KEYS_NOTEBOOK_MOVETABLAST, cb_func_move_tab,
660 0, 0, "move_tablast", _("Move document last"), NULL);
662 group = keybindings_get_core_group(GEANY_KEY_GROUP_DOCUMENT);
664 add_kb(group, GEANY_KEYS_DOCUMENT_LINEWRAP, NULL,
665 0, 0, "menu_linewrap", _("Toggle Line wrapping"), "menu_line_wrapping1");
666 add_kb(group, GEANY_KEYS_DOCUMENT_LINEBREAK, NULL,
667 0, 0, "menu_linebreak", _("Toggle Line breaking"), "line_breaking1");
668 add_kb(group, GEANY_KEYS_DOCUMENT_CLONE, NULL,
669 0, 0, "menu_clone", _("_Clone"), "clone1");
670 add_kb(group, GEANY_KEYS_DOCUMENT_STRIPTRAILINGSPACES, NULL,
671 0, 0, "menu_strip_trailing_spaces", _("_Strip Trailing Spaces"), "strip_trailing_spaces1");
672 add_kb(group, GEANY_KEYS_DOCUMENT_REPLACETABS, NULL,
673 0, 0, "menu_replacetabs", _("Replace tabs with space"), "menu_replace_tabs");
674 add_kb(group, GEANY_KEYS_DOCUMENT_REPLACESPACES, NULL,
675 0, 0, "menu_replacespaces", _("Replace spaces with tabs"), "menu_replace_spaces");
676 add_kb(group, GEANY_KEYS_DOCUMENT_TOGGLEFOLD, NULL,
677 0, 0, "menu_togglefold", _("Toggle current fold"), NULL);
678 add_kb(group, GEANY_KEYS_DOCUMENT_FOLDALL, NULL,
679 0, 0, "menu_foldall", _("Fold all"), "menu_fold_all1");
680 add_kb(group, GEANY_KEYS_DOCUMENT_UNFOLDALL, NULL,
681 0, 0, "menu_unfoldall", _("Unfold all"), "menu_unfold_all1");
682 add_kb(group, GEANY_KEYS_DOCUMENT_RELOADTAGLIST, NULL,
683 GDK_KEY_r, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "reloadtaglist", _("Reload symbol list"), NULL);
684 add_kb(group, GEANY_KEYS_DOCUMENT_REMOVE_MARKERS, NULL,
685 0, 0, "remove_markers", _("Remove Markers"), "remove_markers1");
686 add_kb(group, GEANY_KEYS_DOCUMENT_REMOVE_ERROR_INDICATORS, NULL,
687 0, 0, "remove_error_indicators", _("Remove Error Indicators"), "menu_remove_indicators1");
688 add_kb(group, GEANY_KEYS_DOCUMENT_REMOVE_MARKERS_INDICATORS, NULL,
689 0, 0, "remove_markers_and_indicators", _("Remove Markers and Error Indicators"), NULL);
691 group = keybindings_get_core_group(GEANY_KEY_GROUP_BUILD);
693 add_kb(group, GEANY_KEYS_BUILD_COMPILE, NULL,
694 GDK_KEY_F8, 0, "build_compile", _("Compile"), NULL);
695 add_kb(group, GEANY_KEYS_BUILD_LINK, NULL,
696 GDK_KEY_F9, 0, "build_link", _("Build"), NULL);
697 add_kb(group, GEANY_KEYS_BUILD_MAKE, NULL,
698 GDK_KEY_F9, GDK_SHIFT_MASK, "build_make", _("Make all"), NULL);
699 add_kb(group, GEANY_KEYS_BUILD_MAKEOWNTARGET, NULL,
700 GDK_KEY_F9, GDK_SHIFT_MASK | GEANY_PRIMARY_MOD_MASK, "build_makeowntarget",
701 _("Make custom target"), NULL);
702 add_kb(group, GEANY_KEYS_BUILD_MAKEOBJECT, NULL,
703 GDK_KEY_F8, GDK_SHIFT_MASK, "build_makeobject", _("Make object"), NULL);
704 add_kb(group, GEANY_KEYS_BUILD_NEXTERROR, NULL,
705 0, 0, "build_nexterror", _("Next error"), NULL);
706 add_kb(group, GEANY_KEYS_BUILD_PREVIOUSERROR, NULL,
707 0, 0, "build_previouserror", _("Previous error"), NULL);
708 add_kb(group, GEANY_KEYS_BUILD_RUN, NULL,
709 GDK_KEY_F5, 0, "build_run", _("Run"), NULL);
710 add_kb(group, GEANY_KEYS_BUILD_OPTIONS, NULL,
711 0, 0, "build_options", _("Build options"), NULL);
713 group = keybindings_get_core_group(GEANY_KEY_GROUP_TOOLS);
715 add_kb(group, GEANY_KEYS_TOOLS_OPENCOLORCHOOSER, cb_func_menu_opencolorchooser,
716 0, 0, "menu_opencolorchooser", _("Show Color Chooser"), "menu_choose_color1");
718 group = keybindings_get_core_group(GEANY_KEY_GROUP_HELP);
720 add_kb(group, GEANY_KEYS_HELP_HELP, cb_func_menu_help,
721 GDK_KEY_F1, 0, "menu_help", _("Help"), "help1");
725 static void free_key_group(gpointer item)
727 GeanyKeyGroup *group = item;
729 g_ptr_array_free(group->key_items, TRUE);
731 if (group->plugin)
733 if (group->cb_data_destroy)
734 group->cb_data_destroy(group->cb_data);
735 g_free(group->plugin_keys);
736 /* we allocated those in add_kb_group() as it's a plugin group */
737 g_free((gchar *) group->name);
738 g_free((gchar *) group->label);
739 g_free(group);
744 void keybindings_init(void)
746 memset(binding_ids, 0, sizeof binding_ids);
747 keybinding_groups = g_ptr_array_sized_new(GEANY_KEY_GROUP_COUNT);
748 g_ptr_array_set_free_func(keybinding_groups, free_key_group);
749 kb_accel_group = gtk_accel_group_new();
751 init_default_kb();
752 gtk_window_add_accel_group(GTK_WINDOW(main_widgets.window), kb_accel_group);
754 g_signal_connect(main_widgets.window, "key-press-event", G_CALLBACK(on_key_press_event), NULL);
758 typedef void (*KBItemCallback) (GeanyKeyGroup *group, GeanyKeyBinding *kb, gpointer user_data);
760 static void keybindings_foreach(KBItemCallback cb, gpointer user_data)
762 gsize g, i;
763 GeanyKeyGroup *group;
764 GeanyKeyBinding *kb;
766 foreach_ptr_array(group, g, keybinding_groups)
768 foreach_ptr_array(kb, i, group->key_items)
769 cb(group, kb, user_data);
774 static void load_kb(GeanyKeyGroup *group, GeanyKeyBinding *kb, gpointer user_data)
776 GKeyFile *config = user_data;
777 gchar *val;
778 guint key;
779 GdkModifierType mods;
781 val = g_key_file_get_string(config, group->name, kb->name, NULL);
782 if (val != NULL)
784 gtk_accelerator_parse(val, &key, &mods);
785 kb->key = key;
786 kb->mods = mods;
787 g_free(val);
792 static void load_user_kb(void)
794 gchar *configfile = g_build_filename(app->configdir, "keybindings.conf", NULL);
795 GKeyFile *config = g_key_file_new();
797 /* backwards compatibility with Geany 0.21 defaults */
798 if (!g_file_test(configfile, G_FILE_TEST_EXISTS))
800 gchar *geanyconf = g_build_filename(app->configdir, "geany.conf", NULL);
801 const gchar data[] = "[Bindings]\n"
802 "popup_gototagdefinition=\n"
803 "edit_transposeline=<Control>t\n"
804 "edit_movelineup=\n"
805 "edit_movelinedown=\n"
806 "move_tableft=<Alt>Page_Up\n"
807 "move_tabright=<Alt>Page_Down\n";
809 utils_write_file(configfile, g_file_test(geanyconf, G_FILE_TEST_EXISTS) ?
810 data : "");
811 g_free(geanyconf);
814 /* now load user defined keys */
815 if (g_key_file_load_from_file(config, configfile, G_KEY_FILE_KEEP_COMMENTS, NULL))
817 keybindings_foreach(load_kb, config);
819 g_free(configfile);
820 g_key_file_free(config);
824 static void apply_kb_accel(GeanyKeyGroup *group, GeanyKeyBinding *kb, gpointer user_data)
826 if (kb->key != 0 && kb->menu_item)
828 gtk_widget_add_accelerator(kb->menu_item, "activate", kb_accel_group,
829 kb->key, kb->mods, GTK_ACCEL_VISIBLE);
834 /** Reloads keybinding settings from configuration file.
836 * Normally plugins do not need to call this function as it is called automatically when a
837 * the plugin is activated. However, plugins which need to create keybindings dynamically
838 * and reload them when needed should call this function after all keybindings have been
839 * updated with plugin_set_key_group() and keybindings_set_item() calls - this makes sure
840 * that the corresponding user keybinding shortcuts are applied.
842 * @since 1.32 (API 233) */
843 GEANY_API_SYMBOL
844 void keybindings_load_keyfile(void)
846 load_user_kb();
847 add_popup_menu_accels();
849 /* set menu accels now, after user keybindings have been read */
850 keybindings_foreach(apply_kb_accel, NULL);
854 static void add_menu_accel(GeanyKeyGroup *group, guint kb_id, GtkWidget *menuitem)
856 GeanyKeyBinding *kb = keybindings_get_item(group, kb_id);
858 if (kb->key != 0)
859 gtk_widget_add_accelerator(menuitem, "activate", kb_accel_group,
860 kb->key, kb->mods, GTK_ACCEL_VISIBLE);
864 #define GEANY_ADD_POPUP_ACCEL(kb_id, wid) \
865 add_menu_accel(group, kb_id, ui_lookup_widget(main_widgets.editor_menu, G_STRINGIFY(wid)))
867 /* set the menu item accelerator shortcuts (just for visibility, they are handled anyway) */
868 /* FIXME: update those during runtime */
869 static void add_popup_menu_accels(void)
871 GeanyKeyGroup *group;
873 group = keybindings_get_core_group(GEANY_KEY_GROUP_EDITOR);
874 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_EDITOR_UNDO, undo1);
875 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_EDITOR_REDO, redo1);
876 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_EDITOR_CONTEXTACTION, context_action1);
878 group = keybindings_get_core_group(GEANY_KEY_GROUP_CLIPBOARD);
879 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_CLIPBOARD_CUT, cut1);
880 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_CLIPBOARD_COPY, copy1);
881 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_CLIPBOARD_PASTE, paste1);
883 group = keybindings_get_core_group(GEANY_KEY_GROUP_SELECT);
884 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_SELECT_ALL, menu_select_all2);
886 group = keybindings_get_core_group(GEANY_KEY_GROUP_INSERT);
887 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_INSERT_DATE, insert_date_custom2);
888 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_INSERT_ALTWHITESPACE, insert_alternative_white_space2);
890 group = keybindings_get_core_group(GEANY_KEY_GROUP_FILE);
891 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_FILE_OPENSELECTED, menu_open_selected_file2);
893 group = keybindings_get_core_group(GEANY_KEY_GROUP_SEARCH);
894 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_SEARCH_FINDUSAGE, find_usage2);
895 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_SEARCH_FINDDOCUMENTUSAGE, find_document_usage2);
897 group = keybindings_get_core_group(GEANY_KEY_GROUP_GOTO);
898 GEANY_ADD_POPUP_ACCEL(GEANY_KEYS_GOTO_TAGDEFINITION, goto_tag_definition2);
900 /* Format and Commands share the menu bar submenus */
901 /* Build menu items are set if the build menus are created */
905 static void set_keyfile_kb(GeanyKeyGroup *group, GeanyKeyBinding *kb, gpointer user_data)
907 GKeyFile *config = user_data;
908 gchar *val;
910 val = gtk_accelerator_name(kb->key, kb->mods);
911 g_key_file_set_string(config, group->name, kb->name, val);
912 g_free(val);
916 /* just write the content of the keys array to the config file */
917 void keybindings_write_to_file(void)
919 gchar *configfile = g_build_filename(app->configdir, "keybindings.conf", NULL);
920 gchar *data;
921 GKeyFile *config = g_key_file_new();
923 g_key_file_load_from_file(config, configfile, 0, NULL);
924 keybindings_foreach(set_keyfile_kb, config);
926 /* write the file */
927 data = g_key_file_to_data(config, NULL, NULL);
928 utils_write_file(configfile, data);
930 g_free(data);
931 g_free(configfile);
932 g_key_file_free(config);
936 void keybindings_free(void)
938 GeanyKeyGroup *group;
939 gsize g;
941 foreach_ptr_array(group, g, keybinding_groups)
942 keybindings_free_group(group);
944 g_ptr_array_free(keybinding_groups, TRUE);
948 gchar *keybindings_get_label(GeanyKeyBinding *kb)
950 return utils_str_remove_chars(g_strdup(kb->label), "_");
954 static void fill_shortcut_labels_treeview(GtkWidget *tree)
956 gsize g, i;
957 GeanyKeyBinding *kb;
958 GeanyKeyGroup *group;
959 GtkListStore *store;
960 GtkTreeIter iter;
962 store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_STRING, PANGO_TYPE_WEIGHT);
964 foreach_ptr_array(group, g, keybinding_groups)
966 if (g > 0)
968 gtk_list_store_append(store, &iter);
969 gtk_list_store_set(store, &iter, -1);
971 gtk_list_store_append(store, &iter);
972 gtk_list_store_set(store, &iter, 0, group->label, 2, PANGO_WEIGHT_BOLD, -1);
974 foreach_ptr_array(kb, i, group->key_items)
976 gchar *shortcut, *label;
978 label = keybindings_get_label(kb);
979 shortcut = gtk_accelerator_get_label(kb->key, kb->mods);
981 gtk_list_store_append(store, &iter);
982 gtk_list_store_set(store, &iter, 0, label, 1, shortcut, 2, PANGO_WEIGHT_NORMAL, -1);
984 g_free(shortcut);
985 g_free(label);
988 gtk_tree_view_set_model(GTK_TREE_VIEW(tree), GTK_TREE_MODEL(store));
989 g_object_unref(store);
993 static GtkWidget *create_dialog(void)
995 GtkWidget *dialog, *tree, *label, *swin, *vbox;
996 GtkCellRenderer *text_renderer;
997 GtkTreeViewColumn *column;
999 dialog = gtk_dialog_new_with_buttons(_("Keyboard Shortcuts"), GTK_WINDOW(main_widgets.window),
1000 GTK_DIALOG_DESTROY_WITH_PARENT,
1001 GTK_STOCK_EDIT, GTK_RESPONSE_APPLY,
1002 GTK_STOCK_CLOSE, GTK_RESPONSE_CANCEL, NULL);
1003 vbox = ui_dialog_vbox_new(GTK_DIALOG(dialog));
1004 gtk_box_set_spacing(GTK_BOX(vbox), 6);
1005 gtk_widget_set_name(dialog, "GeanyDialog");
1007 gtk_window_set_default_size(GTK_WINDOW(dialog), -1, GEANY_DEFAULT_DIALOG_HEIGHT);
1009 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_CANCEL);
1011 label = gtk_label_new(_("The following keyboard shortcuts are configurable:"));
1012 gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
1014 tree = gtk_tree_view_new();
1015 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(tree), TRUE);
1016 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(tree), FALSE);
1018 text_renderer = gtk_cell_renderer_text_new();
1019 /* we can't use "weight-set", see https://bugzilla.gnome.org/show_bug.cgi?id=355214 */
1020 column = gtk_tree_view_column_new_with_attributes(
1021 NULL, text_renderer, "text", 0, "weight", 2, NULL);
1022 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column);
1024 text_renderer = gtk_cell_renderer_text_new();
1025 column = gtk_tree_view_column_new_with_attributes(NULL, text_renderer, "text", 1, NULL);
1026 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column);
1028 fill_shortcut_labels_treeview(tree);
1030 swin = gtk_scrolled_window_new(NULL, NULL);
1031 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(swin), GTK_POLICY_NEVER,
1032 GTK_POLICY_AUTOMATIC);
1033 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(swin), GTK_SHADOW_IN);
1034 gtk_container_add(GTK_CONTAINER(swin), tree);
1036 gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 6);
1037 gtk_box_pack_start(GTK_BOX(vbox), swin, TRUE, TRUE, 0);
1039 return dialog;
1043 static void key_dialog_show_prefs(void)
1045 GtkWidget *wid;
1047 prefs_show_dialog();
1048 /* select the KB page */
1049 wid = ui_lookup_widget(ui_widgets.prefs_dialog, "frame22");
1050 if (wid != NULL)
1052 GtkNotebook *nb = GTK_NOTEBOOK(ui_lookup_widget(ui_widgets.prefs_dialog, "notebook2"));
1053 if (nb != NULL)
1055 gtk_notebook_set_current_page(nb, gtk_notebook_page_num(nb, wid));
1061 void keybindings_dialog_show_prefs_scroll(const gchar *name)
1063 key_dialog_show_prefs();
1064 prefs_kb_search_name(name);
1068 /* non-modal keyboard shortcuts dialog, so user can edit whilst seeing the shortcuts */
1069 static GtkWidget *key_dialog = NULL;
1071 static void on_dialog_response(GtkWidget *dialog, gint response, gpointer user_data)
1073 if (response == GTK_RESPONSE_APPLY)
1075 key_dialog_show_prefs();
1077 gtk_widget_destroy(dialog);
1078 key_dialog = NULL;
1082 void keybindings_show_shortcuts(void)
1084 if (key_dialog)
1085 gtk_widget_destroy(key_dialog); /* in case the key_dialog is still visible */
1087 key_dialog = create_dialog();
1088 g_signal_connect(key_dialog, "response", G_CALLBACK(on_dialog_response), NULL);
1089 gtk_widget_show_all(key_dialog);
1093 static gboolean check_fixed_kb(guint keyval, guint state)
1095 /* check alt-0 to alt-9 for setting current notebook page */
1096 if (state == GDK_MOD1_MASK && keyval >= GDK_KEY_0 && keyval <= GDK_KEY_9)
1098 gint page = keyval - GDK_KEY_0 - 1;
1099 gint npages = gtk_notebook_get_n_pages(GTK_NOTEBOOK(main_widgets.notebook));
1101 /* alt-0 is for the rightmost tab */
1102 if (keyval == GDK_KEY_0)
1103 page = npages - 1;
1104 /* invert the order if tabs are added on the other side */
1105 if (swap_alt_tab_order && ! file_prefs.tab_order_ltr)
1106 page = (npages - 1) - page;
1108 gtk_notebook_set_current_page(GTK_NOTEBOOK(main_widgets.notebook), page);
1109 return TRUE;
1111 /* note: these are now overridden by default with move tab bindings */
1112 if (keyval == GDK_KEY_Page_Up || keyval == GDK_KEY_Page_Down)
1114 /* switch to first or last document */
1115 if (state == (GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK))
1117 if (keyval == GDK_KEY_Page_Up)
1118 gtk_notebook_set_current_page(GTK_NOTEBOOK(main_widgets.notebook), 0);
1119 if (keyval == GDK_KEY_Page_Down)
1120 gtk_notebook_set_current_page(GTK_NOTEBOOK(main_widgets.notebook), -1);
1121 return TRUE;
1124 return FALSE;
1128 static gboolean check_snippet_completion(GeanyDocument *doc)
1130 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
1132 g_return_val_if_fail(doc, FALSE);
1134 /* keybinding only valid when scintilla widget has focus */
1135 if (focusw == GTK_WIDGET(doc->editor->sci))
1137 ScintillaObject *sci = doc->editor->sci;
1138 gint pos = sci_get_current_position(sci);
1140 if (editor_prefs.complete_snippets)
1141 return editor_complete_snippet(doc->editor, pos);
1143 return FALSE;
1147 /* Transforms a GdkEventKey event into a GdkEventButton event */
1148 static void trigger_button_event(GtkWidget *widget, guint32 event_time)
1150 GdkEventButton *event;
1151 gboolean ret;
1153 event = g_new0(GdkEventButton, 1);
1155 if (GTK_IS_TEXT_VIEW(widget))
1156 event->window = gtk_text_view_get_window(GTK_TEXT_VIEW(widget), GTK_TEXT_WINDOW_TEXT);
1157 else
1158 event->window = gtk_widget_get_window(widget);
1159 event->time = event_time;
1160 event->type = GDK_BUTTON_PRESS;
1161 event->button = 3;
1163 g_signal_emit_by_name(widget, "button-press-event", event, &ret);
1164 g_signal_emit_by_name(widget, "button-release-event", event, &ret);
1166 g_free(event);
1170 /* Special case for the Menu key and Shift-F10 to show the right-click popup menu for various
1171 * widgets. Without this special handling, the notebook tab list of the documents' notebook
1172 * would be shown. As a very special case, we differentiate between the Menu key and Shift-F10
1173 * if pressed in the editor widget: the Menu key opens the popup menu, Shift-F10 opens the
1174 * notebook tab list. */
1175 static gboolean check_menu_key(GeanyDocument *doc, guint keyval, guint state, guint32 event_time)
1177 g_return_val_if_fail(doc == NULL || doc->is_valid, FALSE);
1179 if ((keyval == GDK_KEY_Menu && state == 0) || (keyval == GDK_KEY_F10 && state == GDK_SHIFT_MASK))
1181 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
1182 if (doc != NULL)
1184 if (focusw == doc->priv->tag_tree)
1186 trigger_button_event(focusw, event_time);
1187 return TRUE;
1189 if (focusw == GTK_WIDGET(doc->editor->sci))
1191 if (keyval == GDK_KEY_Menu)
1192 { /* show editor popup menu */
1193 trigger_button_event(focusw, event_time);
1194 return TRUE;
1196 else
1197 { /* show tab bar menu */
1198 trigger_button_event(main_widgets.notebook, event_time);
1199 return TRUE;
1203 if (focusw == tv.tree_openfiles
1204 || focusw == msgwindow.tree_status
1205 || focusw == msgwindow.tree_compiler
1206 || focusw == msgwindow.tree_msg
1207 || focusw == msgwindow.scribble
1208 #ifdef HAVE_VTE
1209 || (vte_info.have_vte && focusw == vte_config.vte)
1210 #endif
1213 trigger_button_event(focusw, event_time);
1214 return TRUE;
1217 return FALSE;
1221 #ifdef HAVE_VTE
1222 static gboolean set_sensitive(gpointer widget)
1224 gtk_widget_set_sensitive(GTK_WIDGET(widget), TRUE);
1225 return FALSE;
1229 static gboolean check_vte(GdkModifierType state, guint keyval)
1231 guint i;
1232 GeanyKeyBinding *kb;
1233 GeanyKeyGroup *group;
1234 GtkWidget *widget;
1236 if (gtk_window_get_focus(GTK_WINDOW(main_widgets.window)) != vte_config.vte)
1237 return FALSE;
1238 /* let VTE copy/paste override any user keybinding */
1239 if (state == (GEANY_PRIMARY_MOD_MASK | GDK_SHIFT_MASK) && (keyval == GDK_KEY_c || keyval == GDK_KEY_v))
1240 return TRUE;
1241 if (! vte_config.enable_bash_keys)
1242 return FALSE;
1243 /* prevent menubar flickering: */
1244 if (state == GDK_SHIFT_MASK && (keyval >= GDK_KEY_a && keyval <= GDK_KEY_z))
1245 return FALSE;
1246 if (state == 0 && (keyval < GDK_KEY_F1 || keyval > GDK_KEY_F35)) /* e.g. backspace */
1247 return FALSE;
1249 /* make focus commands override any bash commands */
1250 group = keybindings_get_core_group(GEANY_KEY_GROUP_FOCUS);
1251 foreach_ptr_array(kb, i, group->key_items)
1253 if (state == kb->mods && keyval == kb->key)
1254 return FALSE;
1257 /* Temporarily disable the menus to prevent conflicting menu accelerators
1258 * from overriding the VTE bash shortcuts.
1259 * Note: maybe there's a better way of doing this ;-) */
1260 widget = ui_lookup_widget(main_widgets.window, "menubar1");
1261 gtk_widget_set_sensitive(widget, FALSE);
1262 g_idle_add_full(G_PRIORITY_HIGH, set_sensitive, widget, NULL);
1264 widget = main_widgets.editor_menu;
1265 gtk_widget_set_sensitive(widget, FALSE);
1266 g_idle_add(set_sensitive, widget);
1267 return TRUE;
1269 #endif
1272 /* Map the keypad keys to their equivalent functions (taken from ScintillaGTK.cxx) */
1273 static guint key_kp_translate(guint key_in)
1275 switch (key_in)
1277 case GDK_KEY_KP_Down:
1278 return GDK_KEY_Down;
1279 case GDK_KEY_KP_Up:
1280 return GDK_KEY_Up;
1281 case GDK_KEY_KP_Left:
1282 return GDK_KEY_Left;
1283 case GDK_KEY_KP_Right:
1284 return GDK_KEY_Right;
1285 case GDK_KEY_KP_Home:
1286 return GDK_KEY_Home;
1287 case GDK_KEY_KP_End:
1288 return GDK_KEY_End;
1289 case GDK_KEY_KP_Page_Up:
1290 return GDK_KEY_Page_Up;
1291 case GDK_KEY_KP_Page_Down:
1292 return GDK_KEY_Page_Down;
1293 case GDK_KEY_KP_Delete:
1294 return GDK_KEY_Delete;
1295 case GDK_KEY_KP_Insert:
1296 return GDK_KEY_Insert;
1297 default:
1298 return key_in;
1303 /* Check if event keypress matches keybinding combo */
1304 gboolean keybindings_check_event(GdkEventKey *ev, GeanyKeyBinding *kb)
1306 guint state, keyval;
1308 if (ev->keyval == 0)
1309 return FALSE;
1311 keyval = ev->keyval;
1312 state = keybindings_get_modifiers(ev->state);
1313 /* hack to get around that CTRL+Shift+r results in GDK_KEY_R not GDK_KEY_r */
1314 if ((ev->state & GDK_SHIFT_MASK) || (ev->state & GDK_LOCK_MASK))
1315 if (keyval >= GDK_KEY_A && keyval <= GDK_KEY_Z)
1316 keyval += GDK_KEY_a - GDK_KEY_A;
1318 if (keyval >= GDK_KEY_KP_Space && keyval < GDK_KEY_KP_Equal)
1319 keyval = key_kp_translate(keyval);
1321 return (keyval == kb->key && state == kb->mods);
1325 static gboolean run_kb(GeanyKeyBinding *kb, GeanyKeyGroup *group)
1327 gboolean handled = TRUE;
1328 /* call the corresponding handler/callback functions for this shortcut.
1329 * Check the individual keybindings first (handler first, callback second) and
1330 * group second (again handler first, callback second) */
1331 if (kb->cb_func)
1332 handled = kb->cb_func(kb, kb->id, kb->cb_data);
1333 else if (kb->callback)
1334 kb->callback(kb->id);
1335 else if (group->cb_func)
1336 handled = group->cb_func(group, kb->id, group->cb_data);
1337 else if (group->callback)
1338 handled = group->callback(kb->id);
1339 else
1341 g_warning("No callback or handler for keybinding %s: %s!", group->name, kb->name);
1342 return FALSE;
1345 return handled;
1349 /* central keypress event handler, almost all keypress events go to this function */
1350 static gboolean on_key_press_event(GtkWidget *widget, GdkEventKey *ev, gpointer user_data)
1352 guint state, keyval;
1353 gsize g, i;
1354 GeanyDocument *doc;
1355 GeanyKeyGroup *group;
1356 GeanyKeyBinding *kb;
1357 gboolean key_press_ret;
1359 if (ev->keyval == 0)
1360 return FALSE;
1362 g_signal_emit_by_name(geany_object, "key-press", ev, &key_press_ret);
1363 if (key_press_ret)
1364 return TRUE;
1366 doc = document_get_current();
1367 if (doc)
1368 document_check_disk_status(doc, FALSE);
1370 keyval = ev->keyval;
1371 state = keybindings_get_modifiers(ev->state);
1372 /* hack to get around that CTRL+Shift+r results in GDK_KEY_R not GDK_KEY_r */
1373 if ((ev->state & GDK_SHIFT_MASK) || (ev->state & GDK_LOCK_MASK))
1374 if (keyval >= GDK_KEY_A && keyval <= GDK_KEY_Z)
1375 keyval += GDK_KEY_a - GDK_KEY_A;
1377 if (keyval >= GDK_KEY_KP_Space && keyval < GDK_KEY_KP_Equal)
1378 keyval = key_kp_translate(keyval);
1380 /*geany_debug("%d (%d) %d (%d)", keyval, ev->keyval, state, ev->state);*/
1382 /* special cases */
1383 #ifdef HAVE_VTE
1384 if (vte_info.have_vte && check_vte(state, keyval))
1385 return FALSE;
1386 #endif
1387 if (check_menu_key(doc, keyval, state, ev->time))
1388 return TRUE;
1390 foreach_ptr_array(group, g, keybinding_groups)
1392 foreach_ptr_array(kb, i, group->key_items)
1394 if (keyval == kb->key && state == kb->mods)
1396 if (run_kb(kb, group))
1397 return TRUE;
1401 /* fixed keybindings can be overridden by user bindings, so check them last */
1402 if (check_fixed_kb(keyval, state))
1403 return TRUE;
1404 return FALSE;
1408 /* group_id must be a core group, e.g. GEANY_KEY_GROUP_EDITOR
1409 * key_id e.g. GEANY_KEYS_EDITOR_CALLTIP */
1410 GeanyKeyBinding *keybindings_lookup_item(guint group_id, guint key_id)
1412 GeanyKeyGroup *group;
1414 g_return_val_if_fail(group_id < GEANY_KEY_GROUP_COUNT, NULL); /* can't use this for plugin groups */
1416 group = keybindings_get_core_group(group_id);
1418 g_return_val_if_fail(group, NULL);
1419 return keybindings_get_item(group, key_id);
1423 /** Mimics a (built-in only) keybinding action.
1424 * Example: @code keybindings_send_command(GEANY_KEY_GROUP_FILE, GEANY_KEYS_FILE_OPEN); @endcode
1425 * @param group_id @ref GeanyKeyGroupID keybinding group index that contains the @a key_id keybinding.
1426 * @param key_id @ref GeanyKeyBindingID keybinding index. */
1427 GEANY_API_SYMBOL
1428 void keybindings_send_command(guint group_id, guint key_id)
1430 GeanyKeyBinding *kb;
1431 GeanyKeyGroup *group;
1433 kb = keybindings_lookup_item(group_id, key_id);
1434 group = keybindings_get_core_group(group_id);
1435 if (kb && group)
1436 run_kb(kb, group);
1440 /* These are the callback functions, either each group or each shortcut has it's
1441 * own function. */
1444 static gboolean cb_func_file_action(guint key_id)
1446 switch (key_id)
1448 case GEANY_KEYS_FILE_NEW:
1449 document_new_file(NULL, NULL, NULL);
1450 cb_func_switch_action(GEANY_KEYS_FOCUS_EDITOR);
1451 break;
1452 case GEANY_KEYS_FILE_OPEN:
1453 on_open1_activate(NULL, NULL);
1454 break;
1455 case GEANY_KEYS_FILE_OPENSELECTED:
1456 on_menu_open_selected_file1_activate(NULL, NULL);
1457 break;
1458 case GEANY_KEYS_FILE_OPENLASTTAB:
1460 gchar *utf8_filename = g_queue_peek_head(ui_prefs.recent_queue);
1461 gchar *locale_filename = utils_get_locale_from_utf8(utf8_filename);
1462 document_open_file(locale_filename, FALSE, NULL, NULL);
1463 g_free(locale_filename);
1464 break;
1466 case GEANY_KEYS_FILE_SAVE:
1467 on_save1_activate(NULL, NULL);
1468 break;
1469 case GEANY_KEYS_FILE_SAVEAS:
1470 on_save_as1_activate(NULL, NULL);
1471 break;
1472 case GEANY_KEYS_FILE_SAVEALL:
1473 on_save_all1_activate(NULL, NULL);
1474 break;
1475 case GEANY_KEYS_FILE_CLOSE:
1476 on_close1_activate(NULL, NULL);
1477 break;
1478 case GEANY_KEYS_FILE_CLOSEALL:
1479 on_close_all1_activate(NULL, NULL);
1480 break;
1481 case GEANY_KEYS_FILE_RELOAD:
1482 on_toolbutton_reload_clicked(NULL, NULL);
1483 break;
1484 case GEANY_KEYS_FILE_RELOAD_ALL:
1485 on_reload_all(NULL, NULL);
1486 break;
1487 case GEANY_KEYS_FILE_PRINT:
1488 on_print1_activate(NULL, NULL);
1489 break;
1490 case GEANY_KEYS_FILE_PROPERTIES:
1491 on_file_properties_activate(NULL, NULL);
1492 break;
1493 case GEANY_KEYS_FILE_QUIT:
1494 main_quit();
1495 break;
1497 return TRUE;
1501 static gboolean cb_func_project_action(guint key_id)
1503 switch (key_id)
1505 case GEANY_KEYS_PROJECT_NEW:
1506 on_project_new1_activate(NULL, NULL);
1507 break;
1508 case GEANY_KEYS_PROJECT_NEW_FROM_FOLDER:
1509 on_project_new_from_folder1_activate(NULL, NULL);
1510 break;
1511 case GEANY_KEYS_PROJECT_OPEN:
1512 on_project_open1_activate(NULL, NULL);
1513 break;
1514 case GEANY_KEYS_PROJECT_CLOSE:
1515 if (app->project)
1516 on_project_close1_activate(NULL, NULL);
1517 break;
1518 case GEANY_KEYS_PROJECT_PROPERTIES:
1519 if (app->project)
1520 on_project_properties1_activate(NULL, NULL);
1521 break;
1523 return TRUE;
1527 static void cb_func_menu_preferences(guint key_id)
1529 switch (key_id)
1531 case GEANY_KEYS_SETTINGS_PREFERENCES:
1532 on_preferences1_activate(NULL, NULL);
1533 break;
1534 case GEANY_KEYS_SETTINGS_PLUGINPREFERENCES:
1535 on_plugin_preferences1_activate(NULL, NULL);
1536 break;
1541 static void cb_func_menu_help(G_GNUC_UNUSED guint key_id)
1543 on_help1_activate(NULL, NULL);
1547 static gboolean cb_func_search_action(guint key_id)
1549 GeanyDocument *doc = document_get_current();
1550 ScintillaObject *sci;
1552 /* these work without docs */
1553 switch (key_id)
1555 case GEANY_KEYS_SEARCH_FINDINFILES:
1556 on_find_in_files1_activate(NULL, NULL); return TRUE;
1557 case GEANY_KEYS_SEARCH_NEXTMESSAGE:
1558 on_next_message1_activate(NULL, NULL); return TRUE;
1559 case GEANY_KEYS_SEARCH_PREVIOUSMESSAGE:
1560 on_previous_message1_activate(NULL, NULL); return TRUE;
1562 if (!doc)
1563 return TRUE;
1564 sci = doc->editor->sci;
1566 switch (key_id)
1568 case GEANY_KEYS_SEARCH_FIND:
1569 on_find1_activate(NULL, NULL); break;
1570 case GEANY_KEYS_SEARCH_FINDNEXT:
1571 on_find_next1_activate(NULL, NULL); break;
1572 case GEANY_KEYS_SEARCH_FINDPREVIOUS:
1573 on_find_previous1_activate(NULL, NULL); break;
1574 case GEANY_KEYS_SEARCH_FINDPREVSEL:
1575 on_find_prevsel1_activate(NULL, NULL); break;
1576 case GEANY_KEYS_SEARCH_FINDNEXTSEL:
1577 on_find_nextsel1_activate(NULL, NULL); break;
1578 case GEANY_KEYS_SEARCH_REPLACE:
1579 on_replace1_activate(NULL, NULL); break;
1580 case GEANY_KEYS_SEARCH_FINDUSAGE:
1581 on_find_usage1_activate(NULL, NULL); break;
1582 case GEANY_KEYS_SEARCH_FINDDOCUMENTUSAGE:
1583 on_find_document_usage1_activate(NULL, NULL); break;
1584 case GEANY_KEYS_SEARCH_MARKALL:
1586 gchar *text = NULL;
1587 gint pos = sci_get_current_position(sci);
1589 /* clear existing search indicators instead if next to cursor */
1590 if (SSM(sci, SCI_INDICATORVALUEAT,
1591 GEANY_INDICATOR_SEARCH, pos) ||
1592 SSM(sci, SCI_INDICATORVALUEAT,
1593 GEANY_INDICATOR_SEARCH, MAX(pos - 1, 0)))
1595 text = NULL;
1597 else
1599 text = get_current_word_or_sel(doc, TRUE);
1602 if (sci_has_selection(sci))
1603 search_mark_all(doc, text, GEANY_FIND_MATCHCASE);
1604 else
1605 search_mark_all(doc, text, GEANY_FIND_MATCHCASE | GEANY_FIND_WHOLEWORD);
1607 g_free(text);
1608 break;
1611 return TRUE;
1615 static void cb_func_menu_opencolorchooser(G_GNUC_UNUSED guint key_id)
1617 on_show_color_chooser1_activate(NULL, NULL);
1621 static gboolean cb_func_view_action(guint key_id)
1623 switch (key_id)
1625 case GEANY_KEYS_VIEW_TOGGLEALL:
1626 on_menu_toggle_all_additional_widgets1_activate(NULL, NULL);
1627 break;
1628 case GEANY_KEYS_VIEW_SIDEBAR:
1629 on_menu_show_sidebar1_toggled(NULL, NULL);
1630 break;
1631 case GEANY_KEYS_VIEW_ZOOMIN:
1632 on_zoom_in1_activate(NULL, NULL);
1633 break;
1634 case GEANY_KEYS_VIEW_ZOOMOUT:
1635 on_zoom_out1_activate(NULL, NULL);
1636 break;
1637 case GEANY_KEYS_VIEW_ZOOMRESET:
1638 on_normal_size1_activate(NULL, NULL);
1639 break;
1640 default:
1641 break;
1643 return TRUE;
1647 static void cb_func_menu_fullscreen(G_GNUC_UNUSED guint key_id)
1649 GtkCheckMenuItem *c = GTK_CHECK_MENU_ITEM(
1650 ui_lookup_widget(main_widgets.window, "menu_fullscreen1"));
1652 gtk_check_menu_item_set_active(c, ! gtk_check_menu_item_get_active(c));
1656 static void cb_func_menu_messagewindow(G_GNUC_UNUSED guint key_id)
1658 GtkCheckMenuItem *c = GTK_CHECK_MENU_ITEM(
1659 ui_lookup_widget(main_widgets.window, "menu_show_messages_window1"));
1661 gtk_check_menu_item_set_active(c, ! gtk_check_menu_item_get_active(c));
1665 static gboolean read_current_word(GeanyDocument *doc, gboolean sci_word)
1667 g_return_val_if_fail(DOC_VALID(doc), FALSE);
1669 if (sci_word)
1671 editor_find_current_word_sciwc(doc->editor, -1,
1672 editor_info.current_word, GEANY_MAX_WORD_LENGTH);
1674 else
1676 editor_find_current_word(doc->editor, -1,
1677 editor_info.current_word, GEANY_MAX_WORD_LENGTH, NULL);
1680 return (*editor_info.current_word != 0);
1684 static gboolean check_current_word(GeanyDocument *doc, gboolean sci_word)
1686 if (! read_current_word(doc, sci_word))
1688 utils_beep();
1689 return FALSE;
1691 return TRUE;
1695 static gchar *get_current_word_or_sel(GeanyDocument *doc, gboolean sci_word)
1697 ScintillaObject *sci = doc->editor->sci;
1699 if (sci_has_selection(sci))
1700 return sci_get_selection_contents(sci);
1702 return read_current_word(doc, sci_word) ? g_strdup(editor_info.current_word) : NULL;
1706 static void focus_sidebar(void)
1708 if (ui_prefs.sidebar_visible)
1710 gint page_num = gtk_notebook_get_current_page(GTK_NOTEBOOK(main_widgets.sidebar_notebook));
1711 GtkWidget *page = gtk_notebook_get_nth_page(GTK_NOTEBOOK(main_widgets.sidebar_notebook), page_num);
1713 /* gtk_widget_grab_focus() won't work because of the scrolled window containers */
1714 gtk_widget_child_focus(page, GTK_DIR_TAB_FORWARD);
1719 static GtkWidget *find_focus_widget(GtkWidget *widget)
1721 GtkWidget *focus = NULL;
1723 if (GTK_IS_BIN(widget)) /* optimized simple case */
1724 focus = find_focus_widget(gtk_bin_get_child(GTK_BIN(widget)));
1725 else if (GTK_IS_CONTAINER(widget))
1727 GList *children = gtk_container_get_children(GTK_CONTAINER(widget));
1728 GList *node;
1730 for (node = children; node && ! focus; node = node->next)
1731 focus = find_focus_widget(node->data);
1732 g_list_free(children);
1735 /* Some containers handled above might not have children and be what we want to focus
1736 * (e.g. GtkTreeView), so focus that if possible and we don't have anything better */
1737 if (! focus && gtk_widget_get_can_focus(widget))
1738 focus = widget;
1740 return focus;
1744 static void focus_msgwindow(void)
1746 if (ui_prefs.msgwindow_visible)
1748 gint page_num = gtk_notebook_get_current_page(GTK_NOTEBOOK(msgwindow.notebook));
1749 GtkWidget *widget = gtk_notebook_get_nth_page(GTK_NOTEBOOK(msgwindow.notebook), page_num);
1751 widget = find_focus_widget(widget);
1752 if (widget)
1753 gtk_widget_grab_focus(widget);
1754 else
1755 utils_beep();
1760 static gboolean cb_func_switch_action(guint key_id)
1762 switch (key_id)
1764 case GEANY_KEYS_FOCUS_EDITOR:
1766 GeanyDocument *doc = document_get_current();
1767 if (doc != NULL)
1769 GtkWidget *sci = GTK_WIDGET(doc->editor->sci);
1770 if (gtk_widget_has_focus(sci))
1771 ui_update_statusbar(doc, -1);
1772 else
1773 gtk_widget_grab_focus(sci);
1775 break;
1777 case GEANY_KEYS_FOCUS_SCRIBBLE:
1778 msgwin_switch_tab(MSG_SCRATCH, TRUE);
1779 break;
1780 case GEANY_KEYS_FOCUS_SEARCHBAR:
1781 if (toolbar_prefs.visible)
1783 GtkWidget *search_entry = toolbar_get_widget_child_by_name("SearchEntry");
1784 if (search_entry != NULL)
1785 gtk_widget_grab_focus(search_entry);
1787 break;
1788 case GEANY_KEYS_FOCUS_SIDEBAR:
1789 focus_sidebar();
1790 break;
1791 case GEANY_KEYS_FOCUS_VTE:
1792 msgwin_switch_tab(MSG_VTE, TRUE);
1793 break;
1794 case GEANY_KEYS_FOCUS_COMPILER:
1795 msgwin_switch_tab(MSG_COMPILER, TRUE);
1796 break;
1797 case GEANY_KEYS_FOCUS_MESSAGES:
1798 msgwin_switch_tab(MSG_MESSAGE, TRUE);
1799 break;
1800 case GEANY_KEYS_FOCUS_MESSAGE_WINDOW:
1801 focus_msgwindow();
1802 break;
1803 case GEANY_KEYS_FOCUS_SIDEBAR_DOCUMENT_LIST:
1804 sidebar_focus_openfiles_tab();
1805 break;
1806 case GEANY_KEYS_FOCUS_SIDEBAR_SYMBOL_LIST:
1807 sidebar_focus_symbols_tab();
1808 break;
1810 return TRUE;
1814 static void switch_notebook_page(gint direction)
1816 gint page_count, cur_page, pass;
1817 gboolean parent_is_notebook = FALSE;
1818 GtkNotebook *notebook;
1819 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
1821 /* check whether the current widget is a GtkNotebook or a child of a GtkNotebook */
1824 parent_is_notebook = GTK_IS_NOTEBOOK(focusw);
1826 while (! parent_is_notebook && (focusw = gtk_widget_get_parent(focusw)) != NULL);
1828 /* if we found a GtkNotebook widget, use it. Otherwise fallback to the documents notebook */
1829 if (parent_is_notebook)
1830 notebook = GTK_NOTEBOOK(focusw);
1831 else
1832 notebook = GTK_NOTEBOOK(main_widgets.notebook);
1834 /* now switch pages */
1835 page_count = gtk_notebook_get_n_pages(notebook);
1836 cur_page = gtk_notebook_get_current_page(notebook);
1838 /* find the next visible page in the wanted direction, but don't loop
1839 * indefinitely if no pages are visible */
1840 for (pass = 0; pass < page_count; pass++)
1842 GtkWidget *child;
1844 if (direction == GTK_DIR_LEFT)
1846 if (cur_page > 0)
1847 cur_page--;
1848 else
1849 cur_page = page_count - 1;
1851 else if (direction == GTK_DIR_RIGHT)
1853 if (cur_page < page_count - 1)
1854 cur_page++;
1855 else
1856 cur_page = 0;
1859 child = gtk_notebook_get_nth_page (notebook, cur_page);
1860 if (gtk_widget_get_visible (child))
1862 gtk_notebook_set_current_page(notebook, cur_page);
1863 break;
1869 static void cb_func_switch_tableft(G_GNUC_UNUSED guint key_id)
1871 switch_notebook_page(GTK_DIR_LEFT);
1875 static void cb_func_switch_tabright(G_GNUC_UNUSED guint key_id)
1877 switch_notebook_page(GTK_DIR_RIGHT);
1881 static void cb_func_switch_tablastused(G_GNUC_UNUSED guint key_id)
1883 notebook_switch_tablastused();
1887 /* move document left/right/first/last */
1888 static void cb_func_move_tab(guint key_id)
1890 GtkWidget *child;
1891 GtkNotebook *nb = GTK_NOTEBOOK(main_widgets.notebook);
1892 gint cur_page = gtk_notebook_get_current_page(nb);
1894 if (cur_page < 0)
1895 return;
1897 child = gtk_notebook_get_nth_page(nb, cur_page);
1899 switch (key_id)
1901 case GEANY_KEYS_NOTEBOOK_MOVETABLEFT:
1902 gtk_notebook_reorder_child(nb, child, cur_page - 1); /* notebook wraps around by default */
1903 break;
1904 case GEANY_KEYS_NOTEBOOK_MOVETABRIGHT:
1906 gint npage = cur_page + 1;
1908 if (npage == gtk_notebook_get_n_pages(nb))
1909 npage = 0; /* wraparound */
1910 gtk_notebook_reorder_child(nb, child, npage);
1911 break;
1913 case GEANY_KEYS_NOTEBOOK_MOVETABFIRST:
1914 gtk_notebook_reorder_child(nb, child, (file_prefs.tab_order_ltr) ? 0 : -1);
1915 break;
1916 case GEANY_KEYS_NOTEBOOK_MOVETABLAST:
1917 gtk_notebook_reorder_child(nb, child, (file_prefs.tab_order_ltr) ? -1 : 0);
1918 break;
1920 return;
1924 static void goto_matching_brace(GeanyDocument *doc)
1926 gint pos, new_pos;
1927 gint after_brace;
1929 g_return_if_fail(DOC_VALID(doc));
1931 pos = sci_get_current_position(doc->editor->sci);
1932 after_brace = pos > 0 && utils_isbrace(sci_get_char_at(doc->editor->sci, pos - 1), TRUE);
1933 pos -= after_brace; /* set pos to the brace */
1935 new_pos = sci_find_matching_brace(doc->editor->sci, pos);
1936 if (new_pos != -1)
1937 { /* set the cursor at/after the brace */
1938 sci_set_current_position(doc->editor->sci, new_pos + (!after_brace), FALSE);
1939 editor_display_current_line(doc->editor, 0.5F);
1944 static gboolean cb_func_clipboard_action(guint key_id)
1946 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
1948 switch (key_id)
1950 case GEANY_KEYS_CLIPBOARD_CUT:
1951 on_cut1_activate(NULL, NULL);
1952 break;
1953 case GEANY_KEYS_CLIPBOARD_COPY:
1954 on_copy1_activate(NULL, NULL);
1955 break;
1956 case GEANY_KEYS_CLIPBOARD_PASTE:
1957 on_paste1_activate(NULL, NULL);
1958 break;
1959 case GEANY_KEYS_CLIPBOARD_COPYLINE:
1960 if (IS_SCINTILLA(focusw))
1961 sci_send_command(SCINTILLA(focusw), SCI_LINECOPY);
1962 break;
1963 case GEANY_KEYS_CLIPBOARD_CUTLINE:
1964 if (IS_SCINTILLA(focusw))
1965 sci_send_command(SCINTILLA(focusw), SCI_LINECUT);
1966 break;
1968 return TRUE;
1972 static void goto_tag(GeanyDocument *doc, gboolean definition)
1974 gchar *text = get_current_word_or_sel(doc, FALSE);
1976 if (text)
1977 symbols_goto_tag(text, definition);
1978 else
1979 utils_beep();
1981 g_free(text);
1985 /* Common function for goto keybindings, useful even when sci doesn't have focus. */
1986 static gboolean cb_func_goto_action(guint key_id)
1988 gint cur_line;
1989 GeanyDocument *doc = document_get_current();
1991 if (doc == NULL)
1992 return TRUE;
1994 cur_line = sci_get_current_line(doc->editor->sci);
1996 switch (key_id)
1998 case GEANY_KEYS_GOTO_BACK:
1999 navqueue_go_back();
2000 return TRUE;
2001 case GEANY_KEYS_GOTO_FORWARD:
2002 navqueue_go_forward();
2003 return TRUE;
2004 case GEANY_KEYS_GOTO_LINE:
2006 if (toolbar_prefs.visible)
2008 GtkWidget *wid = toolbar_get_widget_child_by_name("GotoEntry");
2010 /* use toolbar item if shown & not in the drop down overflow menu */
2011 if (wid && gtk_widget_get_mapped(wid))
2013 gtk_widget_grab_focus(wid);
2014 return TRUE;
2017 on_go_to_line_activate(NULL, NULL);
2018 return TRUE;
2020 case GEANY_KEYS_GOTO_MATCHINGBRACE:
2021 goto_matching_brace(doc);
2022 return TRUE;
2023 case GEANY_KEYS_GOTO_TOGGLEMARKER:
2025 sci_toggle_marker_at_line(doc->editor->sci, cur_line, 1);
2026 return TRUE;
2028 case GEANY_KEYS_GOTO_NEXTMARKER:
2030 gint mline = sci_marker_next(doc->editor->sci, cur_line + 1, 1 << 1, TRUE);
2032 if (mline != -1)
2034 sci_set_current_line(doc->editor->sci, mline);
2035 editor_display_current_line(doc->editor, 0.5F);
2037 return TRUE;
2039 case GEANY_KEYS_GOTO_PREVIOUSMARKER:
2041 gint mline = sci_marker_previous(doc->editor->sci, cur_line - 1, 1 << 1, TRUE);
2043 if (mline != -1)
2045 sci_set_current_line(doc->editor->sci, mline);
2046 editor_display_current_line(doc->editor, 0.5F);
2048 return TRUE;
2050 case GEANY_KEYS_GOTO_TAGDEFINITION:
2051 goto_tag(doc, TRUE);
2052 return TRUE;
2053 case GEANY_KEYS_GOTO_TAGDECLARATION:
2054 goto_tag(doc, FALSE);
2055 return TRUE;
2057 /* only check editor-sensitive keybindings when editor has focus so home,end still
2058 * work in other widgets */
2059 if (gtk_window_get_focus(GTK_WINDOW(main_widgets.window)) != GTK_WIDGET(doc->editor->sci))
2060 return FALSE;
2062 switch (key_id)
2064 case GEANY_KEYS_GOTO_LINESTART:
2065 sci_send_command(doc->editor->sci, editor_prefs.smart_home_key ? SCI_VCHOME : SCI_HOME);
2066 break;
2067 case GEANY_KEYS_GOTO_LINEEND:
2068 sci_send_command(doc->editor->sci, SCI_LINEEND);
2069 break;
2070 case GEANY_KEYS_GOTO_LINESTARTVISUAL:
2071 sci_send_command(doc->editor->sci, editor_prefs.smart_home_key ? SCI_VCHOMEDISPLAY : SCI_HOMEDISPLAY);
2072 break;
2073 case GEANY_KEYS_GOTO_LINEENDVISUAL:
2074 sci_send_command(doc->editor->sci, SCI_LINEENDDISPLAY);
2075 break;
2076 case GEANY_KEYS_GOTO_PREVWORDPART:
2077 sci_send_command(doc->editor->sci, SCI_WORDPARTLEFT);
2078 break;
2079 case GEANY_KEYS_GOTO_NEXTWORDPART:
2080 sci_send_command(doc->editor->sci, SCI_WORDPARTRIGHT);
2081 break;
2083 return TRUE;
2087 static void duplicate_lines(GeanyEditor *editor)
2089 if (sci_get_lines_selected(editor->sci) > 1)
2090 { /* ignore extra_line because of selecting lines from the line number column */
2091 editor_select_lines(editor, FALSE);
2092 sci_selection_duplicate(editor->sci);
2094 else if (sci_has_selection(editor->sci))
2095 sci_selection_duplicate(editor->sci);
2096 else
2097 sci_line_duplicate(editor->sci);
2101 static void delete_lines(GeanyEditor *editor)
2103 editor_select_lines(editor, TRUE); /* include last line (like cut lines, copy lines do) */
2104 sci_clear(editor->sci); /* (SCI_LINEDELETE only does 1 line) */
2108 /* common function for editor keybindings, only valid when scintilla has focus. */
2109 static gboolean cb_func_editor_action(guint key_id)
2111 GeanyDocument *doc = document_get_current();
2112 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
2114 /* edit keybindings only valid when scintilla widget has focus */
2115 if (doc == NULL || focusw != GTK_WIDGET(doc->editor->sci))
2116 return FALSE; /* also makes tab work outside editor */
2118 switch (key_id)
2120 case GEANY_KEYS_EDITOR_UNDO:
2121 on_undo1_activate(NULL, NULL);
2122 break;
2123 case GEANY_KEYS_EDITOR_REDO:
2124 on_redo1_activate(NULL, NULL);
2125 break;
2126 case GEANY_KEYS_EDITOR_SCROLLTOLINE:
2127 editor_scroll_to_line(doc->editor, -1, 0.5F);
2128 break;
2129 case GEANY_KEYS_EDITOR_SCROLLLINEUP:
2130 sci_send_command(doc->editor->sci, SCI_LINESCROLLUP);
2131 break;
2132 case GEANY_KEYS_EDITOR_SCROLLLINEDOWN:
2133 sci_send_command(doc->editor->sci, SCI_LINESCROLLDOWN);
2134 break;
2135 case GEANY_KEYS_EDITOR_DUPLICATELINE:
2136 duplicate_lines(doc->editor);
2137 break;
2138 case GEANY_KEYS_EDITOR_SNIPPETNEXTCURSOR:
2139 /* allow overloading */
2140 return editor_goto_next_snippet_cursor(doc->editor);
2141 case GEANY_KEYS_EDITOR_DELETELINE:
2142 delete_lines(doc->editor);
2143 break;
2144 case GEANY_KEYS_EDITOR_DELETELINETOEND:
2145 sci_send_command(doc->editor->sci, SCI_DELLINERIGHT);
2146 break;
2147 case GEANY_KEYS_EDITOR_DELETELINETOBEGINNING:
2148 sci_send_command(doc->editor->sci, SCI_DELLINELEFT);
2149 break;
2150 case GEANY_KEYS_EDITOR_TRANSPOSELINE:
2151 sci_send_command(doc->editor->sci, SCI_LINETRANSPOSE);
2152 break;
2153 case GEANY_KEYS_EDITOR_AUTOCOMPLETE:
2154 editor_start_auto_complete(doc->editor, sci_get_current_position(doc->editor->sci), TRUE);
2155 break;
2156 case GEANY_KEYS_EDITOR_CALLTIP:
2157 editor_show_calltip(doc->editor, -1);
2158 break;
2159 case GEANY_KEYS_EDITOR_CONTEXTACTION:
2160 if (check_current_word(doc, FALSE))
2161 on_context_action1_activate(GTK_MENU_ITEM(ui_lookup_widget(main_widgets.editor_menu,
2162 "context_action1")), NULL);
2163 break;
2164 case GEANY_KEYS_EDITOR_COMPLETESNIPPET:
2165 /* allow tab to be overloaded */
2166 return check_snippet_completion(doc);
2168 case GEANY_KEYS_EDITOR_SUPPRESSSNIPPETCOMPLETION:
2170 GeanyKeyBinding *kb = keybindings_lookup_item(GEANY_KEY_GROUP_EDITOR,
2171 GEANY_KEYS_EDITOR_COMPLETESNIPPET);
2173 switch (kb->key)
2175 case GDK_KEY_space:
2176 sci_add_text(doc->editor->sci, " ");
2177 break;
2178 case GDK_KEY_Tab:
2179 sci_send_command(doc->editor->sci, SCI_TAB);
2180 break;
2181 default:
2182 break;
2184 break;
2186 case GEANY_KEYS_EDITOR_WORDPARTCOMPLETION:
2187 return editor_complete_word_part(doc->editor);
2189 case GEANY_KEYS_EDITOR_MOVELINEUP:
2190 sci_move_selected_lines_up(doc->editor->sci);
2191 break;
2192 case GEANY_KEYS_EDITOR_MOVELINEDOWN:
2193 sci_move_selected_lines_down(doc->editor->sci);
2194 break;
2196 return TRUE;
2200 static void join_lines(GeanyEditor *editor)
2202 gint start, end, i;
2204 start = sci_get_line_from_position(editor->sci,
2205 sci_get_selection_start(editor->sci));
2206 end = sci_get_line_from_position(editor->sci,
2207 sci_get_selection_end(editor->sci));
2209 /* remove spaces surrounding the lines so that these spaces
2210 * won't appear within text after joining */
2211 for (i = start; i < end; i++)
2212 editor_strip_line_trailing_spaces(editor, i);
2213 for (i = start + 1; i <= end; i++)
2214 sci_set_line_indentation(editor->sci, i, 0);
2216 sci_set_target_start(editor->sci,
2217 sci_get_position_from_line(editor->sci, start));
2218 sci_set_target_end(editor->sci,
2219 sci_get_position_from_line(editor->sci, end));
2220 sci_lines_join(editor->sci);
2224 static gint get_reflow_column(GeanyEditor *editor)
2226 const GeanyEditorPrefs *eprefs = editor_get_prefs(editor);
2227 if (editor->line_breaking)
2228 return eprefs->line_break_column;
2229 else if (eprefs->long_line_type != 2)
2230 return eprefs->long_line_column;
2231 else
2232 return -1; /* do nothing */
2236 /* Split the line where the cursor is positioned, on `column`,
2237 possibly many times if the line is long.
2238 Return the number of lines added because of the splitting. */
2239 static gint split_line(GeanyEditor *editor, gint column)
2241 ScintillaObject *sci = editor->sci;
2242 gint start_line = sci_get_current_line(sci);
2243 gint line = start_line;
2245 while (TRUE)
2247 gint lstart = sci_get_position_from_line(sci, line);
2248 gint lend = sci_get_line_end_position(sci, line);
2249 gint edge = sci_get_position_from_col(sci, line, column);
2250 gboolean found;
2251 gint pos;
2253 /* don't split on a trailing space of a line */
2254 if (sci_get_char_at(sci, lend - 1) == ' ')
2255 lend--;
2257 /* detect when the line is short enough and no more splitting is needed */
2258 if (sci_get_col_from_position(sci, lend) < column)
2259 break;
2261 /* lookup split position */
2262 found = FALSE;
2263 for (pos = edge - 1; pos > lstart; pos--)
2265 if (sci_get_char_at(sci, pos) == ' ')
2267 found = TRUE;
2268 break;
2271 if (!found)
2273 for (pos = edge; pos < lend; pos++)
2275 if (sci_get_char_at(sci, pos) == ' ')
2277 found = TRUE;
2278 break;
2282 /* don't split right before a space */
2283 while (pos + 1 <= lend && sci_get_char_at(sci, pos + 1) == ' ')
2284 pos++;
2286 if (!found || pos >= lend)
2287 break;
2289 sci_insert_text(sci, pos + 1, editor_get_eol_char(editor));
2290 line++;
2292 return line - start_line;
2296 static void reflow_lines(GeanyEditor *editor, gint column)
2298 gint start, indent, linescount, i;
2300 start = sci_get_line_from_position(editor->sci,
2301 sci_get_selection_start(editor->sci));
2303 /* if several lines are selected, join them. */
2304 if (sci_get_lines_selected(editor->sci) > 1)
2305 join_lines(editor);
2307 /* if this line is short enough, do nothing */
2308 if (column > sci_get_line_end_position(editor->sci, start) -
2309 sci_get_position_from_line(editor->sci, start))
2311 return;
2315 * We have to manipulate line indentation so that indentation
2316 * of the resulting lines would be consistent. For example,
2317 * the result of splitting "[TAB]very long content":
2319 * +-------------+-------------+
2320 * | proper | wrong |
2321 * +-------------+-------------+
2322 * | [TAB]very | [TAB]very |
2323 * | [TAB]long | long |
2324 * | [TAB]content| content |
2325 * +-------------+-------------+
2327 indent = sci_get_line_indentation(editor->sci, start);
2328 sci_set_line_indentation(editor->sci, start, 0);
2330 linescount = split_line(editor, column - indent);
2332 /* Fix indentation. */
2333 for (i = start; i <= start + linescount; i++)
2334 sci_set_line_indentation(editor->sci, i, indent);
2336 /* Remove trailing spaces. */
2337 if (editor_prefs.newline_strip || file_prefs.strip_trailing_spaces)
2339 for (i = start; i <= start + linescount; i++)
2340 editor_strip_line_trailing_spaces(editor, i);
2345 /* deselect last newline of selection, if any */
2346 static void sci_deselect_last_newline(ScintillaObject *sci)
2348 gint start, end;
2350 start = sci_get_selection_start(sci);
2351 end = sci_get_selection_end(sci);
2352 if (end > start && sci_get_col_from_position(sci, end) == 0)
2354 end = sci_get_line_end_position(sci, sci_get_line_from_position(sci, end - 1));
2355 sci_set_selection(sci, start, end);
2360 static void reflow_paragraph(GeanyEditor *editor)
2362 ScintillaObject *sci = editor->sci;
2363 gboolean sel;
2364 gint column;
2366 column = get_reflow_column(editor);
2367 if (column == -1)
2369 utils_beep();
2370 return;
2373 sci_start_undo_action(sci);
2374 sel = sci_has_selection(sci);
2375 if (!sel)
2376 editor_select_indent_block(editor);
2377 sci_deselect_last_newline(sci);
2378 reflow_lines(editor, column);
2379 if (!sel)
2380 sci_set_anchor(sci, -1);
2381 sci_goto_pos(sci, sci_get_line_end_position(sci, sci_get_current_line(sci)), TRUE);
2383 sci_end_undo_action(sci);
2387 static void join_paragraph(GeanyEditor *editor)
2389 ScintillaObject *sci = editor->sci;
2390 gboolean sel;
2392 sci_start_undo_action(sci);
2393 sel = sci_has_selection(sci);
2394 if (!sel)
2395 editor_select_indent_block(editor);
2396 sci_deselect_last_newline(sci);
2397 join_lines(editor);
2398 if (!sel)
2399 sci_set_anchor(sci, -1);
2401 sci_end_undo_action(sci);
2405 /* common function for format keybindings, only valid when scintilla has focus. */
2406 static gboolean cb_func_format_action(guint key_id)
2408 GeanyDocument *doc = document_get_current();
2409 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
2411 /* keybindings only valid when scintilla widget has focus */
2412 if (doc == NULL || focusw != GTK_WIDGET(doc->editor->sci))
2413 return TRUE;
2415 switch (key_id)
2417 case GEANY_KEYS_FORMAT_COMMENTLINETOGGLE:
2418 on_menu_toggle_line_commentation1_activate(NULL, NULL);
2419 break;
2420 case GEANY_KEYS_FORMAT_COMMENTLINE:
2421 on_menu_comment_line1_activate(NULL, NULL);
2422 break;
2423 case GEANY_KEYS_FORMAT_UNCOMMENTLINE:
2424 on_menu_uncomment_line1_activate(NULL, NULL);
2425 break;
2426 case GEANY_KEYS_FORMAT_INCREASEINDENT:
2427 on_menu_increase_indent1_activate(NULL, NULL);
2428 break;
2429 case GEANY_KEYS_FORMAT_DECREASEINDENT:
2430 on_menu_decrease_indent1_activate(NULL, NULL);
2431 break;
2432 case GEANY_KEYS_FORMAT_INCREASEINDENTBYSPACE:
2433 editor_indentation_by_one_space(doc->editor, -1, FALSE);
2434 break;
2435 case GEANY_KEYS_FORMAT_DECREASEINDENTBYSPACE:
2436 editor_indentation_by_one_space(doc->editor, -1, TRUE);
2437 break;
2438 case GEANY_KEYS_FORMAT_AUTOINDENT:
2439 editor_smart_line_indentation(doc->editor);
2440 break;
2441 case GEANY_KEYS_FORMAT_TOGGLECASE:
2442 on_toggle_case1_activate(NULL, NULL);
2443 break;
2444 case GEANY_KEYS_FORMAT_SENDTOCMD1:
2445 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 0)
2446 tools_execute_custom_command(doc, ui_prefs.custom_commands[0]);
2447 break;
2448 case GEANY_KEYS_FORMAT_SENDTOCMD2:
2449 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 1)
2450 tools_execute_custom_command(doc, ui_prefs.custom_commands[1]);
2451 break;
2452 case GEANY_KEYS_FORMAT_SENDTOCMD3:
2453 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 2)
2454 tools_execute_custom_command(doc, ui_prefs.custom_commands[2]);
2455 break;
2456 case GEANY_KEYS_FORMAT_SENDTOCMD4:
2457 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 3)
2458 tools_execute_custom_command(doc, ui_prefs.custom_commands[3]);
2459 break;
2460 case GEANY_KEYS_FORMAT_SENDTOCMD5:
2461 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 4)
2462 tools_execute_custom_command(doc, ui_prefs.custom_commands[4]);
2463 break;
2464 case GEANY_KEYS_FORMAT_SENDTOCMD6:
2465 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 5)
2466 tools_execute_custom_command(doc, ui_prefs.custom_commands[5]);
2467 break;
2468 case GEANY_KEYS_FORMAT_SENDTOCMD7:
2469 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 6)
2470 tools_execute_custom_command(doc, ui_prefs.custom_commands[6]);
2471 break;
2472 case GEANY_KEYS_FORMAT_SENDTOCMD8:
2473 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 7)
2474 tools_execute_custom_command(doc, ui_prefs.custom_commands[7]);
2475 break;
2476 case GEANY_KEYS_FORMAT_SENDTOCMD9:
2477 if (ui_prefs.custom_commands && g_strv_length(ui_prefs.custom_commands) > 8)
2478 tools_execute_custom_command(doc, ui_prefs.custom_commands[8]);
2479 break;
2480 case GEANY_KEYS_FORMAT_SENDTOVTE:
2481 on_send_selection_to_vte1_activate(NULL, NULL);
2482 break;
2483 case GEANY_KEYS_FORMAT_REFLOWPARAGRAPH:
2484 reflow_paragraph(doc->editor);
2485 break;
2486 case GEANY_KEYS_FORMAT_JOINLINES:
2487 join_paragraph(doc->editor);
2488 break;
2490 return TRUE;
2494 /* common function for select keybindings, valid for scintilla and/or gtk_editable objects. */
2495 static gboolean cb_func_select_action(guint key_id)
2497 GeanyDocument *doc = document_get_current();
2498 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
2500 switch (key_id)
2502 case GEANY_KEYS_SELECT_ALL:
2503 on_menu_select_all1_activate(NULL, NULL);
2504 break;
2505 case GEANY_KEYS_SELECT_WORD:
2506 if (doc != NULL)
2507 editor_select_word(doc->editor);
2508 break;
2509 case GEANY_KEYS_SELECT_LINE:
2510 if (doc != NULL)
2511 editor_select_lines(doc->editor, FALSE);
2512 break;
2513 case GEANY_KEYS_SELECT_PARAGRAPH:
2514 if (doc != NULL)
2515 editor_select_paragraph(doc->editor);
2516 break;
2517 case GEANY_KEYS_SELECT_WORDPARTLEFT:
2518 if (IS_SCINTILLA(focusw))
2519 sci_send_command(SCINTILLA(focusw), SCI_WORDPARTLEFTEXTEND);
2520 break;
2521 case GEANY_KEYS_SELECT_WORDPARTRIGHT:
2522 if (IS_SCINTILLA(focusw))
2523 sci_send_command(SCINTILLA(focusw), SCI_WORDPARTRIGHTEXTEND);
2524 break;
2526 return TRUE;
2530 static gboolean cb_func_document_action(guint key_id)
2532 GeanyDocument *doc = document_get_current();
2534 if (doc == NULL)
2535 return TRUE;
2537 switch (key_id)
2539 case GEANY_KEYS_DOCUMENT_REPLACETABS:
2540 on_replace_tabs_activate(NULL, NULL);
2541 break;
2542 case GEANY_KEYS_DOCUMENT_REPLACESPACES:
2543 on_replace_spaces_activate(NULL, NULL);
2544 break;
2545 case GEANY_KEYS_DOCUMENT_LINEBREAK:
2546 on_line_breaking1_activate(NULL, NULL);
2547 ui_document_show_hide(doc);
2548 break;
2549 case GEANY_KEYS_DOCUMENT_LINEWRAP:
2550 on_line_wrapping1_toggled(NULL, NULL);
2551 ui_document_show_hide(doc);
2552 break;
2553 case GEANY_KEYS_DOCUMENT_CLONE:
2554 document_clone(doc);
2555 break;
2556 case GEANY_KEYS_DOCUMENT_RELOADTAGLIST:
2557 document_update_tags(doc);
2558 break;
2559 case GEANY_KEYS_DOCUMENT_FOLDALL:
2560 editor_fold_all(doc->editor);
2561 break;
2562 case GEANY_KEYS_DOCUMENT_UNFOLDALL:
2563 editor_unfold_all(doc->editor);
2564 break;
2565 case GEANY_KEYS_DOCUMENT_TOGGLEFOLD:
2566 if (editor_prefs.folding)
2568 gint line = sci_get_current_line(doc->editor->sci);
2569 editor_toggle_fold(doc->editor, line, 0);
2571 break;
2572 case GEANY_KEYS_DOCUMENT_REMOVE_MARKERS:
2573 on_remove_markers1_activate(NULL, NULL);
2574 break;
2575 case GEANY_KEYS_DOCUMENT_REMOVE_ERROR_INDICATORS:
2576 on_menu_remove_indicators1_activate(NULL, NULL);
2577 break;
2578 case GEANY_KEYS_DOCUMENT_REMOVE_MARKERS_INDICATORS:
2579 on_remove_markers1_activate(NULL, NULL);
2580 on_menu_remove_indicators1_activate(NULL, NULL);
2581 break;
2582 case GEANY_KEYS_DOCUMENT_STRIPTRAILINGSPACES:
2583 editor_strip_trailing_spaces(doc->editor, FALSE);
2584 break;
2586 return TRUE;
2590 static void insert_line_after(GeanyEditor *editor)
2592 ScintillaObject *sci = editor->sci;
2594 sci_send_command(sci, SCI_LINEEND);
2595 sci_send_command(sci, SCI_NEWLINE);
2599 static void insert_line_before(GeanyEditor *editor)
2601 ScintillaObject *sci = editor->sci;
2602 gint line = sci_get_current_line(sci);
2603 gint indentpos = sci_get_line_indent_position(sci, line);
2605 sci_set_current_position(sci, indentpos, TRUE);
2606 sci_send_command(sci, SCI_NEWLINE);
2607 sci_send_command(sci, SCI_LINEUP);
2611 /* common function for insert keybindings, only valid when scintilla has focus. */
2612 static gboolean cb_func_insert_action(guint key_id)
2614 GeanyDocument *doc = document_get_current();
2615 GtkWidget *focusw = gtk_window_get_focus(GTK_WINDOW(main_widgets.window));
2617 /* keybindings only valid when scintilla widget has focus */
2618 if (doc == NULL || focusw != GTK_WIDGET(doc->editor->sci))
2619 return TRUE;
2621 switch (key_id)
2623 case GEANY_KEYS_INSERT_ALTWHITESPACE:
2624 editor_insert_alternative_whitespace(doc->editor);
2625 break;
2626 case GEANY_KEYS_INSERT_DATE:
2627 gtk_menu_item_activate(GTK_MENU_ITEM(
2628 ui_lookup_widget(main_widgets.window, "insert_date_custom1")));
2629 break;
2630 case GEANY_KEYS_INSERT_LINEAFTER:
2631 insert_line_after(doc->editor);
2632 break;
2633 case GEANY_KEYS_INSERT_LINEBEFORE:
2634 insert_line_before(doc->editor);
2635 break;
2637 return TRUE;
2641 /* update key combination */
2642 void keybindings_update_combo(GeanyKeyBinding *kb, guint key, GdkModifierType mods)
2644 GtkWidget *widget = kb->menu_item;
2646 if (widget && kb->key)
2647 gtk_widget_remove_accelerator(widget, kb_accel_group, kb->key, kb->mods);
2649 kb->key = key;
2650 kb->mods = mods;
2652 if (widget && kb->key)
2653 gtk_widget_add_accelerator(widget, "activate", kb_accel_group,
2654 kb->key, kb->mods, GTK_ACCEL_VISIBLE);
2658 /* used for plugins, can be called repeatedly. */
2659 GeanyKeyGroup *keybindings_set_group(GeanyKeyGroup *group, const gchar *section_name,
2660 const gchar *label, gsize count, GeanyKeyGroupCallback callback)
2662 g_return_val_if_fail(section_name, NULL);
2663 g_return_val_if_fail(count, NULL);
2665 /* prevent conflict with core bindings */
2666 g_return_val_if_fail(!g_str_equal(section_name, keybindings_keyfile_group_name), NULL);
2668 if (!group)
2670 group = g_new0(GeanyKeyGroup, 1);
2671 add_kb_group(group, section_name, label, callback, TRUE);
2673 /* Calls free_key_binding() for individual entries for plugins - has to be
2674 * called before g_free(group->plugin_keys) */
2675 g_ptr_array_set_size(group->key_items, 0);
2676 g_free(group->plugin_keys);
2677 group->plugin_keys = g_new0(GeanyKeyBinding, count);
2678 group->plugin_key_count = count;
2679 return group;
2683 void keybindings_free_group(GeanyKeyGroup *group)
2685 g_ptr_array_remove_fast(keybinding_groups, group);