aboutsummaryrefslogtreecommitdiffstats
path: root/char-table.c
diff options
context:
space:
mode:
Diffstat (limited to 'char-table.c')
-rw-r--r--char-table.c330
1 files changed, 255 insertions, 75 deletions
diff --git a/char-table.c b/char-table.c
index 076ef8f..9a4c925 100644
--- a/char-table.c
+++ b/char-table.c
@@ -1,15 +1,11 @@
/* See LICENSE file for copyright and license details. */
#include "common.h"
-#define FIRST_CHARACTER 0x0000u
-#define LAST_CHARACTER 0xFFFFu
#define REFERENCE_CHARACTER "X"
#define CELL_WIDTH_FACTOR 4
#define CELL_HEIGHT_FACTOR 2
#define CELL_X_CLEARANCE 2
#define CELL_Y_CLEARANCE 2
-#define BASE_FONT_FAMILY "Sans"
-#define BASE_FONT_SIZE 22
G_DEFINE_TYPE(CharTable, gcmap_char_table, GTK_TYPE_DRAWING_AREA)
@@ -34,17 +30,11 @@ is_printable_character(gunichar ch)
static void
format_character(gunichar ch, char *buf)
{
- if (is_printable_character(ch))
- buf[g_unichar_to_utf8(ch, buf)] = '\0';
-#ifdef TODO /* this part is just for testing */
- else if (ch == 0x7Fu)
- stpcpy(buf, "DELETE");
- else
- stpcpy(buf, "X");
-#else
- else
+ if (!is_printable_character(ch)) {
buf[0u] = '\0';
-#endif
+ return;
+ }
+ buf[g_unichar_to_utf8(ch, buf)] = '\0';
}
@@ -67,8 +57,8 @@ fit_text(CharTable *this, const char *text, int *text_w_out, int *text_h_out)
/* Set text to measure */
pango_layout_set_text(this->layout, text, -1);
- /* Try default size */
- apply_font_size(this, BASE_FONT_SIZE, text_w_out, text_h_out);
+ /* Try desired size */
+ apply_font_size(this, this->font_size, text_w_out, text_h_out);
if (*text_w_out <= this->max_w && *text_h_out <= this->max_h)
return;
@@ -76,7 +66,7 @@ fit_text(CharTable *this, const char *text, int *text_w_out, int *text_h_out)
scale_x = (double)this->max_w / (double)*text_w_out;
scale_y = (double)this->max_h / (double)*text_h_out;
scale = MIN(scale_x, scale_y);
- scale = MIN(scale, 1) * BASE_FONT_SIZE;
+ scale = MIN(scale, 1) * this->font_size;
size = (int)scale;
size = MAX(size, 1);
@@ -95,7 +85,7 @@ fit_text(CharTable *this, const char *text, int *text_w_out, int *text_h_out)
do {
ok_size++;
start:
- if (ok_size == BASE_FONT_SIZE)
+ if (ok_size == this->font_size)
break;
apply_font_size(this, ok_size + 1, text_w_out, text_h_out);
} while (*text_w_out < this->max_w && *text_h_out < this->max_h);
@@ -107,27 +97,30 @@ fit_text(CharTable *this, const char *text, int *text_w_out, int *text_h_out)
}
-static int
+static gboolean
gcmap_char_table_expose(GtkWidget *widget, GdkEventExpose *event)
{
- CharTable *this = (CharTable *)widget;
+ CharTable *this = GCMAP_CHAR_TABLE(widget);
GdkRectangle area = event->area;
- int r0 = MAX(area.y * this->nrows / widget->allocation.height, 0);
- int c0 = MAX(area.x * this->ncols / widget->allocation.width, 0);
- int r1 = MIN((area.y + area.height - 1) * this->nrows / widget->allocation.height + 1, this->nrows);
- int c1 = MIN((area.x + area.width - 1) * this->ncols / widget->allocation.width + 1, this->ncols);
+ int r0 = MAX(area.y * this->nrows / this->viewport_h, 0);
+ int c0 = MAX(area.x * this->ncols / this->viewport_w, 0);
+ int r1 = MIN((area.y + area.height - 1) * this->nrows / this->viewport_h + 1, r0 + this->nrows);
+ int c1 = MIN((area.x + area.width - 1) * this->ncols / this->viewport_w + 1, c0 + this->ncols);
GtkStyle *style = gtk_widget_get_style(widget);
cairo_t *g;
char text[16];
int r, c, x0, x1, y0, y1;
int text_w, text_h;
- size_t index, index1;
- gunichar ch;
+ size_t skip, range, row_skip;
+ uint32_t cp;
g = gdk_cairo_create(widget->window);
if (!g)
eprintf("gdk_cairo_create:");
+ /* Make sure it still looks good even if starting in the middle of row */
+ r1 += 1;
+
/* Configure drawing area */
gdk_cairo_rectangle(g, &area);
cairo_clip(g);
@@ -137,44 +130,61 @@ gcmap_char_table_expose(GtkWidget *widget, GdkEventExpose *event)
cairo_paint(g);
/* Draw grid lines */
- set_source_colour_blend(g, &style->text[GTK_STATE_NORMAL], 0.5, &style->base[GTK_STATE_NORMAL]);
+ set_source_colour_blend(g, &style->base[GTK_STATE_NORMAL], 0.10, &style->text[GTK_STATE_NORMAL]);
cairo_set_line_width(g, 1.0);
/* horizontal */
- y0 = widget->allocation.height * (r0 + 1) / this->nrows;
- x0 = widget->allocation.width * c0 / this->ncols;
- x1 = widget->allocation.width * c1 / this->ncols;
- for (r = r0 + 1; r < r1; r++, y0 = y1) {
- y1 = widget->allocation.height * (r + 1) / this->nrows;
+ y0 = this->viewport_h * (r0 + 1) / this->nrows;
+ x0 = this->viewport_w * c0 / this->ncols;
+ x1 = this->viewport_w * c1 / this->ncols;
+ for (r = r0 + (area.y > 0); r < r1; r++, y0 = y1) {
+ y1 = this->viewport_h * (r + 1) / this->nrows;
cairo_move_to(g, x0, y0 + 0.5);
cairo_line_to(g, x1, y0 + 0.5);
cairo_stroke(g);
}
/* vertical */
- x0 = widget->allocation.width * (c0 + 1) / this->ncols;
- y0 = widget->allocation.height * r0 / this->nrows;
- y1 = widget->allocation.height * r1 / this->nrows;
- for (c = c0 + 1; c < c1; c++, x0 = x1) {
- x1 = widget->allocation.width * (c + 1) / this->ncols;
+ x0 = this->viewport_w * (c0 + 1) / this->ncols;
+ y0 = this->viewport_h * r0 / this->nrows;
+ y1 = this->viewport_h * r1 / this->nrows;
+ for (c = c0 + (area.x > 0); c < c1; c++, x0 = x1) {
+ x1 = this->viewport_w * (c + 1) / this->ncols;
cairo_move_to(g, x0 + 0.5, y0);
cairo_line_to(g, x0 + 0.5, y1);
cairo_stroke(g);
}
/* Draw gryphs */
+ if (this->nranges == 0u)
+ goto out;
set_source_colour(g, &style->text[GTK_STATE_NORMAL]);
- index = (size_t)r0 * (size_t)this->ncols + (size_t)c0;
- y0 = widget->allocation.height * r0 / this->nrows;
- for (r = r0; r < r1; r++, y0 = y1, index = index1) {
- index1 = index + (size_t)this->ncols;
- x0 = widget->allocation.width * c0 / this->ncols;
- y1 = widget->allocation.height * (r + 1) / this->nrows;
- for (c = c0; c < c1; c++, index++, x0 = x1) {
- ch = FIRST_CHARACTER + (gunichar)index;
- if (ch > LAST_CHARACTER)
- goto out;
- x1 = widget->allocation.width * (c + 1) / this->ncols;
+ y0 = this->viewport_h * r0 / this->nrows;
+ range = 0u;
+ cp = this->ranges[range].first;
+ skip = (size_t)r0 * (size_t)this->ncols + (size_t)c0;
+ row_skip = (size_t)this->ncols - (size_t)(c1 - c0);
+ for (r = r0; r < r1; r++, y0 = y1, skip = row_skip) {
+ while (skip) {
+ size_t max_jump = (size_t)(this->ranges[range].last - cp + 1u);
+ size_t jump = MIN(skip, max_jump);
+ cp += (uint32_t)jump;
+ skip -= jump;
+ if (cp > this->ranges[range].last) {
+ if (++range == this->nranges)
+ goto out;
+ cp = this->ranges[range].first;
+ }
+ }
+ x0 = this->viewport_w * c0 / this->ncols;
+ y1 = this->viewport_h * (r + 1) / this->nrows;
+ for (c = c0; c < c1; c++, cp++, x0 = x1) {
+ if (cp > this->ranges[range].last) {
+ if (++range == this->nranges)
+ goto out;
+ cp = this->ranges[range].first;
+ }
+ x1 = this->viewport_w * (c + 1) / this->ncols;
- format_character(ch, text);
+ format_character((gunichar)cp, text);
fit_text(this, text, &text_w, &text_h);
cairo_move_to(g, (x0 + x1 - text_w) / 2, (y0 + y1 - text_h) / 2);
pango_cairo_show_layout(g, this->layout);
@@ -183,53 +193,122 @@ gcmap_char_table_expose(GtkWidget *widget, GdkEventExpose *event)
out:
cairo_destroy(g);
- return 0;
+ return FALSE;
}
-static void
-gcmap_char_table_size_allocate(GtkWidget *widget, GtkAllocation *allocation)
+static int
+update_cell_size(GtkWidget *widget, int ranges_changed)
{
- CharTable *this = (CharTable *)widget;
+ CharTable *this = GCMAP_CHAR_TABLE(widget);
int text_w, text_h;
int cell_w, cell_h;
+ uint32_t nchars;
+ int table_h;
+ size_t i;
- GTK_WIDGET_CLASS(gcmap_char_table_parent_class)->size_allocate(widget, allocation);
+ int old_ncols = this->ncols;
+ int old_cell_h = this->cell_h;
pango_layout_set_text(this->layout, REFERENCE_CHARACTER, -1);
- apply_font_size(this, BASE_FONT_SIZE, &text_w, &text_h);
+
+ pango_font_description_set_weight(this->font, PANGO_WEIGHT_NORMAL);
+ pango_font_description_set_style(this->font, PANGO_STYLE_NORMAL);
+ apply_font_size(this, this->font_size, &text_w, &text_h);
+ pango_font_description_set_weight(this->font, this->font_weight);
+ pango_font_description_set_style(this->font, this->font_style);
+
cell_w = MAX(text_w, 1) * CELL_WIDTH_FACTOR;
cell_h = MAX(text_h, 1) * CELL_HEIGHT_FACTOR;
- this->ncols = MAX(widget->allocation.width / cell_w, 1);
- this->nrows = MAX(widget->allocation.height / cell_h, 1);
- cell_w = widget->allocation.width / this->ncols - 1;
- cell_h = widget->allocation.height / this->nrows - 1;
+ this->ncols = MAX(this->viewport_w / cell_w, 1);
+ this->nrows = MAX(this->viewport_h / cell_h, 1);
+ cell_w = MAX(this->viewport_w / this->ncols - 1, 1);
+ cell_h = MAX(this->viewport_h / this->nrows - 1, 1);
this->max_w = cell_w - CELL_X_CLEARANCE * 2;
this->max_h = cell_h - CELL_Y_CLEARANCE * 2;
this->max_w = this->max_w < 1 ? cell_w : this->max_w;
this->max_h = this->max_h < 1 ? cell_h : this->max_h;
+ this->cell_h = cell_h;
+
+ if (old_ncols == this->ncols && old_cell_h == this->cell_h && !ranges_changed)
+ return 0;
+
+ nchars = (uint32_t)this->nranges;
+ for (i = 0u; i < this->nranges; i++)
+ nchars += this->ranges[i].last - this->ranges[i].first;
+ this->nchars = nchars;
+ table_h = (int)(nchars / (uint32_t)this->ncols);
+ if (nchars % (uint32_t)this->ncols)
+ table_h += 1;
+ table_h = table_h > INT_MAX / this->cell_h ? INT_MAX : table_h * this->cell_h;
+ if (this->table_h == table_h)
+ return 0;
+ this->table_h = table_h;
+ return 1;
+}
+
+
+static void
+gcmap_char_table_size_request(GtkWidget *widget, GtkRequisition *requisition)
+{
+ CharTable *this = GCMAP_CHAR_TABLE(widget);
+ requisition->width = 1;
+ requisition->height = this->table_h;
}
+static gboolean
+gcmap_char_table_button_press(GtkWidget *widget, GdkEventButton *event)
+{
+ (void) event;
+ gtk_widget_grab_focus(widget);
+ return FALSE;
+}
static void
gcmap_char_table_init(CharTable *this)
{
- this->nrows = 1; /* will be set by gcmap_char_table_size_allocate */
- this->ncols = 1; /* will be set by gcmap_char_table_size_allocate */
+ GtkWidget *widget = GTK_WIDGET(this);
- this->font = pango_font_description_new();
- if (!this->font)
- eprintf("pango_font_description_new:");
- pango_font_description_set_family(this->font, BASE_FONT_FAMILY);
- pango_font_description_set_size(this->font, BASE_FONT_SIZE * PANGO_SCALE);
- pango_font_description_set_weight(this->font, PANGO_WEIGHT_NORMAL);
+ this->nrows = 1;
+ this->ncols = 1;
+ this->max_w = 1;
+ this->max_h = 1;
+ this->cell_h = 1;
+ this->table_h = 1;
+ this->nchars = 0u;
+ this->viewport_w = 1;
+ this->viewport_h = 1;
- this->layout = gtk_widget_create_pango_layout((GtkWidget *)this, "");
- if (!this->layout)
- eprintf("gtk_widget_create_pango_layout:");
+ GTK_WIDGET_SET_FLAGS(widget, GTK_CAN_FOCUS);
+ gtk_widget_set_sensitive(widget, TRUE);
+
+ gtk_widget_add_events(widget,
+ GDK_BUTTON_PRESS_MASK |
+ GDK_KEY_PRESS_MASK |
+ GDK_FOCUS_CHANGE_MASK);
+
+ g_signal_connect(widget, "button-press-event",
+ G_CALLBACK(&gcmap_char_table_button_press), NULL);
+
+ /* TODO scrolling with control held should change font size (attempt to keep hover glyph in place) */
+}
+
+
+static void
+gcmap_char_table_dispose(GObject *object)
+{
+ CharTable *this = GCMAP_CHAR_TABLE(object);
+ g_clear_object(&this->layout);
+ G_OBJECT_CLASS(gcmap_char_table_parent_class)->dispose(object);
+}
- /* TODO on tear down: pango_font_description_free(this->font); */
- /* TODO on tear down: g_object_unref(this->layout); */
+
+static void
+gcmap_char_table_finalize(GObject *object)
+{
+ CharTable *this = GCMAP_CHAR_TABLE(object);
+ pango_font_description_free(this->font);
+ G_OBJECT_CLASS(gcmap_char_table_parent_class)->finalize(object);
}
@@ -237,13 +316,114 @@ static void
gcmap_char_table_class_init(CharTableClass *class)
{
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(class);
+ GObjectClass *object_class = G_OBJECT_CLASS(class);
+ object_class->dispose = &gcmap_char_table_dispose;
+ object_class->finalize = &gcmap_char_table_finalize;
widget_class->expose_event = &gcmap_char_table_expose;
- widget_class->size_allocate = &gcmap_char_table_size_allocate;
+ widget_class->size_request = &gcmap_char_table_size_request;
+}
+
+
+void
+gcmap_char_table_set_font_family(CharTable *this, const char *family)
+{
+ pango_font_description_set_family(this->font, family);
+ pango_layout_set_font_description(this->layout, this->font);
+ pango_layout_context_changed(this->layout);
+ gtk_widget_queue_resize(GTK_WIDGET(this));
+ gtk_widget_queue_draw(GTK_WIDGET(this));
+}
+
+
+void
+gcmap_char_table_set_font_weight(CharTable *this, PangoWeight weight)
+{
+ if (this->font_weight == weight)
+ return;
+ this->font_weight = weight;
+ pango_font_description_set_weight(this->font, this->font_weight);
+ pango_layout_set_font_description(this->layout, this->font);
+ pango_layout_context_changed(this->layout);
+ gtk_widget_queue_draw(GTK_WIDGET(this));
+}
+
+
+void
+gcmap_char_table_set_font_style(CharTable *this, PangoStyle style)
+{
+ if (this->font_style == style)
+ return;
+ this->font_style = style;
+ pango_font_description_set_style(this->font, this->font_style);
+ pango_layout_set_font_description(this->layout, this->font);
+ pango_layout_context_changed(this->layout);
+ gtk_widget_queue_draw(GTK_WIDGET(this));
+}
+
+
+void
+gcmap_char_table_set_font_size(CharTable *this, int size)
+{
+ GtkWidget *widget;
+ if (this->font_size == MAX(size, 1))
+ return;
+ this->font_size = MAX(size, 1);
+ pango_font_description_set_size(this->font, this->font_size * PANGO_SCALE);
+ pango_layout_set_font_description(this->layout, this->font);
+ pango_layout_context_changed(this->layout);
+ widget = GTK_WIDGET(this);
+ gtk_widget_queue_resize(widget);
+ gtk_widget_queue_draw(widget);
+}
+
+
+void
+gcmap_char_table_set_ranges(CharTable *this, const struct libcmap_range *ranges, size_t nranges)
+{
+ GtkWidget *widget;
+ if (this->ranges == ranges && this->nranges == nranges)
+ return;
+ this->ranges = ranges;
+ this->nranges = nranges;
+ widget = GTK_WIDGET(this);
+ if (update_cell_size(widget, 1))
+ gtk_widget_queue_resize(widget);
+ gtk_widget_queue_draw(widget);
+}
+
+
+void
+gcmap_char_table_viewport_updated(CharTable *this, int viewport_w, int viewport_h)
+{
+ this->viewport_w = viewport_w;
+ this->viewport_h = viewport_h;
+ update_cell_size(GTK_WIDGET(this), 0);
}
GtkWidget *
-gcmap_char_table_new(void)
+gcmap_char_table_new(const char *font_family, int font_size, PangoWeight font_weight, PangoStyle font_style)
{
- return g_object_new(GCMAP_TYPE_CHAR_TABLE, NULL);
+ CharTable *this = g_object_new(GCMAP_TYPE_CHAR_TABLE, NULL);
+ GtkWidget *widget = GTK_WIDGET(this);
+
+ this->font_size = font_size;
+ this->font_weight = font_weight;
+ this->font_style = font_style;
+ this->ranges = NULL;
+ this->nranges = 0u;
+
+ this->font = pango_font_description_new();
+ if (!this->font)
+ eprintf("pango_font_description_new:");
+ pango_font_description_set_family(this->font, font_family);
+ pango_font_description_set_size(this->font, this->font_size * PANGO_SCALE);
+ pango_font_description_set_weight(this->font, this->font_weight);
+ pango_font_description_set_style(this->font, this->font_style);
+
+ this->layout = gtk_widget_create_pango_layout(widget, "");
+ if (!this->layout)
+ eprintf("gtk_widget_create_pango_layout:");
+
+ return widget;
}