You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
koffice/kexi/widget/utils/kexigradientwidget.cpp

359 lines
9.5 KiB

/* This file is part of the KDE project
Copyright (C) 2005 Christian Nitschkowski <segfault_ii@web.de>
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU Library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this program; see the file COPYING. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include <tqapplication.h>
#include <tqbitmap.h>
#include <tqimage.h>
#include <tqobjectlist.h>
#include <tqpainter.h>
#include <tqstyle.h>
#include <kimageeffect.h>
#include <kpixmap.h>
#include "kexigradientwidget.h"
KexiGradientWidget::KexiGradientWidget( TQWidget *parent, const char *name, WFlags f )
: TQWidget( parent, name, f ), p_displayMode( NoGradient ),
p_gradientType( VerticalGradient ),
p_color1( TQt::white ), p_color2( TQt::blue ), p_currentChild( 0 ),
p_opacity( 0.5 ), p_cacheDirty( true )
{
p_customBackgroundWidgets.setAutoDelete( false );
p_knownWidgets.setAutoDelete( false );
p_backgroundColor = TQWidget::paletteBackgroundColor();
connect ( &p_rebuildDelayTimer, TQT_SIGNAL( timeout() ), this, TQT_SLOT( setCacheDirty() ) );
installEventFilter( this );
}
KexiGradientWidget::~KexiGradientWidget()
{
}
bool KexiGradientWidget::isValidChildWidget( TQObject* child ) {
const TQWidget* wgt = dynamic_cast<TQWidget*>( child );
if ( wgt == 0L )
return false;
if ( wgt->inherits( "TQScrollView" ) )
return false;
if ( wgt->inherits( "TQComboBox" ) )
return false;
if ( wgt->inherits( "TQLineEdit" ) )
return false;
if ( wgt->inherits( "KexiDBForm" ) )
return false;
return true;
}
void KexiGradientWidget::buildChildrenList( WidgetList& list, TQWidget* p ) {
TQObjectList* objects = p->queryList( "TQWidget", 0, false, false );
for ( TQObjectList::Iterator it = objects->begin(); it != objects->end(); ++it ) {
if ( isValidChildWidget( ( *it ) ) == false )
continue;
list.append( dynamic_cast<TQWidget*>( ( *it ) ) );
buildChildrenList( list, dynamic_cast<TQWidget*>( ( *it ) ) );
}
delete objects;
}
void KexiGradientWidget::rebuildCache( void ) {
WidgetList childWidgetList;
buildChildrenList( childWidgetList, this );
/**
Disable the effect and behave like a normal TQWidget.
*/
if ( p_displayMode == NoGradient ) {
// if ( p_backgroundPixmap.isNull() ) {
//unsetPalette();
//} else {
TQWidget::setPaletteBackgroundPixmap( p_backgroundPixmap );
//}
TQWidget::setPaletteBackgroundColor( p_backgroundColor );
for ( WidgetList::Iterator it = childWidgetList.begin();
it != childWidgetList.end(); ++it ) {
if ( p_customBackgroundWidgets.contains( ( *it ) ) == false ) {
( *it )->unsetPalette();
}
}
/**
The cache is now in a current state.
*/
p_cacheDirty = false;
return;
}
KPixmap tempPixmap;
TQImage gradientImage;
TQImage bgImage;
/**
Draw the gradient
*/
gradientImage = KImageEffect::gradient( size(), p_color1, p_color2,
(KImageEffect::GradientType)p_gradientType );
/**
Draw the widget-background in a pixmap and fade it with the gradient.
*/
if ( p_displayMode == FadedGradient ) {
tempPixmap.resize( size() );
TQPainter p( &tempPixmap, this );
if ( p_backgroundPixmap.isNull() ) {
/*
Need to unset the palette, otherwise the old gradient
will be used as a background, not the widget's default bg.
*/
unsetPalette();
p.fillRect( 0, 0, width(), height(), palette().brush(
isEnabled() ? TQPalette::Active : TQPalette::Disabled,
TQColorGroup::Background ) );
} else {
p.drawTiledPixmap( 0, 0, width(), height(), p_backgroundPixmap );
}
p.end();
bgImage = tempPixmap;
KImageEffect::blend( gradientImage, bgImage, (float)p_opacity );
tempPixmap.convertFromImage( bgImage );
} else if ( p_displayMode == SimpleGradient ) {
/**
Use the gradient as the final background-pixmap
if displaymode is set to SimpleGradient.
*/
tempPixmap.convertFromImage( gradientImage );
}
/**
All children need to have our background set.
*/
KPixmap partPixmap;
TQRect area;
TQWidget* childWidget = 0;
const TQPoint topLeft( 0, 0 );
for ( WidgetList::Iterator it = childWidgetList.begin();
it != childWidgetList.end(); ++it ) {
childWidget = ( *it );
/**
Exclude widgets with a custom palette.
*/
if ( p_customBackgroundWidgets.contains( childWidget ) ) {
continue;
}
partPixmap.resize( childWidget->size() );
/**
Get the part of the tempPixmap that is
under the current child-widget.
*/
if ( TQT_BASE_OBJECT(childWidget->parent()) == TQT_BASE_OBJECT(this) ) {
area = childWidget->geometry();
} else {
area.setTopLeft( childWidget->mapTo( this,
childWidget->clipRegion().boundingRect().topLeft() ) );
area.setSize( childWidget->size() );
}
bitBlt( &partPixmap, topLeft, &tempPixmap, area );
p_currentChild = childWidget;
childWidget->setPaletteBackgroundPixmap( partPixmap );
}
TQWidget::setPaletteBackgroundPixmap( tempPixmap );
/**
Unset the dirty-flag at the end of the method.
TQWidget::setPaletteBackgroundPixmap() causes this
to get set to true again, so set it to false
right after setting the pixmap.
*/
p_cacheDirty = false;
}
void KexiGradientWidget::paintEvent( TQPaintEvent* e ) {
/**
Rebuild the background-pixmap if necessary.
*/
if ( p_cacheDirty == true ) {
rebuildCache();
}
/**
Draw the widget as usual
*/
TQWidget::paintEvent( e );
}
bool KexiGradientWidget::eventFilter( TQObject* object, TQEvent* event ) {
TQWidget* child = dynamic_cast<TQWidget*>( object );
/**
Manage list of child-widgets.
*/
if ( TQT_BASE_OBJECT(object) == TQT_BASE_OBJECT(this) ) {
if ( event->type() == TQEvent::ChildInserted ) {
child = dynamic_cast<TQWidget*>( dynamic_cast<TQChildEvent*>( event )->child() );
if ( isValidChildWidget( TQT_TQOBJECT(child) ) == false ) {
return false;
}
/**
Add the new child-widget to our list of known widgets.
*/
p_knownWidgets.append( child );
/**
... and install 'this' as the child's event-filter.
*/
child->installEventFilter( this );
} else if ( event->type() == TQEvent::ChildRemoved ) {
/**
Remove the child-widget from the list of known widgets.
*/
p_knownWidgets.remove( dynamic_cast<TQWidget*>( dynamic_cast<TQChildEvent*>( event )->child() ) );
}
return false;
}
/**
Manage custombackground-list.
*/
if ( event->type() == TQEvent::PaletteChange ) {
/**
p_currentChild will be == 0L, when the user
sets it's palette manually.
In this case, it has to be added to the customBackground-list.
*/
if ( p_currentChild == 0L && child != 0L ) {
if ( p_customBackgroundWidgets.contains( child ) == false ) {
p_customBackgroundWidgets.append( child );
return false;
}
}
/**
Check if the widget whose PaletteChange-event we handle
isn't the widget we set the background in rebuildCache().
*/
if ( child != p_currentChild && child != 0L ) {
/**
Add the new child to the list of widgets, we don't set
the background ourselves if it isn't in the list.
*/
if ( p_customBackgroundWidgets.contains( child ) == false ) {
if ( child->paletteBackgroundPixmap() != 0L ) {
p_customBackgroundWidgets.append( child );
}
} else {
/**
If the palette is now the default-palette again,
remove it from the "don't set background in rebuildCache()"-list
and rebuild the cache, so it again will get the gradient background.
*/
if ( child->paletteBackgroundPixmap() == 0L ) {
p_customBackgroundWidgets.remove( child );
if ( p_displayMode != NoGradient ) {
p_cacheDirty = true;
}
}
}
}
p_currentChild = 0;
}
if ( event->type() == TQEvent::Move ) {
if ( p_customBackgroundWidgets.contains( child ) == false ) {
updateChildBackground( child );
}
}
return false;
}
void KexiGradientWidget::updateChildBackground( TQWidget* childWidget )
{
KPixmap partPixmap;
KPixmap bgPixmap;
TQRect area;
const TQPoint topLeft( 0, 0 );
bgPixmap = paletteBackgroundPixmap() ? (*paletteBackgroundPixmap()) : TQPixmap();
if ( bgPixmap.isNull() )
return;
/**
Exclude widgtes that don't have a parent.
This happens when children are removed
which are in the knownWidgets-list.
*/
if ( childWidget->parent() == 0L )
return;
/**
Exclude widgets with a custom palette.
*/
if ( p_customBackgroundWidgets.contains( childWidget ) ) {
return;
}
partPixmap.resize( childWidget->size() );
/**
Get the part of the tempPixmap that is
under the current child-widget.
*/
if ( TQT_BASE_OBJECT(childWidget->parent()) == TQT_BASE_OBJECT(this) ) {
area = childWidget->geometry();
} else {
area.setTopLeft( childWidget->mapTo( this,
childWidget->clipRegion().boundingRect().topLeft() ) );
area.setSize( childWidget->size() );
}
bitBlt( &partPixmap, topLeft, &bgPixmap, area );
p_currentChild = childWidget;
childWidget->setPaletteBackgroundPixmap( partPixmap );
}
void KexiGradientWidget::setPaletteBackgroundColor( const TQColor& color )
{
p_backgroundColor = color;
if ( p_displayMode == NoGradient ) {
TQWidget::setPaletteBackgroundColor( p_backgroundColor );
}
}
const TQColor& KexiGradientWidget::paletteBackgroundColor() const
{
return p_backgroundColor;
}
#include "kexigradientwidget.moc"