f6458fa1e8
The problem with KToolInvocation is that it creates a dead lock on Wayland in case kdeinit is not already running. It starts kdeinit and does a QProcess::waitForFinished and our kdeinit needs to interact with the wayland server. So dead lock. As KRun also calls into the dangerous code path it's no option which leaves us with QProcess to start the processes. A nice side-effect is that we no don't need to link KF5::Service any more from kwin_core. Now once Plasma and Notification don't use it any more, it will be gone completely.
1119 lines
36 KiB
C++
1119 lines
36 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
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, see <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
|
|
#include "rules.h"
|
|
|
|
#include <fixx11h.h>
|
|
#include <kconfig.h>
|
|
#include <KXMessages>
|
|
#include <QRegExp>
|
|
#include <QTemporaryFile>
|
|
#include <QFile>
|
|
#include <QFileInfo>
|
|
#include <QDebug>
|
|
|
|
#ifndef KCMRULES
|
|
#include "client.h"
|
|
#include "client_machine.h"
|
|
#include "screens.h"
|
|
#include "workspace.h"
|
|
#endif
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
Rules::Rules()
|
|
: temporary_state(0)
|
|
, wmclassmatch(UnimportantMatch)
|
|
, wmclasscomplete(UnimportantMatch)
|
|
, windowrolematch(UnimportantMatch)
|
|
, titlematch(UnimportantMatch)
|
|
, clientmachinematch(UnimportantMatch)
|
|
, types(NET::AllTypesMask)
|
|
, placementrule(UnusedForceRule)
|
|
, positionrule(UnusedSetRule)
|
|
, sizerule(UnusedSetRule)
|
|
, minsizerule(UnusedForceRule)
|
|
, maxsizerule(UnusedForceRule)
|
|
, opacityactiverule(UnusedForceRule)
|
|
, opacityinactiverule(UnusedForceRule)
|
|
, ignoregeometryrule(UnusedSetRule)
|
|
, desktoprule(UnusedSetRule)
|
|
, screenrule(UnusedSetRule)
|
|
, activityrule(UnusedSetRule)
|
|
, typerule(UnusedForceRule)
|
|
, maximizevertrule(UnusedSetRule)
|
|
, maximizehorizrule(UnusedSetRule)
|
|
, minimizerule(UnusedSetRule)
|
|
, shaderule(UnusedSetRule)
|
|
, skiptaskbarrule(UnusedSetRule)
|
|
, skippagerrule(UnusedSetRule)
|
|
, skipswitcherrule(UnusedSetRule)
|
|
, aboverule(UnusedSetRule)
|
|
, belowrule(UnusedSetRule)
|
|
, fullscreenrule(UnusedSetRule)
|
|
, noborderrule(UnusedSetRule)
|
|
, decocolorrule(UnusedForceRule)
|
|
, blockcompositingrule(UnusedForceRule)
|
|
, fsplevelrule(UnusedForceRule)
|
|
, acceptfocusrule(UnusedForceRule)
|
|
, closeablerule(UnusedForceRule)
|
|
, autogrouprule(UnusedForceRule)
|
|
, autogroupfgrule(UnusedForceRule)
|
|
, autogroupidrule(UnusedForceRule)
|
|
, strictgeometryrule(UnusedForceRule)
|
|
, shortcutrule(UnusedSetRule)
|
|
, disableglobalshortcutsrule(UnusedForceRule)
|
|
{
|
|
}
|
|
|
|
Rules::Rules(const QString& str, bool temporary)
|
|
: temporary_state(temporary ? 2 : 0)
|
|
{
|
|
QTemporaryFile file;
|
|
if (file.open()) {
|
|
QByteArray s = str.toUtf8();
|
|
file.write(s.data(), s.length());
|
|
}
|
|
file.flush();
|
|
KConfig cfg(file.fileName(), KConfig::SimpleConfig);
|
|
readFromCfg(cfg.group(QString()));
|
|
if (description.isEmpty())
|
|
description = QStringLiteral("temporary");
|
|
}
|
|
|
|
#define READ_MATCH_STRING( var, func ) \
|
|
var = cfg.readEntry( #var ) func; \
|
|
var##match = (StringMatch) qMax( FirstStringMatch, \
|
|
qMin( LastStringMatch, static_cast< StringMatch >( cfg.readEntry( #var "match",0 ))));
|
|
|
|
#define READ_SET_RULE( var, func, def ) \
|
|
var = func ( cfg.readEntry( #var, def)); \
|
|
var##rule = readSetRule( cfg, QStringLiteral( #var "rule" ) );
|
|
|
|
#define READ_SET_RULE_DEF( var , func, def ) \
|
|
var = func ( cfg.readEntry( #var, def )); \
|
|
var##rule = readSetRule( cfg, QStringLiteral( #var "rule" ) );
|
|
|
|
#define READ_FORCE_RULE( var, func, def) \
|
|
var = func ( cfg.readEntry( #var, def)); \
|
|
var##rule = readForceRule( cfg, QStringLiteral( #var "rule" ) );
|
|
|
|
#define READ_FORCE_RULE2( var, def, func, funcarg ) \
|
|
var = func ( cfg.readEntry( #var, def),funcarg ); \
|
|
var##rule = readForceRule( cfg, QStringLiteral( #var "rule" ) );
|
|
|
|
|
|
|
|
Rules::Rules(const KConfigGroup& cfg)
|
|
: temporary_state(0)
|
|
{
|
|
readFromCfg(cfg);
|
|
}
|
|
|
|
static int limit0to4(int i)
|
|
{
|
|
return qMax(0, qMin(4, i));
|
|
}
|
|
|
|
void Rules::readFromCfg(const KConfigGroup& cfg)
|
|
{
|
|
description = cfg.readEntry("Description");
|
|
if (description.isEmpty()) // capitalized first, lowercase for backwards compatibility
|
|
description = cfg.readEntry("description");
|
|
READ_MATCH_STRING(wmclass, .toLower().toLatin1());
|
|
wmclasscomplete = cfg.readEntry("wmclasscomplete" , false);
|
|
READ_MATCH_STRING(windowrole, .toLatin1());
|
|
READ_MATCH_STRING(title,);
|
|
READ_MATCH_STRING(clientmachine, .toLower().toLatin1());
|
|
types = NET::WindowTypeMask(cfg.readEntry<uint>("types", NET::AllTypesMask));
|
|
READ_FORCE_RULE2(placement, QString(), Placement::policyFromString, false);
|
|
READ_SET_RULE_DEF(position, , invalidPoint);
|
|
READ_SET_RULE(size, , QSize());
|
|
if (size.isEmpty() && sizerule != (SetRule)Remember)
|
|
sizerule = UnusedSetRule;
|
|
READ_FORCE_RULE(minsize, , QSize());
|
|
if (!minsize.isValid())
|
|
minsize = QSize(1, 1);
|
|
READ_FORCE_RULE(maxsize, , QSize());
|
|
if (maxsize.isEmpty())
|
|
maxsize = QSize(32767, 32767);
|
|
READ_FORCE_RULE(opacityactive, , 0);
|
|
if (opacityactive < 0 || opacityactive > 100)
|
|
opacityactive = 100;
|
|
READ_FORCE_RULE(opacityinactive, , 0);
|
|
if (opacityinactive < 0 || opacityinactive > 100)
|
|
opacityinactive = 100;
|
|
READ_SET_RULE(ignoregeometry, , false);
|
|
READ_SET_RULE(desktop, , 0);
|
|
READ_SET_RULE(screen, , 0);
|
|
READ_SET_RULE(activity, , QString());
|
|
type = readType(cfg, QStringLiteral("type"));
|
|
typerule = type != NET::Unknown ? readForceRule(cfg, QStringLiteral("typerule")) : UnusedForceRule;
|
|
READ_SET_RULE(maximizevert, , false);
|
|
READ_SET_RULE(maximizehoriz, , false);
|
|
READ_SET_RULE(minimize, , false);
|
|
READ_SET_RULE(shade, , false);
|
|
READ_SET_RULE(skiptaskbar, , false);
|
|
READ_SET_RULE(skippager, , false);
|
|
READ_SET_RULE(skipswitcher, , false);
|
|
READ_SET_RULE(above, , false);
|
|
READ_SET_RULE(below, , false);
|
|
READ_SET_RULE(fullscreen, , false);
|
|
READ_SET_RULE(noborder, , false);
|
|
decocolor = readDecoColor(cfg);
|
|
decocolorrule = decocolor.isEmpty() ? UnusedForceRule : readForceRule(cfg, QStringLiteral("decocolorrule"));
|
|
READ_FORCE_RULE(blockcompositing, , false);
|
|
READ_FORCE_RULE(fsplevel, limit0to4, 0); // fsp is 0-4
|
|
READ_FORCE_RULE(acceptfocus, , false);
|
|
READ_FORCE_RULE(closeable, , false);
|
|
READ_FORCE_RULE(autogroup, , false);
|
|
READ_FORCE_RULE(autogroupfg, , true);
|
|
READ_FORCE_RULE(autogroupid, , QString());
|
|
READ_FORCE_RULE(strictgeometry, , false);
|
|
READ_SET_RULE(shortcut, , QString());
|
|
READ_FORCE_RULE(disableglobalshortcuts, , false);
|
|
}
|
|
|
|
#undef READ_MATCH_STRING
|
|
#undef READ_SET_RULE
|
|
#undef READ_FORCE_RULE
|
|
#undef READ_FORCE_RULE2
|
|
|
|
#define WRITE_MATCH_STRING( var, force ) \
|
|
if ( !var.isEmpty() || force ) \
|
|
{ \
|
|
cfg.writeEntry( #var, var ); \
|
|
cfg.writeEntry( #var "match", (int)var##match ); \
|
|
} \
|
|
else \
|
|
{ \
|
|
cfg.deleteEntry( #var ); \
|
|
cfg.deleteEntry( #var "match" ); \
|
|
}
|
|
|
|
#define WRITE_SET_RULE( var, func ) \
|
|
if ( var##rule != UnusedSetRule ) \
|
|
{ \
|
|
cfg.writeEntry( #var, func ( var )); \
|
|
cfg.writeEntry( #var "rule", (int)var##rule ); \
|
|
} \
|
|
else \
|
|
{ \
|
|
cfg.deleteEntry( #var ); \
|
|
cfg.deleteEntry( #var "rule" ); \
|
|
}
|
|
|
|
#define WRITE_FORCE_RULE( var, func ) \
|
|
if ( var##rule != UnusedForceRule ) \
|
|
{ \
|
|
cfg.writeEntry( #var, func ( var )); \
|
|
cfg.writeEntry( #var "rule", (int)var##rule ); \
|
|
} \
|
|
else \
|
|
{ \
|
|
cfg.deleteEntry( #var ); \
|
|
cfg.deleteEntry( #var "rule" ); \
|
|
}
|
|
|
|
void Rules::write(KConfigGroup& cfg) const
|
|
{
|
|
cfg.writeEntry("Description", description);
|
|
// always write wmclass
|
|
WRITE_MATCH_STRING(wmclass, true);
|
|
cfg.writeEntry("wmclasscomplete", wmclasscomplete);
|
|
WRITE_MATCH_STRING(windowrole, false);
|
|
WRITE_MATCH_STRING(title, false);
|
|
WRITE_MATCH_STRING(clientmachine, false);
|
|
if (types != NET::AllTypesMask)
|
|
cfg.writeEntry("types", uint(types));
|
|
else
|
|
cfg.deleteEntry("types");
|
|
WRITE_FORCE_RULE(placement, Placement::policyToString);
|
|
WRITE_SET_RULE(position,);
|
|
WRITE_SET_RULE(size,);
|
|
WRITE_FORCE_RULE(minsize,);
|
|
WRITE_FORCE_RULE(maxsize,);
|
|
WRITE_FORCE_RULE(opacityactive,);
|
|
WRITE_FORCE_RULE(opacityinactive,);
|
|
WRITE_SET_RULE(ignoregeometry,);
|
|
WRITE_SET_RULE(desktop,);
|
|
WRITE_SET_RULE(screen,);
|
|
WRITE_SET_RULE(activity,);
|
|
WRITE_FORCE_RULE(type, int);
|
|
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(skipswitcher,);
|
|
WRITE_SET_RULE(above,);
|
|
WRITE_SET_RULE(below,);
|
|
WRITE_SET_RULE(fullscreen,);
|
|
WRITE_SET_RULE(noborder,);
|
|
auto colorToString = [](const QString &value) -> QString {
|
|
if (value.endsWith(QStringLiteral(".colors"))) {
|
|
return QFileInfo(value).baseName();
|
|
} else {
|
|
return value;
|
|
}
|
|
};
|
|
WRITE_FORCE_RULE(decocolor, colorToString);
|
|
WRITE_FORCE_RULE(blockcompositing,);
|
|
WRITE_FORCE_RULE(fsplevel,);
|
|
WRITE_FORCE_RULE(acceptfocus,);
|
|
WRITE_FORCE_RULE(closeable,);
|
|
WRITE_FORCE_RULE(autogroup,);
|
|
WRITE_FORCE_RULE(autogroupfg,);
|
|
WRITE_FORCE_RULE(autogroupid,);
|
|
WRITE_FORCE_RULE(strictgeometry,);
|
|
WRITE_SET_RULE(shortcut,);
|
|
WRITE_FORCE_RULE(disableglobalshortcuts,);
|
|
}
|
|
|
|
#undef WRITE_MATCH_STRING
|
|
#undef WRITE_SET_RULE
|
|
#undef WRITE_FORCE_RULE
|
|
|
|
// returns true if it doesn't affect anything
|
|
bool Rules::isEmpty() const
|
|
{
|
|
return(placementrule == UnusedForceRule
|
|
&& positionrule == UnusedSetRule
|
|
&& sizerule == UnusedSetRule
|
|
&& minsizerule == UnusedForceRule
|
|
&& maxsizerule == UnusedForceRule
|
|
&& opacityactiverule == UnusedForceRule
|
|
&& opacityinactiverule == UnusedForceRule
|
|
&& ignoregeometryrule == UnusedSetRule
|
|
&& desktoprule == UnusedSetRule
|
|
&& screenrule == UnusedSetRule
|
|
&& activityrule == UnusedSetRule
|
|
&& typerule == UnusedForceRule
|
|
&& maximizevertrule == UnusedSetRule
|
|
&& maximizehorizrule == UnusedSetRule
|
|
&& minimizerule == UnusedSetRule
|
|
&& shaderule == UnusedSetRule
|
|
&& skiptaskbarrule == UnusedSetRule
|
|
&& skippagerrule == UnusedSetRule
|
|
&& skipswitcherrule == UnusedSetRule
|
|
&& aboverule == UnusedSetRule
|
|
&& belowrule == UnusedSetRule
|
|
&& fullscreenrule == UnusedSetRule
|
|
&& noborderrule == UnusedSetRule
|
|
&& decocolorrule == UnusedForceRule
|
|
&& blockcompositingrule == UnusedForceRule
|
|
&& fsplevelrule == UnusedForceRule
|
|
&& acceptfocusrule == UnusedForceRule
|
|
&& closeablerule == UnusedForceRule
|
|
&& autogrouprule == UnusedForceRule
|
|
&& autogroupfgrule == UnusedForceRule
|
|
&& autogroupidrule == UnusedForceRule
|
|
&& strictgeometryrule == UnusedForceRule
|
|
&& shortcutrule == UnusedSetRule
|
|
&& disableglobalshortcutsrule == UnusedForceRule);
|
|
}
|
|
|
|
Rules::SetRule Rules::readSetRule(const KConfigGroup& cfg, const QString& key)
|
|
{
|
|
int v = cfg.readEntry(key, 0);
|
|
if (v >= DontAffect && v <= ForceTemporarily)
|
|
return static_cast< SetRule >(v);
|
|
return UnusedSetRule;
|
|
}
|
|
|
|
Rules::ForceRule Rules::readForceRule(const KConfigGroup& cfg, const QString& key)
|
|
{
|
|
int v = cfg.readEntry(key, 0);
|
|
if (v == DontAffect || v == Force || v == ForceTemporarily)
|
|
return static_cast< ForceRule >(v);
|
|
return UnusedForceRule;
|
|
}
|
|
|
|
NET::WindowType Rules::readType(const KConfigGroup& cfg, const QString& key)
|
|
{
|
|
int v = cfg.readEntry(key, 0);
|
|
if (v >= NET::Normal && v <= NET::Splash)
|
|
return static_cast< NET::WindowType >(v);
|
|
return NET::Unknown;
|
|
}
|
|
|
|
QString Rules::readDecoColor(const KConfigGroup &cfg)
|
|
{
|
|
QString themeName = cfg.readEntry("decocolor", QString());
|
|
if (themeName.isEmpty()) {
|
|
return QString();
|
|
}
|
|
// find the actual scheme file
|
|
return QStandardPaths::locate(QStandardPaths::GenericDataLocation,
|
|
QStringLiteral("color-schemes/") + themeName + QStringLiteral(".colors"));
|
|
}
|
|
|
|
bool Rules::matchType(NET::WindowType match_type) const
|
|
{
|
|
if (types != NET::AllTypesMask) {
|
|
if (match_type == NET::Unknown)
|
|
match_type = NET::Normal; // NET::Unknown->NET::Normal is only here for matching
|
|
if (!NET::typeMatchesMask(match_type, types))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Rules::matchWMClass(const QByteArray& match_class, const QByteArray& match_name) const
|
|
{
|
|
if (wmclassmatch != UnimportantMatch) {
|
|
// TODO optimize?
|
|
QByteArray cwmclass = wmclasscomplete
|
|
? match_name + ' ' + match_class : match_class;
|
|
if (wmclassmatch == RegExpMatch && QRegExp(QString::fromUtf8(wmclass)).indexIn(QString::fromUtf8(cwmclass)) == -1)
|
|
return false;
|
|
if (wmclassmatch == ExactMatch && wmclass != cwmclass)
|
|
return false;
|
|
if (wmclassmatch == SubstringMatch && !cwmclass.contains(wmclass))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Rules::matchRole(const QByteArray& match_role) const
|
|
{
|
|
if (windowrolematch != UnimportantMatch) {
|
|
if (windowrolematch == RegExpMatch && QRegExp(QString::fromUtf8(windowrole)).indexIn(QString::fromUtf8(match_role)) == -1)
|
|
return false;
|
|
if (windowrolematch == ExactMatch && windowrole != match_role)
|
|
return false;
|
|
if (windowrolematch == SubstringMatch && !match_role.contains(windowrole))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Rules::matchTitle(const QString& match_title) const
|
|
{
|
|
if (titlematch != UnimportantMatch) {
|
|
if (titlematch == RegExpMatch && QRegExp(title).indexIn(match_title) == -1)
|
|
return false;
|
|
if (titlematch == ExactMatch && title != match_title)
|
|
return false;
|
|
if (titlematch == SubstringMatch && !match_title.contains(title))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Rules::matchClientMachine(const QByteArray& match_machine, bool local) const
|
|
{
|
|
if (clientmachinematch != UnimportantMatch) {
|
|
// if it's localhost, check also "localhost" before checking hostname
|
|
if (match_machine != "localhost" && local
|
|
&& matchClientMachine("localhost", true))
|
|
return true;
|
|
if (clientmachinematch == RegExpMatch
|
|
&& QRegExp(QString::fromUtf8(clientmachine)).indexIn(QString::fromUtf8(match_machine)) == -1)
|
|
return false;
|
|
if (clientmachinematch == ExactMatch
|
|
&& clientmachine != match_machine)
|
|
return false;
|
|
if (clientmachinematch == SubstringMatch
|
|
&& !match_machine.contains(clientmachine))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
#ifndef KCMRULES
|
|
bool Rules::match(const Client* c) const
|
|
{
|
|
if (!matchType(c->windowType(true)))
|
|
return false;
|
|
if (!matchWMClass(c->resourceClass(), c->resourceName()))
|
|
return false;
|
|
if (!matchRole(c->windowRole()))
|
|
return false;
|
|
if (!matchTitle(c->caption(false)))
|
|
return false;
|
|
if (!matchClientMachine(c->clientMachine()->hostName(), c->clientMachine()->isLocal()))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
#define NOW_REMEMBER(_T_, _V_) ((selection & _T_) && (_V_##rule == (SetRule)Remember))
|
|
|
|
bool Rules::update(Client* c, int selection)
|
|
{
|
|
// TODO check this setting is for this client ?
|
|
bool updated = false;
|
|
if NOW_REMEMBER(Position, position) {
|
|
if (!c->isFullScreen()) {
|
|
QPoint new_pos = position;
|
|
// don't use the position in the direction which is maximized
|
|
if ((c->maximizeMode() & MaximizeHorizontal) == 0)
|
|
new_pos.setX(c->pos().x());
|
|
if ((c->maximizeMode() & MaximizeVertical) == 0)
|
|
new_pos.setY(c->pos().y());
|
|
updated = updated || position != new_pos;
|
|
position = new_pos;
|
|
}
|
|
}
|
|
if NOW_REMEMBER(Size, size) {
|
|
if (!c->isFullScreen()) {
|
|
QSize new_size = size;
|
|
// don't use the position in the direction which is maximized
|
|
if ((c->maximizeMode() & MaximizeHorizontal) == 0)
|
|
new_size.setWidth(c->size().width());
|
|
if ((c->maximizeMode() & MaximizeVertical) == 0)
|
|
new_size.setHeight(c->size().height());
|
|
updated = updated || size != new_size;
|
|
size = new_size;
|
|
}
|
|
}
|
|
if NOW_REMEMBER(Desktop, desktop) {
|
|
updated = updated || desktop != c->desktop();
|
|
desktop = c->desktop();
|
|
}
|
|
if NOW_REMEMBER(Screen, screen) {
|
|
updated = updated || screen != c->screen();
|
|
screen = c->screen();
|
|
}
|
|
if NOW_REMEMBER(Activity, activity) {
|
|
// TODO: ivan - multiple activities support
|
|
const QString & joinedActivities = c->activities().join(QStringLiteral(","));
|
|
updated = updated || activity != joinedActivities;
|
|
activity = joinedActivities;
|
|
}
|
|
if NOW_REMEMBER(MaximizeVert, maximizevert) {
|
|
updated = updated || maximizevert != bool(c->maximizeMode() & MaximizeVertical);
|
|
maximizevert = c->maximizeMode() & MaximizeVertical;
|
|
}
|
|
if NOW_REMEMBER(MaximizeHoriz, maximizehoriz) {
|
|
updated = updated || maximizehoriz != bool(c->maximizeMode() & MaximizeHorizontal);
|
|
maximizehoriz = c->maximizeMode() & MaximizeHorizontal;
|
|
}
|
|
if NOW_REMEMBER(Minimize, minimize) {
|
|
updated = updated || minimize != c->isMinimized();
|
|
minimize = c->isMinimized();
|
|
}
|
|
if NOW_REMEMBER(Shade, shade) {
|
|
updated = updated || (shade != (c->shadeMode() != ShadeNone));
|
|
shade = c->shadeMode() != ShadeNone;
|
|
}
|
|
if NOW_REMEMBER(SkipTaskbar, skiptaskbar) {
|
|
updated = updated || skiptaskbar != c->skipTaskbar();
|
|
skiptaskbar = c->skipTaskbar();
|
|
}
|
|
if NOW_REMEMBER(SkipPager, skippager) {
|
|
updated = updated || skippager != c->skipPager();
|
|
skippager = c->skipPager();
|
|
}
|
|
if NOW_REMEMBER(SkipSwitcher, skipswitcher) {
|
|
updated = updated || skipswitcher != c->skipSwitcher();
|
|
skipswitcher = c->skipSwitcher();
|
|
}
|
|
if NOW_REMEMBER(Above, above) {
|
|
updated = updated || above != c->keepAbove();
|
|
above = c->keepAbove();
|
|
}
|
|
if NOW_REMEMBER(Below, below) {
|
|
updated = updated || below != c->keepBelow();
|
|
below = c->keepBelow();
|
|
}
|
|
if NOW_REMEMBER(Fullscreen, fullscreen) {
|
|
updated = updated || fullscreen != c->isFullScreen();
|
|
fullscreen = c->isFullScreen();
|
|
}
|
|
if NOW_REMEMBER(NoBorder, noborder) {
|
|
updated = updated || noborder != c->noBorder();
|
|
noborder = c->noBorder();
|
|
}
|
|
return updated;
|
|
}
|
|
|
|
#undef NOW_REMEMBER
|
|
|
|
#define APPLY_RULE( var, name, type ) \
|
|
bool Rules::apply##name( type& arg, bool init ) const \
|
|
{ \
|
|
if ( checkSetRule( var##rule, init )) \
|
|
arg = this->var; \
|
|
return checkSetStop( var##rule ); \
|
|
}
|
|
|
|
#define APPLY_FORCE_RULE( var, name, type ) \
|
|
bool Rules::apply##name( type& arg ) const \
|
|
{ \
|
|
if ( checkForceRule( var##rule )) \
|
|
arg = this->var; \
|
|
return checkForceStop( var##rule ); \
|
|
}
|
|
|
|
APPLY_FORCE_RULE(placement, Placement, Placement::Policy)
|
|
|
|
bool Rules::applyGeometry(QRect& rect, bool init) const
|
|
{
|
|
QPoint p = rect.topLeft();
|
|
QSize s = rect.size();
|
|
bool ret = false; // no short-circuiting
|
|
if (applyPosition(p, init)) {
|
|
rect.moveTopLeft(p);
|
|
ret = true;
|
|
}
|
|
if (applySize(s, init)) {
|
|
rect.setSize(s);
|
|
ret = true;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
bool Rules::applyPosition(QPoint& pos, bool init) const
|
|
{
|
|
if (this->position != invalidPoint && checkSetRule(positionrule, init))
|
|
pos = this->position;
|
|
return checkSetStop(positionrule);
|
|
}
|
|
|
|
bool Rules::applySize(QSize& s, bool init) const
|
|
{
|
|
if (this->size.isValid() && checkSetRule(sizerule, init))
|
|
s = this->size;
|
|
return checkSetStop(sizerule);
|
|
}
|
|
|
|
APPLY_FORCE_RULE(minsize, MinSize, QSize)
|
|
APPLY_FORCE_RULE(maxsize, MaxSize, QSize)
|
|
APPLY_FORCE_RULE(opacityactive, OpacityActive, int)
|
|
APPLY_FORCE_RULE(opacityinactive, OpacityInactive, int)
|
|
APPLY_RULE(ignoregeometry, IgnoreGeometry, bool)
|
|
|
|
APPLY_RULE(desktop, Desktop, int)
|
|
APPLY_RULE(screen, Screen, int)
|
|
APPLY_RULE(activity, Activity, QString)
|
|
APPLY_FORCE_RULE(type, Type, NET::WindowType)
|
|
|
|
bool Rules::applyMaximizeHoriz(MaximizeMode& mode, bool init) const
|
|
{
|
|
if (checkSetRule(maximizehorizrule, init))
|
|
mode = static_cast< MaximizeMode >((maximizehoriz ? MaximizeHorizontal : 0) | (mode & MaximizeVertical));
|
|
return checkSetStop(maximizehorizrule);
|
|
}
|
|
|
|
bool Rules::applyMaximizeVert(MaximizeMode& mode, bool init) const
|
|
{
|
|
if (checkSetRule(maximizevertrule, init))
|
|
mode = static_cast< MaximizeMode >((maximizevert ? MaximizeVertical : 0) | (mode & MaximizeHorizontal));
|
|
return checkSetStop(maximizevertrule);
|
|
}
|
|
|
|
APPLY_RULE(minimize, Minimize, bool)
|
|
|
|
bool Rules::applyShade(ShadeMode& sh, bool init) const
|
|
{
|
|
if (checkSetRule(shaderule, init)) {
|
|
if (!this->shade)
|
|
sh = ShadeNone;
|
|
if (this->shade && sh == ShadeNone)
|
|
sh = ShadeNormal;
|
|
}
|
|
return checkSetStop(shaderule);
|
|
}
|
|
|
|
APPLY_RULE(skiptaskbar, SkipTaskbar, bool)
|
|
APPLY_RULE(skippager, SkipPager, bool)
|
|
APPLY_RULE(skipswitcher, SkipSwitcher, bool)
|
|
APPLY_RULE(above, KeepAbove, bool)
|
|
APPLY_RULE(below, KeepBelow, bool)
|
|
APPLY_RULE(fullscreen, FullScreen, bool)
|
|
APPLY_RULE(noborder, NoBorder, bool)
|
|
APPLY_FORCE_RULE(decocolor, DecoColor, QString)
|
|
APPLY_FORCE_RULE(blockcompositing, BlockCompositing, bool)
|
|
APPLY_FORCE_RULE(fsplevel, FSP, int)
|
|
APPLY_FORCE_RULE(acceptfocus, AcceptFocus, bool)
|
|
APPLY_FORCE_RULE(closeable, Closeable, bool)
|
|
APPLY_FORCE_RULE(autogroup, Autogrouping, bool)
|
|
APPLY_FORCE_RULE(autogroupfg, AutogroupInForeground, bool)
|
|
APPLY_FORCE_RULE(autogroupid, AutogroupById, QString)
|
|
APPLY_FORCE_RULE(strictgeometry, StrictGeometry, bool)
|
|
APPLY_RULE(shortcut, Shortcut, QString)
|
|
APPLY_FORCE_RULE(disableglobalshortcuts, DisableGlobalShortcuts, bool)
|
|
|
|
|
|
#undef APPLY_RULE
|
|
#undef APPLY_FORCE_RULE
|
|
|
|
bool Rules::isTemporary() const
|
|
{
|
|
return temporary_state > 0;
|
|
}
|
|
|
|
bool Rules::discardTemporary(bool force)
|
|
{
|
|
if (temporary_state == 0) // not temporary
|
|
return false;
|
|
if (force || --temporary_state == 0) { // too old
|
|
delete this;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
#define DISCARD_USED_SET_RULE( var ) \
|
|
do { \
|
|
if ( var##rule == ( SetRule ) ApplyNow || ( withdrawn && var##rule == ( SetRule ) ForceTemporarily )) \
|
|
var##rule = UnusedSetRule; \
|
|
} while ( false )
|
|
#define DISCARD_USED_FORCE_RULE( var ) \
|
|
do { \
|
|
if ( withdrawn && var##rule == ( ForceRule ) ForceTemporarily ) \
|
|
var##rule = UnusedForceRule; \
|
|
} while ( false )
|
|
|
|
void Rules::discardUsed(bool withdrawn)
|
|
{
|
|
DISCARD_USED_FORCE_RULE(placement);
|
|
DISCARD_USED_SET_RULE(position);
|
|
DISCARD_USED_SET_RULE(size);
|
|
DISCARD_USED_FORCE_RULE(minsize);
|
|
DISCARD_USED_FORCE_RULE(maxsize);
|
|
DISCARD_USED_FORCE_RULE(opacityactive);
|
|
DISCARD_USED_FORCE_RULE(opacityinactive);
|
|
DISCARD_USED_SET_RULE(ignoregeometry);
|
|
DISCARD_USED_SET_RULE(desktop);
|
|
DISCARD_USED_SET_RULE(screen);
|
|
DISCARD_USED_SET_RULE(activity);
|
|
DISCARD_USED_FORCE_RULE(type);
|
|
DISCARD_USED_SET_RULE(maximizevert);
|
|
DISCARD_USED_SET_RULE(maximizehoriz);
|
|
DISCARD_USED_SET_RULE(minimize);
|
|
DISCARD_USED_SET_RULE(shade);
|
|
DISCARD_USED_SET_RULE(skiptaskbar);
|
|
DISCARD_USED_SET_RULE(skippager);
|
|
DISCARD_USED_SET_RULE(skipswitcher);
|
|
DISCARD_USED_SET_RULE(above);
|
|
DISCARD_USED_SET_RULE(below);
|
|
DISCARD_USED_SET_RULE(fullscreen);
|
|
DISCARD_USED_SET_RULE(noborder);
|
|
DISCARD_USED_FORCE_RULE(decocolor);
|
|
DISCARD_USED_FORCE_RULE(blockcompositing);
|
|
DISCARD_USED_FORCE_RULE(fsplevel);
|
|
DISCARD_USED_FORCE_RULE(acceptfocus);
|
|
DISCARD_USED_FORCE_RULE(closeable);
|
|
DISCARD_USED_FORCE_RULE(autogroup);
|
|
DISCARD_USED_FORCE_RULE(autogroupfg);
|
|
DISCARD_USED_FORCE_RULE(autogroupid);
|
|
DISCARD_USED_FORCE_RULE(strictgeometry);
|
|
DISCARD_USED_SET_RULE(shortcut);
|
|
DISCARD_USED_FORCE_RULE(disableglobalshortcuts);
|
|
}
|
|
#undef DISCARD_USED_SET_RULE
|
|
#undef DISCARD_USED_FORCE_RULE
|
|
|
|
#endif
|
|
|
|
QDebug& operator<<(QDebug& stream, const Rules* r)
|
|
{
|
|
return stream << "[" << r->description << ":" << r->wmclass << "]" ;
|
|
}
|
|
|
|
#ifndef KCMRULES
|
|
void WindowRules::discardTemporary()
|
|
{
|
|
QVector< Rules* >::Iterator it2 = rules.begin();
|
|
for (QVector< Rules* >::Iterator it = rules.begin();
|
|
it != rules.end();
|
|
) {
|
|
if ((*it)->discardTemporary(true))
|
|
++it;
|
|
else {
|
|
*it2++ = *it++;
|
|
}
|
|
}
|
|
rules.erase(it2, rules.end());
|
|
}
|
|
|
|
void WindowRules::update(Client* c, int selection)
|
|
{
|
|
bool updated = false;
|
|
for (QVector< Rules* >::ConstIterator it = rules.constBegin();
|
|
it != rules.constEnd();
|
|
++it)
|
|
if ((*it)->update(c, selection)) // no short-circuiting here
|
|
updated = true;
|
|
if (updated)
|
|
RuleBook::self()->requestDiskStorage();
|
|
}
|
|
|
|
#define CHECK_RULE( rule, type ) \
|
|
type WindowRules::check##rule( type arg, bool init ) const \
|
|
{ \
|
|
if ( rules.count() == 0 ) \
|
|
return arg; \
|
|
type ret = arg; \
|
|
for ( QVector< Rules* >::ConstIterator it = rules.constBegin(); \
|
|
it != rules.constEnd(); \
|
|
++it ) \
|
|
{ \
|
|
if ( (*it)->apply##rule( ret, init )) \
|
|
break; \
|
|
} \
|
|
return ret; \
|
|
}
|
|
|
|
#define CHECK_FORCE_RULE( rule, type ) \
|
|
type WindowRules::check##rule( type arg ) const \
|
|
{ \
|
|
if ( rules.count() == 0 ) \
|
|
return arg; \
|
|
type ret = arg; \
|
|
for ( QVector< Rules* >::ConstIterator it = rules.begin(); \
|
|
it != rules.end(); \
|
|
++it ) \
|
|
{ \
|
|
if ( (*it)->apply##rule( ret )) \
|
|
break; \
|
|
} \
|
|
return ret; \
|
|
}
|
|
|
|
CHECK_FORCE_RULE(Placement, Placement::Policy)
|
|
|
|
QRect WindowRules::checkGeometry(QRect rect, bool init) const
|
|
{
|
|
return QRect(checkPosition(rect.topLeft(), init), checkSize(rect.size(), init));
|
|
}
|
|
|
|
CHECK_RULE(Position, QPoint)
|
|
CHECK_RULE(Size, QSize)
|
|
CHECK_FORCE_RULE(MinSize, QSize)
|
|
CHECK_FORCE_RULE(MaxSize, QSize)
|
|
CHECK_FORCE_RULE(OpacityActive, int)
|
|
CHECK_FORCE_RULE(OpacityInactive, int)
|
|
CHECK_RULE(IgnoreGeometry, bool)
|
|
|
|
CHECK_RULE(Desktop, int)
|
|
CHECK_RULE(Activity, QString)
|
|
CHECK_FORCE_RULE(Type, NET::WindowType)
|
|
CHECK_RULE(MaximizeVert, MaximizeMode)
|
|
CHECK_RULE(MaximizeHoriz, MaximizeMode)
|
|
|
|
MaximizeMode WindowRules::checkMaximize(MaximizeMode mode, bool init) const
|
|
{
|
|
bool vert = checkMaximizeVert(mode, init) & MaximizeVertical;
|
|
bool horiz = checkMaximizeHoriz(mode, init) & MaximizeHorizontal;
|
|
return static_cast< MaximizeMode >((vert ? MaximizeVertical : 0) | (horiz ? MaximizeHorizontal : 0));
|
|
}
|
|
|
|
int WindowRules::checkScreen(int screen, bool init) const
|
|
{
|
|
if ( rules.count() == 0 )
|
|
return screen;
|
|
int ret = screen;
|
|
for ( QVector< Rules* >::ConstIterator it = rules.constBegin(); it != rules.constEnd(); ++it ) {
|
|
if ( (*it)->applyScreen( ret, init ))
|
|
break;
|
|
}
|
|
if (ret >= Screens::self()->count())
|
|
ret = screen;
|
|
return ret;
|
|
}
|
|
|
|
CHECK_RULE(Minimize, bool)
|
|
CHECK_RULE(Shade, ShadeMode)
|
|
CHECK_RULE(SkipTaskbar, bool)
|
|
CHECK_RULE(SkipPager, bool)
|
|
CHECK_RULE(SkipSwitcher, bool)
|
|
CHECK_RULE(KeepAbove, bool)
|
|
CHECK_RULE(KeepBelow, bool)
|
|
CHECK_RULE(FullScreen, bool)
|
|
CHECK_RULE(NoBorder, bool)
|
|
CHECK_FORCE_RULE(DecoColor, QString)
|
|
CHECK_FORCE_RULE(BlockCompositing, bool)
|
|
CHECK_FORCE_RULE(FSP, int)
|
|
CHECK_FORCE_RULE(AcceptFocus, bool)
|
|
CHECK_FORCE_RULE(Closeable, bool)
|
|
CHECK_FORCE_RULE(Autogrouping, bool)
|
|
CHECK_FORCE_RULE(AutogroupInForeground, bool)
|
|
CHECK_FORCE_RULE(AutogroupById, QString)
|
|
CHECK_FORCE_RULE(StrictGeometry, bool)
|
|
CHECK_RULE(Shortcut, QString)
|
|
CHECK_FORCE_RULE(DisableGlobalShortcuts, bool)
|
|
|
|
#undef CHECK_RULE
|
|
#undef CHECK_FORCE_RULE
|
|
|
|
// Client
|
|
|
|
void Client::setupWindowRules(bool ignore_temporary)
|
|
{
|
|
client_rules = RuleBook::self()->find(this, ignore_temporary);
|
|
// check only after getting the rules, because there may be a rule forcing window type
|
|
}
|
|
|
|
// Applies Force, ForceTemporarily and ApplyNow rules
|
|
// Used e.g. after the rules have been modified using the kcm.
|
|
void Client::applyWindowRules()
|
|
{
|
|
// apply force rules
|
|
// Placement - does need explicit update, just like some others below
|
|
// Geometry : setGeometry() doesn't check rules
|
|
QRect orig_geom = QRect(pos(), sizeForClientSize(clientSize())); // handle shading
|
|
QRect geom = client_rules.checkGeometry(orig_geom);
|
|
if (geom != orig_geom)
|
|
setGeometry(geom);
|
|
// MinSize, MaxSize handled by Geometry
|
|
// IgnoreGeometry
|
|
setDesktop(desktop());
|
|
workspace()->sendClientToScreen(this, screen());
|
|
setOnActivities(activities());
|
|
// Type
|
|
maximize(maximizeMode());
|
|
// Minimize : functions don't check, and there are two functions
|
|
if (client_rules.checkMinimize(isMinimized()))
|
|
minimize();
|
|
else
|
|
unminimize();
|
|
setShade(shadeMode());
|
|
setOriginalSkipTaskbar(skipTaskbar());
|
|
setSkipPager(skipPager());
|
|
setSkipSwitcher(skipSwitcher());
|
|
setKeepAbove(keepAbove());
|
|
setKeepBelow(keepBelow());
|
|
setFullScreen(isFullScreen(), true);
|
|
setNoBorder(noBorder());
|
|
updateColorScheme();
|
|
// FSP
|
|
// AcceptFocus :
|
|
if (workspace()->mostRecentlyActivatedClient() == this
|
|
&& !client_rules.checkAcceptFocus(true))
|
|
workspace()->activateNextClient(this);
|
|
// Closeable
|
|
QSize s = adjustedSize();
|
|
if (s != size())
|
|
resizeWithChecks(s);
|
|
// Autogrouping : Only checked on window manage
|
|
// AutogroupInForeground : Only checked on window manage
|
|
// AutogroupById : Only checked on window manage
|
|
// StrictGeometry
|
|
setShortcut(rules()->checkShortcut(shortcut().toString()));
|
|
// see also Client::setActive()
|
|
if (isActive()) {
|
|
setOpacity(rules()->checkOpacityActive(qRound(opacity() * 100.0)) / 100.0);
|
|
workspace()->disableGlobalShortcutsForClient(rules()->checkDisableGlobalShortcuts(false));
|
|
} else
|
|
setOpacity(rules()->checkOpacityInactive(qRound(opacity() * 100.0)) / 100.0);
|
|
}
|
|
|
|
void Client::updateWindowRules(Rules::Types selection)
|
|
{
|
|
if (!isManaged()) // not fully setup yet
|
|
return;
|
|
if (RuleBook::self()->areUpdatesDisabled())
|
|
return;
|
|
client_rules.update(this, selection);
|
|
}
|
|
|
|
void Client::finishWindowRules()
|
|
{
|
|
updateWindowRules(Rules::All);
|
|
client_rules = WindowRules();
|
|
}
|
|
|
|
// Workspace
|
|
KWIN_SINGLETON_FACTORY(RuleBook)
|
|
|
|
RuleBook::RuleBook(QObject *parent)
|
|
: QObject(parent)
|
|
, m_updateTimer(new QTimer(this))
|
|
, m_updatesDisabled(false)
|
|
, m_temporaryRulesMessages(new KXMessages(connection(), rootWindow(), "_KDE_NET_WM_TEMPORARY_RULES", nullptr))
|
|
{
|
|
connect(m_temporaryRulesMessages.data(), SIGNAL(gotMessage(QString)), SLOT(temporaryRulesMessage(QString)));
|
|
connect(m_updateTimer, SIGNAL(timeout()), SLOT(save()));
|
|
m_updateTimer->setInterval(1000);
|
|
m_updateTimer->setSingleShot(true);
|
|
}
|
|
|
|
RuleBook::~RuleBook()
|
|
{
|
|
save();
|
|
deleteAll();
|
|
}
|
|
|
|
void RuleBook::deleteAll()
|
|
{
|
|
qDeleteAll(m_rules);
|
|
m_rules.clear();
|
|
}
|
|
|
|
WindowRules RuleBook::find(const Client* c, bool ignore_temporary)
|
|
{
|
|
QVector< Rules* > ret;
|
|
for (QList< Rules* >::Iterator it = m_rules.begin();
|
|
it != m_rules.end();
|
|
) {
|
|
if (ignore_temporary && (*it)->isTemporary()) {
|
|
++it;
|
|
continue;
|
|
}
|
|
if ((*it)->match(c)) {
|
|
Rules* rule = *it;
|
|
qCDebug(KWIN_CORE) << "Rule found:" << rule << ":" << c;
|
|
if (rule->isTemporary())
|
|
it = m_rules.erase(it);
|
|
else
|
|
++it;
|
|
ret.append(rule);
|
|
continue;
|
|
}
|
|
++it;
|
|
}
|
|
return WindowRules(ret);
|
|
}
|
|
|
|
void RuleBook::edit(AbstractClient* c, bool whole_app)
|
|
{
|
|
save();
|
|
QStringList args;
|
|
args << QStringLiteral("--wid") << QString::number(c->window());
|
|
if (whole_app)
|
|
args << QStringLiteral("--whole-app");
|
|
QProcess *p = new QProcess(this);
|
|
p->setArguments(args);
|
|
p->setProcessEnvironment(kwinApp()->processStartupEnvironment());
|
|
p->setProgram(QStringLiteral(KWIN_RULES_DIALOG_BIN));
|
|
connect(p, static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished), this, &QProcess::deleteLater);
|
|
connect(p, static_cast<void (QProcess::*)(QProcess::ProcessError)>(&QProcess::error), this,
|
|
[p] (QProcess::ProcessError e) {
|
|
if (e == QProcess::FailedToStart) {
|
|
qCDebug(KWIN_CORE) << "Failed to start kwin_rules_dialog";
|
|
}
|
|
}
|
|
);
|
|
p->start();
|
|
}
|
|
|
|
void RuleBook::load()
|
|
{
|
|
deleteAll();
|
|
KConfig cfg(QStringLiteral(KWIN_NAME) + QStringLiteral("rulesrc"), KConfig::NoGlobals);
|
|
int count = cfg.group("General").readEntry("count", 0);
|
|
for (int i = 1;
|
|
i <= count;
|
|
++i) {
|
|
KConfigGroup cg(&cfg, QString::number(i));
|
|
Rules* rule = new Rules(cg);
|
|
m_rules.append(rule);
|
|
}
|
|
}
|
|
|
|
void RuleBook::save()
|
|
{
|
|
m_updateTimer->stop();
|
|
KConfig cfg(QStringLiteral(KWIN_NAME) + QStringLiteral("rulesrc"), KConfig::NoGlobals);
|
|
QStringList groups = cfg.groupList();
|
|
for (QStringList::ConstIterator it = groups.constBegin();
|
|
it != groups.constEnd();
|
|
++it)
|
|
cfg.deleteGroup(*it);
|
|
cfg.group("General").writeEntry("count", m_rules.count());
|
|
int i = 1;
|
|
for (QList< Rules* >::ConstIterator it = m_rules.constBegin();
|
|
it != m_rules.constEnd();
|
|
++it) {
|
|
if ((*it)->isTemporary())
|
|
continue;
|
|
KConfigGroup cg(&cfg, QString::number(i));
|
|
(*it)->write(cg);
|
|
++i;
|
|
}
|
|
}
|
|
|
|
void RuleBook::temporaryRulesMessage(const QString& message)
|
|
{
|
|
bool was_temporary = false;
|
|
for (QList< Rules* >::ConstIterator it = m_rules.constBegin();
|
|
it != m_rules.constEnd();
|
|
++it)
|
|
if ((*it)->isTemporary())
|
|
was_temporary = true;
|
|
Rules* rule = new Rules(message, true);
|
|
m_rules.prepend(rule); // highest priority first
|
|
if (!was_temporary)
|
|
QTimer::singleShot(60000, this, SLOT(cleanupTemporaryRules()));
|
|
}
|
|
|
|
void RuleBook::cleanupTemporaryRules()
|
|
{
|
|
bool has_temporary = false;
|
|
for (QList< Rules* >::Iterator it = m_rules.begin();
|
|
it != m_rules.end();
|
|
) {
|
|
if ((*it)->discardTemporary(false)) { // deletes (*it)
|
|
it = m_rules.erase(it);
|
|
} else {
|
|
if ((*it)->isTemporary())
|
|
has_temporary = true;
|
|
++it;
|
|
}
|
|
}
|
|
if (has_temporary)
|
|
QTimer::singleShot(60000, this, SLOT(cleanupTemporaryRules()));
|
|
}
|
|
|
|
void RuleBook::discardUsed(Client* c, bool withdrawn)
|
|
{
|
|
bool updated = false;
|
|
for (QList< Rules* >::Iterator it = m_rules.begin();
|
|
it != m_rules.end();
|
|
) {
|
|
if (c->rules()->contains(*it)) {
|
|
updated = true;
|
|
(*it)->discardUsed(withdrawn);
|
|
if ((*it)->isEmpty()) {
|
|
c->removeRule(*it);
|
|
Rules* r = *it;
|
|
it = m_rules.erase(it);
|
|
delete r;
|
|
continue;
|
|
}
|
|
}
|
|
++it;
|
|
}
|
|
if (updated)
|
|
requestDiskStorage();
|
|
}
|
|
|
|
void RuleBook::requestDiskStorage()
|
|
{
|
|
m_updateTimer->start();
|
|
}
|
|
|
|
void RuleBook::setUpdatesDisabled(bool disable)
|
|
{
|
|
m_updatesDisabled = disable;
|
|
if (!disable) {
|
|
foreach (Client * c, Workspace::self()->clientList())
|
|
c->updateWindowRules(Rules::All);
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
} // namespace
|