]> Pileus Git - ~andy/gtk/blob - gtk/gtkeditable.c
Moved entry->visible into editable class. (Leave it behind deprecated for
[~andy/gtk] / gtk / gtkeditable.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
16  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  */
19 #include <ctype.h>
20 #include <string.h>
21 #ifdef USE_XIM
22 #include "gdk/gdkx.h"
23 #endif
24 #include "gdk/gdkkeysyms.h"
25 #include "gdk/gdki18n.h"
26 #include "gtkeditable.h"
27 #include "gtkmain.h"
28 #include "gtkselection.h"
29 #include "gtksignal.h"
30
31 #define MIN_EDITABLE_WIDTH  150
32 #define DRAW_TIMEOUT     20
33 #define INNER_BORDER     2
34
35 enum {
36   CHANGED,
37   INSERT_TEXT,
38   DELETE_TEXT,
39   /* Binding actions */
40   ACTIVATE,
41   SET_EDITABLE,
42   MOVE_CURSOR,
43   MOVE_WORD,
44   MOVE_PAGE,
45   MOVE_TO_ROW,
46   MOVE_TO_COLUMN,
47   KILL_CHAR,
48   KILL_WORD,
49   KILL_LINE,
50   CUT_CLIPBOARD,
51   COPY_CLIPBOARD,
52   PASTE_CLIPBOARD,
53   LAST_SIGNAL
54 };
55
56 enum {
57   ARG_0,
58   ARG_TEXT_POSITION,
59   ARG_EDITABLE
60 };
61
62 /* values for selection info */
63
64 enum {
65   TARGET_STRING,
66   TARGET_TEXT,
67   TARGET_COMPOUND_TEXT
68 };
69
70 static void gtk_editable_class_init          (GtkEditableClass *klass);
71 static void gtk_editable_init                (GtkEditable      *editable);
72 static void gtk_editable_set_arg             (GtkObject        *object,
73                                               GtkArg           *arg,
74                                               guint             arg_id);
75 static void gtk_editable_get_arg             (GtkObject        *object,
76                                               GtkArg           *arg,
77                                               guint             arg_id);
78 static void *gtk_editable_get_public_chars   (GtkEditable      *editable,
79                                               gint              start,
80                                               gint              end);
81 static gint gtk_editable_selection_clear     (GtkWidget        *widget,
82                                              GdkEventSelection *event);
83 static void gtk_editable_selection_get      (GtkWidget         *widget,
84                                              GtkSelectionData  *selection_data,
85                                              guint              info,
86                                              guint              time);
87 static void gtk_editable_selection_received  (GtkWidget        *widget,
88                                               GtkSelectionData *selection_data,
89                                               guint             time);
90
91 static void gtk_editable_set_selection    (GtkEditable      *editable,
92                                            gint              start,
93                                            gint              end);
94 static guint32 gtk_editable_get_event_time (GtkEditable     *editable);
95
96 static void gtk_editable_real_cut_clipboard   (GtkEditable     *editable);
97 static void gtk_editable_real_copy_clipboard  (GtkEditable     *editable);
98 static void gtk_editable_real_paste_clipboard (GtkEditable     *editable);
99 static void gtk_editable_real_set_editable    (GtkEditable     *editable,
100                                                gboolean         is_editable);
101      
102 static GtkWidgetClass *parent_class = NULL;
103 static guint editable_signals[LAST_SIGNAL] = { 0 };
104
105 static GdkAtom clipboard_atom = GDK_NONE;
106
107 GtkType
108 gtk_editable_get_type (void)
109 {
110   static GtkType editable_type = 0;
111
112   if (!editable_type)
113     {
114       static const GtkTypeInfo editable_info =
115       {
116         "GtkEditable",
117         sizeof (GtkEditable),
118         sizeof (GtkEditableClass),
119         (GtkClassInitFunc) gtk_editable_class_init,
120         (GtkObjectInitFunc) gtk_editable_init,
121         /* reserved_1 */ NULL,
122         /* reserved_2 */ NULL,
123         (GtkClassInitFunc) NULL,
124       };
125
126       editable_type = gtk_type_unique (GTK_TYPE_WIDGET, &editable_info);
127     }
128
129   return editable_type;
130 }
131
132 static void
133 gtk_editable_class_init (GtkEditableClass *class)
134 {
135   GtkObjectClass *object_class;
136   GtkWidgetClass *widget_class;
137
138   object_class = (GtkObjectClass*) class;
139   widget_class = (GtkWidgetClass*) class;
140
141   parent_class = gtk_type_class (GTK_TYPE_WIDGET);
142
143   editable_signals[CHANGED] =
144     gtk_signal_new ("changed",
145                     GTK_RUN_LAST,
146                     object_class->type,
147                     GTK_SIGNAL_OFFSET (GtkEditableClass, changed),
148                     gtk_marshal_NONE__NONE,
149                     GTK_TYPE_NONE, 0);
150
151   editable_signals[INSERT_TEXT] =
152     gtk_signal_new ("insert_text",
153                     GTK_RUN_LAST,
154                     object_class->type,
155                     GTK_SIGNAL_OFFSET (GtkEditableClass, insert_text),
156                     gtk_marshal_NONE__POINTER_INT_POINTER,
157                     GTK_TYPE_NONE,
158                     3,
159                     GTK_TYPE_STRING,
160                     GTK_TYPE_INT,
161                     GTK_TYPE_POINTER);
162
163   editable_signals[DELETE_TEXT] =
164     gtk_signal_new ("delete_text",
165                     GTK_RUN_LAST,
166                     object_class->type,
167                     GTK_SIGNAL_OFFSET (GtkEditableClass, delete_text),
168                     gtk_marshal_NONE__INT_INT,
169                     GTK_TYPE_NONE,
170                     2,
171                     GTK_TYPE_INT,
172                     GTK_TYPE_INT);                  
173
174   editable_signals[ACTIVATE] =
175     gtk_signal_new ("activate",
176                     GTK_RUN_LAST | GTK_RUN_ACTION,
177                     object_class->type,
178                     GTK_SIGNAL_OFFSET (GtkEditableClass, activate),
179                     gtk_marshal_NONE__NONE,
180                     GTK_TYPE_NONE, 0);
181
182   editable_signals[SET_EDITABLE] =
183     gtk_signal_new ("set-editable",
184                     GTK_RUN_LAST | GTK_RUN_ACTION,
185                     object_class->type,
186                     GTK_SIGNAL_OFFSET (GtkEditableClass, set_editable),
187                     gtk_marshal_NONE__BOOL,
188                     GTK_TYPE_NONE, 1,
189                     GTK_TYPE_BOOL);
190
191   editable_signals[MOVE_CURSOR] =
192     gtk_signal_new ("move_cursor",
193                     GTK_RUN_LAST | GTK_RUN_ACTION,
194                     object_class->type,
195                     GTK_SIGNAL_OFFSET (GtkEditableClass, move_cursor),
196                     gtk_marshal_NONE__INT_INT,
197                     GTK_TYPE_NONE, 2, 
198                     GTK_TYPE_INT, 
199                     GTK_TYPE_INT);
200
201   editable_signals[MOVE_WORD] =
202     gtk_signal_new ("move_word",
203                     GTK_RUN_LAST | GTK_RUN_ACTION,
204                     object_class->type,
205                     GTK_SIGNAL_OFFSET (GtkEditableClass, move_word),
206                     gtk_marshal_NONE__INT,
207                     GTK_TYPE_NONE, 1, 
208                     GTK_TYPE_INT);
209
210   editable_signals[MOVE_PAGE] =
211     gtk_signal_new ("move_page",
212                     GTK_RUN_LAST | GTK_RUN_ACTION,
213                     object_class->type,
214                     GTK_SIGNAL_OFFSET (GtkEditableClass, move_page),
215                     gtk_marshal_NONE__INT_INT,
216                     GTK_TYPE_NONE, 2, 
217                     GTK_TYPE_INT, 
218                     GTK_TYPE_INT);
219
220   editable_signals[MOVE_TO_ROW] =
221     gtk_signal_new ("move_to_row",
222                     GTK_RUN_LAST | GTK_RUN_ACTION,
223                     object_class->type,
224                     GTK_SIGNAL_OFFSET (GtkEditableClass, move_to_row),
225                     gtk_marshal_NONE__INT,
226                     GTK_TYPE_NONE, 1, 
227                     GTK_TYPE_INT);
228
229   editable_signals[MOVE_TO_COLUMN] =
230     gtk_signal_new ("move_to_column",
231                     GTK_RUN_LAST | GTK_RUN_ACTION,
232                     object_class->type,
233                     GTK_SIGNAL_OFFSET (GtkEditableClass, move_to_column),
234                     gtk_marshal_NONE__INT,
235                     GTK_TYPE_NONE, 1, 
236                     GTK_TYPE_INT);
237
238   editable_signals[KILL_CHAR] =
239     gtk_signal_new ("kill_char",
240                     GTK_RUN_LAST | GTK_RUN_ACTION,
241                     object_class->type,
242                     GTK_SIGNAL_OFFSET (GtkEditableClass, kill_char),
243                     gtk_marshal_NONE__INT,
244                     GTK_TYPE_NONE, 1, 
245                     GTK_TYPE_INT);
246
247   editable_signals[KILL_WORD] =
248     gtk_signal_new ("kill_word",
249                     GTK_RUN_LAST | GTK_RUN_ACTION,
250                     object_class->type,
251                     GTK_SIGNAL_OFFSET (GtkEditableClass, kill_word),
252                     gtk_marshal_NONE__INT,
253                     GTK_TYPE_NONE, 1, 
254                     GTK_TYPE_INT);
255
256   editable_signals[KILL_LINE] =
257     gtk_signal_new ("kill_line",
258                     GTK_RUN_LAST | GTK_RUN_ACTION,
259                     object_class->type,
260                     GTK_SIGNAL_OFFSET (GtkEditableClass, kill_line),
261                     gtk_marshal_NONE__INT,
262                     GTK_TYPE_NONE, 1, 
263                     GTK_TYPE_INT);
264
265   editable_signals[CUT_CLIPBOARD] =
266     gtk_signal_new ("cut_clipboard",
267                     GTK_RUN_LAST | GTK_RUN_ACTION,
268                     object_class->type,
269                     GTK_SIGNAL_OFFSET (GtkEditableClass, cut_clipboard),
270                     gtk_marshal_NONE__NONE,
271                     GTK_TYPE_NONE, 0);
272
273   editable_signals[COPY_CLIPBOARD] =
274     gtk_signal_new ("copy_clipboard",
275                     GTK_RUN_LAST | GTK_RUN_ACTION,
276                     object_class->type,
277                     GTK_SIGNAL_OFFSET (GtkEditableClass, copy_clipboard),
278                     gtk_marshal_NONE__NONE,
279                     GTK_TYPE_NONE, 0);
280
281   editable_signals[PASTE_CLIPBOARD] =
282     gtk_signal_new ("paste_clipboard",
283                     GTK_RUN_LAST | GTK_RUN_ACTION,
284                     object_class->type,
285                     GTK_SIGNAL_OFFSET (GtkEditableClass, paste_clipboard),
286                     gtk_marshal_NONE__NONE,
287                     GTK_TYPE_NONE, 0);
288
289   gtk_object_class_add_signals (object_class, editable_signals, LAST_SIGNAL);
290
291   gtk_object_add_arg_type ("GtkEditable::text_position", GTK_TYPE_INT, GTK_ARG_READWRITE, ARG_TEXT_POSITION);
292   gtk_object_add_arg_type ("GtkEditable::editable", GTK_TYPE_BOOL, GTK_ARG_READWRITE, ARG_EDITABLE);
293     
294   object_class->set_arg = gtk_editable_set_arg;
295   object_class->get_arg = gtk_editable_get_arg;
296
297   widget_class->selection_clear_event = gtk_editable_selection_clear;
298   widget_class->selection_received = gtk_editable_selection_received;
299   widget_class->selection_get = gtk_editable_selection_get;
300
301   class->insert_text = NULL;
302   class->delete_text = NULL;
303
304   class->activate = NULL;
305   class->set_editable = gtk_editable_real_set_editable;
306
307   class->move_cursor = NULL;
308   class->move_word = NULL;
309   class->move_page = NULL;
310   class->move_to_row = NULL;
311   class->move_to_column = NULL;
312
313   class->kill_char = NULL;
314   class->kill_word = NULL;
315   class->kill_line = NULL;
316
317   class->cut_clipboard = gtk_editable_real_cut_clipboard;
318   class->copy_clipboard = gtk_editable_real_copy_clipboard;
319   class->paste_clipboard = gtk_editable_real_paste_clipboard;
320
321   class->update_text = NULL;
322   class->get_chars = NULL;
323   class->set_selection = NULL;
324   class->set_position = NULL;
325 }
326
327 static void
328 gtk_editable_set_arg (GtkObject      *object,
329                       GtkArg         *arg,
330                       guint           arg_id)
331 {
332   GtkEditable *editable;
333
334   editable = GTK_EDITABLE (object);
335
336   switch (arg_id)
337     {
338     case ARG_TEXT_POSITION:
339       gtk_editable_set_position (editable, GTK_VALUE_INT (*arg));
340       break;
341     case ARG_EDITABLE:
342       gtk_editable_set_editable (editable, GTK_VALUE_BOOL (*arg));
343       break;
344     default:
345       break;
346     }
347 }
348
349 static void
350 gtk_editable_get_arg (GtkObject      *object,
351                       GtkArg         *arg,
352                       guint           arg_id)
353 {
354   GtkEditable *editable;
355
356   editable = GTK_EDITABLE (object);
357
358   switch (arg_id)
359     {
360     case ARG_TEXT_POSITION:
361       GTK_VALUE_INT (*arg) = editable->current_pos;
362       break;
363     case ARG_EDITABLE:
364       GTK_VALUE_BOOL (*arg) = editable->editable;
365       break;
366     default:
367       arg->type = GTK_TYPE_INVALID;
368       break;
369     }
370 }
371
372 static void
373 gtk_editable_init (GtkEditable *editable)
374 {
375   static const GtkTargetEntry targets[] = {
376     { "STRING", 0, TARGET_STRING },
377     { "TEXT",   0, TARGET_TEXT }, 
378     { "COMPOUND_TEXT", 0, TARGET_COMPOUND_TEXT }
379   };
380   static const gint n_targets = sizeof(targets) / sizeof(targets[0]);
381   
382   GTK_WIDGET_SET_FLAGS (editable, GTK_CAN_FOCUS);
383
384   editable->selection_start_pos = 0;
385   editable->selection_end_pos = 0;
386   editable->has_selection = FALSE;
387   editable->editable = 1;
388   editable->visible = 1;
389   editable->clipboard_text = NULL;
390
391 #ifdef USE_XIM
392   editable->ic = NULL;
393 #endif
394
395   if (!clipboard_atom)
396     clipboard_atom = gdk_atom_intern ("CLIPBOARD", FALSE);
397
398   gtk_selection_add_targets (GTK_WIDGET (editable), GDK_SELECTION_PRIMARY,
399                              targets, n_targets);
400   gtk_selection_add_targets (GTK_WIDGET (editable), clipboard_atom,
401                              targets, n_targets);
402 }
403
404 void
405 gtk_editable_insert_text (GtkEditable *editable,
406                           const gchar *new_text,
407                           gint         new_text_length,
408                           gint        *position)
409 {
410   GtkEditableClass *klass;
411
412   gchar buf[64];
413   gchar *text;
414
415   g_return_if_fail (editable != NULL);
416   g_return_if_fail (GTK_IS_EDITABLE (editable));
417
418   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
419
420   if (new_text_length <= 64)
421     text = buf;
422   else
423     text = g_new (gchar, new_text_length);
424
425   strncpy (text, new_text, new_text_length);
426
427   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[INSERT_TEXT], text, new_text_length, position);
428   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[CHANGED]);
429
430   if (new_text_length > 64)
431     g_free (text);
432 }
433
434 void
435 gtk_editable_delete_text (GtkEditable *editable,
436                           gint         start_pos,
437                           gint         end_pos)
438 {
439   GtkEditableClass *klass;
440
441   g_return_if_fail (editable != NULL);
442   g_return_if_fail (GTK_IS_EDITABLE (editable));
443
444   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
445
446   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[DELETE_TEXT], start_pos, end_pos);
447   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[CHANGED]);
448 }
449
450 static void
451 gtk_editable_update_text (GtkEditable *editable,
452                        gint      start_pos,
453                        gint      end_pos)
454 {
455   GtkEditableClass *klass;
456
457   g_return_if_fail (editable != NULL);
458   g_return_if_fail (GTK_IS_EDITABLE (editable));
459
460   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
461
462   klass->update_text (editable, start_pos, end_pos);
463 }
464
465 gchar *    
466 gtk_editable_get_chars      (GtkEditable      *editable,
467                              gint              start,
468                              gint              end)
469 {
470   GtkEditableClass *klass;
471
472   g_return_val_if_fail (editable != NULL, NULL);
473   g_return_val_if_fail (GTK_IS_EDITABLE (editable), NULL);
474
475   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
476
477   return klass->get_chars (editable, start, end);
478 }
479
480 /*
481  * Like gtk_editable_get_chars, but if the editable is not
482  * visible, return asterisks
483  */
484 static void *    
485 gtk_editable_get_public_chars (GtkEditable      *editable,
486                                gint              start,
487                                gint              end)
488 {
489   if (editable->visible)
490     return gtk_editable_get_chars (editable, start, end);
491   else
492     {
493       gint i;
494       gint nchars = end - start;
495       gchar *str;
496        
497       if (nchars < 0)
498         nchars = -nchars;
499
500       str = g_new (gchar, nchars + 1);
501       for (i = 0; i<nchars; i++)
502         str[i] = '*';
503       str[i] = '\0';
504
505       return str;
506     }
507 }
508
509 static void
510 gtk_editable_set_selection (GtkEditable *editable,
511                             gint      start_pos,
512                             gint      end_pos)
513 {
514   GtkEditableClass *klass;
515
516   g_return_if_fail (editable != NULL);
517   g_return_if_fail (GTK_IS_EDITABLE (editable));
518
519   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
520
521   klass->set_selection (editable, start_pos, end_pos);
522 }
523
524 void
525 gtk_editable_set_position (GtkEditable      *editable,
526                            gint              position)
527 {
528   GtkEditableClass *klass;
529
530   g_return_if_fail (editable != NULL);
531   g_return_if_fail (GTK_IS_EDITABLE (editable));
532
533   klass = GTK_EDITABLE_CLASS (GTK_OBJECT (editable)->klass);
534
535   klass->set_position (editable, position);
536 }
537
538 gint
539 gtk_editable_get_position (GtkEditable      *editable)
540 {
541   g_return_val_if_fail (editable != NULL, -1);
542   g_return_val_if_fail (GTK_IS_EDITABLE (editable), -1);
543
544   return editable->current_pos;
545 }
546
547 static gint
548 gtk_editable_selection_clear (GtkWidget         *widget,
549                               GdkEventSelection *event)
550 {
551   GtkEditable *editable;
552   
553   g_return_val_if_fail (widget != NULL, FALSE);
554   g_return_val_if_fail (GTK_IS_EDITABLE (widget), FALSE);
555   g_return_val_if_fail (event != NULL, FALSE);
556   
557   /* Let the selection handling code know that the selection
558    * has been changed, since we've overriden the default handler */
559   if (!gtk_selection_clear (widget, event))
560     return FALSE;
561   
562   editable = GTK_EDITABLE (widget);
563   
564   if (event->selection == GDK_SELECTION_PRIMARY)
565     {
566       if (editable->has_selection)
567         {
568           editable->has_selection = FALSE;
569           gtk_editable_update_text (editable, editable->selection_start_pos,
570                                     editable->selection_end_pos);
571         }
572     }
573   else if (event->selection == clipboard_atom)
574     {
575       g_free (editable->clipboard_text);
576       editable->clipboard_text = NULL;
577     }
578   
579   return TRUE;
580 }
581
582 static void
583 gtk_editable_selection_get (GtkWidget        *widget,
584                             GtkSelectionData *selection_data,
585                             guint             info,
586                             guint             time)
587 {
588   GtkEditable *editable;
589   gint selection_start_pos;
590   gint selection_end_pos;
591
592   gchar *str;
593   gint length;
594
595   g_return_if_fail (widget != NULL);
596   g_return_if_fail (GTK_IS_EDITABLE (widget));
597
598   editable = GTK_EDITABLE (widget);
599
600   if (selection_data->selection == GDK_SELECTION_PRIMARY)
601     {
602       selection_start_pos = MIN (editable->selection_start_pos, editable->selection_end_pos);
603       selection_end_pos = MAX (editable->selection_start_pos, editable->selection_end_pos);
604       str = gtk_editable_get_public_chars(editable, 
605                                           selection_start_pos, 
606                                           selection_end_pos);
607       if (!str)
608          return;                /* Refuse */
609       length = strlen (str);
610     }
611   else                          /* CLIPBOARD */
612     {
613       if (!editable->clipboard_text)
614         return;                 /* Refuse */
615
616       str = editable->clipboard_text;
617       length = strlen (editable->clipboard_text);
618     }
619   
620   if (info == TARGET_STRING)
621     {
622       gtk_selection_data_set (selection_data,
623                               GDK_SELECTION_TYPE_STRING,
624                               8*sizeof(gchar), (guchar *)str, length);
625     }
626   else if ((info == TARGET_TEXT) || (info == TARGET_COMPOUND_TEXT))
627     {
628       guchar *text;
629       gchar c;
630       GdkAtom encoding;
631       gint format;
632       gint new_length;
633
634       c = str[length];
635       str[length] = '\0';
636       gdk_string_to_compound_text (str, &encoding, &format, &text, &new_length);
637       gtk_selection_data_set (selection_data, encoding, format, text, new_length);
638       gdk_free_compound_text (text);
639       str[length] = c;
640     }
641
642   if (str != editable->clipboard_text)
643     g_free (str);
644 }
645
646 static void
647 gtk_editable_selection_received  (GtkWidget         *widget,
648                                   GtkSelectionData  *selection_data,
649                                   guint              time)
650 {
651   GtkEditable *editable;
652   gint reselect;
653   gint old_pos;
654   gint tmp_pos;
655   enum {INVALID, STRING, CTEXT} type;
656
657   g_return_if_fail (widget != NULL);
658   g_return_if_fail (GTK_IS_EDITABLE (widget));
659
660   editable = GTK_EDITABLE (widget);
661
662   if (selection_data->type == GDK_TARGET_STRING)
663     type = STRING;
664   else if ((selection_data->type == gdk_atom_intern ("COMPOUND_TEXT", FALSE)) ||
665            (selection_data->type == gdk_atom_intern ("TEXT", FALSE)))
666     type = CTEXT;
667   else
668     type = INVALID;
669
670   if (type == INVALID || selection_data->length < 0)
671     {
672     /* avoid infinite loop */
673     if (selection_data->target != GDK_TARGET_STRING)
674       gtk_selection_convert (widget, selection_data->selection,
675                              GDK_TARGET_STRING, time);
676     return;
677   }
678
679   reselect = FALSE;
680
681   if ((editable->selection_start_pos != editable->selection_end_pos) && 
682       (!editable->has_selection || 
683        (selection_data->selection == clipboard_atom)))
684     {
685       reselect = TRUE;
686
687       /* Don't want to call gtk_editable_delete_selection here if we are going
688        * to reclaim the selection to avoid extra server traffic */
689       if (editable->has_selection)
690         {
691           gtk_editable_delete_text (editable,
692                                  MIN (editable->selection_start_pos, editable->selection_end_pos),
693                                  MAX (editable->selection_start_pos, editable->selection_end_pos));
694         }
695       else
696         gtk_editable_delete_selection (editable);
697     }
698
699   tmp_pos = old_pos = editable->current_pos;
700
701   switch (type)
702     {
703     case STRING:
704       selection_data->data[selection_data->length] = 0;
705       gtk_editable_insert_text (editable, (gchar *)selection_data->data,
706                                 strlen ((gchar *)selection_data->data), 
707                                 &tmp_pos);
708       editable->current_pos = tmp_pos;
709       break;
710     case CTEXT:
711       {
712         gchar **list;
713         gint count;
714         gint i;
715
716         count = gdk_text_property_to_text_list (selection_data->type,
717                                                 selection_data->format, 
718                                                 selection_data->data,
719                                                 selection_data->length,
720                                                 &list);
721         for (i=0; i<count; i++) 
722           {
723             gtk_editable_insert_text (editable, list[i], strlen (list[i]), &tmp_pos);
724             editable->current_pos = tmp_pos;
725           }
726         if (count > 0)
727           gdk_free_text_list (list);
728       }
729       break;
730     case INVALID:               /* quiet compiler */
731       break;
732     }
733
734   if (reselect)
735     gtk_editable_set_selection (editable, old_pos, editable->current_pos);
736 }
737
738 void
739 gtk_editable_delete_selection (GtkEditable *editable)
740 {
741   guint start;
742   guint end;
743
744   g_return_if_fail (editable != NULL);
745   g_return_if_fail (GTK_IS_EDITABLE (editable));
746
747   if (!editable->editable)
748     return;
749
750   start = editable->selection_start_pos;
751   end = editable->selection_end_pos;
752
753   editable->selection_start_pos = 0;
754   editable->selection_end_pos = 0;
755
756   if (start != end)
757     gtk_editable_delete_text (editable, MIN (start, end), MAX (start,end));
758
759   if (editable->has_selection)
760     {
761       editable->has_selection = FALSE;
762       if (gdk_selection_owner_get (GDK_SELECTION_PRIMARY) == GTK_WIDGET (editable)->window)
763         gtk_selection_owner_set (NULL, GDK_SELECTION_PRIMARY, GDK_CURRENT_TIME);
764     }
765 }
766
767 void
768 gtk_editable_claim_selection (GtkEditable *editable, 
769                               gboolean  claim, 
770                               guint32   time)
771 {
772   g_return_if_fail (editable != NULL);
773   g_return_if_fail (GTK_IS_EDITABLE (editable));
774   g_return_if_fail (GTK_WIDGET_REALIZED (editable));
775
776   editable->has_selection = FALSE;
777   
778   if (claim)
779     {
780       if (gtk_selection_owner_set (GTK_WIDGET(editable), GDK_SELECTION_PRIMARY, time))
781         editable->has_selection = TRUE;
782     }
783   else
784     {
785       if (gdk_selection_owner_get (GDK_SELECTION_PRIMARY) == 
786           GTK_WIDGET(editable)->window)
787         gtk_selection_owner_set (NULL, GDK_SELECTION_PRIMARY, time);
788     }
789 }
790
791 void
792 gtk_editable_select_region (GtkEditable *editable,
793                             gint         start,
794                             gint         end)
795 {
796   g_return_if_fail (editable != NULL);
797   g_return_if_fail (GTK_IS_EDITABLE (editable));
798   
799   if (GTK_WIDGET_REALIZED (editable))
800     gtk_editable_claim_selection (editable, start != end, GDK_CURRENT_TIME);
801   
802   gtk_editable_set_selection (editable, start, end);
803 }
804
805 /* Get the timestamp of the current event. Actually, the only thing
806  * we really care about below is the key event
807  */
808 static guint32
809 gtk_editable_get_event_time (GtkEditable *editable)
810 {
811   GdkEvent *event;
812   guint32 tm = GDK_CURRENT_TIME;
813   
814   event = gtk_get_current_event();
815   
816   if (event)
817     switch (event->type)
818       {
819       case GDK_MOTION_NOTIFY:
820         tm = event->motion.time; break;
821       case GDK_BUTTON_PRESS:
822       case GDK_2BUTTON_PRESS:
823       case GDK_3BUTTON_PRESS:
824       case GDK_BUTTON_RELEASE:
825         tm = event->button.time; break;
826       case GDK_KEY_PRESS:
827       case GDK_KEY_RELEASE:
828         tm = event->key.time; break;
829       case GDK_ENTER_NOTIFY:
830       case GDK_LEAVE_NOTIFY:
831         tm = event->crossing.time; break;
832       case GDK_PROPERTY_NOTIFY:
833         tm = event->property.time; break;
834       case GDK_SELECTION_CLEAR:
835       case GDK_SELECTION_REQUEST:
836       case GDK_SELECTION_NOTIFY:
837         tm = event->selection.time; break;
838       case GDK_PROXIMITY_IN:
839       case GDK_PROXIMITY_OUT:
840         tm = event->proximity.time; break;
841       default:                  /* use current time */
842         break;
843       }
844   gdk_event_free(event);
845   
846   return tm;
847 }
848
849 void
850 gtk_editable_cut_clipboard (GtkEditable *editable)
851 {
852   g_return_if_fail (editable != NULL);
853   g_return_if_fail (GTK_IS_EDITABLE (editable));
854   
855   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[CUT_CLIPBOARD]);
856 }
857
858 void
859 gtk_editable_copy_clipboard (GtkEditable *editable)
860 {
861   g_return_if_fail (editable != NULL);
862   g_return_if_fail (GTK_IS_EDITABLE (editable));
863   
864   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[COPY_CLIPBOARD]);
865 }
866
867 void
868 gtk_editable_paste_clipboard (GtkEditable *editable)
869 {
870   g_return_if_fail (editable != NULL);
871   g_return_if_fail (GTK_IS_EDITABLE (editable));
872   
873   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[PASTE_CLIPBOARD]);
874 }
875
876 void
877 gtk_editable_set_editable (GtkEditable    *editable,
878                            gboolean        is_editable)
879 {
880   g_return_if_fail (editable != NULL);
881   g_return_if_fail (GTK_IS_EDITABLE (editable));
882   
883   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[SET_EDITABLE], is_editable != FALSE);
884 }
885
886 static void
887 gtk_editable_real_set_editable (GtkEditable    *editable,
888                                 gboolean        is_editable)
889 {
890   g_return_if_fail (editable != NULL);
891   g_return_if_fail (GTK_IS_EDITABLE (editable));
892
893   editable->editable = is_editable != FALSE;
894   gtk_widget_queue_draw (GTK_WIDGET (editable));
895 }
896
897 static void
898 gtk_editable_real_cut_clipboard (GtkEditable *editable)
899 {
900   g_return_if_fail (editable != NULL);
901   g_return_if_fail (GTK_IS_EDITABLE (editable));
902   
903   gtk_editable_real_copy_clipboard (editable);
904   gtk_editable_delete_selection (editable);
905 }
906
907 static void
908 gtk_editable_real_copy_clipboard (GtkEditable *editable)
909 {
910   guint32 time;
911   gint selection_start_pos; 
912   gint selection_end_pos;
913
914   g_return_if_fail (editable != NULL);
915   g_return_if_fail (GTK_IS_EDITABLE (editable));
916   
917   time = gtk_editable_get_event_time (editable);
918   selection_start_pos = MIN (editable->selection_start_pos, editable->selection_end_pos);
919   selection_end_pos = MAX (editable->selection_start_pos, editable->selection_end_pos);
920  
921   if (selection_start_pos != selection_end_pos)
922     {
923       if (gtk_selection_owner_set (GTK_WIDGET (editable),
924                                    clipboard_atom,
925                                    time))
926         editable->clipboard_text = gtk_editable_get_public_chars (editable,
927                                                                   selection_start_pos,
928                                                                   selection_end_pos);
929     }
930 }
931
932 static void
933 gtk_editable_real_paste_clipboard (GtkEditable *editable)
934 {
935   guint32 time;
936
937   g_return_if_fail (editable != NULL);
938   g_return_if_fail (GTK_IS_EDITABLE (editable));
939   
940   time = gtk_editable_get_event_time (editable);
941   if (editable->editable)
942     gtk_selection_convert (GTK_WIDGET(editable), 
943                            clipboard_atom, 
944                            gdk_atom_intern ("COMPOUND_TEXT", FALSE), time);
945 }
946
947 void
948 gtk_editable_changed (GtkEditable *editable)
949 {
950   g_return_if_fail (editable != NULL);
951   g_return_if_fail (GTK_IS_EDITABLE (editable));
952   
953   gtk_signal_emit (GTK_OBJECT (editable), editable_signals[CHANGED]);
954 }
955
956 #if 0
957 static void
958 gtk_editable_parent_set (GtkWidget *widget, 
959                          GtkWidget *old_parent, 
960                          GtkWidget *editable)
961 {
962   GtkWidget *parent;
963
964   parent = old_parent;
965   while (parent)
966     {
967       gtk_signal_disconnect_by_func (GTK_OBJECT (parent),
968                                      GTK_SIGNAL_FUNC (gtk_editable_parent_set),
969                                      editable);
970       parent = parent->parent;
971     }
972
973   parent = widget->parent;
974   while (parent)
975     {
976       gtk_signal_connect (GTK_OBJECT (parent), "parent_set",
977                           GTK_SIGNAL_FUNC (gtk_editable_parent_set), 
978                           editable);
979       
980       parent = parent->parent;
981     }
982 }
983 #endif