/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2008 Martin Gräßlin .
*********************************************************************/
#include "flipswitch.h"
#include
#include
#include
#include
#include
#include
#include
#include
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
#include
#endif
namespace KWin
{
KWIN_EFFECT( flipswitch, FlipSwitchEffect )
FlipSwitchEffect::FlipSwitchEffect()
: mActivated( 0 )
, animateFlip( false )
, forward( true )
, start( false )
, stop( false )
, addFullRepaint( false )
, rearrangeWindows( 0 )
, stopRequested( false )
, startRequested( false )
, progress( 0.0 )
{
KConfigGroup conf = effects->effectConfig( "FlipSwitch" );
mFlipDuration = conf.readEntry( "FlipDuration", 300 );
mAnimation = conf.readEntry( "AnimateFlip", true );
}
FlipSwitchEffect::~FlipSwitchEffect()
{
}
void FlipSwitchEffect::prePaintScreen( ScreenPrePaintData& data, int time )
{
if( mActivated || stopRequested || stop )
{
data.mask |= Effect::PAINT_SCREEN_WITH_TRANSFORMED_WINDOWS;
if( mAnimation )
progress = qMin( 1.0, progress + time / double( mFlipDuration ));
}
effects->prePaintScreen(data, time);
}
void FlipSwitchEffect::paintScreen( int mask, QRegion region, ScreenPaintData& data )
{
effects->paintScreen( mask, region, data );
if( mActivated || stopRequested || stop )
{
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
glMatrixMode( GL_PROJECTION );
glPushMatrix();
glPushAttrib( GL_CURRENT_BIT | GL_ENABLE_BIT );
glEnable( GL_BLEND );
glBlendFunc( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA );
glLoadIdentity();
glFrustum(-QApplication::desktop()->geometry().width()*0.5f,
QApplication::desktop()->geometry().width()*0.5f,
QApplication::desktop()->geometry().height()*0.5f,
-QApplication::desktop()->geometry().height()*0.5f, 10, 50);
glMatrixMode( GL_MODELVIEW );
glLoadIdentity();
float xOffset = QApplication::desktop()->geometry().width()*0.33f;
float zOffset = 10.0;
// bring the selected window to the back of the list
QList< EffectWindow* > tempList = effects->currentTabBoxWindowList();
int index = tempList.indexOf( effects->currentTabBoxWindow() );
QList< EffectWindow* > windowList;
for( int i=index; irearrangeWindows; i-- )
{
EffectWindow* w = windowList.front();
windowList.pop_front();
windowList.append( w );
}
}
else
{
for( int i=0; ix()-QApplication::desktop()->geometry().width()*0.5f;
float y = -QApplication::desktop()->geometry().height()*1.5f+w->y()+w->height();
float z = -10.0;
if( w->isMinimized() )
{
// use icon instead of window
x = w->iconGeometry().x()-QApplication::desktop()->geometry().width()*0.5f;
y = -QApplication::desktop()->geometry().height()*1.5f+w->iconGeometry().y()+w->height();
}
// Position of the window in the stack
float stackX = -QApplication::desktop()->geometry().width()*0.25f-(xOffset*windowList.count())+xOffset*(i+1);
float stackY = -QApplication::desktop()->geometry().height()*0.5f;
float stackZ = (-1*zOffset*windowList.count()) -12.5+zOffset*(i+1);
float animateXOffset;
float animateYOffset;
float animateZOffset;
float rotation;
// if start move to stack, if stop move from stack
if( start )
{
animateXOffset = x+progress*(stackX-x);
animateYOffset = y+progress*(stackY-y);
animateZOffset = z+progress*(stackZ-z);
rotation = progress*0.25;
}
else if( stop )
{
animateXOffset = stackX+progress*(x-stackX);
animateYOffset = stackY+progress*(y-stackY);
animateZOffset = stackZ+progress*(z-stackZ);
rotation = 0.25-progress*0.25;
}
// go to current position and rotate by the time based factor
glTranslatef(animateXOffset, animateYOffset, animateZOffset );
glRotatef(rotation, 0.0, 1.0, 0.0);
// top most window has to be painted not drawn.
if( igeometry().width()*0.25f-(xOffset*windowList.count()),
-QApplication::desktop()->geometry().height()*0.5f,
(-1*zOffset*windowList.count()) -12.5);
if( animateFlip && windowList.count() > 1 )
{
if( progress < 1.0 )
{
float animateXOffset = progress*xOffset;
float animateZOffset = progress*zOffset;
if( forward )
{
if( animateXOffset > xOffset ) animateXOffset = xOffset;
if( animateZOffset > zOffset ) animateZOffset = zOffset;
glTranslatef(animateXOffset, 0.0, animateZOffset);
EffectWindow* w = windowList.front();
windowList.pop_front();
windowList.append( w );
}
else
{
animateXOffset = xOffset - animateXOffset;
animateZOffset = zOffset - animateZOffset;
if( animateXOffset < 0.0 ) animateXOffset = 0.0;
if( animateZOffset < 0.0 ) animateZOffset = 0.0;
glTranslatef(animateXOffset, 0.0, animateZOffset);
}
}
else
{
progress = 0.0;
if( rearrangeWindows != 0 )
{
animateFlip = true;
if( rearrangeWindows < 0 )
{
forward = true;
rearrangeWindows++;
localProgress = 0.0;
}
else
{
forward = false;
rearrangeWindows--;
localProgress = 1.0;
}
}
else
{
animateFlip = false;
if( stopRequested )
{
stop = true;
stopRequested = false;
}
}
}
}
for( int i=0; i 1 )
{
if( forward ) opacity = opacity - localProgress*opacity;
else opacity = localProgress*opacity;
}
paintWindowFlip( windowList[i], false, opacity);
}
glPopMatrix();
}
glPopMatrix();
}
glPopAttrib();
glMatrixMode( GL_PROJECTION );
glPopMatrix();
glMatrixMode( GL_MODELVIEW );
// caption of selected window
QColor color_frame;
QColor color_text;
color_frame = KColorScheme( QPalette::Active, KColorScheme::Window ).background().color();
color_frame.setAlphaF( 0.9 );
color_text = KColorScheme( QPalette::Active, KColorScheme::Window ).foreground().color();
QFont text_font;
text_font.setBold( true );
text_font.setPointSize( 20 );
glPushAttrib( GL_CURRENT_BIT );
glColor4f( color_frame.redF(), color_frame.greenF(), color_frame.blueF(), color_frame.alphaF());
QRect frameRect = QRect( QApplication::desktop()->geometry().width()*0.25f,
QApplication::desktop()->geometry().height()*0.9f,
QApplication::desktop()->geometry().width()*0.5f,
QFontMetrics( text_font ).height() * 1.2 );
renderRoundBoxWithEdge( frameRect );
effects->paintText( effects->currentTabBoxWindow()->caption(),
frameRect.center(),
frameRect.width() - 100,
color_text,
text_font );
glPopAttrib();
// icon of selected window
GLTexture* icon = new GLTexture( effects->currentTabBoxWindow()->icon() );
icon->bind();
glPushAttrib( GL_CURRENT_BIT | GL_ENABLE_BIT );
icon->bind();
glEnable( GL_BLEND );
glBlendFunc( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA );
// icon takes 80 % of the height of the frame. So each 10 % space left on the top and botton
QRect iconRect = QRect( frameRect.x() + frameRect.height()*0.1f,
frameRect.y() + frameRect.height()*0.1f,
frameRect.height()*0.8f,
frameRect.height()*0.8f );
icon->render( region, iconRect);
icon->unbind();
glPopAttrib();
#endif
}
}
void FlipSwitchEffect::postPaintScreen()
{
if( (mActivated && ( animateFlip || start )) || stopRequested || stop )
{
effects->addRepaintFull();
}
if( addFullRepaint )
{
addFullRepaint = false;
effects->setActiveFullScreenEffect( 0 );
effects->addRepaintFull();
}
effects->postPaintScreen();
}
void FlipSwitchEffect::paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data )
{
if( mActivated || stopRequested || stop )
{
if( !( mask & PAINT_WINDOW_TRANSFORMED ) && ( !w->isDesktop() ) )
{
return;
}
}
effects->paintWindow( w, mask, region, data );
}
void FlipSwitchEffect::tabBoxAdded( int mode )
{
if( effects->activeFullScreenEffect() && effects->activeFullScreenEffect() != this )
return;
if( !mActivated )
{
// only for windows mode
if( mode == TabBoxWindowsMode && effects->currentTabBoxWindowList().count() > 0 )
{
effects->refTabBox();
effects->setActiveFullScreenEffect( this );
selectedWindow = effects->currentTabBoxWindowList().indexOf(effects->currentTabBoxWindow());
if( !stop && !stopRequested )
{
mActivated = true;
start = true;
effects->addRepaintFull();
}
else
{
// last tabbox effect still running - schedule start effect
startRequested = true;
}
}
}
}
void FlipSwitchEffect::tabBoxClosed()
{
if( mActivated )
{
// if animation than deactivate after animation
mActivated = false;
effects->unrefTabBox();
if( mAnimation )
{
if ( start || animateFlip ) stopRequested = true;
else
{
stop = true;
effects->addRepaintFull();
}
}
else
effects->setActiveFullScreenEffect( 0 );
}
}
void FlipSwitchEffect::tabBoxUpdated()
{
if( mActivated )
{
if( mAnimation && effects->currentTabBoxWindowList().count() > 1 )
{
// determine the switch direction
int index = effects->currentTabBoxWindowList().indexOf(effects->currentTabBoxWindow());
bool direction = false;
int windowCount = effects->currentTabBoxWindowList().count();
if( index > selectedWindow )
{
if( index == windowCount-1 && selectedWindow == 0 ) forward = false;
else direction = true;
}
else if( index == 0 && ( selectedWindow == windowCount-1 ) )
{
direction = true;
}
else if( index == selectedWindow ) return; // nothing changed
else
{
direction = false;
}
selectedWindow = index;
if( !animateFlip && !start )
{
forward = direction;
animateFlip = true;
}
else
{
if( direction ) rearrangeWindows--;
else rearrangeWindows++;
if( rearrangeWindows >= windowCount ) rearrangeWindows = rearrangeWindows % windowCount;
else if( (-1*rearrangeWindows) >= windowCount ) rearrangeWindows = -1*((-1*rearrangeWindows) % windowCount);
}
}
effects->addRepaintFull();
}
}
void FlipSwitchEffect::paintWindowFlip( EffectWindow* w, bool draw, float opacity )
{
WindowPaintData data( w );
int x = 0;
int y = QApplication::desktop()->geometry().height() - w->geometry().height();
QRect thumbnail;
setPositionTransformations( data,
thumbnail, w,
QRect( x, y, w->geometry().width(), w->geometry().height() ),
Qt::KeepAspectRatio );
data.opacity = opacity;
thumbnail = infiniteRegion();
// if paintWindow() is used the window behind the initial selected window is not painted on the stack,
// but painted when it is selected
// if drawWindow() is used the front window does not glide through the monitor during animation
// so use drawWindow() for all windows but the selected and paintWindow() for the selected window
if( draw )
effects->drawWindow( w,
PAINT_WINDOW_TRANSFORMED,
thumbnail, data );
else
effects->paintWindow( w,
PAINT_WINDOW_TRANSFORMED,
thumbnail, data );
}
} // namespace