////////////////////////////////////////////////////////////////////////////// // oxygenshadowcache.cpp // handles caching of TileSet objects to draw shadows // ------------------- // // Copyright (c) 2009 Hugo Pereira Da Costa // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS // IN THE SOFTWARE. ////////////////////////////////////////////////////////////////////////////// #include "oxygenshadowcache.h" #include "oxygenclient.h" #include "oxygenhelper.h" #include #include #include #include namespace Oxygen { //_______________________________________________________ ShadowCache::ShadowCache( DecoHelper& helper ): helper_( helper ), activeShadowConfiguration_( ShadowConfiguration( QPalette::Active ) ), inactiveShadowConfiguration_( ShadowConfiguration( QPalette::Inactive ) ) { setEnabled( true ); setMaxIndex( 256 ); } //_______________________________________________________ bool ShadowCache::shadowConfigurationChanged( const ShadowConfiguration& other ) const { const ShadowConfiguration& local = (other.colorGroup() == QPalette::Active ) ? activeShadowConfiguration_:inactiveShadowConfiguration_; return !(local == other); } //_______________________________________________________ void ShadowCache::setShadowConfiguration( const ShadowConfiguration& other ) { ShadowConfiguration& local = (other.colorGroup() == QPalette::Active ) ? activeShadowConfiguration_:inactiveShadowConfiguration_; local = other; } //_______________________________________________________ TileSet* ShadowCache::tileSet( const Client* client ) { // construct key Key key( client ); // check if tileset already in cache int hash( key.hash() ); if( shadowCache_.contains(hash) ) return shadowCache_.object(hash); // create tileset otherwise qreal size( shadowSize() ); TileSet* tileSet = new TileSet( shadowPixmap( client, key.active ), size, size, 1, 1); shadowCache_.insert( hash, tileSet ); return tileSet; } //_______________________________________________________ TileSet* ShadowCache::tileSet( const Client* client, qreal opacity ) { int index( opacity*maxIndex_ ); assert( index <= maxIndex_ ); // construct key Key key( client ); key.index = index; // check if tileset already in cache int hash( key.hash() ); if( animatedShadowCache_.contains(hash) ) return animatedShadowCache_.object(hash); // create shadow and tileset otherwise qreal size( shadowSize() ); QPixmap shadow( size*2, size*2 ); shadow.fill( Qt::transparent ); QPainter p( &shadow ); p.setRenderHint( QPainter::Antialiasing ); QPixmap inactiveShadow( shadowPixmap( client, false ) ); { QPainter pp( &inactiveShadow ); pp.setRenderHint( QPainter::Antialiasing ); pp.setCompositionMode(QPainter::CompositionMode_DestinationIn); pp.fillRect( inactiveShadow.rect(), QColor( 0, 0, 0, 255*(1.0-opacity ) ) ); } QPixmap activeShadow( shadowPixmap( client, true ) ); { QPainter pp( &activeShadow ); pp.setRenderHint( QPainter::Antialiasing ); pp.setCompositionMode(QPainter::CompositionMode_DestinationIn); pp.fillRect( activeShadow.rect(), QColor( 0, 0, 0, 255*( opacity ) ) ); } p.drawPixmap( QPointF(0,0), inactiveShadow ); p.drawPixmap( QPointF(0,0), activeShadow ); p.end(); TileSet* tileSet = new TileSet(shadow, size, size, 1, 1); animatedShadowCache_.insert( hash, tileSet ); return tileSet; } //_________________________________________________________________ QPixmap ShadowCache::shadowPixmap(const Client* client, bool active ) const { // get window color Key key( client ); QPalette palette( client->backgroundPalette( client->widget(), client->widget()->palette() ) ); QColor color( palette.color( client->widget()->backgroundRole() ) ); return simpleShadowPixmap( color, key, active ); } //_______________________________________________________ QPixmap ShadowCache::simpleShadowPixmap( const QColor& color, const Key& key, bool active ) const { // local reference to relevant shadow configuration const ShadowConfiguration& shadowConfiguration( active && key.useOxygenShadows ? activeShadowConfiguration_:inactiveShadowConfiguration_ ); static const qreal fixedSize = 25.5; qreal size( shadowSize() ); qreal shadowSize( shadowConfiguration.isEnabled() ? shadowConfiguration.shadowSize():0 ); QPixmap shadow = QPixmap( size*2, size*2 ); shadow.fill( Qt::transparent ); QPainter p( &shadow ); p.setRenderHint( QPainter::Antialiasing ); p.setPen( Qt::NoPen ); // some gradients rendering are different at bottom corners if client has no border bool hasBorder( key.hasBorder || key.isShade ); if( shadowSize ) { if( active && key.useOxygenShadows ) { { const qreal gradientSize = qMin( shadowSize, (shadowSize+fixedSize)/2 ); const qreal hoffset = shadowConfiguration.horizontalOffset()*gradientSize/fixedSize; const qreal voffset = shadowConfiguration.verticalOffset()*gradientSize/fixedSize; // inner (shark) gradient const int nPoints = 7; const qreal x[7] = {0, 0.05, 0.1, 0.15, 0.2, 0.3, 0.4 }; const qreal values[7] = {0.8, 0.78, 0.69, 0.42, 0.18, 0.01, 0 }; QRadialGradient rg = QRadialGradient( size+12.0*hoffset, size+12.0*voffset, gradientSize ); QColor c = shadowConfiguration.innerColor(); for( int i = 0; iisActive() || client->isForcedActive(); useOxygenShadows = client->configuration().useOxygenShadows(); isShade = client->isShade(); hasTitleOutline = client->configuration().drawTitleOutline(); hasBorder = ( client->configuration().frameBorder() > Configuration::BorderNone ); } }