2021-09-24 21:45:48 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2021-01-09 22:51:20 +00:00
|
|
|
/*
|
|
|
|
|
* output.c: labwc output and rendering
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2019-2021 Johan Malm
|
|
|
|
|
* Copyright (C) 2020 The Sway authors
|
|
|
|
|
*/
|
|
|
|
|
|
2020-10-15 21:03:04 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
2020-12-30 10:29:21 +00:00
|
|
|
#include "config.h"
|
2021-06-30 20:12:58 +01:00
|
|
|
#include <assert.h>
|
2022-02-11 23:12:45 +00:00
|
|
|
#include <wlr/types/wlr_buffer.h>
|
2020-09-29 19:53:46 +01:00
|
|
|
#include <wlr/types/wlr_xdg_output_v1.h>
|
2021-01-09 22:51:20 +00:00
|
|
|
#include <wlr/types/wlr_output_damage.h>
|
2022-02-20 13:15:58 +00:00
|
|
|
#include <wlr/types/wlr_scene.h>
|
2021-01-09 22:51:20 +00:00
|
|
|
#include <wlr/util/region.h>
|
2021-07-23 21:15:55 +01:00
|
|
|
#include <wlr/util/log.h>
|
2022-02-11 23:12:45 +00:00
|
|
|
#include "buffer.h"
|
2019-12-26 21:37:31 +00:00
|
|
|
#include "labwc.h"
|
2021-03-08 21:56:57 +00:00
|
|
|
#include "layers.h"
|
2020-10-19 22:14:17 +01:00
|
|
|
#include "menu/menu.h"
|
2021-03-21 20:54:55 +00:00
|
|
|
#include "ssd.h"
|
2021-02-21 22:18:34 +00:00
|
|
|
#include "theme.h"
|
2019-12-26 21:37:31 +00:00
|
|
|
|
2021-01-09 22:51:20 +00:00
|
|
|
static void
|
2022-02-11 23:12:45 +00:00
|
|
|
output_frame_notify(struct wl_listener *listener, void *data)
|
2021-01-09 22:51:20 +00:00
|
|
|
{
|
2022-02-11 23:12:45 +00:00
|
|
|
struct output *output = wl_container_of(listener, output, frame);
|
2021-01-09 22:51:20 +00:00
|
|
|
if (!output->wlr_output->enabled) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-11 23:12:45 +00:00
|
|
|
wlr_scene_output_commit(output->scene_output);
|
2020-05-29 22:18:03 +01:00
|
|
|
|
2022-02-11 23:12:45 +00:00
|
|
|
struct timespec now = { 0 };
|
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
|
|
|
wlr_scene_output_send_frame_done(output->scene_output, &now);
|
2021-01-09 22:51:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
output_destroy_notify(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
2021-08-25 20:45:39 +01:00
|
|
|
struct output *output = wl_container_of(listener, output, destroy);
|
|
|
|
|
wl_list_remove(&output->link);
|
2022-02-11 23:12:45 +00:00
|
|
|
wl_list_remove(&output->frame.link);
|
2021-08-25 20:45:39 +01:00
|
|
|
wl_list_remove(&output->destroy.link);
|
2021-01-09 22:51:20 +00:00
|
|
|
}
|
|
|
|
|
|
2020-09-29 19:53:46 +01:00
|
|
|
static void
|
|
|
|
|
new_output_notify(struct wl_listener *listener, void *data)
|
2020-05-29 22:18:03 +01:00
|
|
|
{
|
2021-08-22 14:14:50 +01:00
|
|
|
/*
|
|
|
|
|
* This event is rasied by the backend when a new output (aka display
|
|
|
|
|
* or monitor) becomes available.
|
|
|
|
|
*/
|
2020-05-29 22:18:03 +01:00
|
|
|
struct server *server = wl_container_of(listener, server, new_output);
|
|
|
|
|
struct wlr_output *wlr_output = data;
|
|
|
|
|
|
2021-11-26 19:27:50 +00:00
|
|
|
/*
|
|
|
|
|
* Configures the output created by the backend to use our allocator
|
|
|
|
|
* and our renderer. Must be done once, before commiting the output
|
|
|
|
|
*/
|
|
|
|
|
if (!wlr_output_init_render(wlr_output, server->allocator,
|
|
|
|
|
server->renderer)) {
|
2021-11-21 07:01:16 +00:00
|
|
|
wlr_log(WLR_ERROR, "unable to init output renderer");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-26 10:52:09 -05:00
|
|
|
wlr_log(WLR_DEBUG, "enable output");
|
|
|
|
|
wlr_output_enable(wlr_output, true);
|
|
|
|
|
|
2021-08-22 14:14:50 +01:00
|
|
|
/* The mode is a tuple of (width, height, refresh rate). */
|
|
|
|
|
wlr_log(WLR_DEBUG, "set preferred mode");
|
|
|
|
|
struct wlr_output_mode *preferred_mode =
|
|
|
|
|
wlr_output_preferred_mode(wlr_output);
|
|
|
|
|
wlr_output_set_mode(wlr_output, preferred_mode);
|
|
|
|
|
|
2020-05-29 22:18:03 +01:00
|
|
|
/*
|
2021-08-22 14:14:50 +01:00
|
|
|
* Sometimes the preferred mode is not available due to hardware
|
|
|
|
|
* constraints (e.g. GPU or cable bandwidth limitations). In these
|
|
|
|
|
* cases it's better to fallback to lower modes than to end up with
|
|
|
|
|
* a black screen. See sway@4cdc4ac6
|
2020-05-29 22:18:03 +01:00
|
|
|
*/
|
2021-08-22 14:14:50 +01:00
|
|
|
if (!wlr_output_test(wlr_output)) {
|
2021-08-25 20:45:39 +01:00
|
|
|
wlr_log(WLR_DEBUG,
|
|
|
|
|
"preferred mode rejected, falling back to another mode");
|
2021-08-22 14:14:50 +01:00
|
|
|
struct wlr_output_mode *mode;
|
|
|
|
|
wl_list_for_each(mode, &wlr_output->modes, link) {
|
|
|
|
|
if (mode == preferred_mode) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
wlr_output_set_mode(wlr_output, mode);
|
|
|
|
|
if (wlr_output_test(wlr_output)) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-05-29 22:18:03 +01:00
|
|
|
}
|
|
|
|
|
|
2021-08-22 14:14:50 +01:00
|
|
|
wlr_output_commit(wlr_output);
|
|
|
|
|
|
2020-05-29 22:18:03 +01:00
|
|
|
struct output *output = calloc(1, sizeof(struct output));
|
|
|
|
|
output->wlr_output = wlr_output;
|
2021-10-20 22:32:46 +01:00
|
|
|
wlr_output->data = output;
|
2020-05-29 22:18:03 +01:00
|
|
|
output->server = server;
|
2021-07-12 21:39:09 +01:00
|
|
|
wlr_output_effective_resolution(wlr_output,
|
|
|
|
|
&output->usable_area.width, &output->usable_area.height);
|
2020-05-29 22:18:03 +01:00
|
|
|
wl_list_insert(&server->outputs, &output->link);
|
2021-01-09 22:51:20 +00:00
|
|
|
|
2020-09-29 19:53:46 +01:00
|
|
|
output->destroy.notify = output_destroy_notify;
|
|
|
|
|
wl_signal_add(&wlr_output->events.destroy, &output->destroy);
|
2022-02-11 23:12:45 +00:00
|
|
|
output->frame.notify = output_frame_notify;
|
|
|
|
|
wl_signal_add(&wlr_output->events.frame, &output->frame);
|
2021-01-09 22:51:20 +00:00
|
|
|
|
2022-03-02 20:39:46 +00:00
|
|
|
int nr_layers = sizeof(output->layers) / sizeof(output->layers[0]);
|
|
|
|
|
for (int i = 0; i < nr_layers; i++) {
|
2022-02-20 13:15:58 +00:00
|
|
|
wl_list_init(&output->layers[i]);
|
|
|
|
|
output->layer_tree[i] =
|
|
|
|
|
wlr_scene_tree_create(&server->scene->node);
|
|
|
|
|
output->layer_tree[i]->node.data = output->wlr_output;
|
|
|
|
|
}
|
|
|
|
|
wlr_scene_node_lower_to_bottom(&output->layer_tree[1]->node);
|
|
|
|
|
wlr_scene_node_lower_to_bottom(&output->layer_tree[0]->node);
|
|
|
|
|
wlr_scene_node_raise_to_top(&output->layer_tree[2]->node);
|
|
|
|
|
wlr_scene_node_raise_to_top(&output->layer_tree[3]->node);
|
2021-03-22 21:25:51 +00:00
|
|
|
|
2021-10-22 20:23:09 +01:00
|
|
|
if (rc.adaptive_sync) {
|
|
|
|
|
wlr_log(WLR_INFO, "enable adaptive sync on %s",
|
|
|
|
|
wlr_output->name);
|
2021-03-22 21:25:51 +00:00
|
|
|
wlr_output_enable_adaptive_sync(wlr_output, true);
|
|
|
|
|
}
|
2022-02-09 16:38:07 -05:00
|
|
|
|
2020-05-29 22:18:03 +01:00
|
|
|
wlr_output_layout_add_auto(server->output_layout, wlr_output);
|
2022-02-11 23:12:45 +00:00
|
|
|
|
|
|
|
|
/* TODO: check this makes sense */
|
|
|
|
|
struct wlr_scene_output *scene_output;
|
|
|
|
|
wl_list_for_each (scene_output, &output->server->scene->outputs, link) {
|
|
|
|
|
if (scene_output->output == wlr_output) {
|
|
|
|
|
output->scene_output = scene_output;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(output->scene_output);
|
2020-09-29 19:53:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
output_init(struct server *server)
|
|
|
|
|
{
|
|
|
|
|
server->new_output.notify = new_output_notify;
|
|
|
|
|
wl_signal_add(&server->backend->events.new_output, &server->new_output);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Create an output layout, which is a wlroots utility for working with
|
|
|
|
|
* an arrangement of screens in a physical layout.
|
|
|
|
|
*/
|
|
|
|
|
server->output_layout = wlr_output_layout_create();
|
|
|
|
|
if (!server->output_layout) {
|
|
|
|
|
wlr_log(WLR_ERROR, "unable to create output layout");
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
2022-03-02 20:25:57 +00:00
|
|
|
wlr_scene_attach_output_layout(server->scene, server->output_layout);
|
2020-09-29 19:53:46 +01:00
|
|
|
|
|
|
|
|
/* Enable screen recording with wf-recorder */
|
|
|
|
|
wlr_xdg_output_manager_v1_create(server->wl_display,
|
|
|
|
|
server->output_layout);
|
|
|
|
|
|
|
|
|
|
wl_list_init(&server->outputs);
|
2021-02-27 23:15:02 -05:00
|
|
|
|
|
|
|
|
output_manager_init(server);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 17:30:55 -05:00
|
|
|
static void
|
|
|
|
|
output_update_for_layout_change(struct server *server)
|
|
|
|
|
{
|
|
|
|
|
/* Adjust window positions/sizes */
|
|
|
|
|
struct view *view;
|
|
|
|
|
wl_list_for_each(view, &server->views, link) {
|
|
|
|
|
view_adjust_for_layout_change(view);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* "Move" each wlr_output_cursor (in per-output coordinates) to
|
|
|
|
|
* align with the seat cursor. Set a default cursor image so
|
|
|
|
|
* that the cursor isn't invisible on new outputs.
|
|
|
|
|
*
|
|
|
|
|
* TODO: remember the most recent cursor image (see cursor.c)
|
|
|
|
|
* and set that rather than XCURSOR_DEFAULT
|
|
|
|
|
*/
|
|
|
|
|
wlr_cursor_move(server->seat.cursor, NULL, 0, 0);
|
|
|
|
|
wlr_xcursor_manager_set_cursor_image(server->seat.xcursor_manager,
|
|
|
|
|
XCURSOR_DEFAULT, server->seat.cursor);
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
static void
|
|
|
|
|
output_config_apply(struct server *server,
|
2021-03-06 18:30:53 +00:00
|
|
|
struct wlr_output_configuration_v1 *config)
|
2021-02-27 23:15:02 -05:00
|
|
|
{
|
|
|
|
|
server->pending_output_config = config;
|
|
|
|
|
|
|
|
|
|
struct wlr_output_configuration_head_v1 *head;
|
|
|
|
|
wl_list_for_each(head, &config->heads, link) {
|
|
|
|
|
struct wlr_output *o = head->state.output;
|
|
|
|
|
bool need_to_add = head->state.enabled && !o->enabled;
|
2021-08-25 20:45:39 +01:00
|
|
|
if (need_to_add) {
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_layout_add_auto(server->output_layout, o);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool need_to_remove = !head->state.enabled && o->enabled;
|
2021-08-25 20:45:39 +01:00
|
|
|
if (need_to_remove) {
|
2022-02-11 23:12:45 +00:00
|
|
|
/* TODO: should we output->scene_output = NULL; ?? */
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_layout_remove(server->output_layout, o);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wlr_output_enable(o, head->state.enabled);
|
2021-08-25 20:45:39 +01:00
|
|
|
if (head->state.enabled) {
|
|
|
|
|
if (head->state.mode) {
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_set_mode(o, head->state.mode);
|
|
|
|
|
} else {
|
|
|
|
|
int32_t width = head->state.custom_mode.width;
|
|
|
|
|
int32_t height = head->state.custom_mode.height;
|
|
|
|
|
int32_t refresh = head->state.custom_mode.refresh;
|
2021-09-21 22:05:56 +01:00
|
|
|
wlr_output_set_custom_mode(o, width,
|
|
|
|
|
height, refresh);
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
2021-03-06 18:30:53 +00:00
|
|
|
wlr_output_layout_move(server->output_layout, o,
|
|
|
|
|
head->state.x, head->state.y);
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_set_scale(o, head->state.scale);
|
|
|
|
|
wlr_output_set_transform(o, head->state.transform);
|
|
|
|
|
}
|
|
|
|
|
wlr_output_commit(o);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
server->pending_output_config = NULL;
|
2021-12-31 17:30:55 -05:00
|
|
|
output_update_for_layout_change(server);
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
2021-02-28 00:08:47 -05:00
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
static bool
|
|
|
|
|
verify_output_config_v1(const struct wlr_output_configuration_v1 *config)
|
2021-02-28 00:08:47 -05:00
|
|
|
{
|
2021-08-25 20:45:39 +01:00
|
|
|
/* TODO implement */
|
2021-02-28 00:08:47 -05:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
static void
|
|
|
|
|
handle_output_manager_apply(struct wl_listener *listener, void *data)
|
2021-02-27 23:15:02 -05:00
|
|
|
{
|
2021-09-21 22:05:56 +01:00
|
|
|
struct server *server =
|
|
|
|
|
wl_container_of(listener, server, output_manager_apply);
|
2021-02-27 23:15:02 -05:00
|
|
|
struct wlr_output_configuration_v1 *config = data;
|
2021-02-28 00:08:47 -05:00
|
|
|
|
|
|
|
|
bool config_is_good = verify_output_config_v1(config);
|
|
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
if (config_is_good) {
|
2021-02-28 00:08:47 -05:00
|
|
|
output_config_apply(server, config);
|
|
|
|
|
wlr_output_configuration_v1_send_succeeded(config);
|
|
|
|
|
} else {
|
|
|
|
|
wlr_output_configuration_v1_send_failed(config);
|
|
|
|
|
}
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_configuration_v1_destroy(config);
|
2021-12-17 10:20:57 -05:00
|
|
|
struct output *output;
|
|
|
|
|
wl_list_for_each(output, &server->outputs, link) {
|
|
|
|
|
wlr_xcursor_manager_load(server->seat.xcursor_manager,
|
|
|
|
|
output->wlr_output->scale);
|
|
|
|
|
}
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Take the way outputs are currently configured/layed out and turn that into
|
|
|
|
|
* a struct that we send to clients via the wlr_output_configuration v1
|
|
|
|
|
* interface
|
|
|
|
|
*/
|
2021-08-25 20:45:39 +01:00
|
|
|
static struct
|
|
|
|
|
wlr_output_configuration_v1 *create_output_config(struct server *server)
|
2021-02-27 23:15:02 -05:00
|
|
|
{
|
2021-09-21 22:05:56 +01:00
|
|
|
struct wlr_output_configuration_v1 *config =
|
|
|
|
|
wlr_output_configuration_v1_create();
|
2021-08-25 20:45:39 +01:00
|
|
|
if (!config) {
|
2021-07-23 21:15:55 +01:00
|
|
|
wlr_log(WLR_ERROR, "wlr_output_configuration_v1_create()");
|
2021-02-27 23:15:02 -05:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct output *output;
|
|
|
|
|
wl_list_for_each(output, &server->outputs, link) {
|
2021-03-06 18:30:53 +00:00
|
|
|
struct wlr_output_configuration_head_v1 *head =
|
|
|
|
|
wlr_output_configuration_head_v1_create(config,
|
|
|
|
|
output->wlr_output);
|
2021-08-25 20:45:39 +01:00
|
|
|
if (!head) {
|
2021-09-21 22:05:56 +01:00
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
|
"wlr_output_configuration_head_v1_create()");
|
2021-02-27 23:15:02 -05:00
|
|
|
wlr_output_configuration_v1_destroy(config);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2022-02-14 20:20:16 +00:00
|
|
|
struct wlr_box box;
|
|
|
|
|
wlr_output_layout_get_box(server->output_layout,
|
|
|
|
|
output->wlr_output, &box);
|
|
|
|
|
if (!wlr_box_empty(&box)) {
|
|
|
|
|
head->state.x = box.x;
|
|
|
|
|
head->state.y = box.y;
|
2021-02-27 23:15:02 -05:00
|
|
|
} else {
|
2021-07-23 21:15:55 +01:00
|
|
|
wlr_log(WLR_ERROR, "failed to get output layout box");
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return config;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
static void
|
|
|
|
|
handle_output_layout_change(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
2021-09-21 22:05:56 +01:00
|
|
|
struct server *server =
|
|
|
|
|
wl_container_of(listener, server, output_layout_change);
|
2021-02-27 23:15:02 -05:00
|
|
|
|
|
|
|
|
bool done_changing = server->pending_output_config == NULL;
|
2021-08-25 20:45:39 +01:00
|
|
|
if (done_changing) {
|
2021-09-21 22:05:56 +01:00
|
|
|
struct wlr_output_configuration_v1 *config =
|
|
|
|
|
create_output_config(server);
|
2021-08-25 20:45:39 +01:00
|
|
|
if (config) {
|
2021-09-21 22:05:56 +01:00
|
|
|
wlr_output_manager_v1_set_configuration(
|
|
|
|
|
server->output_manager, config);
|
2021-02-27 23:15:02 -05:00
|
|
|
} else {
|
2021-09-21 22:05:56 +01:00
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
|
"wlr_output_manager_v1_set_configuration()");
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
2021-10-30 10:20:06 -04:00
|
|
|
struct output *output;
|
2022-02-20 13:15:58 +00:00
|
|
|
|
2022-03-02 20:25:57 +00:00
|
|
|
wl_list_for_each(output, &server->outputs, link) {
|
|
|
|
|
if (output) {
|
2022-03-02 20:29:29 +00:00
|
|
|
layers_arrange(output);
|
2022-03-02 20:25:57 +00:00
|
|
|
}
|
|
|
|
|
}
|
2021-12-31 17:30:55 -05:00
|
|
|
output_update_for_layout_change(server);
|
2021-02-27 23:15:02 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-25 20:45:39 +01:00
|
|
|
void
|
|
|
|
|
output_manager_init(struct server *server)
|
2021-02-27 23:15:02 -05:00
|
|
|
{
|
|
|
|
|
server->output_manager = wlr_output_manager_v1_create(server->wl_display);
|
|
|
|
|
|
|
|
|
|
server->output_layout_change.notify = handle_output_layout_change;
|
|
|
|
|
wl_signal_add(&server->output_layout->events.change,
|
2021-03-06 18:30:53 +00:00
|
|
|
&server->output_layout_change);
|
2021-02-27 23:15:02 -05:00
|
|
|
|
|
|
|
|
server->output_manager_apply.notify = handle_output_manager_apply;
|
|
|
|
|
wl_signal_add(&server->output_manager->events.apply,
|
2021-03-06 18:30:53 +00:00
|
|
|
&server->output_manager_apply);
|
2020-05-29 22:18:03 +01:00
|
|
|
}
|
2021-07-12 16:44:30 +01:00
|
|
|
|
|
|
|
|
struct output *
|
|
|
|
|
output_from_wlr_output(struct server *server, struct wlr_output *wlr_output)
|
|
|
|
|
{
|
|
|
|
|
struct output *output;
|
2021-08-25 20:45:39 +01:00
|
|
|
wl_list_for_each (output, &server->outputs, link) {
|
2021-07-12 16:44:30 +01:00
|
|
|
if (output->wlr_output == wlr_output) {
|
|
|
|
|
return output;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2021-07-13 21:50:02 +01:00
|
|
|
|
2021-07-21 22:04:54 +01:00
|
|
|
struct wlr_box
|
|
|
|
|
output_usable_area_in_layout_coords(struct output *output)
|
|
|
|
|
{
|
|
|
|
|
struct wlr_box box = output->usable_area;
|
|
|
|
|
double ox = 0, oy = 0;
|
|
|
|
|
wlr_output_layout_output_coords(output->server->output_layout,
|
|
|
|
|
output->wlr_output, &ox, &oy);
|
|
|
|
|
box.x -= ox;
|
|
|
|
|
box.y -= oy;
|
|
|
|
|
return box;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct wlr_box
|
|
|
|
|
output_usable_area_from_cursor_coords(struct server *server)
|
2021-07-13 21:50:02 +01:00
|
|
|
{
|
|
|
|
|
struct wlr_output *wlr_output;
|
|
|
|
|
wlr_output = wlr_output_layout_output_at(server->output_layout,
|
|
|
|
|
server->seat.cursor->x, server->seat.cursor->y);
|
2021-07-21 22:04:54 +01:00
|
|
|
struct output *output = output_from_wlr_output(server, wlr_output);
|
|
|
|
|
return output_usable_area_in_layout_coords(output);
|
2021-07-13 21:50:02 +01:00
|
|
|
}
|