update emoji autocorrect entries from po-files
[LibreOffice.git] / embeddedobj / source / general / docholder.cxx
blobb720f0e9a28693023a8a25cd3dcd3d55e4c71afd
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <com/sun/star/embed/Aspects.hpp>
21 #include <com/sun/star/frame/TaskCreator.hpp>
22 #include <com/sun/star/frame/XComponentLoader.hpp>
23 #include <com/sun/star/frame/XSynchronousFrameLoader.hpp>
24 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
25 #include <com/sun/star/lang/XSingleServiceFactory.hpp>
26 #include <com/sun/star/lang/XSingleComponentFactory.hpp>
27 #include <com/sun/star/util/XCloseBroadcaster.hpp>
28 #include <com/sun/star/util/XCloseable.hpp>
29 #include <com/sun/star/container/XNameAccess.hpp>
30 #include <com/sun/star/lang/XServiceInfo.hpp>
31 #include <com/sun/star/beans/XPropertySet.hpp>
32 #include <com/sun/star/beans/NamedValue.hpp>
33 #include <com/sun/star/frame/XModel.hpp>
34 #include <com/sun/star/frame/Desktop.hpp>
35 #include <com/sun/star/frame/XFramesSupplier.hpp>
36 #include <com/sun/star/frame/XDispatchHelper.hpp>
37 #include <com/sun/star/frame/FrameSearchFlag.hpp>
38 #include <com/sun/star/frame/XControllerBorder.hpp>
39 #include <com/sun/star/util/XModifyBroadcaster.hpp>
40 #include <com/sun/star/frame/XDispatchProviderInterception.hpp>
41 #include <com/sun/star/awt/Toolkit.hpp>
42 #include <com/sun/star/awt/XTopWindow.hpp>
43 #include <com/sun/star/awt/PosSize.hpp>
44 #include <com/sun/star/awt/XView.hpp>
45 #include <com/sun/star/awt/WindowAttribute.hpp>
46 #include <com/sun/star/awt/VclWindowPeerAttribute.hpp>
47 #include <com/sun/star/bridge/XBridgeSupplier2.hpp>
48 #include <com/sun/star/bridge/ModelDependent.hpp>
49 #include <com/sun/star/embed/XHatchWindow.hpp>
50 #include <com/sun/star/embed/HatchWindowFactory.hpp>
51 #include <com/sun/star/embed/XInplaceClient.hpp>
52 #include <com/sun/star/frame/XLayoutManager.hpp>
53 #include <com/sun/star/frame/XMenuBarMergingAcceptor.hpp>
54 #include <com/sun/star/frame/ModuleManager.hpp>
55 #include <com/sun/star/ui/XDockingAreaAcceptor.hpp>
56 #include <com/sun/star/ui/XUIElementSettings.hpp>
57 #include <com/sun/star/ui/XUIConfigurationManager.hpp>
58 #include <com/sun/star/ui/XUIConfigurationManagerSupplier.hpp>
59 #include <com/sun/star/ui/theModuleUIConfigurationManagerSupplier.hpp>
60 #include <com/sun/star/embed/StateChangeInProgressException.hpp>
62 #include <com/sun/star/embed/EmbedMisc.hpp>
63 #include <com/sun/star/embed/EmbedStates.hpp>
64 #include <osl/diagnose.h>
65 #include <rtl/process.h>
66 #include <vcl/svapp.hxx>
67 #include <svtools/embedhlp.hxx>
69 #include <comphelper/processfactory.hxx>
70 #include <comphelper/namedvaluecollection.hxx>
72 #include "docholder.hxx"
73 #include "commonembobj.hxx"
74 #include "intercept.hxx"
76 #define HATCH_BORDER_WIDTH (((m_pEmbedObj->getStatus(embed::Aspects::MSOLE_CONTENT)&embed::EmbedMisc::MS_EMBED_ACTIVATEWHENVISIBLE) && \
77 m_pEmbedObj->getCurrentState()!=embed::EmbedStates::UI_ACTIVE) ? 0 : 4 )
79 using namespace ::com::sun::star;
83 class IntCounterGuard {
84 sal_Int32& m_nFlag;
85 public:
86 IntCounterGuard( sal_Int32& nFlag )
87 : m_nFlag( nFlag )
89 m_nFlag++;
92 ~IntCounterGuard()
94 if ( m_nFlag )
95 m_nFlag--;
101 static void InsertMenu_Impl( const uno::Reference< container::XIndexContainer >& xTargetMenu,
102 sal_Int32 nTargetIndex,
103 const uno::Reference< container::XIndexAccess >& xSourceMenu,
104 sal_Int32 nSourceIndex,
105 const OUString& aContModuleName,
106 const uno::Reference< frame::XDispatchProvider >& xSourceDisp )
108 sal_Int32 nInd = 0;
109 OUString aModuleIdentPropName( "ModuleIdentifier" );
110 OUString aDispProvPropName( "DispatchProvider" );
111 bool bModuleNameSet = false;
112 bool bDispProvSet = false;
114 uno::Sequence< beans::PropertyValue > aSourceProps;
115 xSourceMenu->getByIndex( nSourceIndex ) >>= aSourceProps;
116 uno::Sequence< beans::PropertyValue > aTargetProps( aSourceProps.getLength() );
117 for ( nInd = 0; nInd < aSourceProps.getLength(); nInd++ )
119 aTargetProps[nInd].Name = aSourceProps[nInd].Name;
120 if ( !aContModuleName.isEmpty() && aTargetProps[nInd].Name.equals( aModuleIdentPropName ) )
122 aTargetProps[nInd].Value <<= aContModuleName;
123 bModuleNameSet = true;
125 else if ( aTargetProps[nInd].Name.equals( aDispProvPropName ) )
127 aTargetProps[nInd].Value <<= xSourceDisp;
128 bDispProvSet = true;
130 else
131 aTargetProps[nInd].Value = aSourceProps[nInd].Value;
134 if ( !bModuleNameSet && !aContModuleName.isEmpty() )
136 aTargetProps.realloc( ++nInd );
137 aTargetProps[nInd-1].Name = aModuleIdentPropName;
138 aTargetProps[nInd-1].Value <<= aContModuleName;
141 if ( !bDispProvSet && xSourceDisp.is() )
143 aTargetProps.realloc( ++nInd );
144 aTargetProps[nInd-1].Name = aDispProvPropName;
145 aTargetProps[nInd-1].Value <<= xSourceDisp;
148 xTargetMenu->insertByIndex( nTargetIndex, uno::makeAny( aTargetProps ) );
152 DocumentHolder::DocumentHolder( const uno::Reference< uno::XComponentContext >& xContext,
153 OCommonEmbeddedObject* pEmbObj )
154 : m_pEmbedObj( pEmbObj ),
155 m_pInterceptor( NULL ),
156 m_xContext( xContext ),
157 m_bReadOnly( false ),
158 m_bWaitForClose( false ),
159 m_bAllowClosing( false ),
160 m_bDesktopTerminated( false ),
161 m_nNoBorderResizeReact( 0 ),
162 m_nNoResizeReact( 0 )
164 m_aOutplaceFrameProps.realloc( 3 );
165 beans::NamedValue aArg;
167 aArg.Name = "TopWindow";
168 aArg.Value <<= sal_True;
169 m_aOutplaceFrameProps[0] <<= aArg;
171 aArg.Name = "MakeVisible";
172 aArg.Value <<= sal_False;
173 m_aOutplaceFrameProps[1] <<= aArg;
175 uno::Reference< frame::XDesktop2 > xDesktop = frame::Desktop::create( m_xContext );
176 m_refCount++;
179 xDesktop->addTerminateListener( this );
181 catch ( const uno::Exception& )
184 m_refCount--;
186 aArg.Name = "ParentFrame";
187 aArg.Value <<= xDesktop; //TODO/LATER: should use parent document frame
188 m_aOutplaceFrameProps[2] <<= aArg;
192 DocumentHolder::~DocumentHolder()
194 m_refCount++; // to allow deregistration as a listener
196 if( m_xFrame.is() )
197 CloseFrame();
199 if ( m_xComponent.is() )
201 try {
202 CloseDocument( true, false );
203 } catch( const uno::Exception& ) {}
206 if ( m_pInterceptor )
208 m_pInterceptor->DisconnectDocHolder();
209 m_pInterceptor->release();
212 if ( !m_bDesktopTerminated )
213 FreeOffice();
217 void DocumentHolder::CloseFrame()
219 uno::Reference< util::XCloseBroadcaster > xCloseBroadcaster( m_xFrame, uno::UNO_QUERY );
220 if ( xCloseBroadcaster.is() )
221 xCloseBroadcaster->removeCloseListener( ( util::XCloseListener* )this );
223 uno::Reference<util::XCloseable> xCloseable(
224 m_xFrame,uno::UNO_QUERY );
225 if( xCloseable.is() )
226 try {
227 xCloseable->close( sal_True );
229 catch( const uno::Exception& ) {
231 else {
232 uno::Reference<lang::XComponent> xComp( m_xFrame,uno::UNO_QUERY );
233 if( xComp.is() )
234 xComp->dispose();
237 uno::Reference< lang::XComponent > xComp( m_xHatchWindow, uno::UNO_QUERY );
238 if ( xComp.is() )
239 xComp->dispose();
241 m_xHatchWindow = uno::Reference< awt::XWindow >();
242 m_xOwnWindow = uno::Reference< awt::XWindow >();
243 m_xFrame = uno::Reference< frame::XFrame >();
247 void DocumentHolder::FreeOffice()
249 uno::Reference< frame::XDesktop2 > xDesktop = frame::Desktop::create( m_xContext );
250 xDesktop->removeTerminateListener( this );
252 // the following code is commented out since for now there is still no completely correct way to detect
253 // whether the office can be terminated, so it is better to have unnecessary process running than
254 // to lose any data
256 // uno::Reference< frame::XFramesSupplier > xFramesSupplier( xDesktop, uno::UNO_QUERY );
257 // if ( xFramesSupplier.is() )
258 // {
259 // uno::Reference< frame::XFrames > xFrames = xFramesSupplier->getFrames();
260 // if ( xFrames.is() && !xFrames->hasElements() )
261 // {
262 // try
263 // {
264 // xDesktop->terminate();
265 // }
266 // catch( uno::Exception & )
267 // {}
268 // }
269 // }
273 void DocumentHolder::CloseDocument( bool bDeliverOwnership, bool bWaitForClose )
275 uno::Reference< util::XCloseBroadcaster > xBroadcaster( m_xComponent, uno::UNO_QUERY );
276 if ( xBroadcaster.is() )
278 uno::Reference< document::XEventBroadcaster > xEventBroadcaster( m_xComponent, uno::UNO_QUERY );
279 if ( xEventBroadcaster.is() )
280 xEventBroadcaster->removeEventListener( ( document::XEventListener* )this );
281 else
283 // the object does not support document::XEventBroadcaster interface
284 // use the workaround, register for modified events
285 uno::Reference< util::XModifyBroadcaster > xModifyBroadcaster( m_xComponent, uno::UNO_QUERY );
286 if ( xModifyBroadcaster.is() )
287 xModifyBroadcaster->removeModifyListener( ( util::XModifyListener* )this );
290 uno::Reference< util::XCloseable > xCloseable( xBroadcaster, uno::UNO_QUERY );
291 if ( xCloseable.is() )
293 m_bAllowClosing = true;
294 m_bWaitForClose = bWaitForClose;
295 xCloseable->close( bDeliverOwnership );
299 m_xComponent = 0;
303 void DocumentHolder::PlaceFrame( const awt::Rectangle& aNewRect )
305 OSL_ENSURE( m_xFrame.is() && m_xOwnWindow.is(),
306 "The object does not have windows required for inplace mode!" );
308 //TODO: may need mutex locking???
309 if ( m_xFrame.is() && m_xOwnWindow.is() )
311 // the frame can be replaced only in inplace mode
312 frame::BorderWidths aOldWidths;
313 IntCounterGuard aGuard( m_nNoBorderResizeReact );
317 aOldWidths = m_aBorderWidths;
319 awt::Rectangle aHatchRect = AddBorderToArea( aNewRect );
321 ResizeWindows_Impl( aHatchRect );
323 } while ( aOldWidths.Left != m_aBorderWidths.Left
324 || aOldWidths.Top != m_aBorderWidths.Top
325 || aOldWidths.Right != m_aBorderWidths.Right
326 || aOldWidths.Bottom != m_aBorderWidths.Bottom );
328 m_aObjRect = aNewRect;
333 void DocumentHolder::ResizeWindows_Impl( const awt::Rectangle& aHatchRect )
335 OSL_ENSURE( m_xFrame.is() && m_xOwnWindow.is() /*&& m_xHatchWindow.is()*/,
336 "The object does not have windows required for inplace mode!" );
337 if ( m_xHatchWindow.is() )
339 m_xOwnWindow->setPosSize( HATCH_BORDER_WIDTH,
340 HATCH_BORDER_WIDTH,
341 aHatchRect.Width - 2*HATCH_BORDER_WIDTH,
342 aHatchRect.Height - 2*HATCH_BORDER_WIDTH,
343 awt::PosSize::POSSIZE );
346 m_xHatchWindow->setPosSize( aHatchRect.X,
347 aHatchRect.Y,
348 aHatchRect.Width,
349 aHatchRect.Height,
350 awt::PosSize::POSSIZE );
352 else
353 m_xOwnWindow->setPosSize( aHatchRect.X + HATCH_BORDER_WIDTH,
354 aHatchRect.Y + HATCH_BORDER_WIDTH,
355 aHatchRect.Width - 2*HATCH_BORDER_WIDTH,
356 aHatchRect.Height - 2*HATCH_BORDER_WIDTH,
357 awt::PosSize::POSSIZE );
361 bool DocumentHolder::SetFrameLMVisibility( const uno::Reference< frame::XFrame >& xFrame, bool bVisible )
363 bool bResult = false;
367 uno::Reference< ::com::sun::star::frame::XLayoutManager > xLayoutManager;
368 uno::Reference< beans::XPropertySet > xPropSet( xFrame, uno::UNO_QUERY_THROW );
369 xPropSet->getPropertyValue("LayoutManager") >>= xLayoutManager;
370 if ( xLayoutManager.is() )
372 xLayoutManager->setVisible( bVisible );
374 // MBA: locking is done only on the container LM, because it is not about hiding windows, it's about
375 // giving up control over the component window (and stopping to listen for resize events of the container window)
376 if ( bVisible )
377 xLayoutManager->unlock();
378 else
379 xLayoutManager->lock();
381 bResult = true;
384 catch( const uno::Exception& )
387 return bResult;
391 bool DocumentHolder::ShowInplace( const uno::Reference< awt::XWindowPeer >& xParent,
392 const awt::Rectangle& aRectangleToShow,
393 const uno::Reference< frame::XDispatchProvider >& xContDisp )
395 OSL_ENSURE( !m_xFrame.is(), "A frame exists already!" );
397 if ( !m_xFrame.is() )
399 uno::Reference < frame::XModel > xModel( GetComponent(), uno::UNO_QUERY );
400 awt::Rectangle aHatchRectangle = AddBorderToArea( aRectangleToShow );
402 awt::Rectangle aOwnRectangle( HATCH_BORDER_WIDTH,
403 HATCH_BORDER_WIDTH,
404 aHatchRectangle.Width - 2*HATCH_BORDER_WIDTH,
405 aHatchRectangle.Height - 2*HATCH_BORDER_WIDTH );
406 uno::Reference< awt::XWindow > xHWindow;
407 uno::Reference< awt::XWindowPeer > xMyParent( xParent );
409 if ( xModel.is() )
412 uno::Reference< embed::XHatchWindowFactory > xHatchFactory =
413 embed::HatchWindowFactory::create(m_xContext);
415 uno::Reference< embed::XHatchWindow > xHatchWindow =
416 xHatchFactory->createHatchWindowInstance( xParent,
417 aHatchRectangle,
418 awt::Size( HATCH_BORDER_WIDTH, HATCH_BORDER_WIDTH ) );
420 uno::Reference< awt::XWindowPeer > xHatchWinPeer( xHatchWindow, uno::UNO_QUERY );
421 xHWindow = uno::Reference< awt::XWindow >( xHatchWinPeer, uno::UNO_QUERY );
422 if ( !xHWindow.is() )
423 throw uno::RuntimeException(); // TODO: can not create own window
425 xHatchWindow->setController( uno::Reference< embed::XHatchWindowController >(
426 static_cast< embed::XHatchWindowController* >( this ) ) );
428 xMyParent = xHatchWinPeer;
430 else
432 aOwnRectangle.X += aHatchRectangle.X;
433 aOwnRectangle.Y += aHatchRectangle.Y;
436 awt::WindowDescriptor aOwnWinDescriptor( awt::WindowClass_TOP,
437 OUString("dockingwindow"),
438 xMyParent,
440 awt::Rectangle(),//aOwnRectangle,
441 awt::WindowAttribute::SHOW | awt::VclWindowPeerAttribute::CLIPCHILDREN );
443 uno::Reference< awt::XToolkit2 > xToolkit = awt::Toolkit::create(m_xContext);
445 uno::Reference< awt::XWindowPeer > xNewWinPeer = xToolkit->createWindow( aOwnWinDescriptor );
446 uno::Reference< awt::XWindow > xOwnWindow( xNewWinPeer, uno::UNO_QUERY );
447 if ( !xOwnWindow.is() )
448 throw uno::RuntimeException(); // TODO: can not create own window
450 // create a frame based on the specified window
451 uno::Reference< lang::XSingleServiceFactory > xFrameFact = frame::TaskCreator::create(m_xContext);
453 uno::Sequence< uno::Any > aArgs( 2 );
454 beans::NamedValue aArg;
456 aArg.Name = "ContainerWindow";
457 aArg.Value <<= xOwnWindow;
458 aArgs[0] <<= aArg;
460 uno::Reference< frame::XFrame > xContFrame( xContDisp, uno::UNO_QUERY );
461 if ( xContFrame.is() )
463 aArg.Name = "ParentFrame";
464 aArg.Value <<= xContFrame;
465 aArgs[1] <<= aArg;
467 else
468 aArgs.realloc( 1 );
470 // the call will create, initialize the frame, and register it in the parent
471 m_xFrame.set( xFrameFact->createInstanceWithArguments( aArgs ), uno::UNO_QUERY_THROW );
473 m_xHatchWindow = xHWindow;
474 m_xOwnWindow = xOwnWindow;
476 if ( !SetFrameLMVisibility( m_xFrame, false ) )
478 OSL_FAIL( "Can't deactivate LayoutManager!\n" );
479 // TODO/LATER: error handling?
482 // m_bIsInplace = sal_True; TODO: ?
484 uno::Reference< util::XCloseBroadcaster > xCloseBroadcaster( m_xFrame, uno::UNO_QUERY );
485 if ( xCloseBroadcaster.is() )
486 xCloseBroadcaster->addCloseListener( ( util::XCloseListener* )this );
488 // TODO: some listeners to the frame and the window ( resize for example )
491 if ( m_xComponent.is() )
493 if ( !LoadDocToFrame( true ) )
495 CloseFrame();
496 return false;
499 uno::Reference< frame::XControllerBorder > xControllerBorder( m_xFrame->getController(), uno::UNO_QUERY );
500 if ( xControllerBorder.is() )
502 m_aBorderWidths = xControllerBorder->getBorder();
503 xControllerBorder->addBorderResizeListener( (frame::XBorderResizeListener*)this );
506 PlaceFrame( aRectangleToShow );
508 if ( m_xHatchWindow.is() )
509 m_xHatchWindow->setVisible( sal_True );
511 return true;
514 return false;
518 uno::Reference< container::XIndexAccess > DocumentHolder::RetrieveOwnMenu_Impl()
520 uno::Reference< container::XIndexAccess > xResult;
522 uno::Reference< ::com::sun::star::ui::XUIConfigurationManagerSupplier > xUIConfSupplier(
523 m_xComponent,
524 uno::UNO_QUERY );
525 uno::Reference< ::com::sun::star::ui::XUIConfigurationManager > xUIConfigManager;
526 if( xUIConfSupplier.is())
528 xUIConfigManager.set(
529 xUIConfSupplier->getUIConfigurationManager(),
530 uno::UNO_QUERY_THROW );
535 if( xUIConfigManager.is())
537 xResult = xUIConfigManager->getSettings(
538 OUString( "private:resource/menubar/menubar" ),
539 sal_False );
542 catch( const uno::Exception& )
545 if ( !xResult.is() )
547 // no internal document configuration, use the one from the module
548 uno::Reference< frame::XModuleManager2 > xModuleMan = frame::ModuleManager::create(m_xContext);
549 OUString aModuleIdent =
550 xModuleMan->identify( uno::Reference< uno::XInterface >( m_xComponent, uno::UNO_QUERY ) );
552 if ( !aModuleIdent.isEmpty() )
554 uno::Reference< ui::XModuleUIConfigurationManagerSupplier > xModConfSupplier =
555 ui::theModuleUIConfigurationManagerSupplier::get(m_xContext);
556 uno::Reference< ::com::sun::star::ui::XUIConfigurationManager > xModUIConfMan(
557 xModConfSupplier->getUIConfigurationManager( aModuleIdent ),
558 uno::UNO_QUERY_THROW );
559 xResult = xModUIConfMan->getSettings(
560 OUString( "private:resource/menubar/menubar" ),
561 sal_False );
565 if ( !xResult.is() )
566 throw uno::RuntimeException();
568 return xResult;
572 void DocumentHolder::FindConnectPoints(
573 const uno::Reference< container::XIndexAccess >& xMenu,
574 sal_Int32 nConnectPoints[2] )
575 throw ( uno::Exception )
577 nConnectPoints[0] = -1;
578 nConnectPoints[1] = -1;
579 for ( sal_Int32 nInd = 0; nInd < xMenu->getCount(); nInd++ )
581 uno::Sequence< beans::PropertyValue > aProps;
582 xMenu->getByIndex( nInd ) >>= aProps;
583 OUString aCommand;
584 for ( sal_Int32 nSeqInd = 0; nSeqInd < aProps.getLength(); nSeqInd++ )
585 if ( aProps[nSeqInd].Name == "CommandURL" )
587 aProps[nSeqInd].Value >>= aCommand;
588 break;
591 if ( aCommand.isEmpty() )
592 throw uno::RuntimeException();
594 if ( aCommand == ".uno:PickList" )
595 nConnectPoints[0] = nInd;
596 else if ( aCommand == ".uno:WindowList" )
597 nConnectPoints[1] = nInd;
602 uno::Reference< container::XIndexAccess > DocumentHolder::MergeMenusForInplace(
603 const uno::Reference< container::XIndexAccess >& xContMenu,
604 const uno::Reference< frame::XDispatchProvider >& xContDisp,
605 const OUString& aContModuleName,
606 const uno::Reference< container::XIndexAccess >& xOwnMenu,
607 const uno::Reference< frame::XDispatchProvider >& xOwnDisp )
608 throw ( uno::Exception )
610 // TODO/LATER: use dispatch providers on merge
612 sal_Int32 nContPoints[2];
613 sal_Int32 nOwnPoints[2];
615 uno::Reference< lang::XSingleComponentFactory > xIndAccessFact( xContMenu, uno::UNO_QUERY_THROW );
617 uno::Reference< container::XIndexContainer > xMergedMenu(
618 xIndAccessFact->createInstanceWithContext(
619 comphelper::getProcessComponentContext() ),
620 uno::UNO_QUERY_THROW );
622 FindConnectPoints( xContMenu, nContPoints );
623 FindConnectPoints( xOwnMenu, nOwnPoints );
625 for ( sal_Int32 nInd = 0; nInd < xOwnMenu->getCount(); nInd++ )
627 if ( nOwnPoints[0] == nInd )
629 if ( nContPoints[0] >= 0 && nContPoints[0] < xContMenu->getCount() )
631 InsertMenu_Impl( xMergedMenu, nInd, xContMenu, nContPoints[0], aContModuleName, xContDisp );
634 else if ( nOwnPoints[1] == nInd )
636 if ( nContPoints[1] >= 0 && nContPoints[1] < xContMenu->getCount() )
638 InsertMenu_Impl( xMergedMenu, nInd, xContMenu, nContPoints[1], aContModuleName, xContDisp );
641 else
642 InsertMenu_Impl( xMergedMenu, nInd, xOwnMenu, nInd, OUString(), xOwnDisp );
645 return uno::Reference< container::XIndexAccess >( xMergedMenu, uno::UNO_QUERY_THROW );
649 bool DocumentHolder::MergeMenus_Impl( const uno::Reference< ::com::sun::star::frame::XLayoutManager >& xOwnLM,
650 const uno::Reference< ::com::sun::star::frame::XLayoutManager >& xContLM,
651 const uno::Reference< frame::XDispatchProvider >& xContDisp,
652 const OUString& aContModuleName )
654 bool bMenuMerged = false;
657 uno::Reference< ::com::sun::star::ui::XUIElementSettings > xUISettings(
658 xContLM->getElement(
659 OUString( "private:resource/menubar/menubar" ) ),
660 uno::UNO_QUERY_THROW );
661 uno::Reference< container::XIndexAccess > xContMenu = xUISettings->getSettings( sal_True );
662 if ( !xContMenu.is() )
663 throw uno::RuntimeException();
665 uno::Reference< container::XIndexAccess > xOwnMenu = RetrieveOwnMenu_Impl();
666 uno::Reference< frame::XDispatchProvider > xOwnDisp( m_xFrame, uno::UNO_QUERY_THROW );
668 uno::Reference< container::XIndexAccess > xMergedMenu = MergeMenusForInplace( xContMenu, xContDisp, aContModuleName, xOwnMenu, xOwnDisp );
669 uno::Reference< ::com::sun::star::frame::XMenuBarMergingAcceptor > xMerge( xOwnLM,
670 uno::UNO_QUERY_THROW );
671 bMenuMerged = xMerge->setMergedMenuBar( xMergedMenu );
673 catch( const uno::Exception& )
676 return bMenuMerged;
679 bool DocumentHolder::ShowUI( const uno::Reference< ::com::sun::star::frame::XLayoutManager >& xContainerLM,
680 const uno::Reference< frame::XDispatchProvider >& xContainerDP,
681 const OUString& aContModuleName )
683 bool bResult = false;
684 if ( xContainerLM.is() )
686 // the LM of the embedded frame and its current DockingAreaAcceptor
687 uno::Reference< ::com::sun::star::frame::XLayoutManager > xOwnLM;
688 uno::Reference< ::com::sun::star::ui::XDockingAreaAcceptor > xDocAreaAcc;
692 uno::Reference< beans::XPropertySet > xPropSet( m_xFrame, uno::UNO_QUERY_THROW );
693 xPropSet->getPropertyValue("LayoutManager") >>= xOwnLM;
694 xDocAreaAcc = xContainerLM->getDockingAreaAcceptor();
696 catch( const uno::Exception& ){}
698 if ( xOwnLM.is() && xDocAreaAcc.is() )
700 // make sure that lock state of LM is correct even if an exception is thrown in between
701 bool bUnlock = false;
702 bool bLock = false;
705 // take over the control over the containers window
706 // as long as the LM is invisible and locked an empty tool space will be used on resizing
707 xOwnLM->setDockingAreaAcceptor( xDocAreaAcc );
709 // try to merge menus; don't do anything else if it fails
710 if ( MergeMenus_Impl( xOwnLM, xContainerLM, xContainerDP, aContModuleName ) )
712 // make sure that the container LM does not control the size of the containers window anymore
713 // this must be done after merging menus as we won't get the container menu otherwise
714 xContainerLM->setDockingAreaAcceptor( uno::Reference < ui::XDockingAreaAcceptor >() );
716 // prevent further changes at this LM
717 xContainerLM->setVisible( sal_False );
718 xContainerLM->lock();
719 bUnlock = true;
721 // by unlocking the LM each layout change will now resize the containers window; pending layouts will be processed now
722 xOwnLM->setVisible( sal_True );
724 uno::Reference< frame::XFramesSupplier > xSupp( m_xFrame->getCreator(), uno::UNO_QUERY );
725 if ( xSupp.is() )
726 xSupp->setActiveFrame( m_xFrame );
728 xOwnLM->unlock();
729 bLock = true;
730 bResult = true;
732 // TODO/LATER: The following action should be done only if the window is not hidden
733 // otherwise the activation must fail, unfortunately currently it is not possible
734 // to detect whether the window is hidden using UNO API
735 m_xOwnWindow->setFocus();
738 catch( const uno::Exception& )
740 // activation failed; reestablish old state
743 uno::Reference< frame::XFramesSupplier > xSupp( m_xFrame->getCreator(), uno::UNO_QUERY );
744 if ( xSupp.is() )
745 xSupp->setActiveFrame( 0 );
747 // remove control about containers window from own LM
748 if ( bLock )
749 xOwnLM->lock();
750 xOwnLM->setVisible( sal_False );
751 xOwnLM->setDockingAreaAcceptor( uno::Reference< ::com::sun::star::ui::XDockingAreaAcceptor >() );
753 // unmerge menu
754 uno::Reference< ::com::sun::star::frame::XMenuBarMergingAcceptor > xMerge( xOwnLM, uno::UNO_QUERY_THROW );
755 xMerge->removeMergedMenuBar();
757 catch( const uno::Exception& ) {}
761 // reestablish control of containers window
762 xContainerLM->setDockingAreaAcceptor( xDocAreaAcc );
763 xContainerLM->setVisible( sal_True );
764 if ( bUnlock )
765 xContainerLM->unlock();
767 catch( const uno::Exception& ) {}
772 return bResult;
776 bool DocumentHolder::HideUI( const uno::Reference< ::com::sun::star::frame::XLayoutManager >& xContainerLM )
778 bool bResult = false;
780 if ( xContainerLM.is() )
782 uno::Reference< ::com::sun::star::frame::XLayoutManager > xOwnLM;
784 try {
785 uno::Reference< beans::XPropertySet > xPropSet( m_xFrame, uno::UNO_QUERY_THROW );
786 xPropSet->getPropertyValue("LayoutManager") >>= xOwnLM;
787 } catch( const uno::Exception& )
790 if ( xOwnLM.is() )
792 try {
793 uno::Reference< frame::XFramesSupplier > xSupp( m_xFrame->getCreator(), uno::UNO_QUERY );
794 if ( xSupp.is() )
795 xSupp->setActiveFrame( 0 );
797 uno::Reference< ::com::sun::star::ui::XDockingAreaAcceptor > xDocAreaAcc = xOwnLM->getDockingAreaAcceptor();
799 xOwnLM->setDockingAreaAcceptor( uno::Reference < ui::XDockingAreaAcceptor >() );
800 xOwnLM->lock();
801 xOwnLM->setVisible( sal_False );
803 uno::Reference< ::com::sun::star::frame::XMenuBarMergingAcceptor > xMerge( xOwnLM, uno::UNO_QUERY_THROW );
804 xMerge->removeMergedMenuBar();
806 xContainerLM->setDockingAreaAcceptor( xDocAreaAcc );
807 xContainerLM->setVisible( sal_True );
808 xContainerLM->unlock();
810 xContainerLM->doLayout();
811 bResult = true;
813 catch( const uno::Exception& )
815 SetFrameLMVisibility( m_xFrame, true );
820 return bResult;
824 uno::Reference< frame::XFrame > DocumentHolder::GetDocFrame()
826 // the frame for outplace activation
827 if ( !m_xFrame.is() )
829 uno::Reference< lang::XSingleServiceFactory > xFrameFact = frame::TaskCreator::create(m_xContext);
831 m_xFrame.set(xFrameFact->createInstanceWithArguments( m_aOutplaceFrameProps ), uno::UNO_QUERY_THROW);
833 uno::Reference< frame::XDispatchProviderInterception > xInterception( m_xFrame, uno::UNO_QUERY );
834 if ( xInterception.is() )
836 if ( m_pInterceptor )
838 m_pInterceptor->DisconnectDocHolder();
839 m_pInterceptor->release();
840 m_pInterceptor = NULL;
843 m_pInterceptor = new Interceptor( this );
844 m_pInterceptor->acquire();
846 // register interceptor from outside
847 if ( m_xOutplaceInterceptor.is() )
848 xInterception->registerDispatchProviderInterceptor( m_xOutplaceInterceptor );
850 xInterception->registerDispatchProviderInterceptor( m_pInterceptor );
853 uno::Reference< util::XCloseBroadcaster > xCloseBroadcaster( m_xFrame, uno::UNO_QUERY );
854 if ( xCloseBroadcaster.is() )
855 xCloseBroadcaster->addCloseListener( ( util::XCloseListener* )this );
858 if ( m_xComponent.is() )
860 uno::Reference< ::com::sun::star::frame::XLayoutManager > xOwnLM;
861 try {
862 uno::Reference< beans::XPropertySet > xPropSet( m_xFrame, uno::UNO_QUERY_THROW );
863 xPropSet->getPropertyValue("LayoutManager") >>= xOwnLM;
864 } catch( const uno::Exception& )
867 if ( xOwnLM.is() )
868 xOwnLM->lock();
870 // TODO/LATER: get it for the real aspect
871 awt::Size aSize;
872 GetExtent( embed::Aspects::MSOLE_CONTENT, &aSize );
873 LoadDocToFrame(false);
875 if ( xOwnLM.is() )
877 xOwnLM->unlock();
878 xOwnLM->lock();
881 SetExtent( embed::Aspects::MSOLE_CONTENT, aSize );
883 if ( xOwnLM.is() )
884 xOwnLM->unlock();
889 uno::Reference< awt::XWindow > xHWindow = m_xFrame->getContainerWindow();
891 if( xHWindow.is() )
893 sal_Int32 nDisplay = Application::GetDisplayBuiltInScreen();
895 Rectangle aWorkRect = Application::GetScreenPosSizePixel( nDisplay );
896 awt::Rectangle aWindowRect = xHWindow->getPosSize();
898 if (( aWindowRect.Width < aWorkRect.GetWidth()) && ( aWindowRect.Height < aWorkRect.GetHeight() ))
900 int OffsetX = ( aWorkRect.GetWidth() - aWindowRect.Width ) / 2 + aWorkRect.Left();
901 int OffsetY = ( aWorkRect.GetHeight() - aWindowRect.Height ) /2 + aWorkRect.Top();
902 xHWindow->setPosSize( OffsetX, OffsetY, aWindowRect.Width, aWindowRect.Height, awt::PosSize::POS );
904 else
906 xHWindow->setPosSize( aWorkRect.Left(), aWorkRect.Top(), aWorkRect.GetWidth(), aWorkRect.GetHeight(), awt::PosSize::POSSIZE );
909 xHWindow->setVisible( sal_True );
912 catch ( const uno::Exception& )
916 return m_xFrame;
920 void DocumentHolder::SetComponent( const uno::Reference< util::XCloseable >& xDoc, bool bReadOnly )
922 if ( m_xComponent.is() )
924 // May be should be improved
925 try {
926 CloseDocument( true, false );
927 } catch( const uno::Exception& )
931 m_xComponent = xDoc;
933 m_bReadOnly = bReadOnly;
934 m_bAllowClosing = false;
936 uno::Reference< util::XCloseBroadcaster > xBroadcaster( m_xComponent, uno::UNO_QUERY );
937 if ( xBroadcaster.is() )
938 xBroadcaster->addCloseListener( ( util::XCloseListener* )this );
940 uno::Reference< document::XEventBroadcaster > xEventBroadcaster( m_xComponent, uno::UNO_QUERY );
941 if ( xEventBroadcaster.is() )
942 xEventBroadcaster->addEventListener( ( document::XEventListener* )this );
943 else
945 // the object does not support document::XEventBroadcaster interface
946 // use the workaround, register for modified events
947 uno::Reference< util::XModifyBroadcaster > xModifyBroadcaster( m_xComponent, uno::UNO_QUERY );
948 if ( xModifyBroadcaster.is() )
949 xModifyBroadcaster->addModifyListener( ( util::XModifyListener* )this );
952 if ( m_xFrame.is() )
953 LoadDocToFrame(false);
957 bool DocumentHolder::LoadDocToFrame( bool bInPlace )
959 if ( m_xFrame.is() && m_xComponent.is() )
961 uno::Reference < frame::XModel > xDoc( m_xComponent, uno::UNO_QUERY );
962 if ( xDoc.is() )
964 // load new document in to the frame
965 uno::Reference< frame::XComponentLoader > xComponentLoader( m_xFrame, uno::UNO_QUERY_THROW );
967 ::comphelper::NamedValueCollection aArgs;
968 aArgs.put( "Model", m_xComponent );
969 aArgs.put( "ReadOnly", m_bReadOnly );
970 if ( bInPlace )
971 aArgs.put( "PluginMode", sal_Int16(1) );
972 OUString sUrl;
973 uno::Reference< lang::XServiceInfo> xServiceInfo(xDoc,uno::UNO_QUERY);
974 if ( xServiceInfo.is()
975 && xServiceInfo->supportsService("com.sun.star.report.ReportDefinition") )
977 sUrl = ".component:DB/ReportDesign";
979 else if( xServiceInfo.is()
980 && xServiceInfo->supportsService("com.sun.star.chart2.ChartDocument"))
981 sUrl = "private:factory/schart";
982 else
983 sUrl = "private:object";
985 xComponentLoader->loadComponentFromURL( sUrl,
986 OUString( "_self" ),
988 aArgs.getPropertyValues() );
990 return true;
992 else
994 uno::Reference < frame::XSynchronousFrameLoader > xLoader( m_xComponent, uno::UNO_QUERY );
995 if ( xLoader.is() )
996 return xLoader->load( uno::Sequence < beans::PropertyValue >(), m_xFrame );
997 else
998 return false;
1002 return true;
1006 void DocumentHolder::Show()
1008 if( m_xFrame.is() )
1010 m_xFrame->activate();
1011 uno::Reference<awt::XTopWindow> xTopWindow( m_xFrame->getContainerWindow(), uno::UNO_QUERY );
1012 if( xTopWindow.is() )
1013 xTopWindow->toFront();
1015 else
1016 GetDocFrame();
1020 bool DocumentHolder::SetExtent( sal_Int64 nAspect, const awt::Size& aSize )
1022 uno::Reference< embed::XVisualObject > xDocVis( m_xComponent, uno::UNO_QUERY );
1023 if ( xDocVis.is() )
1027 xDocVis->setVisualAreaSize( nAspect, aSize );
1028 return true;
1030 catch( const uno::Exception& )
1032 // TODO: Error handling
1036 return false;
1040 bool DocumentHolder::GetExtent( sal_Int64 nAspect, awt::Size *pSize )
1042 uno::Reference< embed::XVisualObject > xDocVis( m_xComponent, uno::UNO_QUERY );
1043 if ( pSize && xDocVis.is() )
1047 *pSize = xDocVis->getVisualAreaSize( nAspect );
1048 return true;
1050 catch( const uno::Exception& )
1052 // TODO: Error handling
1056 return false;
1060 sal_Int32 DocumentHolder::GetMapUnit( sal_Int64 nAspect )
1062 uno::Reference< embed::XVisualObject > xDocVis( m_xComponent, uno::UNO_QUERY );
1063 if ( xDocVis.is() )
1067 return xDocVis->getMapUnit( nAspect );
1069 catch( const uno::Exception& )
1071 // TODO: Error handling
1075 return 0;
1079 awt::Rectangle DocumentHolder::CalculateBorderedArea( const awt::Rectangle& aRect )
1081 return awt::Rectangle( aRect.X + m_aBorderWidths.Left + HATCH_BORDER_WIDTH,
1082 aRect.Y + m_aBorderWidths.Top + HATCH_BORDER_WIDTH,
1083 aRect.Width - m_aBorderWidths.Left - m_aBorderWidths.Right - 2*HATCH_BORDER_WIDTH,
1084 aRect.Height - m_aBorderWidths.Top - m_aBorderWidths.Bottom - 2*HATCH_BORDER_WIDTH );
1088 awt::Rectangle DocumentHolder::AddBorderToArea( const awt::Rectangle& aRect )
1090 return awt::Rectangle( aRect.X - m_aBorderWidths.Left - HATCH_BORDER_WIDTH,
1091 aRect.Y - m_aBorderWidths.Top - HATCH_BORDER_WIDTH,
1092 aRect.Width + m_aBorderWidths.Left + m_aBorderWidths.Right + 2*HATCH_BORDER_WIDTH,
1093 aRect.Height + m_aBorderWidths.Top + m_aBorderWidths.Bottom + 2*HATCH_BORDER_WIDTH );
1097 void SAL_CALL DocumentHolder::disposing( const com::sun::star::lang::EventObject& aSource )
1098 throw (uno::RuntimeException, std::exception)
1100 if ( m_xComponent.is() && m_xComponent == aSource.Source )
1102 m_xComponent = 0;
1103 if ( m_bWaitForClose )
1105 m_bWaitForClose = false;
1106 FreeOffice();
1110 if( m_xFrame.is() && m_xFrame == aSource.Source )
1112 m_xHatchWindow = uno::Reference< awt::XWindow >();
1113 m_xOwnWindow = uno::Reference< awt::XWindow >();
1114 m_xFrame = uno::Reference< frame::XFrame >();
1120 void SAL_CALL DocumentHolder::queryClosing( const lang::EventObject& aSource, sal_Bool /*bGetsOwnership*/ )
1121 throw (util::CloseVetoException, uno::RuntimeException, std::exception)
1123 if ( m_xComponent.is() && m_xComponent == aSource.Source && !m_bAllowClosing )
1124 throw util::CloseVetoException();
1128 void SAL_CALL DocumentHolder::notifyClosing( const lang::EventObject& aSource )
1129 throw (uno::RuntimeException, std::exception)
1131 if ( m_xComponent.is() && m_xComponent == aSource.Source )
1133 m_xComponent = 0;
1134 if ( m_bWaitForClose )
1136 m_bWaitForClose = false;
1137 FreeOffice();
1141 if( m_xFrame.is() && m_xFrame == aSource.Source )
1143 m_xHatchWindow = uno::Reference< awt::XWindow >();
1144 m_xOwnWindow = uno::Reference< awt::XWindow >();
1145 m_xFrame = uno::Reference< frame::XFrame >();
1150 void SAL_CALL DocumentHolder::queryTermination( const lang::EventObject& )
1151 throw (frame::TerminationVetoException, uno::RuntimeException, std::exception)
1153 if ( m_bWaitForClose )
1154 throw frame::TerminationVetoException();
1158 void SAL_CALL DocumentHolder::notifyTermination( const lang::EventObject& aSource )
1159 throw (uno::RuntimeException, std::exception)
1161 OSL_ENSURE( !m_xComponent.is(), "Just a disaster..." );
1163 uno::Reference< frame::XDesktop > xDesktop( aSource.Source, uno::UNO_QUERY );
1164 m_bDesktopTerminated = true;
1165 if ( xDesktop.is() )
1166 xDesktop->removeTerminateListener( ( frame::XTerminateListener* )this );
1170 void SAL_CALL DocumentHolder::modified( const lang::EventObject& aEvent )
1171 throw ( uno::RuntimeException, std::exception )
1173 // if the component does not support document::XEventBroadcaster
1174 // the modify notifications are used as workaround, but only for running state
1175 if( aEvent.Source == m_xComponent && m_pEmbedObj && m_pEmbedObj->getCurrentState() == embed::EmbedStates::RUNNING )
1176 m_pEmbedObj->PostEvent_Impl( OUString( "OnVisAreaChanged" ) );
1180 void SAL_CALL DocumentHolder::notifyEvent( const document::EventObject& Event )
1181 throw ( uno::RuntimeException, std::exception )
1183 if( m_pEmbedObj && Event.Source == m_xComponent )
1185 // for now the ignored events are not forwarded, but sent by the object itself
1186 if ( !Event.EventName.startsWith( "OnSave" )
1187 && !Event.EventName.startsWith( "OnSaveDone" )
1188 && !Event.EventName.startsWith( "OnSaveAs" )
1189 && !Event.EventName.startsWith( "OnSaveAsDone" )
1190 && !( Event.EventName.startsWith( "OnVisAreaChanged" ) && m_nNoResizeReact ) )
1191 m_pEmbedObj->PostEvent_Impl( Event.EventName );
1196 void SAL_CALL DocumentHolder::borderWidthsChanged( const uno::Reference< uno::XInterface >& aObject,
1197 const frame::BorderWidths& aNewSize )
1198 throw ( uno::RuntimeException, std::exception )
1200 // TODO: may require mutex introduction ???
1201 if ( m_pEmbedObj && m_xFrame.is() && aObject == m_xFrame->getController() )
1203 if ( m_aBorderWidths.Left != aNewSize.Left
1204 || m_aBorderWidths.Right != aNewSize.Right
1205 || m_aBorderWidths.Top != aNewSize.Top
1206 || m_aBorderWidths.Bottom != aNewSize.Bottom )
1208 m_aBorderWidths = aNewSize;
1209 if ( !m_nNoBorderResizeReact )
1210 PlaceFrame( m_aObjRect );
1216 void SAL_CALL DocumentHolder::requestPositioning( const awt::Rectangle& aRect )
1217 throw (uno::RuntimeException, std::exception)
1219 // TODO: may require mutex introduction ???
1220 if ( m_pEmbedObj )
1222 // borders should not be counted
1223 awt::Rectangle aObjRect = CalculateBorderedArea( aRect );
1224 IntCounterGuard aGuard( m_nNoResizeReact );
1225 m_pEmbedObj->requestPositioning( aObjRect );
1230 awt::Rectangle SAL_CALL DocumentHolder::calcAdjustedRectangle( const awt::Rectangle& aRect )
1231 throw (uno::RuntimeException, std::exception)
1233 // Solar mutex should be locked already since this is a call from HatchWindow with focus
1234 awt::Rectangle aResult( aRect );
1236 if ( m_xFrame.is() )
1238 // borders should not be counted
1239 uno::Reference< frame::XControllerBorder > xControllerBorder( m_xFrame->getController(), uno::UNO_QUERY );
1240 if ( xControllerBorder.is() )
1242 awt::Rectangle aObjRect = CalculateBorderedArea( aRect );
1243 aObjRect = xControllerBorder->queryBorderedArea( aObjRect );
1244 aResult = AddBorderToArea( aObjRect );
1248 awt::Rectangle aMinRectangle = AddBorderToArea( awt::Rectangle() );
1249 if ( aResult.Width < aMinRectangle.Width + 2 )
1250 aResult.Width = aMinRectangle.Width + 2;
1251 if ( aResult.Height < aMinRectangle.Height + 2 )
1252 aResult.Height = aMinRectangle.Height + 2;
1254 return aResult;
1257 void SAL_CALL DocumentHolder::activated( ) throw (::com::sun::star::uno::RuntimeException, std::exception)
1259 if ( (m_pEmbedObj->getStatus(embed::Aspects::MSOLE_CONTENT)&embed::EmbedMisc::MS_EMBED_ACTIVATEWHENVISIBLE) ||
1260 svt::EmbeddedObjectRef::IsGLChart(m_pEmbedObj) )
1262 if ( m_pEmbedObj->getCurrentState() != embed::EmbedStates::UI_ACTIVE &&
1263 !(m_pEmbedObj->getStatus(embed::Aspects::MSOLE_CONTENT)&embed::EmbedMisc::MS_EMBED_NOUIACTIVATE) )
1267 m_pEmbedObj->changeState( embed::EmbedStates::UI_ACTIVE );
1269 catch ( const com::sun::star::embed::StateChangeInProgressException& )
1271 // must catch this exception because focus is grabbed while UI activation in doVerb()
1273 catch ( const com::sun::star::uno::Exception& )
1275 // no outgoing exceptions specified here
1278 else
1280 uno::Reference< frame::XFramesSupplier > xSupp( m_xFrame->getCreator(), uno::UNO_QUERY );
1281 if ( xSupp.is() )
1282 xSupp->setActiveFrame( m_xFrame );
1287 void DocumentHolder::ResizeHatchWindow()
1289 awt::Rectangle aHatchRect = AddBorderToArea( m_aObjRect );
1290 ResizeWindows_Impl( aHatchRect );
1291 uno::Reference< embed::XHatchWindow > xHatchWindow( m_xHatchWindow, uno::UNO_QUERY );
1292 xHatchWindow->setHatchBorderSize( awt::Size( HATCH_BORDER_WIDTH, HATCH_BORDER_WIDTH ) );
1295 void SAL_CALL DocumentHolder::deactivated( ) throw (::com::sun::star::uno::RuntimeException, std::exception)
1297 // deactivation is too unspecific to be useful; usually we only trigger code from activation
1298 // so UIDeactivation is actively triggered by the container
1301 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */