1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the Free
16 * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
41 GdkAtom seltypes[LAST_SEL_TYPE];
43 typedef struct _Target {
48 GtkSelectionFunction *handler;
51 /* The following is a list of all the selection targets defined
54 static Target targets[] = {
55 { "ADOBE_PORTABLE_DOCUMENT_FORMAT", STRING, 0, 8, NULL },
56 { "APPLE_PICT", APPLE_PICT, 0, 8, NULL },
57 { "BACKGROUND", PIXEL, 0, 32, NULL },
58 { "BITMAP", BITMAP, 0, 32, NULL },
59 { "CHARACTER_POSITION", SPAN, 0, 32, NULL },
60 { "CLASS", TEXT, 0, 8, NULL },
61 { "CLIENT_WINDOW", WINDOW, 0, 32, NULL },
62 { "COLORMAP", COLORMAP, 0, 32, NULL },
63 { "COLUMN_NUMBER", SPAN, 0, 32, NULL },
64 { "COMPOUND_TEXT", COMPOUND_TEXT, 0, 8, NULL },
65 /* { "DELETE", "NULL", 0, ?, NULL }, */
66 { "DRAWABLE", DRAWABLE, 0, 32, NULL },
67 { "ENCAPSULATED_POSTSCRIPT", STRING, 0, 8, NULL },
68 { "ENCAPSULATED_POSTSCRIPT_INTERCHANGE", STRING, 0, 8, NULL },
69 { "FILE_NAME", TEXT, 0, 8, NULL },
70 { "FOREGROUND", PIXEL, 0, 32, NULL },
71 { "HOST_NAME", TEXT, 0, 8, NULL },
72 /* { "INSERT_PROPERTY", "NULL", 0, ? NULL }, */
73 /* { "INSERT_SELECTION", "NULL", 0, ? NULL }, */
74 { "LENGTH", INTEGER, 0, 32, NULL },
75 { "LINE_NUMBER", SPAN, 0, 32, NULL },
76 { "LIST_LENGTH", INTEGER, 0, 32, NULL },
77 { "MODULE", TEXT, 0, 8, NULL },
78 /* { "MULTIPLE", "ATOM_PAIR", 0, 32, NULL }, */
79 { "NAME", TEXT, 0, 8, NULL },
80 { "ODIF", TEXT, 0, 8, NULL },
81 { "OWNER_OS", TEXT, 0, 8, NULL },
82 { "PIXMAP", PIXMAP, 0, 32, NULL },
83 { "POSTSCRIPT", STRING, 0, 8, NULL },
84 { "PROCEDURE", TEXT, 0, 8, NULL },
85 { "PROCESS", INTEGER, 0, 32, NULL },
86 { "STRING", STRING, 0, 8, NULL },
87 { "TARGETS", ATOM, 0, 32, NULL },
88 { "TASK", INTEGER, 0, 32, NULL },
89 { "TEXT", TEXT, 0, 8 , NULL },
90 { "TIMESTAMP", INTEGER, 0, 32, NULL },
91 { "USER", TEXT, 0, 8, NULL },
94 static int num_targets = sizeof(targets)/sizeof(Target);
96 static int have_selection = FALSE;
98 GtkWidget *selection_text;
99 GtkWidget *selection_button;
100 GString *selection_string = NULL;
107 seltypes[SEL_TYPE_NONE] = GDK_NONE;
108 seltypes[APPLE_PICT] = gdk_atom_intern ("APPLE_PICT",FALSE);
109 seltypes[ATOM] = gdk_atom_intern ("ATOM",FALSE);
110 seltypes[ATOM_PAIR] = gdk_atom_intern ("ATOM_PAIR",FALSE);
111 seltypes[BITMAP] = gdk_atom_intern ("BITMAP",FALSE);
112 seltypes[C_STRING] = gdk_atom_intern ("C_STRING",FALSE);
113 seltypes[COLORMAP] = gdk_atom_intern ("COLORMAP",FALSE);
114 seltypes[COMPOUND_TEXT] = gdk_atom_intern ("COMPOUND_TEXT",FALSE);
115 seltypes[DRAWABLE] = gdk_atom_intern ("DRAWABLE",FALSE);
116 seltypes[INTEGER] = gdk_atom_intern ("INTEGER",FALSE);
117 seltypes[PIXEL] = gdk_atom_intern ("PIXEL",FALSE);
118 seltypes[PIXMAP] = gdk_atom_intern ("PIXMAP",FALSE);
119 seltypes[SPAN] = gdk_atom_intern ("SPAN",FALSE);
120 seltypes[STRING] = gdk_atom_intern ("STRING",FALSE);
121 seltypes[TEXT] = gdk_atom_intern ("TEXT",FALSE);
122 seltypes[WINDOW] = gdk_atom_intern ("WINDOW",FALSE);
124 for (i=0; i<num_targets; i++)
125 targets[i].target = gdk_atom_intern (targets[i].target_name, FALSE);
129 selection_toggled (GtkWidget *widget)
131 if (GTK_TOGGLE_BUTTON(widget)->active)
133 have_selection = gtk_selection_owner_set (widget,
134 GDK_SELECTION_PRIMARY,
137 gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
143 if (gdk_selection_owner_get (GDK_SELECTION_PRIMARY) == widget->window)
144 gtk_selection_owner_set (NULL, GDK_SELECTION_PRIMARY,
146 have_selection = FALSE;
152 selection_handle (GtkWidget *widget,
153 GtkSelectionData *selection_data, gpointer data)
158 if (!selection_string)
165 buffer = selection_string->str;
166 len = selection_string->len;
169 gtk_selection_data_set (selection_data,
170 selection_data->target == seltypes[COMPOUND_TEXT] ?
171 seltypes[COMPOUND_TEXT] : seltypes[STRING],
176 selection_clear (GtkWidget *widget, GdkEventSelection *event)
178 have_selection = FALSE;
179 gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
185 stringify_atom (guchar *data, gint *position)
187 gchar *str = gdk_atom_name (*(GdkAtom *)(data+*position));
188 *position += sizeof(GdkAtom);
194 stringify_text (guchar *data, gint *position)
196 gchar *str = g_strdup ((gchar *)(data+*position));
197 *position += strlen (str) + 1;
203 stringify_xid (guchar *data, gint *position)
208 sprintf(buffer,"0x%x",*(guint32 *)(data+*position));
209 str = g_strdup (buffer);
211 *position += sizeof(guint32);
217 stringify_integer (guchar *data, gint *position)
222 sprintf(buffer,"%d",*(int *)(data+*position));
223 str = g_strdup (buffer);
225 *position += sizeof(int);
231 stringify_span (guchar *data, gint *position)
236 sprintf(buffer,"%d - %d",((int *)(data+*position))[0],
237 ((int *)(data+*position))[1]);
238 str = g_strdup (buffer);
240 *position += 2*sizeof(int);
246 selection_received (GtkWidget *widget, GtkSelectionData *data)
253 if (data->length < 0)
255 g_print("Error retrieving selection\n");
259 seltype = SEL_TYPE_NONE;
260 for (i=0; i<LAST_SEL_TYPE; i++)
262 if (seltypes[i] == data->type)
269 if (seltype == SEL_TYPE_NONE)
271 char *name = gdk_atom_name (data->type);
272 g_print("Don't know how to handle type: %s (%ld)\n",
273 name?name:"<unknown>",
278 if (selection_string != NULL)
279 g_string_free (selection_string, TRUE);
281 selection_string = g_string_new (NULL);
283 gtk_text_freeze (GTK_TEXT (selection_text));
284 gtk_text_set_point (GTK_TEXT (selection_text), 0);
285 gtk_text_foreward_delete (GTK_TEXT (selection_text),
286 gtk_text_get_length (GTK_TEXT (selection_text)));
289 while (position < data->length)
294 str = stringify_atom (data->data, &position);
299 str = stringify_text (data->data, &position);
306 str = stringify_xid (data->data, &position);
310 str = stringify_integer (data->data, &position);
313 str = stringify_span (data->data, &position);
317 char *name = gdk_atom_name (data->type);
318 g_print("Can't convert type %s (%ld) to string\n",
319 name?name:"<unknown>",
321 position = data->length;
324 gtk_text_insert (GTK_TEXT (selection_text), NULL,
325 &selection_text->style->black,
327 gtk_text_insert (GTK_TEXT (selection_text), NULL,
328 &selection_text->style->black,
330 g_string_append (selection_string, str);
333 gtk_text_thaw (GTK_TEXT (selection_text));
337 paste (GtkWidget *widget, GtkWidget *entry)
342 name = gtk_entry_get_text (GTK_ENTRY(entry));
343 atom = gdk_atom_intern (name, FALSE);
345 if (atom == GDK_NONE)
347 g_print("Could not create atom: \"%s\"\n",name);
351 gtk_selection_convert (selection_button, GDK_SELECTION_PRIMARY, atom,
362 main (int argc, char *argv[])
369 GtkWidget *hscrollbar;
370 GtkWidget *vscrollbar;
373 gtk_init (&argc, &argv);
377 dialog = gtk_dialog_new ();
378 gtk_widget_set_name (dialog, "Test Input");
379 gtk_container_border_width (GTK_CONTAINER(dialog), 0);
381 gtk_signal_connect (GTK_OBJECT (dialog), "destroy",
382 GTK_SIGNAL_FUNC (quit), NULL);
384 table = gtk_table_new (4, 2, FALSE);
385 gtk_container_border_width (GTK_CONTAINER(table), 10);
387 gtk_table_set_row_spacing (GTK_TABLE (table), 0, 5);
388 gtk_table_set_row_spacing (GTK_TABLE (table), 1, 2);
389 gtk_table_set_row_spacing (GTK_TABLE (table), 2, 2);
390 gtk_table_set_col_spacing (GTK_TABLE (table), 0, 2);
391 gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->vbox),
392 table, TRUE, TRUE, 0);
393 gtk_widget_show (table);
395 selection_button = gtk_toggle_button_new_with_label ("Claim Selection");
396 gtk_table_attach (GTK_TABLE (table), selection_button, 0, 2, 0, 1,
397 GTK_EXPAND | GTK_FILL, 0, 0, 0);
398 gtk_widget_show (selection_button);
400 gtk_signal_connect (GTK_OBJECT(selection_button), "toggled",
401 GTK_SIGNAL_FUNC (selection_toggled), NULL);
402 gtk_signal_connect (GTK_OBJECT(selection_button), "selection_clear_event",
403 GTK_SIGNAL_FUNC (selection_clear), NULL);
404 gtk_signal_connect (GTK_OBJECT(selection_button), "selection_received",
405 GTK_SIGNAL_FUNC (selection_received), NULL);
407 gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
408 seltypes[STRING], selection_handle, NULL, NULL);
410 gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
411 seltypes[TEXT], selection_handle, NULL, NULL);
413 gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
414 seltypes[COMPOUND_TEXT],
415 selection_handle, NULL, NULL);
417 selection_text = gtk_text_new (NULL, NULL);
418 gtk_table_attach_defaults (GTK_TABLE (table), selection_text, 0, 1, 1, 2);
419 gtk_widget_show (selection_text);
421 hscrollbar = gtk_hscrollbar_new (GTK_TEXT (selection_text)->hadj);
422 gtk_table_attach (GTK_TABLE (table), hscrollbar, 0, 1, 2, 3,
423 GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
424 gtk_widget_show (hscrollbar);
426 vscrollbar = gtk_vscrollbar_new (GTK_TEXT (selection_text)->vadj);
427 gtk_table_attach (GTK_TABLE (table), vscrollbar, 1, 2, 1, 2,
428 GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
429 gtk_widget_show (vscrollbar);
431 hbox = gtk_hbox_new (FALSE, 2);
432 gtk_table_attach (GTK_TABLE (table), hbox, 0, 2, 3, 4,
433 GTK_EXPAND | GTK_FILL, 0, 0, 0);
434 gtk_widget_show (hbox);
436 label = gtk_label_new ("Target:");
437 gtk_box_pack_start (GTK_BOX(hbox), label, FALSE, FALSE, 0);
438 gtk_widget_show (label);
440 entry = gtk_entry_new ();
441 gtk_box_pack_start (GTK_BOX(hbox), entry, TRUE, TRUE, 0);
442 gtk_widget_show (entry);
444 /* .. And create some buttons */
445 button = gtk_button_new_with_label ("Paste");
446 gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->action_area),
447 button, TRUE, TRUE, 0);
448 gtk_signal_connect (GTK_OBJECT (button), "clicked",
449 GTK_SIGNAL_FUNC (paste), entry);
450 gtk_widget_show (button);
452 button = gtk_button_new_with_label ("Quit");
453 gtk_box_pack_start (GTK_BOX (GTK_DIALOG(dialog)->action_area),
454 button, TRUE, TRUE, 0);
456 gtk_signal_connect_object (GTK_OBJECT (button), "clicked",
457 GTK_SIGNAL_FUNC (gtk_widget_destroy),
458 GTK_OBJECT (dialog));
459 gtk_widget_show (button);
461 gtk_widget_show (dialog);