/***************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2004 Lubos Lunak You can Freely distribute this program under the GNU General Public License. See the file "COPYING" for the exact licensing terms. ******************************************************************/ #include "rules.h" #include #include #include #include #include #include "client.h" #include "workspace.h" namespace KWinInternal { static WindowRules dummyRules; // dummy used to avoid NULL checks WindowRules::WindowRules() : temporary_state( 0 ) , wmclassregexp( false ) , wmclasscomplete( false ) , windowroleregexp( false ) , titleregexp( false ) , extraroleregexp( false ) , clientmachineregexp( false ) , types( NET::AllTypesMask ) , placementrule( DontCareRule ) , positionrule( DontCareRule ) , sizerule( DontCareRule ) , minsizerule( DontCareRule ) , maxsizerule( DontCareRule ) , desktoprule( DontCareRule ) , typerule( DontCareRule ) , maximizevertrule( DontCareRule ) , maximizehorizrule( DontCareRule ) , minimizerule( DontCareRule ) , shaderule( DontCareRule ) , skiptaskbarrule( DontCareRule ) , skippagerrule( DontCareRule ) , aboverule( DontCareRule ) , belowrule( DontCareRule ) , fullscreenrule( DontCareRule ) , noborderrule( DontCareRule ) , fsplevelrule( DontCareRule ) , acceptfocusrule( DontCareRule ) , moveresizemoderule( DontCareRule ) , closeablerule( DontCareRule ) { } WindowRules::WindowRules( const QString& str, bool temporary ) : temporary_state( temporary ? 2 : 0 ) { KTempFile file; QFile* f = file.file(); if( f != NULL ) { QCString s = str.utf8(); f->writeBlock( s.data(), s.length()); } file.close(); KSimpleConfig cfg( file.name()); readFromCfg( cfg ); file.unlink(); } #define READ_MATCH_STRING( var, func ) \ var = cfg.readEntry( #var ) func; \ var##regexp = cfg.readBoolEntry( #var "regexp" ); #define READ_SET_RULE( var, type, func ) \ var = func ( cfg.read##type##Entry( #var )); \ var##rule = readRule( cfg, #var "rule" ); #define READ_SET_RULE_DEF( var, type, func, def ) \ var = func ( cfg.read##type##Entry( #var, def )); \ var##rule = readRule( cfg, #var "rule" ); #define READ_SET_RULE_2( var, type, func, funcarg ) \ var = func ( cfg.read##type##Entry( #var ), funcarg ); \ var##rule = readRule( cfg, #var "rule" ); #define READ_FORCE_RULE( var, type, func ) \ var = func ( cfg.read##type##Entry( #var )); \ var##rule = readForceRule( cfg, #var "rule" ); WindowRules::WindowRules( KConfig& cfg ) : temporary_state( 0 ) { readFromCfg( cfg ); } static int limit0to4( int i ) { return QMAX( 0, QMIN( 4, i )); } void WindowRules::readFromCfg( KConfig& cfg ) { wmclass = cfg.readEntry( "wmclass" ).lower().latin1(); wmclassregexp = cfg.readBoolEntry( "wmclassregexp" ); wmclasscomplete = cfg.readBoolEntry( "wmclasscomplete" ); READ_MATCH_STRING( windowrole, .lower().latin1() ); READ_MATCH_STRING( title, ); READ_MATCH_STRING( extrarole, .lower().latin1() ); READ_MATCH_STRING( clientmachine, .lower().latin1() ); types = cfg.readUnsignedLongNumEntry( "types", NET::AllTypesMask ); READ_SET_RULE_2( placement,, Placement::policyFromString, false ); READ_SET_RULE_DEF( position, Point,, &invalidPoint ); READ_SET_RULE( size, Size, ); if( size.isEmpty() && sizerule != RememberRule ) sizerule = DontCareRule; READ_SET_RULE( minsize, Size, ); if( !minsize.isValid()) minsizerule = DontCareRule; READ_SET_RULE( maxsize, Size, ); if( maxsize.isEmpty()) maxsizerule = DontCareRule; READ_SET_RULE( desktop, Num, ); type = readType( cfg, "type" ); typerule = type != NET::Unknown ? readForceRule( cfg, "typerule" ) : DontCareRule; READ_SET_RULE( maximizevert, Bool, ); READ_SET_RULE( maximizehoriz, Bool, ); READ_SET_RULE( minimize, Bool, ); READ_SET_RULE( shade, Bool, ); READ_SET_RULE( skiptaskbar, Bool, ); READ_SET_RULE( skippager, Bool, ); READ_SET_RULE( above, Bool, ); READ_SET_RULE( below, Bool, ); READ_SET_RULE( fullscreen, Bool, ); READ_SET_RULE( noborder, Bool, ); READ_FORCE_RULE( fsplevel, Num, limit0to4 ); // fsp is 0-4 READ_FORCE_RULE( acceptfocus, Bool, ); READ_FORCE_RULE( moveresizemode, , Options::stringToMoveResizeMode ); READ_FORCE_RULE( closeable, Bool, ); kdDebug() << "READ RULE:" << wmclass << "/" << title << endl; } #undef READ_MATCH_STRING #undef READ_SET_RULE #undef READ_SET_RULE_2 #undef READ_FORCE_RULE #define WRITE_MATCH_STRING( var, cast ) \ if( !var.isEmpty()) \ { \ cfg.writeEntry( #var, cast var ); \ cfg.writeEntry( #var "regexp", var##regexp ); \ } \ else \ { \ cfg.deleteEntry( #var ); \ cfg.deleteEntry( #var "regexp" ); \ } #define WRITE_SET_RULE( var, func ) \ if( var##rule != DontCareRule ) \ { \ cfg.writeEntry( #var, func ( var )); \ cfg.writeEntry( #var "rule", var##rule ); \ } \ else \ { \ cfg.deleteEntry( #var ); \ cfg.deleteEntry( #var "rule" ); \ } #define WRITE_WITH_DEFAULT( var, default ) \ if( var != default ) \ cfg.writeEntry( #var, var ); \ else \ cfg.deleteEntry( #var ); void WindowRules::write( KConfig& cfg ) const { // always write wmclass cfg.writeEntry( "wmclass", ( const char* )wmclass ); cfg.writeEntry( "wmclassregexp", wmclassregexp ); cfg.writeEntry( "wmclasscomplete", wmclasscomplete ); WRITE_MATCH_STRING( windowrole, (const char*) ); WRITE_MATCH_STRING( title, ); WRITE_MATCH_STRING( extrarole, (const char*) ); WRITE_MATCH_STRING( clientmachine, (const char*) ); WRITE_WITH_DEFAULT( types, NET::AllTypesMask ); WRITE_SET_RULE( placement, Placement::policyToString ); WRITE_SET_RULE( position, ); WRITE_SET_RULE( size, ); WRITE_SET_RULE( minsize, ); WRITE_SET_RULE( maxsize, ); WRITE_SET_RULE( desktop, ); WRITE_SET_RULE( type, ); WRITE_SET_RULE( maximizevert, ); WRITE_SET_RULE( maximizehoriz, ); WRITE_SET_RULE( minimize, ); WRITE_SET_RULE( shade, ); WRITE_SET_RULE( skiptaskbar, ); WRITE_SET_RULE( skippager, ); WRITE_SET_RULE( above, ); WRITE_SET_RULE( below, ); WRITE_SET_RULE( fullscreen, ); WRITE_SET_RULE( noborder, ); WRITE_SET_RULE( fsplevel, ); WRITE_SET_RULE( acceptfocus, ); WRITE_SET_RULE( moveresizemode, Options::moveResizeModeToString ); WRITE_SET_RULE( closeable, ); } #undef WRITE_MATCH_STRING #undef WRITE_SET_RULE #undef WRITE_WITH_DEFAULT SettingRule WindowRules::readRule( KConfig& cfg, const QString& key ) { int v = cfg.readNumEntry( key ); if( v >= DontCareRule && v <= LastRule ) return static_cast< SettingRule >( v ); return DontCareRule; } SettingRule WindowRules::readForceRule( KConfig& cfg, const QString& key ) { int v = cfg.readNumEntry( key ); if( v == DontCareRule || v == ForceRule ) return static_cast< SettingRule >( v ); return DontCareRule; } NET::WindowType WindowRules::readType( KConfig& cfg, const QString& key ) { int v = cfg.readNumEntry( key ); if( v >= NET::Normal && v <= NET::Splash ) return static_cast< NET::WindowType >( v ); return NET::Unknown; } bool WindowRules::match( const Client* c ) const { if( types != NET::AllTypesMask ) { NET::WindowType t = c->windowType( true ); // direct type if( t == NET::Unknown ) t = NET::Normal; // NET::Unknown->NET::Normal is only here for matching if( !NET::typeMatchesMask( t, types )) return false; } if( !wmclass.isEmpty()) { // TODO optimize? QCString cwmclass = wmclasscomplete ? c->resourceName() + ' ' + c->resourceClass() : c->resourceClass(); if( wmclassregexp && QRegExp( wmclass ).search( cwmclass ) == -1 ) return false; if( !wmclassregexp && wmclass != cwmclass ) return false; } if( !windowrole.isEmpty()) { if( windowroleregexp && QRegExp( windowrole ).search( c->windowRole()) == -1 ) return false; if( !windowroleregexp && windowrole != c->windowRole()) return false; } if( !title.isEmpty()) { if( titleregexp && QRegExp( title ).search( c->caption( false )) == -1 ) return false; if( !titleregexp && title != c->caption( false )) return false; } // TODO extrarole if( !clientmachine.isEmpty()) { if( clientmachineregexp && QRegExp( clientmachine ).search( c->wmClientMachine( true )) == -1 && QRegExp( clientmachine ).search( c->wmClientMachine( false )) == -1 ) return false; if( !clientmachineregexp && clientmachine != c->wmClientMachine( true ) && clientmachine != c->wmClientMachine( false )) return false; } return true; } void WindowRules::update( Client* c ) { // TODO check this setting is for this client ? bool updated = false; if( positionrule == RememberRule ) { updated = updated || position != c->pos(); position = c->pos(); } if( sizerule == RememberRule ) { updated = updated || size != c->size(); size = c->size(); } if( desktoprule == RememberRule ) { updated = updated || desktop != c->desktop(); desktop = c->desktop(); } if( maximizevertrule == RememberRule ) { updated = updated || maximizevert != bool( c->maximizeMode() & MaximizeVertical ); maximizevert = c->maximizeMode() & MaximizeVertical; } if( maximizehorizrule == RememberRule ) { updated = updated || maximizehoriz != bool( c->maximizeMode() & MaximizeHorizontal ); maximizehoriz = c->maximizeMode() & MaximizeHorizontal; } if( minimizerule == RememberRule ) { updated = updated || minimize != c->isMinimized(); minimize = c->isMinimized(); } if( shaderule == RememberRule ) { updated = updated || ( shade != ( c->shadeMode() != Client::ShadeNone )); shade = c->shadeMode() != Client::ShadeNone; } if( skiptaskbarrule == RememberRule ) { updated = updated || skiptaskbar != c->skipTaskbar(); skiptaskbar = c->skipTaskbar(); } if( skippagerrule == RememberRule ) { updated = updated || skippager != c->skipPager(); skippager = c->skipPager(); } if( aboverule == RememberRule ) { updated = updated || above != c->keepAbove(); above = c->keepAbove(); } if( belowrule == RememberRule ) { updated = updated || below != c->keepBelow(); below = c->keepBelow(); } if( fullscreenrule == RememberRule ) { updated = updated || fullscreen != c->isFullScreen(); fullscreen = c->isFullScreen(); } if( noborderrule == RememberRule ) { updated = updated || noborder != c->isUserNoBorder(); noborder = c->isUserNoBorder(); } if( updated ) Workspace::self()->rulesUpdated(); } Placement::Policy WindowRules::checkPlacement( Placement::Policy placement ) const { return checkForceRule( placementrule ) ? this->placement : placement; } // TODO at to porad jeste kontroluje min/max size , + udelat override pro min/max? QRect WindowRules::checkGeometry( const QRect& rect, bool init ) const { return QRect( checkRule( positionrule, init ) && this->position != invalidPoint ? this->position : rect.topLeft(), checkRule( sizerule, init ) && this->size.isValid() ? this->size : rect.size()); } QPoint WindowRules::checkPosition( const QPoint& pos, bool init ) const { return checkRule( positionrule, init ) && this->position != invalidPoint ? this->position : pos; } QSize WindowRules::checkSize( const QSize& s, bool init ) const { return checkRule( sizerule, init ) && this->size.isValid() ? this->size : s; } QSize WindowRules::checkMinSize( const QSize& s ) const { return checkForceRule( minsizerule ) ? this->minsize : s; } QSize WindowRules::checkMaxSize( const QSize& s ) const { return checkForceRule( maxsizerule ) ? this->maxsize : s; } int WindowRules::checkDesktop( int req_desktop, bool init ) const { // TODO chaining? return checkRule( desktoprule, init ) ? this->desktop : req_desktop; } NET::WindowType WindowRules::checkType( NET::WindowType req_type ) const { return checkForceRule( typerule ) ? this->type : req_type; } KDecorationDefines::MaximizeMode WindowRules::checkMaximize( MaximizeMode mode, bool init ) const { bool vert = checkRule( maximizevertrule, init ) ? this->maximizevert : bool( mode & MaximizeVertical ); bool horiz = checkRule( maximizehorizrule, init ) ? this->maximizehoriz : bool( mode & MaximizeHorizontal ); return static_cast< MaximizeMode >(( vert ? MaximizeVertical : 0 ) | ( horiz ? MaximizeHorizontal : 0 )); } bool WindowRules::checkMinimize( bool minimize, bool init ) const { return checkRule( minimizerule, init ) ? this->minimize : minimize; } Client::ShadeMode WindowRules::checkShade( Client::ShadeMode shade, bool init ) const { if( checkRule( shaderule, init )) { if( !this->shade ) return Client::ShadeNone; if( this->shade && shade == Client::ShadeNone ) return Client::ShadeNormal; } return shade; } bool WindowRules::checkSkipTaskbar( bool skip, bool init ) const { return checkRule( skiptaskbarrule, init ) ? this->skiptaskbar : skip; } bool WindowRules::checkSkipPager( bool skip, bool init ) const { return checkRule( skippagerrule, init ) ? this->skippager : skip; } bool WindowRules::checkKeepAbove( bool above, bool init ) const { return checkRule( aboverule, init ) ? this->above : above; } bool WindowRules::checkKeepBelow( bool below, bool init ) const { return checkRule( belowrule, init ) ? this->below : below; } bool WindowRules::checkFullScreen( bool fs, bool init ) const { return checkRule( fullscreenrule, init ) ? this->fullscreen : fs; } bool WindowRules::checkNoBorder( bool noborder, bool init ) const { return checkRule( noborderrule, init ) ? this->noborder : noborder; } int WindowRules::checkFSP( int fsp ) const { return checkForceRule( fsplevelrule ) ? this->fsplevel : fsp; } bool WindowRules::checkAcceptFocus( bool focus ) const { return checkForceRule( acceptfocusrule ) ? this->acceptfocus : focus; } Options::MoveResizeMode WindowRules::checkMoveResizeMode( Options::MoveResizeMode mode ) const { return checkForceRule( moveresizemoderule ) ? this->moveresizemode : mode; } bool WindowRules::checkCloseable( bool closeable ) const { return checkForceRule( closeablerule ) ? this->closeable : closeable; } bool WindowRules::isTemporary() const { return temporary_state > 0; } bool WindowRules::discardTemporary( bool force ) { if( temporary_state == 0 ) // not temporary return false; if( force || --temporary_state == 0 ) // too old { kdDebug() << "DISCARD:" << wmclass << "/" << title << endl; delete this; return true; } return false; } // Client void Client::setupWindowRules( bool ignore_temporary ) { client_rules = workspace()->findWindowRules( this, ignore_temporary ); // check only after getting the rules, because there may be a rule forcing window type if( isTopMenu()) // TODO cannot have restrictions client_rules = &dummyRules; } void Client::updateWindowRules() { client_rules->update( this ); } void Client::finishWindowRules() { updateWindowRules(); client_rules = &dummyRules; } // Workspace WindowRules* Workspace::findWindowRules( const Client* c, bool ignore_temporary ) { for( QValueList< WindowRules* >::ConstIterator it = windowRules.begin(); it != windowRules.end(); ++it ) { if( ignore_temporary && (*it)->isTemporary()) continue; // TODO chaining for wildcard matches if( (*it)->match( c )) { kdDebug() << "RULE FOUND:" << c << endl; WindowRules* ret = *it; if( ret->isTemporary()) windowRules.remove( *it ); return ret; } } kdDebug() << "RULE DUMMY:" << c << endl; return &dummyRules; } void Workspace::editWindowRules( Client* c ) { // TODO } void Workspace::loadWindowRules() { while( !windowRules.isEmpty()) { delete windowRules.front(); windowRules.pop_front(); } KConfig cfg( "kwinrulesrc", true ); cfg.setGroup( "General" ); int count = cfg.readNumEntry( "count" ); kdDebug() << "RULES:" << count << endl; for( int i = 1; i <= count; ++i ) { cfg.setGroup( QString::number( i )); WindowRules* setting = new WindowRules( cfg ); windowRules.append( setting ); } } void Workspace::writeWindowRules() { KConfig cfg( "kwinrulesrc" ); cfg.setGroup( "General" ); cfg.writeEntry( "count", windowRules.count()); int i = 1; for( QValueList< WindowRules* >::ConstIterator it = windowRules.begin(); it != windowRules.end(); ++it ) { if( (*it)->isTemporary()) continue; cfg.setGroup( QString::number( i )); (*it)->write( cfg ); ++i; } } void Workspace::gotTemporaryRulesMessage( const QString& message ) { bool was_temporary = false; for( QValueList< WindowRules* >::ConstIterator it = windowRules.begin(); it != windowRules.end(); ++it ) if( (*it)->isTemporary()) was_temporary = true; WindowRules* rule = new WindowRules( message, true ); windowRules.prepend( rule ); // highest priority first if( !was_temporary ) QTimer::singleShot( 60000, this, SLOT( cleanupTemporaryRules())); } void Workspace::cleanupTemporaryRules() { kdDebug() << "CLEANUP" << endl; bool has_temporary = false; for( QValueList< WindowRules* >::Iterator it = windowRules.begin(); it != windowRules.end(); ) { if( (*it)->discardTemporary( false )) it = windowRules.remove( it ); else { if( (*it)->isTemporary()) has_temporary = true; ++it; } } if( has_temporary ) QTimer::singleShot( 60000, this, SLOT( cleanupTemporaryRules())); } void Workspace::rulesUpdated() { rulesUpdatedTimer.start( 1000, true ); } } // namespace