applied patch to use a backing pixmap for drawing in GtkEntry, which removes

the bad flickering. (gtk-wille-980106-0)

-Yosh
This commit is contained in:
Manish Singh 1998-01-08 01:58:39 +00:00
parent 52d5af4d7e
commit 4ccc005197
10 changed files with 195 additions and 72 deletions

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -1,3 +1,9 @@
Wed Jan 7 18:06:17 PST 1998 Manish Singh <yosh@gimp.org>
* gtk/gtkentry.c:
* gtk/gtkentry.h: applied patch to use a backing pixmap for
drawing, which removes the bad flickering. (gtk-wille-980106-0)
Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org> Wed Jan 7 02:14:30 PST 1998 Manish Singh <yosh@gimp.org>
* glib/glib.h * glib/glib.h

View File

@ -397,7 +397,8 @@ GList* g_list_prepend (GList *list,
GList* g_list_insert (GList *list, GList* g_list_insert (GList *list,
gpointer data, gpointer data,
gint position); gint position);
GList* g_list_insert_sorted (GList *list, GList* g_list_insert_sorted
(GList *list,
gpointer data, gpointer data,
GCompareFunc func); GCompareFunc func);
GList* g_list_concat (GList *list1, GList* g_list_concat (GList *list1,
@ -433,7 +434,8 @@ GSList* g_slist_prepend (GSList *list,
GSList* g_slist_insert (GSList *list, GSList* g_slist_insert (GSList *list,
gpointer data, gpointer data,
gint position); gint position);
GSList* g_slist_insert_sorted (GSList *list, GSList* g_slist_insert_sorted
(GSList *list,
gpointer data, gpointer data,
GCompareFunc func); GCompareFunc func);
GSList* g_slist_concat (GSList *list1, GSList* g_slist_concat (GSList *list1,

View File

@ -72,6 +72,8 @@ static void gtk_entry_size_request (GtkWidget *widget,
GtkRequisition *requisition); GtkRequisition *requisition);
static void gtk_entry_size_allocate (GtkWidget *widget, static void gtk_entry_size_allocate (GtkWidget *widget,
GtkAllocation *allocation); GtkAllocation *allocation);
static void gtk_entry_make_backing_pixmap (GtkEntry *entry,
gint width, gint height);
static void gtk_entry_draw (GtkWidget *widget, static void gtk_entry_draw (GtkWidget *widget,
GdkRectangle *area); GdkRectangle *area);
static gint gtk_entry_expose (GtkWidget *widget, static gint gtk_entry_expose (GtkWidget *widget,
@ -97,6 +99,9 @@ static void gtk_entry_selection_received (GtkWidget *widget,
GtkSelectionData *selection_data); GtkSelectionData *selection_data);
static void gtk_entry_draw_text (GtkEntry *entry); static void gtk_entry_draw_text (GtkEntry *entry);
static void gtk_entry_draw_cursor (GtkEntry *entry); static void gtk_entry_draw_cursor (GtkEntry *entry);
static void gtk_entry_draw_cursor_on_drawable
(GtkEntry *entry,
GdkDrawable *drawable);
static void gtk_entry_queue_draw (GtkEntry *entry); static void gtk_entry_queue_draw (GtkEntry *entry);
static gint gtk_entry_timer (gpointer data); static gint gtk_entry_timer (gpointer data);
static gint gtk_entry_position (GtkEntry *entry, static gint gtk_entry_position (GtkEntry *entry,
@ -314,6 +319,7 @@ gtk_entry_init (GtkEntry *entry)
GTK_WIDGET_SET_FLAGS (entry, GTK_CAN_FOCUS); GTK_WIDGET_SET_FLAGS (entry, GTK_CAN_FOCUS);
entry->text_area = NULL; entry->text_area = NULL;
entry->backing_pixmap = NULL;
entry->text = NULL; entry->text = NULL;
entry->text_size = 0; entry->text_size = 0;
entry->text_length = 0; entry->text_length = 0;
@ -538,6 +544,9 @@ gtk_entry_destroy (GtkObject *object)
g_free (entry->text); g_free (entry->text);
entry->text = NULL; entry->text = NULL;
if (entry->backing_pixmap)
gdk_pixmap_unref (entry->backing_pixmap);
if (GTK_OBJECT_CLASS (parent_class)->destroy) if (GTK_OBJECT_CLASS (parent_class)->destroy)
(* GTK_OBJECT_CLASS (parent_class)->destroy) (object); (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
} }
@ -1034,9 +1043,7 @@ gtk_entry_key_press (GtkWidget *widget,
if (event->state & GDK_CONTROL_MASK) if (event->state & GDK_CONTROL_MASK)
gtk_delete_line (entry); gtk_delete_line (entry);
else if (event->state & GDK_SHIFT_MASK) else if (event->state & GDK_SHIFT_MASK)
{
gtk_entry_cut_clipboard (entry, event); gtk_entry_cut_clipboard (entry, event);
}
else else
gtk_delete_forward_character (entry); gtk_delete_forward_character (entry);
} }
@ -1372,6 +1379,33 @@ gtk_entry_selection_received (GtkWidget *widget,
gtk_entry_queue_draw (entry); gtk_entry_queue_draw (entry);
} }
static void
gtk_entry_make_backing_pixmap (GtkEntry *entry, gint width, gint height)
{
gint pixmap_width, pixmap_height;
if (!entry->backing_pixmap)
{
/* allocate */
entry->backing_pixmap = gdk_pixmap_new (entry->text_area,
width, height,
-1);
}
else
{
/* reallocate if sizes don't match */
gdk_window_get_size (entry->backing_pixmap,
&pixmap_width, &pixmap_height);
if ((pixmap_width != width) || (pixmap_height != height))
{
gdk_pixmap_unref (entry->backing_pixmap);
entry->backing_pixmap = gdk_pixmap_new (entry->text_area,
width, height,
-1);
}
}
}
static void static void
gtk_entry_draw_text (GtkEntry *entry) gtk_entry_draw_text (GtkEntry *entry)
{ {
@ -1383,6 +1417,8 @@ gtk_entry_draw_text (GtkEntry *entry)
gint selection_end_xoffset; gint selection_end_xoffset;
gint width, height; gint width, height;
gint y; gint y;
GdkDrawable *drawable;
gint use_backing_pixmap;
g_return_if_fail (entry != NULL); g_return_if_fail (entry != NULL);
g_return_if_fail (GTK_IS_ENTRY (entry)); g_return_if_fail (GTK_IS_ENTRY (entry));
@ -1403,11 +1439,39 @@ gtk_entry_draw_text (GtkEntry *entry)
{ {
widget = GTK_WIDGET (entry); widget = GTK_WIDGET (entry);
gdk_window_clear (entry->text_area); if (!entry->text)
if (entry->text)
{ {
gdk_window_clear (entry->text_area);
if (entry->editable)
gtk_entry_draw_cursor (entry);
return;
}
gdk_window_get_size (entry->text_area, &width, &height); gdk_window_get_size (entry->text_area, &width, &height);
/*
If the widget has focus, draw on a backing pixmap to avoid flickering
and copy it to the text_area.
Otherwise draw to text_area directly for better speed.
*/
use_backing_pixmap = GTK_WIDGET_HAS_FOCUS (widget) && (entry->text != NULL);
if (use_backing_pixmap)
{
gtk_entry_make_backing_pixmap (entry, width, height);
drawable = entry->backing_pixmap;
gdk_draw_rectangle (drawable,
widget->style->white_gc,
TRUE,
0, 0,
width,
height);
}
else
{
drawable = entry->text_area;
gdk_window_clear (entry->text_area);
}
y = (height - (widget->style->font->ascent + widget->style->font->descent)) / 2; y = (height - (widget->style->font->ascent + widget->style->font->descent)) / 2;
y += widget->style->font->ascent; y += widget->style->font->ascent;
@ -1428,12 +1492,12 @@ gtk_entry_draw_text (GtkEntry *entry)
selection_end_pos); selection_end_pos);
if (selection_start_pos > 0) if (selection_start_pos > 0)
gdk_draw_text (entry->text_area, widget->style->font, gdk_draw_text (drawable, widget->style->font,
widget->style->fg_gc[GTK_STATE_NORMAL], widget->style->fg_gc[GTK_STATE_NORMAL],
-entry->scroll_offset, y, -entry->scroll_offset, y,
entry->text, selection_start_pos); entry->text, selection_start_pos);
gdk_draw_rectangle (entry->text_area, gdk_draw_rectangle (drawable,
widget->style->bg_gc[selected_state], widget->style->bg_gc[selected_state],
TRUE, TRUE,
-entry->scroll_offset + selection_start_xoffset, -entry->scroll_offset + selection_start_xoffset,
@ -1441,14 +1505,14 @@ gtk_entry_draw_text (GtkEntry *entry)
selection_end_xoffset - selection_start_xoffset, selection_end_xoffset - selection_start_xoffset,
-1); -1);
gdk_draw_text (entry->text_area, widget->style->font, gdk_draw_text (drawable, widget->style->font,
widget->style->fg_gc[selected_state], widget->style->fg_gc[selected_state],
-entry->scroll_offset + selection_start_xoffset, y, -entry->scroll_offset + selection_start_xoffset, y,
entry->text + selection_start_pos, entry->text + selection_start_pos,
selection_end_pos - selection_start_pos); selection_end_pos - selection_start_pos);
if (selection_end_pos < entry->text_length) if (selection_end_pos < entry->text_length)
gdk_draw_string (entry->text_area, widget->style->font, gdk_draw_string (drawable, widget->style->font,
widget->style->fg_gc[GTK_STATE_NORMAL], widget->style->fg_gc[GTK_STATE_NORMAL],
-entry->scroll_offset + selection_end_xoffset, y, -entry->scroll_offset + selection_end_xoffset, y,
entry->text + selection_end_pos); entry->text + selection_end_pos);
@ -1458,20 +1522,34 @@ gtk_entry_draw_text (GtkEntry *entry)
GdkGCValues values; GdkGCValues values;
gdk_gc_get_values (widget->style->fg_gc[GTK_STATE_NORMAL], &values); gdk_gc_get_values (widget->style->fg_gc[GTK_STATE_NORMAL], &values);
gdk_draw_string (entry->text_area, widget->style->font, gdk_draw_string (drawable, widget->style->font,
widget->style->fg_gc[GTK_STATE_NORMAL], widget->style->fg_gc[GTK_STATE_NORMAL],
-entry->scroll_offset, y, -entry->scroll_offset, y,
entry->text); entry->text);
} }
}
if(entry->editable) if (entry->editable)
gtk_entry_draw_cursor (entry); gtk_entry_draw_cursor_on_drawable (entry, drawable);
if (use_backing_pixmap)
gdk_draw_pixmap(entry->text_area,
widget->style->fg_gc[GTK_STATE_NORMAL],
entry->backing_pixmap,
0, 0, 0, 0, width, height);
} }
} }
static void static void
gtk_entry_draw_cursor (GtkEntry *entry) gtk_entry_draw_cursor (GtkEntry *entry)
{
g_return_if_fail (entry != NULL);
g_return_if_fail (GTK_IS_ENTRY (entry));
gtk_entry_draw_cursor_on_drawable (entry, entry->text_area);
}
static void
gtk_entry_draw_cursor_on_drawable (GtkEntry *entry, GdkDrawable *drawable)
{ {
GtkWidget *widget; GtkWidget *widget;
GdkGC *gc; GdkGC *gc;
@ -1498,7 +1576,7 @@ gtk_entry_draw_cursor (GtkEntry *entry)
gc = widget->style->white_gc; gc = widget->style->white_gc;
gdk_window_get_size (entry->text_area, NULL, &text_area_height); gdk_window_get_size (entry->text_area, NULL, &text_area_height);
gdk_draw_line (entry->text_area, gc, xoffset, 0, xoffset, text_area_height); gdk_draw_line (drawable, gc, xoffset, 0, xoffset, text_area_height);
#ifdef USE_XIM #ifdef USE_XIM
if (gdk_im_ready() && entry->ic && if (gdk_im_ready() && entry->ic &&
gdk_ic_get_style (entry->ic) & GdkIMPreeditPosition) gdk_ic_get_style (entry->ic) & GdkIMPreeditPosition)

View File

@ -41,6 +41,7 @@ struct _GtkEntry
GtkWidget widget; GtkWidget widget;
GdkWindow *text_area; GdkWindow *text_area;
GdkPixmap *backing_pixmap;
gchar *text; gchar *text;
guint16 text_size; guint16 text_size;