xref: /trunk/main/vcl/source/gdi/image.cxx (revision 9f62ea84)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 // MARKER(update_precomp.py): autogen include statement, do not remove
24 #include "precompiled_vcl.hxx"
25 
26 #include <boost/scoped_ptr.hpp>
27 #include <boost/scoped_array.hpp>
28 
29 #include <rtl/logfile.hxx>
30 
31 #include <tools/debug.hxx>
32 #include <tools/stream.hxx>
33 #include <tools/rc.h>
34 #include <tools/rc.hxx>
35 #include <tools/resmgr.hxx>
36 
37 #include <vcl/settings.hxx>
38 #include <vcl/outdev.hxx>
39 #include <vcl/graph.hxx>
40 #include <vcl/svapp.hxx>
41 #include <vcl/image.hxx>
42 
43 #include <impimagetree.hxx>
44 #include <image.h>
45 
46 #if OSL_DEBUG_LEVEL > 0
47 #include <rtl/strbuf.hxx>
48 #endif
49 
50 DBG_NAME( Image )
51 DBG_NAME( ImageList )
52 
53 #define IMAGE_FILE_VERSION 100
54 
55 using namespace ::com::sun::star;
56 
57 // ---------
58 // - Image -
59 // ---------
60 
Image()61 Image::Image() :
62 	mpImplData( NULL )
63 {
64 	DBG_CTOR( Image, NULL );
65 }
66 
67 // -----------------------------------------------------------------------
68 
Image(const ResId & rResId)69 Image::Image( const ResId& rResId ) :
70 	mpImplData( NULL )
71 {
72 	DBG_CTOR( Image, NULL );
73 
74 	rResId.SetRT( RSC_IMAGE );
75 
76 	ResMgr* pResMgr = rResId.GetResMgr();
77 	if( pResMgr && pResMgr->GetResource( rResId ) )
78 	{
79 		pResMgr->Increment( sizeof( RSHEADER_TYPE ) );
80 
81 		BitmapEx	aBmpEx;
82 		sal_uLong 		nObjMask = pResMgr->ReadLong();
83 
84 		if( nObjMask & RSC_IMAGE_IMAGEBITMAP )
85 		{
86 			aBmpEx = BitmapEx( ResId( (RSHEADER_TYPE*)pResMgr->GetClass(), *pResMgr ) );
87 			pResMgr->Increment( pResMgr->GetObjSize( (RSHEADER_TYPE*)pResMgr->GetClass() ) );
88 		}
89 
90         if( nObjMask & RSC_IMAGE_MASKBITMAP )
91         {
92             if( !aBmpEx.IsEmpty() && aBmpEx.GetTransparentType() == TRANSPARENT_NONE )
93             {
94                 const Bitmap aMaskBitmap( ResId( (RSHEADER_TYPE*)pResMgr->GetClass(), *pResMgr ) );
95                 aBmpEx = BitmapEx( aBmpEx.GetBitmap(), aMaskBitmap );
96             }
97 
98             pResMgr->Increment( pResMgr->GetObjSize( (RSHEADER_TYPE*)pResMgr->GetClass() ) );
99         }
100 
101         if( nObjMask & RSC_IMAGE_MASKCOLOR )
102         {
103             if( !aBmpEx.IsEmpty() && aBmpEx.GetTransparentType() == TRANSPARENT_NONE )
104             {
105                 const Color aMaskColor( ResId( (RSHEADER_TYPE*)pResMgr->GetClass(), *pResMgr ) );
106                 aBmpEx = BitmapEx( aBmpEx.GetBitmap(), aMaskColor );
107             }
108 
109             pResMgr->Increment( pResMgr->GetObjSize( (RSHEADER_TYPE*)pResMgr->GetClass() ) );
110         }
111         if( ! aBmpEx.IsEmpty() )
112 			ImplInit( aBmpEx );
113 	}
114 }
115 
116 // -----------------------------------------------------------------------
117 
Image(const Image & rImage)118 Image::Image( const Image& rImage ) :
119 	mpImplData( rImage.mpImplData )
120 {
121 	DBG_CTOR( Image, NULL );
122 
123 	if( mpImplData )
124 		++mpImplData->mnRefCount;
125 }
126 
127 // -----------------------------------------------------------------------
128 
Image(const BitmapEx & rBitmapEx)129 Image::Image( const BitmapEx& rBitmapEx ) :
130 	mpImplData( NULL )
131 {
132 	DBG_CTOR( Image, NULL );
133 
134 	ImplInit( rBitmapEx );
135 }
136 
137 // -----------------------------------------------------------------------
138 
Image(const Bitmap & rBitmap)139 Image::Image( const Bitmap& rBitmap ) :
140 	mpImplData( NULL )
141 {
142 	DBG_CTOR( Image, NULL );
143 
144 	ImplInit( rBitmap );
145 }
146 
147 // -----------------------------------------------------------------------
148 
Image(const Bitmap & rBitmap,const Bitmap & rMaskBitmap)149 Image::Image( const Bitmap& rBitmap, const Bitmap& rMaskBitmap ) :
150 	mpImplData( NULL )
151 {
152 	DBG_CTOR( Image, NULL );
153 
154 	const BitmapEx aBmpEx( rBitmap, rMaskBitmap );
155 
156 	ImplInit( aBmpEx );
157 }
158 
159 // -----------------------------------------------------------------------
160 
Image(const Bitmap & rBitmap,const Color & rColor)161 Image::Image( const Bitmap& rBitmap, const Color& rColor ) :
162 	mpImplData( NULL )
163 {
164 	DBG_CTOR( Image, NULL );
165 
166 	const BitmapEx aBmpEx( rBitmap, rColor );
167 
168 	ImplInit( aBmpEx );
169 }
170 
171 // -----------------------------------------------------------------------
172 
Image(const uno::Reference<graphic::XGraphic> & rxGraphic)173 Image::Image( const uno::Reference< graphic::XGraphic >& rxGraphic ) :
174 	mpImplData( NULL )
175 {
176 	DBG_CTOR( Image, NULL );
177 
178 	const Graphic aGraphic( rxGraphic );
179 	ImplInit( aGraphic.GetBitmapEx() );
180 }
181 
182 // -----------------------------------------------------------------------
183 
~Image()184 Image::~Image()
185 {
186 	DBG_DTOR( Image, NULL );
187 
188 	if( mpImplData && ( 0 == --mpImplData->mnRefCount ) )
189 		delete mpImplData;
190 }
191 
192 // -----------------------------------------------------------------------
193 
ImplInit(const BitmapEx & rBmpEx)194 void Image::ImplInit( const BitmapEx& rBmpEx )
195 {
196 	if( !rBmpEx.IsEmpty() )
197 	{
198 		mpImplData = new ImplImage;
199 		mpImplData->mnRefCount = 1;
200 
201 		if( rBmpEx.GetTransparentType() == TRANSPARENT_NONE )
202 		{
203 			mpImplData->meType = IMAGETYPE_BITMAP;
204 			mpImplData->mpData = new Bitmap( rBmpEx.GetBitmap() );
205 		}
206 		else
207 		{
208 			mpImplData->meType = IMAGETYPE_IMAGE;
209 			mpImplData->mpData = new ImplImageData( rBmpEx );
210 		}
211 	}
212 }
213 
214 // -----------------------------------------------------------------------
215 
GetSizePixel() const216 Size Image::GetSizePixel() const
217 {
218 	DBG_CHKTHIS( Image, NULL );
219 
220 	Size aRet;
221 
222 	if( mpImplData )
223 	{
224 		switch( mpImplData->meType )
225 		{
226 			case IMAGETYPE_BITMAP:
227 				aRet = static_cast< Bitmap* >( mpImplData->mpData )->GetSizePixel();
228 			break;
229 
230 			case IMAGETYPE_IMAGE:
231 				aRet = static_cast< ImplImageData* >( mpImplData->mpData )->maBmpEx.GetSizePixel();
232 			break;
233 		}
234 	}
235 
236 	return aRet;
237 }
238 
239 // -----------------------------------------------------------------------
240 
GetBitmapEx() const241 BitmapEx Image::GetBitmapEx() const
242 {
243 	DBG_CHKTHIS( Image, NULL );
244 
245 	BitmapEx aRet;
246 
247 	if( mpImplData )
248 	{
249 		switch( mpImplData->meType )
250 		{
251 			case IMAGETYPE_BITMAP:
252 				aRet = *static_cast< Bitmap* >( mpImplData->mpData );
253 			break;
254 
255 			case IMAGETYPE_IMAGE:
256 				aRet = static_cast< ImplImageData* >( mpImplData->mpData )->maBmpEx;
257 			break;
258 		}
259 	}
260 
261 	return aRet;
262 }
263 
264 // -----------------------------------------------------------------------
265 
GetXGraphic() const266 uno::Reference< graphic::XGraphic > Image::GetXGraphic() const
267 {
268 	const Graphic aGraphic( GetBitmapEx() );
269 
270 	return aGraphic.GetXGraphic();
271 }
272 
273 // -----------------------------------------------------------------------
274 
GetColorTransformedImage(ImageColorTransform eColorTransform) const275 Image Image::GetColorTransformedImage( ImageColorTransform eColorTransform ) const
276 {
277 	DBG_CHKTHIS( Image, NULL );
278 
279     Image aRet;
280 
281     if( IMAGECOLORTRANSFORM_HIGHCONTRAST == eColorTransform )
282     {
283         BitmapEx aBmpEx( GetBitmapEx() );
284 
285         if( !aBmpEx.IsEmpty() )
286         {
287             Color*  pSrcColors = NULL;
288             Color*  pDstColors = NULL;
289             sal_uLong   nColorCount = 0;
290 
291             Image::GetColorTransformArrays( eColorTransform, pSrcColors, pDstColors, nColorCount );
292 
293             if( nColorCount && pSrcColors && pDstColors )
294             {
295                 aBmpEx.Replace( pSrcColors, pDstColors, nColorCount );
296 				aRet = Image( aBmpEx );
297             }
298 
299             delete[] pSrcColors;
300             delete[] pDstColors;
301         }
302     }
303 	else if( IMAGECOLORTRANSFORM_MONOCHROME_BLACK == eColorTransform ||
304 			 IMAGECOLORTRANSFORM_MONOCHROME_WHITE == eColorTransform  )
305 	{
306         BitmapEx aBmpEx( GetBitmapEx() );
307 
308         if( !aBmpEx.IsEmpty() )
309 			aRet = Image( aBmpEx.GetColorTransformedBitmapEx( ( BmpColorMode )( eColorTransform ) ) );
310 	}
311 
312     if( !aRet )
313         aRet = *this;
314 
315     return aRet;
316 }
317 
318 // -----------------------------------------------------------------------
319 
Invert()320 void Image::Invert()
321 {
322     BitmapEx aInvertedBmp( GetBitmapEx() );
323     aInvertedBmp.Invert();
324     *this = aInvertedBmp;
325 }
326 
327 // -----------------------------------------------------------------------
328 
GetColorTransformArrays(ImageColorTransform eColorTransform,Color * & rpSrcColor,Color * & rpDstColor,sal_uLong & rColorCount)329 void Image::GetColorTransformArrays( ImageColorTransform eColorTransform,
330                                      Color*& rpSrcColor, Color*& rpDstColor, sal_uLong& rColorCount )
331 {
332     if( IMAGECOLORTRANSFORM_HIGHCONTRAST == eColorTransform )
333     {
334         rpSrcColor = new Color[ 4 ];
335         rpDstColor = new Color[ 4 ];
336         rColorCount = 4;
337 
338         rpSrcColor[ 0 ] = Color( COL_BLACK );
339         rpDstColor[ 0 ] = Color( COL_WHITE );
340 
341         rpSrcColor[ 1 ] = Color( COL_WHITE );
342         rpDstColor[ 1 ] = Color( COL_BLACK );
343 
344         rpSrcColor[ 2 ] = Color( COL_BLUE );
345         rpDstColor[ 2 ] = Color( COL_WHITE );
346 
347         rpSrcColor[ 3 ] = Color( COL_LIGHTBLUE );
348         rpDstColor[ 3 ] = Color( COL_WHITE );
349     }
350     else
351     {
352         rpSrcColor = rpDstColor = NULL;
353         rColorCount = 0;
354     }
355 }
356 
357 // -----------------------------------------------------------------------
358 
operator =(const Image & rImage)359 Image& Image::operator=( const Image& rImage )
360 {
361 	DBG_CHKTHIS( Image, NULL );
362 	DBG_CHKOBJ( &rImage, Image, NULL );
363 
364 	if( rImage.mpImplData )
365 		++rImage.mpImplData->mnRefCount;
366 
367 	if( mpImplData && ( 0 == --mpImplData->mnRefCount ) )
368 		delete mpImplData;
369 
370 	mpImplData = rImage.mpImplData;
371 
372 	return *this;
373 }
374 
375 // -----------------------------------------------------------------------
376 
operator ==(const Image & rImage) const377 sal_Bool Image::operator==( const Image& rImage ) const
378 {
379 	DBG_CHKTHIS( Image, NULL );
380 	DBG_CHKOBJ( &rImage, Image, NULL );
381 
382 	bool bRet = false;
383 
384 	if( rImage.mpImplData == mpImplData )
385 		bRet = true;
386 	else if( !rImage.mpImplData || !mpImplData )
387 		bRet = false;
388 	else if( rImage.mpImplData->mpData == mpImplData->mpData )
389 		bRet = true;
390 	else if( rImage.mpImplData->meType == mpImplData->meType )
391 	{
392 		switch( mpImplData->meType )
393 		{
394 			case IMAGETYPE_BITMAP:
395 				bRet = ( *static_cast< Bitmap* >( rImage.mpImplData->mpData ) == *static_cast< Bitmap* >( mpImplData->mpData ) );
396 			break;
397 
398 			case IMAGETYPE_IMAGE:
399 				bRet = static_cast< ImplImageData* >( rImage.mpImplData->mpData )->IsEqual( *static_cast< ImplImageData* >( mpImplData->mpData ) );
400 			break;
401 
402 			default:
403 				bRet = false;
404 			break;
405 		}
406 	}
407 
408 	return bRet;
409 }
410 
411 // -------------
412 // - ImageList -
413 // -------------
414 
ImageList(sal_uInt16 nInit,sal_uInt16 nGrow)415 ImageList::ImageList( sal_uInt16 nInit, sal_uInt16 nGrow ) :
416 	mpImplData( NULL ),
417 	mnInitSize( nInit ),
418 	mnGrowSize( nGrow )
419 {
420 	DBG_CTOR( ImageList, NULL );
421 }
422 
423 // -----------------------------------------------------------------------
424 
ImageList(const ResId & rResId)425 ImageList::ImageList( const ResId& rResId ) :
426 	mpImplData( NULL ),
427 	mnInitSize( 1 ),
428 	mnGrowSize( 4 )
429 {
430     RTL_LOGFILE_CONTEXT( aLog, "vcl: ImageList::ImageList( const ResId& rResId )" );
431 
432 	DBG_CTOR( ImageList, NULL );
433 
434 	rResId.SetRT( RSC_IMAGELIST );
435 
436 	ResMgr* pResMgr = rResId.GetResMgr();
437 
438 	if( pResMgr && pResMgr->GetResource( rResId ) )
439 	{
440 		pResMgr->Increment( sizeof( RSHEADER_TYPE ) );
441 
442 		sal_uLong 								nObjMask = pResMgr->ReadLong();
443 		const String						aPrefix( pResMgr->ReadString() );
444         ::boost::scoped_ptr< Color >        spMaskColor;
445 
446 		if( nObjMask & RSC_IMAGE_MASKCOLOR )
447             spMaskColor.reset( new Color( ResId( (RSHEADER_TYPE*)pResMgr->GetClass(), *pResMgr ) ) );
448 
449 		pResMgr->Increment( pResMgr->GetObjSize( (RSHEADER_TYPE*)pResMgr->GetClass() ) );
450 
451 		if( nObjMask & RSC_IMAGELIST_IDLIST )
452 		{
453 			for( sal_Int32 i = 0, nCount = pResMgr->ReadLong(); i < nCount; ++i )
454 				pResMgr->ReadLong();
455 		}
456 
457 		sal_Int32 nCount = pResMgr->ReadLong();
458 		ImplInit( static_cast< sal_uInt16 >( nCount ), Size() );
459 
460 		BitmapEx aEmpty;
461 		for( sal_Int32 i = 0; i < nCount; ++i )
462 		{
463 			rtl::OUString aName = pResMgr->ReadString();
464 			sal_uInt16 nId = static_cast< sal_uInt16 >( pResMgr->ReadLong() );
465 			mpImplData->AddImage( aName, nId, aEmpty );
466 		}
467 
468 		if( nObjMask & RSC_IMAGELIST_IDCOUNT )
469 			pResMgr->ReadShort();
470 	}
471 }
472 
473 // -----------------------------------------------------------------------
474 
ImageList(const::std::vector<::rtl::OUString> & rNameVector,const::rtl::OUString & rPrefix,const Color *)475 ImageList::ImageList( const ::std::vector< ::rtl::OUString >& rNameVector,
476 					  const ::rtl::OUString& rPrefix,
477 					  const Color* ) :
478 	mpImplData( NULL ),
479 	mnInitSize( 1 ),
480 	mnGrowSize( 4 )
481 {
482     RTL_LOGFILE_CONTEXT( aLog, "vcl: ImageList::ImageList(const vector< OUString >& ..." );
483 
484 	DBG_CTOR( ImageList, NULL );
485 
486     ImplInit( sal::static_int_cast< sal_uInt16 >( rNameVector.size() ), Size() );
487 
488 	mpImplData->maPrefix = rPrefix;
489 	for( sal_uInt32 i = 0; i < rNameVector.size(); ++i )
490 	{
491 //		fprintf (stderr, "List %p [%d]: '%s'\n",
492 //				 this, i, rtl::OUStringToOString( rNameVector[i], RTL_TEXTENCODING_UTF8 ).getStr() );
493 		mpImplData->AddImage( rNameVector[ i ], static_cast< sal_uInt16 >( i ) + 1, BitmapEx() );
494 	}
495 }
496 
497 // -----------------------------------------------------------------------
498 
ImageList(const ImageList & rImageList)499 ImageList::ImageList( const ImageList& rImageList ) :
500 	mpImplData( rImageList.mpImplData ),
501 	mnInitSize( rImageList.mnInitSize ),
502 	mnGrowSize( rImageList.mnGrowSize )
503 {
504 	DBG_CTOR( ImageList, NULL );
505 
506 	if( mpImplData )
507 		++mpImplData->mnRefCount;
508 }
509 
510 // -----------------------------------------------------------------------
511 
~ImageList()512 ImageList::~ImageList()
513 {
514 	DBG_DTOR( ImageList, NULL );
515 
516 	if( mpImplData && ( 0 == --mpImplData->mnRefCount ) )
517 		delete mpImplData;
518 }
519 
ImplInit(sal_uInt16 nItems,const Size & rSize)520 void ImageList::ImplInit( sal_uInt16 nItems, const Size &rSize )
521 {
522 	mpImplData = new ImplImageList;
523 	mpImplData->mnRefCount = 1;
524 	mpImplData->maImages.reserve( nItems );
525 	mpImplData->maImageSize = rSize;
526 }
527 
528 // -----------------------------------------------------------------------
529 
Load(const rtl::OUString & rPrefix)530 void ImageAryData::Load(const rtl::OUString &rPrefix)
531 {
532     static ImplImageTreeSingletonRef aImageTree;
533 
534 	::rtl::OUString aSymbolsStyle = Application::GetSettings().GetStyleSettings().GetCurrentSymbolsStyleName();
535 
536 	BitmapEx aBmpEx;
537 
538 //	fprintf (stderr, "Attempt load of '%s'\n",
539 //			 rtl::OUStringToOString( maName, RTL_TEXTENCODING_UTF8 ).getStr() );
540 
541 	rtl::OUString aFileName = rPrefix;
542 	aFileName += maName;
543 #if OSL_DEBUG_LEVEL > 0
544 	bool bSuccess =
545 #endif
546         aImageTree->loadImage( aFileName, aSymbolsStyle, maBitmapEx, true );
547 #if OSL_DEBUG_LEVEL > 0
548     if ( !bSuccess )
549     {
550         ::rtl::OStringBuffer aMessage;
551         aMessage.append( "ImageAryData::Load: failed to load image '" );
552         aMessage.append( ::rtl::OUStringToOString( aFileName, RTL_TEXTENCODING_UTF8 ).getStr() );
553         aMessage.append( "'" );
554         OSL_ENSURE( false, aMessage.makeStringAndClear().getStr() );
555     }
556 #endif
557 }
558 
559 // -----------------------------------------------------------------------
560 
ImplMakeUnique()561 void ImageList::ImplMakeUnique()
562 {
563 	if( mpImplData && mpImplData->mnRefCount > 1 )
564 	{
565 		--mpImplData->mnRefCount;
566 		mpImplData = new ImplImageList( *mpImplData ) ;
567 	}
568 }
569 
570 // -----------------------------------------------------------------------
571 // Rather a performance hazard:
GetAsHorizontalStrip() const572 BitmapEx ImageList::GetAsHorizontalStrip() const
573 {
574 	Size aSize( mpImplData->maImageSize );
575     sal_uInt16 nCount = GetImageCount();
576 	if( !nCount )
577 		return BitmapEx();
578 	aSize.Width() *= nCount;
579 
580 	// Load any stragglers
581     for (sal_uInt16 nIdx = 0; nIdx < nCount; nIdx++)
582 	{
583 		ImageAryData *pData = mpImplData->maImages[ nIdx ];
584 		if( pData->IsLoadable() )
585 			pData->Load( mpImplData->maPrefix );
586 	}
587 
588 	BitmapEx aTempl = mpImplData->maImages[ 0 ]->maBitmapEx;
589 	BitmapEx aResult;
590 	Bitmap aPixels( aSize, aTempl.GetBitmap().GetBitCount() );
591 	if( aTempl.IsAlpha() )
592 		aResult = BitmapEx( aPixels, AlphaMask( aSize ) );
593 	else if( aTempl.IsTransparent() )
594 		aResult = BitmapEx( aPixels, Bitmap( aSize, aTempl.GetMask().GetBitCount() ) );
595 	else
596 		aResult = BitmapEx( aPixels );
597 
598 	Rectangle aSrcRect( Point( 0, 0 ), mpImplData->maImageSize );
599     for (sal_uInt16 nIdx = 0; nIdx < nCount; nIdx++)
600 	{
601 		Rectangle aDestRect( Point( nIdx * mpImplData->maImageSize.Width(), 0 ),
602 							 mpImplData->maImageSize );
603 		ImageAryData *pData = mpImplData->maImages[ nIdx ];
604 		aResult.CopyPixel( aDestRect, aSrcRect, &pData->maBitmapEx);
605 	}
606 
607 	return aResult;
608 }
609 
610 // -----------------------------------------------------------------------
611 
InsertFromHorizontalStrip(const BitmapEx & rBitmapEx,const std::vector<rtl::OUString> & rNameVector)612 void ImageList::InsertFromHorizontalStrip( const BitmapEx &rBitmapEx,
613 										   const std::vector< rtl::OUString > &rNameVector )
614 {
615     sal_uInt16 nItems = sal::static_int_cast< sal_uInt16 >( rNameVector.size() );
616 
617 //	fprintf (stderr, "InsertFromHorizontalStrip (1) [%d items]\n", nItems);
618 
619 	if (!nItems)
620 			return;
621 
622 	Size aSize( rBitmapEx.GetSizePixel() );
623 	DBG_ASSERT (rBitmapEx.GetSizePixel().Width() % nItems == 0,
624 				"ImageList::InsertFromHorizontalStrip - very odd size");
625 	aSize.Width() /= nItems;
626 	ImplInit( nItems, aSize );
627 
628     for (sal_uInt16 nIdx = 0; nIdx < nItems; nIdx++)
629 	{
630 		BitmapEx aBitmap( rBitmapEx, Point( nIdx * aSize.Width(), 0 ), aSize );
631 		mpImplData->AddImage( rNameVector[ nIdx ], nIdx + 1, aBitmap );
632 	}
633 }
634 
635 // -----------------------------------------------------------------------
636 
InsertFromHorizontalBitmap(const ResId & rResId,sal_uInt16 nCount,const Color * pMaskColor,const Color * pSearchColors,const Color * pReplaceColors,sal_uLong nColorCount)637 void ImageList::InsertFromHorizontalBitmap( const ResId& rResId,
638 											sal_uInt16       nCount,
639 											const Color *pMaskColor,
640 											const Color *pSearchColors,
641 											const Color *pReplaceColors,
642 											sal_uLong        nColorCount)
643 {
644 	BitmapEx aBmpEx( rResId );
645 	if (!aBmpEx.IsTransparent())
646 	{
647         if( pMaskColor )
648             aBmpEx = BitmapEx( aBmpEx.GetBitmap(), *pMaskColor );
649         else
650             aBmpEx = BitmapEx( aBmpEx.GetBitmap() );
651     }
652 	if ( nColorCount && pSearchColors && pReplaceColors )
653 		aBmpEx.Replace( pSearchColors, pReplaceColors, nColorCount );
654 
655 	std::vector< rtl::OUString > aNames( nCount );
656 	InsertFromHorizontalStrip( aBmpEx, aNames );
657 }
658 
659 // -----------------------------------------------------------------------
660 
ImplGetImageId(const::rtl::OUString & rImageName) const661 sal_uInt16 ImageList::ImplGetImageId( const ::rtl::OUString& rImageName ) const
662 {
663 	DBG_CHKTHIS( ImageList, NULL );
664 
665 	ImageAryData *pImg = mpImplData->maNameHash[ rImageName ];
666 	if( pImg )
667 		return pImg->mnId;
668 	else
669 		return 0;
670 }
671 
672 // -----------------------------------------------------------------------
673 
AddImage(sal_uInt16 nId,const Image & rImage)674 void ImageList::AddImage( sal_uInt16 nId, const Image& rImage )
675 {
676 	DBG_CHKTHIS( ImageList, NULL );
677 	DBG_CHKOBJ( &rImage, Image, NULL );
678 	DBG_ASSERT( nId, "ImageList::AddImage(): ImageId == 0" );
679 	DBG_ASSERT( GetImagePos( nId ) == IMAGELIST_IMAGE_NOTFOUND, "ImageList::AddImage() - ImageId already exists" );
680 	DBG_ASSERT( rImage.mpImplData, "ImageList::AddImage(): Wrong Size" );
681 	DBG_ASSERT( !mpImplData || (rImage.GetSizePixel() == mpImplData->maImageSize), "ImageList::AddImage(): Wrong Size" );
682 
683 	if( !mpImplData )
684 		ImplInit( 0, rImage.GetSizePixel() );
685 
686 	mpImplData->AddImage( rtl::OUString(), nId, rImage.GetBitmapEx());
687 }
688 
689 // -----------------------------------------------------------------------
690 
AddImage(const::rtl::OUString & rImageName,const Image & rImage)691 void ImageList::AddImage( const ::rtl::OUString& rImageName, const Image& rImage )
692 {
693 	DBG_ASSERT( GetImagePos( rImageName ) == IMAGELIST_IMAGE_NOTFOUND, "ImageList::AddImage() - ImageName already exists" );
694 
695 	if( !mpImplData )
696 		ImplInit( 0, rImage.GetSizePixel() );
697 
698     mpImplData->AddImage( rImageName, GetImageCount() + 1,
699 						  rImage.GetBitmapEx() );
700 }
701 
702 // -----------------------------------------------------------------------
703 
ReplaceImage(sal_uInt16 nId,const Image & rImage)704 void ImageList::ReplaceImage( sal_uInt16 nId, const Image& rImage )
705 {
706 	DBG_CHKTHIS( ImageList, NULL );
707 	DBG_CHKOBJ( &rImage, Image, NULL );
708 	DBG_ASSERT( GetImagePos( nId ) != IMAGELIST_IMAGE_NOTFOUND, "ImageList::ReplaceImage(): Unknown nId" );
709 
710 	RemoveImage( nId );
711 	AddImage( nId, rImage );
712 }
713 
714 // -----------------------------------------------------------------------
715 
ReplaceImage(const::rtl::OUString & rImageName,const Image & rImage)716 void ImageList::ReplaceImage( const ::rtl::OUString& rImageName, const Image& rImage )
717 {
718     const sal_uInt16 nId = ImplGetImageId( rImageName );
719 
720 	if( nId )
721     {
722         RemoveImage( nId );
723 
724         if( !mpImplData )
725 		    ImplInit( 0, rImage.GetSizePixel() );
726         mpImplData->AddImage( rImageName, nId, rImage.GetBitmapEx());
727     }
728 }
729 
730 // -----------------------------------------------------------------------
731 
ReplaceImage(sal_uInt16 nId,sal_uInt16 nReplaceId)732 void ImageList::ReplaceImage( sal_uInt16 nId, sal_uInt16 nReplaceId )
733 {
734 	DBG_CHKTHIS( ImageList, NULL );
735 	DBG_ASSERT( GetImagePos( nId ) != IMAGELIST_IMAGE_NOTFOUND, "ImageList::ReplaceImage(): Unknown nId" );
736 	DBG_ASSERT( GetImagePos( nReplaceId ) != IMAGELIST_IMAGE_NOTFOUND, "ImageList::ReplaceImage(): Unknown nReplaceId" );
737 
738 	sal_uLong nPosDest = GetImagePos( nId );
739 	sal_uLong nPosSrc = GetImagePos( nReplaceId );
740 	if( nPosDest != IMAGELIST_IMAGE_NOTFOUND &&
741 		nPosSrc != IMAGELIST_IMAGE_NOTFOUND )
742 	{
743 		ImplMakeUnique();
744 		mpImplData->maImages[nPosDest] = mpImplData->maImages[nPosSrc];
745 	}
746 }
747 
748 // -----------------------------------------------------------------------
749 
ReplaceImage(const::rtl::OUString & rImageName,const::rtl::OUString & rReplaceName)750 void ImageList::ReplaceImage( const ::rtl::OUString& rImageName, const ::rtl::OUString& rReplaceName )
751 {
752     const sal_uInt16 nId1 = ImplGetImageId( rImageName ), nId2 = ImplGetImageId( rReplaceName );
753 
754 	if( nId1 && nId2 )
755         ReplaceImage( nId1, nId2 );
756 }
757 
758 // -----------------------------------------------------------------------
759 
RemoveImage(sal_uInt16 nId)760 void ImageList::RemoveImage( sal_uInt16 nId )
761 {
762 	DBG_CHKTHIS( ImageList, NULL );
763 
764 	for( sal_uInt32 i = 0; i < mpImplData->maImages.size(); ++i )
765 	{
766 		if( mpImplData->maImages[ i ]->mnId == nId )
767 		{
768 			mpImplData->RemoveImage( static_cast< sal_uInt16 >( i ) );
769 			break;
770 		}
771 	}
772 }
773 
774 // -----------------------------------------------------------------------
775 
RemoveImage(const::rtl::OUString & rImageName)776 void ImageList::RemoveImage( const ::rtl::OUString& rImageName )
777 {
778     const sal_uInt16 nId = ImplGetImageId( rImageName );
779 
780     if( nId )
781         RemoveImage( nId );
782 }
783 
784 // -----------------------------------------------------------------------
785 
GetImage(sal_uInt16 nId) const786 Image ImageList::GetImage( sal_uInt16 nId ) const
787 {
788 	DBG_CHKTHIS( ImageList, NULL );
789 
790 //	fprintf (stderr, "GetImage %d\n", nId);
791 
792 	Image aRet;
793 
794 	if( mpImplData )
795 	{
796 		std::vector<ImageAryData *>::iterator aIter;
797 		for( aIter = mpImplData->maImages.begin();
798 			 aIter != mpImplData->maImages.end(); aIter++)
799 		{
800 			if ((*aIter)->mnId == nId)
801 			{
802 				if( (*aIter)->IsLoadable() )
803 					(*aIter)->Load( mpImplData->maPrefix );
804 
805 				aRet = Image( (*aIter)->maBitmapEx );
806 			}
807 		}
808 	}
809 
810 	return aRet;
811 }
812 
813 // -----------------------------------------------------------------------
814 
GetImage(const::rtl::OUString & rImageName) const815 Image ImageList::GetImage( const ::rtl::OUString& rImageName ) const
816 {
817 //	fprintf (stderr, "GetImage '%s'\n",
818 //			 rtl::OUStringToOString( rImageName, RTL_TEXTENCODING_UTF8 ).getStr() );
819 
820 	if( mpImplData )
821 	{
822 		ImageAryData *pImg = mpImplData->maNameHash[ rImageName ];
823 
824 		if( pImg )
825 		{
826 			if( pImg->IsLoadable() )
827 				pImg->Load( mpImplData->maPrefix );
828 			return Image( pImg->maBitmapEx );
829 		}
830 	}
831 //	fprintf (stderr, "no such image\n");
832 
833 	return Image();
834 }
835 
836 // -----------------------------------------------------------------------
837 
Clear()838 void ImageList::Clear()
839 {
840 	DBG_CHKTHIS( ImageList, NULL );
841 
842 	if( mpImplData && ( 0 == --mpImplData->mnRefCount ) )
843 		delete mpImplData;
844 
845 	mpImplData = NULL;
846 }
847 
848 // -----------------------------------------------------------------------
849 
GetImageCount() const850 sal_uInt16 ImageList::GetImageCount() const
851 {
852 	DBG_CHKTHIS( ImageList, NULL );
853 
854 	return mpImplData ? static_cast< sal_uInt16 >( mpImplData->maImages.size() ) : 0;
855 }
856 
857 // -----------------------------------------------------------------------
858 
GetImagePos(sal_uInt16 nId) const859 sal_uInt16 ImageList::GetImagePos( sal_uInt16 nId ) const
860 {
861 	DBG_CHKTHIS( ImageList, NULL );
862 
863 	if( mpImplData && nId )
864 	{
865 		for( sal_uInt32 i = 0; i < mpImplData->maImages.size(); ++i )
866 		{
867 			if (mpImplData->maImages[ i ]->mnId == nId)
868 				return static_cast< sal_uInt16 >( i );
869 		}
870 	}
871 
872 	return IMAGELIST_IMAGE_NOTFOUND;
873 }
874 
HasImageAtPos(sal_uInt16 nId) const875 bool ImageList::HasImageAtPos( sal_uInt16 nId ) const
876 {
877 	return GetImagePos( nId ) != IMAGELIST_IMAGE_NOTFOUND;
878 }
879 
880 // -----------------------------------------------------------------------
881 
GetImagePos(const::rtl::OUString & rImageName) const882 sal_uInt16 ImageList::GetImagePos( const ::rtl::OUString& rImageName ) const
883 {
884 	DBG_CHKTHIS( ImageList, NULL );
885 
886 	if( mpImplData && rImageName.getLength() )
887 	{
888 		for( sal_uInt32 i = 0; i < mpImplData->maImages.size(); i++ )
889 		{
890 			if (mpImplData->maImages[i]->maName == rImageName)
891 				return static_cast< sal_uInt16 >( i );
892 		}
893 	}
894 
895 	return IMAGELIST_IMAGE_NOTFOUND;
896 }
897 
898 // -----------------------------------------------------------------------
899 
GetImageId(sal_uInt16 nPos) const900 sal_uInt16 ImageList::GetImageId( sal_uInt16 nPos ) const
901 {
902 	DBG_CHKTHIS( ImageList, NULL );
903 
904     if( mpImplData && (nPos < GetImageCount()) )
905 		return mpImplData->maImages[ nPos ]->mnId;
906 
907 	return 0;
908 }
909 
910 // -----------------------------------------------------------------------
911 
GetImageIds(::std::vector<sal_uInt16> & rIds) const912 void ImageList::GetImageIds( ::std::vector< sal_uInt16 >& rIds ) const
913 {
914     RTL_LOGFILE_CONTEXT( aLog, "vcl: ImageList::GetImageIds" );
915 
916 	DBG_CHKTHIS( ImageList, NULL );
917 
918 	rIds = ::std::vector< sal_uInt16 >();
919 
920 	if( mpImplData )
921 	{
922 		for( sal_uInt32 i = 0; i < mpImplData->maImages.size(); i++ )
923 			rIds.push_back( mpImplData->maImages[i]->mnId );
924 	}
925 }
926 
927 // -----------------------------------------------------------------------
928 
GetImageName(sal_uInt16 nPos) const929 ::rtl::OUString	ImageList::GetImageName( sal_uInt16 nPos ) const
930 {
931 	DBG_CHKTHIS( ImageList, NULL );
932 
933     if( mpImplData && (nPos < GetImageCount()) )
934 		return mpImplData->maImages[ nPos ]->maName;
935 
936 	return ::rtl::OUString();
937 }
938 
939 // -----------------------------------------------------------------------
940 
GetImageNames(::std::vector<::rtl::OUString> & rNames) const941 void ImageList::GetImageNames( ::std::vector< ::rtl::OUString >& rNames ) const
942 {
943     RTL_LOGFILE_CONTEXT( aLog, "vcl: ImageList::GetImageNames" );
944 
945 	DBG_CHKTHIS( ImageList, NULL );
946 
947 	rNames = ::std::vector< ::rtl::OUString >();
948 
949 	if( mpImplData )
950 	{
951 		for( sal_uInt32 i = 0; i < mpImplData->maImages.size(); i++ )
952         {
953             const rtl::OUString& rName( mpImplData->maImages[ i ]->maName );
954             if( rName.getLength() != 0 )
955                 rNames.push_back( rName );
956         }
957 	}
958 }
959 
960 // -----------------------------------------------------------------------
961 
GetImageSize() const962 Size ImageList::GetImageSize() const
963 {
964 	DBG_CHKTHIS( ImageList, NULL );
965 
966 	Size aRet;
967 
968 	if( mpImplData )
969 	{
970 		aRet = mpImplData->maImageSize;
971 
972 		// force load of 1st image to see - uncommon case.
973 		if( aRet.Width() == 0 && aRet.Height() == 0 &&
974             !mpImplData->maImages.empty() )
975 		{
976 			Image aTmp = GetImage( mpImplData->maImages[ 0 ]->mnId );
977 			aRet = mpImplData->maImageSize = aTmp.GetSizePixel();
978 		}
979 	}
980 //	fprintf (stderr, "GetImageSize returns %d, %d\n",
981 //			 aRet.Width(), aRet.Height());
982 
983 	return aRet;
984 }
985 
986 // -----------------------------------------------------------------------
987 
operator =(const ImageList & rImageList)988 ImageList& ImageList::operator=( const ImageList& rImageList )
989 {
990 	DBG_CHKTHIS( ImageList, NULL );
991 	DBG_CHKOBJ( &rImageList, ImageList, NULL );
992 
993 	if( rImageList.mpImplData )
994 		++rImageList.mpImplData->mnRefCount;
995 
996 	if( mpImplData && ( 0 == --mpImplData->mnRefCount ) )
997 		delete mpImplData;
998 
999 	mpImplData = rImageList.mpImplData;
1000 
1001 	return *this;
1002 }
1003 
1004 // -----------------------------------------------------------------------
1005 
operator ==(const ImageList & rImageList) const1006 sal_Bool ImageList::operator==( const ImageList& rImageList ) const
1007 {
1008 	DBG_CHKTHIS( ImageList, NULL );
1009 	DBG_CHKOBJ( &rImageList, ImageList, NULL );
1010 
1011 	bool bRet = false;
1012 
1013 	if( rImageList.mpImplData == mpImplData )
1014 		bRet = true;
1015 	else if( !rImageList.mpImplData || !mpImplData )
1016 		bRet = false;
1017 	else if( rImageList.GetImageCount() == GetImageCount() &&
1018 		 	 rImageList.mpImplData->maImageSize == mpImplData->maImageSize )
1019 		bRet = true; // strange semantic
1020 
1021 	return bRet;
1022 }
1023