]> Pileus Git - ~andy/gtk/blob - tests/testselection.c
Initial revision
[~andy/gtk] / tests / testselection.c
1 /* GTK - The GIMP Toolkit
2  * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3  *
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.
8  *
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.
13  *
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.
17  */
18
19 #include "gtk.h"
20
21 typedef enum {
22   SEL_TYPE_NONE,
23   APPLE_PICT,
24   ATOM,
25   ATOM_PAIR,
26   BITMAP,
27   C_STRING,
28   COLORMAP,
29   COMPOUND_TEXT,
30   DRAWABLE,
31   INTEGER,
32   PIXEL,
33   PIXMAP,
34   SPAN,
35   STRING,
36   TEXT,
37   WINDOW,
38   LAST_SEL_TYPE,
39 } SelType;
40
41 GdkAtom seltypes[LAST_SEL_TYPE];
42
43 typedef struct _Target {
44   gchar *target_name;
45   SelType type;
46   GdkAtom target;
47   gint format;
48   GtkSelectionFunction *handler;
49 } Target;
50
51 /* The following is a list of all the selection targets defined
52    in the ICCCM */
53
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 },
92 };
93
94 static int num_targets = sizeof(targets)/sizeof(Target);
95
96 static int have_selection = FALSE;
97
98 GtkWidget *selection_text;
99 GtkWidget *selection_button;
100 GString *selection_string = NULL;
101
102 static void
103 init_atoms ()
104 {
105   int i;
106
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);
123
124   for (i=0; i<num_targets; i++)
125     targets[i].target = gdk_atom_intern (targets[i].target_name, FALSE);
126 }
127
128 void
129 selection_toggled (GtkWidget *widget)
130 {
131   if (GTK_TOGGLE_BUTTON(widget)->active)
132     {
133       have_selection = gtk_selection_owner_set (widget,
134                                                 GDK_SELECTION_PRIMARY,
135                                                 GDK_CURRENT_TIME);
136       if (!have_selection)
137         gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
138     }
139   else
140     {
141       if (have_selection)
142         {
143           if (gdk_selection_owner_get (GDK_SELECTION_PRIMARY) == widget->window)
144             gtk_selection_owner_set (NULL, GDK_SELECTION_PRIMARY,
145                                      GDK_CURRENT_TIME);
146           have_selection = FALSE;
147         }
148     }
149 }
150
151 void
152 selection_handle (GtkWidget *widget, 
153                   GtkSelectionData *selection_data, gpointer data)
154 {
155   guchar *buffer;
156   gint len;
157
158   if (!selection_string)
159     {
160       buffer = NULL;
161       len = 0;
162     }      
163   else
164     {
165       buffer = selection_string->str;
166       len = selection_string->len;
167     }
168   
169   gtk_selection_data_set (selection_data,
170                           selection_data->target == seltypes[COMPOUND_TEXT] ?
171                                   seltypes[COMPOUND_TEXT] : seltypes[STRING],
172                           8, buffer, len);
173 }
174
175 gint
176 selection_clear (GtkWidget *widget, GdkEventSelection *event)
177 {
178   have_selection = FALSE;
179   gtk_toggle_button_set_state (GTK_TOGGLE_BUTTON(widget), FALSE);
180
181   return TRUE;
182 }
183
184 gchar *
185 stringify_atom (guchar *data, gint *position)
186 {
187   gchar *str = gdk_atom_name (*(GdkAtom *)(data+*position));
188   *position += sizeof(GdkAtom);
189     
190   return str;
191 }
192
193 gchar *
194 stringify_text (guchar *data, gint *position)
195 {
196   gchar *str = g_strdup ((gchar *)(data+*position));
197   *position += strlen (str) + 1;
198     
199   return str;
200 }
201
202 gchar *
203 stringify_xid (guchar *data, gint *position)
204 {
205   gchar buffer[20];
206   gchar *str;
207
208   sprintf(buffer,"0x%x",*(guint32 *)(data+*position));
209   str = g_strdup (buffer);
210
211   *position += sizeof(guint32);
212     
213   return str;
214 }
215
216 gchar *
217 stringify_integer (guchar *data, gint *position)
218 {
219   gchar buffer[20];
220   gchar *str;
221
222   sprintf(buffer,"%d",*(int *)(data+*position));
223   str = g_strdup (buffer);
224
225   *position += sizeof(int);
226     
227   return str;
228 }
229
230 gchar *
231 stringify_span (guchar *data, gint *position)
232 {
233   gchar buffer[42];
234   gchar *str;
235
236   sprintf(buffer,"%d - %d",((int *)(data+*position))[0],
237           ((int *)(data+*position))[1]);
238   str = g_strdup (buffer);
239
240   *position += 2*sizeof(int);
241     
242   return str;
243 }
244
245 void
246 selection_received (GtkWidget *widget, GtkSelectionData *data)
247 {
248   int position;
249   int i;
250   SelType seltype;
251   char *str;
252   
253   if (data->length < 0)
254     {
255       g_print("Error retrieving selection\n");
256       return;
257     }
258
259   seltype = SEL_TYPE_NONE;
260   for (i=0; i<LAST_SEL_TYPE; i++)
261     {
262       if (seltypes[i] == data->type)
263         {
264           seltype = i;
265           break;
266         }
267     }
268
269   if (seltype == SEL_TYPE_NONE)
270     {
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>",
274               data->type);
275       return;
276     }
277
278   if (selection_string != NULL)
279     g_string_free (selection_string, TRUE);
280
281   selection_string = g_string_new (NULL);
282
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)));
287
288   position = 0;
289   while (position < data->length)
290     {
291       switch (seltype)
292         {
293         case ATOM:
294           str = stringify_atom (data->data, &position);
295           break;
296         case COMPOUND_TEXT:
297         case STRING:
298         case TEXT:
299           str = stringify_text (data->data, &position);
300           break;
301         case BITMAP:
302         case DRAWABLE:
303         case PIXMAP:
304         case WINDOW:
305         case COLORMAP:
306           str = stringify_xid (data->data, &position);
307           break;
308         case INTEGER:
309         case PIXEL:
310           str = stringify_integer (data->data, &position);
311           break;
312         case SPAN:
313           str = stringify_span (data->data, &position);
314           break;
315         default:
316           {
317             char *name = gdk_atom_name (data->type);
318             g_print("Can't convert type %s (%ld) to string\n",
319                     name?name:"<unknown>",
320                     data->type);
321             position = data->length;
322           }
323         }
324       gtk_text_insert (GTK_TEXT (selection_text), NULL, 
325                        &selection_text->style->black, 
326                        NULL, str, -1);
327       gtk_text_insert (GTK_TEXT (selection_text), NULL, 
328                        &selection_text->style->black, 
329                        NULL, "\n", -1);
330       g_string_append (selection_string, str);
331       g_free (str);
332     }
333   gtk_text_thaw (GTK_TEXT (selection_text));
334 }
335
336 void
337 paste (GtkWidget *widget, GtkWidget *entry)
338 {
339   char *name;
340   GdkAtom atom;
341
342   name = gtk_entry_get_text (GTK_ENTRY(entry));
343   atom = gdk_atom_intern (name, FALSE);
344
345   if (atom == GDK_NONE)
346     {
347       g_print("Could not create atom: \"%s\"\n",name);
348       return;
349     }
350
351   gtk_selection_convert (selection_button, GDK_SELECTION_PRIMARY, atom, 
352                          GDK_CURRENT_TIME);
353 }
354
355 void
356 quit ()
357 {
358   gtk_exit (0);
359 }
360
361 int
362 main (int argc, char *argv[])
363 {
364   GtkWidget *dialog;
365   GtkWidget *button;
366   GtkWidget *table;
367   GtkWidget *label;
368   GtkWidget *entry;
369   GtkWidget *hscrollbar;
370   GtkWidget *vscrollbar;
371   GtkWidget *hbox;
372   
373   gtk_init (&argc, &argv);
374
375   init_atoms();
376
377   dialog = gtk_dialog_new ();
378   gtk_widget_set_name (dialog, "Test Input");
379   gtk_container_border_width (GTK_CONTAINER(dialog), 0);
380
381   gtk_signal_connect (GTK_OBJECT (dialog), "destroy",
382                       GTK_SIGNAL_FUNC (quit), NULL);
383
384   table = gtk_table_new (4, 2, FALSE);
385   gtk_container_border_width (GTK_CONTAINER(table), 10);
386
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);
394   
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);
399
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);
406
407   gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
408                              seltypes[STRING], selection_handle, NULL, NULL);
409
410   gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
411                              seltypes[TEXT], selection_handle, NULL, NULL);
412
413   gtk_selection_add_handler (selection_button, GDK_SELECTION_PRIMARY,
414                              seltypes[COMPOUND_TEXT],
415                              selection_handle, NULL, NULL);
416
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);
420   
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);
425   
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);
430
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);
435
436   label = gtk_label_new ("Target:");
437   gtk_box_pack_start (GTK_BOX(hbox), label, FALSE, FALSE, 0);
438   gtk_widget_show (label);
439
440   entry = gtk_entry_new ();
441   gtk_box_pack_start (GTK_BOX(hbox), entry, TRUE, TRUE, 0);
442   gtk_widget_show (entry);
443
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);
451
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);
455
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);
460
461   gtk_widget_show (dialog);
462
463   gtk_main ();
464
465   return 0;
466 }