Files
libreoffice/svx/source/xoutdev/xattrbmp.cxx
Noel Grandin 742c0838f2 loplugin:unusedfields in svx part1
Change-Id: I6621e0dda5a5dc3e68c7b1613975c075e2879912
Reviewed-on: https://gerrit.libreoffice.org/39006
Tested-by: Jenkins <ci@libreoffice.org>
Reviewed-by: Noel Grandin <noel.grandin@collabora.co.uk>
2017-06-21 08:42:41 +02:00

510 lines
14 KiB
C++

/* -*- 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 <com/sun/star/awt/XBitmap.hpp>
#include <com/sun/star/graphic/XGraphic.hpp>
#include <tools/stream.hxx>
#include <vcl/window.hxx>
#include <vcl/virdev.hxx>
#include <vcl/bitmapex.hxx>
#include <toolkit/helper/vclunohelper.hxx>
#include <svl/style.hxx>
#include <editeng/memberids.hrc>
#include <svx/dialogs.hrc>
#include "svx/xattr.hxx"
#include <svx/xtable.hxx>
#include <svx/xdef.hxx>
#include <svx/unomid.hxx>
#include <editeng/unoprnms.hxx>
#include <svx/unoapi.hxx>
#include <svx/svdmodel.hxx>
#include <svx/xbitmap.hxx>
#include <com/sun/star/beans/PropertyValue.hpp>
#include <vcl/salbtype.hxx>
#include <vcl/bitmapaccess.hxx>
#include <vcl/dibtools.hxx>
#include <libxml/xmlwriter.h>
using namespace ::com::sun::star;
XOBitmap::XOBitmap( const Bitmap& rBmp ) :
xGraphicObject (new GraphicObject(rBmp)),
bGraphicDirty ( false )
{
}
XOBitmap::~XOBitmap()
{
}
Bitmap XOBitmap::GetBitmap() const
{
return GetGraphicObject().GetGraphic().GetBitmap();
}
const GraphicObject& XOBitmap::GetGraphicObject() const
{
if( bGraphicDirty )
const_cast<XOBitmap*>(this)->Array2Bitmap();
return *xGraphicObject;
}
void XOBitmap::Bitmap2Array()
{
ScopedVclPtrInstance< VirtualDevice > pVDev;
bool bPixelColor = false;
const Bitmap aBitmap( GetBitmap() );
const sal_Int32 nLines = 8; // type dependent
if( !pPixelArray )
pPixelArray.reset( new sal_uInt16[ nLines * nLines ] );
pVDev->SetOutputSizePixel( aBitmap.GetSizePixel() );
pVDev->DrawBitmap( Point(), aBitmap );
aPixelColor = aBckgrColor = pVDev->GetPixel( Point() );
// create array and determine foreground and background color
for (sal_Int32 i = 0; i < nLines; ++i)
{
for (sal_Int32 j = 0; j < nLines; ++j)
{
if ( pVDev->GetPixel( Point( j, i ) ) == aBckgrColor )
pPixelArray[ j + i * nLines ] = 0;
else
{
pPixelArray[ j + i * nLines ] = 1;
if( !bPixelColor )
{
aPixelColor = pVDev->GetPixel( Point( j, i ) );
bPixelColor = true;
}
}
}
}
}
/// convert array, fore- and background color into a bitmap
void XOBitmap::Array2Bitmap()
{
if (!pPixelArray)
return;
ScopedVclPtrInstance< VirtualDevice > pVDev;
const sal_Int32 nLines = 8; // type dependent
pVDev->SetOutputSizePixel( Size( nLines, nLines ) );
// create bitmap
for (sal_Int32 i = 0; i < nLines; ++i)
{
for (sal_Int32 j = 0; j < nLines; ++j)
{
if( pPixelArray[ j + i * nLines ] == 0 )
pVDev->DrawPixel( Point( j, i ), aBckgrColor );
else
pVDev->DrawPixel( Point( j, i ), aPixelColor );
}
}
xGraphicObject.reset(new GraphicObject(pVDev->GetBitmap(Point(), Size(nLines, nLines))));
bGraphicDirty = false;
}
SfxPoolItem* XFillBitmapItem::CreateDefault() { return new XFillBitmapItem; }
XFillBitmapItem::XFillBitmapItem(const OUString& rName, const GraphicObject& rGraphicObject)
: NameOrIndex(XATTR_FILLBITMAP, rName),
maGraphicObject(rGraphicObject)
{
}
XFillBitmapItem::XFillBitmapItem(const XFillBitmapItem& rItem)
: NameOrIndex(rItem),
maGraphicObject(rItem.maGraphicObject)
{
}
Bitmap createHistorical8x8FromArray(const sal_uInt16* pArray, Color aColorPix, Color aColorBack)
{
BitmapPalette aPalette(2);
aPalette[0] = BitmapColor(aColorBack);
aPalette[1] = BitmapColor(aColorPix);
Bitmap aBitmap(Size(8, 8), 1, &aPalette);
Bitmap::ScopedWriteAccess pContent(aBitmap);
if(pContent)
{
for(sal_uInt16 a(0); a < 8; a++)
{
for(sal_uInt16 b(0); b < 8; b++)
{
if(pArray[(a * 8) + b])
{
pContent->SetPixelIndex(a, b, 1);
}
else
{
pContent->SetPixelIndex(a, b, 0);
}
}
}
pContent.reset();
}
return aBitmap;
}
bool isHistorical8x8(const BitmapEx& rBitmapEx, BitmapColor& o_rBack, BitmapColor& o_rFront)
{
bool bRet(false);
if(!rBitmapEx.IsTransparent())
{
Bitmap aBitmap(rBitmapEx.GetBitmap());
if(8 == aBitmap.GetSizePixel().Width() && 8 == aBitmap.GetSizePixel().Height())
{
if(2 == aBitmap.GetColorCount())
{
BitmapReadAccess* pRead = aBitmap.AcquireReadAccess();
if(pRead)
{
if(pRead->HasPalette() && 2 == pRead->GetPaletteEntryCount())
{
const BitmapPalette& rPalette = pRead->GetPalette();
// #i123564# background and foreground were exchanged; of course
// rPalette[0] is the background color
o_rFront = rPalette[1];
o_rBack = rPalette[0];
bRet = true;
}
Bitmap::ReleaseAccess(pRead);
}
}
}
}
return bRet;
}
GraphicObject XFillBitmapItem::makeGraphicObject(SvStream& rIn, sal_uInt16 nVer) const
{
if (!IsIndex())
{
if(0 == nVer)
{
// work with the old bitmap
Bitmap aBmp;
ReadDIB(aBmp, rIn, true);
return Graphic(aBmp);
}
else if(1 == nVer)
{
sal_Int16 iTmp;
rIn.ReadInt16( iTmp ); // former XBitmapStyle
rIn.ReadInt16( iTmp ); // XBitmapType
if(XBitmapType::Import == (XBitmapType)iTmp)
{
Bitmap aBmp;
ReadDIB(aBmp, rIn, true);
return Graphic(aBmp);
}
else if(XBitmapType::N8x8 == (XBitmapType)iTmp)
{
sal_uInt16 aArray[64];
for(sal_uInt16 & i : aArray)
{
rIn.ReadUInt16( i );
}
Color aColorPix;
Color aColorBack;
ReadColor( rIn, aColorPix );
ReadColor( rIn, aColorBack );
const Bitmap aBitmap(createHistorical8x8FromArray(aArray, aColorPix, aColorBack));
return Graphic(aBitmap);
}
}
else if(2 == nVer)
{
BitmapEx aBmpEx;
ReadDIBBitmapEx(aBmpEx, rIn);
return Graphic(aBmpEx);
}
}
return GraphicObject();
}
XFillBitmapItem::XFillBitmapItem(SvStream& rIn, sal_uInt16 nVer)
: NameOrIndex(XATTR_FILLBITMAP, rIn)
, maGraphicObject(makeGraphicObject(rIn, nVer))
{
}
XFillBitmapItem::XFillBitmapItem(const GraphicObject& rGraphicObject)
: NameOrIndex(XATTR_FILLBITMAP, -1)
, maGraphicObject(rGraphicObject)
{
}
SfxPoolItem* XFillBitmapItem::Clone(SfxItemPool* /*pPool*/) const
{
return new XFillBitmapItem(*this);
}
bool XFillBitmapItem::operator==(const SfxPoolItem& rItem) const
{
return (NameOrIndex::operator==(rItem)
&& maGraphicObject == static_cast<const XFillBitmapItem&>(rItem).maGraphicObject);
}
SfxPoolItem* XFillBitmapItem::Create(SvStream& rIn, sal_uInt16 nVer) const
{
return new XFillBitmapItem( rIn, nVer );
}
SvStream& XFillBitmapItem::Store( SvStream& rOut, sal_uInt16 nItemVersion ) const
{
NameOrIndex::Store(rOut, nItemVersion);
if(!IsIndex())
{
WriteDIBBitmapEx(maGraphicObject.GetGraphic().GetBitmapEx(), rOut);
}
return rOut;
}
bool XFillBitmapItem::isPattern() const
{
BitmapColor aBack, aFront;
return isHistorical8x8(GetGraphicObject().GetGraphic().GetBitmap(), aBack, aFront);
}
sal_uInt16 XFillBitmapItem::GetVersion(sal_uInt16 /*nFileFormatVersion*/) const
{
return 2;
}
bool XFillBitmapItem::GetPresentation(
SfxItemPresentation /*ePres*/,
MapUnit /*eCoreUnit*/,
MapUnit /*ePresUnit*/,
OUString& rText,
const IntlWrapper*) const
{
rText += GetName();
return true;
}
bool XFillBitmapItem::QueryValue(css::uno::Any& rVal, sal_uInt8 nMemberId) const
{
nMemberId &= ~CONVERT_TWIPS;
// needed for MID_NAME
OUString aApiName;
// needed for complete item (MID 0)
OUString aInternalName;
OUString aURL;
css::uno::Reference< css::awt::XBitmap > xBmp;
if( nMemberId == MID_NAME )
{
aApiName = SvxUnogetApiNameForItem(Which(), GetName());
}
else if( nMemberId == 0 )
{
aInternalName = GetName();
}
if( nMemberId == MID_GRAFURL ||
nMemberId == 0 )
{
aURL = UNO_NAME_GRAPHOBJ_URLPREFIX;
aURL += OStringToOUString(
GetGraphicObject().GetUniqueID(),
RTL_TEXTENCODING_ASCII_US);
}
if( nMemberId == MID_BITMAP ||
nMemberId == 0 )
{
xBmp.set(VCLUnoHelper::CreateBitmap(GetGraphicObject().GetGraphic().GetBitmapEx()));
}
if( nMemberId == MID_NAME )
rVal <<= aApiName;
else if( nMemberId == MID_GRAFURL )
rVal <<= aURL;
else if( nMemberId == MID_BITMAP )
rVal <<= xBmp;
else
{
// member-id 0 => complete item (e.g. for toolbars)
DBG_ASSERT( nMemberId == 0, "invalid member-id" );
uno::Sequence< beans::PropertyValue > aPropSeq( 3 );
aPropSeq[0].Name = "Name";
aPropSeq[0].Value <<= aInternalName;
aPropSeq[1].Name = "FillBitmapURL";
aPropSeq[1].Value <<= aURL;
aPropSeq[2].Name = "Bitmap";
aPropSeq[2].Value <<= xBmp;
rVal <<= aPropSeq;
}
return true;
}
bool XFillBitmapItem::PutValue( const css::uno::Any& rVal, sal_uInt8 nMemberId )
{
nMemberId &= ~CONVERT_TWIPS;
OUString aName;
OUString aURL;
css::uno::Reference< css::awt::XBitmap > xBmp;
css::uno::Reference< css::graphic::XGraphic > xGraphic;
bool bSetName = false;
bool bSetURL = false;
bool bSetBitmap = false;
if( nMemberId == MID_NAME )
bSetName = (rVal >>= aName);
else if( nMemberId == MID_GRAFURL )
bSetURL = (rVal >>= aURL);
else if( nMemberId == MID_BITMAP )
{
bSetBitmap = (rVal >>= xBmp);
if ( !bSetBitmap )
bSetBitmap = (rVal >>= xGraphic );
}
else
{
DBG_ASSERT( nMemberId == 0, "invalid member-id" );
uno::Sequence< beans::PropertyValue > aPropSeq;
if( rVal >>= aPropSeq )
{
for ( sal_Int32 n = 0; n < aPropSeq.getLength(); n++ )
{
if ( aPropSeq[n].Name == "Name" )
bSetName = (aPropSeq[n].Value >>= aName);
else if ( aPropSeq[n].Name == "FillBitmapURL" )
bSetURL = (aPropSeq[n].Value >>= aURL);
else if ( aPropSeq[n].Name == "Bitmap" )
bSetBitmap = (aPropSeq[n].Value >>= xBmp);
}
}
}
if( bSetName )
{
SetName( aName );
}
if( bSetURL )
{
GraphicObject aGraphicObject = GraphicObject::CreateGraphicObjectFromURL(aURL);
if( aGraphicObject.GetType() != GraphicType::NONE )
maGraphicObject = aGraphicObject;
// #121194# Prefer GraphicObject over bitmap object if both are provided
if(bSetBitmap && GraphicType::NONE != maGraphicObject.GetType())
{
bSetBitmap = false;
}
}
if( bSetBitmap )
{
if(xBmp.is())
{
maGraphicObject.SetGraphic(VCLUnoHelper::GetBitmap(xBmp));
}
else if(xGraphic.is())
{
maGraphicObject.SetGraphic(xGraphic);
}
}
return (bSetName || bSetURL || bSetBitmap);
}
bool XFillBitmapItem::CompareValueFunc( const NameOrIndex* p1, const NameOrIndex* p2 )
{
const GraphicObject& aGraphicObjectA(static_cast<const XFillBitmapItem*>(p1)->GetGraphicObject());
const GraphicObject& aGraphicObjectB(static_cast<const XFillBitmapItem*>(p2)->GetGraphicObject());
return aGraphicObjectA == aGraphicObjectB;
}
XFillBitmapItem* XFillBitmapItem::checkForUniqueItem( SdrModel* pModel ) const
{
if( pModel )
{
XPropertyListType aListType = XPropertyListType::Bitmap;
if(isPattern())
aListType = XPropertyListType::Pattern;
const OUString aUniqueName = NameOrIndex::CheckNamedItem(
this, XATTR_FILLBITMAP, &pModel->GetItemPool(),
pModel->GetStyleSheetPool() ? &pModel->GetStyleSheetPool()->GetPool() : nullptr,
XFillBitmapItem::CompareValueFunc, RID_SVXSTR_BMP21,
pModel->GetPropertyList( aListType ) );
// if the given name is not valid, replace it!
if( aUniqueName != GetName() )
{
return new XFillBitmapItem(aUniqueName, maGraphicObject);
}
}
return nullptr;
}
void XFillBitmapItem::dumpAsXml(xmlTextWriterPtr pWriter) const
{
xmlTextWriterStartElement(pWriter, BAD_CAST("XFillBitmapItem"));
xmlTextWriterWriteAttribute(pWriter, BAD_CAST("whichId"), BAD_CAST(OString::number(Which()).getStr()));
NameOrIndex::dumpAsXml(pWriter);
xmlTextWriterEndElement(pWriter);
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */