/* -*- 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 <sal/log.hxx>
#include <tools/debug.hxx>
#include <vcl/image.hxx>
#include "ImageList.hxx"
ImageList::ImageList()
{
}
ImageList::ImageList(const std::vector< OUString >& rNameVector,
const OUString& rPrefix)
{
SAL_INFO( "vcl", "vcl: ImageList::ImageList(const vector< OUString >& ..." );
maImages.reserve( rNameVector.size() );
maPrefix = rPrefix;
for( size_t i = 0; i < rNameVector.size(); ++i )
ImplAddImage( rPrefix, rNameVector[ i ], static_cast< sal_uInt16 >( i ) + 1, Image() );
}
// FIXME: Rather a performance hazard
BitmapEx ImageList::GetAsHorizontalStrip() const
{
sal_uInt16 nCount = maImages.size();
if( !nCount )
return BitmapEx();
BitmapEx aTempl = maImages[ 0 ]->maImage.GetBitmapEx();
Size aImageSize(aTempl.GetSizePixel());
Size aSize(aImageSize.Width() * nCount, aImageSize.Height());
BitmapEx aResult( aTempl, Point(), aSize );
tools::Rectangle aSrcRect( Point( 0, 0 ), aImageSize );
for (sal_uInt16 nIdx = 0; nIdx < nCount; nIdx++)
{
tools::Rectangle aDestRect( Point( nIdx * aImageSize.Width(), 0 ), aImageSize );
ImageAryData *pData = maImages[ nIdx ].get();
BitmapEx aTmp = pData->maImage.GetBitmapEx();
aResult.CopyPixel( aDestRect, aSrcRect, aTmp);
}
return aResult;
}
void ImageList::InsertFromHorizontalStrip( const BitmapEx &rBitmapEx,
const std::vector< OUString > &rNameVector )
{
sal_uInt16 nItems = sal::static_int_cast< sal_uInt16 >( rNameVector.size() );
if (!nItems)
return;
Size aSize( rBitmapEx.GetSizePixel() );
DBG_ASSERT (rBitmapEx.GetSizePixel().Width() % nItems == 0,
"ImageList::InsertFromHorizontalStrip - very odd size");
aSize.setWidth( aSize.Width() / nItems );
maImages.clear();
maNameHash.clear();
maImages.reserve( nItems );
maPrefix.clear();
for (sal_uInt16 nIdx = 0; nIdx < nItems; nIdx++)
{
BitmapEx aBitmap( rBitmapEx, Point( nIdx * aSize.Width(), 0 ), aSize );
ImplAddImage( maPrefix, rNameVector[ nIdx ], nIdx + 1, Image( aBitmap ) );
}
}
sal_uInt16 ImageList::ImplGetImageId( const OUString& rImageName ) const
{
auto it = maNameHash.find( rImageName );
if (it == maNameHash.end())
return 0;
return it->second->mnId;
}
void ImageList::AddImage( const OUString& rImageName, const Image& rImage )
{
SAL_WARN_IF( GetImagePos( rImageName ) != IMAGELIST_IMAGE_NOTFOUND, "vcl", "ImageList::AddImage() - ImageName already exists" );
ImplAddImage( maPrefix, rImageName, GetImageCount() + 1, rImage );
}
void ImageList::ReplaceImage( const OUString& rImageName, const Image& rImage )
{
const sal_uInt16 nId = ImplGetImageId( rImageName );
if( nId )
{
// Just replace the bitmap rather than doing RemoveImage / AddImage
// which breaks index-based iteration.
ImageAryData *pImg = maNameHash[ rImageName ];
pImg->maImage = rImage;
}
}
void ImageList::RemoveImage( sal_uInt16 nId )
{
for( size_t i = 0; i < maImages.size(); ++i )
{
if( maImages[ i ]->mnId == nId )
{
ImplRemoveImage( static_cast< sal_uInt16 >( i ) );
break;
}
}
}
Image ImageList::GetImage( const OUString& rImageName ) const
{
auto it = maNameHash.find( rImageName );
if (it == maNameHash.end())
return Image();
return it->second->maImage;
}
sal_uInt16 ImageList::GetImageCount() const
{
return static_cast< sal_uInt16 >( maImages.size() );
}
sal_uInt16 ImageList::GetImagePos( std::u16string_view rImageName ) const
{
if( !rImageName.empty() )
{
for( size_t i = 0; i < maImages.size(); i++ )
{
if (maImages[i]->maName == rImageName)
return static_cast< sal_uInt16 >( i );
}
}
return IMAGELIST_IMAGE_NOTFOUND;
}
sal_uInt16 ImageList::GetImageId( sal_uInt16 nPos ) const
{
return maImages[ nPos ]->mnId;
}
const OUString & ImageList::GetImageName( sal_uInt16 nPos ) const
{
return maImages[ nPos ]->maName;
}
void ImageList::GetImageNames( std::vector< OUString >& rNames ) const
{
SAL_INFO( "vcl", "vcl: ImageList::GetImageNames" );
rNames = std::vector< OUString >();
for(auto const & pImage : maImages)
{
const OUString& rName( pImage->maName );
if( !rName.isEmpty())
rNames.push_back( rName );
}
}
void ImageList::ImplAddImage( std::u16string_view aPrefix, const OUString &aName,
sal_uInt16 nId, const Image &aImage )
{
Image aInsert = aImage;
if (!aInsert)
aInsert = Image( OUString::Concat("private:graphicrepository/") + aPrefix + aName );
ImageAryData *pImg = new ImageAryData{ aName, nId, std::move(aInsert) };
maImages.emplace_back( pImg );
if( !aName.isEmpty() )
maNameHash [ aName ] = pImg;
}
void ImageList::ImplRemoveImage( sal_uInt16 nPos )
{
ImageAryData *pImg = maImages[ nPos ].get();
if( !pImg->maName.isEmpty() )
maNameHash.erase( pImg->maName );
maImages.erase( maImages.begin() + nPos );
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
↑ V1029 Numeric Truncation Error. Return value of the 'size' function is written to the 16-bit variable.