2013-02-28 18:55:02 +00:00
|
|
|
/*
|
2013-09-09 16:37:33 +00:00
|
|
|
* This file is part of mpv video player.
|
2013-02-28 18:55:02 +00:00
|
|
|
* Copyright © 2008 Kristian Høgsberg
|
|
|
|
* Copyright © 2012-2013 Collabora, Ltd.
|
2013-09-09 16:37:33 +00:00
|
|
|
* Copyright © 2013 Alexander Preisinger <alexander.preisinger@gmail.com>
|
2013-02-28 18:55:02 +00:00
|
|
|
*
|
2013-09-09 16:37:33 +00:00
|
|
|
* mpv is free software; you can redistribute it and/or modify
|
2013-02-28 18:55:02 +00:00
|
|
|
* 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.
|
|
|
|
*
|
2013-09-09 16:37:33 +00:00
|
|
|
* mpv is distributed in the hope that it will be useful,
|
2013-02-28 18:55:02 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
2013-09-09 16:37:33 +00:00
|
|
|
* with mpv. If not, see <http://www.gnu.org/licenses/>.
|
2013-02-28 18:55:02 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <assert.h>
|
2013-05-01 13:41:33 +00:00
|
|
|
#include <poll.h>
|
2013-02-28 18:55:02 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <linux/input.h>
|
|
|
|
|
|
|
|
#include "config.h"
|
2013-12-17 01:39:45 +00:00
|
|
|
#include "bstr/bstr.h"
|
2013-12-17 01:02:25 +00:00
|
|
|
#include "options/options.h"
|
2013-12-17 01:39:45 +00:00
|
|
|
#include "common/msg.h"
|
2013-02-28 18:55:02 +00:00
|
|
|
#include "libavutil/common.h"
|
|
|
|
#include "talloc.h"
|
|
|
|
|
|
|
|
#include "wayland_common.h"
|
|
|
|
|
|
|
|
#include "vo.h"
|
|
|
|
#include "aspect.h"
|
|
|
|
#include "osdep/timer.h"
|
|
|
|
|
2013-12-17 00:23:09 +00:00
|
|
|
#include "input/input.h"
|
2014-01-04 15:59:22 +00:00
|
|
|
#include "input/event.h"
|
2013-12-17 00:23:09 +00:00
|
|
|
#include "input/keycodes.h"
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
#define MOD_SHIFT_MASK 0x01
|
|
|
|
#define MOD_ALT_MASK 0x02
|
|
|
|
#define MOD_CONTROL_MASK 0x04
|
|
|
|
|
|
|
|
static int lookupkey(int key);
|
|
|
|
|
2013-03-22 18:37:28 +00:00
|
|
|
static void hide_cursor(struct vo_wayland_state * wl);
|
|
|
|
static void show_cursor(struct vo_wayland_state * wl);
|
2013-08-23 09:29:09 +00:00
|
|
|
static void shedule_resize(struct vo_wayland_state *wl,
|
|
|
|
uint32_t edges,
|
|
|
|
int32_t width,
|
|
|
|
int32_t height);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static void vo_wayland_fullscreen (struct vo *vo);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
static const struct mp_keymap keymap[] = {
|
|
|
|
// special keys
|
|
|
|
{XKB_KEY_Pause, MP_KEY_PAUSE}, {XKB_KEY_Escape, MP_KEY_ESC},
|
|
|
|
{XKB_KEY_BackSpace, MP_KEY_BS}, {XKB_KEY_Tab, MP_KEY_TAB},
|
|
|
|
{XKB_KEY_Return, MP_KEY_ENTER}, {XKB_KEY_Menu, MP_KEY_MENU},
|
|
|
|
{XKB_KEY_Print, MP_KEY_PRINT},
|
|
|
|
|
|
|
|
// cursor keys
|
|
|
|
{XKB_KEY_Left, MP_KEY_LEFT}, {XKB_KEY_Right, MP_KEY_RIGHT},
|
|
|
|
{XKB_KEY_Up, MP_KEY_UP}, {XKB_KEY_Down, MP_KEY_DOWN},
|
|
|
|
|
|
|
|
// navigation block
|
|
|
|
{XKB_KEY_Insert, MP_KEY_INSERT}, {XKB_KEY_Delete, MP_KEY_DELETE},
|
|
|
|
{XKB_KEY_Home, MP_KEY_HOME}, {XKB_KEY_End, MP_KEY_END},
|
|
|
|
{XKB_KEY_Page_Up, MP_KEY_PAGE_UP}, {XKB_KEY_Page_Down, MP_KEY_PAGE_DOWN},
|
|
|
|
|
|
|
|
// F-keys
|
|
|
|
{XKB_KEY_F1, MP_KEY_F+1}, {XKB_KEY_F2, MP_KEY_F+2},
|
|
|
|
{XKB_KEY_F3, MP_KEY_F+3}, {XKB_KEY_F4, MP_KEY_F+4},
|
|
|
|
{XKB_KEY_F5, MP_KEY_F+5}, {XKB_KEY_F6, MP_KEY_F+6},
|
|
|
|
{XKB_KEY_F7, MP_KEY_F+7}, {XKB_KEY_F8, MP_KEY_F+8},
|
|
|
|
{XKB_KEY_F9, MP_KEY_F+9}, {XKB_KEY_F10, MP_KEY_F+10},
|
|
|
|
{XKB_KEY_F11, MP_KEY_F+11}, {XKB_KEY_F12, MP_KEY_F+12},
|
|
|
|
|
|
|
|
// numpad independent of numlock
|
|
|
|
{XKB_KEY_KP_Subtract, '-'}, {XKB_KEY_KP_Add, '+'},
|
|
|
|
{XKB_KEY_KP_Multiply, '*'}, {XKB_KEY_KP_Divide, '/'},
|
|
|
|
{XKB_KEY_KP_Enter, MP_KEY_KPENTER},
|
|
|
|
|
|
|
|
// numpad with numlock
|
|
|
|
{XKB_KEY_KP_0, MP_KEY_KP0}, {XKB_KEY_KP_1, MP_KEY_KP1},
|
|
|
|
{XKB_KEY_KP_2, MP_KEY_KP2}, {XKB_KEY_KP_3, MP_KEY_KP3},
|
|
|
|
{XKB_KEY_KP_4, MP_KEY_KP4}, {XKB_KEY_KP_5, MP_KEY_KP5},
|
|
|
|
{XKB_KEY_KP_6, MP_KEY_KP6}, {XKB_KEY_KP_7, MP_KEY_KP7},
|
|
|
|
{XKB_KEY_KP_8, MP_KEY_KP8}, {XKB_KEY_KP_9, MP_KEY_KP9},
|
|
|
|
{XKB_KEY_KP_Decimal, MP_KEY_KPDEC}, {XKB_KEY_KP_Separator, MP_KEY_KPDEC},
|
|
|
|
|
|
|
|
// numpad without numlock
|
|
|
|
{XKB_KEY_KP_Insert, MP_KEY_KPINS}, {XKB_KEY_KP_End, MP_KEY_KP1},
|
|
|
|
{XKB_KEY_KP_Down, MP_KEY_KP2}, {XKB_KEY_KP_Page_Down, MP_KEY_KP3},
|
|
|
|
{XKB_KEY_KP_Left, MP_KEY_KP4}, {XKB_KEY_KP_Begin, MP_KEY_KP5},
|
|
|
|
{XKB_KEY_KP_Right, MP_KEY_KP6}, {XKB_KEY_KP_Home, MP_KEY_KP7},
|
|
|
|
{XKB_KEY_KP_Up, MP_KEY_KP8}, {XKB_KEY_KP_Page_Up, MP_KEY_KP9},
|
|
|
|
{XKB_KEY_KP_Delete, MP_KEY_KPDEL},
|
|
|
|
|
|
|
|
{0, 0}
|
|
|
|
};
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
/** Wayland listeners **/
|
2013-12-04 19:21:15 +00:00
|
|
|
|
|
|
|
static void display_handle_error(void *data,
|
|
|
|
struct wl_display *display,
|
|
|
|
void *object_id,
|
|
|
|
uint32_t code,
|
|
|
|
const char *message)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
const char * error_type_msg = "";
|
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case WL_DISPLAY_ERROR_INVALID_OBJECT:
|
|
|
|
error_type_msg = "Invalid object";
|
|
|
|
break;
|
|
|
|
case WL_DISPLAY_ERROR_INVALID_METHOD:
|
|
|
|
error_type_msg = "Invalid method";
|
|
|
|
break;
|
|
|
|
case WL_DISPLAY_ERROR_NO_MEMORY:
|
|
|
|
error_type_msg = "No memory";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
MP_ERR(wl, "%s: %s\n", error_type_msg, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void display_handle_delete_id(void *data,
|
|
|
|
struct wl_display *display,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
MP_DBG(wl, "Object %u deleted\n", id);
|
|
|
|
}
|
|
|
|
|
2014-01-08 15:15:26 +00:00
|
|
|
static const struct wl_display_listener display_listener = {
|
2013-12-04 19:21:15 +00:00
|
|
|
display_handle_error,
|
|
|
|
display_handle_delete_id
|
|
|
|
};
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
static void ssurface_handle_ping(void *data,
|
|
|
|
struct wl_shell_surface *shell_surface,
|
|
|
|
uint32_t serial)
|
|
|
|
{
|
|
|
|
wl_shell_surface_pong(shell_surface, serial);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ssurface_handle_configure(void *data,
|
|
|
|
struct wl_shell_surface *shell_surface,
|
|
|
|
uint32_t edges,
|
|
|
|
int32_t width,
|
|
|
|
int32_t height)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
2013-08-23 09:29:09 +00:00
|
|
|
shedule_resize(wl, edges, width, height);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ssurface_handle_popup_done(void *data,
|
|
|
|
struct wl_shell_surface *shell_surface)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-01-08 15:15:26 +00:00
|
|
|
static const struct wl_shell_surface_listener shell_surface_listener = {
|
2013-02-28 18:55:02 +00:00
|
|
|
ssurface_handle_ping,
|
|
|
|
ssurface_handle_configure,
|
|
|
|
ssurface_handle_popup_done
|
|
|
|
};
|
|
|
|
|
|
|
|
static void output_handle_geometry(void *data,
|
|
|
|
struct wl_output *wl_output,
|
|
|
|
int32_t x,
|
|
|
|
int32_t y,
|
|
|
|
int32_t physical_width,
|
|
|
|
int32_t physical_height,
|
|
|
|
int32_t subpixel,
|
|
|
|
const char *make,
|
|
|
|
const char *model,
|
|
|
|
int32_t transform)
|
|
|
|
{
|
|
|
|
/* Ignore transforms for now */
|
|
|
|
switch (transform) {
|
|
|
|
case WL_OUTPUT_TRANSFORM_NORMAL:
|
|
|
|
case WL_OUTPUT_TRANSFORM_90:
|
|
|
|
case WL_OUTPUT_TRANSFORM_180:
|
|
|
|
case WL_OUTPUT_TRANSFORM_270:
|
|
|
|
case WL_OUTPUT_TRANSFORM_FLIPPED:
|
|
|
|
case WL_OUTPUT_TRANSFORM_FLIPPED_90:
|
|
|
|
case WL_OUTPUT_TRANSFORM_FLIPPED_180:
|
|
|
|
case WL_OUTPUT_TRANSFORM_FLIPPED_270:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void output_handle_mode(void *data,
|
|
|
|
struct wl_output *wl_output,
|
|
|
|
uint32_t flags,
|
|
|
|
int32_t width,
|
|
|
|
int32_t height,
|
|
|
|
int32_t refresh)
|
|
|
|
{
|
2013-05-02 18:56:59 +00:00
|
|
|
struct vo_wayland_output *output = data;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-02 18:56:59 +00:00
|
|
|
if (!output)
|
|
|
|
return;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-02 18:56:59 +00:00
|
|
|
output->width = width;
|
|
|
|
output->height = height;
|
|
|
|
output->flags = flags;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2014-01-08 15:15:26 +00:00
|
|
|
static const struct wl_output_listener output_listener = {
|
2013-02-28 18:55:02 +00:00
|
|
|
output_handle_geometry,
|
|
|
|
output_handle_mode
|
|
|
|
};
|
|
|
|
|
|
|
|
/* KEYBOARD LISTENER */
|
|
|
|
static void keyboard_handle_keymap(void *data,
|
|
|
|
struct wl_keyboard *wl_keyboard,
|
|
|
|
uint32_t format,
|
|
|
|
int32_t fd,
|
|
|
|
uint32_t size)
|
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
struct vo_wayland_state *wl = data;
|
2013-02-28 18:55:02 +00:00
|
|
|
char *map_str;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (format != WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1) {
|
2013-02-28 18:55:02 +00:00
|
|
|
close(fd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
map_str = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0);
|
|
|
|
if (map_str == MAP_FAILED) {
|
|
|
|
close(fd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-12 23:02:01 +00:00
|
|
|
wl->input.xkb.keymap = xkb_keymap_new_from_string(wl->input.xkb.context,
|
2013-07-18 15:35:28 +00:00
|
|
|
map_str,
|
|
|
|
XKB_KEYMAP_FORMAT_TEXT_V1,
|
|
|
|
0);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
munmap(map_str, size);
|
|
|
|
close(fd);
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!wl->input.xkb.keymap) {
|
2013-09-12 14:29:13 +00:00
|
|
|
MP_ERR(wl, "failed to compile keymap\n");
|
2013-02-28 18:55:02 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->input.xkb.state = xkb_state_new(wl->input.xkb.keymap);
|
|
|
|
if (!wl->input.xkb.state) {
|
2013-09-12 14:29:13 +00:00
|
|
|
MP_ERR(wl, "failed to create XKB state\n");
|
2013-07-18 15:35:28 +00:00
|
|
|
xkb_map_unref(wl->input.xkb.keymap);
|
|
|
|
wl->input.xkb.keymap = NULL;
|
2013-02-28 18:55:02 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void keyboard_handle_enter(void *data,
|
|
|
|
struct wl_keyboard *wl_keyboard,
|
|
|
|
uint32_t serial,
|
|
|
|
struct wl_surface *surface,
|
|
|
|
struct wl_array *keys)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void keyboard_handle_leave(void *data,
|
|
|
|
struct wl_keyboard *wl_keyboard,
|
|
|
|
uint32_t serial,
|
|
|
|
struct wl_surface *surface)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void keyboard_handle_key(void *data,
|
|
|
|
struct wl_keyboard *wl_keyboard,
|
|
|
|
uint32_t serial,
|
|
|
|
uint32_t time,
|
|
|
|
uint32_t key,
|
|
|
|
uint32_t state)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
uint32_t code, num_syms;
|
2013-05-01 11:34:59 +00:00
|
|
|
int mpkey;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
const xkb_keysym_t *syms;
|
|
|
|
xkb_keysym_t sym;
|
|
|
|
|
|
|
|
code = key + 8;
|
2013-07-18 15:35:28 +00:00
|
|
|
num_syms = xkb_key_get_syms(wl->input.xkb.state, code, &syms);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
sym = XKB_KEY_NoSymbol;
|
|
|
|
if (num_syms == 1)
|
|
|
|
sym = syms[0];
|
|
|
|
|
2013-05-01 11:34:59 +00:00
|
|
|
if (sym != XKB_KEY_NoSymbol && (mpkey = lookupkey(sym))) {
|
|
|
|
if (state == WL_KEYBOARD_KEY_STATE_PRESSED)
|
2013-07-02 16:56:50 +00:00
|
|
|
mp_input_put_key(wl->vo->input_ctx, mpkey | MP_KEY_STATE_DOWN);
|
2013-05-01 11:34:59 +00:00
|
|
|
else
|
2013-07-02 16:56:50 +00:00
|
|
|
mp_input_put_key(wl->vo->input_ctx, mpkey | MP_KEY_STATE_UP);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void keyboard_handle_modifiers(void *data,
|
|
|
|
struct wl_keyboard *wl_keyboard,
|
|
|
|
uint32_t serial,
|
|
|
|
uint32_t mods_depressed,
|
|
|
|
uint32_t mods_latched,
|
|
|
|
uint32_t mods_locked,
|
|
|
|
uint32_t group)
|
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
struct vo_wayland_state *wl = data;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
xkb_state_update_mask(wl->input.xkb.state,
|
|
|
|
mods_depressed,
|
|
|
|
mods_latched,
|
|
|
|
mods_locked,
|
|
|
|
0, 0, group);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2014-01-08 15:15:26 +00:00
|
|
|
static const struct wl_keyboard_listener keyboard_listener = {
|
2013-02-28 18:55:02 +00:00
|
|
|
keyboard_handle_keymap,
|
|
|
|
keyboard_handle_enter,
|
|
|
|
keyboard_handle_leave,
|
|
|
|
keyboard_handle_key,
|
|
|
|
keyboard_handle_modifiers
|
|
|
|
};
|
|
|
|
|
|
|
|
/* POINTER LISTENER */
|
|
|
|
static void pointer_handle_enter(void *data,
|
|
|
|
struct wl_pointer *pointer,
|
|
|
|
uint32_t serial,
|
|
|
|
struct wl_surface *surface,
|
|
|
|
wl_fixed_t sx_w,
|
|
|
|
wl_fixed_t sy_w)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->cursor.serial = serial;
|
|
|
|
wl->cursor.pointer = pointer;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-09-24 16:53:19 +00:00
|
|
|
/* Release the left button on pointer enter again
|
|
|
|
* because after moving the shell surface no release event is sent */
|
2013-09-24 16:53:24 +00:00
|
|
|
mp_input_put_key(wl->vo->input_ctx, MP_MOUSE_BTN0 | MP_KEY_STATE_UP);
|
2013-05-19 11:04:59 +00:00
|
|
|
show_cursor(wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pointer_handle_leave(void *data,
|
|
|
|
struct wl_pointer *pointer,
|
|
|
|
uint32_t serial,
|
|
|
|
struct wl_surface *surface)
|
|
|
|
{
|
2013-07-02 15:31:43 +00:00
|
|
|
struct vo_wayland_state *wl = data;
|
2013-07-02 16:56:50 +00:00
|
|
|
mp_input_put_key(wl->vo->input_ctx, MP_KEY_MOUSE_LEAVE);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pointer_handle_motion(void *data,
|
|
|
|
struct wl_pointer *pointer,
|
|
|
|
uint32_t time,
|
|
|
|
wl_fixed_t sx_w,
|
|
|
|
wl_fixed_t sy_w)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->cursor.pointer = pointer;
|
2013-10-25 08:27:58 +00:00
|
|
|
wl->window.mouse_x = wl_fixed_to_int(sx_w);
|
|
|
|
wl->window.mouse_y = wl_fixed_to_int(sy_w);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-10-25 08:27:58 +00:00
|
|
|
vo_mouse_movement(wl->vo, wl->window.mouse_x,
|
|
|
|
wl->window.mouse_y);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pointer_handle_button(void *data,
|
|
|
|
struct wl_pointer *pointer,
|
|
|
|
uint32_t serial,
|
|
|
|
uint32_t time,
|
|
|
|
uint32_t button,
|
|
|
|
uint32_t state)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
2013-07-02 16:56:50 +00:00
|
|
|
mp_input_put_key(wl->vo->input_ctx, MP_MOUSE_BTN0 + (button - BTN_LEFT) |
|
2013-07-18 15:35:28 +00:00
|
|
|
((state == WL_POINTER_BUTTON_STATE_PRESSED)
|
|
|
|
? MP_KEY_STATE_DOWN : MP_KEY_STATE_UP));
|
2013-09-24 16:53:24 +00:00
|
|
|
|
2013-10-25 08:27:58 +00:00
|
|
|
if (!mp_input_test_dragging(wl->vo->input_ctx, wl->window.mouse_x, wl->window.mouse_y) &&
|
2013-09-24 17:15:55 +00:00
|
|
|
(button == BTN_LEFT) && (state == WL_POINTER_BUTTON_STATE_PRESSED))
|
2013-09-24 16:53:24 +00:00
|
|
|
wl_shell_surface_move(wl->window.shell_surface, wl->input.seat, serial);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pointer_handle_axis(void *data,
|
|
|
|
struct wl_pointer *pointer,
|
|
|
|
uint32_t time,
|
|
|
|
uint32_t axis,
|
|
|
|
wl_fixed_t value)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
2013-07-25 16:16:08 +00:00
|
|
|
// value is 10.00 on a normal mouse wheel
|
|
|
|
// scale it down to 1.00 for multipliying it with the commands
|
2013-02-28 18:55:02 +00:00
|
|
|
if (axis == WL_POINTER_AXIS_VERTICAL_SCROLL) {
|
|
|
|
if (value > 0)
|
2013-07-25 16:16:08 +00:00
|
|
|
mp_input_put_axis(wl->vo->input_ctx, MP_AXIS_DOWN,
|
|
|
|
wl_fixed_to_double(value)*0.1);
|
2013-02-28 18:55:02 +00:00
|
|
|
if (value < 0)
|
2013-07-25 16:16:08 +00:00
|
|
|
mp_input_put_axis(wl->vo->input_ctx, MP_AXIS_UP,
|
|
|
|
wl_fixed_to_double(value)*-0.1);
|
|
|
|
}
|
|
|
|
else if (axis == WL_POINTER_AXIS_HORIZONTAL_SCROLL) {
|
|
|
|
if (value > 0)
|
|
|
|
mp_input_put_axis(wl->vo->input_ctx, MP_AXIS_RIGHT,
|
|
|
|
wl_fixed_to_double(value)*0.1);
|
|
|
|
if (value < 0)
|
|
|
|
mp_input_put_axis(wl->vo->input_ctx, MP_AXIS_LEFT,
|
|
|
|
wl_fixed_to_double(value)*-0.1);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_pointer_listener pointer_listener = {
|
|
|
|
pointer_handle_enter,
|
|
|
|
pointer_handle_leave,
|
|
|
|
pointer_handle_motion,
|
|
|
|
pointer_handle_button,
|
|
|
|
pointer_handle_axis,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void seat_handle_capabilities(void *data,
|
|
|
|
struct wl_seat *seat,
|
|
|
|
enum wl_seat_capability caps)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if ((caps & WL_SEAT_CAPABILITY_KEYBOARD) && !wl->input.keyboard) {
|
|
|
|
wl->input.keyboard = wl_seat_get_keyboard(seat);
|
|
|
|
wl_keyboard_add_listener(wl->input.keyboard, &keyboard_listener, wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
2013-07-18 15:35:28 +00:00
|
|
|
else if (!(caps & WL_SEAT_CAPABILITY_KEYBOARD) && wl->input.keyboard) {
|
|
|
|
wl_keyboard_destroy(wl->input.keyboard);
|
|
|
|
wl->input.keyboard = NULL;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
2013-07-18 15:35:28 +00:00
|
|
|
if ((caps & WL_SEAT_CAPABILITY_POINTER) && !wl->input.pointer) {
|
|
|
|
wl->input.pointer = wl_seat_get_pointer(seat);
|
|
|
|
wl_pointer_add_listener(wl->input.pointer, &pointer_listener, wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
2013-07-18 15:35:28 +00:00
|
|
|
else if (!(caps & WL_SEAT_CAPABILITY_POINTER) && wl->input.pointer) {
|
|
|
|
wl_pointer_destroy(wl->input.pointer);
|
|
|
|
wl->input.pointer = NULL;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_seat_listener seat_listener = {
|
|
|
|
seat_handle_capabilities,
|
|
|
|
};
|
|
|
|
|
2014-01-04 12:48:50 +00:00
|
|
|
static void data_offer_handle_offer(void *data,
|
|
|
|
struct wl_data_offer *offer,
|
|
|
|
const char *mime_type)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
if (strcmp(mime_type, "text/uri-list") != 0)
|
|
|
|
MP_VERBOSE(wl, "unsupported mime type for drag and drop: %s\n", mime_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_data_offer_listener data_offer_listener = {
|
|
|
|
data_offer_handle_offer,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void data_device_handle_data_offer(void *data,
|
|
|
|
struct wl_data_device *wl_data_device,
|
|
|
|
struct wl_data_offer *id)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
if (wl->input.offer) {
|
|
|
|
MP_ERR(wl, "There is already a dnd entry point.\n");
|
|
|
|
wl_data_offer_destroy(wl->input.offer);
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->input.offer = id;
|
|
|
|
wl_data_offer_add_listener(id, &data_offer_listener, wl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void data_device_handle_enter(void *data,
|
|
|
|
struct wl_data_device *wl_data_device,
|
|
|
|
uint32_t serial,
|
|
|
|
struct wl_surface *surface,
|
|
|
|
wl_fixed_t x,
|
|
|
|
wl_fixed_t y,
|
|
|
|
struct wl_data_offer *id)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
if (wl->input.offer != id)
|
|
|
|
MP_FATAL(wl, "Fatal dnd error (Please report this issue)\n");
|
|
|
|
|
|
|
|
wl_data_offer_accept(id, serial, "text/uri-list");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void data_device_handle_leave(void *data,
|
|
|
|
struct wl_data_device *wl_data_device)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
if (wl->input.offer) {
|
|
|
|
wl_data_offer_destroy(wl->input.offer);
|
|
|
|
wl->input.offer = NULL;
|
|
|
|
}
|
|
|
|
// dnd fd is closed on POLLHUP
|
|
|
|
}
|
|
|
|
|
|
|
|
static void data_device_handle_motion(void *data,
|
|
|
|
struct wl_data_device *wl_data_device,
|
|
|
|
uint32_t time,
|
|
|
|
wl_fixed_t x,
|
|
|
|
wl_fixed_t y)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void data_device_handle_drop(void *data,
|
|
|
|
struct wl_data_device *wl_data_device)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
|
|
|
int pipefd[2];
|
|
|
|
|
|
|
|
if (pipe(pipefd) == -1) {
|
|
|
|
MP_FATAL(wl, "can't create pipe for dnd communication\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl->input.dnd_fd = pipefd[0];
|
|
|
|
wl_data_offer_receive(wl->input.offer, "text/uri-list", pipefd[1]);
|
|
|
|
close(pipefd[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void data_device_handle_selection(void *data,
|
|
|
|
struct wl_data_device *wl_data_device,
|
|
|
|
struct wl_data_offer *id)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_data_device_listener data_device_listener = {
|
|
|
|
data_device_handle_data_offer,
|
|
|
|
data_device_handle_enter,
|
|
|
|
data_device_handle_leave,
|
|
|
|
data_device_handle_motion,
|
|
|
|
data_device_handle_drop,
|
|
|
|
data_device_handle_selection
|
|
|
|
};
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
static void registry_handle_global (void *data,
|
2014-01-08 15:21:49 +00:00
|
|
|
struct wl_registry *reg,
|
2013-02-28 18:55:02 +00:00
|
|
|
uint32_t id,
|
|
|
|
const char *interface,
|
|
|
|
uint32_t version)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = data;
|
|
|
|
|
|
|
|
if (strcmp(interface, "wl_compositor") == 0) {
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.compositor = wl_registry_bind(reg, id,
|
|
|
|
&wl_compositor_interface, 1);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (strcmp(interface, "wl_shell") == 0) {
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.shell = wl_registry_bind(reg, id, &wl_shell_interface, 1);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (strcmp(interface, "wl_shm") == 0) {
|
|
|
|
|
2013-08-19 12:07:17 +00:00
|
|
|
wl->display.shm = wl_registry_bind(reg, id, &wl_shm_interface, 1);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (strcmp(interface, "wl_output") == 0) {
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
struct vo_wayland_output *output =
|
|
|
|
talloc_zero(wl, struct vo_wayland_output);
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
output->id = id;
|
2013-07-18 15:35:28 +00:00
|
|
|
output->output = wl_registry_bind(reg, id, &wl_output_interface, 1);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-02 18:56:59 +00:00
|
|
|
wl_output_add_listener(output->output, &output_listener, output);
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_list_insert(&wl->display.output_list, &output->link);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2014-01-04 12:48:50 +00:00
|
|
|
else if (strcmp(interface, "wl_data_device_manager") == 0) {
|
|
|
|
|
|
|
|
wl->input.devman = wl_registry_bind(reg,
|
|
|
|
id,
|
|
|
|
&wl_data_device_manager_interface,
|
|
|
|
1);
|
|
|
|
}
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
else if (strcmp(interface, "wl_seat") == 0) {
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->input.seat = wl_registry_bind(reg, id, &wl_seat_interface, 1);
|
|
|
|
wl_seat_add_listener(wl->input.seat, &seat_listener, wl);
|
2014-01-04 12:48:50 +00:00
|
|
|
|
|
|
|
wl->input.datadev = wl_data_device_manager_get_data_device(
|
|
|
|
wl->input.devman, wl->input.seat);
|
|
|
|
wl_data_device_add_listener(wl->input.datadev, &data_device_listener, wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void registry_handle_global_remove (void *data,
|
|
|
|
struct wl_registry *registry,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_registry_listener registry_listener = {
|
|
|
|
registry_handle_global,
|
|
|
|
registry_handle_global_remove
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*** internal functions ***/
|
|
|
|
|
|
|
|
static int lookupkey(int key)
|
|
|
|
{
|
|
|
|
static const char *passthrough_keys
|
|
|
|
= " -+*/<>`~!@#$%^&()_{}:;\"\',.?\\|=[]";
|
|
|
|
|
|
|
|
int mpkey = 0;
|
|
|
|
if ((key >= 'a' && key <= 'z') ||
|
|
|
|
(key >= 'A' && key <= 'Z') ||
|
|
|
|
(key >= '0' && key <= '9') ||
|
|
|
|
(key > 0 && key < 256 && strchr(passthrough_keys, key)))
|
|
|
|
mpkey = key;
|
|
|
|
|
|
|
|
if (!mpkey)
|
|
|
|
mpkey = lookup_keymap_table(keymap, key);
|
|
|
|
|
|
|
|
return mpkey;
|
|
|
|
}
|
|
|
|
|
2013-03-22 18:37:28 +00:00
|
|
|
static void hide_cursor (struct vo_wayland_state *wl)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!wl->cursor.pointer)
|
2013-02-28 18:55:02 +00:00
|
|
|
return;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_pointer_set_cursor(wl->cursor.pointer, wl->cursor.serial, NULL, 0, 0);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-03-22 18:37:28 +00:00
|
|
|
static void show_cursor (struct vo_wayland_state *wl)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!wl->cursor.pointer)
|
2013-02-28 18:55:02 +00:00
|
|
|
return;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
struct wl_cursor_image *image = wl->cursor.default_cursor->images[0];
|
|
|
|
struct wl_buffer *buffer = wl_cursor_image_get_buffer(image);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_pointer_set_cursor(wl->cursor.pointer,
|
|
|
|
wl->cursor.serial,
|
|
|
|
wl->cursor.surface,
|
|
|
|
image->hotspot_x,
|
|
|
|
image->hotspot_y);
|
|
|
|
|
|
|
|
wl_surface_attach(wl->cursor.surface, buffer, 0, 0);
|
|
|
|
wl_surface_damage(wl->cursor.surface, 0, 0, image->width, image->height);
|
|
|
|
wl_surface_commit(wl->cursor.surface);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-08-23 09:29:09 +00:00
|
|
|
static void shedule_resize(struct vo_wayland_state *wl,
|
|
|
|
uint32_t edges,
|
|
|
|
int32_t width,
|
|
|
|
int32_t height)
|
2013-05-07 18:51:09 +00:00
|
|
|
{
|
2013-08-23 09:29:09 +00:00
|
|
|
int32_t minimum_size = 150;
|
|
|
|
int32_t x, y;
|
|
|
|
float temp_aspect = width / (float) MPMAX(height, 1);
|
|
|
|
|
2014-01-07 20:12:24 +00:00
|
|
|
MP_DBG(wl, "shedule resize: %dx%d\n", width, height);
|
|
|
|
|
2013-08-23 09:29:09 +00:00
|
|
|
if (width < minimum_size)
|
|
|
|
width = minimum_size;
|
|
|
|
|
|
|
|
if (height < minimum_size)
|
|
|
|
height = minimum_size;
|
|
|
|
|
2013-10-25 08:22:40 +00:00
|
|
|
// don't keep the aspect ration in fullscreen mode, because the compositor
|
|
|
|
// shows the desktop in the border regions if the video has not the same
|
|
|
|
// aspect ration as the screen
|
2014-01-02 19:30:21 +00:00
|
|
|
/* if only the height is changed we have to calculate the width
|
|
|
|
* in any other case we calculate the height */
|
|
|
|
switch (edges) {
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_TOP:
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_BOTTOM:
|
|
|
|
width = wl->window.aspect * height;
|
|
|
|
break;
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_LEFT:
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_RIGHT:
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_TOP_LEFT: // just a preference
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_TOP_RIGHT:
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_BOTTOM_LEFT:
|
|
|
|
case WL_SHELL_SURFACE_RESIZE_BOTTOM_RIGHT:
|
|
|
|
height = (1 / wl->window.aspect) * width;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (wl->window.aspect < temp_aspect)
|
2013-08-23 09:29:09 +00:00
|
|
|
width = wl->window.aspect * height;
|
2014-01-02 19:30:21 +00:00
|
|
|
else
|
2013-08-23 09:29:09 +00:00
|
|
|
height = (1 / wl->window.aspect) * width;
|
2014-01-02 19:30:21 +00:00
|
|
|
break;
|
2013-05-07 18:51:09 +00:00
|
|
|
}
|
2013-08-23 09:29:09 +00:00
|
|
|
|
|
|
|
if (edges & WL_SHELL_SURFACE_RESIZE_LEFT)
|
|
|
|
x = wl->window.width - width;
|
|
|
|
else
|
|
|
|
x = 0;
|
|
|
|
|
|
|
|
if (edges & WL_SHELL_SURFACE_RESIZE_TOP)
|
|
|
|
y = wl->window.height - height;
|
2013-05-07 18:51:09 +00:00
|
|
|
else
|
2013-08-23 09:29:09 +00:00
|
|
|
y = 0;
|
|
|
|
|
|
|
|
wl->window.sh_width = width;
|
|
|
|
wl->window.sh_height = height;
|
|
|
|
wl->window.sh_x = x;
|
|
|
|
wl->window.sh_y = y;
|
|
|
|
wl->window.events |= VO_EVENT_RESIZE;
|
2013-10-25 08:22:40 +00:00
|
|
|
wl->vo->dwidth = width;
|
|
|
|
wl->vo->dheight = height;
|
2013-05-07 18:51:09 +00:00
|
|
|
}
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
static bool create_display (struct vo_wayland_state *wl)
|
|
|
|
{
|
2014-01-07 00:06:28 +00:00
|
|
|
if (wl->vo->probing && !getenv("XDG_RUNTIME_DIR"))
|
|
|
|
return false;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.display = wl_display_connect(NULL);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-09-10 15:52:29 +00:00
|
|
|
if (!wl->display.display) {
|
2013-09-10 16:33:43 +00:00
|
|
|
MP_MSG(wl, wl->vo->probing ? MSGL_V : MSGL_ERR,
|
|
|
|
"failed to connect to a wayland server: "
|
|
|
|
"check if a wayland compositor is running\n");
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
return false;
|
2013-09-10 15:52:29 +00:00
|
|
|
}
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-12-04 19:21:15 +00:00
|
|
|
wl_display_add_listener(wl->display.display, &display_listener, wl);
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.registry = wl_display_get_registry(wl->display.display);
|
|
|
|
wl_registry_add_listener(wl->display.registry, ®istry_listener, wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_display_dispatch(wl->display.display);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.display_fd = wl_display_get_fd(wl->display.display);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_display (struct vo_wayland_state *wl)
|
|
|
|
{
|
2014-01-08 21:16:41 +00:00
|
|
|
struct vo_wayland_output *output = NULL;
|
|
|
|
struct vo_wayland_output *tmp = NULL;
|
2014-01-08 19:58:32 +00:00
|
|
|
|
2014-01-08 21:16:41 +00:00
|
|
|
wl_list_for_each_safe(output, tmp, &wl->display.output_list, link) {
|
2014-01-08 19:58:32 +00:00
|
|
|
if (output && output->output) {
|
|
|
|
wl_output_destroy(output->output);
|
|
|
|
output->output = NULL;
|
2014-01-08 21:16:41 +00:00
|
|
|
wl_list_remove(&output->link);
|
2014-01-08 19:58:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wl->display.shm)
|
|
|
|
wl_shm_destroy(wl->display.shm);
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (wl->display.shell)
|
|
|
|
wl_shell_destroy(wl->display.shell);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (wl->display.compositor)
|
|
|
|
wl_compositor_destroy(wl->display.compositor);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2014-01-06 23:59:25 +00:00
|
|
|
if (wl->display.registry)
|
|
|
|
wl_registry_destroy(wl->display.registry);
|
2014-01-08 19:58:32 +00:00
|
|
|
|
2014-01-06 23:59:25 +00:00
|
|
|
if (wl->display.display) {
|
|
|
|
wl_display_flush(wl->display.display);
|
|
|
|
wl_display_disconnect(wl->display.display);
|
|
|
|
}
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
static bool create_window (struct vo_wayland_state *wl)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->window.surface = wl_compositor_create_surface(wl->display.compositor);
|
|
|
|
wl->window.shell_surface = wl_shell_get_shell_surface(wl->display.shell,
|
|
|
|
wl->window.surface);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-31 17:19:44 +00:00
|
|
|
if (!wl->window.shell_surface) {
|
2013-09-12 14:29:13 +00:00
|
|
|
MP_ERR(wl, "creating shell surface failed\n");
|
2013-07-18 15:35:28 +00:00
|
|
|
return false;
|
2013-07-31 17:19:44 +00:00
|
|
|
}
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_shell_surface_add_listener(wl->window.shell_surface,
|
|
|
|
&shell_surface_listener, wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_shell_surface_set_toplevel(wl->window.shell_surface);
|
|
|
|
wl_shell_surface_set_class(wl->window.shell_surface, "mpv");
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
return true;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_window (struct vo_wayland_state *wl)
|
|
|
|
{
|
2014-01-06 23:59:25 +00:00
|
|
|
if (wl->window.shell_surface)
|
|
|
|
wl_shell_surface_destroy(wl->window.shell_surface);
|
2014-01-08 19:58:32 +00:00
|
|
|
|
2014-01-06 23:59:25 +00:00
|
|
|
if (wl->window.surface)
|
|
|
|
wl_surface_destroy(wl->window.surface);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
static bool create_cursor (struct vo_wayland_state *wl)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
2013-09-10 15:52:29 +00:00
|
|
|
if (!wl->display.shm) {
|
|
|
|
MP_ERR(wl->vo, "no shm interface available\n");
|
2013-08-19 12:07:17 +00:00
|
|
|
return false;
|
2013-09-10 15:52:29 +00:00
|
|
|
}
|
2013-08-19 12:07:17 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->cursor.surface =
|
|
|
|
wl_compositor_create_surface(wl->display.compositor);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-08-19 12:07:17 +00:00
|
|
|
if (!wl->cursor.surface)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
wl->cursor.theme = wl_cursor_theme_load(NULL, 32, wl->display.shm);
|
|
|
|
wl->cursor.default_cursor = wl_cursor_theme_get_cursor(wl->cursor.theme,
|
|
|
|
"left_ptr");
|
|
|
|
return true;
|
2013-07-18 15:35:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_cursor (struct vo_wayland_state *wl)
|
|
|
|
{
|
|
|
|
if (wl->cursor.theme)
|
|
|
|
wl_cursor_theme_destroy(wl->cursor.theme);
|
|
|
|
|
|
|
|
if (wl->cursor.surface)
|
|
|
|
wl_surface_destroy(wl->cursor.surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool create_input (struct vo_wayland_state *wl)
|
|
|
|
{
|
|
|
|
wl->input.xkb.context = xkb_context_new(0);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!wl->input.xkb.context) {
|
2013-09-12 14:29:13 +00:00
|
|
|
MP_ERR(wl, "failed to initialize input: check xkbcommon\n");
|
2013-07-18 15:35:28 +00:00
|
|
|
return false;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
2013-07-18 15:35:28 +00:00
|
|
|
|
2014-01-04 12:48:50 +00:00
|
|
|
wl->input.dnd_fd = -1;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
return true;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_input (struct vo_wayland_state *wl)
|
|
|
|
{
|
2013-08-26 16:35:57 +00:00
|
|
|
if (wl->input.keyboard) {
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_keyboard_destroy(wl->input.keyboard);
|
2013-08-26 16:35:57 +00:00
|
|
|
xkb_map_unref(wl->input.xkb.keymap);
|
|
|
|
xkb_state_unref(wl->input.xkb.state);
|
|
|
|
}
|
2014-01-08 19:58:32 +00:00
|
|
|
|
2014-01-06 23:59:25 +00:00
|
|
|
if (wl->input.xkb.context)
|
|
|
|
xkb_context_unref(wl->input.xkb.context);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (wl->input.pointer)
|
|
|
|
wl_pointer_destroy(wl->input.pointer);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2014-01-04 12:48:50 +00:00
|
|
|
if (wl->input.datadev)
|
|
|
|
wl_data_device_destroy(wl->input.datadev);
|
|
|
|
|
|
|
|
if (wl->input.devman)
|
|
|
|
wl_data_device_manager_destroy(wl->input.devman);
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (wl->input.seat)
|
|
|
|
wl_seat_destroy(wl->input.seat);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*** mplayer2 interface ***/
|
|
|
|
|
|
|
|
int vo_wayland_init (struct vo *vo)
|
|
|
|
{
|
|
|
|
vo->wayland = talloc_zero(NULL, struct vo_wayland_state);
|
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
|
|
|
wl->vo = vo;
|
2013-09-12 14:29:13 +00:00
|
|
|
wl->log = mp_log_new(wl, vo->log, "wayland");
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2014-01-08 20:49:18 +00:00
|
|
|
wl_list_init(&wl->display.output_list);
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!create_input(wl)
|
|
|
|
|| !create_display(wl)
|
|
|
|
|| !create_window(wl)
|
|
|
|
|| !create_cursor(wl))
|
|
|
|
{
|
2014-01-06 23:59:25 +00:00
|
|
|
vo_wayland_uninit(vo);
|
2013-02-28 18:55:02 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
vo->event_fd = wl->display.display_fd;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
return true;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void vo_wayland_uninit (struct vo *vo)
|
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2013-07-18 15:35:28 +00:00
|
|
|
destroy_cursor(wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
destroy_window(wl);
|
|
|
|
destroy_display(wl);
|
2013-07-18 15:35:28 +00:00
|
|
|
destroy_input(wl);
|
2013-02-28 18:55:02 +00:00
|
|
|
talloc_free(wl);
|
|
|
|
vo->wayland = NULL;
|
|
|
|
}
|
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static void vo_wayland_ontop (struct vo *vo)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
2014-01-07 20:12:24 +00:00
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
|
|
|
MP_DBG(wl, "going ontop\n");
|
|
|
|
vo->opts->ontop = 1;
|
|
|
|
wl_shell_surface_set_toplevel(wl->window.shell_surface);
|
|
|
|
shedule_resize(wl, 0, wl->window.width, wl->window.height);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static void vo_wayland_border (struct vo *vo)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
|
|
|
/* wayland clienst have to do the decorations themself
|
|
|
|
* (client side decorations) but there is no such code implement nor
|
|
|
|
* do I plan on implementing something like client side decorations
|
|
|
|
*
|
|
|
|
* The only exception would be resizing on when clicking and dragging
|
|
|
|
* on the border region of the window but this should be discussed at first
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static void vo_wayland_fullscreen (struct vo *vo)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2014-01-04 12:57:47 +00:00
|
|
|
if (!wl->display.shell)
|
2013-02-28 18:55:02 +00:00
|
|
|
return;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
struct wl_output *fs_output = wl->display.fs_output;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 12:05:39 +00:00
|
|
|
if (vo->opts->fullscreen) {
|
2013-09-12 14:38:44 +00:00
|
|
|
MP_DBG(wl, "going fullscreen\n");
|
2013-10-25 08:22:40 +00:00
|
|
|
wl->window.is_fullscreen = true;
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->window.p_width = wl->window.width;
|
|
|
|
wl->window.p_height = wl->window.height;
|
|
|
|
wl_shell_surface_set_fullscreen(wl->window.shell_surface,
|
2013-07-16 14:51:27 +00:00
|
|
|
WL_SHELL_SURFACE_FULLSCREEN_METHOD_DEFAULT,
|
2013-02-28 18:55:02 +00:00
|
|
|
0, fs_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
2013-09-12 14:38:44 +00:00
|
|
|
MP_DBG(wl, "leaving fullscreen\n");
|
2013-10-25 08:22:40 +00:00
|
|
|
wl->window.is_fullscreen = false;
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_shell_surface_set_toplevel(wl->window.shell_surface);
|
2013-08-23 09:29:09 +00:00
|
|
|
shedule_resize(wl, 0, wl->window.p_width, wl->window.p_height);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static int vo_wayland_check_events (struct vo *vo)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2013-07-18 15:35:28 +00:00
|
|
|
struct wl_display *dp = wl->display.display;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-01 13:41:33 +00:00
|
|
|
wl_display_dispatch_pending(dp);
|
|
|
|
wl_display_flush(dp);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-05-01 13:41:33 +00:00
|
|
|
struct pollfd fd = {
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.display_fd,
|
2013-05-02 21:24:14 +00:00
|
|
|
POLLIN | POLLOUT | POLLERR | POLLHUP,
|
2013-05-01 13:41:33 +00:00
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
/* wl_display_dispatch is blocking
|
2013-05-02 21:24:14 +00:00
|
|
|
* wl_dipslay_dispatch_pending is non-blocking but does not read from the fd
|
2013-05-01 13:41:33 +00:00
|
|
|
*
|
|
|
|
* when pausing no input events get queued so we have to check if there
|
|
|
|
* are events to read from the file descriptor through poll */
|
|
|
|
if (poll(&fd, 1, 0) > 0) {
|
2013-09-19 15:32:22 +00:00
|
|
|
if (fd.revents & POLLERR || fd.revents & POLLHUP) {
|
|
|
|
MP_FATAL(wl, "error occurred on the display fd: "
|
|
|
|
"closing file descriptor\n");
|
|
|
|
close(wl->display.display_fd);
|
|
|
|
mp_input_put_key(vo->input_ctx, MP_KEY_CLOSE_WIN);
|
|
|
|
}
|
2013-05-01 13:41:33 +00:00
|
|
|
if (fd.revents & POLLIN)
|
|
|
|
wl_display_dispatch(dp);
|
|
|
|
if (fd.revents & POLLOUT)
|
|
|
|
wl_display_flush(dp);
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2014-01-04 12:48:50 +00:00
|
|
|
/* If drag & drop was ended poll the file descriptor from the offer if
|
|
|
|
* there is data to read.
|
|
|
|
* We only accept the mime type text/uri-list.
|
|
|
|
*/
|
|
|
|
if (wl->input.dnd_fd != -1) {
|
|
|
|
fd.fd = wl->input.dnd_fd;
|
|
|
|
fd.events = POLLIN | POLLHUP | POLLERR;
|
|
|
|
|
|
|
|
if (poll(&fd, 1, 0) > 0) {
|
|
|
|
if (fd.revents & POLLERR) {
|
|
|
|
MP_ERR(wl, "error occured on the drag&drop fd\n");
|
|
|
|
close(wl->input.dnd_fd);
|
|
|
|
wl->input.dnd_fd = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fd.revents & POLLIN) {
|
|
|
|
int const to_read = 2048;
|
|
|
|
char *buffer = malloc(to_read);
|
|
|
|
size_t buffer_len = to_read;
|
|
|
|
size_t str_len = 0;
|
|
|
|
int has_read = 0;
|
|
|
|
|
2014-01-04 19:28:52 +00:00
|
|
|
while (0 < (has_read = read(fd.fd, buffer+str_len, to_read))) {
|
2014-01-04 12:48:50 +00:00
|
|
|
if (buffer_len + to_read < buffer_len) {
|
|
|
|
MP_ERR(wl, "Integer overflow while reading from fd\n");
|
|
|
|
free(buffer);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
str_len += has_read;
|
|
|
|
buffer_len += to_read;
|
|
|
|
buffer = realloc(buffer, buffer_len);
|
|
|
|
|
|
|
|
if (has_read < to_read) {
|
|
|
|
buffer[str_len] = 0;
|
|
|
|
struct bstr file_list = bstr0(buffer);
|
2014-01-04 18:32:09 +00:00
|
|
|
mp_event_drop_mime_data(vo->input_ctx, "text/uri-list",
|
|
|
|
file_list);
|
2014-01-08 19:58:32 +00:00
|
|
|
free(buffer);
|
2014-01-04 12:48:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fd.revents & POLLHUP) {
|
|
|
|
close(wl->input.dnd_fd);
|
|
|
|
wl->input.dnd_fd = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-23 09:29:09 +00:00
|
|
|
// window events are reset by the resizing code
|
|
|
|
return wl->window.events;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-05-16 14:43:34 +00:00
|
|
|
static void vo_wayland_update_screeninfo (struct vo *vo)
|
2013-02-28 18:55:02 +00:00
|
|
|
{
|
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2013-03-04 21:41:27 +00:00
|
|
|
struct mp_vo_opts *opts = vo->opts;
|
2013-05-02 18:56:59 +00:00
|
|
|
bool mode_received = false;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_display_roundtrip(wl->display.display);
|
2013-02-28 18:55:02 +00:00
|
|
|
|
2013-03-04 16:40:21 +00:00
|
|
|
vo->xinerama_x = vo->xinerama_y = 0;
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
int screen_id = 0;
|
|
|
|
|
|
|
|
struct vo_wayland_output *output;
|
|
|
|
struct vo_wayland_output *first_output = NULL;
|
|
|
|
struct vo_wayland_output *fsscreen_output = NULL;
|
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_list_for_each_reverse(output, &wl->display.output_list, link) {
|
2013-05-02 18:56:59 +00:00
|
|
|
if (!output || !output->width)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
mode_received = true;
|
|
|
|
|
2013-03-04 21:41:27 +00:00
|
|
|
if (opts->fsscreen_id == screen_id)
|
2013-02-28 18:55:02 +00:00
|
|
|
fsscreen_output = output;
|
|
|
|
|
|
|
|
if (!first_output)
|
|
|
|
first_output = output;
|
|
|
|
|
|
|
|
screen_id++;
|
|
|
|
}
|
|
|
|
|
2013-05-02 18:56:59 +00:00
|
|
|
if (!mode_received) {
|
2013-09-12 14:29:13 +00:00
|
|
|
MP_ERR(wl, "no output mode detected\n");
|
2013-05-02 18:56:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-02-28 18:55:02 +00:00
|
|
|
if (fsscreen_output) {
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.fs_output = fsscreen_output->output;
|
2013-03-04 21:41:27 +00:00
|
|
|
opts->screenwidth = fsscreen_output->width;
|
|
|
|
opts->screenheight = fsscreen_output->height;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
else {
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->display.fs_output = NULL; /* current output is always 0 */
|
2013-02-28 18:55:02 +00:00
|
|
|
|
|
|
|
if (first_output) {
|
2013-03-04 21:41:27 +00:00
|
|
|
opts->screenwidth = first_output->width;
|
|
|
|
opts->screenheight = first_output->height;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-07 20:12:24 +00:00
|
|
|
wl->window.fs_width = opts->screenwidth;
|
|
|
|
wl->window.fs_height = opts->screenheight;
|
2013-02-28 18:55:02 +00:00
|
|
|
}
|
|
|
|
|
2013-05-19 11:04:59 +00:00
|
|
|
int vo_wayland_control (struct vo *vo, int *events, int request, void *arg)
|
2013-05-15 16:17:18 +00:00
|
|
|
{
|
2013-05-19 11:04:59 +00:00
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_display_dispatch_pending(wl->display.display);
|
2013-05-19 11:04:59 +00:00
|
|
|
|
2013-05-15 16:17:18 +00:00
|
|
|
switch (request) {
|
|
|
|
case VOCTRL_CHECK_EVENTS:
|
|
|
|
*events |= vo_wayland_check_events(vo);
|
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_FULLSCREEN:
|
|
|
|
vo_wayland_fullscreen(vo);
|
|
|
|
*events |= VO_EVENT_RESIZE;
|
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_ONTOP:
|
|
|
|
vo_wayland_ontop(vo);
|
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_BORDER:
|
|
|
|
vo_wayland_border(vo);
|
|
|
|
*events |= VO_EVENT_RESIZE;
|
|
|
|
return VO_TRUE;
|
|
|
|
case VOCTRL_UPDATE_SCREENINFO:
|
|
|
|
vo_wayland_update_screeninfo(vo);
|
|
|
|
return VO_TRUE;
|
2013-11-12 20:02:49 +00:00
|
|
|
case VOCTRL_GET_WINDOW_SIZE: {
|
|
|
|
int *s = arg;
|
|
|
|
s[0] = wl->window.width;
|
|
|
|
s[1] = wl->window.height;
|
|
|
|
return VO_TRUE;
|
|
|
|
}
|
|
|
|
case VOCTRL_SET_WINDOW_SIZE: {
|
|
|
|
int *s = arg;
|
|
|
|
if (!wl->window.is_fullscreen)
|
|
|
|
shedule_resize(wl, 0, s[0], s[1]);
|
|
|
|
return VO_TRUE;
|
|
|
|
}
|
2013-05-19 11:04:59 +00:00
|
|
|
case VOCTRL_SET_CURSOR_VISIBILITY:
|
|
|
|
if (*(bool *)arg) {
|
2013-07-18 15:35:28 +00:00
|
|
|
if (!wl->cursor.visible)
|
2013-05-19 11:04:59 +00:00
|
|
|
show_cursor(wl);
|
|
|
|
}
|
|
|
|
else {
|
2013-07-18 15:35:28 +00:00
|
|
|
if (wl->cursor.visible)
|
2013-05-19 11:04:59 +00:00
|
|
|
hide_cursor(wl);
|
|
|
|
}
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->cursor.visible = *(bool *)arg;
|
2013-05-19 11:04:59 +00:00
|
|
|
return VO_TRUE;
|
2013-06-16 07:34:31 +00:00
|
|
|
case VOCTRL_UPDATE_WINDOW_TITLE:
|
2013-07-18 15:35:28 +00:00
|
|
|
wl_shell_surface_set_title(wl->window.shell_surface, (char *) arg);
|
2013-06-16 07:34:31 +00:00
|
|
|
return VO_TRUE;
|
2013-05-15 16:17:18 +00:00
|
|
|
}
|
|
|
|
return VO_NOTIMPL;
|
|
|
|
}
|
2013-05-16 14:43:34 +00:00
|
|
|
|
2013-05-19 11:04:59 +00:00
|
|
|
bool vo_wayland_config (struct vo *vo, uint32_t d_width,
|
|
|
|
uint32_t d_height, uint32_t flags)
|
2013-05-16 14:43:34 +00:00
|
|
|
{
|
2013-07-18 15:35:28 +00:00
|
|
|
struct vo_wayland_state *wl = vo->wayland;
|
2013-05-16 14:43:34 +00:00
|
|
|
|
2013-07-18 15:35:28 +00:00
|
|
|
wl->window.p_width = d_width;
|
|
|
|
wl->window.p_height = d_height;
|
2014-01-07 20:12:24 +00:00
|
|
|
wl->window.aspect = d_width / (float) MPMAX(d_height, 1);
|
2013-07-16 15:52:57 +00:00
|
|
|
|
2014-01-07 20:12:24 +00:00
|
|
|
if (!(flags & VOFLAG_HIDDEN)) {
|
|
|
|
if (!wl->window.is_init) {
|
|
|
|
wl->window.width = d_width;
|
|
|
|
wl->window.height = d_height;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vo->opts->fullscreen) {
|
|
|
|
if (wl->window.is_fullscreen)
|
|
|
|
shedule_resize(wl, 0, wl->window.fs_width, wl->window.fs_height);
|
|
|
|
else
|
|
|
|
vo_wayland_fullscreen(vo);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
vo_wayland_ontop(vo);
|
|
|
|
wl->window.is_init = true;
|
|
|
|
}
|
2013-05-16 14:43:34 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|