2007-11-27 19:40:25 +00:00
|
|
|
/********************************************************************
|
2007-11-14 00:09:14 +00:00
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org>
|
2009-02-08 16:04:02 +00:00
|
|
|
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.com>
|
2009-02-08 14:07:00 +00:00
|
|
|
Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com>
|
2007-11-14 00:09:14 +00:00
|
|
|
|
2007-11-27 19:40:25 +00:00
|
|
|
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/>.
|
|
|
|
*********************************************************************/
|
2007-11-14 00:09:14 +00:00
|
|
|
|
|
|
|
#include "logout.h"
|
|
|
|
|
2007-11-14 13:48:16 +00:00
|
|
|
#include "kwinglutils.h"
|
|
|
|
|
2009-11-03 09:47:33 +00:00
|
|
|
#include <math.h>
|
2009-03-06 06:10:00 +00:00
|
|
|
#include <kconfiggroup.h>
|
2007-11-14 00:09:14 +00:00
|
|
|
#include <kdebug.h>
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
KWIN_EFFECT( logout, LogoutEffect )
|
|
|
|
|
|
|
|
LogoutEffect::LogoutEffect()
|
2009-02-08 16:04:02 +00:00
|
|
|
: progress( 0.0 )
|
|
|
|
, logoutWindow( NULL )
|
|
|
|
, logoutWindowClosed( true )
|
2009-05-11 12:14:41 +00:00
|
|
|
, logoutWindowPassed( false )
|
2007-11-14 00:09:14 +00:00
|
|
|
{
|
2008-01-26 19:36:04 +00:00
|
|
|
char net_wm_cm_name[ 100 ];
|
|
|
|
sprintf( net_wm_cm_name, "_NET_WM_CM_S%d", DefaultScreen( display()));
|
|
|
|
Atom net_wm_cm = XInternAtom( display(), net_wm_cm_name, False );
|
|
|
|
Window sel = XGetSelectionOwner( display(), net_wm_cm );
|
|
|
|
Atom hack = XInternAtom( display(), "_KWIN_LOGOUT_EFFECT", False );
|
|
|
|
XChangeProperty( display(), sel, hack, hack, 8, PropModeReplace, (unsigned char*)&hack, 1 );
|
|
|
|
// the atom is not removed when effect is destroyed, this is temporary anyway
|
2009-07-27 12:54:02 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
blurTexture = NULL;
|
|
|
|
blurTarget = NULL;
|
|
|
|
#endif
|
2009-03-06 06:10:00 +00:00
|
|
|
reconfigure( ReconfigureAll );
|
|
|
|
}
|
|
|
|
|
|
|
|
LogoutEffect::~LogoutEffect()
|
|
|
|
{
|
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
delete blurTexture;
|
|
|
|
delete blurTarget;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogoutEffect::reconfigure( ReconfigureFlags )
|
|
|
|
{
|
2009-08-09 16:04:39 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
2009-10-03 07:06:51 +00:00
|
|
|
frameDelay = 0;
|
2009-03-06 06:10:00 +00:00
|
|
|
KConfigGroup conf = effects->effectConfig( "Logout" );
|
2009-07-27 12:54:02 +00:00
|
|
|
useBlur = conf.readEntry( "UseBlur", true );
|
|
|
|
delete blurTexture;
|
2009-02-15 10:34:41 +00:00
|
|
|
blurTexture = NULL;
|
2009-07-27 12:54:02 +00:00
|
|
|
delete blurTarget;
|
2009-02-15 10:34:41 +00:00
|
|
|
blurTarget = NULL;
|
2009-07-27 12:54:02 +00:00
|
|
|
blurSupported = false;
|
2009-02-08 14:07:00 +00:00
|
|
|
#endif
|
2007-11-14 00:09:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LogoutEffect::prePaintScreen( ScreenPrePaintData& data, int time )
|
|
|
|
{
|
2009-07-27 12:54:02 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
2009-11-03 09:47:33 +00:00
|
|
|
if( !logoutWindow && progress == 0.0 )
|
2009-07-27 12:54:02 +00:00
|
|
|
{
|
2009-11-03 09:47:33 +00:00
|
|
|
if( blurTexture )
|
2009-07-27 12:54:02 +00:00
|
|
|
{
|
|
|
|
delete blurTexture;
|
|
|
|
blurTexture = NULL;
|
|
|
|
delete blurTarget;
|
|
|
|
blurTarget = NULL;
|
|
|
|
blurSupported = false;
|
|
|
|
}
|
|
|
|
}
|
2009-11-03 09:47:33 +00:00
|
|
|
else if( !blurTexture )
|
2009-07-27 12:54:02 +00:00
|
|
|
{
|
|
|
|
blurSupported = false;
|
|
|
|
delete blurTarget; // catch as we just tested the texture ;-P
|
|
|
|
if( effects->compositingType() == OpenGLCompositing && GLTexture::NPOTTextureSupported() && useBlur )
|
|
|
|
{ // TODO: It seems that it is not possible to create a GLRenderTarget that has
|
|
|
|
// a different size than the display right now. Most likely a KWin core bug.
|
|
|
|
// Create texture and render target
|
|
|
|
blurTexture = new GLTexture( displayWidth(), displayHeight() );
|
|
|
|
blurTexture->setFilter( GL_LINEAR_MIPMAP_LINEAR );
|
|
|
|
blurTexture->setWrapMode( GL_CLAMP_TO_EDGE );
|
|
|
|
|
|
|
|
blurTarget = new GLRenderTarget( blurTexture );
|
|
|
|
if( blurTarget->valid() )
|
|
|
|
blurSupported = true;
|
2009-10-03 07:06:51 +00:00
|
|
|
|
|
|
|
// As creating the render target takes time it can cause the first two frames of the
|
|
|
|
// blur animation to be jerky. For this reason we only start the animation after the
|
|
|
|
// third frame.
|
|
|
|
frameDelay = 2;
|
2009-07-27 12:54:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-03 07:06:51 +00:00
|
|
|
if( frameDelay )
|
|
|
|
--frameDelay;
|
|
|
|
else
|
2009-10-07 15:10:15 +00:00
|
|
|
#endif
|
2009-10-03 07:06:51 +00:00
|
|
|
{
|
|
|
|
if( logoutWindow != NULL && !logoutWindowClosed )
|
|
|
|
progress = qMin( 1.0, progress + time / animationTime( 2000.0 ));
|
|
|
|
else if( progress > 0.0 )
|
|
|
|
progress = qMax( 0.0, progress - time / animationTime( 500.0 ));
|
|
|
|
}
|
2009-02-08 14:07:00 +00:00
|
|
|
|
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
if( blurSupported && progress > 0.0 )
|
|
|
|
{
|
|
|
|
data.mask |= PAINT_SCREEN_WITH_TRANSFORMED_WINDOWS;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-11-14 00:09:14 +00:00
|
|
|
effects->prePaintScreen( data, time );
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogoutEffect::paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data )
|
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
if( progress > 0.0 )
|
2007-11-14 00:09:14 +00:00
|
|
|
{
|
2009-02-15 16:04:29 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
if( blurSupported && w == logoutWindow )
|
2007-11-14 13:48:16 +00:00
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
windowOpacity = data.opacity;
|
|
|
|
data.opacity = 0.0; // Cheat, we need the opacity for later but don't want to blur it
|
2007-11-14 13:48:16 +00:00
|
|
|
}
|
2009-02-08 16:04:02 +00:00
|
|
|
else
|
|
|
|
{
|
2009-05-11 12:14:41 +00:00
|
|
|
if( w != logoutWindow && !logoutWindowPassed )
|
2009-02-08 16:04:02 +00:00
|
|
|
{
|
2009-11-03 09:47:33 +00:00
|
|
|
if( blurSupported )
|
|
|
|
data.saturation *= ( 1.0 - progress * 0.2 );
|
|
|
|
else
|
2009-02-15 16:12:43 +00:00
|
|
|
{
|
|
|
|
data.saturation *= ( 1.0 - progress * 0.8 );
|
|
|
|
data.brightness *= ( 1.0 - progress * 0.3 );
|
|
|
|
}
|
2009-02-08 16:04:02 +00:00
|
|
|
}
|
2009-05-11 12:14:41 +00:00
|
|
|
if( blurSupported && logoutWindowPassed )
|
|
|
|
{
|
|
|
|
windows.append( w );
|
|
|
|
windowsOpacities[ w ] = data.opacity;
|
|
|
|
data.opacity = 0.0;
|
|
|
|
}
|
2007-11-14 13:48:16 +00:00
|
|
|
}
|
2009-11-03 09:47:33 +00:00
|
|
|
#else
|
|
|
|
if( w != logoutWindow && !logoutWindowPassed )
|
|
|
|
{
|
|
|
|
data.saturation *= ( 1.0 - progress * 0.8 );
|
|
|
|
data.brightness *= ( 1.0 - progress * 0.3 );
|
|
|
|
}
|
2009-02-15 16:04:29 +00:00
|
|
|
#endif
|
2009-05-11 12:14:41 +00:00
|
|
|
if( w == logoutWindow )
|
|
|
|
{
|
|
|
|
// logout window - all following windows are on top and should not be altered
|
|
|
|
logoutWindowPassed = true;
|
|
|
|
}
|
2007-11-14 00:09:14 +00:00
|
|
|
}
|
|
|
|
effects->paintWindow( w, mask, region, data );
|
|
|
|
}
|
|
|
|
|
2009-02-08 16:04:02 +00:00
|
|
|
void LogoutEffect::paintScreen( int mask, QRegion region, ScreenPaintData& data )
|
2007-11-14 00:09:14 +00:00
|
|
|
{
|
2009-05-28 11:37:02 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
if( blurSupported && progress > 0.0 )
|
|
|
|
{
|
|
|
|
effects->pushRenderTarget( blurTarget );
|
|
|
|
}
|
|
|
|
#endif
|
2009-02-08 16:04:02 +00:00
|
|
|
effects->paintScreen( mask, region, data );
|
2009-02-08 14:07:00 +00:00
|
|
|
|
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
|
|
|
if( blurSupported && progress > 0.0 )
|
|
|
|
{
|
2009-04-28 15:00:56 +00:00
|
|
|
GLRenderTarget* target = effects->popRenderTarget();
|
|
|
|
assert( target == blurTarget );
|
2009-10-05 09:47:13 +00:00
|
|
|
Q_UNUSED( target );
|
2009-02-08 14:07:00 +00:00
|
|
|
|
2009-11-03 09:47:33 +00:00
|
|
|
//--------------------------
|
|
|
|
// Render the screen effect
|
|
|
|
|
|
|
|
glPushAttrib( GL_CURRENT_BIT | GL_ENABLE_BIT | GL_TEXTURE_BIT );
|
|
|
|
|
|
|
|
// Unmodified base image
|
2009-02-08 14:07:00 +00:00
|
|
|
blurTexture->bind();
|
2009-11-03 09:47:33 +00:00
|
|
|
glBegin( GL_QUADS );
|
|
|
|
glTexCoord2f( 0.0, 0.0 );
|
|
|
|
glVertex2f( 0.0, displayHeight() );
|
|
|
|
glTexCoord2f( 1.0, 0.0 );
|
|
|
|
glVertex2f( displayWidth(), displayHeight() );
|
|
|
|
glTexCoord2f( 1.0, 1.0 );
|
|
|
|
glVertex2f( displayWidth(), 0.0 );
|
|
|
|
glTexCoord2f( 0.0, 1.0 );
|
|
|
|
glVertex2f( 0.0, 0.0 );
|
|
|
|
glEnd();
|
|
|
|
|
|
|
|
// Blurred image
|
2009-02-08 14:07:00 +00:00
|
|
|
GLfloat bias[1];
|
|
|
|
glGetTexEnvfv( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, bias );
|
2009-11-03 09:47:33 +00:00
|
|
|
glTexEnvf( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, 1.75 );
|
|
|
|
glTexEnvi( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE );
|
|
|
|
glEnable( GL_BLEND );
|
|
|
|
glBlendFunc( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA );
|
|
|
|
glColor4f( 1.0f, 1.0f, 1.0f, progress * 0.4 );
|
2009-02-08 16:04:02 +00:00
|
|
|
glBegin( GL_QUADS );
|
2009-02-08 14:07:00 +00:00
|
|
|
glTexCoord2f( 0.0, 0.0 );
|
|
|
|
glVertex2f( 0.0, displayHeight() );
|
|
|
|
glTexCoord2f( 1.0, 0.0 );
|
|
|
|
glVertex2f( displayWidth(), displayHeight() );
|
|
|
|
glTexCoord2f( 1.0, 1.0 );
|
|
|
|
glVertex2f( displayWidth(), 0.0 );
|
|
|
|
glTexCoord2f( 0.0, 1.0 );
|
|
|
|
glVertex2f( 0.0, 0.0 );
|
|
|
|
glEnd();
|
|
|
|
glTexEnvf( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, bias[0] );
|
|
|
|
blurTexture->unbind();
|
|
|
|
|
2009-11-03 09:47:33 +00:00
|
|
|
// Vignetting (Radial gradient with transparent middle and black edges)
|
|
|
|
for( int screen = 0; screen < effects->numScreens(); screen++ )
|
|
|
|
{ // TODO: Cache
|
|
|
|
QRect screenGeom = effects->clientArea( ScreenArea, screen, 0 );
|
|
|
|
glScissor( screenGeom.x(), displayHeight() - screenGeom.y() - screenGeom.height(),
|
|
|
|
screenGeom.width(), screenGeom.height() ); // GL coords are flipped
|
|
|
|
glEnable( GL_SCISSOR_TEST ); // Geom must be set before enable
|
|
|
|
float ro = float(( screenGeom.width() > screenGeom.height() )
|
|
|
|
? screenGeom.width() : screenGeom.height() ) * 0.8f; // Outer radius
|
|
|
|
glBegin( GL_TRIANGLES );
|
|
|
|
const float a = M_PI / 8.0f; // Angle of increment
|
|
|
|
for( float i = 0.0f; i < M_PI * 1.99f; i += a )
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
|
|
|
|
glColor4f( 0.0f, 0.0f, 0.0f, 0.0f );
|
|
|
|
|
|
|
|
x = screenGeom.x() + screenGeom.width() / 2;
|
|
|
|
y = screenGeom.y() + screenGeom.height() / 2;
|
|
|
|
glVertex3f( x, y, 0 );
|
|
|
|
|
|
|
|
glColor4f( 0.0f, 0.0f, 0.0f, progress * 0.9f );
|
|
|
|
|
|
|
|
x = ro * cos( i ) + screenGeom.x() + screenGeom.width() / 2;
|
|
|
|
y = ro * sin( i ) + screenGeom.y() + screenGeom.height() / 2;
|
|
|
|
glVertex3f( x, y, 0 );
|
|
|
|
|
|
|
|
x = ro * cos( i + a ) + screenGeom.x() + screenGeom.width() / 2;
|
|
|
|
y = ro * sin( i + a ) + screenGeom.y() + screenGeom.height() / 2;
|
|
|
|
glVertex3f( x, y, 0 );
|
|
|
|
}
|
|
|
|
glEnd();
|
|
|
|
glDisable( GL_SCISSOR_TEST );
|
|
|
|
}
|
|
|
|
|
|
|
|
glPopAttrib();
|
|
|
|
|
|
|
|
//--------------------------
|
|
|
|
|
2009-02-08 14:07:00 +00:00
|
|
|
// Render the logout window
|
2009-02-08 16:04:02 +00:00
|
|
|
if( logoutWindow )
|
2009-02-08 14:07:00 +00:00
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
int winMask = logoutWindow->hasAlpha() ? PAINT_WINDOW_TRANSLUCENT : PAINT_WINDOW_OPAQUE;
|
|
|
|
WindowPaintData winData( logoutWindow );
|
|
|
|
winData.opacity = windowOpacity;
|
|
|
|
effects->drawWindow( logoutWindow, winMask, region, winData );
|
2009-02-08 14:07:00 +00:00
|
|
|
}
|
2009-05-11 12:14:41 +00:00
|
|
|
|
|
|
|
// Render all windows on top of logout window
|
|
|
|
foreach( EffectWindow* w, windows )
|
|
|
|
{
|
|
|
|
int winMask = w->hasAlpha() ? PAINT_WINDOW_TRANSLUCENT : PAINT_WINDOW_OPAQUE;
|
|
|
|
WindowPaintData winData( w );
|
|
|
|
winData.opacity = windowsOpacities[ w ];
|
|
|
|
effects->drawWindow( w, winMask, region, winData );
|
|
|
|
}
|
|
|
|
|
|
|
|
windows.clear();
|
|
|
|
windowsOpacities.clear();
|
2009-02-08 14:07:00 +00:00
|
|
|
}
|
|
|
|
#endif
|
2007-11-14 00:09:14 +00:00
|
|
|
}
|
|
|
|
|
2009-02-08 16:04:02 +00:00
|
|
|
void LogoutEffect::postPaintScreen()
|
|
|
|
{
|
2009-10-07 15:10:15 +00:00
|
|
|
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
|
2009-10-03 07:06:51 +00:00
|
|
|
if(( progress != 0.0 && progress != 1.0 ) || frameDelay )
|
2009-02-08 16:04:02 +00:00
|
|
|
effects->addRepaintFull();
|
2009-10-07 15:10:15 +00:00
|
|
|
#else
|
|
|
|
if( progress != 0.0 && progress != 1.0 )
|
|
|
|
effects->addRepaintFull();
|
|
|
|
#endif
|
|
|
|
|
2009-05-25 22:00:30 +00:00
|
|
|
if( progress > 0.0 )
|
2009-05-11 12:14:41 +00:00
|
|
|
logoutWindowPassed = false;
|
2009-02-08 16:04:02 +00:00
|
|
|
effects->postPaintScreen();
|
|
|
|
}
|
|
|
|
|
2007-11-14 00:09:14 +00:00
|
|
|
void LogoutEffect::windowAdded( EffectWindow* w )
|
|
|
|
{
|
|
|
|
if( isLogoutDialog( w ))
|
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
logoutWindow = w;
|
|
|
|
logoutWindowClosed = false; // So we don't blur the window on close
|
|
|
|
progress = 0.0;
|
2007-11-14 00:09:14 +00:00
|
|
|
effects->addRepaintFull();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogoutEffect::windowClosed( EffectWindow* w )
|
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
if( w == logoutWindow )
|
2007-11-14 00:09:14 +00:00
|
|
|
{
|
2009-02-08 16:04:02 +00:00
|
|
|
logoutWindowClosed = true;
|
2007-11-14 00:09:14 +00:00
|
|
|
effects->addRepaintFull();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-08 16:04:02 +00:00
|
|
|
void LogoutEffect::windowDeleted( EffectWindow* w )
|
|
|
|
{
|
|
|
|
if( w == logoutWindow )
|
|
|
|
logoutWindow = NULL;
|
|
|
|
}
|
|
|
|
|
2007-11-14 00:09:14 +00:00
|
|
|
bool LogoutEffect::isLogoutDialog( EffectWindow* w )
|
|
|
|
{ // TODO there should be probably a better way (window type?)
|
2007-11-14 15:58:57 +00:00
|
|
|
if( w->windowClass() == "ksmserver ksmserver"
|
|
|
|
&& ( w->windowRole() == "logoutdialog" || w->windowRole() == "logouteffect" ))
|
|
|
|
{
|
2007-11-14 00:09:14 +00:00
|
|
|
return true;
|
2007-11-14 15:58:57 +00:00
|
|
|
}
|
2007-11-14 00:09:14 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|