/***************************************************************** kwin - the KDE window manager Copyright (C) 1999, 2000 Matthias Ettrich ******************************************************************/ #include #include "main.h" #include "options.h" #include "atoms.h" #include "workspace.h" #include #include #include #include #include #include #include #include #include #include #define INT8 _X11INT8 #define INT32 _X11INT32 #include #undef INT8 #undef INT32 #include #include #include Options* options; Atoms* atoms; Time kwin_time = CurrentTime; static bool initting = FALSE; int x11ErrorHandler(Display *d, XErrorEvent *e){ char msg[80], req[80], number[80]; bool ignore_badwindow = TRUE; //maybe temporary if (initting && ( e->request_code == X_ChangeWindowAttributes || e->request_code == X_GrabKey ) && (e->error_code == BadAccess)) { fprintf(stderr, i18n("kwin: it looks like there's already a window manager running. kwin not started\n")); exit(1); } if (ignore_badwindow && (e->error_code == BadWindow || e->error_code == BadColor)) return 0; XGetErrorText(d, e->error_code, msg, sizeof(msg)); sprintf(number, "%d", e->request_code); XGetErrorDatabaseText(d, "XRequest", number, "", req, sizeof(req)); fprintf(stderr, "kwin: %s(0x%lx): %s\n", req, e->resourceid, msg); if (initting) { fprintf(stderr, i18n("kwin: failure during initialisation; aborting\n")); exit(1); } return 0; } Application::Application( ) : KApplication( ) { initting = TRUE; // startup.... // install X11 error handler XSetErrorHandler( x11ErrorHandler ); // check whether another windowmanager is running XSelectInput(qt_xdisplay(), qt_xrootwin(), SubstructureRedirectMask ); syncX(); // trigger error now options = new Options; atoms = new Atoms; // create a workspace. workspaces += new Workspace( isSessionRestored() ); syncX(); // trigger possible errors, there's still a chance to abort initting = FALSE; // startup done, we are up and running now. } Application::~Application() { for ( WorkspaceList::Iterator it = workspaces.begin(); it != workspaces.end(); ++it) { delete (*it); } delete options; } bool Application::x11EventFilter( XEvent *e ) { switch ( e->type ) { case ButtonPress: case ButtonRelease: case MotionNotify: kwin_time = (e->type == MotionNotify) ? e->xmotion.time : e->xbutton.time; break; case KeyPress: case KeyRelease: kwin_time = e->xkey.time; break; case PropertyNotify: kwin_time = e->xproperty.time; break; case ConfigureNotify: { if ( e->xconfigure.window != e->xconfigure.event ) return TRUE; } break; default: break; } for ( WorkspaceList::Iterator it = workspaces.begin(); it != workspaces.end(); ++it) { if ( (*it)->workspaceEvent( e ) ) return TRUE; } return KApplication::x11EventFilter( e ); } void Application::commitData( QSessionManager& /*sm*/ ) { // nothing to do, really } void Application::saveState( QSessionManager& sm ) { KApplication::saveState( sm ); static bool firstTime = false; if ( firstTime ) { firstTime = false; return; // no need to save this state. } sm.release(); // we really should do phase 2 here, unfortunately qt-2.1beta3 contains a bug. // #######TODO FIXME with final Qt-2.1 /* if ( !sm.isPhase2() ) { sm.requestPhase2(); return; } */ workspaces.first()->storeSession( kapp->sessionConfig() ); kapp->sessionConfig()->sync(); } static void sighandler(int) { QApplication::exit(); } static const char *version = "0.4"; static const char *description = I18N_NOOP( "The KDE window manager." ); int main( int argc, char * argv[] ) { KAboutData aboutData( "kwin", I18N_NOOP("KWin"), version, description, KAboutData::License_BSD, "(c) 1999-2000, The KDE Developers"); aboutData.addAuthor("Matthias Ettrich",0, "ettrich@kde.org"); aboutData.addAuthor("Daniel M. Duley",0, "mosfet@kde.org"); KCmdLineArgs::init(argc, argv, &aboutData); if (signal(SIGTERM, sighandler) == SIG_IGN) signal(SIGTERM, SIG_IGN); if (signal(SIGINT, sighandler) == SIG_IGN) signal(SIGINT, SIG_IGN); if (signal(SIGHUP, sighandler) == SIG_IGN) signal(SIGHUP, SIG_IGN); Application a; fcntl(ConnectionNumber(qt_xdisplay()), F_SETFD, 1); DCOPClient * client = a.dcopClient(); client->attach(); client->registerAs(a.name()); return a.exec(); }