window.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:11k
源码类别:

midi

开发平台:

Unix_Linux

  1. /**
  2.  * @file window.c
  3.  * @brief X C Bindings window provider module for VLC media player
  4.  */
  5. /*****************************************************************************
  6.  * Copyright © 2009 Rémi Denis-Courmont
  7.  *
  8.  * This library is free software; you can redistribute it and/or
  9.  * modify it under the terms of the GNU General Public License
  10.  * as published by the Free Software Foundation; either version 2.0
  11.  * of the License, or (at your option) any later version.
  12.  *
  13.  * This library is distributed in the hope that it will be useful,
  14.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  16.  * GNU General Public License for more details.
  17.  *
  18.  * You should have received a copy of the GNU Lesser General Public
  19.  * License along with this library; if not, write to the Free Software
  20.  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
  21.  ****************************************************************************/
  22. #ifdef HAVE_CONFIG_H
  23. # include <config.h>
  24. #endif
  25. #include <stdarg.h>
  26. #include <assert.h>
  27. #include <poll.h>
  28. #include <unistd.h> /* gethostname() */
  29. #include <limits.h> /* HOST_NAME_MAX */
  30. #include <xcb/xcb.h>
  31. typedef xcb_atom_t Atom;
  32. #include <X11/Xatom.h> /* XA_WM_NAME */
  33. #include <vlc_common.h>
  34. #include <vlc_plugin.h>
  35. #include <vlc_window.h>
  36. #include "xcb_vlc.h"
  37. #define DISPLAY_TEXT N_("X11 display")
  38. #define DISPLAY_LONGTEXT N_( 
  39.     "X11 hardware display to use. By default VLC will " 
  40.     "use the value of the DISPLAY environment variable.")
  41. static int  Open (vlc_object_t *);
  42. static void Close (vlc_object_t *);
  43. /*
  44.  * Module descriptor
  45.  */
  46. vlc_module_begin ()
  47.     set_shortname (N_("XCB window"))
  48.     set_description (N_("(Experimental) XCB video window"))
  49.     set_category (CAT_VIDEO)
  50.     set_subcategory (SUBCAT_VIDEO_VOUT)
  51.     set_capability ("xwindow", 10)
  52.     set_callbacks (Open, Close)
  53.     add_string ("x11-display", NULL, NULL,
  54.                 DISPLAY_TEXT, DISPLAY_LONGTEXT, true)
  55. vlc_module_end ()
  56. static int Control (vout_window_t *, int, va_list ap);
  57. static void *Thread (void *);
  58. struct vout_window_sys_t
  59. {
  60.     xcb_connection_t *conn;
  61.     key_handler_t *keys;
  62.     vlc_thread_t thread;
  63.     xcb_window_t root;
  64.     xcb_atom_t wm_state;
  65.     xcb_atom_t wm_state_above;
  66.     /*xcb_atom_t wmstate_fullscreen;*/
  67. };
  68. static inline
  69. void set_string (xcb_connection_t *conn, xcb_window_t window,
  70.                  xcb_atom_t type, xcb_atom_t atom, const char *str)
  71. {
  72.     xcb_change_property (conn, XCB_PROP_MODE_REPLACE, window, atom, type,
  73.                          /* format */ 8, strlen (str), str);
  74. }
  75. static inline
  76. void set_ascii_prop (xcb_connection_t *conn, xcb_window_t window,
  77.                      xcb_atom_t atom, const char *value)
  78. {
  79.     set_string (conn, window, atom, XA_STRING, value);
  80. }
  81. static inline
  82. void set_hostname_prop (xcb_connection_t *conn, xcb_window_t window)
  83. {
  84.     char hostname[HOST_NAME_MAX];
  85.     if (gethostname (hostname, sizeof (hostname)) == 0)
  86.     {
  87.         hostname[sizeof (hostname) - 1] = '';
  88.         set_ascii_prop (conn, window, XA_WM_CLIENT_MACHINE, hostname);
  89.     }
  90. }
  91. static inline
  92. xcb_intern_atom_cookie_t intern_string (xcb_connection_t *c, const char *s)
  93. {
  94.     return xcb_intern_atom (c, 0, strlen (s), s);
  95. }
  96. static
  97. xcb_atom_t get_atom (xcb_connection_t *conn, xcb_intern_atom_cookie_t ck)
  98. {
  99.     xcb_intern_atom_reply_t *reply;
  100.     xcb_atom_t atom;
  101.     reply = xcb_intern_atom_reply (conn, ck, NULL);
  102.     if (reply == NULL)
  103.         return 0;
  104.     atom = reply->atom;
  105.     free (reply);
  106.     return atom;
  107. }
  108. #define NET_WM_STATE_REMOVE 0
  109. #define NET_WM_STATE_ADD    1
  110. #define NET_WM_STATE_TOGGLE 2
  111. /**
  112.  * Create an X11 window.
  113.  */
  114. static int Open (vlc_object_t *obj)
  115. {
  116.     vout_window_t *wnd = (vout_window_t *)obj;
  117.     vout_window_sys_t *p_sys = malloc (sizeof (*p_sys));
  118.     xcb_generic_error_t *err;
  119.     xcb_void_cookie_t ck;
  120.     if (p_sys == NULL)
  121.         return VLC_ENOMEM;
  122.     /* Connect to X */
  123.     char *display = var_CreateGetNonEmptyString (wnd, "x11-display");
  124.     int snum;
  125.     xcb_connection_t *conn = xcb_connect (display, &snum);
  126.     free (display);
  127.     if (xcb_connection_has_error (conn) /*== NULL*/)
  128.         goto error;
  129.     /* Find configured screen */
  130.     const xcb_setup_t *setup = xcb_get_setup (conn);
  131.     xcb_screen_t *scr = NULL;
  132.     for (xcb_screen_iterator_t i = xcb_setup_roots_iterator (setup);
  133.          i.rem > 0; xcb_screen_next (&i))
  134.     {
  135.         if (snum == 0)
  136.         {
  137.             scr = i.data;
  138.             break;
  139.         }
  140.         snum--;
  141.     }
  142.     if (scr == NULL)
  143.     {
  144.         msg_Err (wnd, "bad X11 screen number");
  145.         goto error;
  146.     }
  147.     /* Create window */
  148.     const uint32_t mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK;
  149.     uint32_t values[2] = {
  150.         /* XCB_CW_BACK_PIXEL */
  151.         scr->black_pixel,
  152.         /* XCB_CW_EVENT_MASK */
  153.         XCB_EVENT_MASK_KEY_PRESS,
  154.     };
  155.     xcb_window_t window = xcb_generate_id (conn);
  156.     ck = xcb_create_window_checked (conn, scr->root_depth, window, scr->root,
  157.                                     0, 0, wnd->width, wnd->height, 0,
  158.                                     XCB_WINDOW_CLASS_INPUT_OUTPUT,
  159.                                     scr->root_visual, mask, values);
  160.     err = xcb_request_check (conn, ck);
  161.     if (err)
  162.     {
  163.         msg_Err (wnd, "creating window: X11 error %d", err->error_code);
  164.         goto error;
  165.     }
  166.     wnd->handle.xid = window;
  167.     wnd->p_sys = p_sys;
  168.     wnd->control = Control;
  169.     p_sys->conn = conn;
  170.     p_sys->keys = CreateKeyHandler (obj, conn);
  171.     p_sys->root = scr->root;
  172.     /* ICCCM
  173.      * No cut&paste nor drag&drop, only Window Manager communication. */
  174.     /* Plain ASCII localization of VLC for ICCCM window name */
  175.     set_ascii_prop (conn, window, XA_WM_NAME,
  176.                   vlc_pgettext ("ASCII", "VLC media player"));
  177.     set_ascii_prop (conn, window, XA_WM_ICON_NAME,
  178.                     vlc_pgettext ("ASCII", "VLC"));
  179.     xcb_change_property (conn, XCB_PROP_MODE_REPLACE, window, XA_WM_CLASS,
  180.                          XA_STRING, 8, 8, "vlcVlc");
  181.     set_hostname_prop (conn, window);
  182.     /* EWMH */
  183.     xcb_intern_atom_cookie_t utf8_string_ck
  184.         = intern_string (conn, "UTF8_STRING");;
  185.     xcb_intern_atom_cookie_t net_wm_name_ck
  186.         = intern_string (conn, "_NET_WM_NAME");
  187.     xcb_intern_atom_cookie_t net_wm_icon_name_ck
  188.         = intern_string (conn, "_NET_WM_ICON_NAME");
  189.     xcb_atom_t utf8 = get_atom (conn, utf8_string_ck);
  190.     xcb_atom_t net_wm_name = get_atom (conn, net_wm_name_ck);
  191.     char *title = var_CreateGetNonEmptyString (wnd, "video-title");
  192.     if (title)
  193.     {
  194.         set_string (conn, window, utf8, net_wm_name, title);
  195.         free (title);
  196.     }
  197.     else
  198.         set_string (conn, window, utf8, net_wm_name, _("VLC media player"));
  199.     xcb_atom_t net_wm_icon_name = get_atom (conn, net_wm_icon_name_ck);
  200.     set_string (conn, window, utf8, net_wm_icon_name, _("VLC"));
  201.     /* Cache any EWMH atom we may need later */
  202.     xcb_intern_atom_cookie_t wm_state_ck, wm_state_above_ck;
  203.     wm_state_ck = intern_string (conn, "_NET_WM_STATE");
  204.     wm_state_above_ck = intern_string (conn, "_NET_WM_STATE_ABOVE");
  205.     p_sys->wm_state = get_atom (conn, wm_state_ck);
  206.     p_sys->wm_state_above = get_atom (conn, wm_state_above_ck);
  207.     /* Create the event thread. It will dequeue all events, so any checked
  208.      * request from this thread must be completed at this point. */
  209.     if ((p_sys->keys != NULL)
  210.      && vlc_clone (&p_sys->thread, Thread, wnd, VLC_THREAD_PRIORITY_LOW))
  211.         DestroyKeyHandler (p_sys->keys);
  212.     /* Make sure the window is ready */
  213.     xcb_map_window (conn, window);
  214.     xcb_flush (conn);
  215.     return VLC_SUCCESS;
  216. error:
  217.     xcb_disconnect (conn);
  218.     free (p_sys);
  219.     return VLC_EGENERIC;
  220. }
  221. /**
  222.  * Destroys the X11 window.
  223.  */
  224. static void Close (vlc_object_t *obj)
  225. {
  226.     vout_window_t *wnd = (vout_window_t *)obj;
  227.     vout_window_sys_t *p_sys = wnd->p_sys;
  228.     xcb_connection_t *conn = p_sys->conn;
  229.     xcb_window_t window = wnd->handle.xid;
  230.     if (p_sys->keys)
  231.     {
  232.         vlc_cancel (p_sys->thread);
  233.         vlc_join (p_sys->thread, NULL);
  234.         DestroyKeyHandler (p_sys->keys);
  235.     }
  236.     xcb_unmap_window (conn, window);
  237.     xcb_destroy_window (conn, window);
  238.     xcb_disconnect (conn);
  239.     free (p_sys);
  240. }
  241. static void *Thread (void *data)
  242. {
  243.     vout_window_t *wnd = data;
  244.     vout_window_sys_t *p_sys = wnd->p_sys;
  245.     xcb_connection_t *conn = p_sys->conn;
  246.     int fd = xcb_get_file_descriptor (conn);
  247.     if (fd == -1)
  248.         return NULL;
  249.     for (;;)
  250.     {
  251.         xcb_generic_event_t *ev;
  252.         struct pollfd ufd = { .fd = fd, .events = POLLIN, };
  253.         poll (&ufd, 1, -1);
  254.         int canc = vlc_savecancel ();
  255.         while ((ev = xcb_poll_for_event (conn)) != NULL)
  256.         {
  257.             if (ProcessKeyEvent (p_sys->keys, ev) == 0)
  258.                 continue;
  259.             msg_Dbg (wnd, "unhandled event: %"PRIu8, ev->response_type);
  260.             free (ev);
  261.         }
  262.         vlc_restorecancel (canc);
  263.         if (xcb_connection_has_error (conn))
  264.         {
  265.             msg_Err (wnd, "X server failure");
  266.             break;
  267.         }
  268.     }
  269.     return NULL;
  270. }
  271. #include <vlc_vout.h>
  272. static int Control (vout_window_t *wnd, int cmd, va_list ap)
  273. {
  274.     vout_window_sys_t *p_sys = wnd->p_sys;
  275.     xcb_connection_t *conn = p_sys->conn;
  276.     switch (cmd)
  277.     {
  278.         case VOUT_SET_SIZE:
  279.         {
  280.             unsigned width = va_arg (ap, unsigned);
  281.             unsigned height = va_arg (ap, unsigned);
  282.             const uint32_t values[] = { width, height, };
  283.             xcb_configure_window (conn, wnd->handle.xid,
  284.                                   XCB_CONFIG_WINDOW_WIDTH |
  285.                                   XCB_CONFIG_WINDOW_HEIGHT, values);
  286.             xcb_flush (conn);
  287.             break;
  288.         }
  289.         case VOUT_SET_STAY_ON_TOP:
  290.         {   /* From EWMH "_WM_STATE" */
  291.             xcb_client_message_event_t ev = {
  292.                 .response_type = XCB_CLIENT_MESSAGE,
  293.                 .format = 32,
  294.                 .window = wnd->handle.xid,
  295.                 .type = p_sys->wm_state,
  296.             };
  297.             bool on = va_arg (ap, int);
  298.             ev.data.data32[0] = on ? NET_WM_STATE_ADD : NET_WM_STATE_REMOVE;
  299.             ev.data.data32[1] = p_sys->wm_state_above;
  300.             ev.data.data32[2] = 0;
  301.             ev.data.data32[3] = 1;
  302.             /* From ICCCM "Changing Window State" */
  303.             xcb_send_event (p_sys->conn, 0, p_sys->root,
  304.                             XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY |
  305.                             XCB_EVENT_MASK_SUBSTRUCTURE_REDIRECT,
  306.                             (const char *)&ev);
  307.             xcb_flush (p_sys->conn);
  308.             return VLC_SUCCESS;
  309.         }
  310.         default:
  311.             msg_Err (wnd, "request %d not implemented", cmd);
  312.             return VLC_EGENERIC;
  313.     }
  314.     return VLC_SUCCESS;
  315. }