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  
24  // MARKER(update_precomp.py): autogen include statement, do not remove
25  #include "precompiled_framework.hxx"
26  
27  #include <imagemanagerimpl.hxx>
28  #include <threadhelp/resetableguard.hxx>
29  #include <xml/imagesconfiguration.hxx>
30  #include <uiconfiguration/graphicnameaccess.hxx>
31  #include <services.h>
32  
33  #include "properties.h"
34  
35  //_________________________________________________________________________________________________________________
36  //	interface includes
37  //_________________________________________________________________________________________________________________
38  #include <com/sun/star/ui/UIElementType.hpp>
39  #include <com/sun/star/ui/ConfigurationEvent.hpp>
40  #include <com/sun/star/lang/DisposedException.hpp>
41  #include <com/sun/star/beans/XPropertySet.hpp>
42  #include <com/sun/star/beans/PropertyValue.hpp>
43  #include <com/sun/star/embed/ElementModes.hpp>
44  #include <com/sun/star/io/XStream.hpp>
45  #include <com/sun/star/ui/ImageType.hpp>
46  
47  //_________________________________________________________________________________________________________________
48  //	other includes
49  //_________________________________________________________________________________________________________________
50  
51  #include <vcl/svapp.hxx>
52  #include <rtl/ustrbuf.hxx>
53  #include <osl/mutex.hxx>
54  #include <comphelper/sequence.hxx>
55  #include <tools/urlobj.hxx>
56  #include <unotools/ucbstreamhelper.hxx>
57  #include <vcl/pngread.hxx>
58  #include <vcl/pngwrite.hxx>
59  #include <rtl/logfile.hxx>
60  #include "svtools/miscopt.hxx"
61  
62  //_________________________________________________________________________________________________________________
63  //	namespaces
64  //_________________________________________________________________________________________________________________
65  
66  using ::rtl::OUString;
67  using ::com::sun::star::uno::Sequence;
68  using ::com::sun::star::uno::XInterface;
69  using ::com::sun::star::uno::Exception;
70  using ::com::sun::star::uno::RuntimeException;
71  using ::com::sun::star::uno::UNO_QUERY;
72  using ::com::sun::star::uno::Any;
73  using ::com::sun::star::uno::makeAny;
74  using ::com::sun::star::graphic::XGraphic;
75  using namespace ::com::sun::star;
76  using namespace ::com::sun::star::io;
77  using namespace ::com::sun::star::embed;
78  using namespace ::com::sun::star::lang;
79  using namespace ::com::sun::star::container;
80  using namespace ::com::sun::star::beans;
81  using namespace ::com::sun::star::ui;
82  using namespace ::cppu;
83  
84  // Image sizes for our toolbars/menus
85  const sal_Int32 IMAGE_SIZE_NORMAL         = 16;
86  const sal_Int32 IMAGE_SIZE_LARGE          = 26;
87  const sal_Int16 MAX_IMAGETYPE_VALUE       = ::com::sun::star::ui::ImageType::COLOR_HIGHCONTRAST|
88                                              ::com::sun::star::ui::ImageType::SIZE_LARGE;
89  
90  static const char   IMAGE_FOLDER[]        = "images";
91  static const char   BITMAPS_FOLDER[]      = "Bitmaps";
92  static const char   IMAGE_EXTENSION[]     = ".png";
93  
94  static const char*  IMAGELIST_XML_FILE[]  =
95  {
96      "sc_imagelist.xml",
97      "lc_imagelist.xml",
98      "sch_imagelist.xml",
99      "lch_imagelist.xml"
100  };
101  
102  static const char*  BITMAP_FILE_NAMES[]   =
103  {
104      "sc_userimages.png",
105      "lc_userimages.png",
106      "sch_userimages.png",
107      "lch_userimages.png"
108  };
109  
110  namespace framework
111  {
112      static char ModuleImageList[] = "private:resource/images/moduleimages";
113      static osl::Mutex*          pImageListWrapperMutex = 0;
114      static GlobalImageList*     pGlobalImageList = 0;
115      static const char* ImageType_Prefixes[ImageType_COUNT] =
116      {
117          "res/commandimagelist/sc_",
118          "res/commandimagelist/lc_",
119          "res/commandimagelist/sch_",
120          "res/commandimagelist/lch_"
121      };
122  
123  typedef GraphicNameAccess CmdToXGraphicNameAccess;
124  
125  static osl::Mutex& getGlobalImageListMutex()
126  {
127      if ( pImageListWrapperMutex == 0 )
128      {
129          osl::MutexGuard aGuard( osl::Mutex::getGlobalMutex() ) ;
130          if ( pImageListWrapperMutex == 0 )
131              pImageListWrapperMutex = new osl::Mutex;
132      }
133  
134      return *pImageListWrapperMutex;
135  }
136  
137  static GlobalImageList* getGlobalImageList( const uno::Reference< XMultiServiceFactory >& rServiceManager )
138  {
139      osl::MutexGuard guard( getGlobalImageListMutex() );
140  
141      if ( pGlobalImageList == 0 )
142          pGlobalImageList = new GlobalImageList( rServiceManager );
143  
144      return pGlobalImageList;
145  }
146  
147  static rtl::OUString getCanonicalName( const rtl::OUString& rFileName )
148  {
149      bool               bRemoveSlash( true );
150      sal_Int32          nLength = rFileName.getLength();
151      const sal_Unicode* pString = rFileName.getStr();
152  
153      rtl::OUStringBuffer aBuf( nLength );
154      for ( sal_Int32 i = 0; i < nLength; i++ )
155      {
156          const sal_Unicode c = pString[i];
157          switch ( c )
158          {
159              // map forbidden characters to escape
160              case '/' : if ( !bRemoveSlash )
161                           aBuf.appendAscii( "%2f" );
162                         break;
163              case '\\': aBuf.appendAscii( "%5c" ); bRemoveSlash = false; break;
164              case ':' : aBuf.appendAscii( "%3a" ); bRemoveSlash = false; break;
165              case '*' : aBuf.appendAscii( "%2a" ); bRemoveSlash = false; break;
166              case '?' : aBuf.appendAscii( "%3f" ); bRemoveSlash = false; break;
167              case '<' : aBuf.appendAscii( "%3c" ); bRemoveSlash = false; break;
168              case '>' : aBuf.appendAscii( "%3e" ); bRemoveSlash = false; break;
169              case '|' : aBuf.appendAscii( "%7c" ); bRemoveSlash = false; break;
170              default: aBuf.append( c ); bRemoveSlash = false;
171          }
172      }
173      return aBuf.makeStringAndClear();
174  }
175  
176  //_________________________________________________________________________________________________________________
177  
178  CmdImageList::CmdImageList( const uno::Reference< XMultiServiceFactory >& rServiceManager, const rtl::OUString& aModuleIdentifier ) :
179      m_bVectorInit( sal_False ),
180      m_aModuleIdentifier( aModuleIdentifier ),
181      m_xServiceManager( rServiceManager ),
182      m_nSymbolsStyle( SvtMiscOptions().GetCurrentSymbolsStyle() )
183  {
184      for ( sal_Int32 n=0; n < ImageType_COUNT; n++ )
185          m_pImageList[n] = 0;
186  }
187  
188  CmdImageList::~CmdImageList()
189  {
190      for ( sal_Int32 n=0; n < ImageType_COUNT; n++ )
191          delete m_pImageList[n];
192  }
193  
194  void CmdImageList::impl_fillCommandToImageNameMap()
195  {
196      RTL_LOGFILE_CONTEXT( aLog, "framework: CmdImageList::impl_fillCommandToImageNameMap" );
197  
198      if ( !m_bVectorInit )
199      {
200          const rtl::OUString aCommandImageList( RTL_CONSTASCII_USTRINGPARAM( UICOMMANDDESCRIPTION_NAMEACCESS_COMMANDIMAGELIST ));
201          Sequence< OUString > aCmdImageSeq;
202          uno::Reference< XNameAccess > xCmdDesc( m_xServiceManager->createInstance(
203                                                  SERVICENAME_UICOMMANDDESCRIPTION ),
204                                              UNO_QUERY );
205  
206          if ( m_aModuleIdentifier.getLength() > 0 )
207          {
208              // If we have a module identifier - use to retrieve the command image name list from it.
209              // Otherwise we will use the global command image list
210              try
211              {
212                  xCmdDesc->getByName( m_aModuleIdentifier ) >>= xCmdDesc;
213                  if ( xCmdDesc.is() )
214                      xCmdDesc->getByName( aCommandImageList ) >>= aCmdImageSeq;
215              }
216              catch ( NoSuchElementException& )
217              {
218                  // Module unknown we will work with an empty command image list!
219                  return;
220              }
221          }
222  
223          if ( xCmdDesc.is() )
224          {
225              try
226              {
227                  xCmdDesc->getByName( aCommandImageList ) >>= aCmdImageSeq;
228              }
229              catch ( NoSuchElementException& )
230              {
231              }
232              catch ( WrappedTargetException& )
233              {
234              }
235          }
236  
237          // We have to map commands which uses special characters like '/',':','?','\','<'.'>','|'
238          String aExt = String::CreateFromAscii( IMAGE_EXTENSION );
239          m_aImageCommandNameVector.resize(aCmdImageSeq.getLength() );
240          m_aImageNameVector.resize( aCmdImageSeq.getLength() );
241  
242          ::std::copy( aCmdImageSeq.getConstArray(),
243                       aCmdImageSeq.getConstArray()+aCmdImageSeq.getLength(),
244                       m_aImageCommandNameVector.begin() );
245  
246          // Create a image name vector that must be provided to the vcl imagelist. We also need
247          // a command to image name map to speed up access time for image retrieval.
248          OUString aUNOString( RTL_CONSTASCII_USTRINGPARAM( ".uno:" ));
249          String   aEmptyString;
250          const sal_uInt32 nCount = m_aImageCommandNameVector.size();
251  		for ( sal_uInt32 i = 0; i < nCount; i++ )
252          {
253              OUString aCommandName( m_aImageCommandNameVector[i] );
254              String   aImageName;
255  
256              if ( aCommandName.indexOf( aUNOString ) != 0 )
257              {
258                  INetURLObject aUrlObject( aCommandName, INetURLObject::ENCODE_ALL );
259                  aImageName = aUrlObject.GetURLPath();
260                  aImageName = getCanonicalName( aImageName ); // convert to valid filename
261              }
262              else
263              {
264                  // just remove the schema
265                  if ( aCommandName.getLength() > 5 )
266                      aImageName = aCommandName.copy( 5 );
267                  else
268                      aImageName = aEmptyString;
269  
270                  // Search for query part.
271                  sal_Int32 nIndex = aImageName.Search( '?' );
272                  if ( nIndex != STRING_NOTFOUND )
273                      aImageName = getCanonicalName( aImageName ); // convert to valid filename
274              }
275              // Image names are not case-dependent. Always use lower case characters to
276              // reflect this.
277              aImageName += aExt;
278              aImageName.ToLowerAscii();
279  
280              m_aImageNameVector[i] = aImageName;
281              m_aCommandToImageNameMap.insert( CommandToImageNameMap::value_type( aCommandName, aImageName ));
282          }
283  
284          m_bVectorInit = sal_True;
285      }
286  }
287  
288  ImageList* CmdImageList::impl_getImageList( sal_Int16 nImageType )
289  {
290      SvtMiscOptions aMiscOptions;
291  
292      sal_Int16 nSymbolsStyle = aMiscOptions.GetCurrentSymbolsStyle();
293      if ( nSymbolsStyle != m_nSymbolsStyle )
294      {
295          m_nSymbolsStyle = nSymbolsStyle;
296          for ( sal_Int32 n=0; n < ImageType_COUNT; n++ )
297              delete m_pImageList[n], m_pImageList[n] = NULL;
298      }
299  
300      if ( !m_pImageList[nImageType] )
301      {
302          m_pImageList[nImageType] = new ImageList( m_aImageNameVector,
303                                                    OUString::createFromAscii( ImageType_Prefixes[nImageType] ) );
304      }
305  
306      return m_pImageList[nImageType];
307  }
308  
309  std::vector< ::rtl::OUString >& CmdImageList::impl_getImageNameVector()
310  {
311      return m_aImageNameVector;
312  }
313  
314  std::vector< rtl::OUString >& CmdImageList::impl_getImageCommandNameVector()
315  {
316      return m_aImageCommandNameVector;
317  }
318  
319  Image CmdImageList::getImageFromCommandURL( sal_Int16 nImageType, const rtl::OUString& rCommandURL )
320  {
321      impl_fillCommandToImageNameMap();
322      CommandToImageNameMap::const_iterator pIter = m_aCommandToImageNameMap.find( rCommandURL );
323      if ( pIter != m_aCommandToImageNameMap.end() )
324      {
325          ImageList* pImageList = impl_getImageList( nImageType );
326          return pImageList->GetImage( pIter->second );
327      }
328  
329      return Image();
330  }
331  
332  bool CmdImageList::hasImage( sal_Int16 /*nImageType*/, const rtl::OUString& rCommandURL )
333  {
334      impl_fillCommandToImageNameMap();
335      CommandToImageNameMap::const_iterator pIter = m_aCommandToImageNameMap.find( rCommandURL );
336      if ( pIter != m_aCommandToImageNameMap.end() )
337          return true;
338      else
339          return false;
340  }
341  
342  ::std::vector< rtl::OUString >& CmdImageList::getImageNames()
343  {
344      return impl_getImageNameVector();
345  }
346  
347  ::std::vector< rtl::OUString >& CmdImageList::getImageCommandNames()
348  {
349      return impl_getImageCommandNameVector();
350  }
351  
352  //_________________________________________________________________________________________________________________
353  
354  GlobalImageList::GlobalImageList( const uno::Reference< XMultiServiceFactory >& rServiceManager ) :
355      CmdImageList( rServiceManager, rtl::OUString() ),
356      m_nRefCount( 0 )
357  {
358  }
359  
360  GlobalImageList::~GlobalImageList()
361  {
362  }
363  
364  Image GlobalImageList::getImageFromCommandURL( sal_Int16 nImageType, const rtl::OUString& rCommandURL )
365  {
366      osl::MutexGuard guard( getGlobalImageListMutex() );
367      return CmdImageList::getImageFromCommandURL( nImageType, rCommandURL );
368  }
369  
370  bool GlobalImageList::hasImage( sal_Int16 nImageType, const rtl::OUString& rCommandURL )
371  {
372      osl::MutexGuard guard( getGlobalImageListMutex() );
373      return CmdImageList::hasImage( nImageType, rCommandURL );
374  }
375  
376  ::std::vector< rtl::OUString >& GlobalImageList::getImageNames()
377  {
378      osl::MutexGuard guard( getGlobalImageListMutex() );
379      return impl_getImageNameVector();
380  }
381  
382  ::std::vector< rtl::OUString >& GlobalImageList::getImageCommandNames()
383  {
384      osl::MutexGuard guard( getGlobalImageListMutex() );
385      return impl_getImageCommandNameVector();
386  }
387  
388  oslInterlockedCount GlobalImageList::acquire()
389  {
390      osl_incrementInterlockedCount( &m_nRefCount );
391      return m_nRefCount;
392  }
393  
394  oslInterlockedCount GlobalImageList::release()
395  {
396      osl::MutexGuard guard( getGlobalImageListMutex() );
397  
398      if ( !osl_decrementInterlockedCount( &m_nRefCount ))
399      {
400          oslInterlockedCount nCount( m_nRefCount );
401          // remove global pointer as we destroy the object now
402          pGlobalImageList = 0;
403  		delete this;
404          return nCount;
405      }
406  
407      return m_nRefCount;
408  }
409  
410  static sal_Bool implts_checkAndScaleGraphic( uno::Reference< XGraphic >& rOutGraphic, const uno::Reference< XGraphic >& rInGraphic, sal_Int16 nImageType )
411  {
412      static Size   aNormSize( IMAGE_SIZE_NORMAL, IMAGE_SIZE_NORMAL );
413      static Size   aLargeSize( IMAGE_SIZE_LARGE, IMAGE_SIZE_LARGE );
414  
415      if ( !rInGraphic.is() )
416      {
417          rOutGraphic = Image().GetXGraphic();
418          return sal_False;
419      }
420  
421      // Check size and scale it
422      Image  aImage( rInGraphic );
423      Size   aSize = aImage.GetSizePixel();
424      bool   bMustScale( false );
425  
426      if (( nImageType == ImageType_Color_Large ) ||
427          ( nImageType == ImageType_HC_Large ))
428          bMustScale = ( aSize != aLargeSize );
429      else
430          bMustScale = ( aSize != aNormSize );
431  
432      if ( bMustScale )
433      {
434          BitmapEx aBitmap = aImage.GetBitmapEx();
435          aBitmap.Scale( aNormSize );
436          aImage = Image( aBitmap );
437          rOutGraphic = aImage.GetXGraphic();
438      }
439      else
440          rOutGraphic = rInGraphic;
441      return sal_True;
442  }
443  
444  static sal_Int16 implts_convertImageTypeToIndex( sal_Int16 nImageType )
445  {
446      sal_Int16 nIndex( 0 );
447      if ( nImageType & ::com::sun::star::ui::ImageType::SIZE_LARGE )
448          nIndex += 1;
449      if ( nImageType & ::com::sun::star::ui::ImageType::COLOR_HIGHCONTRAST )
450          nIndex += 2;
451      return nIndex;
452  }
453  
454  ImageList* ImageManagerImpl::implts_getUserImageList( ImageType nImageType )
455  {
456      ResetableGuard aGuard( m_aLock );
457      if ( !m_pUserImageList[nImageType] )
458          implts_loadUserImages( nImageType, m_xUserImageStorage, m_xUserBitmapsStorage );
459  
460      return m_pUserImageList[nImageType];
461  }
462  
463  void ImageManagerImpl::implts_initialize()
464  {
465      // Initialize the top-level structures with the storage data
466      if ( m_xUserConfigStorage.is() )
467      {
468          long nModes = m_bReadOnly ? ElementModes::READ : ElementModes::READWRITE;
469  
470          try
471          {
472              m_xUserImageStorage = m_xUserConfigStorage->openStorageElement( OUString::createFromAscii( IMAGE_FOLDER ),
473                                                                              nModes );
474              if ( m_xUserImageStorage.is() )
475              {
476                  m_xUserBitmapsStorage = m_xUserImageStorage->openStorageElement( OUString::createFromAscii( BITMAPS_FOLDER ),
477                                                                                   nModes );
478              }
479          }
480          catch ( com::sun::star::container::NoSuchElementException& )
481          {
482          }
483          catch ( ::com::sun::star::embed::InvalidStorageException& )
484          {
485          }
486          catch ( ::com::sun::star::lang::IllegalArgumentException& )
487          {
488          }
489          catch ( ::com::sun::star::io::IOException& )
490          {
491          }
492          catch ( ::com::sun::star::embed::StorageWrappedTargetException& )
493          {
494          }
495      }
496  }
497  
498  sal_Bool ImageManagerImpl::implts_loadUserImages(
499      ImageType nImageType,
500      const uno::Reference< XStorage >& xUserImageStorage,
501      const uno::Reference< XStorage >& xUserBitmapsStorage )
502  {
503      ResetableGuard aGuard( m_aLock );
504  
505      if ( xUserImageStorage.is() && xUserBitmapsStorage.is() )
506      {
507          try
508          {
509              uno::Reference< XStream > xStream = xUserImageStorage->openStreamElement( rtl::OUString::createFromAscii( IMAGELIST_XML_FILE[nImageType] ),
510                                                                                        ElementModes::READ );
511              uno::Reference< XInputStream > xInputStream = xStream->getInputStream();
512  
513              ImageListsDescriptor aUserImageListInfo;
514              ImagesConfiguration::LoadImages( m_xServiceManager,
515                                               xInputStream,
516                                               aUserImageListInfo );
517              if (( aUserImageListInfo.pImageList != 0 ) &&
518                  ( aUserImageListInfo.pImageList->Count() > 0 ))
519              {
520                  ImageListItemDescriptor* pList = aUserImageListInfo.pImageList->GetObject(0);
521                  sal_Int32 nCount = pList->pImageItemList->Count();
522                  std::vector< OUString > aUserImagesVector;
523                  aUserImagesVector.reserve(nCount);
524                  for ( sal_uInt16 i=0; i < nCount; i++ )
525                  {
526                      const ImageItemDescriptor* pItem = pList->pImageItemList->GetObject(i);
527                      aUserImagesVector.push_back( pItem->aCommandURL );
528                  }
529  
530                  uno::Reference< XStream > xBitmapStream = xUserBitmapsStorage->openStreamElement(
531                                                          rtl::OUString::createFromAscii( BITMAP_FILE_NAMES[nImageType] ),
532                                                          ElementModes::READ );
533  
534                  if ( xBitmapStream.is() )
535                  {
536                      SvStream* pSvStream( 0 );
537  		            BitmapEx aUserBitmap;
538  		            {
539  		                pSvStream = utl::UcbStreamHelper::CreateStream( xBitmapStream );
540  		                vcl::PNGReader aPngReader( *pSvStream );
541  		                aUserBitmap = aPngReader.Read();
542  		            }
543                      delete pSvStream;
544  
545                      // Delete old image list and create a new one from the read bitmap
546                      delete m_pUserImageList[nImageType];
547                      m_pUserImageList[nImageType] = new ImageList();
548  					m_pUserImageList[nImageType]->InsertFromHorizontalStrip
549  						( aUserBitmap, aUserImagesVector );
550                      return sal_True;
551                  }
552              }
553          }
554          catch ( com::sun::star::container::NoSuchElementException& )
555          {
556          }
557          catch ( ::com::sun::star::embed::InvalidStorageException& )
558          {
559          }
560          catch ( ::com::sun::star::lang::IllegalArgumentException& )
561          {
562          }
563          catch ( ::com::sun::star::io::IOException& )
564          {
565          }
566          catch ( ::com::sun::star::embed::StorageWrappedTargetException& )
567          {
568          }
569      }
570  
571      // Destroy old image list - create a new empty one
572      delete m_pUserImageList[nImageType];
573      m_pUserImageList[nImageType] = new ImageList;
574  
575      return sal_True;
576  }
577  
578  sal_Bool ImageManagerImpl::implts_storeUserImages(
579      ImageType nImageType,
580      const uno::Reference< XStorage >& xUserImageStorage,
581      const uno::Reference< XStorage >& xUserBitmapsStorage )
582  {
583      ResetableGuard aGuard( m_aLock );
584  
585      if ( m_bModified )
586      {
587          ImageList* pImageList = implts_getUserImageList( nImageType );
588          if ( pImageList->GetImageCount() > 0 )
589          {
590              ImageListsDescriptor aUserImageListInfo;
591              aUserImageListInfo.pImageList = new ImageListDescriptor;
592  
593              ImageListItemDescriptor* pList = new ImageListItemDescriptor;
594              aUserImageListInfo.pImageList->Insert( pList, 0 );
595  
596              pList->pImageItemList = new ImageItemListDescriptor;
597              for ( sal_uInt16 i=0; i < pImageList->GetImageCount(); i++ )
598              {
599                  ImageItemDescriptor* pItem = new ::framework::ImageItemDescriptor;
600  
601                  pItem->nIndex = i;
602                  pItem->aCommandURL = pImageList->GetImageName( i );
603                  pList->pImageItemList->Insert( pItem, pList->pImageItemList->Count() );
604              }
605  
606              pList->aURL = String::CreateFromAscii("Bitmaps/");
607              pList->aURL += String::CreateFromAscii( BITMAP_FILE_NAMES[nImageType] );
608  
609              uno::Reference< XTransactedObject > xTransaction;
610              uno::Reference< XOutputStream >     xOutputStream;
611              uno::Reference< XStream > xStream = xUserImageStorage->openStreamElement( rtl::OUString::createFromAscii( IMAGELIST_XML_FILE[nImageType] ),
612                                                                                        ElementModes::WRITE|ElementModes::TRUNCATE );
613              if ( xStream.is() )
614              {
615                  uno::Reference< XStream > xBitmapStream =
616                      xUserBitmapsStorage->openStreamElement( rtl::OUString::createFromAscii( BITMAP_FILE_NAMES[nImageType] ),
617                                                              ElementModes::WRITE|ElementModes::TRUNCATE );
618                  if ( xBitmapStream.is() )
619                  {
620                      SvStream* pSvStream = utl::UcbStreamHelper::CreateStream( xBitmapStream );
621                      {
622                          vcl::PNGWriter aPngWriter( pImageList->GetAsHorizontalStrip() );
623                          aPngWriter.Write( *pSvStream );
624                      }
625                      delete pSvStream;
626  
627                      // Commit user bitmaps storage
628                      xTransaction = uno::Reference< XTransactedObject >( xUserBitmapsStorage, UNO_QUERY );
629  					if ( xTransaction.is() )
630                      	xTransaction->commit();
631                  }
632  
633                  xOutputStream = xStream->getOutputStream();
634                  if ( xOutputStream.is() )
635                      ImagesConfiguration::StoreImages( m_xServiceManager, xOutputStream, aUserImageListInfo );
636  
637                  // Commit user image storage
638                  xTransaction = uno::Reference< XTransactedObject >( xUserImageStorage, UNO_QUERY );
639  				if ( xTransaction.is() )
640                  	xTransaction->commit();
641              }
642  
643              return sal_True;
644          }
645          else
646          {
647              // Remove the streams from the storage, if we have no data. We have to catch
648              // the NoSuchElementException as it can be possible that there is no stream at all!
649              try
650              {
651                  xUserImageStorage->removeElement( rtl::OUString::createFromAscii( IMAGELIST_XML_FILE[nImageType] ));
652              }
653              catch ( ::com::sun::star::container::NoSuchElementException& )
654              {
655              }
656  
657              try
658              {
659                  xUserBitmapsStorage->removeElement( rtl::OUString::createFromAscii( BITMAP_FILE_NAMES[nImageType] ));
660              }
661              catch ( ::com::sun::star::container::NoSuchElementException& )
662              {
663              }
664  
665              uno::Reference< XTransactedObject > xTransaction;
666  
667              // Commit user image storage
668              xTransaction = uno::Reference< XTransactedObject >( xUserImageStorage, UNO_QUERY );
669  			if ( xTransaction.is() )
670                  xTransaction->commit();
671  
672              // Commit user bitmaps storage
673              xTransaction = uno::Reference< XTransactedObject >( xUserBitmapsStorage, UNO_QUERY );
674  			if ( xTransaction.is() )
675                  xTransaction->commit();
676  
677              return sal_True;
678          }
679      }
680  
681      return sal_False;
682  }
683  const rtl::Reference< GlobalImageList >& ImageManagerImpl::implts_getGlobalImageList()
684  {
685      ResetableGuard aGuard( m_aLock );
686  
687      if ( !m_pGlobalImageList.is() )
688          m_pGlobalImageList = getGlobalImageList( m_xServiceManager );
689      return m_pGlobalImageList;
690  }
691  
692  CmdImageList* ImageManagerImpl::implts_getDefaultImageList()
693  {
694      ResetableGuard aGuard( m_aLock );
695  
696      if ( !m_pDefaultImageList )
697          m_pDefaultImageList = new CmdImageList( m_xServiceManager, m_aModuleIdentifier );
698  
699      return m_pDefaultImageList;
700  }
701  
702  ImageManagerImpl::ImageManagerImpl( const uno::Reference< XMultiServiceFactory >& xServiceManager,const uno::Reference< XInterface >& _xOwner,bool _bUseGlobal ) :
703      ThreadHelpBase( &Application::GetSolarMutex() )
704      , m_xServiceManager( xServiceManager )
705      , m_xOwner(_xOwner)
706      , m_pDefaultImageList( 0 )
707      , m_aXMLPostfix( RTL_CONSTASCII_USTRINGPARAM( ".xml" ))
708      , m_aResourceString( RTL_CONSTASCII_USTRINGPARAM( ModuleImageList ))
709      , m_aListenerContainer( m_aLock.getShareableOslMutex() )
710      , m_bUseGlobal(_bUseGlobal)
711      , m_bReadOnly( true )
712      , m_bInitialized( false )
713      , m_bModified( false )
714      , m_bConfigRead( false )
715      , m_bDisposed( false )
716  {
717      for ( sal_Int32 n=0; n < ImageType_COUNT; n++ )
718      {
719          m_pUserImageList[n] = 0;
720          m_bUserImageListModified[n] = false;
721      }
722  }
723  
724  ImageManagerImpl::~ImageManagerImpl()
725  {
726      clear();
727  }
728  
729  void ImageManagerImpl::dispose()
730  {
731      css::lang::EventObject aEvent( m_xOwner );
732      m_aListenerContainer.disposeAndClear( aEvent );
733  
734  	{
735  	    ResetableGuard aGuard( m_aLock );
736          m_xUserConfigStorage.clear();
737          m_xUserImageStorage.clear();
738          m_xUserRootCommit.clear();
739          m_bConfigRead = false;
740          m_bModified = false;
741          m_bDisposed = true;
742  
743          // delete user and default image list on dispose
744          for ( sal_Int32 n=0; n < ImageType_COUNT; n++ )
745          {
746              delete m_pUserImageList[n];
747              m_pUserImageList[n] = 0;
748          }
749          delete m_pDefaultImageList;
750          m_pDefaultImageList = 0;
751      }
752  
753  }
754  void ImageManagerImpl::addEventListener( const uno::Reference< XEventListener >& xListener ) throw (::com::sun::star::uno::RuntimeException)
755  {
756      {
757          ResetableGuard aGuard( m_aLock );
758  
759  	    /* SAFE AREA ----------------------------------------------------------------------------------------------- */
760          if ( m_bDisposed )
761              throw DisposedException();
762      }
763  
764      m_aListenerContainer.addInterface( ::getCppuType( ( const uno::Reference< XEventListener >* ) NULL ), xListener );
765  }
766  
767  void ImageManagerImpl::removeEventListener( const uno::Reference< XEventListener >& xListener ) throw (::com::sun::star::uno::RuntimeException)
768  {
769      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
770      m_aListenerContainer.removeInterface( ::getCppuType( ( const uno::Reference< XEventListener >* ) NULL ), xListener );
771  }
772  
773  // XInitialization
774  void ImageManagerImpl::initialize( const Sequence< Any >& aArguments )
775  {
776      ResetableGuard aLock( m_aLock );
777  
778      if ( !m_bInitialized )
779      {
780          for ( sal_Int32 n = 0; n < aArguments.getLength(); n++ )
781          {
782              PropertyValue aPropValue;
783              if ( aArguments[n] >>= aPropValue )
784              {
785                  if ( aPropValue.Name.equalsAscii( "UserConfigStorage" ))
786                  {
787                      aPropValue.Value >>= m_xUserConfigStorage;
788                  }
789                  else if ( aPropValue.Name.equalsAscii( "ModuleIdentifier" ))
790                  {
791                      aPropValue.Value >>= m_aModuleIdentifier;
792                  }
793                  else if ( aPropValue.Name.equalsAscii( "UserRootCommit" ))
794                  {
795                      aPropValue.Value >>= m_xUserRootCommit;
796                  }
797              }
798          }
799  
800          if ( m_xUserConfigStorage.is() )
801          {
802              uno::Reference< XPropertySet > xPropSet( m_xUserConfigStorage, UNO_QUERY );
803              if ( xPropSet.is() )
804              {
805                  long nOpenMode = 0;
806                  if ( xPropSet->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "OpenMode" ))) >>= nOpenMode )
807                      m_bReadOnly = !( nOpenMode & ElementModes::WRITE );
808              }
809          }
810  
811          implts_initialize();
812  
813          m_bInitialized = true;
814      }
815  }
816  
817  // XImageManagerImpl
818  void ImageManagerImpl::reset()
819  throw (::com::sun::star::uno::RuntimeException)
820  {
821      ResetableGuard aLock( m_aLock );
822  
823      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
824      if ( m_bDisposed )
825          throw DisposedException();
826  
827      std::vector< OUString > aUserImageNames;
828  
829      for ( sal_Int32 i = 0; i < ImageType_COUNT; i++ )
830      {
831          aUserImageNames.clear();
832          ImageList* pImageList = implts_getUserImageList( ImageType(i));
833          pImageList->GetImageNames( aUserImageNames );
834  
835          Sequence< rtl::OUString > aRemoveList( aUserImageNames.size() );
836          const sal_uInt32 nCount = aUserImageNames.size();
837  		for ( sal_uInt32 j = 0; j < nCount; j++ )
838              aRemoveList[j] = aUserImageNames[j];
839  
840          // Remove images
841          removeImages( sal_Int16( i ), aRemoveList );
842          m_bUserImageListModified[i] = true;
843      }
844  
845      m_bModified = sal_True;
846  }
847  
848  Sequence< ::rtl::OUString > ImageManagerImpl::getAllImageNames( ::sal_Int16 nImageType )
849  throw (::com::sun::star::uno::RuntimeException)
850  {
851      ResetableGuard aLock( m_aLock );
852  
853      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
854      if ( m_bDisposed )
855          throw DisposedException();
856  
857      ImageNameMap aImageCmdNameMap;
858  
859      sal_Int16 nIndex = implts_convertImageTypeToIndex( nImageType );
860  
861      sal_uInt32 i( 0 );
862      if ( m_bUseGlobal )
863      {
864          rtl::Reference< GlobalImageList > rGlobalImageList = implts_getGlobalImageList();
865  
866          const std::vector< OUString >& rGlobalImageNameVector = rGlobalImageList->getImageCommandNames();
867          const sal_uInt32 nGlobalCount = rGlobalImageNameVector.size();
868          for ( i = 0; i < nGlobalCount; i++ )
869              aImageCmdNameMap.insert( ImageNameMap::value_type( rGlobalImageNameVector[i], sal_True ));
870  
871          const std::vector< OUString >& rModuleImageNameVector = implts_getDefaultImageList()->getImageCommandNames();
872          const sal_uInt32 nModuleCount = rModuleImageNameVector.size();
873          for ( i = 0; i < nModuleCount; i++ )
874              aImageCmdNameMap.insert( ImageNameMap::value_type( rModuleImageNameVector[i], sal_True ));
875      }
876  
877      ImageList* pImageList = implts_getUserImageList( ImageType( nIndex ));
878      std::vector< OUString > rUserImageNames;
879      pImageList->GetImageNames( rUserImageNames );
880      const sal_uInt32 nUserCount = rUserImageNames.size();
881      for ( i = 0; i < nUserCount; i++ )
882          aImageCmdNameMap.insert( ImageNameMap::value_type( rUserImageNames[i], sal_True ));
883  
884      Sequence< OUString > aImageNameSeq( aImageCmdNameMap.size() );
885      ImageNameMap::const_iterator pIter;
886      i = 0;
887      for ( pIter = aImageCmdNameMap.begin(); pIter != aImageCmdNameMap.end(); pIter++ )
888          aImageNameSeq[i++] = pIter->first;
889  
890      return aImageNameSeq;
891  }
892  
893  ::sal_Bool ImageManagerImpl::hasImage( ::sal_Int16 nImageType, const ::rtl::OUString& aCommandURL )
894  throw (::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::uno::RuntimeException)
895  {
896      ResetableGuard aLock( m_aLock );
897  
898      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
899      if ( m_bDisposed )
900          throw DisposedException();
901  
902      if (( nImageType < 0 ) || ( nImageType > MAX_IMAGETYPE_VALUE ))
903          throw IllegalArgumentException();
904  
905      sal_Int16 nIndex = implts_convertImageTypeToIndex( nImageType );
906      if ( m_bUseGlobal && implts_getGlobalImageList()->hasImage( nIndex, aCommandURL ))
907          return sal_True;
908      else
909      {
910          if ( m_bUseGlobal && implts_getDefaultImageList()->hasImage( nIndex, aCommandURL ))
911              return sal_True;
912          else
913          {
914              // User layer
915              ImageList* pImageList = implts_getUserImageList( ImageType( nIndex ));
916              if ( pImageList )
917                  return ( pImageList->GetImagePos( aCommandURL ) != IMAGELIST_IMAGE_NOTFOUND );
918          }
919      }
920  
921      return sal_False;
922  }
923  
924  Sequence< uno::Reference< XGraphic > > ImageManagerImpl::getImages(
925      ::sal_Int16 nImageType,
926      const Sequence< ::rtl::OUString >& aCommandURLSequence )
927  throw ( ::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::uno::RuntimeException )
928  {
929      ResetableGuard aLock( m_aLock );
930  
931      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
932      if ( m_bDisposed )
933          throw DisposedException();
934  
935      if (( nImageType < 0 ) || ( nImageType > MAX_IMAGETYPE_VALUE ))
936          throw IllegalArgumentException();
937  
938      Sequence< uno::Reference< XGraphic > > aGraphSeq( aCommandURLSequence.getLength() );
939  
940      const rtl::OUString* aStrArray = aCommandURLSequence.getConstArray();
941  
942      sal_Int16                         nIndex            = implts_convertImageTypeToIndex( nImageType );
943      rtl::Reference< GlobalImageList > rGlobalImageList;
944      CmdImageList*                     pDefaultImageList = NULL;
945      if ( m_bUseGlobal )
946      {
947          rGlobalImageList  = implts_getGlobalImageList();
948          pDefaultImageList = implts_getDefaultImageList();
949      }
950      ImageList*                        pUserImageList    = implts_getUserImageList( ImageType( nIndex ));
951  
952      // We have to search our image list in the following order:
953      // 1. user image list (read/write)
954      // 2. module image list (read)
955      // 3. global image list (read)
956      for ( sal_Int32 n = 0; n < aCommandURLSequence.getLength(); n++ )
957      {
958          Image aImage = pUserImageList->GetImage( aStrArray[n] );
959          if ( !aImage && m_bUseGlobal )
960          {
961              aImage = pDefaultImageList->getImageFromCommandURL( nIndex, aStrArray[n] );
962              if ( !aImage )
963                  aImage = rGlobalImageList->getImageFromCommandURL( nIndex, aStrArray[n] );
964          }
965  
966          aGraphSeq[n] = aImage.GetXGraphic();
967      }
968  
969      return aGraphSeq;
970  }
971  
972  void ImageManagerImpl::replaceImages(
973      ::sal_Int16 nImageType,
974      const Sequence< ::rtl::OUString >& aCommandURLSequence,
975      const Sequence< uno::Reference< XGraphic > >& aGraphicsSequence )
976  throw ( ::com::sun::star::lang::IllegalArgumentException,
977          ::com::sun::star::lang::IllegalAccessException,
978          ::com::sun::star::uno::RuntimeException)
979  {
980      CmdToXGraphicNameAccess* pInsertedImages( 0 );
981      CmdToXGraphicNameAccess* pReplacedImages( 0 );
982  
983      {
984          ResetableGuard aLock( m_aLock );
985  
986          /* SAFE AREA ----------------------------------------------------------------------------------------------- */
987          if ( m_bDisposed )
988              throw DisposedException();
989  
990          if (( aCommandURLSequence.getLength() != aGraphicsSequence.getLength() ) ||
991              (( nImageType < 0 ) || ( nImageType > MAX_IMAGETYPE_VALUE )))
992              throw IllegalArgumentException();
993  
994          if ( m_bReadOnly )
995              throw IllegalAccessException();
996  
997          sal_Int16 nIndex = implts_convertImageTypeToIndex( nImageType );
998          ImageList* pImageList = implts_getUserImageList( ImageType( nIndex ));
999  
1000          uno::Reference< XGraphic > xGraphic;
1001          for ( sal_Int32 i = 0; i < aCommandURLSequence.getLength(); i++ )
1002          {
1003              // Check size and scale. If we don't have any graphics ignore it
1004              if ( !implts_checkAndScaleGraphic( xGraphic, aGraphicsSequence[i], nIndex ))
1005                  continue;
1006  
1007              sal_uInt16 nPos = pImageList->GetImagePos( aCommandURLSequence[i] );
1008              if ( nPos == IMAGELIST_IMAGE_NOTFOUND )
1009              {
1010                  pImageList->AddImage( aCommandURLSequence[i], xGraphic );
1011                  if ( !pInsertedImages )
1012                      pInsertedImages = new CmdToXGraphicNameAccess();
1013                  pInsertedImages->addElement( aCommandURLSequence[i], xGraphic );
1014              }
1015              else
1016              {
1017                  pImageList->ReplaceImage( aCommandURLSequence[i], xGraphic );
1018                  if ( !pReplacedImages )
1019                      pReplacedImages = new CmdToXGraphicNameAccess();
1020                  pReplacedImages->addElement( aCommandURLSequence[i], xGraphic );
1021              }
1022          }
1023  
1024          if (( pInsertedImages != 0 ) || (  pReplacedImages != 0 ))
1025          {
1026              m_bModified = sal_True;
1027              m_bUserImageListModified[nIndex] = true;
1028          }
1029      }
1030  
1031      // Notify listeners
1032      if ( pInsertedImages != 0 )
1033      {
1034          ConfigurationEvent aInsertEvent;
1035          aInsertEvent.aInfo           <<= nImageType;
1036          aInsertEvent.Accessor        <<= m_xOwner;
1037          aInsertEvent.Source          = m_xOwner;
1038          aInsertEvent.ResourceURL     = m_aResourceString;
1039          aInsertEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1040                                          static_cast< OWeakObject *>( pInsertedImages ), UNO_QUERY ));
1041          implts_notifyContainerListener( aInsertEvent, NotifyOp_Insert );
1042      }
1043      if ( pReplacedImages != 0 )
1044      {
1045          ConfigurationEvent aReplaceEvent;
1046          aReplaceEvent.aInfo           <<= nImageType;
1047          aReplaceEvent.Accessor        <<= m_xOwner;
1048          aReplaceEvent.Source          = m_xOwner;
1049          aReplaceEvent.ResourceURL     = m_aResourceString;
1050          aReplaceEvent.ReplacedElement = Any();
1051          aReplaceEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1052                                              static_cast< OWeakObject *>( pReplacedImages ), UNO_QUERY ));
1053          implts_notifyContainerListener( aReplaceEvent, NotifyOp_Replace );
1054      }
1055  }
1056  
1057  void ImageManagerImpl::removeImages( ::sal_Int16 nImageType, const Sequence< ::rtl::OUString >& aCommandURLSequence )
1058  throw ( ::com::sun::star::lang::IllegalArgumentException,
1059          ::com::sun::star::lang::IllegalAccessException,
1060          ::com::sun::star::uno::RuntimeException)
1061  {
1062      CmdToXGraphicNameAccess* pRemovedImages( 0 );
1063      CmdToXGraphicNameAccess* pReplacedImages( 0 );
1064  
1065      {
1066          ResetableGuard aLock( m_aLock );
1067  
1068          /* SAFE AREA ----------------------------------------------------------------------------------------------- */
1069          if ( m_bDisposed )
1070              throw DisposedException();
1071  
1072          if (( nImageType < 0 ) || ( nImageType > MAX_IMAGETYPE_VALUE ))
1073              throw IllegalArgumentException();
1074  
1075          if ( m_bReadOnly )
1076              throw IllegalAccessException();
1077  
1078          sal_Int16 nIndex = implts_convertImageTypeToIndex( nImageType );
1079          rtl::Reference< GlobalImageList > rGlobalImageList;
1080          CmdImageList*                     pDefaultImageList = NULL;
1081          if ( m_bUseGlobal )
1082          {
1083              rGlobalImageList  = implts_getGlobalImageList();
1084              pDefaultImageList = implts_getDefaultImageList();
1085          }
1086          ImageList*                        pImageList        = implts_getUserImageList( ImageType( nIndex ));
1087          uno::Reference< XGraphic >        xEmptyGraphic( Image().GetXGraphic() );
1088  
1089          for ( sal_Int32 i = 0; i < aCommandURLSequence.getLength(); i++ )
1090          {
1091              sal_uInt16 nPos = pImageList->GetImagePos( aCommandURLSequence[i] );
1092              if ( nPos != IMAGELIST_IMAGE_NOTFOUND )
1093              {
1094                  Image aImage = pImageList->GetImage( nPos );
1095                  sal_uInt16 nId   = pImageList->GetImageId( nPos );
1096                  pImageList->RemoveImage( nId );
1097  
1098                  if ( m_bUseGlobal )
1099                  {
1100                      // Check, if we have a image in our module/global image list. If we find one =>
1101                      // this is a replace instead of a remove operation!
1102                      Image aNewImage = pDefaultImageList->getImageFromCommandURL( nIndex, aCommandURLSequence[i] );
1103                      if ( !aNewImage )
1104                          aNewImage = rGlobalImageList->getImageFromCommandURL( nIndex, aCommandURLSequence[i] );
1105                      if ( !aNewImage )
1106                      {
1107                          if ( !pRemovedImages )
1108                              pRemovedImages = new CmdToXGraphicNameAccess();
1109                          pRemovedImages->addElement( aCommandURLSequence[i], xEmptyGraphic );
1110                      }
1111                      else
1112                      {
1113                          if ( !pReplacedImages )
1114                              pReplacedImages = new CmdToXGraphicNameAccess();
1115                          pReplacedImages->addElement( aCommandURLSequence[i], aNewImage.GetXGraphic() );
1116                      }
1117                  } // if ( m_bUseGlobal )
1118                  else
1119                  {
1120                      if ( !pRemovedImages )
1121                          pRemovedImages = new CmdToXGraphicNameAccess();
1122                      pRemovedImages->addElement( aCommandURLSequence[i], xEmptyGraphic );
1123                  }
1124              }
1125          }
1126  
1127          if (( pReplacedImages != 0 ) || ( pRemovedImages != 0 ))
1128          {
1129              m_bModified = sal_True;
1130              m_bUserImageListModified[nIndex] = true;
1131          }
1132      }
1133  
1134      // Notify listeners
1135      if ( pRemovedImages != 0 )
1136      {
1137          ConfigurationEvent aRemoveEvent;
1138          aRemoveEvent.aInfo           = uno::makeAny( nImageType );
1139          aRemoveEvent.Accessor        = uno::makeAny( m_xOwner );
1140          aRemoveEvent.Source          = m_xOwner;
1141          aRemoveEvent.ResourceURL     = m_aResourceString;
1142          aRemoveEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1143                                              static_cast< OWeakObject *>( pRemovedImages ), UNO_QUERY ));
1144          implts_notifyContainerListener( aRemoveEvent, NotifyOp_Remove );
1145      }
1146      if ( pReplacedImages != 0 )
1147      {
1148          ConfigurationEvent aReplaceEvent;
1149          aReplaceEvent.aInfo           = uno::makeAny( nImageType );
1150          aReplaceEvent.Accessor        = uno::makeAny( m_xOwner );
1151          aReplaceEvent.Source          = m_xOwner;
1152          aReplaceEvent.ResourceURL     = m_aResourceString;
1153          aReplaceEvent.ReplacedElement = Any();
1154          aReplaceEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1155                                              static_cast< OWeakObject *>( pReplacedImages ), UNO_QUERY ));
1156          implts_notifyContainerListener( aReplaceEvent, NotifyOp_Replace );
1157      }
1158  }
1159  
1160  void ImageManagerImpl::insertImages( ::sal_Int16 nImageType, const Sequence< ::rtl::OUString >& aCommandURLSequence, const Sequence< uno::Reference< XGraphic > >& aGraphicSequence )
1161  throw ( ::com::sun::star::container::ElementExistException,
1162          ::com::sun::star::lang::IllegalArgumentException,
1163          ::com::sun::star::lang::IllegalAccessException,
1164          ::com::sun::star::uno::RuntimeException)
1165  {
1166      replaceImages(nImageType,aCommandURLSequence,aGraphicSequence);
1167  }
1168  
1169  
1170  // XUIConfigurationPersistence
1171  void ImageManagerImpl::reload()
1172  throw ( ::com::sun::star::uno::Exception,
1173          ::com::sun::star::uno::RuntimeException )
1174  {
1175      ResetableGuard aGuard( m_aLock );
1176  
1177      if ( m_bDisposed )
1178          throw DisposedException();
1179  
1180      CommandMap                   aOldUserCmdImageSet;
1181      std::vector< rtl::OUString > aNewUserCmdImageSet;
1182  
1183      if ( m_bModified )
1184      {
1185          for ( sal_Int16 i = 0; i < sal_Int16( ImageType_COUNT ); i++ )
1186          {
1187              if ( !m_bDisposed && m_bUserImageListModified[i] )
1188              {
1189                  std::vector< rtl::OUString > aOldUserCmdImageVector;
1190                  ImageList* pImageList = implts_getUserImageList( (ImageType)i );
1191                  pImageList->GetImageNames( aOldUserCmdImageVector );
1192  
1193                  // Fill hash map to speed up search afterwards
1194                  sal_uInt32 j( 0 );
1195                  const sal_uInt32 nOldCount = aOldUserCmdImageVector.size();
1196  				for ( j = 0; j < nOldCount; j++ )
1197                      aOldUserCmdImageSet.insert( CommandMap::value_type( aOldUserCmdImageVector[j], false ));
1198  
1199                  // Attention: This can make the old image list pointer invalid!
1200                  implts_loadUserImages( (ImageType)i, m_xUserImageStorage, m_xUserBitmapsStorage );
1201                  pImageList = implts_getUserImageList( (ImageType)i );
1202                  pImageList->GetImageNames( aNewUserCmdImageSet );
1203  
1204                  CmdToXGraphicNameAccess* pInsertedImages( 0 );
1205                  CmdToXGraphicNameAccess* pReplacedImages( 0 );
1206                  CmdToXGraphicNameAccess* pRemovedImages( 0 );
1207  
1208                  const sal_uInt32 nNewCount = aNewUserCmdImageSet.size();
1209  				for ( j = 0; j < nNewCount; j++ )
1210                  {
1211                      CommandMap::iterator pIter = aOldUserCmdImageSet.find( aNewUserCmdImageSet[j] );
1212                      if ( pIter != aOldUserCmdImageSet.end() )
1213                      {
1214                          pIter->second = true; // mark entry as replaced
1215                          if ( !pReplacedImages )
1216                              pReplacedImages = new CmdToXGraphicNameAccess();
1217                          pReplacedImages->addElement( aNewUserCmdImageSet[j],
1218                                                       pImageList->GetImage( aNewUserCmdImageSet[j] ).GetXGraphic() );
1219                      }
1220                      else
1221                      {
1222                          if ( !pInsertedImages )
1223                              pInsertedImages = new CmdToXGraphicNameAccess();
1224                          pInsertedImages->addElement( aNewUserCmdImageSet[j],
1225                                                       pImageList->GetImage( aNewUserCmdImageSet[j] ).GetXGraphic() );
1226                      }
1227                  }
1228  
1229                  // Search map for unmarked entries => they have been removed from the user list
1230                  // through this reload operation.
1231                  // We have to search the module and global image list!
1232                  rtl::Reference< GlobalImageList > rGlobalImageList;
1233                  CmdImageList*                     pDefaultImageList = NULL;
1234                  if ( m_bUseGlobal )
1235                  {
1236                      rGlobalImageList  = implts_getGlobalImageList();
1237                      pDefaultImageList = implts_getDefaultImageList();
1238                  }
1239                  uno::Reference< XGraphic >        xEmptyGraphic( Image().GetXGraphic() );
1240                  CommandMap::const_iterator        pIter = aOldUserCmdImageSet.begin();
1241                  while ( pIter != aOldUserCmdImageSet.end() )
1242                  {
1243                      if ( !pIter->second )
1244                      {
1245                          if ( m_bUseGlobal )
1246                          {
1247                              Image aImage = pDefaultImageList->getImageFromCommandURL( i, pIter->first );
1248                              if ( !aImage )
1249                                  aImage = rGlobalImageList->getImageFromCommandURL( i, pIter->first );
1250  
1251                              if ( !aImage )
1252                              {
1253                                  // No image in the module/global image list => remove user image
1254                                  if ( !pRemovedImages )
1255                                      pRemovedImages = new CmdToXGraphicNameAccess();
1256                                  pRemovedImages->addElement( pIter->first, xEmptyGraphic );
1257                              }
1258                              else
1259                              {
1260                                  // Image has been found in the module/global image list => replace user image
1261                                  if ( !pReplacedImages )
1262                                      pReplacedImages = new CmdToXGraphicNameAccess();
1263                                  pReplacedImages->addElement( pIter->first, aImage.GetXGraphic() );
1264                              }
1265                          } // if ( m_bUseGlobal )
1266                          else
1267                          {
1268                              // No image in the user image list => remove user image
1269                              if ( !pRemovedImages )
1270                                  pRemovedImages = new CmdToXGraphicNameAccess();
1271                              pRemovedImages->addElement( pIter->first, xEmptyGraphic );
1272                          }
1273                      }
1274                      ++pIter;
1275                  }
1276  
1277                  aGuard.unlock();
1278  
1279                  // Now notify our listeners. Unlock mutex to prevent deadlocks
1280                  if ( pInsertedImages != 0 )
1281                  {
1282                      ConfigurationEvent aInsertEvent;
1283                      aInsertEvent.aInfo           = uno::makeAny( i );
1284                      aInsertEvent.Accessor        = uno::makeAny( m_xOwner );
1285                      aInsertEvent.Source          = m_xOwner;
1286                      aInsertEvent.ResourceURL     = m_aResourceString;
1287                      aInsertEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1288                                                      static_cast< OWeakObject *>( pInsertedImages ), UNO_QUERY ));
1289                      implts_notifyContainerListener( aInsertEvent, NotifyOp_Insert );
1290                  }
1291                  if ( pReplacedImages != 0 )
1292                  {
1293                      ConfigurationEvent aReplaceEvent;
1294                      aReplaceEvent.aInfo           = uno::makeAny( i );
1295                      aReplaceEvent.Accessor        = uno::makeAny( m_xOwner );
1296                      aReplaceEvent.Source          = m_xOwner;
1297                      aReplaceEvent.ResourceURL     = m_aResourceString;
1298                      aReplaceEvent.ReplacedElement = Any();
1299                      aReplaceEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1300                                                      static_cast< OWeakObject *>( pReplacedImages ), UNO_QUERY ));
1301                      implts_notifyContainerListener( aReplaceEvent, NotifyOp_Replace );
1302                  }
1303                  if ( pRemovedImages != 0 )
1304                  {
1305                      ConfigurationEvent aRemoveEvent;
1306                      aRemoveEvent.aInfo           = uno::makeAny( i );
1307                      aRemoveEvent.Accessor        = uno::makeAny( m_xOwner );
1308                      aRemoveEvent.Source          = m_xOwner;
1309                      aRemoveEvent.ResourceURL     = m_aResourceString;
1310                      aRemoveEvent.Element         = uno::makeAny( uno::Reference< XNameAccess >(
1311                                                          static_cast< OWeakObject *>( pRemovedImages ), UNO_QUERY ));
1312                      implts_notifyContainerListener( aRemoveEvent, NotifyOp_Remove );
1313                  }
1314  
1315                  aGuard.lock();
1316              }
1317          }
1318      }
1319  }
1320  
1321  void ImageManagerImpl::store()
1322  throw (::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException)
1323  {
1324      ResetableGuard aGuard( m_aLock );
1325  
1326      if ( m_bDisposed )
1327          throw DisposedException();
1328  
1329      if ( m_bModified )
1330      {
1331          sal_Bool bWritten( sal_False );
1332          for ( sal_Int32 i = 0; i < ImageType_COUNT; i++ )
1333          {
1334              sal_Bool bSuccess = implts_storeUserImages( ImageType(i), m_xUserImageStorage, m_xUserBitmapsStorage );
1335              if ( bSuccess )
1336                  bWritten = sal_True;
1337              m_bUserImageListModified[i] = false;
1338          }
1339  
1340          if ( bWritten &&
1341               m_xUserConfigStorage.is() )
1342          {
1343              uno::Reference< XTransactedObject > xUserConfigStorageCommit( m_xUserConfigStorage, UNO_QUERY );
1344  			if ( xUserConfigStorageCommit.is() )
1345              	xUserConfigStorageCommit->commit();
1346              if ( m_xUserRootCommit.is() )
1347                  m_xUserRootCommit->commit();
1348          }
1349  
1350          m_bModified = sal_False;
1351      }
1352  }
1353  
1354  void ImageManagerImpl::storeToStorage( const uno::Reference< XStorage >& Storage )
1355  throw (::com::sun::star::uno::Exception, ::com::sun::star::uno::RuntimeException)
1356  {
1357      ResetableGuard aGuard( m_aLock );
1358  
1359      if ( m_bDisposed )
1360          throw DisposedException();
1361  
1362      if ( m_bModified && Storage.is() )
1363      {
1364          long nModes = ElementModes::READWRITE;
1365  
1366          uno::Reference< XStorage > xUserImageStorage = Storage->openStorageElement( OUString::createFromAscii( IMAGE_FOLDER ),
1367                                                                                      nModes );
1368          if ( xUserImageStorage.is() )
1369          {
1370              uno::Reference< XStorage > xUserBitmapsStorage = xUserImageStorage->openStorageElement( OUString::createFromAscii( BITMAPS_FOLDER ),
1371                                                                                                      nModes );
1372              for ( sal_Int32 i = 0; i < ImageType_COUNT; i++ )
1373              {
1374                  implts_getUserImageList( (ImageType)i );
1375                  implts_storeUserImages( (ImageType)i, xUserImageStorage, xUserBitmapsStorage );
1376              }
1377  
1378              uno::Reference< XTransactedObject > xTransaction( Storage, UNO_QUERY );
1379  			if ( xTransaction.is() )
1380              	xTransaction->commit();
1381          }
1382      }
1383  }
1384  
1385  sal_Bool ImageManagerImpl::isModified()
1386  throw (::com::sun::star::uno::RuntimeException)
1387  {
1388      ResetableGuard aGuard( m_aLock );
1389      return m_bModified;
1390  }
1391  
1392  sal_Bool ImageManagerImpl::isReadOnly() throw (::com::sun::star::uno::RuntimeException)
1393  {
1394      ResetableGuard aGuard( m_aLock );
1395      return m_bReadOnly;
1396  }
1397  // XUIConfiguration
1398  void ImageManagerImpl::addConfigurationListener( const uno::Reference< ::com::sun::star::ui::XUIConfigurationListener >& xListener )
1399  throw (::com::sun::star::uno::RuntimeException)
1400  {
1401      {
1402          ResetableGuard aGuard( m_aLock );
1403  
1404          /* SAFE AREA ----------------------------------------------------------------------------------------------- */
1405          if ( m_bDisposed )
1406              throw DisposedException();
1407      }
1408  
1409      m_aListenerContainer.addInterface( ::getCppuType( ( const uno::Reference< XUIConfigurationListener >* ) NULL ), xListener );
1410  }
1411  
1412  void ImageManagerImpl::removeConfigurationListener( const uno::Reference< ::com::sun::star::ui::XUIConfigurationListener >& xListener )
1413  throw (::com::sun::star::uno::RuntimeException)
1414  {
1415      /* SAFE AREA ----------------------------------------------------------------------------------------------- */
1416      m_aListenerContainer.removeInterface( ::getCppuType( ( const uno::Reference< XUIConfigurationListener >* ) NULL ), xListener );
1417  }
1418  
1419  
1420  void ImageManagerImpl::implts_notifyContainerListener( const ConfigurationEvent& aEvent, NotifyOp eOp )
1421  {
1422      ::cppu::OInterfaceContainerHelper* pContainer = m_aListenerContainer.getContainer(
1423                                          ::getCppuType( ( const css::uno::Reference< ::com::sun::star::ui::XUIConfigurationListener >*) NULL ) );
1424      if ( pContainer != NULL )
1425  	{
1426          ::cppu::OInterfaceIteratorHelper pIterator( *pContainer );
1427          while ( pIterator.hasMoreElements() )
1428          {
1429              try
1430              {
1431                  switch ( eOp )
1432                  {
1433                      case NotifyOp_Replace:
1434                          ((::com::sun::star::ui::XUIConfigurationListener*)pIterator.next())->elementReplaced( aEvent );
1435                          break;
1436                      case NotifyOp_Insert:
1437                          ((::com::sun::star::ui::XUIConfigurationListener*)pIterator.next())->elementInserted( aEvent );
1438                          break;
1439                      case NotifyOp_Remove:
1440                          ((::com::sun::star::ui::XUIConfigurationListener*)pIterator.next())->elementRemoved( aEvent );
1441                          break;
1442                  }
1443              }
1444              catch( css::uno::RuntimeException& )
1445              {
1446                  pIterator.remove();
1447              }
1448          }
1449  	}
1450  }
1451  void ImageManagerImpl::clear()
1452  {
1453      ResetableGuard aGuard( m_aLock );
1454      for ( sal_Int32 n = 0; n < ImageType_COUNT; n++ )
1455      {
1456          delete m_pUserImageList[n];
1457          m_pUserImageList[n] = 0;
1458      }
1459  }
1460  } // namespace framework
1461