/* * gui-over.c - GUI: overlays * * Written 2016 by Werner Almesberger * Copyright 2016 by Werner Almesberger * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. */ /* * Resources: * * http://zetcode.com/gfx/cairo/cairobackends/ * https://developer.gnome.org/gtk3/stable/gtk-migrating-2-to-3.html * https://www.cairographics.org/samples/rounded_rectangle/ * * Section "Description" in * https://developer.gnome.org/pango/stable/pango-Cairo-Rendering.html */ #include #include #include #include #include #include #include #include "util.h" #include "fmt-pango.h" #include "gui-aoi.h" #include "gui-over.h" #define OVER_BORDER 8 #define OVER_RADIUS 6 #define OVER_SEP 8 struct overlay { const char *s; struct overlay_style style; struct aoi **aois; bool (*hover)(void *user, bool on); void (*click)(void *user); void *user; struct aoi *aoi; struct overlay *next; }; #define NORMAL_FONT "Helvetica 10" #define BOLD_FONT "Helvetica Bold 10" #define NORMAL_PAD 8 #define NORMAL_RADIUS 6 #define NORMAL_SKIP 8 #define NORMAL_WIDTH 2 #define DENSE_PAD 4 #define DENSE_RADIUS 3 #define DENSE_SKIP 5 #define DENSE_WIDTH 1 #define BG_STANDARD { 0.8, 0.9, 1.0, 0.8 } #define FG_STANDARD { 0.0, 0.0, 0.0, 1.0 } #define FRAME_STANDARD { 0.5, 0.5, 1.0, 0.7 } #define BG_SELECTED BG_STANDARD #define FG_SELECTED FG_STANDARD #define FRAME_SELECTED { 0.0, 0.0, 1.0, 0.8 } #define BG_DIFF_NEW BG_STANDARD #define FG_DIFF_NEW { 0.0, 0.6, 0.0, 1.0 } #define FRAME_DIFF_NEW FRAME_STANDARD #define BG_DIFF_NEW_SELECTED BG_DIFF_NEW #define FG_DIFF_NEW_SELECTED FG_DIFF_NEW #define FRAME_DIFF_NEW_SELECTED FRAME_SELECTED #define BG_DIFF_OLD BG_STANDARD #define FG_DIFF_OLD { 0.8, 0.0, 0.0, 1.0 } #define FRAME_DIFF_OLD FRAME_STANDARD #define BG_DIFF_OLD_SELECTED BG_DIFF_OLD #define FG_DIFF_OLD_SELECTED FG_DIFF_OLD #define FRAME_DIFF_OLD_SELECTED FRAME_SELECTED #define BOX_ATTRS(style) \ .pad = style##_PAD, \ .radius = style##_RADIUS, \ .skip = style##_SKIP, \ .width = style##_WIDTH #define NORMAL BOX_ATTRS(NORMAL) #define DENSE BOX_ATTRS(DENSE) #define COLOR_ATTRS(style) \ .bg = BG_##style, \ .fg = FG_##style, \ .frame = FRAME_##style #define STANDARD COLOR_ATTRS(STANDARD) #define SELECTED COLOR_ATTRS(SELECTED) #define DIFF_NEW COLOR_ATTRS(DIFF_NEW) #define DIFF_NEW_SELECTED COLOR_ATTRS(DIFF_NEW_SELECTED) #define DIFF_OLD COLOR_ATTRS(DIFF_OLD) #define DIFF_OLD_SELECTED COLOR_ATTRS(DIFF_OLD_SELECTED) struct overlay_style overlay_style_default = { .font = NORMAL_FONT, NORMAL, STANDARD, }, overlay_style_dense = { .font = NORMAL_FONT, DENSE, STANDARD, }, overlay_style_dense_selected = { .font = BOLD_FONT, DENSE, SELECTED, }, overlay_style_diff_new = { .font = NORMAL_FONT, NORMAL, DIFF_NEW, }, overlay_style_diff_old = { .font = NORMAL_FONT, NORMAL, DIFF_OLD, }, overlay_style_dense_diff_new = { .font = BOLD_FONT, DENSE, DIFF_NEW_SELECTED, }, overlay_style_dense_diff_old = { .font = BOLD_FONT, DENSE, DIFF_OLD_SELECTED, }; static void rrect(cairo_t *cr, int x, int y, int w, int h, int r) { const double deg = M_PI / 180.0; cairo_new_path(cr); cairo_arc(cr, x + w - r, y + r, r, -90 * deg, 0); cairo_arc(cr, x + w - r, y + h - r, r, 0, 90 * deg); cairo_arc(cr, x + r, y + h - r, r, 90 * deg, 180 * deg); cairo_arc(cr, x + r, y + r, r, 180 * deg, 270 * deg); cairo_close_path(cr); } struct overlay *overlay_draw(struct overlay *over, cairo_t *cr, int *x, int *y) { const struct overlay_style *style = &over->style; const double *fg = style->fg; const double *bg = style->bg; const double *frame = style->frame; unsigned ink_w, ink_h; /* effectively used text area size */ unsigned w, h; /* box size */ int tx, ty; /* text start position */ int sx, sy; /* box start position */ PangoLayout *layout; PangoFontDescription *desc; PangoRectangle ink_rect; desc = pango_font_description_from_string(style->font); layout = pango_cairo_create_layout(cr); pango_layout_set_font_description(layout, desc); pango_layout_set_markup(layout, over->s, -1); pango_font_description_free(desc); pango_layout_get_extents(layout, &ink_rect, NULL); #if 0 fprintf(stderr, "%d + %d %d + %d\n", ink_rect.x / PANGO_SCALE, ink_rect.width / PANGO_SCALE, ink_rect.y / PANGO_SCALE, ink_rect.height / PANGO_SCALE); #endif ink_w = ink_rect.width / PANGO_SCALE; ink_h = ink_rect.height / PANGO_SCALE; ink_w = ink_w > style->wmin ? ink_w : style->wmin; ink_w = !style->wmax || ink_w < style->wmax ? ink_w : style->wmax; w = ink_w + 2 * style->pad; h = ink_h + 2 * style->pad; sx = *x; sy = *y; if (sx < 0 || sy < 0) { double x1, y1, x2, y2; int sw, sh; cairo_clip_extents(cr, &x1, &y1, &x2, &y2); sw = x2 - x1; sh = y2 - y1; if (sx < 0) sx = sw + sx - w; if (sy < 0) sy = sh + sy - h; } tx = sx - ink_rect.x / PANGO_SCALE + style->pad; ty = sy - ink_rect.y / PANGO_SCALE + style->pad; rrect(cr, sx, sy, w, h, style->radius); cairo_set_source_rgba(cr, bg[0], bg[1], bg[2], bg[3]); cairo_fill_preserve(cr); cairo_set_source_rgba(cr, frame[0], frame[1], frame[2], frame[3]); cairo_set_line_width(cr, style->width); cairo_stroke(cr); if (style->wmax) { cairo_new_path(cr); #if 0 fprintf(stderr, "%u(%d) %u %.60s\n", ty, ink_rect.y / PANGO_SCALE, ink_h, over->s); #endif /* * @@@ for some mysterious reason, we get * ink_h = ink_rect.height / PANGO_SCALE = 5 * instead of 2 if using overlay_style_dense_selected. Strangely, changing * overlay_style_dense_selected such that it becomes more like * overlay_style_dense has no effect. * * This causes the text to be cut vertically, roughly in the middle. We hack * around this problem by growind the clipping area vertically. This works, * since we're currently only concerned about horizontal clipping anyway. */ cairo_rectangle(cr, tx, ty, ink_w, ink_h + 20); cairo_clip(cr); } cairo_set_source_rgba(cr, fg[0], fg[1], fg[2], fg[3]); cairo_move_to(cr, tx, ty); pango_cairo_update_layout(cr, layout); pango_cairo_show_layout(cr, layout); cairo_reset_clip(cr); g_object_unref(layout); if (over->hover || over->click) { struct aoi aoi_cfg = { .x = sx, .y = sy, .w = w, .h = h, .hover = over->hover, .click = over->click, .user = over->user, }; if (over->aoi) aoi_update(over->aoi, &aoi_cfg); else over->aoi = aoi_add(over->aois, &aoi_cfg); } if (*y >= 0) *y += h + style->skip; else *y -= h + style->skip; return over->next; } void overlay_draw_all(struct overlay *overlays, cairo_t *cr, int x, int y) { struct overlay *over; for (over = overlays; over; over = over->next) overlay_draw(over, cr, &x, &y); } struct overlay *overlay_add(struct overlay **overlays, struct aoi **aois, bool (*hover)(void *user, bool on), void (*click)(void *user), void *user) { struct overlay *over; struct overlay **anchor; over = alloc_type(struct overlay); over->s = NULL; over->style = overlay_style_default; over->aois = aois; over->hover = hover; over->click = click; over->user = user; over->aoi = NULL; for (anchor = overlays; *anchor; anchor = &(*anchor)->next); over->next = NULL; *anchor = over; return over; } void overlay_style(struct overlay *over, const struct overlay_style *style) { over->style = *style; } void overlay_text_raw(struct overlay *over, const char *s) { free((char *) over->s); over->s = stralloc(s); } void overlay_text(struct overlay *over, const char *fmt, ...) { va_list ap; va_start(ap, fmt); overlay_text_raw(over, vfmt_pango(fmt, ap)); va_end(ap); } static void overlay_free(struct overlay *over) { if (over->aoi) aoi_remove(over->aois, over->aoi); free((void *) over->s); free(over); } void overlay_remove(struct overlay **overlays, struct overlay *over) { struct overlay **anchor; for (anchor = overlays; *anchor; anchor = &(*anchor)->next) if (*anchor == over) { *anchor = over->next; overlay_free(over); return; } abort(); } void overlay_remove_all(struct overlay **overlays) { struct overlay *next; while (*overlays) { next = (*overlays)->next; overlay_free(*overlays); *overlays = next; } }