kwin/effects/maketransparent.cpp

233 lines
7.8 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 "maketransparent.h"
#include <kconfiggroup.h>
namespace KWin
{
KWIN_EFFECT( maketransparent, MakeTransparentEffect )
MakeTransparentEffect::MakeTransparentEffect()
: fadeout( NULL )
, current( NULL )
, previous( NULL )
{
reconfigure( ReconfigureAll );
active = effects->activeWindow();
}
void MakeTransparentEffect::reconfigure( ReconfigureFlags )
{
KConfigGroup conf = effects->effectConfig("MakeTransparent");
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 ) );
}
void MakeTransparentEffect::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(( 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 MakeTransparentEffect::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 MakeTransparentEffect::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 MakeTransparentEffect::windowUserMovedResized( EffectWindow* w, bool first, bool last )
{
if( moveresize != 1.0 && ( first || last ))
{
moveresize_timeline.setProgress(0.0);
w->addRepaintFull();
}
}
void MakeTransparentEffect::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