/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* caja-clipboard.c * * Caja Clipboard support. For now, routines to support component cut * and paste. * * Copyright (C) 1999, 2000 Free Software Foundaton * Copyright (C) 2000, 2001 Eazel, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. * * Authors: Rebecca Schulman <rebecka@eazel.com>, * Darin Adler <darin@bentspoon.com> */ #include <config.h> #include "caja-clipboard.h" #include "caja-file-utilities.h" #include <glib/gi18n.h> #include <gtk/gtk.h> #include <string.h> typedef struct _TargetCallbackData TargetCallbackData; typedef void (* SelectAllCallback) (gpointer target); typedef void (* ConnectCallbacksFunc) (GObject *object, TargetCallbackData *target_data); static void selection_changed_callback (GtkWidget *widget, gpointer callback_data); static void owner_change_callback (GtkClipboard *clipboard, GdkEventOwnerChange *event, gpointer callback_data); struct _TargetCallbackData { GtkUIManager *ui_manager; GtkActionGroup *action_group; gboolean shares_selection_changes; SelectAllCallback select_all_callback; ConnectCallbacksFunc connect_callbacks; ConnectCallbacksFunc disconnect_callbacks; }; static void cut_callback (gpointer target) { g_assert (target != NULL); g_signal_emit_by_name (target, "cut-clipboard"); } static void copy_callback (gpointer target) { g_assert (target != NULL); g_signal_emit_by_name (target, "copy-clipboard"); } static void paste_callback (gpointer target) { g_assert (target != NULL); g_signal_emit_by_name (target, "paste-clipboard"); } static void editable_select_all_callback (gpointer target) { GtkEditable *editable; editable = GTK_EDITABLE (target); g_assert (editable != NULL); gtk_editable_set_position (editable, -1); gtk_editable_select_region (editable, 0, -1); } static void text_view_select_all_callback (gpointer target) { g_assert (GTK_IS_TEXT_VIEW (target)); g_signal_emit_by_name (target, "select-all", TRUE); } static void action_cut_callback (GtkAction *action, gpointer callback_data) { cut_callback (callback_data); } static void action_copy_callback (GtkAction *action, gpointer callback_data) { copy_callback (callback_data); } static void action_paste_callback (GtkAction *action, gpointer callback_data) { paste_callback (callback_data); } static void action_select_all_callback (GtkAction *action, gpointer callback_data) { TargetCallbackData *target_data; g_assert (callback_data != NULL); target_data = g_object_get_data (callback_data, "Caja:clipboard_target_data"); g_assert (target_data != NULL); target_data->select_all_callback (callback_data); } static void received_clipboard_contents (GtkClipboard *clipboard, GtkSelectionData *selection_data, gpointer data) { GtkActionGroup *action_group; GtkAction *action; action_group = data; action = gtk_action_group_get_action (action_group, "Paste"); if (action != NULL) { gtk_action_set_sensitive (action, gtk_selection_data_targets_include_text (selection_data)); } g_object_unref (action_group); } static void set_paste_sensitive_if_clipboard_contains_data (GtkActionGroup *action_group) { GtkAction *action; if (gdk_display_supports_selection_notification (gdk_display_get_default ())) { gtk_clipboard_request_contents (gtk_clipboard_get (GDK_SELECTION_CLIPBOARD), gdk_atom_intern ("TARGETS", FALSE), received_clipboard_contents, g_object_ref (action_group)); } else { /* If selection notification isn't supported, always activate Paste */ action = gtk_action_group_get_action (action_group, "Paste"); gtk_action_set_sensitive (action, TRUE); } } static void set_clipboard_menu_items_sensitive (GtkActionGroup *action_group) { GtkAction *action; action = gtk_action_group_get_action (action_group, "Cut"); gtk_action_set_sensitive (action, TRUE); action = gtk_action_group_get_action (action_group, "Copy"); gtk_action_set_sensitive (action, TRUE); } static void set_clipboard_menu_items_insensitive (GtkActionGroup *action_group) { GtkAction *action; action = gtk_action_group_get_action (action_group, "Cut"); gtk_action_set_sensitive (action, FALSE); action = gtk_action_group_get_action (action_group, "Copy"); gtk_action_set_sensitive (action, FALSE); } static gboolean clipboard_items_are_merged_in (GtkWidget *widget) { return GPOINTER_TO_INT (g_object_get_data (G_OBJECT (widget), "Caja:clipboard_menu_items_merged")); } static void set_clipboard_items_are_merged_in (GObject *widget_as_object, gboolean merged_in) { g_object_set_data (widget_as_object, "Caja:clipboard_menu_items_merged", GINT_TO_POINTER (merged_in)); } static void editable_connect_callbacks (GObject *object, TargetCallbackData *target_data) { g_signal_connect_after (object, "selection_changed", G_CALLBACK (selection_changed_callback), target_data); selection_changed_callback (GTK_WIDGET (object), target_data); } static void editable_disconnect_callbacks (GObject *object, TargetCallbackData *target_data) { g_signal_handlers_disconnect_matched (object, G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA, 0, 0, NULL, G_CALLBACK (selection_changed_callback), target_data); } static void text_buffer_update_sensitivity (GtkTextBuffer *buffer, TargetCallbackData *target_data) { g_assert (GTK_IS_TEXT_BUFFER (buffer)); g_assert (target_data != NULL); if (gtk_text_buffer_get_selection_bounds (buffer, NULL, NULL)) { set_clipboard_menu_items_sensitive (target_data->action_group); } else { set_clipboard_menu_items_insensitive (target_data->action_group); } } static void text_buffer_delete_range (GtkTextBuffer *buffer, GtkTextIter *iter1, GtkTextIter *iter2, TargetCallbackData *target_data) { text_buffer_update_sensitivity (buffer, target_data); } static void text_buffer_mark_set (GtkTextBuffer *buffer, GtkTextIter *iter, GtkTextMark *mark, TargetCallbackData *target_data) { /* anonymous marks with NULL names refer to cursor moves */ if (gtk_text_mark_get_name (mark) != NULL) { text_buffer_update_sensitivity (buffer, target_data); } } static void text_view_connect_callbacks (GObject *object, TargetCallbackData *target_data) { GtkTextBuffer *buffer; buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (object)); g_assert (buffer); g_signal_connect_after (buffer, "mark-set", G_CALLBACK (text_buffer_mark_set), target_data); g_signal_connect_after (buffer, "delete-range", G_CALLBACK (text_buffer_delete_range), target_data); text_buffer_update_sensitivity (buffer, target_data); } static void text_view_disconnect_callbacks (GObject *object, TargetCallbackData *target_data) { GtkTextBuffer *buffer; buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (object)); g_assert (buffer); g_signal_handlers_disconnect_matched (buffer, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, target_data); } static void merge_in_clipboard_menu_items (GObject *widget_as_object, TargetCallbackData *target_data) { gboolean add_selection_callback; g_assert (target_data != NULL); add_selection_callback = target_data->shares_selection_changes; gtk_ui_manager_insert_action_group (target_data->ui_manager, target_data->action_group, 0); set_paste_sensitive_if_clipboard_contains_data (target_data->action_group); g_signal_connect (gtk_clipboard_get (GDK_SELECTION_CLIPBOARD), "owner_change", G_CALLBACK (owner_change_callback), target_data); if (add_selection_callback) { target_data->connect_callbacks (widget_as_object, target_data); } else { /* If we don't use sensitivity, everything should be on */ set_clipboard_menu_items_sensitive (target_data->action_group); } set_clipboard_items_are_merged_in (widget_as_object, TRUE); } static void merge_out_clipboard_menu_items (GObject *widget_as_object, TargetCallbackData *target_data) { gboolean selection_callback_was_added; g_assert (target_data != NULL); gtk_ui_manager_remove_action_group (target_data->ui_manager, target_data->action_group); g_signal_handlers_disconnect_matched (gtk_clipboard_get (GDK_SELECTION_CLIPBOARD), G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA, 0, 0, NULL, G_CALLBACK (owner_change_callback), target_data); selection_callback_was_added = target_data->shares_selection_changes; if (selection_callback_was_added) { target_data->disconnect_callbacks (widget_as_object, target_data); } set_clipboard_items_are_merged_in (widget_as_object, FALSE); } static gboolean focus_changed_callback (GtkWidget *widget, GdkEventAny *event, gpointer callback_data) { /* Connect the component to the container if the widget has focus. */ if (gtk_widget_has_focus (widget)) { if (!clipboard_items_are_merged_in (widget)) { merge_in_clipboard_menu_items (G_OBJECT (widget), callback_data); } } else { if (clipboard_items_are_merged_in (widget)) { merge_out_clipboard_menu_items (G_OBJECT (widget), callback_data); } } return FALSE; } static void selection_changed_callback (GtkWidget *widget, gpointer callback_data) { TargetCallbackData *target_data; GtkEditable *editable; int start, end; target_data = (TargetCallbackData *) callback_data; g_assert (target_data != NULL); editable = GTK_EDITABLE (widget); g_assert (editable != NULL); if (gtk_editable_get_selection_bounds (editable, &start, &end) && start != end) { set_clipboard_menu_items_sensitive (target_data->action_group); } else { set_clipboard_menu_items_insensitive (target_data->action_group); } } static void owner_change_callback (GtkClipboard *clipboard, GdkEventOwnerChange *event, gpointer callback_data) { TargetCallbackData *target_data; g_assert (callback_data != NULL); target_data = callback_data; set_paste_sensitive_if_clipboard_contains_data (target_data->action_group); } static void target_destroy_callback (GtkWidget *object, gpointer callback_data) { g_assert (callback_data != NULL); if (clipboard_items_are_merged_in (object)) { merge_out_clipboard_menu_items (G_OBJECT (object), callback_data); } } static void target_data_free (TargetCallbackData *target_data) { g_object_unref (target_data->action_group); g_free (target_data); } static const GtkActionEntry clipboard_entries[] = { /* name, icon name */ { "Cut", "edit-cut", /* label, accelerator */ N_("Cu_t"), NULL, /* tooltip */ N_("Cut the selected text to the clipboard"), G_CALLBACK (action_cut_callback) }, /* name, icon name */ { "Copy", "edit-copy", /* label, accelerator */ N_("_Copy"), NULL, /* tooltip */ N_("Copy the selected text to the clipboard"), G_CALLBACK (action_copy_callback) }, /* name, icon name */ { "Paste", "edit-paste", /* label, accelerator */ N_("_Paste"), NULL, /* tooltip */ N_("Paste the text stored on the clipboard"), G_CALLBACK (action_paste_callback) }, /* name, icon name */ { "Select All", NULL, /* label, accelerator */ N_("Select _All"), "<control>A", /* tooltip */ N_("Select all the text in a text field"), G_CALLBACK (action_select_all_callback) }, }; static TargetCallbackData * initialize_clipboard_component_with_callback_data (GtkEditable *target, GtkUIManager *ui_manager, gboolean shares_selection_changes, SelectAllCallback select_all_callback, ConnectCallbacksFunc connect_callbacks, ConnectCallbacksFunc disconnect_callbacks) { GtkActionGroup *action_group; TargetCallbackData *target_data; action_group = gtk_action_group_new ("ClipboardActions"); gtk_action_group_set_translation_domain (action_group, GETTEXT_PACKAGE); gtk_action_group_add_actions (action_group, clipboard_entries, G_N_ELEMENTS (clipboard_entries), target); /* Do the actual connection of the UI to the container at * focus time, and disconnect at both focus and destroy * time. */ target_data = g_new (TargetCallbackData, 1); target_data->ui_manager = ui_manager; target_data->action_group = action_group; target_data->shares_selection_changes = shares_selection_changes; target_data->select_all_callback = select_all_callback; target_data->connect_callbacks = connect_callbacks; target_data->disconnect_callbacks = disconnect_callbacks; return target_data; } static void caja_clipboard_real_set_up (gpointer target, GtkUIManager *ui_manager, gboolean shares_selection_changes, SelectAllCallback select_all_callback, ConnectCallbacksFunc connect_callbacks, ConnectCallbacksFunc disconnect_callbacks) { TargetCallbackData *target_data; if (g_object_get_data (G_OBJECT (target), "Caja:clipboard_target_data") != NULL) { return; } target_data = initialize_clipboard_component_with_callback_data (target, ui_manager, shares_selection_changes, select_all_callback, connect_callbacks, disconnect_callbacks); g_signal_connect (target, "focus_in_event", G_CALLBACK (focus_changed_callback), target_data); g_signal_connect (target, "focus_out_event", G_CALLBACK (focus_changed_callback), target_data); g_signal_connect (target, "destroy", G_CALLBACK (target_destroy_callback), target_data); g_object_set_data_full (G_OBJECT (target), "Caja:clipboard_target_data", target_data, (GDestroyNotify) target_data_free); /* Call the focus changed callback once to merge if the window is * already in focus. */ focus_changed_callback (GTK_WIDGET (target), NULL, target_data); } void caja_clipboard_set_up_editable (GtkEditable *target, GtkUIManager *ui_manager, gboolean shares_selection_changes) { g_return_if_fail (GTK_IS_EDITABLE (target)); g_return_if_fail (GTK_IS_UI_MANAGER (ui_manager)); caja_clipboard_real_set_up (target, ui_manager, shares_selection_changes, editable_select_all_callback, editable_connect_callbacks, editable_disconnect_callbacks); } void caja_clipboard_set_up_text_view (GtkTextView *target, GtkUIManager *ui_manager) { g_return_if_fail (GTK_IS_TEXT_VIEW (target)); g_return_if_fail (GTK_IS_UI_MANAGER (ui_manager)); caja_clipboard_real_set_up (target, ui_manager, TRUE, text_view_select_all_callback, text_view_connect_callbacks, text_view_disconnect_callbacks); } static GList * convert_lines_to_str_list (char **lines, gboolean *cut) { int i; GList *result; if (cut) { *cut = FALSE; } if (lines[0] == NULL) { return NULL; } if (strcmp (lines[0], "cut") == 0) { if (cut) { *cut = TRUE; } } else if (strcmp (lines[0], "copy") != 0) { return NULL; } result = NULL; for (i = 1; lines[i] != NULL; i++) { result = g_list_prepend (result, g_strdup (lines[i])); } return g_list_reverse (result); } GList* caja_clipboard_get_uri_list_from_selection_data (GtkSelectionData *selection_data, gboolean *cut, GdkAtom copied_files_atom) { GList *items; char **lines; if (gtk_selection_data_get_data_type (selection_data) != copied_files_atom || gtk_selection_data_get_length (selection_data) <= 0) { items = NULL; } else { guchar *data; /* Not sure why it's legal to assume there's an extra byte * past the end of the selection data that it's safe to write * to. But gtk_editable_selection_received does this, so I * think it is OK. */ data = (guchar *) gtk_selection_data_get_data (selection_data); data[gtk_selection_data_get_length (selection_data)] = '\0'; lines = g_strsplit (data, "\n", 0); items = convert_lines_to_str_list (lines, cut); g_strfreev (lines); } return items; } GtkClipboard * caja_clipboard_get (GtkWidget *widget) { return gtk_clipboard_get_for_display (gtk_widget_get_display (GTK_WIDGET (widget)), GDK_SELECTION_CLIPBOARD); } void caja_clipboard_clear_if_colliding_uris (GtkWidget *widget, const GList *item_uris, GdkAtom copied_files_atom) { GtkSelectionData *data; GList *clipboard_item_uris, *l; gboolean collision; collision = FALSE; data = gtk_clipboard_wait_for_contents (caja_clipboard_get (widget), copied_files_atom); if (data == NULL) { return; } clipboard_item_uris = caja_clipboard_get_uri_list_from_selection_data (data, NULL, copied_files_atom); for (l = (GList *) item_uris; l; l = l->next) { if (g_list_find_custom ((GList *) item_uris, l->data, (GCompareFunc) g_strcmp0)) { collision = TRUE; break; } } if (collision) { gtk_clipboard_clear (caja_clipboard_get (widget)); } if (clipboard_item_uris) { g_list_free_full (clipboard_item_uris, g_free); } }