244 lines
5.3 KiB
C
244 lines
5.3 KiB
C
/* Copyright (c), Charlotte Meyer <dev@buffet.sh>
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include "desktop/view.h"
|
|
|
|
#include <wlr/types/wlr_cursor.h>
|
|
#include <wlr/util/log.h>
|
|
|
|
#include "desktop/output.h"
|
|
#include "input/cursor.h"
|
|
#include "input/seat.h"
|
|
#include "server.h"
|
|
|
|
void
|
|
view_close(struct kiwmi_view *view)
|
|
{
|
|
if (view->impl->close) {
|
|
view->impl->close(view);
|
|
}
|
|
}
|
|
|
|
void
|
|
view_for_each_surface(
|
|
struct kiwmi_view *view,
|
|
wlr_surface_iterator_func_t iterator,
|
|
void *user_data)
|
|
{
|
|
if (view->impl->for_each_surface) {
|
|
view->impl->for_each_surface(view, iterator, user_data);
|
|
}
|
|
}
|
|
|
|
pid_t
|
|
view_get_pid(struct kiwmi_view *view)
|
|
{
|
|
if (view->impl->get_pid) {
|
|
return view->impl->get_pid(view);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void
|
|
view_get_size(struct kiwmi_view *view, uint32_t *width, uint32_t *height)
|
|
{
|
|
if (view->impl->get_size) {
|
|
view->impl->get_size(view, width, height);
|
|
}
|
|
}
|
|
|
|
const char *
|
|
view_get_app_id(struct kiwmi_view *view)
|
|
{
|
|
if (view->impl->get_string_prop) {
|
|
return view->impl->get_string_prop(view, KIWMI_VIEW_PROP_APP_ID);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
const char *
|
|
view_get_title(struct kiwmi_view *view)
|
|
{
|
|
if (view->impl->get_string_prop) {
|
|
return view->impl->get_string_prop(view, KIWMI_VIEW_PROP_TITLE);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
view_set_activated(struct kiwmi_view *view, bool activated)
|
|
{
|
|
if (view->impl->set_activated) {
|
|
view->impl->set_activated(view, activated);
|
|
}
|
|
}
|
|
|
|
void
|
|
view_set_size(struct kiwmi_view *view, uint32_t width, uint32_t height)
|
|
{
|
|
if (view->impl->set_size) {
|
|
view->impl->set_size(view, width, height);
|
|
}
|
|
}
|
|
|
|
void
|
|
view_set_tiled(struct kiwmi_view *view, enum wlr_edges edges)
|
|
{
|
|
if (view->impl->set_tiled) {
|
|
view->impl->set_tiled(view, edges);
|
|
}
|
|
}
|
|
|
|
struct wlr_surface *
|
|
view_surface_at(
|
|
struct kiwmi_view *view,
|
|
double sx,
|
|
double sy,
|
|
double *sub_x,
|
|
double *sub_y)
|
|
{
|
|
if (view->impl->surface_at) {
|
|
return view->impl->surface_at(view, sx, sy, sub_x, sub_y);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static bool
|
|
surface_at(
|
|
struct kiwmi_view *view,
|
|
struct wlr_surface **surface,
|
|
double lx,
|
|
double ly,
|
|
double *sx,
|
|
double *sy)
|
|
{
|
|
double view_sx = lx - view->x + view->geom.x;
|
|
double view_sy = ly - view->y + view->geom.y;
|
|
|
|
double _sx;
|
|
double _sy;
|
|
struct wlr_surface *_surface =
|
|
view_surface_at(view, view_sx, view_sy, &_sx, &_sy);
|
|
|
|
if (_surface) {
|
|
*sx = _sx;
|
|
*sy = _sy;
|
|
*surface = _surface;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
struct kiwmi_view *
|
|
view_at(
|
|
struct kiwmi_desktop *desktop,
|
|
double lx,
|
|
double ly,
|
|
struct wlr_surface **surface,
|
|
double *sx,
|
|
double *sy)
|
|
{
|
|
struct kiwmi_view *view;
|
|
wl_list_for_each (view, &desktop->views, link) {
|
|
if (view->hidden || !view->mapped) {
|
|
continue;
|
|
}
|
|
|
|
if (surface_at(view, surface, lx, ly, sx, sy)) {
|
|
return view;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
view_begin_interactive(
|
|
struct kiwmi_view *view,
|
|
enum kiwmi_cursor_mode mode,
|
|
uint32_t edges)
|
|
{
|
|
struct kiwmi_desktop *desktop = view->desktop;
|
|
struct kiwmi_server *server = wl_container_of(desktop, server, desktop);
|
|
struct kiwmi_cursor *cursor = server->input.cursor;
|
|
struct wlr_surface *focused_surface =
|
|
server->input.seat->seat->pointer_state.focused_surface;
|
|
struct wlr_surface *wlr_surface = view->wlr_surface;
|
|
|
|
if (wlr_surface != focused_surface) {
|
|
return;
|
|
}
|
|
|
|
uint32_t width;
|
|
uint32_t height;
|
|
view_get_size(view, &width, &height);
|
|
|
|
cursor->cursor_mode = mode;
|
|
cursor->grabbed.view = view;
|
|
|
|
if (mode == KIWMI_CURSOR_MOVE) {
|
|
cursor->grabbed.orig_x = cursor->cursor->x - view->x;
|
|
cursor->grabbed.orig_y = cursor->cursor->y - view->y;
|
|
} else {
|
|
cursor->grabbed.orig_x = cursor->cursor->x;
|
|
cursor->grabbed.orig_y = cursor->cursor->y;
|
|
cursor->grabbed.resize_edges = edges;
|
|
}
|
|
|
|
cursor->grabbed.orig_geom.x = view->x;
|
|
cursor->grabbed.orig_geom.y = view->y;
|
|
cursor->grabbed.orig_geom.width = width;
|
|
cursor->grabbed.orig_geom.height = height;
|
|
}
|
|
|
|
void
|
|
view_move(struct kiwmi_view *view)
|
|
{
|
|
view_begin_interactive(view, KIWMI_CURSOR_MOVE, 0);
|
|
}
|
|
|
|
void
|
|
view_resize(struct kiwmi_view *view, uint32_t edges)
|
|
{
|
|
view_begin_interactive(view, KIWMI_CURSOR_RESIZE, edges);
|
|
}
|
|
|
|
struct kiwmi_view *
|
|
view_create(
|
|
struct kiwmi_desktop *desktop,
|
|
enum kiwmi_view_type type,
|
|
const struct kiwmi_view_impl *impl)
|
|
{
|
|
struct kiwmi_view *view = malloc(sizeof(*view));
|
|
if (!view) {
|
|
wlr_log(WLR_ERROR, "Failed to allocate view");
|
|
return NULL;
|
|
}
|
|
|
|
view->desktop = desktop;
|
|
view->type = type;
|
|
view->impl = impl;
|
|
view->mapped = false;
|
|
view->hidden = true;
|
|
view->decoration = NULL;
|
|
|
|
view->x = 0;
|
|
view->y = 0;
|
|
|
|
wl_signal_init(&view->events.unmap);
|
|
wl_signal_init(&view->events.request_move);
|
|
wl_signal_init(&view->events.request_resize);
|
|
wl_signal_init(&view->events.post_render);
|
|
wl_signal_init(&view->events.pre_render);
|
|
|
|
return view;
|
|
}
|