/* * Copyright (C) 2004 Lubos Lunak <l.lunak@kde.org> * * This program is free software; you can redistribute it and/or modify * 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. * * This program is distributed in the hope that it will be useful, * 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 with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "kdetrayproxy.h" #include <tdeapplication.h> #include <kdebug.h> #include <netwm.h> #include <X11/Xlib.h> #include <sys/select.h> #include <sys/time.h> #include <sys/types.h> #include <unistd.h> #include <assert.h> KDETrayProxy::KDETrayProxy() : selection( makeSelectionAtom()) { connect( &selection, TQT_SIGNAL( newOwner( Window )), TQT_SLOT( newOwner( Window ))); connect( &module, TQT_SIGNAL( windowAdded( WId )), TQT_SLOT( windowAdded( WId ))); selection.owner(); for( TQValueList< WId >::ConstIterator it = module.windows().begin(); it != module.windows().end(); ++it ) windowAdded( *it ); kapp->installX11EventFilter( this ); // XSelectInput( StructureNotifyMask ) on windows is done by KWinModule // kdDebug() << "Init done" << endl; } Atom KDETrayProxy::makeSelectionAtom() { return XInternAtom( tqt_xdisplay(), "_NET_SYSTEM_TRAY_S" + TQCString().setNum( tqt_xscreen()), False ); } void KDETrayProxy::windowAdded( WId w ) { NETWinInfo ni( tqt_xdisplay(), w, tqt_xrootwin(), NET::WMKDESystemTrayWinFor ); WId trayWinFor = ni.kdeSystemTrayWinFor(); if ( !trayWinFor ) // not a KDE tray window return; // kdDebug() << "New tray window:" << w << endl; if( !tray_windows.contains( w )) tray_windows.append( w ); withdrawWindow( w ); // window will be removed from pending_windows when after docked if( !pending_windows.contains( w )) pending_windows.append( w ); docked_windows.remove( w ); Window owner = selection.owner(); if( owner == None ) // no tray owner, sorry { // kdDebug() << "No owner, left in pending" << endl; return; } dockWindow( w, owner ); } void KDETrayProxy::newOwner( Window owner ) { // kdDebug() << "New owner:" << owner << endl; for( TQValueList< Window >::ConstIterator it = pending_windows.begin(); it != pending_windows.end(); ++it ) dockWindow( *it, owner ); // remove from pending_windows only in windowRemoved(), after it's really docked } bool KDETrayProxy::x11Event( XEvent* e ) { if( tray_windows.isEmpty()) return false; if( e->type == DestroyNotify && tray_windows.contains( e->xdestroywindow.window )) { tray_windows.remove( e->xdestroywindow.window ); pending_windows.remove( e->xdestroywindow.window ); docked_windows.remove( e->xdestroywindow.window ); } if( e->type == ReparentNotify && tray_windows.contains( e->xreparent.window )) { if( e->xreparent.parent == tqt_xrootwin()) { if( !docked_windows.contains( e->xreparent.window ) || e->xreparent.serial >= docked_windows[ e->xreparent.window ] ) { // kdDebug() << "Window released:" << e->xreparent.window << endl; docked_windows.remove( e->xreparent.window ); if( !pending_windows.contains( e->xreparent.window )) pending_windows.append( e->xreparent.window ); } } else { // kdDebug() << "Window away:" << e->xreparent.window << ":" << e->xreparent.parent << endl; pending_windows.remove( e->xreparent.window ); } } if( e->type == UnmapNotify && tray_windows.contains( e->xunmap.window )) { if( docked_windows.contains( e->xunmap.window ) && e->xunmap.serial >= docked_windows[ e->xunmap.window ] ) { // kdDebug() << "Window unmapped:" << e->xunmap.window << endl; XReparentWindow( tqt_xdisplay(), e->xunmap.window, tqt_xrootwin(), 0, 0 ); // ReparentNotify will take care of the rest } } return false; } void KDETrayProxy::dockWindow( Window w, Window owner ) { // kdDebug() << "Docking " << w << " into " << owner << endl; docked_windows[ w ] = XNextRequest( tqt_xdisplay()); static Atom prop = XInternAtom( tqt_xdisplay(), "_XEMBED_INFO", False ); long data[ 2 ] = { 0, 1 }; XChangeProperty( tqt_xdisplay(), w, prop, prop, 32, PropModeReplace, (unsigned char*)data, 2 ); XSizeHints hints; hints.flags = PMinSize | PMaxSize; hints.min_width = 24; hints.max_width = 24; hints.min_height = 24; hints.max_height = 24; XSetWMNormalHints( tqt_xdisplay(), w, &hints ); // kxerrorhandler ? XEvent ev; memset(&ev, 0, sizeof( ev )); static Atom atom = XInternAtom( tqt_xdisplay(), "_NET_SYSTEM_TRAY_OPCODE", False ); ev.xclient.type = ClientMessage; ev.xclient.window = owner; ev.xclient.message_type = atom; ev.xclient.format = 32; ev.xclient.data.l[ 0 ] = GET_QT_X_TIME(); ev.xclient.data.l[ 1 ] = 0; // SYSTEM_TRAY_REQUEST_DOCK ev.xclient.data.l[ 2 ] = w; ev.xclient.data.l[ 3 ] = 0; // unused ev.xclient.data.l[ 4 ] = 0; // unused XSendEvent( tqt_xdisplay(), owner, False, NoEventMask, &ev ); } void KDETrayProxy::withdrawWindow( Window w ) { XWithdrawWindow( tqt_xdisplay(), w, tqt_xscreen()); static Atom wm_state = XInternAtom( tqt_xdisplay(), "WM_STATE", False ); for(;;) { Atom type; int format; unsigned long length, after; unsigned char *data; int r = XGetWindowProperty( tqt_xdisplay(), w, wm_state, 0, 2, False, AnyPropertyType, &type, &format, &length, &after, &data ); bool withdrawn = true; if ( r == Success && data && format == 32 ) { withdrawn = ( *( long* )data == WithdrawnState ); XFree( (char *)data ); } if( withdrawn ) return; // ---> struct timeval tm; tm.tv_sec = 0; tm.tv_usec = 10 * 1000; // 10ms select(0, NULL, NULL, NULL, &tm); } } #include "kdetrayproxy.moc" #if 0 #include <tdecmdlineargs.h> int main( int argc, char* argv[] ) { TDECmdLineArgs::init( argc, argv, "a", "b", "c", "d" ); TDEApplication app( false ); // no styles app.disableSessionManagement(); KDETrayProxy proxy; return app.exec(); } #endif