/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include ImplWallpaper::ImplWallpaper() : maColor( COL_TRANSPARENT ) { mnRefCount = 1; mpBitmap = NULL; mpCache = NULL; mpGradient = NULL; mpRect = NULL; meStyle = WALLPAPER_NULL; } ImplWallpaper::ImplWallpaper( const ImplWallpaper& rImplWallpaper ) : maColor( rImplWallpaper.maColor ) { mnRefCount = 1; meStyle = rImplWallpaper.meStyle; if ( rImplWallpaper.mpBitmap ) mpBitmap = new BitmapEx( *rImplWallpaper.mpBitmap ); else mpBitmap = NULL; if( rImplWallpaper.mpCache ) mpCache = new BitmapEx( *rImplWallpaper.mpCache ); else mpCache = NULL; if ( rImplWallpaper.mpGradient ) mpGradient = new Gradient( *rImplWallpaper.mpGradient ); else mpGradient = NULL; if ( rImplWallpaper.mpRect ) mpRect = new Rectangle( *rImplWallpaper.mpRect ); else mpRect = NULL; } ImplWallpaper::~ImplWallpaper() { delete mpBitmap; delete mpCache; delete mpGradient; delete mpRect; } void ImplWallpaper::ImplSetCachedBitmap( BitmapEx& rBmp ) { if( !mpCache ) mpCache = new BitmapEx( rBmp ); else *mpCache = rBmp; } void ImplWallpaper::ImplReleaseCachedBitmap() { delete mpCache; mpCache = NULL; } SvStream& ReadImplWallpaper( SvStream& rIStm, ImplWallpaper& rImplWallpaper ) { VersionCompat aCompat( rIStm, StreamMode::READ ); sal_uInt16 nTmp16; delete rImplWallpaper.mpRect; rImplWallpaper.mpRect = NULL; delete rImplWallpaper.mpGradient; rImplWallpaper.mpGradient = NULL; delete rImplWallpaper.mpBitmap; rImplWallpaper.mpBitmap = NULL; // version 1 ReadColor( rIStm, rImplWallpaper.maColor ); rIStm.ReadUInt16( nTmp16 ); rImplWallpaper.meStyle = (WallpaperStyle) nTmp16; // version 2 if( aCompat.GetVersion() >= 2 ) { bool bRect, bGrad, bBmp, bDummy; rIStm.ReadCharAsBool( bRect ).ReadCharAsBool( bGrad ).ReadCharAsBool( bBmp ).ReadCharAsBool( bDummy ).ReadCharAsBool( bDummy ).ReadCharAsBool( bDummy ); if( bRect ) { rImplWallpaper.mpRect = new Rectangle; ReadRectangle( rIStm, *rImplWallpaper.mpRect ); } if( bGrad ) { rImplWallpaper.mpGradient = new Gradient; ReadGradient( rIStm, *rImplWallpaper.mpGradient ); } if( bBmp ) { rImplWallpaper.mpBitmap = new BitmapEx; ReadDIBBitmapEx(*rImplWallpaper.mpBitmap, rIStm); } // version 3 (new color format) if( aCompat.GetVersion() >= 3 ) { rImplWallpaper.maColor.Read( rIStm, true ); } } return rIStm; } SvStream& WriteImplWallpaper( SvStream& rOStm, const ImplWallpaper& rImplWallpaper ) { VersionCompat aCompat( rOStm, StreamMode::WRITE, 3 ); bool bRect = ( rImplWallpaper.mpRect != NULL ); bool bGrad = ( rImplWallpaper.mpGradient != NULL ); bool bBmp = ( rImplWallpaper.mpBitmap != NULL ); bool bDummy = false; // version 1 WriteColor( rOStm, rImplWallpaper.maColor ); rOStm.WriteUInt16( rImplWallpaper.meStyle ); // version 2 rOStm.WriteBool( bRect ).WriteBool( bGrad ).WriteBool( bBmp ).WriteBool( bDummy ).WriteBool( bDummy ).WriteBool( bDummy ); if( bRect ) WriteRectangle( rOStm, *rImplWallpaper.mpRect ); if( bGrad ) WriteGradient( rOStm, *rImplWallpaper.mpGradient ); if( bBmp ) WriteDIBBitmapEx(*rImplWallpaper.mpBitmap, rOStm); // version 3 (new color format) ( (Color&) rImplWallpaper.maColor ).Write( rOStm, true ); return rOStm; } inline void Wallpaper::ImplMakeUnique( bool bReleaseCache ) { // copy them if other references exist if ( mpImplWallpaper->mnRefCount != 1 ) { if ( mpImplWallpaper->mnRefCount ) mpImplWallpaper->mnRefCount--; mpImplWallpaper = new ImplWallpaper( *(mpImplWallpaper) ); } if( bReleaseCache ) mpImplWallpaper->ImplReleaseCachedBitmap(); } Wallpaper::Wallpaper() { static ImplWallpaper aStaticImplWallpaper; aStaticImplWallpaper.mnRefCount = 0; mpImplWallpaper = &aStaticImplWallpaper; } Wallpaper::Wallpaper( const Wallpaper& rWallpaper ) { DBG_ASSERT( rWallpaper.mpImplWallpaper->mnRefCount < 0xFFFFFFFE, "Wallpaper: RefCount overflow" ); // use Instance data and increment reference counter mpImplWallpaper = rWallpaper.mpImplWallpaper; // RefCount == 0 for static objekts if ( mpImplWallpaper->mnRefCount ) mpImplWallpaper->mnRefCount++; } Wallpaper::Wallpaper( const Color& rColor ) { mpImplWallpaper = new ImplWallpaper; mpImplWallpaper->maColor = rColor; mpImplWallpaper->meStyle = WALLPAPER_TILE; } Wallpaper::Wallpaper( const BitmapEx& rBmpEx ) { mpImplWallpaper = new ImplWallpaper; mpImplWallpaper->mpBitmap = new BitmapEx( rBmpEx ); mpImplWallpaper->meStyle = WALLPAPER_TILE; } Wallpaper::Wallpaper( const Gradient& rGradient ) { mpImplWallpaper = new ImplWallpaper; mpImplWallpaper->mpGradient = new Gradient( rGradient ); mpImplWallpaper->meStyle = WALLPAPER_TILE; } Wallpaper::~Wallpaper() { // if ImpData are not static then delete them if it is the last reference, // otherwise decrement reference counter if ( mpImplWallpaper->mnRefCount ) { if ( mpImplWallpaper->mnRefCount == 1 ) delete mpImplWallpaper; else mpImplWallpaper->mnRefCount--; } } void Wallpaper::SetColor( const Color& rColor ) { ImplMakeUnique(); mpImplWallpaper->maColor = rColor; if( WALLPAPER_NULL == mpImplWallpaper->meStyle || WALLPAPER_APPLICATIONGRADIENT == mpImplWallpaper->meStyle ) mpImplWallpaper->meStyle = WALLPAPER_TILE; } const Color& Wallpaper::GetColor() const { return mpImplWallpaper->maColor; } void Wallpaper::SetStyle( WallpaperStyle eStyle ) { ImplMakeUnique( false ); if( eStyle == WALLPAPER_APPLICATIONGRADIENT ) // set a dummy gradient, the correct gradient // will be created dynamically in GetGradient() SetGradient( ImplGetApplicationGradient() ); mpImplWallpaper->meStyle = eStyle; } WallpaperStyle Wallpaper::GetStyle() const { return mpImplWallpaper->meStyle; } void Wallpaper::SetBitmap( const BitmapEx& rBitmap ) { if ( !rBitmap ) { if ( mpImplWallpaper->mpBitmap ) { ImplMakeUnique(); delete mpImplWallpaper->mpBitmap; mpImplWallpaper->mpBitmap = NULL; } } else { ImplMakeUnique(); if ( mpImplWallpaper->mpBitmap ) *(mpImplWallpaper->mpBitmap) = rBitmap; else mpImplWallpaper->mpBitmap = new BitmapEx( rBitmap ); } if( WALLPAPER_NULL == mpImplWallpaper->meStyle || WALLPAPER_APPLICATIONGRADIENT == mpImplWallpaper->meStyle) mpImplWallpaper->meStyle = WALLPAPER_TILE; } BitmapEx Wallpaper::GetBitmap() const { if ( mpImplWallpaper->mpBitmap ) return *(mpImplWallpaper->mpBitmap); else { BitmapEx aBmp; return aBmp; } } bool Wallpaper::IsBitmap() const { return (mpImplWallpaper->mpBitmap != 0); } void Wallpaper::SetGradient( const Gradient& rGradient ) { ImplMakeUnique(); if ( mpImplWallpaper->mpGradient ) *(mpImplWallpaper->mpGradient) = rGradient; else mpImplWallpaper->mpGradient = new Gradient( rGradient ); if( WALLPAPER_NULL == mpImplWallpaper->meStyle || WALLPAPER_APPLICATIONGRADIENT == mpImplWallpaper->meStyle ) mpImplWallpaper->meStyle = WALLPAPER_TILE; } Gradient Wallpaper::GetGradient() const { if( WALLPAPER_APPLICATIONGRADIENT == mpImplWallpaper->meStyle ) return ImplGetApplicationGradient(); else if ( mpImplWallpaper->mpGradient ) return *(mpImplWallpaper->mpGradient); else { Gradient aGradient; return aGradient; } } bool Wallpaper::IsGradient() const { return (mpImplWallpaper->mpGradient != 0); } Gradient Wallpaper::ImplGetApplicationGradient() const { Gradient g; g.SetAngle( 900 ); g.SetStyle( GradientStyle_LINEAR ); g.SetStartColor( Application::GetSettings().GetStyleSettings().GetFaceColor() ); // no 'extreme' gradient when high contrast if( Application::GetSettings().GetStyleSettings().GetHighContrastMode() ) g.SetEndColor( Application::GetSettings().GetStyleSettings().GetFaceColor() ); else g.SetEndColor( Application::GetSettings().GetStyleSettings().GetFaceGradientColor() ); return g; } void Wallpaper::SetRect( const Rectangle& rRect ) { ImplMakeUnique( false ); if ( rRect.IsEmpty() ) { if ( mpImplWallpaper->mpRect ) { delete mpImplWallpaper->mpRect; mpImplWallpaper->mpRect = NULL; } } else { if ( mpImplWallpaper->mpRect ) *(mpImplWallpaper->mpRect) = rRect; else mpImplWallpaper->mpRect = new Rectangle( rRect ); } } Rectangle Wallpaper::GetRect() const { if ( mpImplWallpaper->mpRect ) return *(mpImplWallpaper->mpRect); else { Rectangle aRect; return aRect; } } bool Wallpaper::IsRect() const { return (mpImplWallpaper->mpRect != 0); } bool Wallpaper::IsFixed() const { if ( mpImplWallpaper->meStyle == WALLPAPER_NULL ) return false; else return (!mpImplWallpaper->mpBitmap && !mpImplWallpaper->mpGradient); } bool Wallpaper::IsScrollable() const { if ( mpImplWallpaper->meStyle == WALLPAPER_NULL ) return false; else if ( !mpImplWallpaper->mpBitmap && !mpImplWallpaper->mpGradient ) return true; else if ( mpImplWallpaper->mpBitmap ) return (mpImplWallpaper->meStyle == WALLPAPER_TILE); else return false; } Wallpaper& Wallpaper::operator=( const Wallpaper& rWallpaper ) { DBG_ASSERT( rWallpaper.mpImplWallpaper->mnRefCount < 0xFFFFFFFE, "Wallpaper: RefCount overflow" ); // first increment reference counter, in order to self assign if ( rWallpaper.mpImplWallpaper->mnRefCount ) rWallpaper.mpImplWallpaper->mnRefCount++; // if ImpData are not static then delete them if it is the last reference, // otherwise decrement reference counter if ( mpImplWallpaper->mnRefCount ) { if ( mpImplWallpaper->mnRefCount == 1 ) delete mpImplWallpaper; else mpImplWallpaper->mnRefCount--; } mpImplWallpaper = rWallpaper.mpImplWallpaper; return *this; } bool Wallpaper::operator==( const Wallpaper& rWallpaper ) const { if ( mpImplWallpaper == rWallpaper.mpImplWallpaper ) return true; if ( ( mpImplWallpaper->meStyle != rWallpaper.mpImplWallpaper->meStyle ) || ( mpImplWallpaper->maColor != rWallpaper.mpImplWallpaper->maColor ) ) return false; if ( mpImplWallpaper->mpRect != rWallpaper.mpImplWallpaper->mpRect && ( !mpImplWallpaper->mpRect || !rWallpaper.mpImplWallpaper->mpRect || *(mpImplWallpaper->mpRect) != *(rWallpaper.mpImplWallpaper->mpRect) ) ) return false; if ( mpImplWallpaper->mpBitmap != rWallpaper.mpImplWallpaper->mpBitmap && ( !mpImplWallpaper->mpBitmap || !rWallpaper.mpImplWallpaper->mpBitmap || *(mpImplWallpaper->mpBitmap) != *(rWallpaper.mpImplWallpaper->mpBitmap) ) ) return false; if ( mpImplWallpaper->mpGradient != rWallpaper.mpImplWallpaper->mpGradient && ( !mpImplWallpaper->mpGradient || !rWallpaper.mpImplWallpaper->mpGradient || *(mpImplWallpaper->mpGradient) != *(rWallpaper.mpImplWallpaper->mpGradient) ) ) return false; return true; } SvStream& ReadWallpaper( SvStream& rIStm, Wallpaper& rWallpaper ) { rWallpaper.ImplMakeUnique(); return ReadImplWallpaper( rIStm, *rWallpaper.mpImplWallpaper ); } SvStream& WriteWallpaper( SvStream& rOStm, const Wallpaper& rWallpaper ) { return WriteImplWallpaper( rOStm, *rWallpaper.mpImplWallpaper ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */