kwin/effects/glide/glide.cpp

243 lines
7.7 KiB
C++
Raw Normal View History

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2007 Philip Falkner <philip.falkner@gmail.com>
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.com>
Copyright (C) 2010 Alexandre Pereira <pereira.alex@gmail.com>
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 "glide.h"
#include <kconfiggroup.h>
#include <QtCore/QTimeLine>
// Effect is based on fade effect by Philip Falkner
namespace KWin
{
2011-01-30 14:34:42 +00:00
KWIN_EFFECT(glide, GlideEffect)
KWIN_EFFECT_SUPPORTED(glide, GlideEffect::supported())
static const int IsGlideWindow = 0x22A982D4;
GlideEffect::GlideEffect()
2011-01-30 14:34:42 +00:00
{
reconfigure(ReconfigureAll);
connect(effects, SIGNAL(windowAdded(EffectWindow*)), this, SLOT(slotWindowAdded(EffectWindow*)));
connect(effects, SIGNAL(windowClosed(EffectWindow*)), this, SLOT(slotWindowClosed(EffectWindow*)));
connect(effects, SIGNAL(windowDeleted(EffectWindow*)), this, SLOT(slotWindowDeleted(EffectWindow*)));
2011-01-30 14:34:42 +00:00
}
bool GlideEffect::supported()
2011-01-30 14:34:42 +00:00
{
return effects->compositingType() == OpenGLCompositing;
2011-01-30 14:34:42 +00:00
}
2011-01-30 14:34:42 +00:00
void GlideEffect::reconfigure(ReconfigureFlags)
{
KConfigGroup conf = effects->effectConfig("Glide");
duration = animationTime(conf, "AnimationTime", 350);
effect = (EffectStyle) conf.readEntry("GlideEffect", 0);
angle = conf.readEntry("GlideAngle", -90);
}
2011-01-30 14:34:42 +00:00
void GlideEffect::prePaintScreen(ScreenPrePaintData& data, int time)
{
if (!windows.isEmpty())
data.mask |= PAINT_SCREEN_WITH_TRANSFORMED_WINDOWS;
2011-01-30 14:34:42 +00:00
effects->prePaintScreen(data, time);
}
2011-01-30 14:34:42 +00:00
void GlideEffect::prePaintWindow(EffectWindow* w, WindowPrePaintData& data, int time)
{
InfoHash::iterator info = windows.find(w);
if (info != windows.end()) {
data.setTransformed();
2011-01-30 14:34:42 +00:00
if (info->added)
info->timeLine->setCurrentTime(info->timeLine->currentTime() + time);
2011-01-30 14:34:42 +00:00
else if (info->closed) {
info->timeLine->setCurrentTime(info->timeLine->currentTime() - time);
2011-01-30 14:34:42 +00:00
if (info->deleted)
w->enablePainting(EffectWindow::PAINT_DISABLED_BY_DELETE);
}
2011-01-30 14:34:42 +00:00
}
effects->prePaintWindow(w, data, time);
// if the window isn't to be painted, then let's make sure
// to track its progress
2011-01-30 14:34:42 +00:00
if (info != windows.end() && !w->isPaintingEnabled() && !effects->activeFullScreenEffect())
w->addRepaintFull();
2011-01-30 14:34:42 +00:00
}
2011-01-30 14:34:42 +00:00
void GlideEffect::paintWindow(EffectWindow* w, int mask, QRegion region, WindowPaintData& data)
{
InfoHash::const_iterator info = windows.constFind(w);
if (info != windows.constEnd()) {
const double progress = info->timeLine->currentValue();
RotationData rot;
rot.axis = RotationData::XAxis;
2011-01-30 14:34:42 +00:00
rot.angle = angle * (1 - progress);
data.rotation = &rot;
data.opacity *= progress;
2011-01-30 14:34:42 +00:00
switch(effect) {
default:
case GlideInOut:
if (info->added)
glideIn(w, data);
else if (info->closed)
glideOut(w, data);
break;
case GlideOutIn:
if (info->added)
glideOut(w, data);
if (info->closed)
glideIn(w, data);
break;
case GlideIn: glideIn(w, data); break;
case GlideOut: glideOut(w, data); break;
}
}
2011-01-30 14:34:42 +00:00
effects->paintWindow(w, mask, region, data);
}
2011-01-30 14:34:42 +00:00
void GlideEffect::glideIn(EffectWindow* w, WindowPaintData& data)
{
InfoHash::const_iterator info = windows.constFind(w);
if (info == windows.constEnd())
return;
const double progress = info->timeLine->currentValue();
data.xScale *= progress;
data.yScale *= progress;
data.zScale *= progress;
2011-01-30 14:34:42 +00:00
data.xTranslate += int(w->width() / 2 * (1 - progress));
data.yTranslate += int(w->height() / 2 * (1 - progress));
}
2011-01-30 14:34:42 +00:00
void GlideEffect::glideOut(EffectWindow* w, WindowPaintData& data)
{
InfoHash::const_iterator info = windows.constFind(w);
if (info == windows.constEnd())
return;
const double progress = info->timeLine->currentValue();
2011-01-30 14:34:42 +00:00
data.xScale *= (2 - progress);
data.yScale *= (2 - progress);
data.zScale *= (2 - progress);
data.xTranslate -= int(w->width() / 2 * (1 - progress));
data.yTranslate -= int(w->height() / 2 * (1 - progress));
}
void GlideEffect::postPaintWindow(EffectWindow* w)
{
InfoHash::iterator info = windows.find(w);
if (info != windows.end()) {
if (info->added && info->timeLine->currentValue() == 1.0) {
2011-01-30 14:34:42 +00:00
windows.remove(w);
effects->addRepaintFull();
} else if (info->closed && info->timeLine->currentValue() == 0.0) {
info->closed = false;
2011-01-30 14:34:42 +00:00
if (info->deleted) {
windows.remove(w);
w->unrefWindow();
}
2011-01-30 14:34:42 +00:00
effects->addRepaintFull();
}
2011-01-30 14:34:42 +00:00
if (info->added || info->closed)
w->addRepaintFull();
}
2011-01-30 14:34:42 +00:00
effects->postPaintWindow(w);
}
void GlideEffect::slotWindowAdded(EffectWindow* w)
2011-01-30 14:34:42 +00:00
{
if (!isGlideWindow(w))
return;
2011-01-30 14:34:42 +00:00
w->setData(IsGlideWindow, true);
const void *addGrab = w->data(WindowAddedGrabRole).value<void*>();
if (addGrab && addGrab != this)
return;
2011-01-30 14:34:42 +00:00
w->setData(WindowAddedGrabRole, QVariant::fromValue(static_cast<void*>(this)));
InfoHash::iterator it = windows.find(w);
WindowInfo *info = (it == windows.end()) ? &windows[w] : &it.value();
info->added = true;
info->closed = false;
info->deleted = false;
delete info->timeLine;
info->timeLine = new QTimeLine(duration);
info->timeLine->setCurveShape(QTimeLine::EaseOutCurve);
w->addRepaintFull();
2011-01-30 14:34:42 +00:00
}
void GlideEffect::slotWindowClosed(EffectWindow* w)
2011-01-30 14:34:42 +00:00
{
if (!isGlideWindow(w))
return;
2011-01-30 14:34:42 +00:00
const void *closeGrab = w->data(WindowClosedGrabRole).value<void*>();
if (closeGrab && closeGrab != this)
return;
w->refWindow();
2011-01-30 14:34:42 +00:00
w->setData(WindowClosedGrabRole, QVariant::fromValue(static_cast<void*>(this)));
InfoHash::iterator it = windows.find(w);
WindowInfo *info = (it == windows.end()) ? &windows[w] : &it.value();
info->added = false;
info->closed = true;
info->deleted = true;
delete info->timeLine;
info->timeLine = new QTimeLine(duration);
info->timeLine->setCurveShape(QTimeLine::EaseInCurve);
info->timeLine->setCurrentTime(info->timeLine->duration());
w->addRepaintFull();
2011-01-30 14:34:42 +00:00
}
void GlideEffect::slotWindowDeleted(EffectWindow* w)
2011-01-30 14:34:42 +00:00
{
windows.remove(w);
}
2011-01-30 14:34:42 +00:00
bool GlideEffect::isGlideWindow(EffectWindow* w)
{
if (effects->activeFullScreenEffect())
return false;
2011-01-30 14:34:42 +00:00
if (w->data(IsGlideWindow).toBool())
return true;
2011-01-30 14:34:42 +00:00
if (w->hasDecoration())
return true;
2011-01-30 14:34:42 +00:00
if (!w->isManaged() || w->isMenu() || w->isNotification() || w->isDesktop() ||
w->isDock() || w->isSplash() || w->isTopMenu() || w->isToolbar() ||
w->windowClass() == "dashboard dashboard")
return false;
return true;
2011-01-30 14:34:42 +00:00
}
GlideEffect::WindowInfo::WindowInfo()
: deleted(false)
, added(false)
, closed(false)
, timeLine(0)
{
}
GlideEffect::WindowInfo::~WindowInfo()
{
delete timeLine;
}
} // namespace