renderer: Split mutable geometry and place it directly in GtkKeyboard

Geometry is now permanently married to the widget rather the renderer. While geometry is not always defined, C doesn't support sum types, so checks won't be enforced by the compiler. It's OK to pretend there's always some geometry to avoid crashes.
This commit is contained in:
Dorota Czaplejewicz
2021-03-17 13:56:34 +00:00
parent 46f8790fc0
commit 24c3fac505
4 changed files with 71 additions and 51 deletions

View File

@ -45,6 +45,8 @@
typedef struct _EekGtkKeyboardPrivate typedef struct _EekGtkKeyboardPrivate
{ {
EekRenderer *renderer; // owned, nullable EekRenderer *renderer; // owned, nullable
struct render_geometry render_geometry; // mutable
EekboardContextService *eekboard_context; // unowned reference EekboardContextService *eekboard_context; // unowned reference
struct submission *submission; // unowned reference struct submission *submission; // unowned reference
@ -72,12 +74,23 @@ eek_gtk_keyboard_real_realize (GtkWidget *self)
GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->realize (self); GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->realize (self);
} }
static void set_allocation_size(EekGtkKeyboard *gtk_keyboard,
struct squeek_layout *layout, gdouble width, gdouble height)
{
// This is where size-dependent surfaces would be released
EekGtkKeyboardPrivate *priv =
eek_gtk_keyboard_get_instance_private (gtk_keyboard);
priv->render_geometry = eek_render_geometry_from_allocation_size(
layout, width, height);
}
static gboolean static gboolean
eek_gtk_keyboard_real_draw (GtkWidget *self, eek_gtk_keyboard_real_draw (GtkWidget *self,
cairo_t *cr) cairo_t *cr)
{ {
EekGtkKeyboard *keyboard = EEK_GTK_KEYBOARD (self);
EekGtkKeyboardPrivate *priv = EekGtkKeyboardPrivate *priv =
eek_gtk_keyboard_get_instance_private (EEK_GTK_KEYBOARD (self)); eek_gtk_keyboard_get_instance_private (keyboard);
GtkAllocation allocation; GtkAllocation allocation;
gtk_widget_get_allocation (self, &allocation); gtk_widget_get_allocation (self, &allocation);
@ -92,15 +105,14 @@ eek_gtk_keyboard_real_draw (GtkWidget *self,
priv->keyboard, priv->keyboard,
pcontext); pcontext);
eek_renderer_set_allocation_size (priv->renderer, set_allocation_size (keyboard, priv->keyboard->layout,
priv->keyboard->layout, allocation.width, allocation.height);
allocation.width,
allocation.height);
eek_renderer_set_scale_factor (priv->renderer, eek_renderer_set_scale_factor (priv->renderer,
gtk_widget_get_scale_factor (self)); gtk_widget_get_scale_factor (self));
} }
eek_renderer_render_keyboard (priv->renderer, priv->renderer->widget_to_layout, priv->submission, cr, priv->keyboard); eek_renderer_render_keyboard (priv->renderer, priv->render_geometry,
priv->submission, cr, priv->keyboard);
return FALSE; return FALSE;
} }
@ -117,8 +129,9 @@ static void
eek_gtk_keyboard_real_size_allocate (GtkWidget *self, eek_gtk_keyboard_real_size_allocate (GtkWidget *self,
GtkAllocation *allocation) GtkAllocation *allocation)
{ {
EekGtkKeyboard *keyboard = EEK_GTK_KEYBOARD (self);
EekGtkKeyboardPrivate *priv = EekGtkKeyboardPrivate *priv =
eek_gtk_keyboard_get_instance_private (EEK_GTK_KEYBOARD (self)); eek_gtk_keyboard_get_instance_private (keyboard);
// check if the change would switch types // check if the change would switch types
enum squeek_arrangement_kind new_type = get_type( enum squeek_arrangement_kind new_type = get_type(
(uint32_t)(allocation->width - allocation->x), (uint32_t)(allocation->width - allocation->x),
@ -130,10 +143,8 @@ eek_gtk_keyboard_real_size_allocate (GtkWidget *self,
} }
if (priv->renderer) { if (priv->renderer) {
eek_renderer_set_allocation_size (priv->renderer, set_allocation_size (keyboard, priv->keyboard->layout,
priv->keyboard->layout, allocation->width, allocation->height);
allocation->width,
allocation->height);
} }
GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)-> GTK_WIDGET_CLASS (eek_gtk_keyboard_parent_class)->
@ -162,7 +173,7 @@ static void depress(EekGtkKeyboard *self,
} }
squeek_layout_depress(priv->keyboard->layout, squeek_layout_depress(priv->keyboard->layout,
priv->submission, priv->submission,
x, y, eek_renderer_get_transformation(priv->renderer), time, self); x, y, priv->render_geometry.widget_to_layout, time, self);
} }
static void drag(EekGtkKeyboard *self, static void drag(EekGtkKeyboard *self,
@ -174,7 +185,7 @@ static void drag(EekGtkKeyboard *self,
} }
squeek_layout_drag(eekboard_context_service_get_keyboard(priv->eekboard_context)->layout, squeek_layout_drag(eekboard_context_service_get_keyboard(priv->eekboard_context)->layout,
priv->submission, priv->submission,
x, y, eek_renderer_get_transformation(priv->renderer), time, x, y, priv->render_geometry.widget_to_layout, time,
priv->eekboard_context, self); priv->eekboard_context, self);
} }
@ -185,8 +196,7 @@ static void release(EekGtkKeyboard *self, guint32 time)
return; return;
} }
squeek_layout_release(eekboard_context_service_get_keyboard(priv->eekboard_context)->layout, squeek_layout_release(eekboard_context_service_get_keyboard(priv->eekboard_context)->layout,
priv->submission, priv->submission, priv->render_geometry.widget_to_layout, time,
eek_renderer_get_transformation(priv->renderer), time,
priv->eekboard_context, self); priv->eekboard_context, self);
} }
@ -396,6 +406,24 @@ eek_gtk_keyboard_new (EekboardContextService *eekservice,
priv->submission = submission; priv->submission = submission;
priv->layout = layout; priv->layout = layout;
priv->renderer = NULL; priv->renderer = NULL;
// This should really be done on initialization.
// Before the widget is allocated,
// we don't really know what geometry it takes.
// When it's off the screen, we also kinda don't.
struct render_geometry initial_geometry = {
// Set to 100 just to make sure if there's any attempt to use it,
// it actually gives plausible results instead of blowing up,
// e.g. on zero division.
.allocation_width = 100,
.allocation_height = 100,
.widget_to_layout = {
.origin_x = 0,
.origin_y = 0,
.scale = 1,
},
};
priv->render_geometry = initial_geometry;
g_signal_connect (eekservice, g_signal_connect (eekservice,
"notify::keyboard", "notify::keyboard",
G_CALLBACK(on_notify_keyboard), G_CALLBACK(on_notify_keyboard),

View File

@ -28,6 +28,7 @@
#include <glib.h> #include <glib.h>
#include <gtk/gtk.h> #include <gtk/gtk.h>
#include "eek/eek-renderer.h"
#include "eek/eek-types.h" #include "eek/eek-types.h"
struct submission; struct submission;

View File

@ -194,23 +194,23 @@ render_button_label (cairo_t *cr,
// FIXME: Pass just the active modifiers instead of entire submission // FIXME: Pass just the active modifiers instead of entire submission
void void
eek_renderer_render_keyboard (EekRenderer *self, eek_renderer_render_keyboard (EekRenderer *self,
struct transformation widget_to_layout, struct render_geometry geometry,
struct submission *submission, struct submission *submission,
cairo_t *cr, cairo_t *cr,
LevelKeyboard *keyboard) LevelKeyboard *keyboard)
{ {
g_return_if_fail (self->allocation_width > 0.0); g_return_if_fail (geometry.allocation_width > 0.0);
g_return_if_fail (self->allocation_height > 0.0); g_return_if_fail (geometry.allocation_height > 0.0);
/* Paint the background covering the entire widget area */ /* Paint the background covering the entire widget area */
gtk_render_background (self->view_context, gtk_render_background (self->view_context,
cr, cr,
0, 0, 0, 0,
self->allocation_width, self->allocation_height); geometry.allocation_width, geometry.allocation_height);
cairo_save(cr); cairo_save(cr);
cairo_translate (cr, widget_to_layout.origin_x, widget_to_layout.origin_y); cairo_translate (cr, geometry.widget_to_layout.origin_x, geometry.widget_to_layout.origin_y);
cairo_scale (cr, widget_to_layout.scale, widget_to_layout.scale); cairo_scale (cr, geometry.widget_to_layout.scale, geometry.widget_to_layout.scale);
squeek_draw_layout_base_view(keyboard->layout, self, cr); squeek_draw_layout_base_view(keyboard->layout, self, cr);
squeek_layout_draw_all_changed(keyboard->layout, self, cr, submission); squeek_layout_draw_all_changed(keyboard->layout, self, cr, submission);
@ -262,8 +262,6 @@ static void
renderer_init (EekRenderer *self) renderer_init (EekRenderer *self)
{ {
self->pcontext = NULL; self->pcontext = NULL;
self->allocation_width = 0.0;
self->allocation_height = 0.0;
self->scale_factor = 1; self->scale_factor = 1;
self->css_provider = squeek_load_style(); self->css_provider = squeek_load_style();
@ -310,22 +308,18 @@ eek_renderer_new (LevelKeyboard *keyboard,
return renderer; return renderer;
} }
void struct render_geometry
eek_renderer_set_allocation_size (EekRenderer *renderer, eek_render_geometry_from_allocation_size (struct squeek_layout *layout,
struct squeek_layout *layout,
gdouble width, gdouble width,
gdouble height) gdouble height)
{ {
g_return_if_fail (width > 0.0 && height > 0.0); struct render_geometry ret = {
.allocation_width = width,
renderer->allocation_width = width; .allocation_height = height,
renderer->allocation_height = height; .widget_to_layout = squeek_layout_calculate_transformation(
layout, width, height),
renderer->widget_to_layout = squeek_layout_calculate_transformation( };
layout, return ret;
renderer->allocation_width, renderer->allocation_height);
// This is where size-dependent surfaces would be released
} }
void void
@ -357,8 +351,3 @@ eek_renderer_get_icon_surface (const gchar *icon_name,
} }
return surface; return surface;
} }
struct transformation
eek_renderer_get_transformation (EekRenderer *renderer) {
return renderer->widget_to_layout;
}

View File

@ -41,22 +41,23 @@ typedef struct EekRenderer
gchar *extra_style; // owned gchar *extra_style; // owned
// Mutable state // Mutable state
gint scale_factor; /* the outputs scale factor */
} EekRenderer;
/// Mutable part of the renderer state.
/// TODO: Possibly should include scale factor.
struct render_geometry {
/// Background extents /// Background extents
gdouble allocation_width; gdouble allocation_width;
gdouble allocation_height; gdouble allocation_height;
gint scale_factor; /* the outputs scale factor */
/// Coords transformation /// Coords transformation
struct transformation widget_to_layout; struct transformation widget_to_layout;
} EekRenderer; };
GType eek_renderer_get_type (void) G_GNUC_CONST; GType eek_renderer_get_type (void) G_GNUC_CONST;
EekRenderer *eek_renderer_new (LevelKeyboard *keyboard, EekRenderer *eek_renderer_new (LevelKeyboard *keyboard,
PangoContext *pcontext); PangoContext *pcontext);
void eek_renderer_set_allocation_size
(EekRenderer *renderer, struct squeek_layout *layout,
gdouble width,
gdouble height);
void eek_renderer_set_scale_factor (EekRenderer *renderer, void eek_renderer_set_scale_factor (EekRenderer *renderer,
gint scale); gint scale);
@ -64,13 +65,14 @@ cairo_surface_t *eek_renderer_get_icon_surface(const gchar *icon_name,
gint size, gint size,
gint scale); gint scale);
void eek_renderer_render_keyboard (EekRenderer *renderer, struct transformation widget_to_layout, struct submission *submission, void eek_renderer_render_keyboard (EekRenderer *renderer, struct render_geometry geometry, struct submission *submission,
cairo_t *cr, LevelKeyboard *keyboard); cairo_t *cr, LevelKeyboard *keyboard);
void void
eek_renderer_free (EekRenderer *self); eek_renderer_free (EekRenderer *self);
struct transformation struct render_geometry
eek_renderer_get_transformation (EekRenderer *renderer); eek_render_geometry_from_allocation_size (struct squeek_layout *layout,
gdouble width, gdouble height);
G_END_DECLS G_END_DECLS
#endif /* EEK_RENDERER_H */ #endif /* EEK_RENDERER_H */