kwin/effects/translucency/translucency.cpp
2011-01-31 20:07:03 +01:00

212 lines
7.4 KiB
C++

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2006 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 "translucency.h"
#include <kconfiggroup.h>
namespace KWin
{
KWIN_EFFECT(translucency, TranslucencyEffect)
TranslucencyEffect::TranslucencyEffect()
: fadeout(NULL)
, current(NULL)
, previous(NULL)
{
reconfigure(ReconfigureAll);
active = effects->activeWindow();
}
void TranslucencyEffect::reconfigure(ReconfigureFlags)
{
KConfigGroup conf = effects->effectConfig("Translucency");
decoration = conf.readEntry("Decoration", 1.0);
moveresize = conf.readEntry("MoveResize", 0.8);
dialogs = conf.readEntry("Dialogs", 1.0);
inactive = conf.readEntry("Inactive", 1.0);
comboboxpopups = conf.readEntry("ComboboxPopups", 1.0);
menus = conf.readEntry("Menus", 1.0);
individualmenuconfig = conf.readEntry("IndividualMenuConfig", false);
if (individualmenuconfig) {
dropdownmenus = conf.readEntry("DropdownMenus", 1.0);
popupmenus = conf.readEntry("PopupMenus", 1.0);
tornoffmenus = conf.readEntry("TornOffMenus", 1.0);
} else {
dropdownmenus = menus;
popupmenus = menus;
tornoffmenus = menus;
}
moveresize_timeline.setCurveShape(TimeLine::EaseOutCurve);
moveresize_timeline.setDuration(animationTime(conf, "Duration", 800));
activeinactive_timeline.setCurveShape(TimeLine::EaseInOutCurve);
activeinactive_timeline.setDuration(animationTime(conf, "Duration", 800));
// Repaint the screen just in case the user changed the inactive opacity
effects->addRepaintFull();
}
void TranslucencyEffect::prePaintWindow(EffectWindow* w, WindowPrePaintData& data, int time)
{
moveresize_timeline.addTime(time);
activeinactive_timeline.addTime(time);
if (decoration != 1.0 && w->hasDecoration()) {
data.mask |= PAINT_WINDOW_TRANSLUCENT;
// don't clear PAINT_WINDOW_OPAQUE, contents are not affected
data.clip &= w->contentsRect().translated(w->pos()); // decoration cannot clip
}
if (inactive != 1.0 && isInactive(w))
data.setTranslucent();
if ((moveresize != 1.0 && (w->isUserMove() || w->isUserResize()))
|| (dialogs != 1.0 && w->isDialog())) {
data.setTranslucent();
}
if ((dropdownmenus != 1.0 && w->isDropdownMenu())
|| (popupmenus != 1.0 && w->isPopupMenu())
|| (tornoffmenus != 1.0 && w->isMenu())
|| (comboboxpopups != 1.0 && w->isComboBox())) {
data.setTranslucent();
}
effects->prePaintWindow(w, data, time);
}
void TranslucencyEffect::paintWindow(EffectWindow* w, int mask, QRegion region, WindowPaintData& data)
{
// We keep track of the windows that was last active so we know
// which one to fade out and which ones to paint as fully inactive
if (w == active && w != current) {
previous = current;
current = w;
}
if (w->isDesktop() || w->isDock()) {
effects->paintWindow(w, mask, region, data);
return;
}
// Handling active and inactive windows
if (inactive != 1.0 && isInactive(w)) {
data.opacity *= inactive;
if (w == previous) {
data.opacity *= (inactive + ((1.0 - inactive) * (1.0 - activeinactive_timeline.value())));
if (activeinactive_timeline.value() < 1.0)
w->addRepaintFull();
}
} else {
// Fading in
if (!isInactive(w) && !w->isDesktop()) {
data.opacity *= (inactive + ((1.0 - inactive) * activeinactive_timeline.value()));
if (activeinactive_timeline.value() < 1.0)
w->addRepaintFull();
}
// decoration and dialogs
if (decoration != 1.0 && w->hasDecoration())
data.decoration_opacity *= decoration;
if (dialogs != 1.0 && w->isDialog())
data.opacity *= dialogs;
// Handling moving and resizing
if (moveresize != 1.0 && !w->isDesktop() && !w->isDock()) {
double progress = moveresize_timeline.value();
if (w->isUserMove() || w->isUserResize()) {
// Fading to translucent
data.opacity *= (moveresize + ((1.0 - moveresize) * (1.0 - progress)));
if (progress < 1.0 && progress > 0.0) {
w->addRepaintFull();
if (fadeout != w)
fadeout = w;
}
} else {
// Fading back to more opaque
if (w == fadeout && !w->isUserMove() && !w->isUserResize()) {
data.opacity *= (moveresize + ((1.0 - moveresize) * (progress)));
if (progress == 1.0 || progress == 0.0)
fadeout = NULL;
else
w->addRepaintFull();
}
}
}
// Menus and combos
if (dropdownmenus != 1.0 && w->isDropdownMenu())
data.opacity *= dropdownmenus;
if (popupmenus != 1.0 && w->isPopupMenu())
data.opacity *= popupmenus;
if (tornoffmenus != 1.0 && w->isMenu())
data.opacity *= tornoffmenus;
if (comboboxpopups != 1.0 && w->isComboBox())
data.opacity *= comboboxpopups;
}
effects->paintWindow(w, mask, region, data);
}
bool TranslucencyEffect::isInactive(const EffectWindow* w) const
{
if (active == w || w->isDock() || !w->isManaged())
return false;
if (NULL != active && NULL != active->group())
if (active->group() == w->group())
return false;
if (!w->isNormalWindow() && !w->isDialog() && !w->isDock())
return false;
return true;
}
void TranslucencyEffect::windowUserMovedResized(EffectWindow* w, bool first, bool last)
{
if (moveresize != 1.0 && (first || last)) {
moveresize_timeline.setProgress(0.0);
w->addRepaintFull();
}
}
void TranslucencyEffect::windowActivated(EffectWindow* w)
{
if (inactive != 1.0) {
activeinactive_timeline.setProgress(0.0);
if (NULL != active && active != w) {
if ((NULL == w || w->group() != active->group()) &&
NULL != active->group()) {
// Active group has changed. so repaint old group
foreach (EffectWindow * tmp, active->group()->members())
tmp->addRepaintFull();
} else
active->addRepaintFull();
}
if (NULL != w) {
if (NULL != w->group()) {
// Repaint windows in new group
foreach (EffectWindow * tmp, w->group()->members())
tmp->addRepaintFull();
} else
w->addRepaintFull();
}
}
active = w;
}
} // namespace