summaryrefslogtreecommitdiff
path: root/gtk/testselection.c
diff options
context:
space:
mode:
Diffstat (limited to 'gtk/testselection.c')
-rw-r--r--gtk/testselection.c466
1 files changed, 466 insertions, 0 deletions
diff --git a/gtk/testselection.c b/gtk/testselection.c
new file mode 100644
index 000000000..3377cc623
--- /dev/null
+++ b/gtk/testselection.c
@@ -0,0 +1,466 @@
+/* GTK - The GIMP Toolkit
+ * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
+ *
+ * This library 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 library 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 library; if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+#include "gtk.h"
+
+typedef enum {
+ SEL_TYPE_NONE,
+ APPLE_PICT,
+ ATOM,
+ ATOM_PAIR,
+ BITMAP,
+ C_STRING,
+ COLORMAP,
+ COMPOUND_TEXT,
+ DRAWABLE,
+ INTEGER,
+ PIXEL,
+ PIXMAP,
+ SPAN,
+ STRING,
+ TEXT,
+ WINDOW,
+ LAST_SEL_TYPE,
+} SelType;
+
+GdkAtom seltypes[LAST_SEL_TYPE];
+
+typedef struct _Target {
+ gchar *target_name;
+ SelType type;
+ GdkAtom target;
+ gint format;
+ GtkSelectionFunction *handler;
+} Target;
+
+/* The following is a list of all the selection targets defined
+ in the ICCCM */
+
+static Target targets[] = {
+ { "ADOBE_PORTABLE_DOCUMENT_FORMAT", STRING, 0, 8, NULL },
+ { "APPLE_PICT", APPLE_PICT, 0, 8, NULL },
+ { "BACKGROUND", PIXEL, 0, 32, NULL },
+ { "BITMAP", BITMAP, 0, 32, NULL },
+ { "CHARACTER_POSITION", SPAN, 0, 32, NULL },
+ { "CLASS", TEXT, 0, 8, NULL },
+ { "CLIENT_WINDOW", WINDOW, 0, 32, NULL },
+ { "COLORMAP", COLORMAP, 0, 32, NULL },
+ { "COLUMN_NUMBER", SPAN, 0, 32, NULL },
+ { "COMPOUND_TEXT", COMPOUND_TEXT, 0, 8, NULL },
+ /* { "DELETE", "NULL", 0, ?, NULL }, */
+ { "DRAWABLE", DRAWABLE, 0, 32, NULL },
+ { "ENCAPSULATED_POSTSCRIPT", STRING, 0, 8, NULL },
+ { "ENCAPSULATED_POSTSCRIPT_INTERCHANGE", STRING, 0, 8, NULL },
+ { "FILE_NAME", TEXT, 0, 8, NULL },
+ { "FOREGROUND", PIXEL, 0, 32, NULL },
+ { "HOST_NAME", TEXT, 0, 8, NULL },
+ /* { "INSERT_PROPERTY", "NULL", 0, ? NULL }, */
+ /* { "INSERT_SELECTION", "NULL", 0, ? NULL }, */
+ { "LENGTH", INTEGER, 0, 32, NULL },
+ { "LINE_NUMBER", SPAN, 0, 32, NULL },
+ { "LIST_LENGTH", INTEGER, 0, 32, NULL },
+ { "MODULE", TEXT, 0, 8, NULL },
+ /* { "MULTIPLE", "ATOM_PAIR", 0, 32, NULL }, */
+ { "NAME", TEXT, 0, 8, NULL },
+ { "ODIF", TEXT, 0, 8, NULL },
+ { "OWNER_OS", TEXT, 0, 8, NULL },
+ { "PIXMAP", PIXMAP, 0, 32, NULL },
+ { "POSTSCRIPT", STRING, 0, 8, NULL },
+ { "PROCEDURE", TEXT, 0, 8, NULL },
+ { "PROCESS", INTEGER, 0, 32, NULL },
+ { "STRING", STRING, 0, 8, NULL },
+ { "TARGETS", ATOM, 0, 32, NULL },
+ { "TASK", INTEGER, 0, 32, NULL },
+ { "TEXT", TEXT, 0, 8 , NULL },
+ { "TIMESTAMP", INTEGER, 0, 32, NULL },
+ { "USER", TEXT, 0, 8, NULL },
+};
+
+static int num_targets = sizeof(targets)/sizeof(Target);
+
+static int have_selection = FALSE;
+
+GtkWidget *selection_text;
+GtkWidget *selection_button;
+GString *selection_string = NULL;
+
+static void
+init_atoms ()
+{
+ int i;
+
+ seltypes[SEL_TYPE_NONE] = GDK_NONE;
+ seltypes[APPLE_PICT] = gdk_atom_intern ("APPLE_PICT",FALSE);
+ seltypes[ATOM] = gdk_atom_intern ("ATOM",FALSE);
+ seltypes[ATOM_PAIR] = gdk_atom_intern ("ATOM_PAIR",FALSE);
+ seltypes[BITMAP] = gdk_atom_intern ("BITMAP",FALSE);
+ seltypes[C_STRING] = gdk_atom_intern ("C_STRING",FALSE);
+ seltypes[COLORMAP] = gdk_atom_intern ("COLORMAP",FALSE);
+ seltypes[COMPOUND_TEXT] = gdk_atom_intern ("COMPOUND_TEXT",FALSE);
+ seltypes[DRAWABLE] = gdk_atom_intern ("DRAWABLE",FALSE);
+ seltypes[INTEGER] = gdk_atom_intern ("INTEGER",FALSE);
+ seltypes[PIXEL] = gdk_atom_intern ("PIXEL",FALSE);
+ seltypes[PIXMAP] = gdk_atom_intern ("PIXMAP",FALSE);
+ seltypes[SPAN] = gdk_atom_intern ("SPAN",FALSE);
+ seltypes[STRING] = gdk_atom_intern ("STRING",FALSE);
+ seltypes[TEXT] = gdk_atom_intern ("TEXT",FALSE);
+ seltypes[WINDOW] = gdk_atom_intern ("WINDOW",FALSE);
+
+ for (i=0; i<num_targets; i++)
+ targets[i].target = gdk_atom_intern (targets[i].target_name, FALSE);
+}
+
+void
+selection_toggled (GtkWidget *widget)
+{
+ if (GTK_TOGGLE_BUTTON(widget)->active)
+ {
+ have_selection = gtk_selection_owner_set (widget,
+ GDK_SELECTION_PRIMARY,
+ GDK_CURRENT_TIME);
+ if (!have_selection)
+ gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
+ }
+ else
+ {
+ if (have_selection)
+ {
+ if (gdk_selection_owner_get (GDK_SELECTION_PRIMARY) == widget->window)
+ gtk_selection_owner_set (NULL, GDK_SELECTION_PRIMARY,
+ GDK_CURRENT_TIME);
+ have_selection = FALSE;
+ }
+ }
+}
+
+void
+selection_handle (GtkWidget *widget,
+ GtkSelectionData *selection_data, gpointer data)
+{
+ guchar *buffer;
+ gint len;
+
+ if (!selection_string)
+ {
+ buffer = NULL;
+ len = 0;
+ }
+ else
+ {
+ buffer = selection_string->str;
+ len = selection_string->len;
+ }
+
+ gtk_selection_data_set (selection_data,
+ selection_data->target == seltypes[COMPOUND_TEXT] ?
+ seltypes[COMPOUND_TEXT] : seltypes[STRING],
+ 8, buffer, len);
+}
+
+gint
+selection_clear (GtkWidget *widget, GdkEventSelection *event)
+{
+ have_selection = FALSE;
+ gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
+
+ return TRUE;
+}
+
+gchar *
+stringify_atom (guchar *data, gint *position)
+{
+ gchar *str = gdk_atom_name (*(GdkAtom *)(data+*position));
+ *position += sizeof(GdkAtom);
+
+ return str;
+}
+
+gchar *
+stringify_text (guchar *data, gint *position)
+{
+ gchar *str = g_strdup ((gchar *)(data+*position));
+ *position += strlen (str) + 1;
+
+ return str;
+}
+
+gchar *
+stringify_xid (guchar *data, gint *position)
+{
+ gchar buffer[20];
+ gchar *str;
+
+ sprintf(buffer,"0x%x",*(guint32 *)(data+*position));
+ str = g_strdup (buffer);
+
+ *position += sizeof(guint32);
+
+ return str;
+}
+
+gchar *
+stringify_integer (guchar *data, gint *position)
+{
+ gchar buffer[20];
+ gchar *str;
+
+ sprintf(buffer,"%d",*(int *)(data+*position));
+ str = g_strdup (buffer);
+
+ *position += sizeof(int);
+
+ return str;
+}
+
+gchar *
+stringify_span (guchar *data, gint *position)
+{
+ gchar buffer[42];
+ gchar *str;
+
+ sprintf(buffer,"%d - %d",((int *)(data+*position))[0],
+ ((int *)(data+*position))[1]);
+ str = g_strdup (buffer);
+
+ *position += 2*sizeof(int);
+
+ return str;
+}
+
+void
+selection_received (GtkWidget *widget, GtkSelectionData *data)
+{
+ int position;
+ int i;
+ SelType seltype;
+ char *str;
+
+ if (data->length < 0)
+ {
+ g_print("Error retrieving selection\n");
+ return;
+ }
+
+ seltype = SEL_TYPE_NONE;
+ for (i=0; i<LAST_SEL_TYPE; i++)
+ {
+ if (seltypes[i] == data->type)
+ {
+ seltype = i;
+ break;
+ }
+ }
+
+ if (seltype == SEL_TYPE_NONE)
+ {
+ char *name = gdk_atom_name (data->type);
+ g_print("Don't know how to handle type: %s (%ld)\n",
+ name?name:"<unknown>",
+ data->type);
+ return;
+ }
+
+ if (selection_string != NULL)
+ g_string_free (selection_string, TRUE);
+
+ selection_string = g_string_new (NULL);
+
+ gtk_text_freeze (GTK_TEXT (selection_text));
+ gtk_text_set_point (GTK_TEXT (selection_text), 0);
+ gtk_text_foreward_delete (GTK_TEXT (selection_text),
+ gtk_text_get_length (GTK_TEXT (selection_text)));
+
+ position = 0;
+ while (position < data->length)
+ {
+ switch (seltype)
+ {
+ case ATOM:
+ str = stringify_atom (data->data, &position);
+ break;
+ case COMPOUND_TEXT:
+ case STRING:
+ case TEXT:
+ str = stringify_text (data->data, &position);
+ break;
+ case BITMAP:
+ case DRAWABLE:
+ case PIXMAP:
+ case WINDOW:
+ case COLORMAP:
+ str = stringify_xid (data->data, &position);
+ break;
+ case INTEGER:
+ case PIXEL:
+ str = stringify_integer (data->data, &position);
+ break;
+ case SPAN:
+ str = stringify_span (data->data, &position);
+ break;
+ default:
+ {
+ char *name = gdk_atom_name (data->type);
+ g_print("Can't convert type %s (%ld) to string\n",
+ name?name:"<unknown>",
+ data->type);
+ position = data->length;
+ }
+ }
+ gtk_text_insert (GTK_TEXT (selection_text), NULL,
+ &selection_text->style->black,
+ NULL, str, -1);
+ gtk_text_insert (GTK_TEXT (selection_text), NULL,
+ &selection_text->style->black,
+ NULL, "\n", -1);
+ g_string_append (selection_string, str);
+ g_free (str);
+ }
+ gtk_text_thaw (GTK_TEXT (selection_text));
+}
+
+void
+paste (GtkWidget *widget, GtkWidget *entry)
+{
+ char *name;
+ GdkAtom atom;
+
+ name = gtk_entry_get_text (GTK_ENTRY(entry));
+ atom = gdk_atom_intern (name, FALSE);
+
+ if (atom == GDK_NONE)
+ {
+ g_print("Could not create atom: \"%s\"\n",name);
+ return;
+ }
+
+ gtk_selection_convert (selection_button, GDK_SELECTION_PRIMARY, atom,
+ GDK_CURRENT_TIME);
+}
+
+void
+quit ()
+{
+ gtk_exit (0);
+}
+
+int
+main (int argc, char *argv[])
+{
+ GtkWidget *dialog;
+ GtkWidget *button;
+ GtkWidget *table;
+ GtkWidget *label;
+ GtkWidget *entry;
+ GtkWidget *hscrollbar;
+ GtkWidget *vscrollbar;
+ GtkWidget *hbox;
+
+ gtk_init (&argc, &argv);
+
+ init_atoms();
+
+ dialog = gtk_dialog_new ();
+ gtk_widget_set_name (dialog, "Test Input");
+ gtk_container_border_width (GTK_CONTAINER(dialog), 0);
+
+ gtk_signal_connect (GTK_OBJECT (dialog), "destroy",
+ GTK_SIGNAL_FUNC (quit), NULL);
+
+ table = gtk_table_new (4, 2, FALSE);
+ gtk_container_border_width (GTK_CONTAINER(table), 10);
+
+ gtk_table_set_row_spacing (GTK_TABLE (table), 0, 5);
+ gtk_table_set_row_spacing (GTK_TABLE (table), 1, 2);
+ gtk_table_set_row_spacing (GTK_TABLE (table), 2, 2);
+ gtk_table_set_col_spacing (GTK_TABLE (table), 0, 2);
+ gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->vbox),
+ table, TRUE, TRUE, 0);
+ gtk_widget_show (table);
+
+ selection_button = gtk_toggle_button_new_with_label ("Claim Selection");
+ gtk_table_attach (GTK_TABLE (table), selection_button, 0, 2, 0, 1,
+ GTK_EXPAND | GTK_FILL, 0, 0, 0);
+ gtk_widget_show (selection_button);
+
+ gtk_signal_connect (GTK_OBJECT(selection_button), "toggled",
+ GTK_SIGNAL_FUNC (selection_toggled), NULL);
+ gtk_signal_connect (GTK_OBJECT(selection_button), "selection_clear_event",
+ GTK_SIGNAL_FUNC (selection_clear), NULL);
+ gtk_signal_connect (GTK_OBJECT(selection_button), "selection_received",
+ GTK_SIGNAL_FUNC (selection_received), NULL);
+
+ gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
+ seltypes[STRING], selection_handle, NULL, NULL);
+
+ gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
+ seltypes[TEXT], selection_handle, NULL, NULL);
+
+ gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
+ seltypes[COMPOUND_TEXT],
+ selection_handle, NULL, NULL);
+
+ selection_text = gtk_text_new (NULL, NULL);
+ gtk_table_attach_defaults (GTK_TABLE (table), selection_text, 0, 1, 1, 2);
+ gtk_widget_show (selection_text);
+
+ hscrollbar = gtk_hscrollbar_new (GTK_TEXT (selection_text)->hadj);
+ gtk_table_attach (GTK_TABLE (table), hscrollbar, 0, 1, 2, 3,
+ GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
+ gtk_widget_show (hscrollbar);
+
+ vscrollbar = gtk_vscrollbar_new (GTK_TEXT (selection_text)->vadj);
+ gtk_table_attach (GTK_TABLE (table), vscrollbar, 1, 2, 1, 2,
+ GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
+ gtk_widget_show (vscrollbar);
+
+ hbox = gtk_hbox_new (FALSE, 2);
+ gtk_table_attach (GTK_TABLE (table), hbox, 0, 2, 3, 4,
+ GTK_EXPAND | GTK_FILL, 0, 0, 0);
+ gtk_widget_show (hbox);
+
+ label = gtk_label_new ("Target:");
+ gtk_box_pack_start (GTK_BOX(hbox), label, FALSE, FALSE, 0);
+ gtk_widget_show (label);
+
+ entry = gtk_entry_new ();
+ gtk_box_pack_start (GTK_BOX(hbox), entry, TRUE, TRUE, 0);
+ gtk_widget_show (entry);
+
+ /* .. And create some buttons */
+ button = gtk_button_new_with_label ("Paste");
+ gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->action_area),
+ button, TRUE, TRUE, 0);
+ gtk_signal_connect (GTK_OBJECT (button), "clicked",
+ GTK_SIGNAL_FUNC (paste), entry);
+ gtk_widget_show (button);
+
+ button = gtk_button_new_with_label ("Quit");
+ gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->action_area),
+ button, TRUE, TRUE, 0);
+
+ gtk_signal_connect_object (GTK_OBJECT (button), "clicked",
+ GTK_SIGNAL_FUNC (gtk_widget_destroy),
+ GTK_OBJECT (dialog));
+ gtk_widget_show (button);
+
+ gtk_widget_show (dialog);
+
+ gtk_main ();
+
+ return 0;
+}