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.
 
 
 
 

215 lines
7.4 KiB

/*************************************************************************
* Copyright (C) 2014 by Hugo Pereira Da Costa <hugo.pereira@free.fr> *
* *
* 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, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA . *
*************************************************************************/
#include "breezemdiwindowshadow.h"
#include "breezemdiwindowshadow.moc"
#include "breeze.h"
#include "breezestyleconfigdata.h"
#include <QMdiArea>
#include <QMdiSubWindow>
#include <QPainter>
#include <QTextStream>
namespace Breeze
{
//____________________________________________________________________
void MdiWindowShadow::updateGeometry( void )
{
if( !_widget ) return;
const int size( Metrics::Shadow_Size - Metrics::Shadow_Overlap );
const int topSize( size - Metrics::Shadow_Offset );
const int bottomSize( size );
const int leftSize( size - Metrics::Shadow_Offset );
const int rightSize( size );
// get tileSet rect
_shadowTilesRect = _widget->frameGeometry().adjusted( -leftSize, -topSize, rightSize, bottomSize );
// get parent MDI area's viewport
QWidget *parent( parentWidget() );
if (parent && !qobject_cast<QMdiArea *>(parent) && qobject_cast<QMdiArea*>(parent->parentWidget()))
{ parent = parent->parentWidget(); }
if( qobject_cast<QAbstractScrollArea *>( parent ) )
{ parent = qobject_cast<QAbstractScrollArea *>( parent )->viewport(); }
// set geometry
QRect geometry( _shadowTilesRect );
if( parent ) geometry &= parent->rect();
setGeometry( geometry );
// translate rendering rect
_shadowTilesRect.translate( -geometry.topLeft() );
}
//____________________________________________________________________
void MdiWindowShadow::updateZOrder( void )
{ stackUnder( _widget ); }
//____________________________________________________________________
void MdiWindowShadow::paintEvent( QPaintEvent* event )
{
if( !_shadowTiles.isValid() ) return;
QPainter painter( this );
painter.setRenderHints( QPainter::Antialiasing );
painter.setClipRegion( event->region() );
_shadowTiles.render( _shadowTilesRect, &painter );
}
//____________________________________________________________________
MdiWindowShadowFactory::MdiWindowShadowFactory( QObject* parent ):
QObject( parent )
{}
//____________________________________________________________________________________
bool MdiWindowShadowFactory::registerWidget( QWidget* widget )
{
// check widget type
QMdiSubWindow* subwindow( qobject_cast<QMdiSubWindow*>( widget ) );
if( !subwindow ) return false;
if( subwindow->widget() && subwindow->widget()->inherits( "KMainWindow" ) ) return false;
// make sure widget is not already registered
if( isRegistered( widget ) ) return false;
// store in set
_registeredWidgets.insert( widget );
widget->installEventFilter( this );
// catch object destruction
connect( widget, SIGNAL(destroyed(QObject*)), SLOT(widgetDestroyed(QObject*)) );
return true;
}
//____________________________________________________________________________________
void MdiWindowShadowFactory::unregisterWidget( QWidget* widget )
{
if( !isRegistered( widget ) ) return;
widget->removeEventFilter( this );
_registeredWidgets.remove( widget );
removeShadow( widget );
}
//____________________________________________________________________________________
bool MdiWindowShadowFactory::eventFilter( QObject* object, QEvent* event )
{
switch( event->type() )
{
// TODO: possibly implement ZOrderChange event, to make sure that
// the shadow is always painted on top
case QEvent::ZOrderChange:
updateShadowZOrder( object );
break;
case QEvent::Destroy:
if( isRegistered( object ) )
{
_registeredWidgets.remove( object );
removeShadow( object );
}
break;
case QEvent::Hide:
hideShadows( object );
break;
case QEvent::Show:
installShadow( object );
updateShadowGeometry( object );
updateShadowZOrder( object );
break;
case QEvent::Move:
case QEvent::Resize:
updateShadowGeometry( object );
break;
default: break;
}
return QObject::eventFilter( object, event );
}
//____________________________________________________________________________________
MdiWindowShadow* MdiWindowShadowFactory::findShadow( QObject* object ) const
{
// check object,
if( !object->parent() ) return 0L;
// find existing window shadows
const QList<QObject* > children = object->parent()->children();
foreach( QObject *child, children )
{
if( MdiWindowShadow* shadow = qobject_cast<MdiWindowShadow*>(child) )
{ if( shadow->widget() == object ) return shadow; }
}
return 0L;
}
//____________________________________________________________________________________
void MdiWindowShadowFactory::installShadow( QObject* object )
{
// cast
QWidget* widget( static_cast<QWidget*>( object ) );
if( !widget->parentWidget() ) return;
// make sure shadow is not already installed
if( findShadow( object ) ) return;
// create new shadow
MdiWindowShadow* windowShadow( new MdiWindowShadow( widget->parentWidget(), _shadowTiles ) );
windowShadow->setWidget( widget );
windowShadow->show();
return;
}
//____________________________________________________________________________________
void MdiWindowShadowFactory::removeShadow( QObject* object )
{
if( MdiWindowShadow* windowShadow = findShadow( object ) )
{
windowShadow->hide();
windowShadow->deleteLater();
}
}
//____________________________________________________________________________________
void MdiWindowShadowFactory::widgetDestroyed( QObject* object )
{ _registeredWidgets.remove( object ); }
}