Cleaned up the logout effect and fixed a whole pile of bugs that I

introduced when adding the blur animation.

svn path=/trunk/KDE/kdebase/workspace/; revision=923329
This commit is contained in:
Lucas Murray 2009-02-08 16:04:02 +00:00
parent 57f57adaa4
commit b11d863033
3 changed files with 67 additions and 60 deletions

View file

@ -3,6 +3,7 @@
This file is part of the KDE project. This file is part of the KDE project.
Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org> Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org>
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.com>
Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com> Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com>
This program is free software; you can redistribute it and/or modify This program is free software; you can redistribute it and/or modify
@ -31,8 +32,9 @@ namespace KWin
KWIN_EFFECT( logout, LogoutEffect ) KWIN_EFFECT( logout, LogoutEffect )
LogoutEffect::LogoutEffect() LogoutEffect::LogoutEffect()
: progress( 0 ) : progress( 0.0 )
, logout_window( NULL ) , logoutWindow( NULL )
, logoutWindowClosed( true )
{ {
char net_wm_cm_name[ 100 ]; char net_wm_cm_name[ 100 ];
sprintf( net_wm_cm_name, "_NET_WM_CM_S%d", DefaultScreen( display())); sprintf( net_wm_cm_name, "_NET_WM_CM_S%d", DefaultScreen( display()));
@ -44,26 +46,18 @@ LogoutEffect::LogoutEffect()
#ifdef KWIN_HAVE_OPENGL_COMPOSITING #ifdef KWIN_HAVE_OPENGL_COMPOSITING
blurSupported = false; blurSupported = false;
if( effects->compositingType() == OpenGLCompositing && GLTexture::NPOTTextureSupported() )
{ // 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 );
// If NPOT textures are not supported use the nearest power-of-two sized blurTarget = new GLRenderTarget( blurTexture );
// texture. It wastes memory, but it's possible to support systems without if( blurTarget->valid() )
// NPOT textures this way. blurSupported = true;
int texw = displayWidth();
int texh = displayHeight();
if( !GLTexture::NPOTTextureSupported() )
{
kWarning( 1212 ) << "NPOT textures not supported, wasting some memory";
texw = nearestPowerOfTwo( texw );
texh = nearestPowerOfTwo( texh );
} }
// Create texture and render target
blurTexture = new GLTexture( texw, texh );
blurTexture->setFilter( GL_LINEAR_MIPMAP_LINEAR );
blurTexture->setWrapMode( GL_CLAMP_TO_EDGE );
blurTarget = new GLRenderTarget( blurTexture );
if( blurTarget->valid() )
blurSupported = true;
#endif #endif
} }
@ -77,10 +71,10 @@ LogoutEffect::~LogoutEffect()
void LogoutEffect::prePaintScreen( ScreenPrePaintData& data, int time ) void LogoutEffect::prePaintScreen( ScreenPrePaintData& data, int time )
{ {
if( logout_window != NULL ) if( logoutWindow != NULL && !logoutWindowClosed )
progress = qBound( 0., progress + time / animationTime( 2000. ), 1. ); progress = qMin( 1.0, progress + time / animationTime( 2000.0 ));
else if( progress != 0 ) else if( progress > 0.0 )
progress = qBound( 0., progress - time / animationTime( 500. ), 1. ); progress = qMax( 0.0, progress - time / animationTime( 500.0 ));
#ifdef KWIN_HAVE_OPENGL_COMPOSITING #ifdef KWIN_HAVE_OPENGL_COMPOSITING
if( blurSupported && progress > 0.0 ) if( blurSupported && progress > 0.0 )
@ -93,37 +87,32 @@ void LogoutEffect::prePaintScreen( ScreenPrePaintData& data, int time )
effects->prePaintScreen( data, time ); effects->prePaintScreen( data, time );
} }
void LogoutEffect::prePaintWindow( EffectWindow* w, WindowPrePaintData& data, int time )
{
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
if( blurSupported && progress > 0.0 && w == logout_window )
w->disablePainting( EffectWindow::PAINT_DISABLED );
#endif
effects->prePaintWindow( w, data, time );
}
void LogoutEffect::paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data ) void LogoutEffect::paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data )
{ {
if( w != logout_window && progress != 0 ) if( progress > 0.0 )
{ {
if( effects->saturationSupported() ) if( w == logoutWindow )
{ {
data.saturation *= ( 1 - progress * 0.8 ); windowOpacity = data.opacity;
data.brightness *= ( 1 - progress * 0.3 ); data.opacity = 0.0; // Cheat, we need the opacity for later but don't want to blur it
} }
else // When saturation isn't supported then reduce brightness a bit more else
{ {
data.brightness *= ( 1 - progress * 0.6 ); if( effects->saturationSupported() )
{
data.saturation *= ( 1.0 - progress * 0.8 );
data.brightness *= ( 1.0 - progress * 0.3 );
}
else // When saturation isn't supported then reduce brightness a bit more
data.brightness *= ( 1.0 - progress * 0.6 );
} }
} }
effects->paintWindow( w, mask, region, data ); effects->paintWindow( w, mask, region, data );
} }
void LogoutEffect::postPaintScreen() void LogoutEffect::paintScreen( int mask, QRegion region, ScreenPaintData& data )
{ {
if( progress != 0 && progress != 1 ) effects->paintScreen( mask, region, data );
effects->addRepaintFull();
effects->postPaintScreen();
#ifdef KWIN_HAVE_OPENGL_COMPOSITING #ifdef KWIN_HAVE_OPENGL_COMPOSITING
if( blurSupported && progress > 0.0 ) if( blurSupported && progress > 0.0 )
@ -135,7 +124,7 @@ void LogoutEffect::postPaintScreen()
GLfloat bias[1]; GLfloat bias[1];
glGetTexEnvfv( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, bias ); glGetTexEnvfv( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, bias );
glTexEnvf( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, progress * 2.75 ); glTexEnvf( GL_TEXTURE_FILTER_CONTROL, GL_TEXTURE_LOD_BIAS, progress * 2.75 );
glBegin(GL_QUADS); glBegin( GL_QUADS );
glTexCoord2f( 0.0, 0.0 ); glTexCoord2f( 0.0, 0.0 );
glVertex2f( 0.0, displayHeight() ); glVertex2f( 0.0, displayHeight() );
glTexCoord2f( 1.0, 0.0 ); glTexCoord2f( 1.0, 0.0 );
@ -149,36 +138,50 @@ void LogoutEffect::postPaintScreen()
blurTexture->unbind(); blurTexture->unbind();
// Render the logout window // Render the logout window
if( logout_window ) if( logoutWindow )
{ {
int mask = PAINT_WINDOW_OPAQUE; // TODO: Can we have a translucent logout window? int winMask = logoutWindow->hasAlpha() ? PAINT_WINDOW_TRANSLUCENT : PAINT_WINDOW_OPAQUE;
QRect region = infiniteRegion(); WindowPaintData winData( logoutWindow );
WindowPaintData data( logout_window ); winData.opacity = windowOpacity;
effects->drawWindow( logout_window, mask, region, data ); effects->drawWindow( logoutWindow, winMask, region, winData );
} }
} }
#endif #endif
} }
void LogoutEffect::postPaintScreen()
{
if( progress != 0.0 && progress != 1.0 )
effects->addRepaintFull();
effects->postPaintScreen();
}
void LogoutEffect::windowAdded( EffectWindow* w ) void LogoutEffect::windowAdded( EffectWindow* w )
{ {
if( isLogoutDialog( w )) if( isLogoutDialog( w ))
{ {
logout_window = w; logoutWindow = w;
progress = 0; logoutWindowClosed = false; // So we don't blur the window on close
progress = 0.0;
effects->addRepaintFull(); effects->addRepaintFull();
} }
} }
void LogoutEffect::windowClosed( EffectWindow* w ) void LogoutEffect::windowClosed( EffectWindow* w )
{ {
if( w == logout_window ) if( w == logoutWindow )
{ {
logout_window = NULL; logoutWindowClosed = true;
effects->addRepaintFull(); effects->addRepaintFull();
} }
} }
void LogoutEffect::windowDeleted( EffectWindow* w )
{
if( w == logoutWindow )
logoutWindow = NULL;
}
bool LogoutEffect::isLogoutDialog( EffectWindow* w ) bool LogoutEffect::isLogoutDialog( EffectWindow* w )
{ // TODO there should be probably a better way (window type?) { // TODO there should be probably a better way (window type?)
if( w->windowClass() == "ksmserver ksmserver" if( w->windowClass() == "ksmserver ksmserver"

View file

@ -122,13 +122,13 @@ Comment[zh_TW]=顯示登出視窗時將桌面淡化
Type=Service Type=Service
X-KDE-ServiceTypes=KWin/Effect X-KDE-ServiceTypes=KWin/Effect
X-KDE-PluginInfo-Author=Lubos Lunak X-KDE-PluginInfo-Author=Lubos Lunak & Lucas Murray
X-KDE-PluginInfo-Email=l.lunak@kde.org X-KDE-PluginInfo-Email=l.lunak@kde.org & lmurray@undefinedfire.com
X-KDE-PluginInfo-Name=kwin4_effect_logout X-KDE-PluginInfo-Name=kwin4_effect_logout
X-KDE-PluginInfo-Version=0.1.0 X-KDE-PluginInfo-Version=0.2.0
X-KDE-PluginInfo-Category=Appearance X-KDE-PluginInfo-Category=Appearance
X-KDE-PluginInfo-Depends= X-KDE-PluginInfo-Depends=
X-KDE-PluginInfo-License=GPL X-KDE-PluginInfo-License=GPL
X-KDE-PluginInfo-EnabledByDefault=true X-KDE-PluginInfo-EnabledByDefault=true
X-KDE-Library=kwin4_effect_builtins X-KDE-Library=kwin4_effect_builtins
X-KDE-Ordering=40 X-KDE-Ordering=85

View file

@ -3,6 +3,7 @@
This file is part of the KDE project. This file is part of the KDE project.
Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org> Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org>
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.com>
Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com> Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com>
This program is free software; you can redistribute it and/or modify This program is free software; you can redistribute it and/or modify
@ -38,20 +39,23 @@ class LogoutEffect
LogoutEffect(); LogoutEffect();
~LogoutEffect(); ~LogoutEffect();
virtual void prePaintScreen( ScreenPrePaintData& data, int time ); virtual void prePaintScreen( ScreenPrePaintData& data, int time );
virtual void paintScreen( int mask, QRegion region, ScreenPaintData& data );
virtual void postPaintScreen(); virtual void postPaintScreen();
virtual void prePaintWindow( EffectWindow* w, WindowPrePaintData& data, int time );
virtual void paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data ); virtual void paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data );
virtual void windowAdded( EffectWindow* w ); virtual void windowAdded( EffectWindow* w );
virtual void windowClosed( EffectWindow* w ); virtual void windowClosed( EffectWindow* w );
virtual void windowDeleted( EffectWindow* w );
private: private:
bool isLogoutDialog( EffectWindow* w ); bool isLogoutDialog( EffectWindow* w );
double progress; // 0-1 double progress; // 0-1
EffectWindow* logout_window; EffectWindow* logoutWindow;
bool logoutWindowClosed;
#ifdef KWIN_HAVE_OPENGL_COMPOSITING #ifdef KWIN_HAVE_OPENGL_COMPOSITING
bool blurSupported; bool blurSupported;
GLTexture* blurTexture; GLTexture* blurTexture;
GLRenderTarget* blurTarget; GLRenderTarget* blurTarget;
double windowOpacity;
#endif #endif
}; };