labwc/src/ssd/resize-indicator.c
tokyo4j ebce406b11 font: remove 4px padding on the right
Added `menu.items.padding.x` padding between item text and arrow instead.

Replaced `if (!string)` with `if (string_null_or_empty(string))` in
`font_extents()` just as a minor optimization.
2025-09-22 18:23:33 +01:00

218 lines
6 KiB
C

// SPDX-License-Identifier: GPL-2.0-only
#include <assert.h>
#include <wlr/types/wlr_scene.h>
#include <wlr/util/box.h>
#include <wlr/util/log.h>
#include "config/rcxml.h"
#include "labwc.h"
#include "resize-indicator.h"
#include "resize-outlines.h"
#include "scaled-buffer/scaled-font-buffer.h"
#include "ssd.h"
#include "theme.h"
#include "view.h"
#define PADDING rc.theme->osd_window_switcher_classic.padding
static void
resize_indicator_reconfigure_view(struct resize_indicator *indicator)
{
assert(indicator->tree);
struct theme *theme = rc.theme;
indicator->height = font_height(&rc.font_osd)
+ 2 * PADDING
+ 2 * theme->osd_border_width;
/* Static positions */
wlr_scene_node_set_position(&indicator->background->node,
theme->osd_border_width, theme->osd_border_width);
wlr_scene_node_set_position(&indicator->text->scene_buffer->node,
theme->osd_border_width + PADDING,
theme->osd_border_width + PADDING);
/* Colors */
wlr_scene_rect_set_color(indicator->border, theme->osd_border_color);
wlr_scene_rect_set_color(indicator->background, theme->osd_bg_color);
}
static void
resize_indicator_init(struct view *view)
{
assert(view);
struct resize_indicator *indicator = &view->resize_indicator;
assert(!indicator->tree);
indicator->tree = wlr_scene_tree_create(view->scene_tree);
indicator->border = wlr_scene_rect_create(
indicator->tree, 0, 0, rc.theme->osd_border_color);
indicator->background = wlr_scene_rect_create(
indicator->tree, 0, 0, rc.theme->osd_bg_color);
indicator->text = scaled_font_buffer_create(indicator->tree);
wlr_scene_node_set_enabled(&indicator->tree->node, false);
resize_indicator_reconfigure_view(indicator);
}
static bool
wants_indicator(struct view *view)
{
assert(view);
if (rc.resize_indicator == LAB_RESIZE_INDICATOR_NON_PIXEL) {
if (view->server->input_mode != LAB_INPUT_STATE_RESIZE) {
return false;
}
struct view_size_hints hints = view_get_size_hints(view);
if (hints.width_inc && hints.height_inc) {
return true;
}
}
return rc.resize_indicator == LAB_RESIZE_INDICATOR_ALWAYS;
}
void
resize_indicator_reconfigure(struct server *server)
{
struct view *view;
wl_list_for_each(view, &server->views, link) {
struct resize_indicator *indicator = &view->resize_indicator;
if (indicator->tree) {
resize_indicator_reconfigure_view(indicator);
}
if (view != server->grabbed_view) {
continue;
}
/* This view is currently in an interactive move/resize operation */
if (indicator->tree && indicator->tree->node.enabled) {
/* Indicator was active while reloading the config */
if (wants_indicator(view)) {
/* Apply new font setting */
resize_indicator_update(view);
} else {
/* Indicator was disabled in config */
resize_indicator_hide(view);
}
} else if (wants_indicator(view)) {
/* Indicator not yet active */
resize_indicator_show(view);
}
}
}
static void
resize_indicator_set_size(struct resize_indicator *indicator, int width)
{
assert(indicator->tree);
/* We are not using a width-cache-early-out here to allow for theme changes */
indicator->width = width
+ 2 * PADDING
+ 2 * rc.theme->osd_border_width;
wlr_scene_rect_set_size(indicator->border, indicator->width, indicator->height);
wlr_scene_rect_set_size(indicator->background,
indicator->width - 2 * rc.theme->osd_border_width,
indicator->height - 2 * rc.theme->osd_border_width);
}
void
resize_indicator_show(struct view *view)
{
assert(view);
if (!wants_indicator(view)) {
return;
}
struct resize_indicator *indicator = &view->resize_indicator;
if (!indicator->tree) {
/* Lazy initialize */
resize_indicator_init(view);
}
wlr_scene_node_raise_to_top(&indicator->tree->node);
wlr_scene_node_set_enabled(&indicator->tree->node, true);
resize_indicator_update(view);
}
void
resize_indicator_update(struct view *view)
{
assert(view);
assert(view == view->server->grabbed_view);
if (!wants_indicator(view)) {
return;
}
struct resize_indicator *indicator = &view->resize_indicator;
if (!indicator->tree) {
/*
* Future-proofs this routine:
*
* This can only happen when either src/interactive.c
* stops calling resize_indicator_show(), there is a
* bug in this file or resize_indicator_reconfigure()
* gets changed.
*/
wlr_log(WLR_INFO, "Warning: resize_indicator has to use a fallback path");
resize_indicator_show(view);
}
char text[32]; /* 12345 x 12345 would be 13 chars + 1 null byte */
struct wlr_box view_box;
if (resize_outlines_enabled(view)) {
view_box = view->resize_outlines.view_geo;
} else {
view_box = view->current;
view_box.height = view_effective_height(view, /* use_pending */ false);
}
if (view->server->input_mode == LAB_INPUT_STATE_RESIZE) {
struct view_size_hints hints = view_get_size_hints(view);
snprintf(text, sizeof(text), "%d x %d",
MAX(0, view_box.width - hints.base_width)
/ MAX(1, hints.width_inc),
MAX(0, view_box.height - hints.base_height)
/ MAX(1, hints.height_inc));
} else if (view->server->input_mode == LAB_INPUT_STATE_MOVE) {
struct border margin = ssd_get_margin(view->ssd);
snprintf(text, sizeof(text), "%d , %d",
view_box.x - margin.left,
view_box.y - margin.top);
} else {
wlr_log(WLR_ERROR, "Invalid input mode for indicator update %u",
view->server->input_mode);
return;
}
/* Let the indicator change width as required by the content */
int width = font_width(&rc.font_osd, text);
resize_indicator_set_size(indicator, width);
/* Center the indicator in the window */
int x = view_box.x - view->current.x + (view_box.width - indicator->width) / 2;
int y = view_box.y - view->current.y + (view_box.height - indicator->height) / 2;
wlr_scene_node_set_position(&indicator->tree->node, x, y);
scaled_font_buffer_update(indicator->text, text, width, &rc.font_osd,
rc.theme->osd_label_text_color, rc.theme->osd_bg_color);
}
void
resize_indicator_hide(struct view *view)
{
assert(view);
struct resize_indicator *indicator = &view->resize_indicator;
if (!indicator->tree) {
return;
}
wlr_scene_node_set_enabled(&indicator->tree->node, false);
}