2021-11-13 21:56:53 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2020-08-05 20:14:17 +01:00
|
|
|
#include <cairo.h>
|
2021-08-07 08:35:46 +01:00
|
|
|
#include <drm_fourcc.h>
|
2020-08-05 20:14:17 +01:00
|
|
|
#include <pango/pangocairo.h>
|
2021-08-07 08:35:46 +01:00
|
|
|
#include <wlr/render/wlr_renderer.h>
|
|
|
|
|
#include <wlr/util/box.h>
|
|
|
|
|
#include <wlr/util/log.h>
|
2020-08-05 20:14:17 +01:00
|
|
|
#include "common/font.h"
|
2022-08-26 04:50:54 +02:00
|
|
|
#include "common/graphic-helpers.h"
|
2024-01-19 19:06:07 +00:00
|
|
|
#include "common/string-helpers.h"
|
2021-08-07 08:35:46 +01:00
|
|
|
#include "labwc.h"
|
2022-02-17 01:46:32 +01:00
|
|
|
#include "buffer.h"
|
2020-08-05 20:14:17 +01:00
|
|
|
|
2022-09-15 10:53:49 -04:00
|
|
|
PangoFontDescription *
|
|
|
|
|
font_to_pango_desc(struct font *font)
|
|
|
|
|
{
|
|
|
|
|
PangoFontDescription *desc = pango_font_description_new();
|
|
|
|
|
pango_font_description_set_family(desc, font->name);
|
|
|
|
|
pango_font_description_set_size(desc, font->size * PANGO_SCALE);
|
|
|
|
|
if (font->slant == FONT_SLANT_ITALIC) {
|
|
|
|
|
pango_font_description_set_style(desc, PANGO_STYLE_ITALIC);
|
|
|
|
|
}
|
2024-08-21 12:27:07 -05:00
|
|
|
if (font->slant == FONT_SLANT_OBLIQUE) {
|
|
|
|
|
pango_font_description_set_style(desc, PANGO_STYLE_OBLIQUE);
|
|
|
|
|
}
|
2022-09-15 10:53:49 -04:00
|
|
|
if (font->weight == FONT_WEIGHT_BOLD) {
|
|
|
|
|
pango_font_description_set_weight(desc, PANGO_WEIGHT_BOLD);
|
|
|
|
|
}
|
|
|
|
|
return desc;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static PangoRectangle
|
2021-08-20 20:20:49 +01:00
|
|
|
font_extents(struct font *font, const char *string)
|
2020-08-05 20:14:17 +01:00
|
|
|
{
|
2022-08-02 22:00:24 +01:00
|
|
|
PangoRectangle rect = { 0 };
|
|
|
|
|
if (!string) {
|
|
|
|
|
return rect;
|
|
|
|
|
}
|
2020-08-05 20:14:17 +01:00
|
|
|
cairo_surface_t *surface;
|
|
|
|
|
cairo_t *c;
|
|
|
|
|
PangoLayout *layout;
|
|
|
|
|
|
|
|
|
|
surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 1, 1);
|
|
|
|
|
c = cairo_create(surface);
|
|
|
|
|
layout = pango_cairo_create_layout(c);
|
2024-07-14 08:55:30 +01:00
|
|
|
pango_context_set_round_glyph_positions(pango_layout_get_context(layout), false);
|
2022-09-15 10:53:49 -04:00
|
|
|
PangoFontDescription *desc = font_to_pango_desc(font);
|
2020-08-05 20:14:17 +01:00
|
|
|
|
2021-08-20 20:20:49 +01:00
|
|
|
pango_layout_set_font_description(layout, desc);
|
2020-08-05 20:14:17 +01:00
|
|
|
pango_layout_set_text(layout, string, -1);
|
|
|
|
|
pango_layout_set_single_paragraph_mode(layout, TRUE);
|
|
|
|
|
pango_layout_set_width(layout, -1);
|
|
|
|
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_MIDDLE);
|
|
|
|
|
pango_layout_get_extents(layout, NULL, &rect);
|
|
|
|
|
pango_extents_to_pixels(&rect, NULL);
|
|
|
|
|
|
|
|
|
|
/* we put a 2 px edge on each side - because Openbox does it :) */
|
2023-08-17 18:59:29 +02:00
|
|
|
/* TODO: remove the 4 pixel addition and always do the padding by the caller */
|
2020-08-05 20:14:17 +01:00
|
|
|
rect.width += 4;
|
|
|
|
|
|
2020-08-14 17:57:18 +01:00
|
|
|
cairo_destroy(c);
|
|
|
|
|
cairo_surface_destroy(surface);
|
2021-08-20 20:20:49 +01:00
|
|
|
pango_font_description_free(desc);
|
2020-08-14 17:57:18 +01:00
|
|
|
g_object_unref(layout);
|
2020-08-05 20:14:17 +01:00
|
|
|
return rect;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
int
|
2021-08-20 20:20:49 +01:00
|
|
|
font_height(struct font *font)
|
2020-08-05 20:14:17 +01:00
|
|
|
{
|
2021-08-20 20:20:49 +01:00
|
|
|
PangoRectangle rectangle = font_extents(font, "abcdefg");
|
2020-08-05 20:14:17 +01:00
|
|
|
return rectangle.height;
|
|
|
|
|
}
|
2020-10-31 15:41:06 +00:00
|
|
|
|
2022-06-15 01:16:32 +02:00
|
|
|
int
|
|
|
|
|
font_width(struct font *font, const char *string)
|
|
|
|
|
{
|
|
|
|
|
PangoRectangle rectangle = font_extents(font, string);
|
|
|
|
|
return rectangle.width;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-17 01:46:32 +01:00
|
|
|
void
|
|
|
|
|
font_buffer_create(struct lab_data_buffer **buffer, int max_width,
|
2024-03-16 22:20:40 -04:00
|
|
|
const char *text, struct font *font, const float *color,
|
|
|
|
|
const float *bg_color, const char *arrow, double scale)
|
2021-08-07 08:35:46 +01:00
|
|
|
{
|
2022-08-02 22:00:24 +01:00
|
|
|
/* Allow a minimum of one pixel each for text and arrow */
|
|
|
|
|
if (max_width < 2) {
|
|
|
|
|
max_width = 2;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-19 19:06:07 +00:00
|
|
|
if (string_null_or_empty(text)) {
|
2021-08-07 08:35:46 +01:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-02 22:00:24 +01:00
|
|
|
PangoRectangle text_extents = font_extents(font, text);
|
|
|
|
|
PangoRectangle arrow_extents = font_extents(font, arrow);
|
|
|
|
|
|
|
|
|
|
if (arrow) {
|
|
|
|
|
if (arrow_extents.width >= max_width - 1) {
|
|
|
|
|
/* It would be weird to get here, but just in case */
|
|
|
|
|
arrow_extents.width = max_width - 1;
|
|
|
|
|
text_extents.width = 1;
|
|
|
|
|
} else {
|
|
|
|
|
text_extents.width = max_width - arrow_extents.width;
|
|
|
|
|
}
|
|
|
|
|
} else if (text_extents.width > max_width) {
|
|
|
|
|
text_extents.width = max_width;
|
2021-08-07 08:35:46 +01:00
|
|
|
}
|
2022-08-02 22:00:24 +01:00
|
|
|
|
|
|
|
|
*buffer = buffer_create_cairo(text_extents.width + arrow_extents.width,
|
2024-10-06 01:42:54 -04:00
|
|
|
text_extents.height, scale);
|
2022-05-26 15:34:08 +02:00
|
|
|
if (!*buffer) {
|
2022-08-02 22:00:24 +01:00
|
|
|
wlr_log(WLR_ERROR, "Failed to create font buffer");
|
2022-05-26 15:34:08 +02:00
|
|
|
return;
|
|
|
|
|
}
|
2021-08-07 08:35:46 +01:00
|
|
|
|
2022-02-17 01:46:32 +01:00
|
|
|
cairo_t *cairo = (*buffer)->cairo;
|
|
|
|
|
cairo_surface_t *surf = cairo_get_target(cairo);
|
2021-08-07 08:35:46 +01:00
|
|
|
|
2024-03-16 22:20:40 -04:00
|
|
|
/*
|
2024-04-07 03:26:16 -04:00
|
|
|
* Fill with the background color first IF the background color
|
|
|
|
|
* is opaque. This is necessary for subpixel rendering to work
|
|
|
|
|
* properly (it does not work on top of transparency).
|
|
|
|
|
*
|
|
|
|
|
* However, if the background color is not opaque, leave the
|
|
|
|
|
* buffer unfilled (completely transparent) since the background
|
|
|
|
|
* is already rendered by the scene element underneath. In this
|
|
|
|
|
* case we have to disable subpixel rendering.
|
|
|
|
|
*
|
|
|
|
|
* Note: the 0.999 cutoff was chosen to be greater than 254/255
|
|
|
|
|
* (about 0.996) but leave some margin for rounding errors.
|
2024-03-16 22:20:40 -04:00
|
|
|
*/
|
2024-04-07 03:26:16 -04:00
|
|
|
bool opaque_bg = (bg_color[3] > 0.999f);
|
|
|
|
|
if (opaque_bg) {
|
|
|
|
|
set_cairo_color(cairo, bg_color);
|
|
|
|
|
cairo_paint(cairo);
|
|
|
|
|
}
|
2024-03-16 22:20:40 -04:00
|
|
|
|
2022-08-26 04:50:54 +02:00
|
|
|
set_cairo_color(cairo, color);
|
2021-08-07 08:35:46 +01:00
|
|
|
cairo_move_to(cairo, 0, 0);
|
|
|
|
|
|
|
|
|
|
PangoLayout *layout = pango_cairo_create_layout(cairo);
|
2024-07-14 08:55:30 +01:00
|
|
|
pango_context_set_round_glyph_positions(pango_layout_get_context(layout), false);
|
2022-08-02 22:00:24 +01:00
|
|
|
pango_layout_set_width(layout, text_extents.width * PANGO_SCALE);
|
2021-08-07 08:35:46 +01:00
|
|
|
pango_layout_set_text(layout, text, -1);
|
|
|
|
|
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
|
|
|
|
|
|
2024-04-07 03:26:16 -04:00
|
|
|
if (!opaque_bg) {
|
|
|
|
|
/* disable subpixel rendering */
|
|
|
|
|
cairo_font_options_t *opts = cairo_font_options_create();
|
|
|
|
|
cairo_font_options_set_antialias(opts, CAIRO_ANTIALIAS_GRAY);
|
|
|
|
|
PangoContext *ctx = pango_layout_get_context(layout);
|
|
|
|
|
pango_cairo_context_set_font_options(ctx, opts);
|
|
|
|
|
cairo_font_options_destroy(opts);
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-15 10:53:49 -04:00
|
|
|
PangoFontDescription *desc = font_to_pango_desc(font);
|
2021-08-07 08:35:46 +01:00
|
|
|
pango_layout_set_font_description(layout, desc);
|
|
|
|
|
pango_font_description_free(desc);
|
|
|
|
|
pango_cairo_update_layout(cairo, layout);
|
|
|
|
|
pango_cairo_show_layout(cairo, layout);
|
2022-08-02 22:00:24 +01:00
|
|
|
|
|
|
|
|
if (arrow) {
|
|
|
|
|
cairo_move_to(cairo, text_extents.width, 0);
|
|
|
|
|
pango_layout_set_width(layout, arrow_extents.width * PANGO_SCALE);
|
|
|
|
|
pango_layout_set_text(layout, arrow, -1);
|
|
|
|
|
pango_cairo_show_layout(cairo, layout);
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-07 08:35:46 +01:00
|
|
|
g_object_unref(layout);
|
|
|
|
|
|
|
|
|
|
cairo_surface_flush(surf);
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-31 15:41:06 +00:00
|
|
|
void
|
|
|
|
|
font_finish(void)
|
|
|
|
|
{
|
|
|
|
|
pango_cairo_font_map_set_default(NULL);
|
|
|
|
|
}
|