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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * x11_window.cpp
  3.  *****************************************************************************
  4.  * Copyright (C) 2003 the VideoLAN team
  5.  * $Id: 3c205922f6d9501394f8ac25ad166379ca0c8e2f $
  6.  *
  7.  * Authors: Cyril Deguet     <asmax@via.ecp.fr>
  8.  *          Olivier Teulière <ipkiss@via.ecp.fr>
  9.  *
  10.  * This program is free software; you can redistribute it and/or modify
  11.  * it under the terms of the GNU General Public License as published by
  12.  * the Free Software Foundation; either version 2 of the License, or
  13.  * (at your option) any later version.
  14.  *
  15.  * This program is distributed in the hope that it will be useful,
  16.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  * GNU General Public License for more details.
  19.  *
  20.  * You should have received a copy of the GNU General Public License
  21.  * along with this program; if not, write to the Free Software
  22.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  23.  *****************************************************************************/
  24. #ifdef X11_SKINS
  25. #include <X11/Xatom.h>
  26. #include "../src/generic_window.hpp"
  27. #include "../src/vlcproc.hpp"
  28. #include "x11_window.hpp"
  29. #include "x11_display.hpp"
  30. #include "x11_graphics.hpp"
  31. #include "x11_dragdrop.hpp"
  32. #include "x11_factory.hpp"
  33. X11Window::X11Window( intf_thread_t *pIntf, GenericWindow &rWindow,
  34.                       X11Display &rDisplay, bool dragDrop, bool playOnDrop,
  35.                       X11Window *pParentWindow ):
  36.     OSWindow( pIntf ), m_rDisplay( rDisplay ), m_pParent( pParentWindow ),
  37.     m_dragDrop( dragDrop )
  38. {
  39.     if (pParentWindow)
  40.     {
  41.         m_wnd_parent = pParentWindow->m_wnd;
  42.     }
  43.     else
  44.     {
  45.         m_wnd_parent = DefaultRootWindow( XDISPLAY );
  46.     }
  47.     XSetWindowAttributes attr;
  48.     attr.event_mask = ExposureMask | StructureNotifyMask;
  49.     // Create the window
  50.     m_wnd = XCreateWindow( XDISPLAY, m_wnd_parent, -10, 0, 1, 1, 0, 0,
  51.                            InputOutput, CopyFromParent, CWEventMask, &attr );
  52.     // Make sure window is created before returning
  53.     XMapWindow( XDISPLAY, m_wnd );
  54.     bool b_map_notify = false;
  55.     do
  56.     {
  57.         XEvent xevent;
  58.         XWindowEvent( XDISPLAY, m_wnd, SubstructureNotifyMask |
  59.                       StructureNotifyMask, &xevent);
  60.         if( (xevent.type == MapNotify)
  61.              && (xevent.xmap.window == m_wnd ) )
  62.         {
  63.             b_map_notify = true;
  64.         }
  65.     } while( ! b_map_notify );
  66.     XUnmapWindow( XDISPLAY, m_wnd );
  67.     // Set the colormap for 8bpp mode
  68.     if( XPIXELSIZE == 1 )
  69.     {
  70.         XSetWindowColormap( XDISPLAY, m_wnd, m_rDisplay.getColormap() );
  71.     }
  72.     // Select events received by the window
  73.     XSelectInput( XDISPLAY, m_wnd, ExposureMask|KeyPressMask|
  74.                   PointerMotionMask|ButtonPressMask|ButtonReleaseMask|
  75.                   LeaveWindowMask|FocusChangeMask );
  76.     // Store a pointer on the generic window in a map
  77.     X11Factory *pFactory = (X11Factory*)X11Factory::instance( getIntf() );
  78.     pFactory->m_windowMap[m_wnd] = &rWindow;
  79.     // Changing decorations
  80.     struct {
  81.         unsigned long flags;
  82.         unsigned long functions;
  83.         unsigned long decorations;
  84.         signed   long input_mode;
  85.         unsigned long status;
  86.     } motifWmHints;
  87.     Atom hints_atom = XInternAtom( XDISPLAY, "_MOTIF_WM_HINTS", False );
  88.     motifWmHints.flags = 2;    // MWM_HINTS_DECORATIONS;
  89.     motifWmHints.decorations = 0;
  90.     XChangeProperty( XDISPLAY, m_wnd, hints_atom, hints_atom, 32,
  91.                      PropModeReplace, (unsigned char *)&motifWmHints,
  92.                      sizeof( motifWmHints ) / sizeof( uint32_t ) );
  93.     // Drag & drop
  94.     if( m_dragDrop )
  95.     {
  96.         // Create a Dnd object for this window
  97.         m_pDropTarget = new X11DragDrop( getIntf(), m_rDisplay, m_wnd,
  98.                                          playOnDrop );
  99.         // Register the window as a drop target
  100.         Atom xdndAtom = XInternAtom( XDISPLAY, "XdndAware", False );
  101.         char xdndVersion = 4;
  102.         XChangeProperty( XDISPLAY, m_wnd, xdndAtom, XA_ATOM, 32,
  103.                          PropModeReplace, (unsigned char *)&xdndVersion, 1 );
  104.         // Store a pointer to be used in X11Loop
  105.         pFactory->m_dndMap[m_wnd] = m_pDropTarget;
  106.     }
  107.     // Change the window title
  108.     XStoreName( XDISPLAY, m_wnd, "VLC" );
  109.     // Associate the window to the main "parent" window
  110.     XSetTransientForHint( XDISPLAY, m_wnd, m_rDisplay.getMainWindow() );
  111. }
  112. X11Window::~X11Window()
  113. {
  114.     X11Factory *pFactory = (X11Factory*)X11Factory::instance( getIntf() );
  115.     pFactory->m_windowMap[m_wnd] = NULL;
  116.     pFactory->m_dndMap[m_wnd] = NULL;
  117.     if( m_dragDrop )
  118.     {
  119.         delete m_pDropTarget;
  120.     }
  121.     XDestroyWindow( XDISPLAY, m_wnd );
  122.     XSync( XDISPLAY, False );
  123. }
  124. void X11Window::reparent( void* OSHandle, int x, int y, int w, int h )
  125. {
  126.     // Reparent the window
  127.     Window new_parent =
  128.            OSHandle ? (Window) OSHandle : DefaultRootWindow( XDISPLAY );
  129.     if( w && h )
  130.         XResizeWindow( XDISPLAY, m_wnd, w, h );
  131.     XReparentWindow( XDISPLAY, m_wnd, new_parent, x, y);
  132.     m_wnd_parent = new_parent;
  133. }
  134. void X11Window::show( int left, int top ) const
  135. {
  136.     // Map the window
  137.     XMapRaised( XDISPLAY, m_wnd );
  138.     XMoveWindow( XDISPLAY, m_wnd, left, top );
  139. }
  140. void X11Window::hide() const
  141. {
  142.     // Unmap the window
  143.     XUnmapWindow( XDISPLAY, m_wnd );
  144. }
  145. void X11Window::moveResize( int left, int top, int width, int height ) const
  146. {
  147.     if( width && height )
  148.         XMoveResizeWindow( XDISPLAY, m_wnd, left, top, width, height );
  149.     else
  150.         XMoveWindow( XDISPLAY, m_wnd, left, top );
  151. }
  152. void X11Window::raise() const
  153. {
  154.     XRaiseWindow( XDISPLAY, m_wnd );
  155. }
  156. void X11Window::setOpacity( uint8_t value ) const
  157. {
  158.     // Sorry, the opacity cannot be changed :)
  159. }
  160. void X11Window::toggleOnTop( bool onTop ) const
  161. {
  162.     int i_ret, i_format;
  163.     unsigned long i, i_items, i_bytesafter;
  164.     Atom net_wm_supported, net_wm_state, net_wm_state_on_top,net_wm_state_above;
  165.     union { Atom *p_atom; unsigned char *p_char; } p_args;
  166.     p_args.p_atom = NULL;
  167.     net_wm_supported = XInternAtom( XDISPLAY, "_NET_SUPPORTED", False );
  168.     i_ret = XGetWindowProperty( XDISPLAY, DefaultRootWindow( XDISPLAY ),
  169.                                 net_wm_supported,
  170.                                 0, 16384, False, AnyPropertyType,
  171.                                 &net_wm_supported,
  172.                                 &i_format, &i_items, &i_bytesafter,
  173.                                 (unsigned char **)&p_args );
  174.     if( i_ret != Success || i_items == 0 ) return; /* Not supported */
  175.     net_wm_state = XInternAtom( XDISPLAY, "_NET_WM_STATE", False );
  176.     net_wm_state_on_top = XInternAtom( XDISPLAY, "_NET_WM_STATE_STAYS_ON_TOP",
  177.                                        False );
  178.     for( i = 0; i < i_items; i++ )
  179.     {
  180.         if( p_args.p_atom[i] == net_wm_state_on_top ) break;
  181.     }
  182.     if( i == i_items )
  183.     { /* use _NET_WM_STATE_ABOVE if window manager
  184.        * doesn't handle _NET_WM_STATE_STAYS_ON_TOP */
  185.         net_wm_state_above = XInternAtom( XDISPLAY, "_NET_WM_STATE_ABOVE",
  186.                                           False);
  187.         for( i = 0; i < i_items; i++ )
  188.         {
  189.             if( p_args.p_atom[i] == net_wm_state_above ) break;
  190.         }
  191.  
  192.         XFree( p_args.p_atom );
  193.         if( i == i_items )
  194.             return; /* Not supported */
  195.         /* Switch "on top" status */
  196.         XClientMessageEvent event;
  197.         memset( &event, 0, sizeof( XClientMessageEvent ) );
  198.         event.type = ClientMessage;
  199.         event.message_type = net_wm_state;
  200.         event.display = XDISPLAY;
  201.         event.window = m_wnd;
  202.         event.format = 32;
  203.         event.data.l[ 0 ] = onTop; /* set property */
  204.         event.data.l[ 1 ] = net_wm_state_above;
  205.         XSendEvent( XDISPLAY, DefaultRootWindow( XDISPLAY ),
  206.                     False, SubstructureRedirectMask, (XEvent*)&event );
  207.         return;
  208.     }
  209.     XFree( p_args.p_atom );
  210.     /* Switch "on top" status */
  211.     XClientMessageEvent event;
  212.     memset( &event, 0, sizeof( XClientMessageEvent ) );
  213.     event.type = ClientMessage;
  214.     event.message_type = net_wm_state;
  215.     event.display = XDISPLAY;
  216.     event.window = m_wnd;
  217.     event.format = 32;
  218.     event.data.l[ 0 ] = onTop; /* set property */
  219.     event.data.l[ 1 ] = net_wm_state_on_top;
  220.     XSendEvent( XDISPLAY, DefaultRootWindow( XDISPLAY ),
  221.                 False, SubstructureRedirectMask, (XEvent*)&event );
  222. }
  223. #endif