2 * tools.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.
22 * Miscellaneous code for the built-in Tools menu items, and custom command code.
23 * For Plugins code see plugins.c.
33 #include "keybindings.h"
34 #include "sciwrappers.h"
41 #include "gtkcompat.h"
59 /* custom commands code*/
64 GtkTreeViewColumn
*edit_column
;
66 GtkTreeSelection
*selection
;
67 GtkWidget
*button_add
;
68 GtkWidget
*button_remove
;
70 GtkWidget
*button_down
;
74 /* update STATUS and TOOLTIP columns according to cmd */
75 static void cc_dialog_update_row_status(GtkListStore
*store
, GtkTreeIter
*iter
, const gchar
*cmd
)
78 const gchar
*stock_id
= GTK_STOCK_NO
;
79 gchar
*tooltip
= NULL
;
81 if (EMPTY(cmd
) || spawn_check_command(cmd
, TRUE
, &err
))
82 stock_id
= GTK_STOCK_YES
;
85 tooltip
= g_strdup_printf(_("Invalid command: %s"), err
->message
);
89 gtk_list_store_set(store
, iter
, CC_COLUMN_STATUS
, stock_id
, CC_COLUMN_TOOLTIP
, tooltip
, -1);
94 /* adds a new row for custom command @p idx, or an new empty one if < 0 */
95 static void cc_dialog_add_command(struct cc_dialog
*cc
, gint idx
, gboolean start_editing
)
98 const gchar
*cmd
= NULL
;
99 const gchar
*label
= NULL
;
100 guint id
= cc
->count
;
104 cmd
= ui_prefs
.custom_commands
[idx
];
105 label
= ui_prefs
.custom_commands_labels
[idx
];
109 gtk_list_store_append(cc
->store
, &iter
);
110 gtk_list_store_set(cc
->store
, &iter
, CC_COLUMN_ID
, id
, CC_COLUMN_CMD
, cmd
, CC_COLUMN_LABEL
, label
, -1);
111 cc_dialog_update_row_status(cc
->store
, &iter
, cmd
);
117 gtk_widget_grab_focus(cc
->view
);
118 path
= gtk_tree_model_get_path(GTK_TREE_MODEL(cc
->store
), &iter
);
119 gtk_tree_view_set_cursor(GTK_TREE_VIEW(cc
->view
), path
, cc
->edit_column
, TRUE
);
120 gtk_tree_path_free(path
);
125 static void cc_on_dialog_add_clicked(GtkButton
*button
, struct cc_dialog
*cc
)
127 cc_dialog_add_command(cc
, -1, TRUE
);
131 static void scroll_to_cursor(GtkTreeView
*view
)
134 GtkTreeViewColumn
*column
;
136 gtk_tree_view_get_cursor(view
, &path
, &column
);
139 gtk_tree_view_scroll_to_cell(view
, path
, column
, FALSE
, 1.0, 1.0);
140 gtk_tree_path_free(path
);
144 static void cc_on_dialog_remove_clicked(GtkButton
*button
, struct cc_dialog
*cc
)
148 if (gtk_tree_selection_get_selected(cc
->selection
, NULL
, &iter
))
150 gtk_list_store_remove(cc
->store
, &iter
);
151 scroll_to_cursor(GTK_TREE_VIEW(cc
->view
));
156 static void cc_on_dialog_move_up_clicked(GtkButton
*button
, struct cc_dialog
*cc
)
160 if (gtk_tree_selection_get_selected(cc
->selection
, NULL
, &iter
))
165 path
= gtk_tree_model_get_path(GTK_TREE_MODEL(cc
->store
), &iter
);
166 if (gtk_tree_path_prev(path
) &&
167 gtk_tree_model_get_iter(GTK_TREE_MODEL(cc
->store
), &prev
, path
))
169 gtk_list_store_move_before(cc
->store
, &iter
, &prev
);
170 scroll_to_cursor(GTK_TREE_VIEW(cc
->view
));
172 gtk_tree_path_free(path
);
177 static void cc_on_dialog_move_down_clicked(GtkButton
*button
, struct cc_dialog
*cc
)
181 if (gtk_tree_selection_get_selected(cc
->selection
, NULL
, &iter
))
183 GtkTreeIter next
= iter
;
185 if (gtk_tree_model_iter_next(GTK_TREE_MODEL(cc
->store
), &next
))
187 gtk_list_store_move_after(cc
->store
, &iter
, &next
);
188 scroll_to_cursor(GTK_TREE_VIEW(cc
->view
));
194 /* Executes command (which should include all necessary command line args) and passes the current
195 * selection through the standard input of command. The whole output of command replaces the
196 * current selection. */
197 void tools_execute_custom_command(GeanyDocument
*doc
, const gchar
*command
)
199 GError
*error
= NULL
;
201 SpawnWriteData input
;
206 g_return_if_fail(doc
!= NULL
&& command
!= NULL
);
208 if (! sci_has_selection(doc
->editor
->sci
))
209 editor_select_lines(doc
->editor
, FALSE
);
211 sel
= sci_get_selection_contents(doc
->editor
->sci
);
213 input
.size
= strlen(sel
);
214 output
= g_string_sized_new(256);
215 errors
= g_string_new(NULL
);
216 ui_set_statusbar(TRUE
, _("Passing data and executing custom command: %s"), command
);
218 if (spawn_sync(NULL
, command
, NULL
, NULL
, &input
, output
, errors
, &status
, &error
))
222 g_warning("%s: %s\n", command
, errors
->str
);
223 ui_set_statusbar(TRUE
,
224 _("The executed custom command returned an error. "
225 "Your selection was not changed. Error message: %s"),
228 else if (!SPAWN_WIFEXITED(status
) || SPAWN_WEXITSTATUS(status
) != EXIT_SUCCESS
)
230 /* TODO maybe include the exit code in the error message */
231 ui_set_statusbar(TRUE
,
232 _("The executed custom command exited with an unsuccessful exit code."));
235 { /* Command completed successfully */
236 sci_replace_sel(doc
->editor
->sci
, output
->str
);
241 ui_set_statusbar(TRUE
, _("Cannot execute custom command \"%s\": %s. "
242 "Check the path setting in Custom Commands."), command
, error
->message
);
246 g_string_free(output
, TRUE
);
247 g_string_free(errors
, TRUE
);
252 static void cc_dialog_on_command_edited(GtkCellRendererText
*renderer
, gchar
*path
, gchar
*text
,
253 struct cc_dialog
*cc
)
257 gtk_tree_model_get_iter_from_string(GTK_TREE_MODEL(cc
->store
), &iter
, path
);
258 gtk_list_store_set(cc
->store
, &iter
, CC_COLUMN_CMD
, text
, -1);
259 cc_dialog_update_row_status(cc
->store
, &iter
, text
);
263 static void cc_dialog_on_label_edited(GtkCellRendererText
*renderer
, gchar
*path
, gchar
*text
,
264 struct cc_dialog
*cc
)
268 gtk_tree_model_get_iter_from_string(GTK_TREE_MODEL(cc
->store
), &iter
, path
);
269 gtk_list_store_set(cc
->store
, &iter
, CC_COLUMN_LABEL
, text
, -1);
273 /* re-compute IDs to reflect the current store state */
274 static void cc_dialog_update_ids(struct cc_dialog
*cc
)
279 if (! gtk_tree_model_get_iter_first(GTK_TREE_MODEL(cc
->store
), &iter
))
284 gtk_list_store_set(cc
->store
, &iter
, CC_COLUMN_ID
, cc
->count
, -1);
287 while (gtk_tree_model_iter_next(GTK_TREE_MODEL(cc
->store
), &iter
));
291 /* update sensitiveness of the buttons according to the selection */
292 static void cc_dialog_update_sensitive(struct cc_dialog
*cc
)
295 gboolean has_selection
= FALSE
;
296 gboolean first_selected
= FALSE
;
297 gboolean last_selected
= FALSE
;
299 if ((has_selection
= gtk_tree_selection_get_selected(cc
->selection
, NULL
, &iter
)))
304 path
= gtk_tree_model_get_path(GTK_TREE_MODEL(cc
->store
), &iter
);
305 copy
= gtk_tree_path_copy(path
);
306 first_selected
= ! gtk_tree_path_prev(copy
);
307 gtk_tree_path_free(copy
);
308 gtk_tree_path_next(path
);
309 last_selected
= ! gtk_tree_model_get_iter(GTK_TREE_MODEL(cc
->store
), &iter
, path
);
310 gtk_tree_path_free(path
);
313 gtk_widget_set_sensitive(cc
->button_remove
, has_selection
);
314 gtk_widget_set_sensitive(cc
->button_up
, has_selection
&& ! first_selected
);
315 gtk_widget_set_sensitive(cc
->button_down
, has_selection
&& ! last_selected
);
319 static void cc_dialog_on_tree_selection_changed(GtkTreeSelection
*selection
, struct cc_dialog
*cc
)
321 cc_dialog_update_sensitive(cc
);
325 static void cc_dialog_on_row_inserted(GtkTreeModel
*model
, GtkTreePath
*path
, GtkTreeIter
*iter
,
326 struct cc_dialog
*cc
)
328 cc_dialog_update_ids(cc
);
329 cc_dialog_update_sensitive(cc
);
333 static void cc_dialog_on_row_deleted(GtkTreeModel
*model
, GtkTreePath
*path
, struct cc_dialog
*cc
)
335 cc_dialog_update_ids(cc
);
336 cc_dialog_update_sensitive(cc
);
340 static void cc_dialog_on_rows_reordered(GtkTreeModel
*model
, GtkTreePath
*path
, GtkTreeIter
*iter
,
341 gpointer new_order
, struct cc_dialog
*cc
)
343 cc_dialog_update_ids(cc
);
344 cc_dialog_update_sensitive(cc
);
348 static void cc_show_dialog_custom_commands(void)
350 GtkWidget
*dialog
, *label
, *vbox
, *scroll
, *buttonbox
;
351 GtkCellRenderer
*renderer
;
352 GtkTreeViewColumn
*column
;
356 dialog
= gtk_dialog_new_with_buttons(_("Set Custom Commands"), GTK_WINDOW(main_widgets
.window
),
357 GTK_DIALOG_DESTROY_WITH_PARENT
, GTK_STOCK_CANCEL
, GTK_RESPONSE_CANCEL
,
358 GTK_STOCK_OK
, GTK_RESPONSE_ACCEPT
, NULL
);
359 gtk_window_set_default_size(GTK_WINDOW(dialog
), 300, 300); /* give a reasonable minimal default size */
360 vbox
= ui_dialog_vbox_new(GTK_DIALOG(dialog
));
361 gtk_box_set_spacing(GTK_BOX(vbox
), 6);
362 gtk_widget_set_name(dialog
, "GeanyDialog");
364 label
= gtk_label_new(_("You can send the current selection to any of these commands and the output of the command replaces the current selection."));
365 gtk_label_set_line_wrap(GTK_LABEL(label
), TRUE
);
366 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0.5);
367 gtk_box_pack_start(GTK_BOX(vbox
), label
, FALSE
, FALSE
, 0);
370 cc
.store
= gtk_list_store_new(CC_COLUMN_COUNT
, G_TYPE_UINT
, G_TYPE_STRING
, G_TYPE_STRING
,
371 G_TYPE_STRING
, G_TYPE_STRING
);
372 cc
.view
= gtk_tree_view_new_with_model(GTK_TREE_MODEL(cc
.store
));
373 ui_tree_view_set_tooltip_text_column(GTK_TREE_VIEW(cc
.view
), CC_COLUMN_TOOLTIP
);
374 gtk_tree_view_set_reorderable(GTK_TREE_VIEW(cc
.view
), TRUE
);
375 cc
.selection
= gtk_tree_view_get_selection(GTK_TREE_VIEW(cc
.view
));
377 renderer
= gtk_cell_renderer_text_new();
378 column
= gtk_tree_view_column_new_with_attributes(_("ID"), renderer
, "text", CC_COLUMN_ID
, NULL
);
379 gtk_tree_view_append_column(GTK_TREE_VIEW(cc
.view
), column
);
380 /* command column, holding status and command display */
381 column
= g_object_new(GTK_TYPE_TREE_VIEW_COLUMN
, "title", _("Command"), "expand", TRUE
, "resizable", TRUE
, NULL
);
382 renderer
= gtk_cell_renderer_pixbuf_new();
383 gtk_tree_view_column_pack_start(column
, renderer
, FALSE
);
384 gtk_tree_view_column_set_attributes(column
, renderer
, "stock-id", CC_COLUMN_STATUS
, NULL
);
385 renderer
= gtk_cell_renderer_text_new();
386 g_object_set(renderer
, "editable", TRUE
, "ellipsize", PANGO_ELLIPSIZE_END
, NULL
);
387 g_signal_connect(renderer
, "edited", G_CALLBACK(cc_dialog_on_command_edited
), &cc
);
388 gtk_tree_view_column_pack_start(column
, renderer
, TRUE
);
389 gtk_tree_view_column_set_attributes(column
, renderer
, "text", CC_COLUMN_CMD
, NULL
);
390 cc
.edit_column
= column
;
391 gtk_tree_view_append_column(GTK_TREE_VIEW(cc
.view
), column
);
393 renderer
= gtk_cell_renderer_text_new();
394 g_object_set(renderer
, "editable", TRUE
, "ellipsize", PANGO_ELLIPSIZE_END
, NULL
);
395 g_signal_connect(renderer
, "edited", G_CALLBACK(cc_dialog_on_label_edited
), &cc
);
396 column
= gtk_tree_view_column_new_with_attributes(_("Label"), renderer
, "text", CC_COLUMN_LABEL
, NULL
);
397 g_object_set(column
, "expand", TRUE
, "resizable", TRUE
, NULL
);
398 gtk_tree_view_append_column(GTK_TREE_VIEW(cc
.view
), column
);
400 scroll
= gtk_scrolled_window_new(NULL
, NULL
);
401 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll
), GTK_POLICY_AUTOMATIC
,
402 GTK_POLICY_AUTOMATIC
);
403 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scroll
), GTK_SHADOW_IN
);
404 gtk_container_add(GTK_CONTAINER(scroll
), cc
.view
);
405 gtk_box_pack_start(GTK_BOX(vbox
), scroll
, TRUE
, TRUE
, 0);
407 if (ui_prefs
.custom_commands
!= NULL
)
410 guint len
= g_strv_length(ui_prefs
.custom_commands
);
412 for (i
= 0; i
< len
; i
++)
414 if (EMPTY(ui_prefs
.custom_commands
[i
]))
415 continue; /* skip empty fields */
417 cc_dialog_add_command(&cc
, i
, FALSE
);
420 /* focus the first row if any */
421 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(cc
.store
), &iter
))
423 GtkTreePath
*path
= gtk_tree_model_get_path(GTK_TREE_MODEL(cc
.store
), &iter
);
425 gtk_tree_view_set_cursor(GTK_TREE_VIEW(cc
.view
), path
, cc
.edit_column
, FALSE
);
426 gtk_tree_path_free(path
);
430 buttonbox
= gtk_hbutton_box_new();
431 gtk_box_set_spacing(GTK_BOX(buttonbox
), 6);
432 gtk_box_pack_start(GTK_BOX(vbox
), buttonbox
, FALSE
, FALSE
, 0);
433 cc
.button_add
= gtk_button_new_from_stock(GTK_STOCK_ADD
);
434 g_signal_connect(cc
.button_add
, "clicked", G_CALLBACK(cc_on_dialog_add_clicked
), &cc
);
435 gtk_container_add(GTK_CONTAINER(buttonbox
), cc
.button_add
);
436 cc
.button_remove
= gtk_button_new_from_stock(GTK_STOCK_REMOVE
);
437 g_signal_connect(cc
.button_remove
, "clicked", G_CALLBACK(cc_on_dialog_remove_clicked
), &cc
);
438 gtk_container_add(GTK_CONTAINER(buttonbox
), cc
.button_remove
);
439 cc
.button_up
= gtk_button_new_from_stock(GTK_STOCK_GO_UP
);
440 g_signal_connect(cc
.button_up
, "clicked", G_CALLBACK(cc_on_dialog_move_up_clicked
), &cc
);
441 gtk_container_add(GTK_CONTAINER(buttonbox
), cc
.button_up
);
442 cc
.button_down
= gtk_button_new_from_stock(GTK_STOCK_GO_DOWN
);
443 g_signal_connect(cc
.button_down
, "clicked", G_CALLBACK(cc_on_dialog_move_down_clicked
), &cc
);
444 gtk_container_add(GTK_CONTAINER(buttonbox
), cc
.button_down
);
446 cc_dialog_update_sensitive(&cc
);
448 /* only connect the selection signal when all other cc_dialog fields are set */
449 g_signal_connect(cc
.selection
, "changed", G_CALLBACK(cc_dialog_on_tree_selection_changed
), &cc
);
450 g_signal_connect(cc
.store
, "row-inserted", G_CALLBACK(cc_dialog_on_row_inserted
), &cc
);
451 g_signal_connect(cc
.store
, "row-deleted", G_CALLBACK(cc_dialog_on_row_deleted
), &cc
);
452 g_signal_connect(cc
.store
, "rows-reordered", G_CALLBACK(cc_dialog_on_rows_reordered
), &cc
);
454 gtk_widget_show_all(vbox
);
456 if (gtk_dialog_run(GTK_DIALOG(dialog
)) == GTK_RESPONSE_ACCEPT
)
458 GSList
*cmd_list
= NULL
;
459 GSList
*lbl_list
= NULL
;
461 gchar
**commands
= NULL
;
462 gchar
**labels
= NULL
;
465 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(cc
.store
), &iter
))
472 gtk_tree_model_get(GTK_TREE_MODEL(cc
.store
), &iter
, CC_COLUMN_CMD
, &cmd
, CC_COLUMN_LABEL
, &lbl
, -1);
475 cmd_list
= g_slist_prepend(cmd_list
, cmd
);
476 lbl_list
= g_slist_prepend(lbl_list
, lbl
);
485 while (gtk_tree_model_iter_next(GTK_TREE_MODEL(cc
.store
), &iter
));
487 cmd_list
= g_slist_reverse(cmd_list
);
488 lbl_list
= g_slist_reverse(lbl_list
);
489 /* create a new null-terminated array but only if there is any commands defined */
493 GSList
*cmd_node
, *lbl_node
;
495 commands
= g_new(gchar
*, len
+ 1);
496 labels
= g_new(gchar
*, len
+ 1);
497 /* walk commands and labels lists */
498 for (cmd_node
= cmd_list
, lbl_node
= lbl_list
; cmd_node
!= NULL
; cmd_node
= cmd_node
->next
, lbl_node
= lbl_node
->next
)
500 commands
[j
] = (gchar
*) cmd_node
->data
;
501 labels
[j
] = (gchar
*) lbl_node
->data
;
504 /* null-terminate the arrays */
508 /* set the new arrays */
509 g_strfreev(ui_prefs
.custom_commands
);
510 ui_prefs
.custom_commands
= commands
;
511 g_strfreev(ui_prefs
.custom_commands_labels
);
512 ui_prefs
.custom_commands_labels
= labels
;
513 /* rebuild the menu items */
514 tools_create_insert_custom_command_menu_items();
516 g_slist_free(cmd_list
);
517 g_slist_free(lbl_list
);
519 gtk_widget_destroy(dialog
);
523 static void cc_on_custom_command_activate(GtkMenuItem
*menuitem
, gpointer user_data
)
525 GeanyDocument
*doc
= document_get_current();
528 g_return_if_fail(DOC_VALID(doc
));
530 command_idx
= GPOINTER_TO_INT(user_data
);
532 if (ui_prefs
.custom_commands
== NULL
||
533 command_idx
< 0 || command_idx
> (gint
) g_strv_length(ui_prefs
.custom_commands
))
535 cc_show_dialog_custom_commands();
539 /* send it through the command and when the command returned the output the current selection
540 * will be replaced */
541 tools_execute_custom_command(doc
, ui_prefs
.custom_commands
[command_idx
]);
545 static void cc_insert_custom_command_items(GtkMenu
*me
, const gchar
*label
, const gchar
*tooltip
, gint idx
)
552 case 0: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD1
; break;
553 case 1: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD2
; break;
554 case 2: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD3
; break;
555 case 3: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD4
; break;
556 case 4: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD5
; break;
557 case 5: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD6
; break;
558 case 6: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD7
; break;
559 case 7: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD8
; break;
560 case 8: key_idx
= GEANY_KEYS_FORMAT_SENDTOCMD9
; break;
563 item
= gtk_menu_item_new_with_label(label
);
564 gtk_widget_set_tooltip_text(item
, tooltip
);
567 GeanyKeyBinding
*kb
= keybindings_lookup_item(GEANY_KEY_GROUP_FORMAT
, key_idx
);
571 gtk_widget_add_accelerator(item
, "activate", gtk_accel_group_new(),
572 kb
->key
, kb
->mods
, GTK_ACCEL_VISIBLE
);
575 gtk_container_add(GTK_CONTAINER(me
), item
);
576 gtk_widget_show(item
);
577 g_signal_connect(item
, "activate", G_CALLBACK(cc_on_custom_command_activate
),
578 GINT_TO_POINTER(idx
));
582 void tools_create_insert_custom_command_menu_items(void)
584 GtkMenu
*menu_edit
= GTK_MENU(ui_lookup_widget(main_widgets
.window
, "send_selection_to2_menu"));
586 GList
*me_children
, *node
;
588 /* first clean the menus to be able to rebuild them */
589 me_children
= gtk_container_get_children(GTK_CONTAINER(menu_edit
));
590 foreach_list(node
, me_children
)
591 gtk_widget_destroy(GTK_WIDGET(node
->data
));
592 g_list_free(me_children
);
594 if (ui_prefs
.custom_commands
== NULL
|| g_strv_length(ui_prefs
.custom_commands
) == 0)
596 item
= gtk_menu_item_new_with_label(_("No custom commands defined."));
597 gtk_container_add(GTK_CONTAINER(menu_edit
), item
);
598 gtk_widget_set_sensitive(item
, FALSE
);
599 gtk_widget_show(item
);
605 len
= g_strv_length(ui_prefs
.custom_commands
);
606 for (i
= 0; i
< len
; i
++)
608 const gchar
*label
= ui_prefs
.custom_commands_labels
[i
];
611 label
= ui_prefs
.custom_commands
[i
];
612 if (!EMPTY(label
)) /* skip empty items */
614 cc_insert_custom_command_items(menu_edit
, label
, ui_prefs
.custom_commands
[i
], idx
);
620 /* separator and Set menu item */
621 item
= gtk_separator_menu_item_new();
622 gtk_container_add(GTK_CONTAINER(menu_edit
), item
);
623 gtk_widget_show(item
);
625 cc_insert_custom_command_items(menu_edit
, _("Set Custom Commands"), NULL
, -1);
629 /* (stolen from bluefish, thanks)
630 * Returns number of characters, lines and words in the supplied gchar*.
631 * Handles UTF-8 correctly. Input must be properly encoded UTF-8.
632 * Words are defined as any characters grouped, separated with spaces. */
633 static void word_count(gchar
*text
, guint
*chars
, guint
*lines
, guint
*words
)
639 return; /* politely refuse to operate on NULL */
641 *chars
= *words
= *lines
= 0;
642 while (*text
!= '\0')
664 utext
= g_utf8_get_char_validated(text
, 2); /* This might be an utf-8 char */
665 if (g_unichar_isspace(utext
)) /* Unicode encoded space? */
666 goto mb_word_separator
;
667 if (g_unichar_isgraph(utext
)) /* Is this something printable? */
671 /* Even if the current char is 2 bytes, this will iterate correctly. */
672 text
= g_utf8_next_char(text
);
675 /* Capture last word, if there's no whitespace at the end of the file. */
678 /* We start counting line numbers from 1 */
684 void tools_word_count(void)
686 GtkWidget
*dialog
, *label
, *vbox
, *table
;
688 guint chars
= 0, lines
= 0, words
= 0;
692 doc
= document_get_current();
693 g_return_if_fail(doc
!= NULL
);
695 dialog
= gtk_dialog_new_with_buttons(_("Word Count"), GTK_WINDOW(main_widgets
.window
),
696 GTK_DIALOG_DESTROY_WITH_PARENT
,
697 GTK_STOCK_CLOSE
, GTK_RESPONSE_CANCEL
, NULL
);
698 vbox
= ui_dialog_vbox_new(GTK_DIALOG(dialog
));
699 gtk_widget_set_name(dialog
, "GeanyDialog");
701 if (sci_has_selection(doc
->editor
->sci
))
703 text
= sci_get_selection_contents(doc
->editor
->sci
);
704 range
= _("selection");
708 text
= sci_get_contents(doc
->editor
->sci
, -1);
709 range
= _("whole document");
711 word_count(text
, &chars
, &lines
, &words
);
714 table
= gtk_table_new(4, 2, FALSE
);
715 gtk_table_set_row_spacings(GTK_TABLE(table
), 5);
716 gtk_table_set_col_spacings(GTK_TABLE(table
), 10);
718 label
= gtk_label_new(_("Range:"));
719 gtk_table_attach(GTK_TABLE(table
), label
, 0, 1, 0, 1,
720 (GtkAttachOptions
) (GTK_FILL
),
721 (GtkAttachOptions
) (0), 0, 0);
722 gtk_misc_set_alignment(GTK_MISC(label
), 1, 0);
724 label
= gtk_label_new(range
);
725 gtk_table_attach(GTK_TABLE(table
), label
, 1, 2, 0, 1,
726 (GtkAttachOptions
) (GTK_FILL
),
727 (GtkAttachOptions
) (0), 20, 0);
728 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0);
730 label
= gtk_label_new(_("Lines:"));
731 gtk_table_attach(GTK_TABLE(table
), label
, 0, 1, 1, 2,
732 (GtkAttachOptions
) (GTK_FILL
),
733 (GtkAttachOptions
) (0), 0, 0);
734 gtk_misc_set_alignment(GTK_MISC(label
), 1, 0);
736 text
= g_strdup_printf("%d", lines
);
737 label
= gtk_label_new(text
);
738 gtk_table_attach(GTK_TABLE(table
), label
, 1, 2, 1, 2,
739 (GtkAttachOptions
) (GTK_FILL
),
740 (GtkAttachOptions
) (0), 20, 0);
741 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0);
744 label
= gtk_label_new(_("Words:"));
745 gtk_table_attach(GTK_TABLE(table
), label
, 0, 1, 2, 3,
746 (GtkAttachOptions
) (GTK_FILL
),
747 (GtkAttachOptions
) (0), 0, 0);
748 gtk_misc_set_alignment(GTK_MISC(label
), 1, 0);
750 text
= g_strdup_printf("%d", words
);
751 label
= gtk_label_new(text
);
752 gtk_table_attach(GTK_TABLE(table
), label
, 1, 2, 2, 3,
753 (GtkAttachOptions
) (GTK_FILL
),
754 (GtkAttachOptions
) (0), 20, 0);
755 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0);
758 label
= gtk_label_new(_("Characters:"));
759 gtk_table_attach(GTK_TABLE(table
), label
, 0, 1, 3, 4,
760 (GtkAttachOptions
) (GTK_FILL
),
761 (GtkAttachOptions
) (0), 0, 0);
762 gtk_misc_set_alignment(GTK_MISC(label
), 1, 0);
764 text
= g_strdup_printf("%d", chars
);
765 label
= gtk_label_new(text
);
766 gtk_table_attach(GTK_TABLE(table
), label
, 1, 2, 3, 4,
767 (GtkAttachOptions
) (GTK_FILL
),
768 (GtkAttachOptions
) (0), 20, 0);
769 gtk_misc_set_alignment(GTK_MISC(label
), 0, 0);
772 gtk_container_add(GTK_CONTAINER(vbox
), table
);
774 g_signal_connect(dialog
, "response", G_CALLBACK(gtk_widget_destroy
), dialog
);
775 g_signal_connect(dialog
, "delete-event", G_CALLBACK(gtk_widget_destroy
), dialog
);
777 gtk_widget_show_all(dialog
);
782 * color dialog callbacks
784 static void on_color_dialog_response(GtkDialog
*dialog
, gint response
, gpointer user_data
)
788 case GTK_RESPONSE_OK
:
789 gtk_widget_hide(ui_widgets
.open_colorsel
);
791 case GTK_RESPONSE_APPLY
:
794 GeanyDocument
*doc
= document_get_current();
798 g_return_if_fail(doc
!= NULL
);
800 colorsel
= gtk_color_selection_dialog_get_color_selection(GTK_COLOR_SELECTION_DIALOG(ui_widgets
.open_colorsel
));
801 gtk_color_selection_get_current_color(GTK_COLOR_SELECTION(colorsel
), &color
);
803 hex
= utils_get_hex_from_color(&color
);
804 editor_insert_color(doc
->editor
, hex
);
810 gtk_widget_hide(ui_widgets
.open_colorsel
);
815 static void on_color_selection_change_palette_with_screen(GdkScreen
*screen
, const GdkColor
*colors
, gint n_colors
)
817 GtkSettings
*settings
;
819 /* Get the updated palette */
820 g_free(ui_prefs
.color_picker_palette
);
821 ui_prefs
.color_picker_palette
= gtk_color_selection_palette_to_string(colors
, n_colors
);
823 /* Update the gtk-color-palette setting so all GtkColorSelection widgets will be modified */
824 settings
= gtk_settings_get_for_screen(screen
);
825 g_object_set(G_OBJECT(settings
), "gtk-color-palette", ui_prefs
.color_picker_palette
, NULL
);
829 /* This shows the color selection dialog to choose a color. */
830 void tools_color_chooser(const gchar
*color
)
836 if (interface_prefs
.use_native_windows_dialogs
)
838 win32_show_color_dialog(color
);
843 if (ui_widgets
.open_colorsel
== NULL
)
845 ui_widgets
.open_colorsel
= gtk_color_selection_dialog_new(_("Color Chooser"));
846 gtk_dialog_add_button(GTK_DIALOG(ui_widgets
.open_colorsel
), GTK_STOCK_APPLY
, GTK_RESPONSE_APPLY
);
847 ui_dialog_set_primary_button_order(GTK_DIALOG(ui_widgets
.open_colorsel
),
848 GTK_RESPONSE_APPLY
, GTK_RESPONSE_CANCEL
, GTK_RESPONSE_OK
, -1);
849 gtk_widget_set_name(ui_widgets
.open_colorsel
, "GeanyDialog");
850 gtk_window_set_transient_for(GTK_WINDOW(ui_widgets
.open_colorsel
), GTK_WINDOW(main_widgets
.window
));
851 colorsel
= gtk_color_selection_dialog_get_color_selection(GTK_COLOR_SELECTION_DIALOG(ui_widgets
.open_colorsel
));
852 gtk_color_selection_set_has_palette(GTK_COLOR_SELECTION(colorsel
), TRUE
);
853 gtk_color_selection_set_change_palette_with_screen_hook(on_color_selection_change_palette_with_screen
);
855 g_signal_connect(ui_widgets
.open_colorsel
, "response",
856 G_CALLBACK(on_color_dialog_response
), NULL
);
857 g_signal_connect(ui_widgets
.open_colorsel
, "delete-event",
858 G_CALLBACK(gtk_widget_hide_on_delete
), NULL
);
861 colorsel
= gtk_color_selection_dialog_get_color_selection(GTK_COLOR_SELECTION_DIALOG(ui_widgets
.open_colorsel
));
862 /* if color is non-NULL set it in the dialog as preselected color */
863 if (color
!= NULL
&& utils_parse_color(color
, &gc
))
865 gtk_color_selection_set_current_color(GTK_COLOR_SELECTION(colorsel
), &gc
);
866 gtk_color_selection_set_previous_color(GTK_COLOR_SELECTION(colorsel
), &gc
);
869 /* We make sure the dialog is visible. */
870 gtk_window_present(GTK_WINDOW(ui_widgets
.open_colorsel
));