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 
28 //_________________________________________________________________________________________________________________
29 //	my own includes
30 //_________________________________________________________________________________________________________________
31 #include <uielement/menubarmanager.hxx>
32 #include <framework/menuconfiguration.hxx>
33 #include <framework/bmkmenu.hxx>
34 #include <framework/addonmenu.hxx>
35 #include <framework/imageproducer.hxx>
36 #include <threadhelp/resetableguard.hxx>
37 #include "framework/addonsoptions.hxx"
38 #include <classes/fwkresid.hxx>
39 #include <classes/menumanager.hxx>
40 #include <framework/acceleratorinfo.hxx>
41 #include <helper/mischelper.hxx>
42 #include <framework/menuextensionsupplier.hxx>
43 #include <classes/resource.hrc>
44 #include <services.h>
45 
46 //_________________________________________________________________________________________________________________
47 //	interface includes
48 //_________________________________________________________________________________________________________________
49 #include <com/sun/star/frame/XDispatch.hpp>
50 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
51 #include <com/sun/star/lang/DisposedException.hpp>
52 #include <com/sun/star/beans/XPropertySet.hpp>
53 #include <com/sun/star/frame/XFramesSupplier.hpp>
54 #include <com/sun/star/frame/XDesktop.hpp>
55 #include <com/sun/star/container/XEnumeration.hpp>
56 #include <com/sun/star/util/XStringWidth.hpp>
57 #include <com/sun/star/uno/XComponentContext.hpp>
58 #include <com/sun/star/lang/XMultiComponentFactory.hpp>
59 #include <com/sun/star/frame/XPopupMenuController.hpp>
60 #include <com/sun/star/frame/XUIControllerRegistration.hpp>
61 #ifndef _COM_SUN_STAR_LANG_XSYSTEMDEPENDENT_HPP_
62 #include <com/sun/star/lang/SystemDependent.hpp>
63 #endif
64 #include <com/sun/star/ui/ItemType.hpp>
65 #include <com/sun/star/ui/ImageType.hpp>
66 #include <com/sun/star/container/XNameAccess.hpp>
67 #include <com/sun/star/frame/XModuleManager.hpp>
68 #include <com/sun/star/ui/XModuleUIConfigurationManagerSupplier.hpp>
69 #include <com/sun/star/ui/XUIConfigurationManagerSupplier.hpp>
70 #include <com/sun/star/ui/ItemStyle.hpp>
71 #include <com/sun/star/frame/status/Visibility.hpp>
72 
73 //_________________________________________________________________________________________________________________
74 //	includes of other projects
75 //_________________________________________________________________________________________________________________
76 #include <comphelper/processfactory.hxx>
77 #include <comphelper/extract.hxx>
78 #include <svtools/menuoptions.hxx>
79 #include <unotools/historyoptions.hxx>
80 #include <unotools/pathoptions.hxx>
81 #include <unotools/cmdoptions.hxx>
82 #include <unotools/localfilehelper.hxx>
83 #ifndef _TOOLKIT_HELPER_VCLUNOHELPER_HXX_
84 #include <toolkit/unohlp.hxx>
85 #endif
86 #include <tools/urlobj.hxx>
87 #include <vcl/svapp.hxx>
88 #include <vcl/window.hxx>
89 #include <vos/mutex.hxx>
90 #include <vcl/svapp.hxx>
91 #include <osl/file.hxx>
92 #include <cppuhelper/implbase1.hxx>
93 #include <svtools/acceleratorexecute.hxx>
94 #include <rtl/logfile.hxx>
95 #include "svtools/miscopt.hxx"
96 #include <framework/addonmenu.hxx>
97 #include <uielement/menubarmerger.hxx>
98 #include <dispatch/uieventloghelper.hxx>
99 
100 // Be careful removing this "bad" construct. There are serious problems
101 // with #define STRICT and including windows.h. Changing this needs some
102 // redesign on other projects, too. Especially sal/main.h which defines
103 // HINSTANCE depending on STRCIT!!!!!!!!!!!!!!!
104 struct SystemMenuData
105 {
106     unsigned long nSize;
107 	long          hMenu;
108 };
109 
110 //_________________________________________________________________________________________________________________
111 //	namespace
112 //_________________________________________________________________________________________________________________
113 
114 using namespace ::cppu;
115 using namespace ::vos;
116 using namespace ::com::sun::star;
117 using namespace ::com::sun::star::uno;
118 using namespace ::com::sun::star::util;
119 using namespace ::com::sun::star::beans;
120 using namespace ::com::sun::star::frame;
121 using namespace ::com::sun::star::container;
122 using namespace ::com::sun::star::lang;
123 using namespace ::com::sun::star::frame;
124 using namespace ::com::sun::star::ui;
125 
126 static const char ITEM_DESCRIPTOR_COMMANDURL[]        = "CommandURL";
127 static const char ITEM_DESCRIPTOR_HELPURL[]           = "HelpURL";
128 static const char ITEM_DESCRIPTOR_CONTAINER[]         = "ItemDescriptorContainer";
129 static const char ITEM_DESCRIPTOR_LABEL[]             = "Label";
130 static const char ITEM_DESCRIPTOR_TYPE[]              = "Type";
131 static const char ITEM_DESCRIPTOR_MODULEIDENTIFIER[]  = "ModuleIdentifier";
132 static const char ITEM_DESCRIPTOR_DISPATCHPROVIDER[]  = "DispatchProvider";
133 static const char ITEM_DESCRIPTOR_STYLE[]             = "Style";
134 static const char ITEM_DESCRIPTOR_ISVISIBLE[]         = "IsVisible";
135 static const char ITEM_DESCRIPTOR_ENABLED[]           = "Enabled";
136 
137 static const sal_Int32 LEN_DESCRIPTOR_COMMANDURL       = 10;
138 static const sal_Int32 LEN_DESCRIPTOR_HELPURL          = 7;
139 static const sal_Int32 LEN_DESCRIPTOR_CONTAINER        = 23;
140 static const sal_Int32 LEN_DESCRIPTOR_LABEL            = 5;
141 static const sal_Int32 LEN_DESCRIPTOR_TYPE             = 4;
142 static const sal_Int32 LEN_DESCRIPTOR_MODULEIDENTIFIER = 16;
143 static const sal_Int32 LEN_DESCRIPTOR_DISPATCHPROVIDER = 16;
144 static const sal_Int32 LEN_DESCRIPTOR_STYLE            = 5;
145 static const sal_Int32 LEN_DESCRIPTOR_ISVISIBLE        = 9;
146 static const sal_Int32 LEN_DESCRIPTOR_ENABLED          = 7;
147 
148 const sal_uInt16 ADDONMENU_MERGE_ITEMID_START = 1500;
149 
150 class StringLength : public ::cppu::WeakImplHelper1< ::com::sun::star::util::XStringWidth >
151 {
152 	public:
153 		StringLength() {}
154 		virtual ~StringLength() {}
155 
156 		// XStringWidth
157 		sal_Int32 SAL_CALL queryStringWidth( const ::rtl::OUString& aString )
158 			throw (RuntimeException)
159 		{
160 			return aString.getLength();
161 		}
162 };
163 
164 namespace framework
165 {
166 
167 // special menu ids/command ids for dynamic popup menus
168 #define SID_SFX_START			5000
169 #define SID_NEWDOCDIRECT		(SID_SFX_START + 537)
170 #define SID_AUTOPILOTMENU		(SID_SFX_START + 1381)
171 #define SID_PICKLIST			(SID_SFX_START + 510)
172 #define SID_MDIWINDOWLIST		(SID_SFX_START + 610)
173 #define SID_ADDONLIST			(SID_SFX_START + 1677)
174 #define SID_HELPMENU			(SID_SFX_START + 410)
175 
176 #define SFX_REFERER_USER		"private:user"
177 
178 const ::rtl::OUString aCmdHelpIndex( RTL_CONSTASCII_USTRINGPARAM( ".uno:HelpIndex" ));
179 const ::rtl::OUString aCmdToolsMenu( RTL_CONSTASCII_USTRINGPARAM( ".uno:ToolsMenu" ));
180 const ::rtl::OUString aCmdHelpMenu( RTL_CONSTASCII_USTRINGPARAM( ".uno:HelpMenu" ));
181 const ::rtl::OUString aSlotHelpMenu( RTL_CONSTASCII_USTRINGPARAM( "slot:5410" ));
182 
183 const ::rtl::OUString aSpecialFileMenu( RTL_CONSTASCII_USTRINGPARAM( "file" ));
184 const ::rtl::OUString aSpecialWindowMenu( RTL_CONSTASCII_USTRINGPARAM( "window" ));
185 const ::rtl::OUString aSlotSpecialFileMenu( RTL_CONSTASCII_USTRINGPARAM( "slot:5510" ));
186 const ::rtl::OUString aSlotSpecialWindowMenu( RTL_CONSTASCII_USTRINGPARAM( "slot:5610" ));
187 const ::rtl::OUString aSlotSpecialToolsMenu( RTL_CONSTASCII_USTRINGPARAM( "slot:6677" ));
188 
189 // special uno commands for picklist and window list
190 const ::rtl::OUString aSpecialFileCommand( RTL_CONSTASCII_USTRINGPARAM( ".uno:PickList" ));
191 const ::rtl::OUString aSpecialWindowCommand( RTL_CONSTASCII_USTRINGPARAM( ".uno:WindowList" ));
192 
193 const ::rtl::OUString UNO_COMMAND( RTL_CONSTASCII_USTRINGPARAM( ".uno:" ));
194 
195 static sal_Int16 getImageTypeFromBools( sal_Bool bBig, sal_Bool bHighContrast )
196 {
197     sal_Int16 n( 0 );
198     if ( bBig )
199         n |= ::com::sun::star::ui::ImageType::SIZE_LARGE;
200     if ( bHighContrast )
201         n |= ::com::sun::star::ui::ImageType::COLOR_HIGHCONTRAST;
202     return n;
203 }
204 
205 // #110897#
206 MenuBarManager::MenuBarManager(
207 	const Reference< XMultiServiceFactory >& xServiceFactory,
208 	const Reference< XFrame >& rFrame,
209     const Reference< XURLTransformer >& _xURLTransformer,
210     const Reference< XDispatchProvider >& rDispatchProvider,
211     const rtl::OUString& rModuleIdentifier,
212     Menu* pMenu, sal_Bool bDelete, sal_Bool bDeleteChildren )
213 : ThreadHelpBase( &Application::GetSolarMutex() ), OWeakObject()
214     , m_bDisposed( sal_False )
215     , m_bRetrieveImages( sal_False )
216     , m_bAcceleratorCfg( sal_False )
217     , m_bModuleIdentified( sal_False )
218     , m_aListenerContainer( m_aLock.getShareableOslMutex() )
219     , mxServiceFactory(xServiceFactory)
220     , m_xURLTransformer(_xURLTransformer)
221     , m_nSymbolsStyle( SvtMiscOptions().GetCurrentSymbolsStyle() )
222 {
223     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::MenuBarManager" );
224     m_xPopupMenuControllerRegistration = Reference< ::com::sun::star::frame::XUIControllerRegistration >(
225 		getServiceFactory()->createInstance( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.frame.PopupMenuControllerFactory" ))),
226 		UNO_QUERY );
227 	FillMenuManager( pMenu, rFrame, rDispatchProvider, rModuleIdentifier, bDelete, bDeleteChildren );
228 }
229 
230 // #110897#
231 MenuBarManager::MenuBarManager(
232 	const Reference< XMultiServiceFactory >& xServiceFactory,
233 	const Reference< XFrame >& rFrame,
234     const Reference< XURLTransformer >& _xURLTransformer,
235     AddonMenu* pAddonMenu,
236     sal_Bool bDelete,
237     sal_Bool bDeleteChildren )
238 :   ThreadHelpBase( &Application::GetSolarMutex() )
239     , OWeakObject()
240     , m_bDisposed( sal_False )
241     , m_bRetrieveImages( sal_True )
242     , m_bAcceleratorCfg( sal_False )
243     , m_bModuleIdentified( sal_False )
244     , m_aListenerContainer( m_aLock.getShareableOslMutex() )
245     , mxServiceFactory(xServiceFactory)
246     , m_xURLTransformer(_xURLTransformer)
247     , m_nSymbolsStyle( SvtMiscOptions().GetCurrentSymbolsStyle() )
248 {
249     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::MenuBarManager" );
250     Init(rFrame,pAddonMenu,bDelete,bDeleteChildren);
251 }
252 
253 // #110897#
254 MenuBarManager::MenuBarManager(
255 	const Reference< XMultiServiceFactory >& xServiceFactory,
256 	const Reference< XFrame >& rFrame,
257     const Reference< XURLTransformer >& _xURLTransformer,
258     AddonPopupMenu* pAddonPopupMenu,
259     sal_Bool bDelete,
260     sal_Bool bDeleteChildren )
261 :     ThreadHelpBase( &Application::GetSolarMutex() )
262     , OWeakObject()
263     , m_bDisposed( sal_False )
264     , m_bRetrieveImages( sal_True )
265     , m_bAcceleratorCfg( sal_False )
266     , m_bModuleIdentified( sal_False )
267     , m_aListenerContainer( m_aLock.getShareableOslMutex() )
268     , mxServiceFactory(xServiceFactory)
269     , m_xURLTransformer(_xURLTransformer)
270     , m_nSymbolsStyle( SvtMiscOptions().GetCurrentSymbolsStyle() )
271 {
272     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::MenuBarManager" );
273     Init(rFrame,pAddonPopupMenu,bDelete,bDeleteChildren,true);
274 }
275 
276 Any SAL_CALL MenuBarManager::queryInterface( const Type & rType ) throw ( RuntimeException )
277 {
278 	Any a = ::cppu::queryInterface(
279 				rType ,
280 				SAL_STATIC_CAST( ::com::sun::star::frame::XStatusListener*, this ),
281 				SAL_STATIC_CAST( ::com::sun::star::frame::XFrameActionListener*, this ),
282                 SAL_STATIC_CAST( ::com::sun::star::ui::XUIConfigurationListener*, this ),
283 				SAL_STATIC_CAST( XEventListener*, (XStatusListener *)this ),
284 				SAL_STATIC_CAST( XComponent*, this ),
285 				SAL_STATIC_CAST( ::com::sun::star::awt::XSystemDependentMenuPeer*, this ));
286 
287 	if ( a.hasValue() )
288 		return a;
289 
290 	return OWeakObject::queryInterface( rType );
291 }
292 
293 
294 void SAL_CALL MenuBarManager::acquire() throw()
295 {
296     OWeakObject::acquire();
297 }
298 
299 
300 void SAL_CALL MenuBarManager::release() throw()
301 {
302     OWeakObject::release();
303 }
304 
305 
306 Any SAL_CALL MenuBarManager::getMenuHandle( const Sequence< sal_Int8 >& /*ProcessId*/, sal_Int16 SystemType ) throw (RuntimeException)
307 {
308     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::getMenuHandle" );
309 	ResetableGuard aGuard( m_aLock );
310 
311 	if ( m_bDisposed )
312 	    throw com::sun::star::lang::DisposedException();
313 
314     Any a;
315 
316     if ( m_pVCLMenu )
317     {
318         OGuard	aSolarGuard( Application::GetSolarMutex() );
319 
320         SystemMenuData aSystemMenuData;
321         aSystemMenuData.nSize = sizeof( SystemMenuData );
322 
323         m_pVCLMenu->GetSystemMenuData( &aSystemMenuData );
324 #ifdef QUARTZ
325         if( SystemType == SystemDependent::SYSTEM_MAC )
326         {
327         }
328 #elif (defined WNT)
329 		if( SystemType == SystemDependent::SYSTEM_WIN32 )
330 		{
331             a <<= (long) aSystemMenuData.hMenu;
332 		}
333 #elif (defined UNX)
334 		if( SystemType == SystemDependent::SYSTEM_XWINDOW )
335 		{
336 		}
337 #endif
338     }
339 
340     return a;
341 }
342 
343 MenuBarManager::~MenuBarManager()
344 {
345     // stop asynchronous settings timer
346     m_xDeferedItemContainer.clear();
347     m_aAsyncSettingsTimer.Stop();
348 
349     DBG_ASSERT( OWeakObject::m_refCount == 0, "Who wants to delete an object with refcount > 0!" );
350 }
351 
352 void MenuBarManager::Destroy()
353 {
354     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::Destroy" );
355     OGuard	aGuard( Application::GetSolarMutex() );
356 
357     if ( !m_bDisposed )
358     {
359         // stop asynchronous settings timer and
360         // release defered item container reference
361         m_aAsyncSettingsTimer.Stop();
362         m_xDeferedItemContainer.clear();
363         RemoveListener();
364 
365         std::vector< MenuItemHandler* >::iterator p;
366 	    for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
367 	    {
368             MenuItemHandler* pItemHandler = *p;
369 		    pItemHandler->xMenuItemDispatch.clear();
370 		    pItemHandler->xSubMenuManager.clear();
371 	        pItemHandler->xPopupMenu.clear();
372 		    delete pItemHandler;
373 	    }
374         m_aMenuItemHandlerVector.clear();
375 
376 	    if ( m_bDeleteMenu )
377         {
378 		    delete m_pVCLMenu;
379             m_pVCLMenu = 0;
380         }
381     }
382 }
383 
384 // XComponent
385 void SAL_CALL MenuBarManager::dispose() throw( RuntimeException )
386 {
387     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::dispose" );
388     Reference< XComponent > xThis( static_cast< OWeakObject* >(this), UNO_QUERY );
389 
390     EventObject aEvent( xThis );
391     m_aListenerContainer.disposeAndClear( aEvent );
392 
393 	{
394 	    ResetableGuard aGuard( m_aLock );
395 //        RemoveListener();
396         Destroy();
397         m_bDisposed = sal_True;
398 
399         if ( m_xDocImageManager.is() )
400         {
401             try
402             {
403                 m_xDocImageManager->removeConfigurationListener(
404                     Reference< XUIConfigurationListener >(
405                         static_cast< OWeakObject* >( this ), UNO_QUERY ));
406             }
407             catch ( Exception& )
408             {
409             }
410         }
411         if ( m_xModuleImageManager.is() )
412         {
413             try
414             {
415                 m_xModuleImageManager->removeConfigurationListener(
416                     Reference< XUIConfigurationListener >(
417                         static_cast< OWeakObject* >( this ), UNO_QUERY ));
418             }
419             catch ( Exception& )
420             {
421             }
422         }
423         m_xDocImageManager.clear();
424         m_xModuleImageManager.clear();
425         m_xGlobalAcceleratorManager.clear();
426         m_xModuleAcceleratorManager.clear();
427         m_xDocAcceleratorManager.clear();
428         m_xUICommandLabels.clear();
429         m_xPopupMenuControllerRegistration.clear();
430         mxServiceFactory.clear();
431     }
432 }
433 
434 void SAL_CALL MenuBarManager::addEventListener( const Reference< XEventListener >& xListener ) throw( RuntimeException )
435 {
436     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::addEventListener" );
437 	ResetableGuard aGuard( m_aLock );
438 
439 	/* SAFE AREA ----------------------------------------------------------------------------------------------- */
440     if ( m_bDisposed )
441         throw DisposedException();
442 
443     m_aListenerContainer.addInterface( ::getCppuType( ( const Reference< XEventListener >* ) NULL ), xListener );
444 }
445 
446 void SAL_CALL MenuBarManager::removeEventListener( const Reference< XEventListener >& xListener ) throw( RuntimeException )
447 {
448     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::removeEventListener" );
449 	ResetableGuard aGuard( m_aLock );
450 	/* SAFE AREA ----------------------------------------------------------------------------------------------- */
451     m_aListenerContainer.removeInterface( ::getCppuType( ( const Reference< XEventListener >* ) NULL ), xListener );
452 }
453 
454 void SAL_CALL MenuBarManager::elementInserted( const ::com::sun::star::ui::ConfigurationEvent& Event )
455 throw (RuntimeException)
456 {
457     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::elementInserted" );
458 	ResetableGuard aGuard( m_aLock );
459 
460     /* SAFE AREA ----------------------------------------------------------------------------------------------- */
461     if ( m_bDisposed )
462         return;
463 
464     sal_Int16 nImageType = sal_Int16();
465     sal_Int16 nCurrentImageType = getImageTypeFromBools( sal_False, m_bWasHiContrast );
466     if (( Event.aInfo >>= nImageType ) &&
467         ( nImageType == nCurrentImageType ))
468         RequestImages();
469 }
470 
471 void SAL_CALL MenuBarManager::elementRemoved( const ::com::sun::star::ui::ConfigurationEvent& Event )
472 throw (RuntimeException)
473 {
474     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::elementRemoved" );
475 	elementInserted(Event);
476 }
477 
478 void SAL_CALL MenuBarManager::elementReplaced( const ::com::sun::star::ui::ConfigurationEvent& Event )
479 throw (RuntimeException)
480 {
481     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::elementReplaced" );
482 	elementInserted(Event);
483 }
484 
485 // XFrameActionListener
486 void SAL_CALL MenuBarManager::frameAction( const FrameActionEvent& Action )
487 throw ( RuntimeException )
488 {
489     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::frameAction" );
490     ResetableGuard aGuard( m_aLock );
491 
492 	if ( m_bDisposed )
493 	    throw com::sun::star::lang::DisposedException();
494 
495     if ( Action.Action == FrameAction_CONTEXT_CHANGED )
496     {
497         std::vector< MenuItemHandler* >::iterator p;
498 	    for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
499 	    {
500             // Clear dispatch reference as we will requery it later o
501             MenuItemHandler* pItemHandler = *p;
502 		    pItemHandler->xMenuItemDispatch.clear();
503         }
504     }
505 }
506 
507 // XStatusListener
508 void SAL_CALL MenuBarManager::statusChanged( const FeatureStateEvent& Event )
509 throw ( RuntimeException )
510 {
511     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::statusChanged" );
512 	::rtl::OUString aFeatureURL = Event.FeatureURL.Complete;
513 
514     OGuard	aSolarGuard( Application::GetSolarMutex() );
515 	{
516 		ResetableGuard aGuard( m_aLock );
517 
518 		if ( m_bDisposed )
519 		    return;
520 
521         // We have to check all menu entries as there can be identical entries in a popup menu.
522         std::vector< MenuItemHandler* >::iterator p;
523 		for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
524 		{
525 			MenuItemHandler* pMenuItemHandler = *p;
526 			if ( pMenuItemHandler->aMenuItemURL == aFeatureURL )
527 		    {
528 			    sal_Bool            bCheckmark( sal_False );
529 			    sal_Bool            bMenuItemEnabled( m_pVCLMenu->IsItemEnabled( pMenuItemHandler->nItemId ));
530                 sal_Bool            bEnabledItem( Event.IsEnabled );
531                 rtl::OUString       aItemText;
532                 status::Visibility  aVisibilityStatus;
533 
534                 #ifdef UNIX
535                 // #b6673979# enable some slots hardly, because UNIX clipboard does not notify all changes
536                 // Can be removed if follow up task will be fixed directly within applications.
537                 if (
538                     ( pMenuItemHandler->aMenuItemURL.equalsAscii (".uno:Paste"         ) ) ||
539                     ( pMenuItemHandler->aMenuItemURL.equalsAscii (".uno:PasteSpecial"  ) ) ||
540                     ( pMenuItemHandler->aMenuItemURL.equalsAscii (".uno:PasteClipboard") )      // special for draw/impress
541                    )
542                     bEnabledItem = sal_True;
543                 #endif
544 
545                 // Enable/disable item
546 			    if ( bEnabledItem != bMenuItemEnabled )
547 			        m_pVCLMenu->EnableItem( pMenuItemHandler->nItemId, bEnabledItem );
548 
549 			    if ( Event.State >>= bCheckmark )
550                 {
551                     // Checkmark or RadioButton
552                     m_pVCLMenu->ShowItem( pMenuItemHandler->nItemId, sal_True );
553                     m_pVCLMenu->CheckItem( pMenuItemHandler->nItemId, bCheckmark );
554 
555                     MenuItemBits nBits = m_pVCLMenu->GetItemBits( pMenuItemHandler->nItemId );
556                     //If not already designated RadioButton set as CheckMark
557                     if (!(nBits & MIB_RADIOCHECK))
558                         m_pVCLMenu->SetItemBits( pMenuItemHandler->nItemId, nBits | MIB_CHECKABLE );
559                 }
560                 else if ( Event.State >>= aItemText )
561                 {
562                     // Replacement for place holders
563                     if ( aItemText.matchAsciiL( "($1)", 4 ))
564                     {
565 					    String aResStr = String( FwkResId( STR_UPDATEDOC ));
566                         rtl::OUString aTmp( aResStr );
567                         aTmp += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( " " ));
568                         aTmp += aItemText.copy( 4 );
569                         aItemText = aTmp;
570                     }
571                     else if ( aItemText.matchAsciiL( "($2)", 4 ))
572                     {
573 					    String aResStr = String( FwkResId( STR_CLOSEDOC_ANDRETURN ));
574                         rtl::OUString aTmp( aResStr );
575                         aTmp += aItemText.copy( 4 );
576                         aItemText = aTmp;
577                     }
578                     else if ( aItemText.matchAsciiL( "($3)", 4 ))
579                     {
580 					    String aResStr = String( FwkResId( STR_SAVECOPYDOC ));
581                         rtl::OUString aTmp( aResStr );
582                         aTmp += aItemText.copy( 4 );
583                         aItemText = aTmp;
584                     }
585 
586                     m_pVCLMenu->ShowItem( pMenuItemHandler->nItemId, sal_True );
587                     m_pVCLMenu->SetItemText( pMenuItemHandler->nItemId, aItemText );
588                 }
589                 else if ( Event.State >>= aVisibilityStatus )
590                 {
591                     // Visibility
592                     m_pVCLMenu->ShowItem( pMenuItemHandler->nItemId, aVisibilityStatus.bVisible );
593                 }
594                 else
595                     m_pVCLMenu->ShowItem( pMenuItemHandler->nItemId, sal_True );
596 		    }
597 
598 		    if ( Event.Requery )
599 		    {
600                 // Release dispatch object - will be requeried on the next activate!
601                 pMenuItemHandler->xMenuItemDispatch.clear();
602 		    }
603         }
604 	}
605 }
606 
607 // Helper to retrieve own structure from item ID
608 MenuBarManager::MenuItemHandler* MenuBarManager::GetMenuItemHandler( sal_uInt16 nItemId )
609 {
610     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::GetMenuItemHandler" );
611 	ResetableGuard aGuard( m_aLock );
612 
613 	std::vector< MenuItemHandler* >::iterator p;
614 	for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
615 	{
616 		MenuItemHandler* pItemHandler = *p;
617 		if ( pItemHandler->nItemId == nItemId )
618 			return pItemHandler;
619 	}
620 
621 	return 0;
622 }
623 
624 // Helper to set request images flag
625 void MenuBarManager::RequestImages()
626 {
627     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::RequestImages" );
628     // must be locked from callee
629 	// ResetableGuard aGuard( m_aLock );
630 
631     m_bRetrieveImages = sal_True;
632     const sal_uInt32 nCount = m_aMenuItemHandlerVector.size();
633     for ( sal_uInt32 i = 0; i < nCount; ++i )
634     {
635 		MenuItemHandler* pItemHandler = m_aMenuItemHandlerVector[i];
636 		if ( pItemHandler->xSubMenuManager.is() )
637         {
638             MenuBarManager* pMenuBarManager = (MenuBarManager*)(pItemHandler->xSubMenuManager.get());
639             pMenuBarManager->RequestImages();
640         }
641 	}
642 }
643 
644 // Helper to reset objects to prepare shutdown
645 void MenuBarManager::RemoveListener()
646 {
647     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::RemoveListener" );
648 	ResetableGuard aGuard( m_aLock );
649 
650     // Check service manager reference. Remove listener can be called due
651     // to a disposing call from the frame and therefore we already removed
652     // our listeners and release the service manager reference!
653     Reference< XMultiServiceFactory > xServiceManager = getServiceFactory();
654     if ( xServiceManager.is() )
655     {
656 	    std::vector< MenuItemHandler* >::iterator p;
657 	    for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
658 	    {
659 		    MenuItemHandler* pItemHandler = *p;
660 		    if ( pItemHandler->xMenuItemDispatch.is() )
661 		    {
662 			    URL aTargetURL;
663 			    aTargetURL.Complete	= pItemHandler->aMenuItemURL;
664 			    m_xURLTransformer->parseStrict( aTargetURL );
665 
666 			    pItemHandler->xMenuItemDispatch->removeStatusListener(
667 				    static_cast< XStatusListener* >( this ), aTargetURL );
668 		    }
669 
670 		    pItemHandler->xMenuItemDispatch.clear();
671 		    if ( pItemHandler->xPopupMenu.is() )
672             {
673 		        {
674                     // Remove popup menu from menu structure
675 		            OGuard	aGuard2( Application::GetSolarMutex() );
676                     m_pVCLMenu->SetPopupMenu( pItemHandler->nItemId, 0 );
677                 }
678 
679                 Reference< com::sun::star::lang::XEventListener > xEventListener( pItemHandler->xPopupMenuController, UNO_QUERY );
680                 if ( xEventListener.is() )
681                 {
682                     EventObject aEventObject;
683                     aEventObject.Source = (OWeakObject *)this;
684                     xEventListener->disposing( aEventObject );
685                 }
686 
687                 // We now provide a popup menu controller to external code.
688                 // Therefore the life-time must be explicitly handled via
689                 // dispose!!
690                 try
691                 {
692                     Reference< XComponent > xComponent( pItemHandler->xPopupMenuController, UNO_QUERY );
693                     if ( xComponent.is() )
694                         xComponent->dispose();
695                 }
696                 catch ( RuntimeException& )
697                 {
698                     throw;
699                 }
700                 catch ( Exception& )
701                 {
702                 }
703 
704                 // Release references to controller and popup menu
705                 pItemHandler->xPopupMenuController.clear();
706                 pItemHandler->xPopupMenu.clear();
707             }
708 
709 		    Reference< XComponent > xComponent( pItemHandler->xSubMenuManager, UNO_QUERY );
710 		    if ( xComponent.is() )
711 		        xComponent->dispose();
712 	    }
713     }
714 
715     try
716     {
717         if ( m_xFrame.is() )
718             m_xFrame->removeFrameActionListener( Reference< XFrameActionListener >(
719                                                     static_cast< OWeakObject* >( this ), UNO_QUERY ));
720     }
721     catch ( Exception& )
722     {
723     }
724 
725 	m_xFrame = 0;
726 }
727 
728 void SAL_CALL MenuBarManager::disposing( const EventObject& Source ) throw ( RuntimeException )
729 {
730     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::disposing(evt)" );
731     MenuItemHandler* pMenuItemDisposing = NULL;
732 
733     ResetableGuard aGuard( m_aLock );
734 
735 	std::vector< MenuItemHandler* >::iterator p;
736 	for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
737 	{
738 		MenuItemHandler* pMenuItemHandler = *p;
739 		if ( pMenuItemHandler->xMenuItemDispatch.is() &&
740              pMenuItemHandler->xMenuItemDispatch == Source.Source )
741 		{
742 	        // disposing called from menu item dispatcher, remove listener
743 			pMenuItemDisposing = pMenuItemHandler;
744 			break;
745 		}
746 	}
747 
748     if ( pMenuItemDisposing )
749 	{
750         // Release references to the dispatch object
751 		URL aTargetURL;
752 		aTargetURL.Complete	= pMenuItemDisposing->aMenuItemURL;
753 
754         // Check reference of service manager before we use it. Reference could
755         // be cleared due to RemoveListener call!
756         Reference< XMultiServiceFactory > xServiceManager( getServiceFactory() );
757         if ( xServiceManager.is() )
758         {
759 		    m_xURLTransformer->parseStrict( aTargetURL );
760 
761 		    pMenuItemDisposing->xMenuItemDispatch->removeStatusListener(
762 			    static_cast< XStatusListener* >( this ), aTargetURL );
763 		    pMenuItemDisposing->xMenuItemDispatch = Reference< XDispatch >();
764 		    if ( pMenuItemDisposing->xPopupMenu.is() )
765             {
766                 Reference< com::sun::star::lang::XEventListener > xEventListener( pMenuItemDisposing->xPopupMenuController, UNO_QUERY );
767                 if ( xEventListener.is() )
768                     xEventListener->disposing( Source );
769 
770                 {
771                     // Remove popup menu from menu structure as we release our reference to
772                     // the controller.
773 		            OGuard	aGuard2( Application::GetSolarMutex() );
774                     m_pVCLMenu->SetPopupMenu( pMenuItemDisposing->nItemId, 0 );
775                 }
776 
777                 pMenuItemDisposing->xPopupMenuController.clear();
778                 pMenuItemDisposing->xPopupMenu.clear();
779             }
780         }
781         return;
782 	}
783 	else if ( Source.Source == m_xFrame )
784 	{
785         // Our frame gets disposed. We have to remove all our listeners
786 	    RemoveListener();
787 	}
788     else if ( Source.Source == Reference< XInterface >( m_xDocImageManager, UNO_QUERY ))
789         m_xDocImageManager.clear();
790     else if ( Source.Source == Reference< XInterface >( m_xModuleImageManager, UNO_QUERY ))
791         m_xModuleImageManager.clear();
792 }
793 
794 
795 void MenuBarManager::CheckAndAddMenuExtension( Menu* pMenu )
796 {
797     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::CheckAndAddMenuExtension" );
798     static const char REFERENCECOMMAND_AFTER[]          = ".uno:OnlineRegistrationDlg";
799     static const char REFERENCECOMMAND_BEFORE[]         = ".uno:About";
800 
801     // retrieve menu extension item
802     MenuExtensionItem aMenuItem( GetMenuExtension() );
803     if (( aMenuItem.aURL.getLength() > 0 ) &&
804         ( aMenuItem.aLabel.getLength() > 0 ))
805     {
806         // remove all old window list entries from menu
807         sal_uInt16 nNewItemId( 0 );
808         sal_uInt16 nInsertPos( MENU_APPEND );
809         sal_uInt16 nAfterPos( MENU_APPEND );
810         sal_uInt16 nBeforePos( MENU_APPEND );
811         String     aCommandAfter( String::CreateFromAscii ( REFERENCECOMMAND_AFTER ));
812         String     aCommandBefore( String::CreateFromAscii ( REFERENCECOMMAND_BEFORE ));
813         for ( sal_uInt16 n = 0; n < pMenu->GetItemCount(); n++ )
814         {
815             sal_uInt16 nItemId = pMenu->GetItemId( n );
816             nNewItemId = std::max( nItemId, nNewItemId );
817             if ( pMenu->GetItemCommand( nItemId ) == aCommandAfter )
818                 nAfterPos = n+1;
819             else if ( pMenu->GetItemCommand( nItemId ) == aCommandBefore )
820                 nBeforePos = n;
821         }
822         ++nNewItemId;
823 
824         if ( nAfterPos != MENU_APPEND )
825             nInsertPos = nAfterPos;
826         else if ( nBeforePos != MENU_APPEND )
827             nInsertPos = nBeforePos;
828 
829         pMenu->InsertItem( nNewItemId, aMenuItem.aLabel, 0, nInsertPos );
830         pMenu->SetItemCommand( nNewItemId, aMenuItem.aURL );
831     }
832 }
833 
834 static void lcl_CheckForChildren(Menu* pMenu, sal_uInt16 nItemId)
835 {
836     if (PopupMenu* pThisPopup = pMenu->GetPopupMenu( nItemId ))
837         pMenu->EnableItem( nItemId, pThisPopup->GetItemCount() ? true : false );
838 }
839 
840 //_________________________________________________________________________________________________________________
841 // vcl handler
842 //_________________________________________________________________________________________________________________
843 
844 IMPL_LINK( MenuBarManager, Activate, Menu *, pMenu )
845 {
846     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::Activate" );
847 	if ( pMenu == m_pVCLMenu )
848 	{
849 		// set/unset hiding disabled menu entries
850 		sal_Bool bDontHide			 = SvtMenuOptions().IsEntryHidingEnabled();
851 		const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
852 		sal_Bool bShowMenuImages	 = rSettings.GetUseImagesInMenus();
853         sal_Bool bHasDisabledEntries = SvtCommandOptions().HasEntries( SvtCommandOptions::CMDOPTION_DISABLED );
854 
855         ResetableGuard aGuard( m_aLock );
856 
857 		sal_uInt16 nFlag = pMenu->GetMenuFlags();
858 		if ( bDontHide )
859 			nFlag &= ~MENU_FLAG_HIDEDISABLEDENTRIES;
860 		else
861 			nFlag |= MENU_FLAG_HIDEDISABLEDENTRIES;
862 		pMenu->SetMenuFlags( nFlag );
863 
864 		if ( m_bActive )
865 			return 0;
866 
867 		m_bActive = sal_True;
868 
869 		::rtl::OUString aMenuCommand( m_aMenuItemCommand );
870         if ( m_aMenuItemCommand == aSpecialWindowMenu ||
871              m_aMenuItemCommand == aSlotSpecialWindowMenu ||
872 			 aMenuCommand == aSpecialWindowCommand )
873              MenuManager::UpdateSpecialWindowMenu( pMenu,getServiceFactory(),m_aLock );
874 
875 		// Check if some modes have changed so we have to update our menu images
876 		sal_Bool bIsHiContrast = rSettings.GetHighContrastMode();
877 		sal_Int16 nSymbolsStyle = SvtMiscOptions().GetCurrentSymbolsStyle();
878 
879 		if ( m_bRetrieveImages ||
880              m_bWasHiContrast != bIsHiContrast ||
881              bShowMenuImages != m_bShowMenuImages ||
882              nSymbolsStyle != m_nSymbolsStyle )
883 		{
884 			// The mode changed so we have to replace all images
885 			m_bWasHiContrast	= bIsHiContrast;
886 			m_bShowMenuImages	= bShowMenuImages;
887 			m_bRetrieveImages	= sal_False;
888 			m_nSymbolsStyle		= nSymbolsStyle;
889             MenuManager::FillMenuImages(m_xFrame,pMenu,bIsHiContrast,bShowMenuImages);
890 		}
891 
892         // Try to map commands to labels
893         for ( sal_uInt16 nPos = 0; nPos < pMenu->GetItemCount(); nPos++ )
894         {
895             sal_uInt16 nItemId = pMenu->GetItemId( nPos );
896             if (( pMenu->GetItemType( nPos ) != MENUITEM_SEPARATOR ) &&
897                 ( pMenu->GetItemText( nItemId ).Len() == 0 ))
898             {
899                 String aCommand = pMenu->GetItemCommand( nItemId );
900                 if ( aCommand.Len() > 0 )
901                     pMenu->SetItemText( nItemId, RetrieveLabelFromCommand( aCommand ));
902             }
903         }
904 
905         // Try to set accelerator keys
906         {
907             RetrieveShortcuts( m_aMenuItemHandlerVector );
908             std::vector< MenuItemHandler* >::iterator p;
909 		    for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
910 		    {
911 		        MenuItemHandler* pMenuItemHandler = *p;
912 
913                 // Set key code, workaround for hard-coded shortcut F1 mapped to .uno:HelpIndex
914                 // Only non-popup menu items can have a short-cut
915                 if ( pMenuItemHandler->aMenuItemURL == aCmdHelpIndex )
916                 {
917                     KeyCode aKeyCode( KEY_F1 );
918                     pMenu->SetAccelKey( pMenuItemHandler->nItemId, aKeyCode );
919                 }
920                 else if ( pMenu->GetPopupMenu( pMenuItemHandler->nItemId ) == 0 )
921                     pMenu->SetAccelKey( pMenuItemHandler->nItemId, pMenuItemHandler->aKeyCode );
922             }
923         }
924 
925 		URL aTargetURL;
926 
927         // Use provided dispatch provider => fallback to frame as dispatch provider
928         Reference< XDispatchProvider > xDispatchProvider;
929         if ( m_xDispatchProvider.is() )
930             xDispatchProvider = m_xDispatchProvider;
931         else
932             xDispatchProvider = Reference< XDispatchProvider >( m_xFrame, UNO_QUERY );
933 
934 		if ( xDispatchProvider.is() )
935 		{
936             KeyCode             aEmptyKeyCode;
937             SvtCommandOptions   aCmdOptions;
938 			std::vector< MenuItemHandler* >::iterator p;
939 			for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
940 			{
941 				MenuItemHandler* pMenuItemHandler = *p;
942 				if ( pMenuItemHandler )
943                 {
944                     if ( !pMenuItemHandler->xMenuItemDispatch.is() &&
945                          !pMenuItemHandler->xSubMenuManager.is()      )
946                     {
947 					    // There is no dispatch mechanism for the special window list menu items,
948 					    // because they are handled directly through XFrame->activate!!!
949                         // Don't update dispatches for special file menu items.
950 					    if ( !(( pMenuItemHandler->nItemId >= START_ITEMID_WINDOWLIST &&
951 					             pMenuItemHandler->nItemId < END_ITEMID_WINDOWLIST )))
952 					    {
953                             Reference< XDispatch > xMenuItemDispatch;
954 
955                             ::rtl::OUString aItemCommand = pMenu->GetItemCommand( pMenuItemHandler->nItemId );
956 						    if ( !aItemCommand.getLength() )
957 						    {
958 							    aItemCommand = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "slot:" ));
959 							    aItemCommand += ::rtl::OUString::valueOf( (sal_Int32)pMenuItemHandler->nItemId );
960 							    pMenu->SetItemCommand( pMenuItemHandler->nItemId, aItemCommand );
961 						    }
962 
963                             aTargetURL.Complete = aItemCommand;
964 
965                             m_xURLTransformer->parseStrict( aTargetURL );
966 
967                             if ( bHasDisabledEntries )
968                             {
969                                 if ( aCmdOptions.Lookup( SvtCommandOptions::CMDOPTION_DISABLED, aTargetURL.Path ))
970                                     pMenu->HideItem( pMenuItemHandler->nItemId );
971                             }
972 
973 						    if ( m_bIsBookmarkMenu )
974 							    xMenuItemDispatch = xDispatchProvider->queryDispatch( aTargetURL, pMenuItemHandler->aTargetFrame, 0 );
975 						    else
976 							    xMenuItemDispatch = xDispatchProvider->queryDispatch( aTargetURL, ::rtl::OUString(), 0 );
977 
978                             sal_Bool bPopupMenu( sal_False );
979                             if ( !pMenuItemHandler->xPopupMenuController.is() &&
980                                  m_xPopupMenuControllerRegistration->hasController( aItemCommand, rtl::OUString() ))
981                             {
982                                 bPopupMenu = CreatePopupMenuController( pMenuItemHandler );
983                             }
984                             else if ( pMenuItemHandler->xPopupMenuController.is() )
985                             {
986                                 // Force update of popup menu
987                                 pMenuItemHandler->xPopupMenuController->updatePopupMenu();
988                                 bPopupMenu = sal_True;
989 								if (PopupMenu*  pThisPopup = pMenu->GetPopupMenu( pMenuItemHandler->nItemId ))
990                                     pMenu->EnableItem( pMenuItemHandler->nItemId, pThisPopup->GetItemCount() ? true : false );
991                             }
992 
993 							lcl_CheckForChildren(pMenu, pMenuItemHandler->nItemId);
994 
995                             if ( xMenuItemDispatch.is() )
996 						    {
997 							    pMenuItemHandler->xMenuItemDispatch = xMenuItemDispatch;
998 							    pMenuItemHandler->aMenuItemURL		= aTargetURL.Complete;
999 
1000                                 if ( !bPopupMenu )
1001                                 {
1002                                     // We need only an update to reflect the current state
1003                                     xMenuItemDispatch->addStatusListener( static_cast< XStatusListener* >( this ), aTargetURL );
1004                                     xMenuItemDispatch->removeStatusListener( static_cast< XStatusListener* >( this ), aTargetURL );
1005                                 }
1006 						    }
1007 						    else if ( !bPopupMenu )
1008 							    pMenu->EnableItem( pMenuItemHandler->nItemId, sal_False );
1009 					    }
1010 				    }
1011                     else if ( pMenuItemHandler->xPopupMenuController.is() )
1012                     {
1013                         // Force update of popup menu
1014                         pMenuItemHandler->xPopupMenuController->updatePopupMenu();
1015 						lcl_CheckForChildren(pMenu, pMenuItemHandler->nItemId);
1016                     }
1017                     else if ( pMenuItemHandler->xMenuItemDispatch.is() )
1018                     {
1019                         // We need an update to reflect the current state
1020                         try
1021                         {
1022                             aTargetURL.Complete = pMenuItemHandler->aMenuItemURL;
1023                             m_xURLTransformer->parseStrict( aTargetURL );
1024 
1025                             pMenuItemHandler->xMenuItemDispatch->addStatusListener(
1026                                                                     static_cast< XStatusListener* >( this ), aTargetURL );
1027                             pMenuItemHandler->xMenuItemDispatch->removeStatusListener(
1028                                                                     static_cast< XStatusListener* >( this ), aTargetURL );
1029                         }
1030                         catch ( Exception& )
1031                         {
1032                         }
1033                     }
1034                     else if ( pMenuItemHandler->xSubMenuManager.is() )
1035 						lcl_CheckForChildren(pMenu, pMenuItemHandler->nItemId);
1036                 }
1037 			}
1038 		}
1039 	}
1040 
1041 	return 1;
1042 }
1043 
1044 
1045 IMPL_LINK( MenuBarManager, Deactivate, Menu *, pMenu )
1046 {
1047     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::Deactivate" );
1048 	if ( pMenu == m_pVCLMenu )
1049     {
1050         m_bActive = sal_False;
1051         if ( pMenu->IsMenuBar() && m_xDeferedItemContainer.is() )
1052         {
1053             // Start timer to handle settings asynchronous
1054             // Changing the menu inside this handler leads to
1055             // a crash under X!
1056             m_aAsyncSettingsTimer.SetTimeoutHdl(LINK(this, MenuBarManager, AsyncSettingsHdl));
1057             m_aAsyncSettingsTimer.SetTimeout(10);
1058             m_aAsyncSettingsTimer.Start();
1059         }
1060     }
1061 
1062 	return 1;
1063 }
1064 
1065 IMPL_LINK( MenuBarManager, AsyncSettingsHdl, Timer*,)
1066 {
1067     OGuard	aGuard( Application::GetSolarMutex() );
1068     Reference< XInterface > xSelfHold(
1069         static_cast< ::cppu::OWeakObject* >( this ), UNO_QUERY_THROW );
1070 
1071     m_aAsyncSettingsTimer.Stop();
1072     if ( !m_bActive && m_xDeferedItemContainer.is() )
1073     {
1074         SetItemContainer( m_xDeferedItemContainer );
1075         m_xDeferedItemContainer.clear();
1076     }
1077 
1078 	return 0;
1079 }
1080 
1081 IMPL_LINK( MenuBarManager, Select, Menu *, pMenu )
1082 {
1083     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::Select" );
1084 	URL						aTargetURL;
1085 	Sequence<PropertyValue>	aArgs;
1086 	Reference< XDispatch >	xDispatch;
1087 
1088 	{
1089 		ResetableGuard aGuard( m_aLock );
1090 
1091 		sal_uInt16 nCurItemId = pMenu->GetCurItemId();
1092         sal_uInt16 nCurPos    = pMenu->GetItemPos( nCurItemId );
1093 		if ( pMenu == m_pVCLMenu &&
1094 			 pMenu->GetItemType( nCurPos ) != MENUITEM_SEPARATOR )
1095 		{
1096 			if ( nCurItemId >= START_ITEMID_WINDOWLIST &&
1097 				 nCurItemId <= END_ITEMID_WINDOWLIST )
1098 			{
1099 				// window list menu item selected
1100 
1101 				// #110897#
1102                 // Reference< XFramesSupplier > xDesktop( ::comphelper::getProcessServiceFactory()->createInstance( DESKTOP_SERVICE ), UNO_QUERY );
1103                 Reference< XFramesSupplier > xDesktop( getServiceFactory()->createInstance( SERVICENAME_DESKTOP ), UNO_QUERY );
1104 
1105 				if ( xDesktop.is() )
1106 				{
1107 					sal_uInt16 nTaskId = START_ITEMID_WINDOWLIST;
1108                     Reference< XIndexAccess > xList( xDesktop->getFrames(), UNO_QUERY );
1109                     sal_Int32 nCount = xList->getCount();
1110                     for ( sal_Int32 i=0; i<nCount; ++i )
1111 					{
1112                         Reference< XFrame > xFrame;
1113                         xList->getByIndex(i) >>= xFrame;
1114                         if ( xFrame.is() && nTaskId == nCurItemId )
1115 						{
1116                             Window* pWin = VCLUnoHelper::GetWindow( xFrame->getContainerWindow() );
1117 							pWin->GrabFocus();
1118 							pWin->ToTop( TOTOP_RESTOREWHENMIN );
1119 							break;
1120 						}
1121 
1122 						nTaskId++;
1123 					}
1124 				}
1125 			}
1126 			else
1127 			{
1128 				MenuItemHandler* pMenuItemHandler = GetMenuItemHandler( nCurItemId );
1129 				if ( pMenuItemHandler && pMenuItemHandler->xMenuItemDispatch.is() )
1130 				{
1131 					aTargetURL.Complete = pMenuItemHandler->aMenuItemURL;
1132                     m_xURLTransformer->parseStrict( aTargetURL );
1133 
1134                     if ( m_bIsBookmarkMenu )
1135 					{
1136 						// bookmark menu item selected
1137 						aArgs.realloc( 1 );
1138 						aArgs[0].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Referer" ));
1139 						aArgs[0].Value <<= ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SFX_REFERER_USER ));
1140 					}
1141 
1142 					xDispatch = pMenuItemHandler->xMenuItemDispatch;
1143 				}
1144 			}
1145 		}
1146 	}
1147 
1148 	if ( xDispatch.is() )
1149     {
1150         const sal_uInt32 nRef = Application::ReleaseSolarMutex();
1151         if(::comphelper::UiEventsLogger::isEnabled()) //#i88653#
1152             UiEventLogHelper(::rtl::OUString::createFromAscii("MenuBarManager")).log(getServiceFactory(), m_xFrame, aTargetURL, aArgs);
1153 		xDispatch->dispatch( aTargetURL, aArgs );
1154         Application::AcquireSolarMutex( nRef );
1155     }
1156 
1157 	return 1;
1158 }
1159 
1160 
1161 IMPL_LINK( MenuBarManager, Highlight, Menu *, EMPTYARG )
1162 {
1163 	return 0;
1164 }
1165 
1166 sal_Bool MenuBarManager::MustBeHidden( PopupMenu* pPopupMenu, const Reference< XURLTransformer >& rTransformer )
1167 {
1168     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::MustBeHidden" );
1169     if ( pPopupMenu )
1170     {
1171         URL               aTargetURL;
1172         SvtCommandOptions aCmdOptions;
1173 
1174         sal_uInt16 nCount = pPopupMenu->GetItemCount();
1175         sal_uInt16 nHideCount( 0 );
1176 
1177         for ( sal_uInt16 i = 0; i < nCount; i++ )
1178         {
1179             sal_uInt16 nId = pPopupMenu->GetItemId( i );
1180             if ( nId > 0 )
1181             {
1182                 PopupMenu* pSubPopupMenu = pPopupMenu->GetPopupMenu( nId );
1183                 if ( pSubPopupMenu )
1184                 {
1185                     if ( MustBeHidden( pSubPopupMenu, rTransformer ))
1186                     {
1187                         pPopupMenu->HideItem( nId );
1188                         ++nHideCount;
1189                     }
1190                 }
1191                 else
1192                 {
1193                     aTargetURL.Complete = pPopupMenu->GetItemCommand( nId );
1194                     rTransformer->parseStrict( aTargetURL );
1195 
1196                     if ( aCmdOptions.Lookup( SvtCommandOptions::CMDOPTION_DISABLED, aTargetURL.Path ))
1197                         ++nHideCount;
1198                 }
1199             }
1200             else
1201                 ++nHideCount;
1202         }
1203 
1204         return ( nCount == nHideCount );
1205     }
1206 
1207     return sal_True;
1208 }
1209 String MenuBarManager::RetrieveLabelFromCommand( const String& aCmdURL )
1210 {
1211     return framework::RetrieveLabelFromCommand(aCmdURL,mxServiceFactory,m_xUICommandLabels,m_xFrame,m_aModuleIdentifier,m_bModuleIdentified,"Label");
1212 }
1213 
1214 
1215 
1216 sal_Bool MenuBarManager::CreatePopupMenuController( MenuItemHandler* pMenuItemHandler )
1217 {
1218     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::CreatePopupMenuController" );
1219     rtl::OUString aItemCommand( pMenuItemHandler->aMenuItemURL );
1220 
1221     // Try instanciate a popup menu controller. It is stored in the menu item handler.
1222     Reference< XMultiComponentFactory > xPopupMenuControllerFactory( m_xPopupMenuControllerRegistration, UNO_QUERY );
1223     if ( xPopupMenuControllerFactory.is() )
1224     {
1225         Sequence< Any > aSeq( 2 );
1226         PropertyValue aPropValue;
1227 
1228         aPropValue.Name         = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "ModuleName" ));
1229         aPropValue.Value      <<= m_aModuleIdentifier;
1230         aSeq[0] <<= aPropValue;
1231         aPropValue.Name         = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Frame" ));
1232         aPropValue.Value      <<= m_xFrame;
1233         aSeq[1] <<= aPropValue;
1234 
1235         Reference< XComponentContext > xComponentContext;
1236         Reference< XPropertySet >      xProps( getServiceFactory(), UNO_QUERY );
1237 
1238 		xProps->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "DefaultContext" ))) >>=
1239             xComponentContext;
1240 
1241         Reference< XPopupMenuController > xPopupMenuController(
1242                                                 xPopupMenuControllerFactory->createInstanceWithArgumentsAndContext(
1243                                                     aItemCommand,
1244                                                     aSeq,
1245                                                     xComponentContext ),
1246                                                 UNO_QUERY );
1247 
1248         if ( xPopupMenuController.is() )
1249         {
1250             // Provide our awt popup menu to the popup menu controller
1251             pMenuItemHandler->xPopupMenuController = xPopupMenuController;
1252             xPopupMenuController->setPopupMenu( pMenuItemHandler->xPopupMenu );
1253             return sal_True;
1254         }
1255     }
1256 
1257     return sal_False;
1258 }
1259 
1260 void MenuBarManager::FillMenuManager( Menu* pMenu, const Reference< XFrame >& rFrame, const Reference< XDispatchProvider >& rDispatchProvider, const rtl::OUString& rModuleIdentifier, sal_Bool bDelete, sal_Bool bDeleteChildren )
1261 {
1262     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::FillMenuManager" );
1263 	m_xFrame			= rFrame;
1264 	m_bActive			= sal_False;
1265 	m_bDeleteMenu		= bDelete;
1266 	m_bDeleteChildren	= bDeleteChildren;
1267 	m_pVCLMenu			= pMenu;
1268 	m_bInitialized		= sal_False;
1269 	m_bIsBookmarkMenu	= sal_False;
1270     m_xDispatchProvider = rDispatchProvider;
1271 
1272 	const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
1273 	m_bWasHiContrast	= rSettings.GetHighContrastMode();
1274 	m_bShowMenuImages	= rSettings.GetUseImagesInMenus();
1275     m_bRetrieveImages   = sal_False;
1276 
1277 	sal_Int32 nAddonsURLPrefixLength = ADDONSPOPUPMENU_URL_PREFIX.getLength();
1278 
1279     // Add root as ui configuration listener
1280     RetrieveImageManagers();
1281 
1282     if ( pMenu->IsMenuBar() && rFrame.is() )
1283 	{
1284         // First merge all addon popup menus into our structure
1285         sal_uInt16 nPos = 0;
1286         for ( nPos = 0; nPos < pMenu->GetItemCount(); nPos++ )
1287         {
1288             sal_uInt16          nItemId  = pMenu->GetItemId( nPos );
1289             ::rtl::OUString aCommand = pMenu->GetItemCommand( nItemId );
1290             if ( nItemId == SID_MDIWINDOWLIST ||
1291                  aCommand == aSpecialWindowCommand )
1292             {
1293 		        // Retrieve addon popup menus and add them to our menu bar
1294 		        Reference< com::sun::star::frame::XModel >		xModel;
1295 		        Reference< com::sun::star::frame::XController >	xController( rFrame->getController(), UNO_QUERY );
1296 		        if ( xController.is() )
1297 			        xModel = Reference< com::sun::star::frame::XModel >( xController->getModel(), UNO_QUERY );
1298 		        framework::AddonMenuManager::MergeAddonPopupMenus( rFrame, xModel, nPos, (MenuBar *)pMenu );
1299                 break;
1300             }
1301         }
1302 
1303         // Merge the Add-Ons help menu items into the Office help menu
1304         framework::AddonMenuManager::MergeAddonHelpMenu( rFrame, (MenuBar *)pMenu );
1305     }
1306 
1307     String      aEmpty;
1308     sal_Bool    bAccessibilityEnabled( Application::GetSettings().GetMiscSettings().GetEnableATToolSupport() );
1309     sal_uInt16 nItemCount = pMenu->GetItemCount();
1310     ::rtl::OUString aItemCommand;
1311     m_aMenuItemHandlerVector.reserve(nItemCount);
1312 	for ( sal_uInt16 i = 0; i < nItemCount; i++ )
1313 	{
1314         sal_uInt16 nItemId = FillItemCommand(aItemCommand,pMenu, i );
1315 
1316         // Set module identifier when provided from outside
1317         if ( rModuleIdentifier.getLength() > 0 )
1318         {
1319             m_aModuleIdentifier = rModuleIdentifier;
1320             m_bModuleIdentified = sal_True;
1321         }
1322 
1323         if (( pMenu->IsMenuBar() || bAccessibilityEnabled ) &&
1324 			( pMenu->GetItemText( nItemId ).Len() == 0 ))
1325         {
1326             if ( aItemCommand.getLength() > 0 )
1327                 pMenu->SetItemText( nItemId, RetrieveLabelFromCommand( aItemCommand ));
1328         }
1329 
1330         Reference< XDispatch > xDispatch;
1331 		Reference< XStatusListener > xStatusListener;
1332 		PopupMenu* pPopup = pMenu->GetPopupMenu( nItemId );
1333         bool bItemShowMenuImages = m_bShowMenuImages;
1334         MenuItemBits nBits =  pMenu->GetItemBits( nItemId );
1335         // overwrite the show icons on menu option?
1336         if ( nBits )
1337             bItemShowMenuImages = ( ( nBits & MIB_ICON ) == MIB_ICON );
1338 		if ( pPopup )
1339 		{
1340             // Retrieve module identifier from Help Command entry
1341             rtl::OUString aModuleIdentifier( rModuleIdentifier );
1342             if ( pMenu->GetHelpCommand( nItemId ).Len() > 0 )
1343             {
1344                 aModuleIdentifier = pMenu->GetHelpCommand( nItemId );
1345                 pMenu->SetHelpCommand( nItemId, aEmpty );
1346             }
1347 
1348             if ( m_xPopupMenuControllerRegistration.is() &&
1349                  pPopup->GetItemCount() == 0 &&
1350                  m_xPopupMenuControllerRegistration->hasController( aItemCommand, rtl::OUString() )
1351                   )
1352             {
1353                 // Check if we have to create a popup menu for a uno based popup menu controller.
1354                 // We have to set an empty popup menu into our menu structure so the controller also
1355                 // works with inplace OLE. Remove old dummy popup menu!
1356                 MenuItemHandler* pItemHandler = new MenuItemHandler( nItemId, xStatusListener, xDispatch );
1357                 VCLXPopupMenu* pVCLXPopupMenu = new VCLXPopupMenu;
1358                 PopupMenu* pNewPopupMenu = (PopupMenu *)pVCLXPopupMenu->GetMenu();
1359                 pMenu->SetPopupMenu( nItemId, pNewPopupMenu );
1360                 pItemHandler->xPopupMenu = Reference< com::sun::star::awt::XPopupMenu >( (OWeakObject *)pVCLXPopupMenu, UNO_QUERY );
1361                 pItemHandler->aMenuItemURL = aItemCommand;
1362                 m_aMenuItemHandlerVector.push_back( pItemHandler );
1363                 delete pPopup;
1364 
1365                 if ( bAccessibilityEnabled )
1366                 {
1367                     if ( CreatePopupMenuController( pItemHandler ))
1368                         pItemHandler->xPopupMenuController->updatePopupMenu();
1369                 }
1370 				lcl_CheckForChildren(pMenu, nItemId);
1371             }
1372             else if (( aItemCommand.getLength() > nAddonsURLPrefixLength ) &&
1373 				     ( aItemCommand.indexOf( ADDONSPOPUPMENU_URL_PREFIX ) == 0 ))
1374 			{
1375 				// A special addon popup menu, must be created with a different ctor
1376 				// #110897#
1377                 MenuBarManager* pSubMenuManager = new MenuBarManager( getServiceFactory(), m_xFrame, m_xURLTransformer,(AddonPopupMenu *)pPopup, bDeleteChildren, bDeleteChildren );
1378                 AddMenu(pSubMenuManager,aItemCommand,nItemId);
1379 			}
1380 			else
1381 			{
1382                 Reference< XDispatchProvider > xPopupMenuDispatchProvider( rDispatchProvider );
1383 
1384                 // Retrieve possible attributes struct
1385 				MenuConfiguration::Attributes* pAttributes = (MenuConfiguration::Attributes *)(pMenu->GetUserValue( nItemId ));
1386                 if ( pAttributes )
1387                     xPopupMenuDispatchProvider = pAttributes->xDispatchProvider;
1388 
1389                 // Check if this is the help menu. Add menu item if needed
1390                 if ( nItemId == SID_HELPMENU || aItemCommand == aSlotHelpMenu || aItemCommand == aCmdHelpMenu )
1391                 {
1392                     // Check if this is the help menu. Add menu item if needed
1393                     CheckAndAddMenuExtension( pPopup );
1394                 }
1395                 else if (( nItemId == SID_ADDONLIST || aItemCommand == aSlotSpecialToolsMenu || aItemCommand == aCmdToolsMenu ) &&
1396 					     AddonMenuManager::HasAddonMenuElements() )
1397                 {
1398                     // Create addon popup menu if there exist elements and this is the tools popup menu
1399 					sal_uInt16      nCount   = 0;
1400 					AddonMenu*  pSubMenu = AddonMenuManager::CreateAddonMenu( rFrame );
1401 					if ( pSubMenu && ( pSubMenu->GetItemCount() > 0 ))
1402 					{
1403 						if ( pPopup->GetItemType( nCount-1 ) != MENUITEM_SEPARATOR )
1404 							pPopup->InsertSeparator();
1405 
1406 					    // Use resource to load popup menu title
1407 					    String aAddonsStrRes = String( FwkResId( STR_MENU_ADDONS ));
1408 					    pPopup->InsertItem( ITEMID_ADDONLIST, aAddonsStrRes );
1409 					    pPopup->SetPopupMenu( ITEMID_ADDONLIST, pSubMenu );
1410 
1411 					    // Set item command for popup menu to enable it for GetImageFromURL
1412 						const ::rtl::OUString aSlotString( RTL_CONSTASCII_USTRINGPARAM( "slot:" ));
1413                         ::rtl::OUString aNewItemCommand( aSlotString );
1414 					    aNewItemCommand += ::rtl::OUString::valueOf( (sal_Int32)ITEMID_ADDONLIST );
1415 					    pPopup->SetItemCommand( ITEMID_ADDONLIST, aNewItemCommand );
1416 					}
1417 					else
1418 					    delete pSubMenu;
1419 				}
1420 
1421                 if ( nItemId == ITEMID_ADDONLIST )
1422                 {
1423 			        // Create control structure within the "Tools" sub menu for the Add-Ons popup menu
1424                     // #110897# MenuBarManager* pSubMenuManager = new MenuBarManager( rFrame, pSubMenu, sal_True, sal_False );
1425                     AddonMenu* pSubMenu = dynamic_cast< AddonMenu* >( pPopup );
1426                     if ( pSubMenu )
1427                     {
1428                         MenuBarManager* pSubMenuManager = new MenuBarManager( getServiceFactory(), m_xFrame, m_xURLTransformer,pSubMenu, sal_True, sal_False );
1429                         AddMenu(pSubMenuManager,aItemCommand,nItemId);
1430                         pSubMenuManager->m_aMenuItemCommand = ::rtl::OUString();
1431 
1432 			            // Set image for the addon popup menu item
1433 			            if ( bItemShowMenuImages && !pPopup->GetItemImage( ITEMID_ADDONLIST ))
1434 			            {
1435                             Reference< XFrame > xTemp( rFrame );
1436                             Image aImage = GetImageFromURL( xTemp, aItemCommand, sal_False, m_bWasHiContrast );
1437         		            if ( !!aImage )
1438            			            pPopup->SetItemImage( ITEMID_ADDONLIST, aImage );
1439 			            }
1440                     }
1441                 }
1442                 else
1443                 {
1444                     // #110897# MenuBarManager* pSubMenuManager = new MenuBarManager( rFrame, pPopupMenu, bDeleteChildren, bDeleteChildren );
1445                     MenuBarManager* pSubMenuMgr = new MenuBarManager( getServiceFactory(), rFrame, m_xURLTransformer,rDispatchProvider, aModuleIdentifier, pPopup, bDeleteChildren, bDeleteChildren );
1446                     AddMenu(pSubMenuMgr,aItemCommand,nItemId);
1447                 }
1448 			}
1449 		}
1450         else if ( pMenu->GetItemType( i ) != MENUITEM_SEPARATOR )
1451 		{
1452             if ( bItemShowMenuImages )
1453 			{
1454 			    if ( AddonMenuManager::IsAddonMenuId( nItemId ))
1455 			    {
1456                     // Add-Ons uses images from different places
1457                     Image           aImage;
1458                     rtl::OUString   aImageId;
1459 
1460 					MenuConfiguration::Attributes* pMenuAttributes =
1461 						(MenuConfiguration::Attributes*)pMenu->GetUserValue( nItemId );
1462 
1463 					if ( pMenuAttributes && pMenuAttributes->aImageId.getLength() > 0 )
1464 					{
1465 						// Retrieve image id from menu attributes
1466 						aImage = GetImageFromURL( m_xFrame, aImageId, sal_False, m_bWasHiContrast );
1467                     }
1468 
1469 	                if ( !aImage )
1470 	                {
1471 						aImage = GetImageFromURL( m_xFrame, aItemCommand, sal_False, m_bWasHiContrast );
1472 	                    if ( !aImage )
1473                             aImage = AddonsOptions().GetImageFromURL( aItemCommand, sal_False, m_bWasHiContrast );
1474                     }
1475 
1476 		            if ( !!aImage )
1477 		                pMenu->SetItemImage( nItemId, aImage );
1478                     else
1479                         m_bRetrieveImages = sal_True;
1480 			    }
1481                 m_bRetrieveImages = sal_True;
1482             }
1483 
1484             MenuItemHandler* pItemHandler = new MenuItemHandler( nItemId, xStatusListener, xDispatch );
1485             pItemHandler->aMenuItemURL = aItemCommand;
1486 
1487             if ( m_xPopupMenuControllerRegistration.is() &&
1488                  m_xPopupMenuControllerRegistration->hasController( aItemCommand, rtl::OUString() ))
1489             {
1490                 // Check if we have to create a popup menu for a uno based popup menu controller.
1491                 // We have to set an empty popup menu into our menu structure so the controller also
1492                 // works with inplace OLE.
1493                 VCLXPopupMenu* pVCLXPopupMenu = new VCLXPopupMenu;
1494                 PopupMenu* pPopupMenu = (PopupMenu *)pVCLXPopupMenu->GetMenu();
1495                 pMenu->SetPopupMenu( pItemHandler->nItemId, pPopupMenu );
1496                 pItemHandler->xPopupMenu = Reference< com::sun::star::awt::XPopupMenu >( (OWeakObject *)pVCLXPopupMenu, UNO_QUERY );
1497 
1498                 if ( bAccessibilityEnabled && CreatePopupMenuController( pItemHandler ) )
1499                 {
1500                     pItemHandler->xPopupMenuController->updatePopupMenu();
1501 				}
1502 
1503 				lcl_CheckForChildren(pMenu, pItemHandler->nItemId);
1504             }
1505 
1506 			m_aMenuItemHandlerVector.push_back( pItemHandler );
1507 		}
1508 	}
1509 
1510     if ( bAccessibilityEnabled )
1511     {
1512         RetrieveShortcuts( m_aMenuItemHandlerVector );
1513         std::vector< MenuItemHandler* >::iterator p;
1514 		for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
1515 		{
1516 		    MenuItemHandler* pMenuItemHandler = *p;
1517 
1518             // Set key code, workaround for hard-coded shortcut F1 mapped to .uno:HelpIndex
1519             // Only non-popup menu items can have a short-cut
1520             if ( pMenuItemHandler->aMenuItemURL == aCmdHelpIndex )
1521             {
1522                 KeyCode aKeyCode( KEY_F1 );
1523                 pMenu->SetAccelKey( pMenuItemHandler->nItemId, aKeyCode );
1524             }
1525             else if ( pMenu->GetPopupMenu( pMenuItemHandler->nItemId ) == 0 )
1526                 pMenu->SetAccelKey( pMenuItemHandler->nItemId, pMenuItemHandler->aKeyCode );
1527         }
1528     }
1529 
1530     SetHdl();
1531 }
1532 
1533 void MenuBarManager::impl_RetrieveShortcutsFromConfiguration(
1534     const Reference< XAcceleratorConfiguration >& rAccelCfg,
1535     const Sequence< rtl::OUString >& rCommands,
1536     std::vector< MenuItemHandler* >& aMenuShortCuts )
1537 {
1538     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::impl_RetrieveShortcutsFromConfiguration" );
1539     if ( rAccelCfg.is() )
1540     {
1541         try
1542         {
1543             com::sun::star::awt::KeyEvent aKeyEvent;
1544             Sequence< Any > aSeqKeyCode = rAccelCfg->getPreferredKeyEventsForCommandList( rCommands );
1545             for ( sal_Int32 i = 0; i < aSeqKeyCode.getLength(); i++ )
1546             {
1547                 if ( aSeqKeyCode[i] >>= aKeyEvent )
1548                     aMenuShortCuts[i]->aKeyCode = svt::AcceleratorExecute::st_AWTKey2VCLKey( aKeyEvent );
1549             }
1550         }
1551         catch ( IllegalArgumentException& )
1552         {
1553         }
1554     }
1555 }
1556 
1557 void MenuBarManager::RetrieveShortcuts( std::vector< MenuItemHandler* >& aMenuShortCuts )
1558 {
1559     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::RetrieveShortcuts" );
1560     if ( !m_bModuleIdentified )
1561     {
1562         m_bModuleIdentified = sal_True;
1563         Reference< XModuleManager > xModuleManager;
1564         xModuleManager = Reference< XModuleManager >( getServiceFactory()->createInstance( SERVICENAME_MODULEMANAGER ), UNO_QUERY_THROW );
1565 
1566         try
1567         {
1568             m_aModuleIdentifier = xModuleManager->identify( m_xFrame );
1569         }
1570         catch( Exception& )
1571         {
1572         }
1573     }
1574 
1575     if ( m_bModuleIdentified )
1576     {
1577         Reference< XAcceleratorConfiguration > xDocAccelCfg( m_xDocAcceleratorManager );
1578         Reference< XAcceleratorConfiguration > xModuleAccelCfg( m_xModuleAcceleratorManager );
1579         Reference< XAcceleratorConfiguration > xGlobalAccelCfg( m_xGlobalAcceleratorManager );
1580 
1581         if ( !m_bAcceleratorCfg )
1582         {
1583             // Retrieve references on demand
1584             m_bAcceleratorCfg = sal_True;
1585             if ( !xDocAccelCfg.is() )
1586             {
1587                 Reference< XController > xController = m_xFrame->getController();
1588                 Reference< XModel > xModel;
1589                 if ( xController.is() )
1590                 {
1591                     xModel = xController->getModel();
1592                     if ( xModel.is() )
1593                     {
1594                         Reference< XUIConfigurationManagerSupplier > xSupplier( xModel, UNO_QUERY );
1595                         if ( xSupplier.is() )
1596                         {
1597                             Reference< XUIConfigurationManager > xDocUICfgMgr( xSupplier->getUIConfigurationManager(), UNO_QUERY );
1598                             if ( xDocUICfgMgr.is() )
1599                             {
1600                                 xDocAccelCfg = Reference< XAcceleratorConfiguration >( xDocUICfgMgr->getShortCutManager(), UNO_QUERY );
1601                                 m_xDocAcceleratorManager = xDocAccelCfg;
1602                             }
1603                         }
1604                     }
1605                 }
1606             }
1607 
1608             if ( !xModuleAccelCfg.is() )
1609             {
1610                 Reference< XModuleUIConfigurationManagerSupplier > xModuleCfgMgrSupplier( getServiceFactory()->createInstance(
1611                                                                                             SERVICENAME_MODULEUICONFIGURATIONMANAGERSUPPLIER ),
1612                                                                                         UNO_QUERY );
1613                 try
1614                 {
1615                     Reference< XUIConfigurationManager > xUICfgMgr = xModuleCfgMgrSupplier->getUIConfigurationManager( m_aModuleIdentifier );
1616                     if ( xUICfgMgr.is() )
1617                     {
1618                         xModuleAccelCfg = Reference< XAcceleratorConfiguration >( xUICfgMgr->getShortCutManager(), UNO_QUERY );
1619                         m_xModuleAcceleratorManager = xModuleAccelCfg;
1620                     }
1621                 }
1622                 catch ( RuntimeException& )
1623                 {
1624                     throw;
1625                 }
1626                 catch ( Exception& )
1627                 {
1628                 }
1629             }
1630 
1631             if ( !xGlobalAccelCfg.is() )
1632             {
1633                 xGlobalAccelCfg = Reference< XAcceleratorConfiguration >( getServiceFactory()->createInstance(
1634                                                                             SERVICENAME_GLOBALACCELERATORCONFIGURATION ),
1635                                                                           UNO_QUERY );
1636                 m_xGlobalAcceleratorManager = xGlobalAccelCfg;
1637             }
1638         }
1639 
1640         KeyCode aEmptyKeyCode;
1641         Sequence< rtl::OUString > aSeq( aMenuShortCuts.size() );
1642         const sal_uInt32 nCount = aMenuShortCuts.size();
1643         for ( sal_uInt32 i = 0; i < nCount; ++i )
1644         {
1645             aSeq[i] = aMenuShortCuts[i]->aMenuItemURL;
1646             aMenuShortCuts[i]->aKeyCode = aEmptyKeyCode;
1647         }
1648 
1649         if ( m_xGlobalAcceleratorManager.is() )
1650             impl_RetrieveShortcutsFromConfiguration( xGlobalAccelCfg, aSeq, aMenuShortCuts );
1651         if ( m_xModuleAcceleratorManager.is() )
1652             impl_RetrieveShortcutsFromConfiguration( xModuleAccelCfg, aSeq, aMenuShortCuts );
1653         if ( m_xDocAcceleratorManager.is() )
1654             impl_RetrieveShortcutsFromConfiguration( xGlobalAccelCfg, aSeq, aMenuShortCuts );
1655     }
1656 }
1657 
1658 void MenuBarManager::RetrieveImageManagers()
1659 {
1660     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::RetrieveImageManagers" );
1661     if ( !m_xDocImageManager.is() )
1662     {
1663         Reference< XController > xController = m_xFrame->getController();
1664         Reference< XModel > xModel;
1665         if ( xController.is() )
1666         {
1667             xModel = xController->getModel();
1668             if ( xModel.is() )
1669             {
1670                 Reference< XUIConfigurationManagerSupplier > xSupplier( xModel, UNO_QUERY );
1671                 if ( xSupplier.is() )
1672                 {
1673                     Reference< XUIConfigurationManager > xDocUICfgMgr( xSupplier->getUIConfigurationManager(), UNO_QUERY );
1674                     m_xDocImageManager = Reference< XImageManager >( xDocUICfgMgr->getImageManager(), UNO_QUERY );
1675                     m_xDocImageManager->addConfigurationListener(
1676                                             Reference< XUIConfigurationListener >(
1677                                                 static_cast< OWeakObject* >( this ), UNO_QUERY ));
1678                 }
1679             }
1680         }
1681     }
1682 
1683     Reference< XModuleManager > xModuleManager;
1684     if ( m_aModuleIdentifier.getLength() == 0 )
1685         xModuleManager.set( getServiceFactory()->createInstance( SERVICENAME_MODULEMANAGER ), UNO_QUERY_THROW );
1686 
1687     try
1688     {
1689         if ( xModuleManager.is() )
1690             m_aModuleIdentifier = xModuleManager->identify( Reference< XInterface >( m_xFrame, UNO_QUERY ) );
1691     }
1692     catch( Exception& )
1693     {
1694     }
1695 
1696     if ( !m_xModuleImageManager.is() )
1697     {
1698         Reference< XModuleUIConfigurationManagerSupplier > xModuleCfgMgrSupplier( getServiceFactory()->createInstance(
1699                                                                                     SERVICENAME_MODULEUICONFIGURATIONMANAGERSUPPLIER ),
1700                                                                                   UNO_QUERY );
1701         Reference< XUIConfigurationManager > xUICfgMgr = xModuleCfgMgrSupplier->getUIConfigurationManager( m_aModuleIdentifier );
1702         m_xModuleImageManager.set( xUICfgMgr->getImageManager(), UNO_QUERY );
1703         m_xModuleImageManager->addConfigurationListener( Reference< XUIConfigurationListener >(
1704                                                             static_cast< OWeakObject* >( this ), UNO_QUERY ));
1705     }
1706 }
1707 
1708 void MenuBarManager::FillMenuWithConfiguration(
1709     sal_uInt16&                             nId,
1710     Menu*                               pMenu,
1711     const ::rtl::OUString&              rModuleIdentifier,
1712     const Reference< XIndexAccess >&    rItemContainer,
1713     const Reference< XURLTransformer >& rTransformer )
1714 {
1715     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::FillMenuWithConfiguration" );
1716     Reference< XDispatchProvider > xEmptyDispatchProvider;
1717     MenuBarManager::FillMenu( nId, pMenu, rModuleIdentifier, rItemContainer, xEmptyDispatchProvider );
1718 
1719     // Merge add-on menu entries into the menu bar
1720     MenuBarManager::MergeAddonMenus( static_cast< Menu* >( pMenu ),
1721                                      AddonsOptions().GetMergeMenuInstructions(),
1722                                      rModuleIdentifier );
1723 
1724     sal_Bool bHasDisabledEntries = SvtCommandOptions().HasEntries( SvtCommandOptions::CMDOPTION_DISABLED );
1725     if ( bHasDisabledEntries )
1726     {
1727         sal_uInt16 nCount = pMenu->GetItemCount();
1728 	    for ( sal_uInt16 i = 0; i < nCount; i++ )
1729         {
1730             sal_uInt16 nID = pMenu->GetItemId( i );
1731             if ( nID > 0 )
1732             {
1733                 PopupMenu* pPopupMenu = pMenu->GetPopupMenu( nID );
1734                 if ( pPopupMenu )
1735                 {
1736                     if ( MustBeHidden( pPopupMenu, rTransformer ))
1737                         pMenu->HideItem( nId );
1738                 }
1739             }
1740         }
1741     }
1742 }
1743 
1744 void MenuBarManager::FillMenu(
1745     sal_uInt16&                               nId,
1746     Menu*                                 pMenu,
1747     const rtl::OUString&                  rModuleIdentifier,
1748     const Reference< XIndexAccess >&      rItemContainer,
1749     const Reference< XDispatchProvider >& rDispatchProvider )
1750 {
1751     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::FillMenu" );
1752     // Fill menu bar with container contents
1753     for ( sal_Int32 n = 0; n < rItemContainer->getCount(); n++ )
1754     {
1755         Sequence< PropertyValue >       aProp;
1756         rtl::OUString                   aCommandURL;
1757         rtl::OUString                   aLabel;
1758         rtl::OUString                   aHelpURL;
1759         rtl::OUString                   aModuleIdentifier( rModuleIdentifier );
1760 	    sal_Bool                        bShow(sal_True);
1761 	    sal_Bool                        bEnabled(sal_True);
1762         sal_uInt16                      nType = 0;
1763         Reference< XIndexAccess >       xIndexContainer;
1764         Reference< XDispatchProvider >  xDispatchProvider( rDispatchProvider );
1765         sal_Int16 nStyle = 0;
1766         try
1767         {
1768             if ( rItemContainer->getByIndex( n ) >>= aProp )
1769             {
1770                 for ( int i = 0; i < aProp.getLength(); i++ )
1771                 {
1772                     rtl::OUString aPropName = aProp[i].Name;
1773                     if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_COMMANDURL, LEN_DESCRIPTOR_COMMANDURL ))
1774                         aProp[i].Value >>= aCommandURL;
1775                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_HELPURL, LEN_DESCRIPTOR_HELPURL ))
1776                         aProp[i].Value >>= aHelpURL;
1777                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_CONTAINER, LEN_DESCRIPTOR_CONTAINER ))
1778                         aProp[i].Value >>= xIndexContainer;
1779                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_LABEL, LEN_DESCRIPTOR_LABEL ))
1780                         aProp[i].Value >>= aLabel;
1781                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_TYPE, LEN_DESCRIPTOR_TYPE ))
1782                         aProp[i].Value >>= nType;
1783                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_MODULEIDENTIFIER, LEN_DESCRIPTOR_MODULEIDENTIFIER ))
1784                         aProp[i].Value >>= aModuleIdentifier;
1785                     else if ( aPropName.equalsAsciiL( ITEM_DESCRIPTOR_DISPATCHPROVIDER, LEN_DESCRIPTOR_DISPATCHPROVIDER ))
1786                         aProp[i].Value >>= xDispatchProvider;
1787                     else if ( aProp[i].Name.equalsAsciiL( ITEM_DESCRIPTOR_STYLE, LEN_DESCRIPTOR_STYLE ))
1788                         aProp[i].Value >>= nStyle;
1789                     else if ( aProp[i].Name.equalsAsciiL( ITEM_DESCRIPTOR_ISVISIBLE, LEN_DESCRIPTOR_ISVISIBLE ))
1790 		                aProp[i].Value >>= bShow;
1791                     else if ( aProp[i].Name.equalsAsciiL( ITEM_DESCRIPTOR_ENABLED, LEN_DESCRIPTOR_ENABLED ))
1792 		                aProp[i].Value >>= bEnabled;
1793                 }
1794 
1795                 if ( nType == ::com::sun::star::ui::ItemType::DEFAULT )
1796                 {
1797                     pMenu->InsertItem( nId, aLabel );
1798                     pMenu->SetItemCommand( nId, aCommandURL );
1799 
1800                     if ( nStyle )
1801                     {
1802                         MenuItemBits nBits = pMenu->GetItemBits( nId );
1803                         if ( nStyle & ::com::sun::star::ui::ItemStyle::ICON )
1804                            nBits |= MIB_ICON;
1805                         if ( nStyle & ::com::sun::star::ui::ItemStyle::TEXT )
1806                            nBits |= MIB_TEXT;
1807                         if ( nStyle & ::com::sun::star::ui::ItemStyle::RADIO_CHECK )
1808                            nBits |= MIB_RADIOCHECK;
1809                         pMenu->SetItemBits( nId, nBits );
1810                     }
1811 
1812                     if ( !bShow )
1813 		                pMenu->HideItem( nId );
1814 
1815                     if ( !bEnabled)
1816                         pMenu->EnableItem( nId, sal_False );
1817 
1818                     if ( xIndexContainer.is() )
1819                     {
1820                         PopupMenu* pNewPopupMenu = new PopupMenu;
1821                         pMenu->SetPopupMenu( nId, pNewPopupMenu );
1822 
1823                         if ( xDispatchProvider.is() )
1824                         {
1825 				            // Use attributes struct to transport special dispatch provider
1826                             MenuConfiguration::Attributes* pAttributes = new MenuConfiguration::Attributes;
1827                             pAttributes->xDispatchProvider = xDispatchProvider;
1828                             pMenu->SetUserValue( nId, (sal_uIntPtr)( pAttributes ));
1829                         }
1830 
1831                         // Use help command to transport module identifier
1832                         if ( aModuleIdentifier.getLength() > 0 )
1833                             pMenu->SetHelpCommand( nId, aModuleIdentifier );
1834 
1835                         ++nId;
1836                         FillMenu( nId, pNewPopupMenu, aModuleIdentifier, xIndexContainer, xDispatchProvider );
1837                     }
1838                     else
1839                         ++nId;
1840                 }
1841                 else
1842                 {
1843                     pMenu->InsertSeparator();
1844                     ++nId;
1845                 }
1846             }
1847         }
1848         catch ( IndexOutOfBoundsException& )
1849         {
1850             break;
1851         }
1852     }
1853 }
1854 
1855 void MenuBarManager::MergeAddonMenus(
1856     Menu* pMenuBar,
1857     const MergeMenuInstructionContainer& aMergeInstructionContainer,
1858     const ::rtl::OUString& rModuleIdentifier )
1859 {
1860     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::MergeAddonMenus" );
1861     // set start value for the item ID for the new addon menu items
1862     sal_uInt16 nItemId = ADDONMENU_MERGE_ITEMID_START;
1863 
1864     const sal_uInt32 nCount = aMergeInstructionContainer.size();
1865     for ( sal_uInt32 i = 0; i < nCount; i++ )
1866     {
1867         const MergeMenuInstruction& rMergeInstruction = aMergeInstructionContainer[i];
1868 
1869         if ( MenuBarMerger::IsCorrectContext( rMergeInstruction.aMergeContext, rModuleIdentifier ))
1870         {
1871             ::std::vector< ::rtl::OUString > aMergePath;
1872 
1873             // retrieve the merge path from the merge point string
1874             MenuBarMerger::RetrieveReferencePath( rMergeInstruction.aMergePoint, aMergePath );
1875 
1876             // convert the sequence/sequence property value to a more convenient vector<>
1877             AddonMenuContainer aMergeMenuItems;
1878             MenuBarMerger::GetSubMenu( rMergeInstruction.aMergeMenu, aMergeMenuItems );
1879 
1880             // try to find the reference point for our merge operation
1881             Menu* pMenu = pMenuBar;
1882             ReferencePathInfo aResult = MenuBarMerger::FindReferencePath( aMergePath, pMenu );
1883 
1884             if ( aResult.eResult == RP_OK )
1885             {
1886                 // normal merge operation
1887                 MenuBarMerger::ProcessMergeOperation( aResult.pPopupMenu,
1888                                                       aResult.nPos,
1889                                                       nItemId,
1890                                                       rMergeInstruction.aMergeCommand,
1891                                                       rMergeInstruction.aMergeCommandParameter,
1892                                                       rModuleIdentifier,
1893                                                       aMergeMenuItems );
1894             }
1895             else
1896             {
1897                 // fallback
1898                 MenuBarMerger::ProcessFallbackOperation( aResult,
1899                                                          nItemId,
1900                                                          rMergeInstruction.aMergeCommand,
1901                                                          rMergeInstruction.aMergeFallback,
1902                                                          aMergePath,
1903                                                          rModuleIdentifier,
1904                                                          aMergeMenuItems );
1905             }
1906         }
1907     }
1908 }
1909 
1910 void MenuBarManager::SetItemContainer( const Reference< XIndexAccess >& rItemContainer )
1911 {
1912     RTL_LOGFILE_CONTEXT( aLog, "framework (cd100003) ::MenuBarManager::SetItemContainer" );
1913 
1914     ResetableGuard aGuard( m_aLock );
1915 
1916     Reference< XFrame > xFrame = m_xFrame;
1917 
1918     if ( !m_bModuleIdentified )
1919     {
1920         m_bModuleIdentified = sal_True;
1921         Reference< XModuleManager > xModuleManager;
1922         xModuleManager = Reference< XModuleManager >( getServiceFactory()->createInstance( SERVICENAME_MODULEMANAGER ), UNO_QUERY_THROW );
1923 
1924         try
1925         {
1926             m_aModuleIdentifier = xModuleManager->identify( xFrame );
1927         }
1928         catch( Exception& )
1929         {
1930         }
1931     }
1932 
1933     // Clear MenuBarManager structures
1934     {
1935         vos::OGuard aSolarMutexGuard( Application::GetSolarMutex() );
1936 
1937         // Check active state as we cannot change our VCL menu during activation by the user
1938         if ( m_bActive )
1939         {
1940             m_xDeferedItemContainer = rItemContainer;
1941             return;
1942         }
1943 
1944         RemoveListener();
1945 	    std::vector< MenuItemHandler* >::iterator p;
1946 	    for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
1947 	    {
1948 		    MenuItemHandler* pItemHandler = *p;
1949 		    pItemHandler->xMenuItemDispatch.clear();
1950 		    pItemHandler->xSubMenuManager.clear();
1951 		    delete pItemHandler;
1952 	    }
1953         m_aMenuItemHandlerVector.clear();
1954 
1955         // Remove top-level parts
1956 	    m_pVCLMenu->Clear();
1957 
1958         sal_uInt16          nId = 1;
1959 
1960         // Fill menu bar with container contents
1961         FillMenuWithConfiguration( nId, (Menu *)m_pVCLMenu, m_aModuleIdentifier, rItemContainer, m_xURLTransformer );
1962 
1963         // Refill menu manager again
1964         Reference< XDispatchProvider > xDispatchProvider;
1965         FillMenuManager( m_pVCLMenu, xFrame, xDispatchProvider, m_aModuleIdentifier, sal_False, sal_True );
1966 
1967         // add itself as frame action listener
1968         m_xFrame->addFrameActionListener( Reference< XFrameActionListener >( static_cast< OWeakObject* >( this ), UNO_QUERY ));
1969     }
1970 }
1971 
1972 void MenuBarManager::GetPopupController( PopupControllerCache& rPopupController )
1973 {
1974     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::GetPopupController" );
1975     String aPopupScheme = String::CreateFromAscii( "vnd.sun.star.popup:" );
1976 
1977     vos::OGuard aSolarMutexGuard( Application::GetSolarMutex() );
1978 
1979     std::vector< MenuItemHandler* >::iterator p;
1980     for ( p = m_aMenuItemHandlerVector.begin(); p != m_aMenuItemHandlerVector.end(); p++ )
1981     {
1982         MenuItemHandler* pItemHandler = *p;
1983         if ( pItemHandler->xPopupMenuController.is() )
1984         {
1985             Reference< XDispatchProvider > xDispatchProvider( pItemHandler->xPopupMenuController, UNO_QUERY );
1986 
1987             PopupControllerEntry aPopupControllerEntry;
1988             aPopupControllerEntry.m_xDispatchProvider = xDispatchProvider;
1989 
1990             // Just use the main part of the URL for popup menu controllers
1991             sal_Int32     nQueryPart( 0 );
1992             sal_Int32     nSchemePart( 0 );
1993             rtl::OUString aMainURL( RTL_CONSTASCII_USTRINGPARAM( "vnd.sun.star.popup:" ));
1994             rtl::OUString aMenuURL( pItemHandler->aMenuItemURL );
1995 
1996             nSchemePart = aMenuURL.indexOf( ':' );
1997             if (( nSchemePart > 0 ) &&
1998                 ( aMenuURL.getLength() > ( nSchemePart+1 )))
1999             {
2000                 nQueryPart  = aMenuURL.indexOf( '?', nSchemePart );
2001                 if ( nQueryPart > 0 )
2002                     aMainURL += aMenuURL.copy( nSchemePart, nQueryPart-nSchemePart );
2003                 else if ( nQueryPart == -1 )
2004                     aMainURL += aMenuURL.copy( nSchemePart+1 );
2005 
2006                 rPopupController.insert( PopupControllerCache::value_type(
2007                                            aMainURL, aPopupControllerEntry ));
2008             }
2009         }
2010         if ( pItemHandler->xSubMenuManager.is() )
2011         {
2012             MenuBarManager* pMenuBarManager = (MenuBarManager*)(pItemHandler->xSubMenuManager.get());
2013             if ( pMenuBarManager )
2014                 pMenuBarManager->GetPopupController( rPopupController );
2015         }
2016     }
2017 }
2018 
2019 // #110897#
2020 const Reference< XMultiServiceFactory >& MenuBarManager::getServiceFactory()
2021 {
2022 	// #110897#
2023 	return mxServiceFactory;
2024 }
2025 
2026 void MenuBarManager::AddMenu(MenuBarManager* pSubMenuManager,const ::rtl::OUString& _sItemCommand,sal_uInt16 _nItemId)
2027 {
2028     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::AddMenu" );
2029 	Reference< XStatusListener > xSubMenuManager( static_cast< OWeakObject *>( pSubMenuManager ), UNO_QUERY );
2030     m_xFrame->addFrameActionListener( Reference< XFrameActionListener >( xSubMenuManager, UNO_QUERY ));
2031 
2032 	// store menu item command as we later have to know which menu is active (see Activate handler)
2033 	pSubMenuManager->m_aMenuItemCommand = _sItemCommand;
2034     Reference< XDispatch > xDispatch;
2035 	MenuItemHandler* pMenuItemHandler = new MenuItemHandler(
2036 												_nItemId,
2037 												xSubMenuManager,
2038 												xDispatch );
2039     pMenuItemHandler->aMenuItemURL = _sItemCommand;
2040 	m_aMenuItemHandlerVector.push_back( pMenuItemHandler );
2041 }
2042 
2043 sal_uInt16 MenuBarManager::FillItemCommand(::rtl::OUString& _rItemCommand,Menu* _pMenu,sal_uInt16 _nIndex) const
2044 {
2045     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::FillItemCommand" );
2046     sal_uInt16 nItemId = _pMenu->GetItemId( _nIndex );
2047 
2048 	_rItemCommand = _pMenu->GetItemCommand( nItemId );
2049 	if ( !_rItemCommand.getLength() )
2050 	{
2051         const static ::rtl::OUString aSlotString( RTL_CONSTASCII_USTRINGPARAM( "slot:" ));
2052 		_rItemCommand = aSlotString;
2053 		_rItemCommand += ::rtl::OUString::valueOf( (sal_Int32)nItemId );
2054 		_pMenu->SetItemCommand( nItemId, _rItemCommand );
2055 	}
2056     return nItemId;
2057 }
2058 void MenuBarManager::Init(const Reference< XFrame >& rFrame,AddonMenu* pAddonMenu,sal_Bool bDelete,sal_Bool bDeleteChildren,bool _bHandlePopUp)
2059 {
2060     RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "framework", "Ocke.Janssen@sun.com", "MenuBarManager::Init" );
2061 	m_bActive			= sal_False;
2062 	m_bDeleteMenu		= bDelete;
2063 	m_bDeleteChildren	= bDeleteChildren;
2064 	m_pVCLMenu			= pAddonMenu;
2065 	m_xFrame			= rFrame;
2066 	m_bInitialized		= sal_False;
2067 	m_bIsBookmarkMenu	= sal_True;
2068 
2069     rtl::OUString aModuleIdentifier;
2070     m_xPopupMenuControllerRegistration = Reference< ::com::sun::star::frame::XUIControllerRegistration >(
2071 		getServiceFactory()->createInstance( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.frame.PopupMenuControllerFactory" ))),
2072 		UNO_QUERY );
2073 
2074 	const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
2075 	m_bWasHiContrast	= rSettings.GetHighContrastMode();
2076 
2077     Reference< XStatusListener > xStatusListener;
2078     Reference< XDispatch > xDispatch;
2079 	sal_uInt16 nItemCount = pAddonMenu->GetItemCount();
2080     ::rtl::OUString aItemCommand;
2081     m_aMenuItemHandlerVector.reserve(nItemCount);
2082 	for ( sal_uInt16 i = 0; i < nItemCount; i++ )
2083 	{
2084         sal_uInt16 nItemId = FillItemCommand(aItemCommand,pAddonMenu, i );
2085 
2086 		PopupMenu* pPopupMenu = pAddonMenu->GetPopupMenu( nItemId );
2087 		if ( pPopupMenu )
2088 		{
2089 			// #110897#
2090 			Reference< XDispatchProvider > xDispatchProvider;
2091             MenuBarManager* pSubMenuManager = new MenuBarManager( getServiceFactory(), rFrame, m_xURLTransformer,xDispatchProvider, aModuleIdentifier, pPopupMenu, _bHandlePopUp ? sal_False : bDeleteChildren, _bHandlePopUp ? sal_False : bDeleteChildren );
2092 
2093 		    Reference< XStatusListener > xSubMenuManager( static_cast< OWeakObject *>( pSubMenuManager ), UNO_QUERY );
2094 
2095 		    // store menu item command as we later have to know which menu is active (see Acivate handler)
2096 		    pSubMenuManager->m_aMenuItemCommand = aItemCommand;
2097 
2098 		    MenuItemHandler* pMenuItemHandler = new MenuItemHandler(
2099 													    nItemId,
2100 													    xSubMenuManager,
2101 													    xDispatch );
2102             m_aMenuItemHandlerVector.push_back( pMenuItemHandler );
2103 		}
2104 		else
2105 		{
2106 			if ( pAddonMenu->GetItemType( i ) != MENUITEM_SEPARATOR )
2107 			{
2108 				MenuConfiguration::Attributes* pAddonAttributes = (MenuConfiguration::Attributes *)(pAddonMenu->GetUserValue( nItemId ));
2109 				MenuItemHandler* pMenuItemHandler = new MenuItemHandler( nItemId, xStatusListener, xDispatch );
2110 
2111 				if ( pAddonAttributes )
2112 				{
2113 					// read additional attributes from attributes struct and AddonMenu implementation will delete all attributes itself!!
2114 					pMenuItemHandler->aTargetFrame = pAddonAttributes->aTargetFrame;
2115 				}
2116 
2117                 pMenuItemHandler->aMenuItemURL = aItemCommand;
2118                 if ( _bHandlePopUp )
2119                 {
2120                     // Check if we have to create a popup menu for a uno based popup menu controller.
2121                     // We have to set an empty popup menu into our menu structure so the controller also
2122                     // works with inplace OLE.
2123                     if ( m_xPopupMenuControllerRegistration.is() &&
2124                         m_xPopupMenuControllerRegistration->hasController( aItemCommand, rtl::OUString() ))
2125                     {
2126                         VCLXPopupMenu* pVCLXPopupMenu = new VCLXPopupMenu;
2127                         PopupMenu* pCtlPopupMenu = (PopupMenu *)pVCLXPopupMenu->GetMenu();
2128                         pAddonMenu->SetPopupMenu( pMenuItemHandler->nItemId, pCtlPopupMenu );
2129                         pMenuItemHandler->xPopupMenu = Reference< com::sun::star::awt::XPopupMenu >( (OWeakObject *)pVCLXPopupMenu, UNO_QUERY );
2130 
2131                     }
2132                 }
2133 				m_aMenuItemHandlerVector.push_back( pMenuItemHandler );
2134 			}
2135 		}
2136 	}
2137 
2138 	SetHdl();
2139 }
2140 
2141 void MenuBarManager::SetHdl()
2142 {
2143 	m_pVCLMenu->SetHighlightHdl( LINK( this, MenuBarManager, Highlight ));
2144 	m_pVCLMenu->SetActivateHdl( LINK( this, MenuBarManager, Activate ));
2145 	m_pVCLMenu->SetDeactivateHdl( LINK( this, MenuBarManager, Deactivate ));
2146 	m_pVCLMenu->SetSelectHdl( LINK( this, MenuBarManager, Select ));
2147 
2148     if ( !m_xURLTransformer.is() && mxServiceFactory.is() )
2149         m_xURLTransformer.set( mxServiceFactory->createInstance(
2150                                                                 SERVICENAME_URLTRANSFORMER),
2151                                                              UNO_QUERY );
2152 }
2153 
2154 }
2155