File: | home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx |
Warning: | line 2636, column 48 Use of memory after it is freed |
Press '?' to see keyboard shortcuts
Keyboard shortcuts:
1 | /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ | |||
2 | /* | |||
3 | * This file is part of the LibreOffice project. | |||
4 | * | |||
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/. | |||
8 | * | |||
9 | * This file incorporates work covered by the following license notice: | |||
10 | * | |||
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 . | |||
18 | */ | |||
19 | ||||
20 | #include "toolbarlayoutmanager.hxx" | |||
21 | #include <uiconfiguration/windowstateproperties.hxx> | |||
22 | #include <uielement/addonstoolbarwrapper.hxx> | |||
23 | #include "helpers.hxx" | |||
24 | #include <services/layoutmanager.hxx> | |||
25 | #include <strings.hrc> | |||
26 | #include <classes/fwkresid.hxx> | |||
27 | ||||
28 | #include <com/sun/star/awt/PosSize.hpp> | |||
29 | #include <com/sun/star/beans/XPropertySet.hpp> | |||
30 | #include <com/sun/star/ui/UIElementType.hpp> | |||
31 | #include <com/sun/star/container/XNameReplace.hpp> | |||
32 | #include <com/sun/star/container/XNameContainer.hpp> | |||
33 | #include <com/sun/star/ui/XUIElementSettings.hpp> | |||
34 | #include <com/sun/star/ui/XUIFunctionListener.hpp> | |||
35 | ||||
36 | #include <cppuhelper/queryinterface.hxx> | |||
37 | #include <unotools/cmdoptions.hxx> | |||
38 | #include <toolkit/helper/vclunohelper.hxx> | |||
39 | #include <toolkit/helper/convert.hxx> | |||
40 | #include <vcl/i18nhelp.hxx> | |||
41 | #include <vcl/dockingarea.hxx> | |||
42 | #include <vcl/settings.hxx> | |||
43 | #include <vcl/svapp.hxx> | |||
44 | #include <sal/log.hxx> | |||
45 | #include <tools/gen.hxx> | |||
46 | ||||
47 | ||||
48 | using namespace ::com::sun::star; | |||
49 | ||||
50 | namespace framework | |||
51 | { | |||
52 | ||||
53 | ToolbarLayoutManager::ToolbarLayoutManager( | |||
54 | const uno::Reference< uno::XComponentContext >& rxContext, | |||
55 | const uno::Reference< ui::XUIElementFactory >& xUIElementFactory, | |||
56 | LayoutManager* pParentLayouter ): | |||
57 | m_xContext( rxContext), | |||
58 | m_xUIElementFactoryManager( xUIElementFactory ), | |||
59 | m_pParentLayouter( pParentLayouter ), | |||
60 | m_eDockOperation( DOCKOP_ON_COLROW ), | |||
61 | m_ePreviewDetection( PREVIEWFRAME_UNKNOWN ), | |||
62 | m_bComponentAttached( false ), | |||
63 | m_bLayoutDirty( false ), | |||
64 | m_bGlobalSettings( false ), | |||
65 | m_bDockingInProgress( false ), | |||
66 | m_bLayoutInProgress( false ), | |||
67 | m_bToolbarCreation( false ) | |||
68 | { | |||
69 | // initialize rectangles to zero values | |||
70 | setZeroRectangle( m_aDockingAreaOffsets ); | |||
71 | setZeroRectangle( m_aDockingArea ); | |||
72 | } | |||
73 | ||||
74 | ToolbarLayoutManager::~ToolbarLayoutManager() | |||
75 | { | |||
76 | m_pGlobalSettings.reset(); | |||
77 | m_pAddonOptions.reset(); | |||
78 | } | |||
79 | ||||
80 | // XInterface | |||
81 | ||||
82 | void SAL_CALL ToolbarLayoutManager::acquire() throw() | |||
83 | { | |||
84 | OWeakObject::acquire(); | |||
85 | } | |||
86 | ||||
87 | void SAL_CALL ToolbarLayoutManager::release() throw() | |||
88 | { | |||
89 | OWeakObject::release(); | |||
90 | } | |||
91 | ||||
92 | uno::Any SAL_CALL ToolbarLayoutManager::queryInterface( const uno::Type & rType ) | |||
93 | { | |||
94 | uno::Any a = ::cppu::queryInterface( rType, | |||
95 | static_cast< awt::XDockableWindowListener* >(this), | |||
96 | static_cast< ui::XUIConfigurationListener* >(this), | |||
97 | static_cast< awt::XWindowListener* >(this)); | |||
98 | ||||
99 | if ( a.hasValue() ) | |||
100 | return a; | |||
101 | ||||
102 | return OWeakObject::queryInterface( rType ); | |||
103 | } | |||
104 | ||||
105 | void SAL_CALL ToolbarLayoutManager::disposing( const lang::EventObject& aEvent ) | |||
106 | { | |||
107 | if ( aEvent.Source == m_xFrame ) | |||
108 | { | |||
109 | // Reset all internal references | |||
110 | reset(); | |||
111 | implts_destroyDockingAreaWindows(); | |||
112 | } | |||
113 | } | |||
114 | ||||
115 | awt::Rectangle ToolbarLayoutManager::getDockingArea() | |||
116 | { | |||
117 | SolarMutexResettableGuard aWriteLock; | |||
118 | tools::Rectangle aNewDockingArea( m_aDockingArea ); | |||
119 | aWriteLock.clear(); | |||
120 | ||||
121 | if ( isLayoutDirty() ) | |||
122 | aNewDockingArea = implts_calcDockingArea(); | |||
123 | ||||
124 | aWriteLock.reset(); | |||
125 | m_aDockingArea = aNewDockingArea; | |||
126 | aWriteLock.clear(); | |||
127 | ||||
128 | return putRectangleValueToAWT(aNewDockingArea); | |||
129 | } | |||
130 | ||||
131 | void ToolbarLayoutManager::setDockingArea( const awt::Rectangle& rDockingArea ) | |||
132 | { | |||
133 | SolarMutexGuard g; | |||
134 | m_aDockingArea = putAWTToRectangle( rDockingArea ); | |||
135 | m_bLayoutDirty = true; | |||
136 | } | |||
137 | ||||
138 | void ToolbarLayoutManager::implts_setDockingAreaWindowSizes( const awt::Rectangle& rBorderSpace ) | |||
139 | { | |||
140 | SolarMutexClearableGuard aReadLock; | |||
141 | tools::Rectangle aDockOffsets = m_aDockingAreaOffsets; | |||
142 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
143 | uno::Reference< awt::XWindow > xTopDockAreaWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
144 | uno::Reference< awt::XWindow > xBottomDockAreaWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] ); | |||
145 | uno::Reference< awt::XWindow > xLeftDockAreaWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
146 | uno::Reference< awt::XWindow > xRightDockAreaWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] ); | |||
147 | aReadLock.clear(); | |||
148 | ||||
149 | uno::Reference< awt::XDevice > xDevice( xContainerWindow, uno::UNO_QUERY ); | |||
150 | ||||
151 | // Convert relative size to output size. | |||
152 | awt::Rectangle aRectangle = xContainerWindow->getPosSize(); | |||
153 | awt::DeviceInfo aInfo = xDevice->getInfo(); | |||
154 | awt::Size aContainerClientSize( aRectangle.Width - aInfo.LeftInset - aInfo.RightInset , | |||
155 | aRectangle.Height - aInfo.TopInset - aInfo.BottomInset ); | |||
156 | long aStatusBarHeight = aDockOffsets.GetHeight(); | |||
157 | ||||
158 | sal_Int32 nLeftRightDockingAreaHeight( aContainerClientSize.Height ); | |||
159 | if ( rBorderSpace.Y >= 0 ) | |||
160 | { | |||
161 | // Top docking area window | |||
162 | xTopDockAreaWindow->setPosSize( 0, 0, aContainerClientSize.Width, rBorderSpace.Y, awt::PosSize::POSSIZE ); | |||
163 | xTopDockAreaWindow->setVisible( true ); | |||
164 | nLeftRightDockingAreaHeight -= rBorderSpace.Y; | |||
165 | } | |||
166 | ||||
167 | if ( rBorderSpace.Height >= 0 ) | |||
168 | { | |||
169 | // Bottom docking area window | |||
170 | sal_Int32 nBottomPos = std::max( sal_Int32( aContainerClientSize.Height - rBorderSpace.Height - aStatusBarHeight + 1 ), sal_Int32( 0 )); | |||
171 | sal_Int32 nHeight = ( nBottomPos == 0 ) ? 0 : rBorderSpace.Height; | |||
172 | ||||
173 | xBottomDockAreaWindow->setPosSize( 0, nBottomPos, aContainerClientSize.Width, nHeight, awt::PosSize::POSSIZE ); | |||
174 | xBottomDockAreaWindow->setVisible( true ); | |||
175 | nLeftRightDockingAreaHeight -= nHeight - 1; | |||
176 | } | |||
177 | ||||
178 | nLeftRightDockingAreaHeight -= aStatusBarHeight; | |||
179 | if ( rBorderSpace.X >= 0 || nLeftRightDockingAreaHeight > 0 ) | |||
180 | { | |||
181 | // Left docking area window | |||
182 | // We also have to change our right docking area window if the top or bottom area has changed. They have a higher priority! | |||
183 | sal_Int32 nHeight = std::max<sal_Int32>( 0, nLeftRightDockingAreaHeight ); | |||
184 | ||||
185 | xLeftDockAreaWindow->setPosSize( 0, rBorderSpace.Y, rBorderSpace.X, nHeight, awt::PosSize::POSSIZE ); | |||
186 | xLeftDockAreaWindow->setVisible( true ); | |||
187 | } | |||
188 | if ( rBorderSpace.Width >= 0 || nLeftRightDockingAreaHeight > 0 ) | |||
189 | { | |||
190 | // Right docking area window | |||
191 | // We also have to change our right docking area window if the top or bottom area has changed. They have a higher priority! | |||
192 | sal_Int32 nLeftPos = std::max<sal_Int32>( 0, aContainerClientSize.Width - rBorderSpace.Width ); | |||
193 | sal_Int32 nHeight = std::max<sal_Int32>( 0, nLeftRightDockingAreaHeight ); | |||
194 | sal_Int32 nWidth = ( nLeftPos == 0 ) ? 0 : rBorderSpace.Width; | |||
195 | ||||
196 | xRightDockAreaWindow->setPosSize( nLeftPos, rBorderSpace.Y, nWidth, nHeight, awt::PosSize::POSSIZE ); | |||
197 | xRightDockAreaWindow->setVisible( true ); | |||
198 | } | |||
199 | } | |||
200 | ||||
201 | ||||
202 | void ToolbarLayoutManager::doLayout(const ::Size& aContainerSize) | |||
203 | { | |||
204 | SolarMutexResettableGuard aWriteLock; | |||
205 | bool bLayoutInProgress( m_bLayoutInProgress ); | |||
206 | m_bLayoutInProgress = true; | |||
207 | awt::Rectangle aDockingArea = putRectangleValueToAWT( m_aDockingArea ); | |||
208 | aWriteLock.clear(); | |||
209 | ||||
210 | if ( bLayoutInProgress ) | |||
211 | return; | |||
212 | ||||
213 | // Retrieve row/column dependent data from all docked user-interface elements | |||
214 | for ( sal_Int32 i = 0; i < DOCKINGAREAS_COUNT; i++ ) | |||
215 | { | |||
216 | bool bReverse( isReverseOrderDockingArea( i )); | |||
217 | std::vector< SingleRowColumnWindowData > aRowColumnsWindowData; | |||
218 | ||||
219 | implts_getDockingAreaElementInfos( static_cast<ui::DockingArea>(i), aRowColumnsWindowData ); | |||
220 | ||||
221 | sal_Int32 nOffset( 0 ); | |||
222 | const sal_uInt32 nCount = aRowColumnsWindowData.size(); | |||
223 | for ( sal_uInt32 j = 0; j < nCount; ++j ) | |||
224 | { | |||
225 | sal_uInt32 nIndex = bReverse ? nCount-j-1 : j; | |||
226 | implts_calcWindowPosSizeOnSingleRowColumn( i, nOffset, aRowColumnsWindowData[nIndex], aContainerSize ); | |||
227 | nOffset += aRowColumnsWindowData[j].nStaticSize; | |||
228 | } | |||
229 | } | |||
230 | ||||
231 | implts_setDockingAreaWindowSizes( aDockingArea ); | |||
232 | ||||
233 | aWriteLock.reset(); | |||
234 | m_bLayoutDirty = false; | |||
235 | m_bLayoutInProgress = false; | |||
236 | aWriteLock.clear(); | |||
237 | } | |||
238 | ||||
239 | bool ToolbarLayoutManager::implts_isParentWindowVisible() const | |||
240 | { | |||
241 | SolarMutexGuard g; | |||
242 | bool bVisible( false ); | |||
243 | if ( m_xContainerWindow.is() ) | |||
244 | bVisible = m_xContainerWindow->isVisible(); | |||
245 | ||||
246 | return bVisible; | |||
247 | } | |||
248 | ||||
249 | tools::Rectangle ToolbarLayoutManager::implts_calcDockingArea() | |||
250 | { | |||
251 | SolarMutexClearableGuard aReadLock; | |||
252 | UIElementVector aWindowVector( m_aUIElements ); | |||
253 | aReadLock.clear(); | |||
254 | ||||
255 | tools::Rectangle aBorderSpace; | |||
256 | sal_Int32 nCurrRowColumn( 0 ); | |||
257 | sal_Int32 nCurrPos( 0 ); | |||
258 | ui::DockingArea nCurrDockingArea( ui::DockingArea_DOCKINGAREA_TOP ); | |||
259 | std::vector< sal_Int32 > aRowColumnSizes[DOCKINGAREAS_COUNT]; | |||
260 | ||||
261 | // initialize rectangle with zero values! | |||
262 | aBorderSpace.setWidth(0); | |||
263 | aBorderSpace.setHeight(0); | |||
264 | ||||
265 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].clear(); | |||
266 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].push_back( 0 ); | |||
267 | ||||
268 | for (auto const& window : aWindowVector) | |||
269 | { | |||
270 | if ( window.m_xUIElement.is() ) | |||
271 | { | |||
272 | uno::Reference< awt::XWindow > xWindow( window.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
273 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
274 | if ( xWindow.is() && xDockWindow.is() ) | |||
275 | { | |||
276 | SolarMutexGuard aGuard; | |||
277 | ||||
278 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
279 | if ( pWindow && !xDockWindow->isFloating() && window.m_bVisible && !window.m_bMasterHide ) | |||
280 | { | |||
281 | awt::Rectangle aPosSize = xWindow->getPosSize(); | |||
282 | if ( window.m_aDockedData.m_nDockedArea != nCurrDockingArea ) | |||
283 | { | |||
284 | nCurrDockingArea = window.m_aDockedData.m_nDockedArea; | |||
285 | nCurrRowColumn = 0; | |||
286 | nCurrPos = 0; | |||
287 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].clear(); | |||
288 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].push_back( 0 ); | |||
289 | } | |||
290 | ||||
291 | if ( window.m_aDockedData.m_nDockedArea == nCurrDockingArea ) | |||
292 | { | |||
293 | if ( isHorizontalDockingArea( window.m_aDockedData.m_nDockedArea )) | |||
294 | { | |||
295 | if ( window.m_aDockedData.m_aPos.Y > nCurrPos ) | |||
296 | { | |||
297 | ++nCurrRowColumn; | |||
298 | nCurrPos = window.m_aDockedData.m_aPos.Y; | |||
299 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].push_back( 0 ); | |||
300 | } | |||
301 | ||||
302 | if ( aPosSize.Height > aRowColumnSizes[static_cast<int>(nCurrDockingArea)][nCurrRowColumn] ) | |||
303 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)][nCurrRowColumn] = aPosSize.Height; | |||
304 | } | |||
305 | else | |||
306 | { | |||
307 | if ( window.m_aDockedData.m_aPos.X > nCurrPos ) | |||
308 | { | |||
309 | ++nCurrRowColumn; | |||
310 | nCurrPos = window.m_aDockedData.m_aPos.X; | |||
311 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)].push_back( 0 ); | |||
312 | } | |||
313 | ||||
314 | if ( aPosSize.Width > aRowColumnSizes[static_cast<int>(nCurrDockingArea)][nCurrRowColumn] ) | |||
315 | aRowColumnSizes[static_cast<int>(nCurrDockingArea)][nCurrRowColumn] = aPosSize.Width; | |||
316 | } | |||
317 | } | |||
318 | } | |||
319 | } | |||
320 | } | |||
321 | } | |||
322 | ||||
323 | // Sum up max heights from every row/column | |||
324 | if ( !aWindowVector.empty() ) | |||
325 | { | |||
326 | for ( sal_Int32 i = 0; i <= sal_Int32(ui::DockingArea_DOCKINGAREA_RIGHT); i++ ) | |||
327 | { | |||
328 | sal_Int32 nSize( 0 ); | |||
329 | const sal_uInt32 nCount = aRowColumnSizes[i].size(); | |||
330 | for ( sal_uInt32 j = 0; j < nCount; j++ ) | |||
331 | nSize += aRowColumnSizes[i][j]; | |||
332 | ||||
333 | if ( i == sal_Int32(ui::DockingArea_DOCKINGAREA_TOP) ) | |||
334 | aBorderSpace.SetTop( nSize ); | |||
335 | else if ( i == sal_Int32(ui::DockingArea_DOCKINGAREA_BOTTOM) ) | |||
336 | aBorderSpace.SetBottom( nSize ); | |||
337 | else if ( i == sal_Int32(ui::DockingArea_DOCKINGAREA_LEFT) ) | |||
338 | aBorderSpace.SetLeft( nSize ); | |||
339 | else | |||
340 | aBorderSpace.SetRight( nSize ); | |||
341 | } | |||
342 | } | |||
343 | ||||
344 | return aBorderSpace; | |||
345 | } | |||
346 | ||||
347 | void ToolbarLayoutManager::reset() | |||
348 | { | |||
349 | { | |||
350 | SolarMutexGuard aWriteLock; | |||
351 | m_xModuleCfgMgr.clear(); | |||
352 | m_xDocCfgMgr.clear(); | |||
353 | m_ePreviewDetection = PREVIEWFRAME_UNKNOWN; | |||
354 | m_bComponentAttached = false; | |||
355 | } | |||
356 | ||||
357 | destroyToolbars(); | |||
358 | resetDockingArea(); | |||
359 | } | |||
360 | ||||
361 | void ToolbarLayoutManager::attach( | |||
362 | const uno::Reference< frame::XFrame >& xFrame, | |||
363 | const uno::Reference< ui::XUIConfigurationManager >& xModuleCfgMgr, | |||
364 | const uno::Reference< ui::XUIConfigurationManager >& xDocCfgMgr, | |||
365 | const uno::Reference< container::XNameAccess >& xPersistentWindowState ) | |||
366 | { | |||
367 | // reset toolbar manager if we lose our current frame | |||
368 | if ( m_xFrame.is() && m_xFrame != xFrame ) | |||
369 | reset(); | |||
370 | ||||
371 | SolarMutexGuard g; | |||
372 | m_xFrame = xFrame; | |||
373 | m_xModuleCfgMgr = xModuleCfgMgr; | |||
374 | m_xDocCfgMgr = xDocCfgMgr; | |||
375 | m_xPersistentWindowState = xPersistentWindowState; | |||
376 | m_bComponentAttached = true; | |||
377 | } | |||
378 | ||||
379 | bool ToolbarLayoutManager::isPreviewFrame() | |||
380 | { | |||
381 | SolarMutexGuard g; | |||
382 | if (m_ePreviewDetection == PREVIEWFRAME_UNKNOWN) | |||
383 | { | |||
384 | uno::Reference< frame::XFrame > xFrame( m_xFrame ); | |||
385 | ||||
386 | uno::Reference< frame::XModel > xModel( impl_getModelFromFrame( xFrame )); | |||
387 | ||||
388 | m_ePreviewDetection = (implts_isPreviewModel( xModel ) ? PREVIEWFRAME_YES : PREVIEWFRAME_NO); | |||
389 | } | |||
390 | return m_ePreviewDetection == PREVIEWFRAME_YES; | |||
391 | } | |||
392 | ||||
393 | void ToolbarLayoutManager::createStaticToolbars() | |||
394 | { | |||
395 | resetDockingArea(); | |||
396 | implts_createCustomToolBars(); | |||
397 | implts_createAddonsToolBars(); | |||
398 | implts_createNonContextSensitiveToolBars(); | |||
399 | implts_sortUIElements(); | |||
400 | } | |||
401 | ||||
402 | bool ToolbarLayoutManager::requestToolbar( const OUString& rResourceURL ) | |||
403 | { | |||
404 | if (isPreviewFrame()) | |||
405 | return false; // no toolbars for preview frame! | |||
406 | ||||
407 | bool bNotify( false ); | |||
408 | bool bMustCallCreate( false ); | |||
409 | uno::Reference< ui::XUIElement > xUIElement; | |||
410 | ||||
411 | UIElement aRequestedToolbar = impl_findToolbar( rResourceURL ); | |||
412 | if ( aRequestedToolbar.m_aName != rResourceURL ) | |||
413 | { | |||
414 | bMustCallCreate = true; | |||
415 | aRequestedToolbar.m_aName = rResourceURL; | |||
416 | aRequestedToolbar.m_aType = UIRESOURCETYPE_TOOLBAR"toolbar"; | |||
417 | aRequestedToolbar.m_xUIElement = xUIElement; | |||
418 | implts_readWindowStateData( rResourceURL, aRequestedToolbar ); | |||
419 | } | |||
420 | ||||
421 | xUIElement = aRequestedToolbar.m_xUIElement; | |||
422 | if ( !xUIElement.is() ) | |||
423 | bMustCallCreate = true; | |||
424 | ||||
425 | bool bCreateOrShowToolbar( aRequestedToolbar.m_bVisible && !aRequestedToolbar.m_bMasterHide ); | |||
426 | ||||
427 | if ( bCreateOrShowToolbar ) | |||
428 | bNotify = bMustCallCreate ? createToolbar( rResourceURL ) : showToolbar( rResourceURL ); | |||
429 | ||||
430 | return bNotify; | |||
431 | } | |||
432 | ||||
433 | bool ToolbarLayoutManager::createToolbar( const OUString& rResourceURL ) | |||
434 | { | |||
435 | bool bNotify( false ); | |||
436 | ||||
437 | uno::Reference<frame::XFrame> xFrame; | |||
438 | uno::Reference<awt::XWindow2> xContainerWindow; | |||
439 | { | |||
440 | SolarMutexGuard aReadLock; | |||
441 | xFrame.set(m_xFrame); | |||
442 | xContainerWindow.set(m_xContainerWindow); | |||
443 | } | |||
444 | ||||
445 | if ( !xFrame.is() || !xContainerWindow.is() ) | |||
446 | return false; | |||
447 | ||||
448 | UIElement aToolbarElement = implts_findToolbar( rResourceURL ); | |||
449 | if ( !aToolbarElement.m_xUIElement.is() ) | |||
450 | { | |||
451 | uno::Reference< ui::XUIElement > xUIElement; | |||
452 | ||||
453 | uno::Sequence< beans::PropertyValue > aPropSeq( 2 ); | |||
454 | aPropSeq[0].Name = "Frame"; | |||
455 | aPropSeq[0].Value <<= xFrame; | |||
456 | aPropSeq[1].Name = "Persistent"; | |||
457 | aPropSeq[1].Value <<= true; | |||
458 | uno::Reference<ui::XUIElementFactory> xUIElementFactory; | |||
459 | { | |||
460 | SolarMutexGuard aReadLock; | |||
461 | xUIElementFactory.set(m_xUIElementFactoryManager); | |||
462 | } | |||
463 | ||||
464 | implts_setToolbarCreation(); | |||
465 | try | |||
466 | { | |||
467 | if ( xUIElementFactory.is() ) | |||
468 | xUIElement = xUIElementFactory->createUIElement( rResourceURL, aPropSeq ); | |||
469 | } | |||
470 | catch (const container::NoSuchElementException&) | |||
471 | { | |||
472 | } | |||
473 | catch (const lang::IllegalArgumentException&) | |||
474 | { | |||
475 | } | |||
476 | implts_setToolbarCreation( false ); | |||
477 | ||||
478 | if ( xUIElement.is() ) | |||
479 | { | |||
480 | uno::Reference< awt::XWindow > xWindow( xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
481 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
482 | if ( xDockWindow.is() && xWindow.is() ) | |||
483 | { | |||
484 | try | |||
485 | { | |||
486 | xDockWindow->addDockableWindowListener( uno::Reference< awt::XDockableWindowListener >( | |||
487 | static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
488 | xWindow->addWindowListener( uno::Reference< awt::XWindowListener >( | |||
489 | static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
490 | xDockWindow->enableDocking( true ); | |||
491 | } | |||
492 | catch (const uno::Exception&) | |||
493 | { | |||
494 | } | |||
495 | } | |||
496 | ||||
497 | bool bVisible = false; | |||
498 | bool bFloating = false; | |||
499 | ||||
500 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
501 | { | |||
502 | SolarMutexClearableGuard aWriteLock; | |||
503 | ||||
504 | UIElement& rElement = impl_findToolbar(rResourceURL); | |||
505 | if (rElement.m_xUIElement.is()) | |||
506 | { | |||
507 | // somebody else must have created it while we released | |||
508 | // the SolarMutex - just dispose our new instance and | |||
509 | // do nothing. (We have to dispose either the new or the | |||
510 | // existing m_xUIElement.) | |||
511 | aWriteLock.clear(); | |||
512 | uno::Reference<lang::XComponent> const xC(xUIElement, uno::UNO_QUERY); | |||
513 | xC->dispose(); | |||
514 | return false; | |||
515 | } | |||
516 | if (!rElement.m_aName.isEmpty()) | |||
517 | { | |||
518 | // Reuse a local entry so we are able to use the latest | |||
519 | // UI changes for this document. | |||
520 | implts_setElementData(rElement, xDockWindow); | |||
521 | rElement.m_xUIElement = xUIElement; | |||
522 | bVisible = rElement.m_bVisible; | |||
523 | bFloating = rElement.m_bFloating; | |||
524 | } | |||
525 | else | |||
526 | { | |||
527 | // Create new UI element and try to read its state data | |||
528 | UIElement aNewToolbar(rResourceURL, UIRESOURCETYPE_TOOLBAR"toolbar", xUIElement); | |||
529 | implts_readWindowStateData(rResourceURL, aNewToolbar); | |||
530 | implts_setElementData(aNewToolbar, xDockWindow); | |||
531 | implts_insertToolbar(aNewToolbar); | |||
532 | bVisible = aNewToolbar.m_bVisible; | |||
533 | bFloating = rElement.m_bFloating; | |||
534 | } | |||
535 | } | |||
536 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
537 | ||||
538 | // set toolbar menu style according to customize command state | |||
539 | SvtCommandOptions aCmdOptions; | |||
540 | ||||
541 | SolarMutexGuard aGuard; | |||
542 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
543 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
544 | { | |||
545 | ToolBox* pToolbar = static_cast<ToolBox *>(pWindow.get()); | |||
546 | ToolBoxMenuType nMenuType = pToolbar->GetMenuType(); | |||
547 | if ( aCmdOptions.Lookup( SvtCommandOptions::CMDOPTION_DISABLED, "ConfigureDialog" )) | |||
548 | pToolbar->SetMenuType( nMenuType & ~ToolBoxMenuType::Customize ); | |||
549 | else | |||
550 | pToolbar->SetMenuType( nMenuType | ToolBoxMenuType::Customize ); | |||
551 | } | |||
552 | bNotify = true; | |||
553 | ||||
554 | implts_sortUIElements(); | |||
555 | ||||
556 | if ( bVisible && !bFloating ) | |||
557 | implts_setLayoutDirty(); | |||
558 | } | |||
559 | } | |||
560 | ||||
561 | return bNotify; | |||
562 | } | |||
563 | ||||
564 | bool ToolbarLayoutManager::destroyToolbar( const OUString& rResourceURL ) | |||
565 | { | |||
566 | uno::Reference< lang::XComponent > xComponent; | |||
567 | ||||
568 | bool bNotify( false ); | |||
569 | bool bMustBeSorted( false ); | |||
570 | bool bMustLayouted( false ); | |||
571 | bool bMustBeDestroyed( !rResourceURL.startsWith("private:resource/toolbar/addon_") ); | |||
572 | ||||
573 | { | |||
574 | SolarMutexGuard aWriteLock; | |||
575 | for (auto & elem : m_aUIElements) | |||
576 | { | |||
577 | if (elem.m_aName == rResourceURL) | |||
578 | { | |||
579 | xComponent.set(elem.m_xUIElement, uno::UNO_QUERY); | |||
580 | if (bMustBeDestroyed) | |||
581 | elem.m_xUIElement.clear(); | |||
582 | else | |||
583 | elem.m_bVisible = false; | |||
584 | break; | |||
585 | } | |||
586 | } | |||
587 | } | |||
588 | ||||
589 | uno::Reference< ui::XUIElement > xUIElement( xComponent, uno::UNO_QUERY ); | |||
590 | if ( xUIElement.is() ) | |||
591 | { | |||
592 | uno::Reference< awt::XWindow > xWindow( xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
593 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
594 | ||||
595 | if ( bMustBeDestroyed ) | |||
596 | { | |||
597 | try | |||
598 | { | |||
599 | if ( xWindow.is() ) | |||
600 | xWindow->removeWindowListener( uno::Reference< awt::XWindowListener >( | |||
601 | static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
602 | } | |||
603 | catch (const uno::Exception&) | |||
604 | { | |||
605 | } | |||
606 | ||||
607 | try | |||
608 | { | |||
609 | if ( xDockWindow.is() ) | |||
610 | xDockWindow->removeDockableWindowListener( uno::Reference< awt::XDockableWindowListener >( | |||
611 | static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
612 | } | |||
613 | catch (const uno::Exception&) | |||
614 | { | |||
615 | } | |||
616 | } | |||
617 | else | |||
618 | { | |||
619 | if ( xWindow.is() ) | |||
620 | xWindow->setVisible( false ); | |||
621 | bNotify = true; | |||
622 | } | |||
623 | ||||
624 | if ( !xDockWindow->isFloating() ) | |||
625 | bMustLayouted = true; | |||
626 | bMustBeSorted = true; | |||
627 | } | |||
628 | ||||
629 | if ( bMustBeDestroyed ) | |||
630 | { | |||
631 | if ( xComponent.is() ) | |||
632 | xComponent->dispose(); | |||
633 | bNotify = true; | |||
634 | } | |||
635 | ||||
636 | if ( bMustLayouted ) | |||
637 | implts_setLayoutDirty(); | |||
638 | ||||
639 | if ( bMustBeSorted ) | |||
640 | implts_sortUIElements(); | |||
641 | ||||
642 | return bNotify; | |||
643 | } | |||
644 | ||||
645 | void ToolbarLayoutManager::destroyToolbars() | |||
646 | { | |||
647 | UIElementVector aUIElementVector; | |||
648 | implts_getUIElementVectorCopy( aUIElementVector ); | |||
649 | ||||
650 | { | |||
651 | SolarMutexGuard aWriteLock; | |||
652 | m_aUIElements.clear(); | |||
653 | m_bLayoutDirty = true; | |||
654 | } | |||
655 | ||||
656 | for (auto const& elem : aUIElementVector) | |||
657 | { | |||
658 | uno::Reference< lang::XComponent > xComponent( elem.m_xUIElement, uno::UNO_QUERY ); | |||
659 | if ( xComponent.is() ) | |||
660 | xComponent->dispose(); | |||
661 | } | |||
662 | } | |||
663 | ||||
664 | bool ToolbarLayoutManager::showToolbar( const OUString& rResourceURL ) | |||
665 | { | |||
666 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
667 | ||||
668 | SolarMutexGuard aGuard; | |||
669 | vcl::Window* pWindow = getWindowFromXUIElement( aUIElement.m_xUIElement ); | |||
670 | ||||
671 | // Addons appear to need to be populated at start, but we don't | |||
672 | // want to populate them with (scaled) images until later. | |||
673 | AddonsToolBarWrapper *pAddOns; | |||
674 | pAddOns = dynamic_cast<AddonsToolBarWrapper *>( aUIElement.m_xUIElement.get()); | |||
675 | if (pAddOns) | |||
676 | pAddOns->populateImages(); | |||
677 | ||||
678 | if ( pWindow ) | |||
679 | { | |||
680 | if ( !aUIElement.m_bFloating ) | |||
681 | implts_setLayoutDirty(); | |||
682 | else | |||
683 | pWindow->Show( true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
684 | ||||
685 | aUIElement.m_bVisible = true; | |||
686 | implts_writeWindowStateData( aUIElement ); | |||
687 | implts_setToolbar( aUIElement ); | |||
688 | implts_sortUIElements(); | |||
689 | return true; | |||
690 | } | |||
691 | ||||
692 | return false; | |||
693 | } | |||
694 | ||||
695 | bool ToolbarLayoutManager::hideToolbar( const OUString& rResourceURL ) | |||
696 | { | |||
697 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
698 | ||||
699 | SolarMutexGuard aGuard; | |||
700 | vcl::Window* pWindow = getWindowFromXUIElement( aUIElement.m_xUIElement ); | |||
701 | if ( pWindow ) | |||
702 | { | |||
703 | pWindow->Show( false ); | |||
704 | if ( !aUIElement.m_bFloating ) | |||
705 | implts_setLayoutDirty(); | |||
706 | ||||
707 | aUIElement.m_bVisible = false; | |||
708 | implts_writeWindowStateData( aUIElement ); | |||
709 | implts_setToolbar( aUIElement ); | |||
710 | return true; | |||
711 | } | |||
712 | ||||
713 | return false; | |||
714 | } | |||
715 | ||||
716 | void ToolbarLayoutManager::refreshToolbarsVisibility( bool bAutomaticToolbars ) | |||
717 | { | |||
718 | UIElementVector aUIElementVector; | |||
719 | ||||
720 | if ( !bAutomaticToolbars ) | |||
721 | return; | |||
722 | ||||
723 | implts_getUIElementVectorCopy( aUIElementVector ); | |||
724 | ||||
725 | UIElement aUIElement; | |||
726 | SolarMutexGuard aGuard; | |||
727 | for (auto const& elem : aUIElementVector) | |||
728 | { | |||
729 | if ( implts_readWindowStateData( elem.m_aName, aUIElement ) && | |||
730 | ( elem.m_bVisible != aUIElement.m_bVisible ) && !elem.m_bMasterHide ) | |||
731 | { | |||
732 | SolarMutexGuard g; | |||
733 | UIElement& rUIElement = impl_findToolbar( elem.m_aName ); | |||
734 | if ( rUIElement.m_aName == elem.m_aName ) | |||
735 | { | |||
736 | rUIElement.m_bVisible = aUIElement.m_bVisible; | |||
737 | implts_setLayoutDirty(); | |||
738 | } | |||
739 | } | |||
740 | } | |||
741 | } | |||
742 | ||||
743 | void ToolbarLayoutManager::setFloatingToolbarsVisibility( bool bVisible ) | |||
744 | { | |||
745 | UIElementVector aUIElementVector; | |||
746 | implts_getUIElementVectorCopy( aUIElementVector ); | |||
747 | ||||
748 | SolarMutexGuard aGuard; | |||
749 | for (auto const& elem : aUIElementVector) | |||
750 | { | |||
751 | vcl::Window* pWindow = getWindowFromXUIElement( elem.m_xUIElement ); | |||
752 | if ( pWindow && elem.m_bFloating ) | |||
753 | { | |||
754 | if ( bVisible ) | |||
755 | { | |||
756 | if ( elem.m_bVisible && !elem.m_bMasterHide ) | |||
757 | pWindow->Show( true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
758 | } | |||
759 | else | |||
760 | pWindow->Show( false ); | |||
761 | } | |||
762 | } | |||
763 | } | |||
764 | ||||
765 | void ToolbarLayoutManager::setVisible( bool bVisible ) | |||
766 | { | |||
767 | UIElementVector aUIElementVector; | |||
768 | implts_getUIElementVectorCopy( aUIElementVector ); | |||
769 | ||||
770 | SolarMutexGuard aGuard; | |||
771 | for (auto & elem : aUIElementVector) | |||
772 | { | |||
773 | if (!elem.m_bFloating) | |||
774 | { | |||
775 | elem.m_bMasterHide = !bVisible; | |||
776 | implts_setToolbar(elem); | |||
777 | implts_setLayoutDirty(); | |||
778 | } | |||
779 | ||||
780 | vcl::Window* pWindow = getWindowFromXUIElement( elem.m_xUIElement ); | |||
781 | if ( pWindow ) | |||
782 | { | |||
783 | bool bSetVisible( elem.m_bVisible && bVisible ); | |||
784 | if ( !bSetVisible ) | |||
785 | pWindow->Hide(); | |||
786 | else | |||
787 | { | |||
788 | if ( elem.m_bFloating ) | |||
789 | pWindow->Show(true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
790 | } | |||
791 | } | |||
792 | } | |||
793 | ||||
794 | if ( !bVisible ) | |||
795 | resetDockingArea(); | |||
796 | } | |||
797 | ||||
798 | bool ToolbarLayoutManager::dockToolbar( const OUString& rResourceURL, ui::DockingArea eDockingArea, const awt::Point& aPos ) | |||
799 | { | |||
800 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
801 | ||||
802 | if ( aUIElement.m_xUIElement.is() ) | |||
803 | { | |||
804 | try | |||
805 | { | |||
806 | uno::Reference< awt::XWindow > xWindow( aUIElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
807 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
808 | if ( xDockWindow.is() ) | |||
809 | { | |||
810 | if ( eDockingArea != ui::DockingArea_DOCKINGAREA_DEFAULT ) | |||
811 | aUIElement.m_aDockedData.m_nDockedArea = eDockingArea; | |||
812 | ||||
813 | if ( !isDefaultPos( aPos )) | |||
814 | aUIElement.m_aDockedData.m_aPos = aPos; | |||
815 | ||||
816 | if ( !xDockWindow->isFloating() ) | |||
817 | { | |||
818 | vcl::Window* pWindow( nullptr ); | |||
819 | ToolBox* pToolBox( nullptr ); | |||
820 | ||||
821 | { | |||
822 | SolarMutexGuard aGuard; | |||
823 | pWindow = VCLUnoHelper::GetWindow( xWindow ).get(); | |||
824 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
825 | { | |||
826 | pToolBox = static_cast<ToolBox *>(pWindow); | |||
827 | ||||
828 | // We have to set the alignment of the toolbox. It's possible that the toolbox is moved from a | |||
829 | // horizontal to a vertical docking area! | |||
830 | pToolBox->SetAlign( ImplConvertAlignment( aUIElement.m_aDockedData.m_nDockedArea )); | |||
831 | } | |||
832 | } | |||
833 | ||||
834 | if ( hasDefaultPosValue( aUIElement.m_aDockedData.m_aPos )) | |||
835 | { | |||
836 | // Docking on its default position without a preset position - | |||
837 | // we have to find a good place for it. | |||
838 | ::Size aSize; | |||
839 | ||||
840 | SolarMutexGuard aGuard; | |||
841 | { | |||
842 | if (pToolBox) | |||
843 | aSize = pToolBox->CalcWindowSizePixel( 1, ImplConvertAlignment( aUIElement.m_aDockedData.m_nDockedArea ) ); | |||
844 | else if (pWindow) | |||
845 | aSize = pWindow->GetSizePixel(); | |||
846 | } | |||
847 | ||||
848 | ::Point aPixelPos; | |||
849 | awt::Point aDockPos; | |||
850 | implts_findNextDockingPos(aUIElement.m_aDockedData.m_nDockedArea, aSize, aDockPos, aPixelPos ); | |||
851 | aUIElement.m_aDockedData.m_aPos = aDockPos; | |||
852 | } | |||
853 | } | |||
854 | ||||
855 | implts_setToolbar( aUIElement ); | |||
856 | ||||
857 | if ( xDockWindow->isFloating() ) | |||
858 | { | |||
859 | // ATTENTION: This will call toggleFloatingMode() via notifications which | |||
860 | // sets the floating member of the UIElement correctly! | |||
861 | xDockWindow->setFloatingMode( false ); | |||
862 | } | |||
863 | else | |||
864 | { | |||
865 | implts_writeWindowStateData( aUIElement ); | |||
866 | implts_sortUIElements(); | |||
867 | ||||
868 | if ( aUIElement.m_bVisible ) | |||
869 | implts_setLayoutDirty(); | |||
870 | } | |||
871 | return true; | |||
872 | } | |||
873 | } | |||
874 | catch (const lang::DisposedException&) | |||
875 | { | |||
876 | } | |||
877 | } | |||
878 | ||||
879 | return false; | |||
880 | } | |||
881 | ||||
882 | bool ToolbarLayoutManager::dockAllToolbars() | |||
883 | { | |||
884 | std::vector< OUString > aToolBarNameVector; | |||
885 | ||||
886 | { | |||
887 | SolarMutexGuard aReadLock; | |||
888 | for (auto const& elem : m_aUIElements) | |||
889 | { | |||
890 | if (elem.m_aType == "toolbar" && elem.m_xUIElement.is() && elem.m_bFloating | |||
891 | && elem.m_bVisible) | |||
892 | aToolBarNameVector.push_back(elem.m_aName); | |||
893 | } | |||
894 | } | |||
895 | ||||
896 | bool bResult(true); | |||
897 | const sal_uInt32 nCount = aToolBarNameVector.size(); | |||
898 | for ( sal_uInt32 i = 0; i < nCount; ++i ) | |||
899 | { | |||
900 | awt::Point aPoint; | |||
901 | aPoint.X = aPoint.Y = SAL_MAX_INT32((sal_Int32) 0x7FFFFFFF); | |||
902 | bResult &= dockToolbar( aToolBarNameVector[i], ui::DockingArea_DOCKINGAREA_DEFAULT, aPoint ); | |||
903 | } | |||
904 | ||||
905 | return bResult; | |||
906 | } | |||
907 | ||||
908 | void ToolbarLayoutManager::childWindowEvent( VclSimpleEvent const * pEvent ) | |||
909 | { | |||
910 | // To enable toolbar controllers to change their image when a sub-toolbar function | |||
911 | // is activated, we need this mechanism. We have NO connection between these toolbars | |||
912 | // anymore! | |||
913 | auto pWindowEvent = dynamic_cast< const VclWindowEvent* >(pEvent); | |||
914 | if (!pWindowEvent) | |||
915 | return; | |||
916 | ||||
917 | if ( pEvent->GetId() == VclEventId::ToolboxSelect ) | |||
918 | { | |||
919 | OUString aToolbarName; | |||
920 | OUString aCommand; | |||
921 | ToolBox* pToolBox = getToolboxPtr( pWindowEvent->GetWindow() ); | |||
922 | ||||
923 | if ( pToolBox ) | |||
924 | { | |||
925 | aToolbarName = retrieveToolbarNameFromHelpURL( pToolBox ); | |||
926 | sal_uInt16 nId = pToolBox->GetCurItemId(); | |||
927 | if ( nId > 0 ) | |||
928 | aCommand = pToolBox->GetItemCommand( nId ); | |||
929 | } | |||
930 | ||||
931 | if ( !aToolbarName.isEmpty() && !aCommand.isEmpty() ) | |||
932 | { | |||
933 | SolarMutexClearableGuard aReadLock; | |||
934 | ::std::vector< uno::Reference< ui::XUIFunctionListener > > aListenerArray; | |||
935 | ||||
936 | for (auto const& elem : m_aUIElements) | |||
937 | { | |||
938 | if ( elem.m_xUIElement.is() ) | |||
939 | { | |||
940 | uno::Reference< ui::XUIFunctionListener > xListener( elem.m_xUIElement, uno::UNO_QUERY ); | |||
941 | if ( xListener.is() ) | |||
942 | aListenerArray.push_back( xListener ); | |||
943 | } | |||
944 | } | |||
945 | aReadLock.clear(); | |||
946 | ||||
947 | const sal_uInt32 nCount = aListenerArray.size(); | |||
948 | for ( sal_uInt32 i = 0; i < nCount; ++i ) | |||
949 | { | |||
950 | try | |||
951 | { | |||
952 | aListenerArray[i]->functionExecute( aToolbarName, aCommand ); | |||
953 | } | |||
954 | catch (const uno::RuntimeException&) | |||
955 | { | |||
956 | throw; | |||
957 | } | |||
958 | catch (const uno::Exception&) | |||
959 | { | |||
960 | } | |||
961 | } | |||
962 | } | |||
963 | } | |||
964 | else if ( pEvent->GetId() == VclEventId::ToolboxFormatChanged ) | |||
965 | { | |||
966 | if ( !implts_isToolbarCreationActive() ) | |||
967 | { | |||
968 | ToolBox* pToolBox = getToolboxPtr( static_cast<VclWindowEvent const *>(pEvent)->GetWindow() ); | |||
969 | if ( pToolBox ) | |||
970 | { | |||
971 | OUString aToolbarName = retrieveToolbarNameFromHelpURL( pToolBox ); | |||
972 | if ( !aToolbarName.isEmpty() ) | |||
973 | { | |||
974 | OUString aToolbarUrl = "private:resource/toolbar/" + aToolbarName; | |||
975 | ||||
976 | UIElement aToolbar = implts_findToolbar( aToolbarUrl ); | |||
977 | if ( aToolbar.m_xUIElement.is() && !aToolbar.m_bFloating ) | |||
978 | { | |||
979 | implts_setLayoutDirty(); | |||
980 | m_pParentLayouter->requestLayout(); | |||
981 | } | |||
982 | } | |||
983 | } | |||
984 | } | |||
985 | } | |||
986 | } | |||
987 | ||||
988 | void ToolbarLayoutManager::resetDockingArea() | |||
989 | { | |||
990 | SolarMutexClearableGuard aReadLock; | |||
991 | uno::Reference< awt::XWindow > xTopDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
992 | uno::Reference< awt::XWindow > xLeftDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
993 | uno::Reference< awt::XWindow > xRightDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] ); | |||
994 | uno::Reference< awt::XWindow > xBottomDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] ); | |||
995 | aReadLock.clear(); | |||
996 | ||||
997 | if ( xTopDockingWindow.is() ) | |||
998 | xTopDockingWindow->setPosSize( 0, 0, 0, 0, awt::PosSize::POSSIZE ); | |||
999 | if ( xLeftDockingWindow.is() ) | |||
1000 | xLeftDockingWindow->setPosSize( 0, 0, 0, 0, awt::PosSize::POSSIZE ); | |||
1001 | if ( xRightDockingWindow.is() ) | |||
1002 | xRightDockingWindow->setPosSize( 0, 0, 0, 0, awt::PosSize::POSSIZE ); | |||
1003 | if ( xBottomDockingWindow.is() ) | |||
1004 | xBottomDockingWindow->setPosSize( 0, 0, 0, 0, awt::PosSize::POSSIZE ); | |||
1005 | } | |||
1006 | ||||
1007 | void ToolbarLayoutManager::setParentWindow( | |||
1008 | const uno::Reference< awt::XWindowPeer >& xParentWindow ) | |||
1009 | { | |||
1010 | static const char DOCKINGAREASTRING[] = "dockingarea"; | |||
1011 | ||||
1012 | uno::Reference< awt::XWindow > xTopDockWindow( createToolkitWindow( m_xContext, xParentWindow, DOCKINGAREASTRING ), uno::UNO_QUERY ); | |||
1013 | uno::Reference< awt::XWindow > xLeftDockWindow( createToolkitWindow( m_xContext, xParentWindow, DOCKINGAREASTRING ), uno::UNO_QUERY ); | |||
1014 | uno::Reference< awt::XWindow > xRightDockWindow( createToolkitWindow( m_xContext, xParentWindow, DOCKINGAREASTRING ), uno::UNO_QUERY ); | |||
1015 | uno::Reference< awt::XWindow > xBottomDockWindow( createToolkitWindow( m_xContext, xParentWindow, DOCKINGAREASTRING ), uno::UNO_QUERY ); | |||
1016 | ||||
1017 | { | |||
1018 | SolarMutexGuard aWriteLock; | |||
1019 | m_xContainerWindow.set(xParentWindow, uno::UNO_QUERY); | |||
1020 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] = xTopDockWindow; | |||
1021 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] = xLeftDockWindow; | |||
1022 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] = xRightDockWindow; | |||
1023 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] = xBottomDockWindow; | |||
1024 | } | |||
1025 | ||||
1026 | if ( xParentWindow.is() ) | |||
1027 | { | |||
1028 | SolarMutexGuard aGuard; | |||
1029 | VclPtr< ::DockingAreaWindow > pWindow = dynamic_cast< ::DockingAreaWindow* >(VCLUnoHelper::GetWindow( xTopDockWindow ).get() ); | |||
1030 | if( pWindow ) | |||
1031 | pWindow->SetAlign( WindowAlign::Top ); | |||
1032 | pWindow = dynamic_cast< ::DockingAreaWindow* >(VCLUnoHelper::GetWindow( xBottomDockWindow ).get() ); | |||
1033 | if( pWindow ) | |||
1034 | pWindow->SetAlign( WindowAlign::Bottom ); | |||
1035 | pWindow = dynamic_cast< ::DockingAreaWindow* >(VCLUnoHelper::GetWindow( xLeftDockWindow ).get() ); | |||
1036 | if( pWindow ) | |||
1037 | pWindow->SetAlign( WindowAlign::Left ); | |||
1038 | pWindow = dynamic_cast< ::DockingAreaWindow* >(VCLUnoHelper::GetWindow( xRightDockWindow ).get() ); | |||
1039 | if( pWindow ) | |||
1040 | pWindow->SetAlign( WindowAlign::Right ); | |||
1041 | implts_reparentToolbars(); | |||
1042 | } | |||
1043 | else | |||
1044 | { | |||
1045 | destroyToolbars(); | |||
1046 | resetDockingArea(); | |||
1047 | } | |||
1048 | } | |||
1049 | ||||
1050 | void ToolbarLayoutManager::setDockingAreaOffsets( const ::tools::Rectangle& rOffsets ) | |||
1051 | { | |||
1052 | SolarMutexGuard g; | |||
1053 | m_aDockingAreaOffsets = rOffsets; | |||
1054 | m_bLayoutDirty = true; | |||
1055 | } | |||
1056 | ||||
1057 | OUString ToolbarLayoutManager::implts_generateGenericAddonToolbarTitle( sal_Int32 nNumber ) const | |||
1058 | { | |||
1059 | OUString aAddonGenericTitle(FwkResId(STR_TOOLBAR_TITLE_ADDONreinterpret_cast<char const *>("STR_TOOLBAR_TITLE_ADDON" "\004" u8"Add-On %num%"))); | |||
1060 | const vcl::I18nHelper& rI18nHelper = Application::GetSettings().GetUILocaleI18nHelper(); | |||
1061 | ||||
1062 | OUString aNumStr = rI18nHelper.GetNum( nNumber, 0, false, false ); | |||
1063 | aAddonGenericTitle = aAddonGenericTitle.replaceFirst( "%num%", aNumStr ); | |||
1064 | ||||
1065 | return aAddonGenericTitle; | |||
1066 | } | |||
1067 | ||||
1068 | void ToolbarLayoutManager::implts_createAddonsToolBars() | |||
1069 | { | |||
1070 | SolarMutexClearableGuard aWriteLock; | |||
1071 | if ( !m_pAddonOptions ) | |||
1072 | m_pAddonOptions.reset( new AddonsOptions ); | |||
1073 | ||||
1074 | uno::Reference< ui::XUIElementFactory > xUIElementFactory( m_xUIElementFactoryManager ); | |||
1075 | uno::Reference< frame::XFrame > xFrame( m_xFrame ); | |||
1076 | aWriteLock.clear(); | |||
1077 | ||||
1078 | if (isPreviewFrame()) | |||
1079 | return; // no addon toolbars for preview frame! | |||
1080 | ||||
1081 | uno::Sequence< uno::Sequence< beans::PropertyValue > > aAddonToolBarData; | |||
1082 | uno::Reference< ui::XUIElement > xUIElement; | |||
1083 | ||||
1084 | sal_uInt32 nCount = m_pAddonOptions->GetAddonsToolBarCount(); | |||
1085 | ||||
1086 | uno::Sequence< beans::PropertyValue > aPropSeq( 2 ); | |||
1087 | aPropSeq[0].Name = "Frame"; | |||
1088 | aPropSeq[0].Value <<= xFrame; | |||
1089 | aPropSeq[1].Name = "ConfigurationData"; | |||
1090 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
1091 | { | |||
1092 | OUString aAddonToolBarName( "private:resource/toolbar/addon_" + | |||
1093 | m_pAddonOptions->GetAddonsToolbarResourceName(i) ); | |||
1094 | aAddonToolBarData = m_pAddonOptions->GetAddonsToolBarPart( i ); | |||
1095 | aPropSeq[1].Value <<= aAddonToolBarData; | |||
1096 | ||||
1097 | UIElement aElement = implts_findToolbar( aAddonToolBarName ); | |||
1098 | ||||
1099 | // #i79828 | |||
1100 | // It's now possible that we are called more than once. Be sure to not create | |||
1101 | // add-on toolbars more than once! | |||
1102 | if ( aElement.m_xUIElement.is() ) | |||
1103 | continue; | |||
1104 | ||||
1105 | try | |||
1106 | { | |||
1107 | xUIElement = xUIElementFactory->createUIElement( aAddonToolBarName, aPropSeq ); | |||
1108 | if ( xUIElement.is() ) | |||
1109 | { | |||
1110 | uno::Reference< awt::XDockableWindow > xDockWindow( xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
1111 | if ( xDockWindow.is() ) | |||
1112 | { | |||
1113 | try | |||
1114 | { | |||
1115 | xDockWindow->addDockableWindowListener( uno::Reference< awt::XDockableWindowListener >( static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
1116 | xDockWindow->enableDocking( true ); | |||
1117 | uno::Reference< awt::XWindow > xWindow( xDockWindow, uno::UNO_QUERY ); | |||
1118 | if ( xWindow.is() ) | |||
1119 | xWindow->addWindowListener( uno::Reference< awt::XWindowListener >( static_cast< OWeakObject * >( this ), uno::UNO_QUERY )); | |||
1120 | } | |||
1121 | catch (const uno::Exception&) | |||
1122 | { | |||
1123 | } | |||
1124 | } | |||
1125 | ||||
1126 | OUString aGenericAddonTitle = implts_generateGenericAddonToolbarTitle( i+1 ); | |||
1127 | ||||
1128 | if ( !aElement.m_aName.isEmpty() ) | |||
1129 | { | |||
1130 | // Reuse a local entry so we are able to use the latest | |||
1131 | // UI changes for this document. | |||
1132 | implts_setElementData( aElement, xDockWindow ); | |||
1133 | aElement.m_xUIElement = xUIElement; | |||
1134 | if ( aElement.m_aUIName.isEmpty() ) | |||
1135 | { | |||
1136 | aElement.m_aUIName = aGenericAddonTitle; | |||
1137 | implts_writeWindowStateData( aElement ); | |||
1138 | } | |||
1139 | } | |||
1140 | else | |||
1141 | { | |||
1142 | // Create new UI element and try to read its state data | |||
1143 | UIElement aNewToolbar( aAddonToolBarName, "toolbar", xUIElement ); | |||
1144 | aNewToolbar.m_bFloating = true; | |||
1145 | implts_readWindowStateData( aAddonToolBarName, aNewToolbar ); | |||
1146 | implts_setElementData( aNewToolbar, xDockWindow ); | |||
1147 | if ( aNewToolbar.m_aUIName.isEmpty() ) | |||
1148 | { | |||
1149 | aNewToolbar.m_aUIName = aGenericAddonTitle; | |||
1150 | implts_writeWindowStateData( aNewToolbar ); | |||
1151 | } | |||
1152 | implts_insertToolbar( aNewToolbar ); | |||
1153 | } | |||
1154 | ||||
1155 | uno::Reference< awt::XWindow > xWindow( xDockWindow, uno::UNO_QUERY ); | |||
1156 | if ( xWindow.is() ) | |||
1157 | { | |||
1158 | // Set generic title for add-on toolbar | |||
1159 | SolarMutexGuard aGuard; | |||
1160 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
1161 | if ( pWindow->GetText().isEmpty() ) | |||
1162 | pWindow->SetText( aGenericAddonTitle ); | |||
1163 | if ( pWindow->GetType() == WindowType::TOOLBOX ) | |||
1164 | { | |||
1165 | ToolBox* pToolbar = static_cast<ToolBox *>(pWindow.get()); | |||
1166 | pToolbar->SetMenuType(); | |||
1167 | } | |||
1168 | } | |||
1169 | } | |||
1170 | } | |||
1171 | catch (const container::NoSuchElementException&) | |||
1172 | { | |||
1173 | } | |||
1174 | catch (const lang::IllegalArgumentException&) | |||
1175 | { | |||
1176 | } | |||
1177 | } | |||
1178 | } | |||
1179 | ||||
1180 | void ToolbarLayoutManager::implts_createCustomToolBars() | |||
1181 | { | |||
1182 | SolarMutexClearableGuard aReadLock; | |||
1183 | if ( !m_bComponentAttached ) | |||
1184 | return; | |||
1185 | ||||
1186 | uno::Reference< frame::XFrame > xFrame( m_xFrame ); | |||
1187 | uno::Reference< ui::XUIConfigurationManager > xModuleCfgMgr = m_xModuleCfgMgr; | |||
1188 | uno::Reference< ui::XUIConfigurationManager > xDocCfgMgr = m_xDocCfgMgr; | |||
1189 | aReadLock.clear(); | |||
1190 | ||||
1191 | if ( !xFrame.is() ) | |||
1192 | return; | |||
1193 | ||||
1194 | if (isPreviewFrame()) | |||
1195 | return; // no custom toolbars for preview frame! | |||
1196 | ||||
1197 | uno::Sequence< uno::Sequence< beans::PropertyValue > > aTbxSeq; | |||
1198 | if ( xDocCfgMgr.is() ) | |||
1199 | { | |||
1200 | aTbxSeq = xDocCfgMgr->getUIElementsInfo( ui::UIElementType::TOOLBAR ); | |||
1201 | implts_createCustomToolBars( aTbxSeq ); // first create all document based toolbars | |||
1202 | } | |||
1203 | if ( xModuleCfgMgr.is() ) | |||
1204 | { | |||
1205 | aTbxSeq = xModuleCfgMgr->getUIElementsInfo( ui::UIElementType::TOOLBAR ); | |||
1206 | implts_createCustomToolBars( aTbxSeq ); // second create module based toolbars | |||
1207 | } | |||
1208 | } | |||
1209 | ||||
1210 | void ToolbarLayoutManager::implts_createNonContextSensitiveToolBars() | |||
1211 | { | |||
1212 | SolarMutexClearableGuard aReadLock; | |||
1213 | ||||
1214 | if ( !m_xPersistentWindowState.is() || !m_xFrame.is() || !m_bComponentAttached ) | |||
1215 | return; | |||
1216 | ||||
1217 | uno::Reference< container::XNameAccess > xPersistentWindowState( m_xPersistentWindowState ); | |||
1218 | aReadLock.clear(); | |||
1219 | ||||
1220 | if (isPreviewFrame()) | |||
1221 | return; | |||
1222 | ||||
1223 | std::vector< OUString > aMakeVisibleToolbars; | |||
1224 | ||||
1225 | try | |||
1226 | { | |||
1227 | const uno::Sequence< OUString > aToolbarNames = xPersistentWindowState->getElementNames(); | |||
1228 | ||||
1229 | if ( aToolbarNames.hasElements() ) | |||
1230 | { | |||
1231 | OUString aElementType; | |||
1232 | OUString aElementName; | |||
1233 | ||||
1234 | aMakeVisibleToolbars.reserve(aToolbarNames.getLength()); | |||
1235 | ||||
1236 | SolarMutexGuard g; | |||
1237 | ||||
1238 | for ( OUString const & aName : aToolbarNames ) | |||
1239 | { | |||
1240 | parseResourceURL( aName, aElementType, aElementName ); | |||
1241 | ||||
1242 | // Check that we only create: | |||
1243 | // - Toolbars (the statusbar is also member of the persistent window state) | |||
1244 | // - Not custom toolbars, there are created with their own method (implts_createCustomToolbars) | |||
1245 | if ( aElementType.equalsIgnoreAsciiCase("toolbar") && | |||
1246 | aElementName.indexOf( "custom_" ) == -1 ) | |||
1247 | { | |||
1248 | UIElement aNewToolbar = implts_findToolbar( aName ); | |||
1249 | bool bFound = ( aNewToolbar.m_aName == aName ); | |||
1250 | if ( !bFound ) | |||
1251 | implts_readWindowStateData( aName, aNewToolbar ); | |||
1252 | ||||
1253 | if ( aNewToolbar.m_bVisible && !aNewToolbar.m_bContextSensitive ) | |||
1254 | { | |||
1255 | if ( !bFound ) | |||
1256 | implts_insertToolbar( aNewToolbar ); | |||
1257 | aMakeVisibleToolbars.push_back( aName ); | |||
1258 | } | |||
1259 | } | |||
1260 | } | |||
1261 | } | |||
1262 | } | |||
1263 | catch (const uno::RuntimeException&) | |||
1264 | { | |||
1265 | throw; | |||
1266 | } | |||
1267 | catch (const uno::Exception&) | |||
1268 | { | |||
1269 | } | |||
1270 | ||||
1271 | for (auto const& rURL : aMakeVisibleToolbars) | |||
1272 | { | |||
1273 | requestToolbar(rURL); | |||
1274 | } | |||
1275 | } | |||
1276 | ||||
1277 | void ToolbarLayoutManager::implts_createCustomToolBars( const uno::Sequence< uno::Sequence< beans::PropertyValue > >& aTbxSeqSeq ) | |||
1278 | { | |||
1279 | for ( const uno::Sequence< beans::PropertyValue >& rTbxSeq : aTbxSeqSeq ) | |||
1280 | { | |||
1281 | OUString aTbxResName; | |||
1282 | OUString aTbxTitle; | |||
1283 | for ( const beans::PropertyValue& rProp : rTbxSeq ) | |||
1284 | { | |||
1285 | if ( rProp.Name == "ResourceURL" ) | |||
1286 | rProp.Value >>= aTbxResName; | |||
1287 | else if ( rProp.Name == "UIName" ) | |||
1288 | rProp.Value >>= aTbxTitle; | |||
1289 | } | |||
1290 | ||||
1291 | // Only create custom toolbars. Their name have to start with "custom_"! | |||
1292 | if ( !aTbxResName.isEmpty() && ( aTbxResName.indexOf( "custom_" ) != -1 ) ) | |||
1293 | implts_createCustomToolBar( aTbxResName, aTbxTitle ); | |||
1294 | } | |||
1295 | } | |||
1296 | ||||
1297 | void ToolbarLayoutManager::implts_createCustomToolBar( const OUString& aTbxResName, const OUString& aTitle ) | |||
1298 | { | |||
1299 | if ( aTbxResName.isEmpty() ) | |||
1300 | return; | |||
1301 | ||||
1302 | if ( !createToolbar( aTbxResName ) ) | |||
1303 | SAL_WARN("fwk.uielement", "ToolbarLayoutManager cannot create custom toolbar")do { if (true) { switch (sal_detail_log_report(::SAL_DETAIL_LOG_LEVEL_WARN , "fwk.uielement")) { case SAL_DETAIL_LOG_ACTION_IGNORE: break ; case SAL_DETAIL_LOG_ACTION_LOG: if (sizeof ::sal::detail::getResult ( ::sal::detail::StreamStart() << "ToolbarLayoutManager cannot create custom toolbar" ) == 1) { ::sal_detail_log( (::SAL_DETAIL_LOG_LEVEL_WARN), ("fwk.uielement" ), ("/home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx" ":" "1303" ": "), ::sal::detail::unwrapStream( ::sal::detail ::StreamStart() << "ToolbarLayoutManager cannot create custom toolbar" ), 0); } else { ::std::ostringstream sal_detail_stream; sal_detail_stream << "ToolbarLayoutManager cannot create custom toolbar" ; ::sal::detail::log( (::SAL_DETAIL_LOG_LEVEL_WARN), ("fwk.uielement" ), ("/home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx" ":" "1303" ": "), sal_detail_stream, 0); }; break; case SAL_DETAIL_LOG_ACTION_FATAL : if (sizeof ::sal::detail::getResult( ::sal::detail::StreamStart () << "ToolbarLayoutManager cannot create custom toolbar" ) == 1) { ::sal_detail_log( (::SAL_DETAIL_LOG_LEVEL_WARN), ("fwk.uielement" ), ("/home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx" ":" "1303" ": "), ::sal::detail::unwrapStream( ::sal::detail ::StreamStart() << "ToolbarLayoutManager cannot create custom toolbar" ), 0); } else { ::std::ostringstream sal_detail_stream; sal_detail_stream << "ToolbarLayoutManager cannot create custom toolbar" ; ::sal::detail::log( (::SAL_DETAIL_LOG_LEVEL_WARN), ("fwk.uielement" ), ("/home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx" ":" "1303" ": "), sal_detail_stream, 0); }; std::abort(); break ; } } } while (false); | |||
1304 | ||||
1305 | uno::Reference< ui::XUIElement > xUIElement = getToolbar( aTbxResName ); | |||
1306 | ||||
1307 | if ( !aTitle.isEmpty() && xUIElement.is() ) | |||
1308 | { | |||
1309 | SolarMutexGuard aGuard; | |||
1310 | ||||
1311 | vcl::Window* pWindow = getWindowFromXUIElement( xUIElement ); | |||
1312 | if ( pWindow ) | |||
1313 | pWindow->SetText( aTitle ); | |||
1314 | } | |||
1315 | } | |||
1316 | ||||
1317 | void ToolbarLayoutManager::implts_reparentToolbars() | |||
1318 | { | |||
1319 | SolarMutexClearableGuard aWriteLock; | |||
1320 | UIElementVector aUIElementVector = m_aUIElements; | |||
1321 | VclPtr<vcl::Window> pContainerWindow = VCLUnoHelper::GetWindow( m_xContainerWindow ); | |||
1322 | VclPtr<vcl::Window> pTopDockWindow = VCLUnoHelper::GetWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
1323 | VclPtr<vcl::Window> pBottomDockWindow = VCLUnoHelper::GetWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] ); | |||
1324 | VclPtr<vcl::Window> pLeftDockWindow = VCLUnoHelper::GetWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
1325 | VclPtr<vcl::Window> pRightDockWindow = VCLUnoHelper::GetWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] ); | |||
1326 | aWriteLock.clear(); | |||
1327 | ||||
1328 | SolarMutexGuard aGuard; | |||
1329 | if ( !pContainerWindow ) | |||
1330 | return; | |||
1331 | ||||
1332 | for (auto const& elem : aUIElementVector) | |||
1333 | { | |||
1334 | uno::Reference< ui::XUIElement > xUIElement( elem.m_xUIElement ); | |||
1335 | if ( xUIElement.is() ) | |||
1336 | { | |||
1337 | uno::Reference< awt::XWindow > xWindow; | |||
1338 | try | |||
1339 | { | |||
1340 | // We have to retrieve the window reference with try/catch as it is | |||
1341 | // possible that all elements have been disposed! | |||
1342 | xWindow.set( xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
1343 | } | |||
1344 | catch (const uno::RuntimeException&) | |||
1345 | { | |||
1346 | throw; | |||
1347 | } | |||
1348 | catch (const uno::Exception&) | |||
1349 | { | |||
1350 | } | |||
1351 | ||||
1352 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
1353 | if ( pWindow ) | |||
1354 | { | |||
1355 | // Reparent our child windows according to their current state. | |||
1356 | if ( elem.m_bFloating ) | |||
1357 | pWindow->SetParent( pContainerWindow ); | |||
1358 | else | |||
1359 | { | |||
1360 | if ( elem.m_aDockedData.m_nDockedArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
1361 | pWindow->SetParent( pTopDockWindow ); | |||
1362 | else if ( elem.m_aDockedData.m_nDockedArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) | |||
1363 | pWindow->SetParent( pBottomDockWindow ); | |||
1364 | else if ( elem.m_aDockedData.m_nDockedArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
1365 | pWindow->SetParent( pLeftDockWindow ); | |||
1366 | else | |||
1367 | pWindow->SetParent( pRightDockWindow ); | |||
1368 | } | |||
1369 | } | |||
1370 | } | |||
1371 | } | |||
1372 | } | |||
1373 | ||||
1374 | void ToolbarLayoutManager::implts_setToolbarCreation( bool bStart ) | |||
1375 | { | |||
1376 | SolarMutexGuard g; | |||
1377 | m_bToolbarCreation = bStart; | |||
1378 | } | |||
1379 | ||||
1380 | bool ToolbarLayoutManager::implts_isToolbarCreationActive() | |||
1381 | { | |||
1382 | SolarMutexGuard g; | |||
1383 | return m_bToolbarCreation; | |||
1384 | } | |||
1385 | ||||
1386 | void ToolbarLayoutManager::implts_setElementData( UIElement& rElement, const uno::Reference< awt::XDockableWindow >& rDockWindow ) | |||
1387 | { | |||
1388 | SolarMutexClearableGuard aReadLock; | |||
1389 | bool bShowElement( rElement.m_bVisible && !rElement.m_bMasterHide && implts_isParentWindowVisible() ); | |||
1390 | aReadLock.clear(); | |||
1391 | ||||
1392 | uno::Reference< awt::XWindow2 > xWindow( rDockWindow, uno::UNO_QUERY ); | |||
1393 | ||||
1394 | vcl::Window* pWindow( nullptr ); | |||
1395 | ToolBox* pToolBox( nullptr ); | |||
1396 | ||||
1397 | if ( !(rDockWindow.is() && xWindow.is()) ) | |||
1398 | return; | |||
1399 | ||||
1400 | { | |||
1401 | SolarMutexGuard aGuard; | |||
1402 | pWindow = VCLUnoHelper::GetWindow( xWindow ).get(); | |||
1403 | if ( pWindow ) | |||
1404 | { | |||
1405 | OUString aText = pWindow->GetText(); | |||
1406 | if ( aText.isEmpty() ) | |||
1407 | pWindow->SetText( rElement.m_aUIName ); | |||
1408 | if ( rElement.m_bNoClose ) | |||
1409 | pWindow->SetStyle( pWindow->GetStyle() & ~WB_CLOSEABLE ); | |||
1410 | if ( pWindow->GetType() == WindowType::TOOLBOX ) | |||
1411 | pToolBox = static_cast<ToolBox *>(pWindow); | |||
1412 | } | |||
1413 | if ( pToolBox ) | |||
1414 | { | |||
1415 | pToolBox->SetButtonType( rElement.m_nStyle ); | |||
1416 | if ( rElement.m_bNoClose ) | |||
1417 | pToolBox->SetFloatStyle( pToolBox->GetFloatStyle() & ~WB_CLOSEABLE ); | |||
1418 | } | |||
1419 | } | |||
1420 | ||||
1421 | if ( rElement.m_bFloating ) | |||
1422 | { | |||
1423 | if ( pWindow ) | |||
1424 | { | |||
1425 | SolarMutexGuard aGuard; | |||
1426 | OUString aText = pWindow->GetText(); | |||
1427 | if ( aText.isEmpty() ) | |||
1428 | pWindow->SetText( rElement.m_aUIName ); | |||
1429 | } | |||
1430 | ||||
1431 | awt::Point aPos(rElement.m_aFloatingData.m_aPos); | |||
1432 | bool bWriteData( false ); | |||
1433 | bool bUndefPos = hasDefaultPosValue( rElement.m_aFloatingData.m_aPos ); | |||
1434 | bool bSetSize = ( rElement.m_aFloatingData.m_aSize.Width != 0 && | |||
1435 | rElement.m_aFloatingData.m_aSize.Height != 0 ); | |||
1436 | rDockWindow->setFloatingMode( true ); | |||
1437 | if ( bUndefPos ) | |||
1438 | { | |||
1439 | aPos = implts_findNextCascadeFloatingPos(); | |||
1440 | rElement.m_aFloatingData.m_aPos = aPos; // set new cascaded position | |||
1441 | bWriteData = true; | |||
1442 | } | |||
1443 | ||||
1444 | if( bSetSize ) | |||
1445 | xWindow->setOutputSize(rElement.m_aFloatingData.m_aSize); | |||
1446 | else | |||
1447 | { | |||
1448 | if( pToolBox ) | |||
1449 | { | |||
1450 | // set an optimal initial floating size | |||
1451 | SolarMutexGuard aGuard; | |||
1452 | ::Size aSize( pToolBox->CalcFloatingWindowSizePixel() ); | |||
1453 | pToolBox->SetOutputSizePixel( aSize ); | |||
1454 | } | |||
1455 | } | |||
1456 | ||||
1457 | // #i60882# IMPORTANT: Set position after size as it is | |||
1458 | // possible that we position some part of the toolbar | |||
1459 | // outside of the desktop. A default constructed toolbar | |||
1460 | // always has one line. Now VCL automatically | |||
1461 | // position the toolbar back into the desktop. Therefore | |||
1462 | // we resize the toolbar with the new (wrong) position. | |||
1463 | // To fix this problem we have to set the size BEFORE the | |||
1464 | // position. | |||
1465 | xWindow->setPosSize( aPos.X, aPos.Y, 0, 0, awt::PosSize::POS ); | |||
1466 | ||||
1467 | if ( bWriteData ) | |||
1468 | implts_writeWindowStateData( rElement ); | |||
1469 | if ( bShowElement && pWindow ) | |||
1470 | { | |||
1471 | SolarMutexGuard aGuard; | |||
1472 | pWindow->Show( true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
1473 | } | |||
1474 | } | |||
1475 | else | |||
1476 | { | |||
1477 | bool bSetSize( false ); | |||
1478 | ::Point aPixelPos; | |||
1479 | ::Size aSize; | |||
1480 | ||||
1481 | if ( pToolBox ) | |||
1482 | { | |||
1483 | SolarMutexGuard aGuard; | |||
1484 | pToolBox->SetAlign( ImplConvertAlignment(rElement.m_aDockedData.m_nDockedArea ) ); | |||
1485 | pToolBox->SetLineCount( 1 ); | |||
1486 | rDockWindow->setFloatingMode( false ); | |||
1487 | if ( rElement.m_aDockedData.m_bLocked ) | |||
1488 | rDockWindow->lock(); | |||
1489 | aSize = pToolBox->CalcWindowSizePixel(); | |||
1490 | bSetSize = true; | |||
1491 | ||||
1492 | if ( isDefaultPos( rElement.m_aDockedData.m_aPos )) | |||
1493 | { | |||
1494 | awt::Point aDockPos; | |||
1495 | implts_findNextDockingPos( rElement.m_aDockedData.m_nDockedArea, aSize, aDockPos, aPixelPos ); | |||
1496 | rElement.m_aDockedData.m_aPos = aDockPos; | |||
1497 | } | |||
1498 | } | |||
1499 | ||||
1500 | xWindow->setPosSize( aPixelPos.X(), aPixelPos.Y(), 0, 0, awt::PosSize::POS ); | |||
1501 | if( bSetSize ) | |||
1502 | xWindow->setOutputSize( AWTSize( aSize) ); | |||
1503 | ||||
1504 | if ( pWindow ) | |||
1505 | { | |||
1506 | SolarMutexGuard aGuard; | |||
1507 | if ( !bShowElement ) | |||
1508 | pWindow->Hide(); | |||
1509 | } | |||
1510 | } | |||
1511 | } | |||
1512 | ||||
1513 | void ToolbarLayoutManager::implts_destroyDockingAreaWindows() | |||
1514 | { | |||
1515 | SolarMutexClearableGuard aWriteLock; | |||
1516 | uno::Reference< awt::XWindow > xTopDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
1517 | uno::Reference< awt::XWindow > xLeftDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
1518 | uno::Reference< awt::XWindow > xRightDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] ); | |||
1519 | uno::Reference< awt::XWindow > xBottomDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] ); | |||
1520 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)].clear(); | |||
1521 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)].clear(); | |||
1522 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)].clear(); | |||
1523 | m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)].clear(); | |||
1524 | aWriteLock.clear(); | |||
1525 | ||||
1526 | // destroy windows | |||
1527 | xTopDockingWindow->dispose(); | |||
1528 | xLeftDockingWindow->dispose(); | |||
1529 | xRightDockingWindow->dispose(); | |||
1530 | xBottomDockingWindow->dispose(); | |||
1531 | } | |||
1532 | ||||
1533 | // persistence methods | |||
1534 | ||||
1535 | bool ToolbarLayoutManager::implts_readWindowStateData( const OUString& aName, UIElement& rElementData ) | |||
1536 | { | |||
1537 | return LayoutManager::readWindowStateData( aName, rElementData, m_xPersistentWindowState, | |||
1538 | m_pGlobalSettings, m_bGlobalSettings, m_xContext ); | |||
1539 | } | |||
1540 | ||||
1541 | void ToolbarLayoutManager::implts_writeWindowStateData( const UIElement& rElementData ) | |||
1542 | { | |||
1543 | SolarMutexClearableGuard aWriteLock; | |||
1544 | uno::Reference< container::XNameAccess > xPersistentWindowState( m_xPersistentWindowState ); | |||
1545 | aWriteLock.clear(); | |||
1546 | ||||
1547 | bool bPersistent( false ); | |||
1548 | uno::Reference< beans::XPropertySet > xPropSet( rElementData.m_xUIElement, uno::UNO_QUERY ); | |||
1549 | if ( xPropSet.is() ) | |||
1550 | { | |||
1551 | try | |||
1552 | { | |||
1553 | // Check persistent flag of the user interface element | |||
1554 | xPropSet->getPropertyValue("Persistent") >>= bPersistent; | |||
1555 | } | |||
1556 | catch (const beans::UnknownPropertyException&) | |||
1557 | { | |||
1558 | bPersistent = true; // Non-configurable elements should at least store their dimension/position | |||
1559 | } | |||
1560 | catch (const lang::WrappedTargetException&) | |||
1561 | { | |||
1562 | } | |||
1563 | } | |||
1564 | ||||
1565 | if ( !(bPersistent && xPersistentWindowState.is()) ) | |||
1566 | return; | |||
1567 | ||||
1568 | try | |||
1569 | { | |||
1570 | uno::Sequence< beans::PropertyValue > aWindowState( 9 ); | |||
1571 | ||||
1572 | aWindowState[0].Name = WINDOWSTATE_PROPERTY_DOCKED"Docked"; | |||
1573 | aWindowState[0].Value <<= !rElementData.m_bFloating; | |||
1574 | aWindowState[1].Name = WINDOWSTATE_PROPERTY_VISIBLE"Visible"; | |||
1575 | aWindowState[1].Value <<= rElementData.m_bVisible; | |||
1576 | aWindowState[2].Name = WINDOWSTATE_PROPERTY_DOCKINGAREA"DockingArea"; | |||
1577 | aWindowState[2].Value <<= rElementData.m_aDockedData.m_nDockedArea; | |||
1578 | ||||
1579 | awt::Point aPos = rElementData.m_aDockedData.m_aPos; | |||
1580 | aWindowState[3].Name = WINDOWSTATE_PROPERTY_DOCKPOS"DockPos"; | |||
1581 | aWindowState[3].Value <<= aPos; | |||
1582 | ||||
1583 | aPos = rElementData.m_aFloatingData.m_aPos; | |||
1584 | aWindowState[4].Name = WINDOWSTATE_PROPERTY_POS"Pos"; | |||
1585 | aWindowState[4].Value <<= aPos; | |||
1586 | ||||
1587 | aWindowState[5].Name = WINDOWSTATE_PROPERTY_SIZE"Size"; | |||
1588 | aWindowState[5].Value <<= rElementData.m_aFloatingData.m_aSize; | |||
1589 | aWindowState[6].Name = WINDOWSTATE_PROPERTY_UINAME"UIName"; | |||
1590 | aWindowState[6].Value <<= rElementData.m_aUIName; | |||
1591 | aWindowState[7].Name = WINDOWSTATE_PROPERTY_LOCKED"Locked"; | |||
1592 | aWindowState[7].Value <<= rElementData.m_aDockedData.m_bLocked; | |||
1593 | aWindowState[8].Name = WINDOWSTATE_PROPERTY_STYLE"Style"; | |||
1594 | aWindowState[8].Value <<= static_cast<sal_uInt16>(rElementData.m_nStyle); | |||
1595 | ||||
1596 | OUString aName = rElementData.m_aName; | |||
1597 | if ( xPersistentWindowState->hasByName( aName )) | |||
1598 | { | |||
1599 | uno::Reference< container::XNameReplace > xReplace( xPersistentWindowState, uno::UNO_QUERY ); | |||
1600 | xReplace->replaceByName( aName, uno::makeAny( aWindowState )); | |||
1601 | } | |||
1602 | else | |||
1603 | { | |||
1604 | uno::Reference< container::XNameContainer > xInsert( xPersistentWindowState, uno::UNO_QUERY ); | |||
1605 | xInsert->insertByName( aName, uno::makeAny( aWindowState )); | |||
1606 | } | |||
1607 | } | |||
1608 | catch (const uno::Exception&) | |||
1609 | { | |||
1610 | } | |||
1611 | } | |||
1612 | ||||
1613 | /****************************************************************************** | |||
1614 | LOOKUP PART FOR TOOLBARS | |||
1615 | ******************************************************************************/ | |||
1616 | ||||
1617 | UIElement& ToolbarLayoutManager::impl_findToolbar( const OUString& aName ) | |||
1618 | { | |||
1619 | static UIElement aEmptyElement; | |||
1620 | ||||
1621 | SolarMutexGuard g; | |||
1622 | for (auto & elem : m_aUIElements) | |||
1623 | { | |||
1624 | if ( elem.m_aName == aName ) | |||
1625 | return elem; | |||
1626 | } | |||
1627 | ||||
1628 | return aEmptyElement; | |||
1629 | } | |||
1630 | ||||
1631 | UIElement ToolbarLayoutManager::implts_findToolbar( const OUString& aName ) | |||
1632 | { | |||
1633 | SolarMutexGuard g; | |||
1634 | return impl_findToolbar( aName ); | |||
1635 | } | |||
1636 | ||||
1637 | UIElement ToolbarLayoutManager::implts_findToolbar( const uno::Reference< uno::XInterface >& xToolbar ) | |||
1638 | { | |||
1639 | UIElement aToolbar; | |||
1640 | ||||
1641 | SolarMutexGuard g; | |||
1642 | for (auto const& elem : m_aUIElements) | |||
1643 | { | |||
1644 | if ( elem.m_xUIElement.is() ) | |||
1645 | { | |||
1646 | uno::Reference< uno::XInterface > xIfac( elem.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
1647 | if ( xIfac == xToolbar ) | |||
1648 | { | |||
1649 | aToolbar = elem; | |||
1650 | break; | |||
1651 | } | |||
1652 | } | |||
1653 | } | |||
1654 | ||||
1655 | return aToolbar; | |||
1656 | } | |||
1657 | ||||
1658 | uno::Reference< awt::XWindow > ToolbarLayoutManager::implts_getXWindow( const OUString& aName ) | |||
1659 | { | |||
1660 | uno::Reference< awt::XWindow > xWindow; | |||
1661 | ||||
1662 | SolarMutexGuard g; | |||
1663 | for (auto const& elem : m_aUIElements) | |||
1664 | { | |||
1665 | if ( elem.m_aName == aName && elem.m_xUIElement.is() ) | |||
1666 | { | |||
1667 | xWindow.set( elem.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
1668 | break; | |||
1669 | } | |||
1670 | } | |||
1671 | ||||
1672 | return xWindow; | |||
1673 | } | |||
1674 | ||||
1675 | vcl::Window* ToolbarLayoutManager::implts_getWindow( const OUString& aName ) | |||
1676 | { | |||
1677 | uno::Reference< awt::XWindow > xWindow = implts_getXWindow( aName ); | |||
1678 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
1679 | ||||
1680 | return pWindow; | |||
1681 | } | |||
1682 | ||||
1683 | bool ToolbarLayoutManager::implts_insertToolbar( const UIElement& rUIElement ) | |||
1684 | { | |||
1685 | UIElement aTempData; | |||
1686 | bool bFound( false ); | |||
1687 | bool bResult( false ); | |||
1688 | ||||
1689 | aTempData = implts_findToolbar( rUIElement.m_aName ); | |||
1690 | if ( aTempData.m_aName == rUIElement.m_aName ) | |||
1691 | bFound = true; | |||
1692 | ||||
1693 | if ( !bFound ) | |||
1694 | { | |||
1695 | SolarMutexGuard g; | |||
1696 | m_aUIElements.push_back( rUIElement ); | |||
1697 | bResult = true; | |||
1698 | } | |||
1699 | ||||
1700 | return bResult; | |||
1701 | } | |||
1702 | ||||
1703 | void ToolbarLayoutManager::implts_setToolbar( const UIElement& rUIElement ) | |||
1704 | { | |||
1705 | SolarMutexGuard g; | |||
1706 | UIElement& rData = impl_findToolbar( rUIElement.m_aName ); | |||
1707 | if ( rData.m_aName == rUIElement.m_aName ) | |||
1708 | rData = rUIElement; | |||
1709 | else | |||
1710 | m_aUIElements.push_back( rUIElement ); | |||
1711 | } | |||
1712 | ||||
1713 | /****************************************************************************** | |||
1714 | LAYOUT CODE PART FOR TOOLBARS | |||
1715 | ******************************************************************************/ | |||
1716 | ||||
1717 | awt::Point ToolbarLayoutManager::implts_findNextCascadeFloatingPos() | |||
1718 | { | |||
1719 | const sal_Int32 nHotZoneX = 50; | |||
1720 | const sal_Int32 nHotZoneY = 50; | |||
1721 | const sal_Int32 nCascadeIndentX = 15; | |||
1722 | const sal_Int32 nCascadeIndentY = 15; | |||
1723 | ||||
1724 | SolarMutexClearableGuard aReadLock; | |||
1725 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
1726 | uno::Reference< awt::XWindow > xTopDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
1727 | uno::Reference< awt::XWindow > xLeftDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
1728 | aReadLock.clear(); | |||
1729 | ||||
1730 | awt::Point aStartPos( nCascadeIndentX, nCascadeIndentY ); | |||
1731 | awt::Point aCurrPos( aStartPos ); | |||
1732 | ||||
1733 | if ( xContainerWindow.is() ) | |||
1734 | { | |||
1735 | SolarMutexGuard aGuard; | |||
1736 | VclPtr<vcl::Window> pContainerWindow = VCLUnoHelper::GetWindow( xContainerWindow ); | |||
1737 | if ( pContainerWindow ) | |||
1738 | aStartPos = AWTPoint(pContainerWindow->OutputToScreenPixel(VCLPoint(aStartPos))); | |||
1739 | } | |||
1740 | ||||
1741 | // Determine size of top and left docking area | |||
1742 | awt::Rectangle aTopRect( xTopDockingWindow->getPosSize() ); | |||
1743 | awt::Rectangle aLeftRect( xLeftDockingWindow->getPosSize() ); | |||
1744 | ||||
1745 | aStartPos.X += aLeftRect.Width + nCascadeIndentX; | |||
1746 | aStartPos.Y += aTopRect.Height + nCascadeIndentY; | |||
1747 | aCurrPos = aStartPos; | |||
1748 | ||||
1749 | // Try to find a cascaded position for the new floating window | |||
1750 | for (auto const& elem : m_aUIElements) | |||
1751 | { | |||
1752 | if ( elem.m_xUIElement.is() ) | |||
1753 | { | |||
1754 | uno::Reference< awt::XDockableWindow > xDockWindow( elem.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
1755 | uno::Reference< awt::XWindow > xWindow( xDockWindow, uno::UNO_QUERY ); | |||
1756 | if ( xDockWindow.is() && xDockWindow->isFloating() ) | |||
1757 | { | |||
1758 | SolarMutexGuard aGuard; | |||
1759 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
1760 | if ( pWindow && pWindow->IsVisible() ) | |||
1761 | { | |||
1762 | awt::Rectangle aFloatRect = xWindow->getPosSize(); | |||
1763 | if ((( aFloatRect.X - nHotZoneX ) <= aCurrPos.X ) && | |||
1764 | ( aFloatRect.X >= aCurrPos.X ) && | |||
1765 | (( aFloatRect.Y - nHotZoneY ) <= aCurrPos.Y ) && | |||
1766 | ( aFloatRect.Y >= aCurrPos.Y )) | |||
1767 | { | |||
1768 | aCurrPos.X = aFloatRect.X + nCascadeIndentX; | |||
1769 | aCurrPos.Y = aFloatRect.Y + nCascadeIndentY; | |||
1770 | } | |||
1771 | } | |||
1772 | } | |||
1773 | } | |||
1774 | } | |||
1775 | ||||
1776 | return aCurrPos; | |||
1777 | } | |||
1778 | ||||
1779 | void ToolbarLayoutManager::implts_sortUIElements() | |||
1780 | { | |||
1781 | SolarMutexGuard g; | |||
1782 | ||||
1783 | std::stable_sort( m_aUIElements.begin(), m_aUIElements.end()); // first created element should first | |||
1784 | ||||
1785 | // We have to reset our temporary flags. | |||
1786 | for (auto & elem : m_aUIElements) | |||
1787 | elem.m_bUserActive = false; | |||
1788 | } | |||
1789 | ||||
1790 | void ToolbarLayoutManager::implts_getUIElementVectorCopy( UIElementVector& rCopy ) | |||
1791 | { | |||
1792 | SolarMutexGuard g; | |||
1793 | rCopy = m_aUIElements; | |||
1794 | } | |||
1795 | ||||
1796 | ::Size ToolbarLayoutManager::implts_getTopBottomDockingAreaSizes() | |||
1797 | { | |||
1798 | ::Size aSize; | |||
1799 | uno::Reference< awt::XWindow > xTopDockingAreaWindow; | |||
1800 | uno::Reference< awt::XWindow > xBottomDockingAreaWindow; | |||
1801 | ||||
1802 | { | |||
1803 | SolarMutexGuard aReadLock; | |||
1804 | xTopDockingAreaWindow = m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)]; | |||
1805 | xBottomDockingAreaWindow = m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)]; | |||
1806 | } | |||
1807 | ||||
1808 | if ( xTopDockingAreaWindow.is() ) | |||
1809 | aSize.setWidth( xTopDockingAreaWindow->getPosSize().Height ); | |||
1810 | if ( xBottomDockingAreaWindow.is() ) | |||
1811 | aSize.setHeight( xBottomDockingAreaWindow->getPosSize().Height ); | |||
1812 | ||||
1813 | return aSize; | |||
1814 | } | |||
1815 | ||||
1816 | void ToolbarLayoutManager::implts_getDockingAreaElementInfos( ui::DockingArea eDockingArea, std::vector< SingleRowColumnWindowData >& rRowColumnsWindowData ) | |||
1817 | { | |||
1818 | std::vector< UIElement > aWindowVector; | |||
1819 | ||||
1820 | if (( eDockingArea < ui::DockingArea_DOCKINGAREA_TOP ) || ( eDockingArea > ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
1821 | eDockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
1822 | ||||
1823 | uno::Reference< awt::XWindow > xDockAreaWindow; | |||
1824 | ||||
1825 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
1826 | { | |||
1827 | SolarMutexGuard aReadLock; | |||
1828 | aWindowVector.reserve(m_aUIElements.size()); | |||
1829 | xDockAreaWindow = m_xDockAreaWindows[static_cast<int>(eDockingArea)]; | |||
1830 | for (auto const& elem : m_aUIElements) | |||
1831 | { | |||
1832 | if (elem.m_aDockedData.m_nDockedArea == eDockingArea && elem.m_bVisible) | |||
1833 | { | |||
1834 | uno::Reference<ui::XUIElement> xUIElement(elem.m_xUIElement); | |||
1835 | if (xUIElement.is()) | |||
1836 | { | |||
1837 | uno::Reference<awt::XWindow> xWindow(xUIElement->getRealInterface(), | |||
1838 | uno::UNO_QUERY); | |||
1839 | uno::Reference<awt::XDockableWindow> xDockWindow(xWindow, uno::UNO_QUERY); | |||
1840 | if (xDockWindow.is()) | |||
1841 | { | |||
1842 | if (!elem.m_bFloating) | |||
1843 | { | |||
1844 | // docked windows | |||
1845 | aWindowVector.push_back(elem); | |||
1846 | } | |||
1847 | else | |||
1848 | { | |||
1849 | // floating windows | |||
1850 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow(xWindow); | |||
1851 | DockingManager* pDockMgr = vcl::Window::GetDockingManager(); | |||
1852 | if (pDockMgr != nullptr) | |||
1853 | { | |||
1854 | ImplDockingWindowWrapper* pWrapper | |||
1855 | = pDockMgr->GetDockingWindowWrapper(pWindow); | |||
1856 | if (pWrapper != nullptr && pWrapper->GetFloatingWindow()) | |||
1857 | { | |||
1858 | // update the position data of the floating window | |||
1859 | if (pWrapper->GetFloatingWindow()->UpdatePositionData()) | |||
1860 | { | |||
1861 | awt::Rectangle aTmpRect = xWindow->getPosSize(); | |||
1862 | UIElement uiElem = elem; | |||
1863 | uiElem.m_aFloatingData.m_aPos | |||
1864 | = awt::Point(aTmpRect.X, aTmpRect.Y); | |||
1865 | implts_setToolbar(uiElem); | |||
1866 | implts_writeWindowStateData(uiElem); | |||
1867 | } | |||
1868 | } | |||
1869 | } | |||
1870 | } | |||
1871 | } | |||
1872 | } | |||
1873 | } | |||
1874 | } | |||
1875 | } | |||
1876 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
1877 | ||||
1878 | rRowColumnsWindowData.clear(); | |||
1879 | ||||
1880 | // Collect data from windows that are on the same row/column | |||
1881 | sal_Int32 j; | |||
1882 | sal_Int32 nIndex( 0 ); | |||
1883 | sal_Int32 nLastPos( 0 ); | |||
1884 | sal_Int32 nCurrPos( -1 ); | |||
1885 | sal_Int32 nLastRowColPixelPos( 0 ); | |||
1886 | awt::Rectangle aDockAreaRect; | |||
1887 | ||||
1888 | if ( xDockAreaWindow.is() ) | |||
1889 | aDockAreaRect = xDockAreaWindow->getPosSize(); | |||
1890 | ||||
1891 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
1892 | nLastRowColPixelPos = 0; | |||
1893 | else if ( eDockingArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) | |||
1894 | nLastRowColPixelPos = aDockAreaRect.Height; | |||
1895 | else if ( eDockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
1896 | nLastRowColPixelPos = 0; | |||
1897 | else | |||
1898 | nLastRowColPixelPos = aDockAreaRect.Width; | |||
1899 | ||||
1900 | const sal_uInt32 nCount = aWindowVector.size(); | |||
1901 | for ( j = 0; j < sal_Int32( nCount); j++ ) | |||
1902 | { | |||
1903 | const UIElement& rElement = aWindowVector[j]; | |||
1904 | uno::Reference< awt::XWindow > xWindow; | |||
1905 | uno::Reference< ui::XUIElement > xUIElement( rElement.m_xUIElement ); | |||
1906 | awt::Rectangle aPosSize; | |||
1907 | ||||
1908 | if ( !lcl_checkUIElement(xUIElement,aPosSize,xWindow) ) | |||
1909 | continue; | |||
1910 | if ( isHorizontalDockingArea( eDockingArea )) | |||
1911 | { | |||
1912 | if ( nCurrPos == -1 ) | |||
1913 | { | |||
1914 | nCurrPos = rElement.m_aDockedData.m_aPos.Y; | |||
1915 | nLastPos = 0; | |||
1916 | ||||
1917 | SingleRowColumnWindowData aRowColumnWindowData; | |||
1918 | aRowColumnWindowData.nRowColumn = nCurrPos; | |||
1919 | rRowColumnsWindowData.push_back( aRowColumnWindowData ); | |||
1920 | } | |||
1921 | ||||
1922 | sal_Int32 nSpace( 0 ); | |||
1923 | if ( rElement.m_aDockedData.m_aPos.Y != nCurrPos ) | |||
1924 | { | |||
1925 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
1926 | nLastRowColPixelPos += rRowColumnsWindowData[nIndex].nStaticSize; | |||
1927 | else | |||
1928 | nLastRowColPixelPos -= rRowColumnsWindowData[nIndex].nStaticSize; | |||
1929 | ++nIndex; | |||
1930 | nLastPos = 0; | |||
1931 | nCurrPos = rElement.m_aDockedData.m_aPos.Y; | |||
1932 | SingleRowColumnWindowData aRowColumnWindowData; | |||
1933 | aRowColumnWindowData.nRowColumn = nCurrPos; | |||
1934 | rRowColumnsWindowData.push_back( aRowColumnWindowData ); | |||
1935 | } | |||
1936 | ||||
1937 | // Calc space before an element and store it | |||
1938 | nSpace = ( rElement.m_aDockedData.m_aPos.X - nLastPos ); | |||
1939 | if ( rElement.m_aDockedData.m_aPos.X >= nLastPos ) | |||
1940 | { | |||
1941 | rRowColumnsWindowData[nIndex].nSpace += nSpace; | |||
1942 | nLastPos = rElement.m_aDockedData.m_aPos.X + aPosSize.Width; | |||
1943 | } | |||
1944 | else | |||
1945 | { | |||
1946 | nSpace = 0; | |||
1947 | nLastPos += aPosSize.Width; | |||
1948 | } | |||
1949 | rRowColumnsWindowData[nIndex].aRowColumnSpace.push_back( nSpace ); | |||
1950 | ||||
1951 | rRowColumnsWindowData[nIndex].aRowColumnWindows.push_back( xWindow ); | |||
1952 | rRowColumnsWindowData[nIndex].aUIElementNames.push_back( rElement.m_aName ); | |||
1953 | rRowColumnsWindowData[nIndex].aRowColumnWindowSizes.emplace_back( | |||
1954 | rElement.m_aDockedData.m_aPos.X, | |||
1955 | rElement.m_aDockedData.m_aPos.Y, | |||
1956 | aPosSize.Width, | |||
1957 | aPosSize.Height ); | |||
1958 | if ( rRowColumnsWindowData[nIndex].nStaticSize < aPosSize.Height ) | |||
1959 | rRowColumnsWindowData[nIndex].nStaticSize = aPosSize.Height; | |||
1960 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
1961 | rRowColumnsWindowData[nIndex].aRowColumnRect = awt::Rectangle( 0, nLastRowColPixelPos, | |||
1962 | aDockAreaRect.Width, aPosSize.Height ); | |||
1963 | else | |||
1964 | rRowColumnsWindowData[nIndex].aRowColumnRect = awt::Rectangle( 0, ( nLastRowColPixelPos - aPosSize.Height ), | |||
1965 | aDockAreaRect.Width, aPosSize.Height ); | |||
1966 | rRowColumnsWindowData[nIndex].nVarSize += aPosSize.Width + nSpace; | |||
1967 | } | |||
1968 | else | |||
1969 | { | |||
1970 | if ( nCurrPos == -1 ) | |||
1971 | { | |||
1972 | nCurrPos = rElement.m_aDockedData.m_aPos.X; | |||
1973 | nLastPos = 0; | |||
1974 | ||||
1975 | SingleRowColumnWindowData aRowColumnWindowData; | |||
1976 | aRowColumnWindowData.nRowColumn = nCurrPos; | |||
1977 | rRowColumnsWindowData.push_back( aRowColumnWindowData ); | |||
1978 | } | |||
1979 | ||||
1980 | sal_Int32 nSpace( 0 ); | |||
1981 | if ( rElement.m_aDockedData.m_aPos.X != nCurrPos ) | |||
1982 | { | |||
1983 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
1984 | nLastRowColPixelPos += rRowColumnsWindowData[nIndex].nStaticSize; | |||
1985 | else | |||
1986 | nLastRowColPixelPos -= rRowColumnsWindowData[nIndex].nStaticSize; | |||
1987 | ++nIndex; | |||
1988 | nLastPos = 0; | |||
1989 | nCurrPos = rElement.m_aDockedData.m_aPos.X; | |||
1990 | SingleRowColumnWindowData aRowColumnWindowData; | |||
1991 | aRowColumnWindowData.nRowColumn = nCurrPos; | |||
1992 | rRowColumnsWindowData.push_back( aRowColumnWindowData ); | |||
1993 | } | |||
1994 | ||||
1995 | // Calc space before an element and store it | |||
1996 | nSpace = ( rElement.m_aDockedData.m_aPos.Y - nLastPos ); | |||
1997 | if ( rElement.m_aDockedData.m_aPos.Y > nLastPos ) | |||
1998 | { | |||
1999 | rRowColumnsWindowData[nIndex].nSpace += nSpace; | |||
2000 | nLastPos = rElement.m_aDockedData.m_aPos.Y + aPosSize.Height; | |||
2001 | } | |||
2002 | else | |||
2003 | { | |||
2004 | nSpace = 0; | |||
2005 | nLastPos += aPosSize.Height; | |||
2006 | } | |||
2007 | rRowColumnsWindowData[nIndex].aRowColumnSpace.push_back( nSpace ); | |||
2008 | ||||
2009 | rRowColumnsWindowData[nIndex].aRowColumnWindows.push_back( xWindow ); | |||
2010 | rRowColumnsWindowData[nIndex].aUIElementNames.push_back( rElement.m_aName ); | |||
2011 | rRowColumnsWindowData[nIndex].aRowColumnWindowSizes.emplace_back( | |||
2012 | rElement.m_aDockedData.m_aPos.X, | |||
2013 | rElement.m_aDockedData.m_aPos.Y, | |||
2014 | aPosSize.Width, | |||
2015 | aPosSize.Height ); | |||
2016 | if ( rRowColumnsWindowData[nIndex].nStaticSize < aPosSize.Width ) | |||
2017 | rRowColumnsWindowData[nIndex].nStaticSize = aPosSize.Width; | |||
2018 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
2019 | rRowColumnsWindowData[nIndex].aRowColumnRect = awt::Rectangle( nLastRowColPixelPos, 0, | |||
2020 | aPosSize.Width, aDockAreaRect.Height ); | |||
2021 | else | |||
2022 | rRowColumnsWindowData[nIndex].aRowColumnRect = awt::Rectangle( ( nLastRowColPixelPos - aPosSize.Width ), 0, | |||
2023 | aPosSize.Width, aDockAreaRect.Height ); | |||
2024 | rRowColumnsWindowData[nIndex].nVarSize += aPosSize.Height + nSpace; | |||
2025 | } | |||
2026 | } | |||
2027 | } | |||
2028 | ||||
2029 | void ToolbarLayoutManager::implts_getDockingAreaElementInfoOnSingleRowCol( ui::DockingArea eDockingArea, sal_Int32 nRowCol, SingleRowColumnWindowData& rRowColumnWindowData ) | |||
2030 | { | |||
2031 | std::vector< UIElement > aWindowVector; | |||
2032 | ||||
2033 | if (( eDockingArea < ui::DockingArea_DOCKINGAREA_TOP ) || ( eDockingArea > ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
2034 | eDockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
2035 | ||||
2036 | bool bHorzDockArea = isHorizontalDockingArea( eDockingArea ); | |||
2037 | ||||
2038 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
2039 | { | |||
2040 | SolarMutexGuard aReadLock; | |||
2041 | for (auto const& elem : m_aUIElements) | |||
2042 | { | |||
2043 | if (elem.m_aDockedData.m_nDockedArea == eDockingArea) | |||
2044 | { | |||
2045 | bool bSameRowCol = bHorzDockArea ? (elem.m_aDockedData.m_aPos.Y == nRowCol) | |||
2046 | : (elem.m_aDockedData.m_aPos.X == nRowCol); | |||
2047 | uno::Reference<ui::XUIElement> xUIElement(elem.m_xUIElement); | |||
2048 | ||||
2049 | if (bSameRowCol && xUIElement.is()) | |||
2050 | { | |||
2051 | uno::Reference<awt::XWindow> xWindow(xUIElement->getRealInterface(), | |||
2052 | uno::UNO_QUERY); | |||
2053 | if (xWindow.is()) | |||
2054 | { | |||
2055 | SolarMutexGuard aGuard; | |||
2056 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow(xWindow); | |||
2057 | uno::Reference<awt::XDockableWindow> xDockWindow(xWindow, uno::UNO_QUERY); | |||
2058 | if (pWindow && elem.m_bVisible && xDockWindow.is() && !elem.m_bFloating) | |||
2059 | aWindowVector.push_back(elem); // docked windows | |||
2060 | } | |||
2061 | } | |||
2062 | } | |||
2063 | } | |||
2064 | } | |||
2065 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
2066 | ||||
2067 | // Initialize structure | |||
2068 | rRowColumnWindowData.aUIElementNames.clear(); | |||
2069 | rRowColumnWindowData.aRowColumnWindows.clear(); | |||
2070 | rRowColumnWindowData.aRowColumnWindowSizes.clear(); | |||
2071 | rRowColumnWindowData.aRowColumnSpace.clear(); | |||
2072 | rRowColumnWindowData.nVarSize = 0; | |||
2073 | rRowColumnWindowData.nStaticSize = 0; | |||
2074 | rRowColumnWindowData.nSpace = 0; | |||
2075 | rRowColumnWindowData.nRowColumn = nRowCol; | |||
2076 | ||||
2077 | // Collect data from windows that are on the same row/column | |||
2078 | sal_Int32 j; | |||
2079 | sal_Int32 nLastPos( 0 ); | |||
2080 | ||||
2081 | const sal_uInt32 nCount = aWindowVector.size(); | |||
2082 | for ( j = 0; j < sal_Int32( nCount); j++ ) | |||
2083 | { | |||
2084 | const UIElement& rElement = aWindowVector[j]; | |||
2085 | uno::Reference< awt::XWindow > xWindow; | |||
2086 | uno::Reference< ui::XUIElement > xUIElement( rElement.m_xUIElement ); | |||
2087 | awt::Rectangle aPosSize; | |||
2088 | if ( !lcl_checkUIElement(xUIElement,aPosSize,xWindow) ) | |||
2089 | continue; | |||
2090 | ||||
2091 | sal_Int32 nSpace; | |||
2092 | if ( isHorizontalDockingArea( eDockingArea )) | |||
2093 | { | |||
2094 | nSpace = ( rElement.m_aDockedData.m_aPos.X - nLastPos ); | |||
2095 | ||||
2096 | // Calc space before an element and store it | |||
2097 | if ( rElement.m_aDockedData.m_aPos.X > nLastPos ) | |||
2098 | rRowColumnWindowData.nSpace += nSpace; | |||
2099 | else | |||
2100 | nSpace = 0; | |||
2101 | ||||
2102 | nLastPos = rElement.m_aDockedData.m_aPos.X + aPosSize.Width; | |||
2103 | ||||
2104 | rRowColumnWindowData.aRowColumnWindowSizes.emplace_back( | |||
2105 | rElement.m_aDockedData.m_aPos.X, rElement.m_aDockedData.m_aPos.Y, | |||
2106 | aPosSize.Width, aPosSize.Height ); | |||
2107 | if ( rRowColumnWindowData.nStaticSize < aPosSize.Height ) | |||
2108 | rRowColumnWindowData.nStaticSize = aPosSize.Height; | |||
2109 | rRowColumnWindowData.nVarSize += aPosSize.Width; | |||
2110 | } | |||
2111 | else | |||
2112 | { | |||
2113 | // Calc space before an element and store it | |||
2114 | nSpace = ( rElement.m_aDockedData.m_aPos.Y - nLastPos ); | |||
2115 | if ( rElement.m_aDockedData.m_aPos.Y > nLastPos ) | |||
2116 | rRowColumnWindowData.nSpace += nSpace; | |||
2117 | else | |||
2118 | nSpace = 0; | |||
2119 | ||||
2120 | nLastPos = rElement.m_aDockedData.m_aPos.Y + aPosSize.Height; | |||
2121 | ||||
2122 | rRowColumnWindowData.aRowColumnWindowSizes.emplace_back( | |||
2123 | rElement.m_aDockedData.m_aPos.X, rElement.m_aDockedData.m_aPos.Y, | |||
2124 | aPosSize.Width, aPosSize.Height ); | |||
2125 | if ( rRowColumnWindowData.nStaticSize < aPosSize.Width ) | |||
2126 | rRowColumnWindowData.nStaticSize = aPosSize.Width; | |||
2127 | rRowColumnWindowData.nVarSize += aPosSize.Height; | |||
2128 | } | |||
2129 | ||||
2130 | rRowColumnWindowData.aUIElementNames.push_back( rElement.m_aName ); | |||
2131 | rRowColumnWindowData.aRowColumnWindows.push_back( xWindow ); | |||
2132 | rRowColumnWindowData.aRowColumnSpace.push_back( nSpace ); | |||
2133 | rRowColumnWindowData.nVarSize += nSpace; | |||
2134 | } | |||
2135 | } | |||
2136 | ||||
2137 | ::tools::Rectangle ToolbarLayoutManager::implts_getWindowRectFromRowColumn( | |||
2138 | ui::DockingArea DockingArea, | |||
2139 | const SingleRowColumnWindowData& rRowColumnWindowData, | |||
2140 | const ::Point& rMousePos, | |||
2141 | const OUString& rExcludeElementName ) | |||
2142 | { | |||
2143 | ::tools::Rectangle aWinRect; | |||
2144 | ||||
2145 | if (( DockingArea < ui::DockingArea_DOCKINGAREA_TOP ) || ( DockingArea > ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
2146 | DockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
2147 | ||||
2148 | if ( rRowColumnWindowData.aRowColumnWindows.empty() ) | |||
2149 | return aWinRect; | |||
2150 | else | |||
2151 | { | |||
2152 | SolarMutexClearableGuard aReadLock; | |||
2153 | VclPtr<vcl::Window> pContainerWindow( VCLUnoHelper::GetWindow( m_xContainerWindow )); | |||
2154 | VclPtr<vcl::Window> pDockingAreaWindow( VCLUnoHelper::GetWindow( m_xDockAreaWindows[static_cast<int>(DockingArea)] )); | |||
2155 | aReadLock.clear(); | |||
2156 | ||||
2157 | // Calc correct position of the column/row rectangle to be able to compare it with mouse pos/tracking rect | |||
2158 | SolarMutexGuard aGuard; | |||
2159 | ||||
2160 | // Retrieve output size from container Window | |||
2161 | if ( pDockingAreaWindow && pContainerWindow ) | |||
2162 | { | |||
2163 | const sal_uInt32 nCount = rRowColumnWindowData.aRowColumnWindows.size(); | |||
2164 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
2165 | { | |||
2166 | awt::Rectangle aWindowRect = rRowColumnWindowData.aRowColumnWindows[i]->getPosSize(); | |||
2167 | ::tools::Rectangle aRect( aWindowRect.X, aWindowRect.Y, aWindowRect.X+aWindowRect.Width, aWindowRect.Y+aWindowRect.Height ); | |||
2168 | aRect.SetPos( pContainerWindow->ScreenToOutputPixel( pDockingAreaWindow->OutputToScreenPixel( aRect.TopLeft() ))); | |||
2169 | if ( aRect.IsInside( rMousePos )) | |||
2170 | { | |||
2171 | // Check if we have found the excluded element. If yes, we have to provide an empty rectangle. | |||
2172 | // We prevent that a toolbar cannot be moved when the mouse pointer is inside its own rectangle! | |||
2173 | if ( rExcludeElementName != rRowColumnWindowData.aUIElementNames[i] ) | |||
2174 | return aRect; | |||
2175 | else | |||
2176 | break; | |||
2177 | } | |||
2178 | } | |||
2179 | } | |||
2180 | } | |||
2181 | ||||
2182 | return aWinRect; | |||
2183 | } | |||
2184 | ||||
2185 | ::tools::Rectangle ToolbarLayoutManager::implts_determineFrontDockingRect( | |||
2186 | ui::DockingArea eDockingArea, | |||
2187 | sal_Int32 nRowCol, | |||
2188 | const ::tools::Rectangle& rDockedElementRect, | |||
2189 | const OUString& rMovedElementName, | |||
2190 | const ::tools::Rectangle& rMovedElementRect ) | |||
2191 | { | |||
2192 | SingleRowColumnWindowData aRowColumnWindowData; | |||
2193 | ||||
2194 | bool bHorzDockArea( isHorizontalDockingArea( eDockingArea )); | |||
2195 | implts_getDockingAreaElementInfoOnSingleRowCol( eDockingArea, nRowCol, aRowColumnWindowData ); | |||
2196 | if ( aRowColumnWindowData.aRowColumnWindows.empty() ) | |||
2197 | return rMovedElementRect; | |||
2198 | else | |||
2199 | { | |||
2200 | sal_Int32 nSpace( 0 ); | |||
2201 | ::tools::Rectangle aFrontDockingRect( rMovedElementRect ); | |||
2202 | const sal_uInt32 nCount = aRowColumnWindowData.aRowColumnWindows.size(); | |||
2203 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
2204 | { | |||
2205 | if ( bHorzDockArea ) | |||
2206 | { | |||
2207 | if ( aRowColumnWindowData.aRowColumnWindowSizes[i].X >= rDockedElementRect.Left() ) | |||
2208 | { | |||
2209 | nSpace += aRowColumnWindowData.aRowColumnSpace[i]; | |||
2210 | break; | |||
2211 | } | |||
2212 | else if ( aRowColumnWindowData.aUIElementNames[i] == rMovedElementName ) | |||
2213 | nSpace += aRowColumnWindowData.aRowColumnWindowSizes[i].Width + | |||
2214 | aRowColumnWindowData.aRowColumnSpace[i]; | |||
2215 | else | |||
2216 | nSpace = 0; | |||
2217 | } | |||
2218 | else | |||
2219 | { | |||
2220 | if ( aRowColumnWindowData.aRowColumnWindowSizes[i].Y >= rDockedElementRect.Top() ) | |||
2221 | { | |||
2222 | nSpace += aRowColumnWindowData.aRowColumnSpace[i]; | |||
2223 | break; | |||
2224 | } | |||
2225 | else if ( aRowColumnWindowData.aUIElementNames[i] == rMovedElementName ) | |||
2226 | nSpace += aRowColumnWindowData.aRowColumnWindowSizes[i].Height + | |||
2227 | aRowColumnWindowData.aRowColumnSpace[i]; | |||
2228 | else | |||
2229 | nSpace = 0; | |||
2230 | } | |||
2231 | } | |||
2232 | ||||
2233 | if ( nSpace > 0 ) | |||
2234 | { | |||
2235 | sal_Int32 nMove = std::min( nSpace, static_cast<sal_Int32>(aFrontDockingRect.getWidth()) ); | |||
2236 | if ( bHorzDockArea ) | |||
2237 | aFrontDockingRect.Move( -nMove, 0 ); | |||
2238 | else | |||
2239 | aFrontDockingRect.Move( 0, -nMove ); | |||
2240 | } | |||
2241 | ||||
2242 | return aFrontDockingRect; | |||
2243 | } | |||
2244 | } | |||
2245 | ||||
2246 | void ToolbarLayoutManager::implts_findNextDockingPos( ui::DockingArea DockingArea, const ::Size& aUIElementSize, awt::Point& rVirtualPos, ::Point& rPixelPos ) | |||
2247 | { | |||
2248 | SolarMutexClearableGuard aReadLock; | |||
2249 | if (( DockingArea < ui::DockingArea_DOCKINGAREA_TOP ) || ( DockingArea > ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
2250 | DockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
2251 | uno::Reference< awt::XWindow > xDockingWindow( m_xDockAreaWindows[static_cast<int>(DockingArea)] ); | |||
2252 | ::Size aDockingWinSize; | |||
2253 | ||||
2254 | // Retrieve output size from container Window | |||
2255 | vcl::Window* pDockingWindow = VCLUnoHelper::GetWindow( xDockingWindow ).get(); | |||
2256 | if ( pDockingWindow ) | |||
2257 | aDockingWinSize = pDockingWindow->GetOutputSizePixel(); | |||
2258 | aReadLock.clear(); | |||
2259 | ||||
2260 | sal_Int32 nFreeRowColPixelPos( 0 ); | |||
2261 | sal_Int32 nMaxSpace( 0 ); | |||
2262 | sal_Int32 nNeededSpace( 0 ); | |||
2263 | sal_Int32 nTopDockingAreaSize( 0 ); | |||
2264 | ||||
2265 | if ( isHorizontalDockingArea( DockingArea )) | |||
2266 | { | |||
2267 | nMaxSpace = aDockingWinSize.Width(); | |||
2268 | nNeededSpace = aUIElementSize.Width(); | |||
2269 | } | |||
2270 | else | |||
2271 | { | |||
2272 | nMaxSpace = aDockingWinSize.Height(); | |||
2273 | nNeededSpace = aUIElementSize.Height(); | |||
2274 | nTopDockingAreaSize = implts_getTopBottomDockingAreaSizes().Width(); | |||
2275 | } | |||
2276 | ||||
2277 | std::vector< SingleRowColumnWindowData > aRowColumnsWindowData; | |||
2278 | ||||
2279 | implts_getDockingAreaElementInfos( DockingArea, aRowColumnsWindowData ); | |||
2280 | sal_Int32 nPixelPos( 0 ); | |||
2281 | const sal_uInt32 nCount = aRowColumnsWindowData.size(); | |||
2282 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
2283 | { | |||
2284 | SingleRowColumnWindowData& rRowColumnWindowData = aRowColumnsWindowData[i]; | |||
2285 | ||||
2286 | if (( DockingArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) || | |||
2287 | ( DockingArea == ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
2288 | nPixelPos += rRowColumnWindowData.nStaticSize; | |||
2289 | ||||
2290 | if ((( nMaxSpace - rRowColumnWindowData.nVarSize ) >= nNeededSpace ) || | |||
2291 | ( rRowColumnWindowData.nSpace >= nNeededSpace )) | |||
2292 | { | |||
2293 | // Check current row where we can find the needed space | |||
2294 | sal_Int32 nCurrPos( 0 ); | |||
2295 | const sal_uInt32 nWindowSizesCount = rRowColumnWindowData.aRowColumnWindowSizes.size(); | |||
2296 | for ( sal_uInt32 j = 0; j < nWindowSizesCount; j++ ) | |||
2297 | { | |||
2298 | awt::Rectangle rRect = rRowColumnWindowData.aRowColumnWindowSizes[j]; | |||
2299 | sal_Int32& rSpace = rRowColumnWindowData.aRowColumnSpace[j]; | |||
2300 | if ( isHorizontalDockingArea( DockingArea )) | |||
2301 | { | |||
2302 | if ( rSpace >= nNeededSpace ) | |||
2303 | { | |||
2304 | rVirtualPos = awt::Point( nCurrPos, rRowColumnWindowData.nRowColumn ); | |||
2305 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
2306 | rPixelPos = ::Point( nCurrPos, nPixelPos ); | |||
2307 | else | |||
2308 | rPixelPos = ::Point( nCurrPos, aDockingWinSize.Height() - nPixelPos ); | |||
2309 | return; | |||
2310 | } | |||
2311 | nCurrPos = rRect.X + rRect.Width; | |||
2312 | } | |||
2313 | else | |||
2314 | { | |||
2315 | if ( rSpace >= nNeededSpace ) | |||
2316 | { | |||
2317 | rVirtualPos = awt::Point( rRowColumnWindowData.nRowColumn, nCurrPos ); | |||
2318 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
2319 | rPixelPos = ::Point( nPixelPos, nTopDockingAreaSize + nCurrPos ); | |||
2320 | else | |||
2321 | rPixelPos = ::Point( aDockingWinSize.Width() - nPixelPos , nTopDockingAreaSize + nCurrPos ); | |||
2322 | return; | |||
2323 | } | |||
2324 | nCurrPos = rRect.Y + rRect.Height; | |||
2325 | } | |||
2326 | } | |||
2327 | ||||
2328 | if (( nCurrPos + nNeededSpace ) <= nMaxSpace ) | |||
2329 | { | |||
2330 | if ( isHorizontalDockingArea( DockingArea )) | |||
2331 | { | |||
2332 | rVirtualPos = awt::Point( nCurrPos, rRowColumnWindowData.nRowColumn ); | |||
2333 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
2334 | rPixelPos = ::Point( nCurrPos, nPixelPos ); | |||
2335 | else | |||
2336 | rPixelPos = ::Point( nCurrPos, aDockingWinSize.Height() - nPixelPos ); | |||
2337 | return; | |||
2338 | } | |||
2339 | else | |||
2340 | { | |||
2341 | rVirtualPos = awt::Point( rRowColumnWindowData.nRowColumn, nCurrPos ); | |||
2342 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
2343 | rPixelPos = ::Point( nPixelPos, nTopDockingAreaSize + nCurrPos ); | |||
2344 | else | |||
2345 | rPixelPos = ::Point( aDockingWinSize.Width() - nPixelPos , nTopDockingAreaSize + nCurrPos ); | |||
2346 | return; | |||
2347 | } | |||
2348 | } | |||
2349 | } | |||
2350 | ||||
2351 | if (( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) || ( DockingArea == ui::DockingArea_DOCKINGAREA_LEFT )) | |||
2352 | nPixelPos += rRowColumnWindowData.nStaticSize; | |||
2353 | } | |||
2354 | ||||
2355 | sal_Int32 nNextFreeRowCol( 0 ); | |||
2356 | sal_Int32 nRowColumnsCount = aRowColumnsWindowData.size(); | |||
2357 | if ( nRowColumnsCount > 0 ) | |||
2358 | nNextFreeRowCol = aRowColumnsWindowData[nRowColumnsCount-1].nRowColumn+1; | |||
2359 | else | |||
2360 | nNextFreeRowCol = 0; | |||
2361 | ||||
2362 | if ( nNextFreeRowCol == 0 ) | |||
2363 | { | |||
2364 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) | |||
2365 | nFreeRowColPixelPos = aDockingWinSize.Height() - aUIElementSize.Height(); | |||
2366 | else if ( DockingArea == ui::DockingArea_DOCKINGAREA_RIGHT ) | |||
2367 | nFreeRowColPixelPos = aDockingWinSize.Width() - aUIElementSize.Width(); | |||
2368 | } | |||
2369 | ||||
2370 | if ( isHorizontalDockingArea( DockingArea )) | |||
2371 | { | |||
2372 | rVirtualPos = awt::Point( 0, nNextFreeRowCol ); | |||
2373 | if ( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
2374 | rPixelPos = ::Point( 0, nFreeRowColPixelPos ); | |||
2375 | else | |||
2376 | rPixelPos = ::Point( 0, aDockingWinSize.Height() - nFreeRowColPixelPos ); | |||
2377 | } | |||
2378 | else | |||
2379 | { | |||
2380 | rVirtualPos = awt::Point( nNextFreeRowCol, 0 ); | |||
2381 | rPixelPos = ::Point( aDockingWinSize.Width() - nFreeRowColPixelPos, 0 ); | |||
2382 | } | |||
2383 | } | |||
2384 | ||||
2385 | void ToolbarLayoutManager::implts_calcWindowPosSizeOnSingleRowColumn( | |||
2386 | sal_Int32 nDockingArea, | |||
2387 | sal_Int32 nOffset, | |||
2388 | SingleRowColumnWindowData& rRowColumnWindowData, | |||
2389 | const ::Size& rContainerSize ) | |||
2390 | { | |||
2391 | sal_Int32 nDiff(0); | |||
2392 | sal_Int32 nRCSpace( rRowColumnWindowData.nSpace ); | |||
2393 | sal_Int32 nContainerClientSize(0); | |||
2394 | ||||
2395 | if ( rRowColumnWindowData.aRowColumnWindows.empty() ) | |||
2396 | return; | |||
2397 | ||||
2398 | if ( isHorizontalDockingArea( nDockingArea )) | |||
2399 | { | |||
2400 | nContainerClientSize = rContainerSize.Width(); | |||
2401 | nDiff = nContainerClientSize - rRowColumnWindowData.nVarSize; | |||
2402 | } | |||
2403 | else | |||
2404 | { | |||
2405 | sal_Int32 nTopDockingAreaSize = implts_getTopBottomDockingAreaSizes().Width(); | |||
2406 | sal_Int32 nBottomDockingAreaSize = implts_getTopBottomDockingAreaSizes().Height(); | |||
2407 | nContainerClientSize = ( rContainerSize.Height() - nTopDockingAreaSize - nBottomDockingAreaSize ); | |||
2408 | nDiff = nContainerClientSize - rRowColumnWindowData.nVarSize; | |||
2409 | } | |||
2410 | ||||
2411 | const sal_uInt32 nCount = rRowColumnWindowData.aRowColumnWindowSizes.size(); | |||
2412 | if (( nDiff < 0 ) && ( nRCSpace > 0 )) | |||
2413 | { | |||
2414 | // First we try to reduce the size of blank space before/behind docked windows | |||
2415 | sal_Int32 i = nCount - 1; | |||
2416 | while ( i >= 0 ) | |||
2417 | { | |||
2418 | sal_Int32 nSpace = rRowColumnWindowData.aRowColumnSpace[i]; | |||
2419 | if ( nSpace >= -nDiff ) | |||
2420 | { | |||
2421 | if ( isHorizontalDockingArea( nDockingArea )) | |||
2422 | { | |||
2423 | // Try to move this and all user elements behind with the calculated difference | |||
2424 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2425 | rRowColumnWindowData.aRowColumnWindowSizes[j].X += nDiff; | |||
2426 | } | |||
2427 | else | |||
2428 | { | |||
2429 | // Try to move this and all user elements behind with the calculated difference | |||
2430 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2431 | rRowColumnWindowData.aRowColumnWindowSizes[j].Y += nDiff; | |||
2432 | } | |||
2433 | nDiff = 0; | |||
2434 | ||||
2435 | break; | |||
2436 | } | |||
2437 | else if ( nSpace > 0 ) | |||
2438 | { | |||
2439 | if ( isHorizontalDockingArea( nDockingArea )) | |||
2440 | { | |||
2441 | // Try to move this and all user elements behind with the calculated difference | |||
2442 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2443 | rRowColumnWindowData.aRowColumnWindowSizes[j].X -= nSpace; | |||
2444 | } | |||
2445 | else | |||
2446 | { | |||
2447 | // Try to move this and all user elements behind with the calculated difference | |||
2448 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2449 | rRowColumnWindowData.aRowColumnWindowSizes[j].Y -= nSpace; | |||
2450 | } | |||
2451 | nDiff += nSpace; | |||
2452 | } | |||
2453 | --i; | |||
2454 | } | |||
2455 | } | |||
2456 | ||||
2457 | // Check if we have to reduce further | |||
2458 | if ( nDiff < 0 ) | |||
2459 | { | |||
2460 | // Now we have to reduce the size of certain docked windows | |||
2461 | sal_Int32 i = sal_Int32( nCount - 1 ); | |||
2462 | while ( i >= 0 ) | |||
2463 | { | |||
2464 | awt::Rectangle& rWinRect = rRowColumnWindowData.aRowColumnWindowSizes[i]; | |||
2465 | ::Size aMinSize; | |||
2466 | ||||
2467 | SolarMutexGuard aGuard; | |||
2468 | { | |||
2469 | uno::Reference< awt::XWindow > xWindow = rRowColumnWindowData.aRowColumnWindows[i]; | |||
2470 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
2471 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
2472 | aMinSize = static_cast<ToolBox *>(pWindow.get())->CalcMinimumWindowSizePixel(); | |||
2473 | } | |||
2474 | ||||
2475 | if ( !aMinSize.IsEmpty() ) | |||
2476 | { | |||
2477 | if ( isHorizontalDockingArea( nDockingArea )) | |||
2478 | { | |||
2479 | sal_Int32 nMaxReducation = rWinRect.Width - aMinSize.Width(); | |||
2480 | if ( nMaxReducation >= -nDiff ) | |||
2481 | { | |||
2482 | rWinRect.Width = rWinRect.Width + nDiff; | |||
2483 | nDiff = 0; | |||
2484 | } | |||
2485 | else | |||
2486 | { | |||
2487 | rWinRect.Width = aMinSize.Width(); | |||
2488 | nDiff += nMaxReducation; | |||
2489 | } | |||
2490 | ||||
2491 | // Try to move this and all user elements behind with the calculated difference | |||
2492 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2493 | rRowColumnWindowData.aRowColumnWindowSizes[j].X += nDiff; | |||
2494 | } | |||
2495 | else | |||
2496 | { | |||
2497 | sal_Int32 nMaxReducation = rWinRect.Height - aMinSize.Height(); | |||
2498 | if ( nMaxReducation >= -nDiff ) | |||
2499 | { | |||
2500 | rWinRect.Height = rWinRect.Height + nDiff; | |||
2501 | nDiff = 0; | |||
2502 | } | |||
2503 | else | |||
2504 | { | |||
2505 | rWinRect.Height = aMinSize.Height(); | |||
2506 | nDiff += nMaxReducation; | |||
2507 | } | |||
2508 | ||||
2509 | // Try to move this and all user elements behind with the calculated difference | |||
2510 | for ( sal_uInt32 j = i; j < nCount; j++ ) | |||
2511 | rRowColumnWindowData.aRowColumnWindowSizes[j].Y += nDiff; | |||
2512 | } | |||
2513 | } | |||
2514 | ||||
2515 | if ( nDiff >= 0 ) | |||
2516 | break; | |||
2517 | ||||
2518 | --i; | |||
2519 | } | |||
2520 | } | |||
2521 | ||||
2522 | SolarMutexClearableGuard aReadLock; | |||
2523 | VclPtr<vcl::Window> pDockAreaWindow = VCLUnoHelper::GetWindow( m_xDockAreaWindows[nDockingArea] ); | |||
2524 | aReadLock.clear(); | |||
2525 | ||||
2526 | sal_Int32 nCurrPos( 0 ); | |||
2527 | ||||
2528 | SolarMutexGuard aGuard; | |||
2529 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
2530 | { | |||
2531 | uno::Reference< awt::XWindow > xWindow = rRowColumnWindowData.aRowColumnWindows[i]; | |||
2532 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
2533 | vcl::Window* pOldParentWindow = pWindow->GetParent(); | |||
2534 | ||||
2535 | if ( pDockAreaWindow != pOldParentWindow ) | |||
2536 | pWindow->SetParent( pDockAreaWindow ); | |||
2537 | ||||
2538 | awt::Rectangle aWinRect = rRowColumnWindowData.aRowColumnWindowSizes[i]; | |||
2539 | if ( isHorizontalDockingArea( nDockingArea )) | |||
2540 | { | |||
2541 | if ( aWinRect.X < nCurrPos ) | |||
2542 | aWinRect.X = nCurrPos; | |||
2543 | pWindow->SetPosSizePixel( ::Point( aWinRect.X, nOffset ), ::Size( aWinRect.Width, rRowColumnWindowData.nStaticSize )); | |||
2544 | pWindow->Show( true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
2545 | nCurrPos += ( aWinRect.X - nCurrPos ) + aWinRect.Width; | |||
2546 | } | |||
2547 | else | |||
2548 | { | |||
2549 | if ( aWinRect.Y < nCurrPos ) | |||
2550 | aWinRect.Y = nCurrPos; | |||
2551 | pWindow->SetPosSizePixel( ::Point( nOffset, aWinRect.Y ), ::Size( rRowColumnWindowData.nStaticSize, aWinRect.Height )); | |||
2552 | pWindow->Show( true, ShowFlags::NoFocusChange | ShowFlags::NoActivate ); | |||
2553 | nCurrPos += ( aWinRect.Y - nCurrPos ) + aWinRect.Height; | |||
2554 | } | |||
2555 | } | |||
2556 | } | |||
2557 | ||||
2558 | void ToolbarLayoutManager::implts_setLayoutDirty() | |||
2559 | { | |||
2560 | SolarMutexGuard g; | |||
2561 | m_bLayoutDirty = true; | |||
2562 | } | |||
2563 | ||||
2564 | void ToolbarLayoutManager::implts_setLayoutInProgress( bool bInProgress ) | |||
2565 | { | |||
2566 | SolarMutexGuard g; | |||
2567 | m_bLayoutInProgress = bInProgress; | |||
2568 | } | |||
2569 | ||||
2570 | ::tools::Rectangle ToolbarLayoutManager::implts_calcHotZoneRect( const ::tools::Rectangle& rRect, sal_Int32 nHotZoneOffset ) | |||
2571 | { | |||
2572 | ::tools::Rectangle aRect( rRect ); | |||
2573 | ||||
2574 | aRect.AdjustLeft( -nHotZoneOffset ); | |||
2575 | aRect.AdjustTop( -nHotZoneOffset ); | |||
2576 | aRect.AdjustRight(nHotZoneOffset ); | |||
2577 | aRect.AdjustBottom(nHotZoneOffset ); | |||
2578 | ||||
2579 | return aRect; | |||
2580 | } | |||
2581 | ||||
2582 | void ToolbarLayoutManager::implts_calcDockingPosSize( | |||
2583 | UIElement& rUIElement, | |||
2584 | DockingOperation& rDockingOperation, | |||
2585 | ::tools::Rectangle& rTrackingRect, | |||
2586 | const Point& rMousePos ) | |||
2587 | { | |||
2588 | SolarMutexResettableGuard aReadLock; | |||
2589 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
2590 | ::Size aContainerWinSize; | |||
2591 | vcl::Window* pContainerWindow( nullptr ); | |||
2592 | ::tools::Rectangle aDockingAreaOffsets( m_aDockingAreaOffsets ); | |||
2593 | aReadLock.clear(); | |||
2594 | ||||
2595 | if ( !rUIElement.m_xUIElement.is() ) | |||
| ||||
2596 | { | |||
2597 | rTrackingRect = ::tools::Rectangle(); | |||
2598 | return; | |||
2599 | } | |||
2600 | ||||
2601 | { | |||
2602 | // Retrieve output size from container Window | |||
2603 | SolarMutexGuard aGuard; | |||
2604 | pContainerWindow = VCLUnoHelper::GetWindow( xContainerWindow ).get(); | |||
2605 | aContainerWinSize = pContainerWindow->GetOutputSizePixel(); | |||
2606 | } | |||
2607 | ||||
2608 | vcl::Window* pDockingAreaWindow( nullptr ); | |||
2609 | uno::Reference< awt::XWindow > xWindow( rUIElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
2610 | uno::Reference< awt::XWindow > xDockingAreaWindow; | |||
2611 | ::tools::Rectangle aTrackingRect( rTrackingRect ); | |||
2612 | ui::DockingArea eDockedArea( rUIElement.m_aDockedData.m_nDockedArea ); | |||
2613 | sal_Int32 nTopDockingAreaSize( implts_getTopBottomDockingAreaSizes().Width() ); | |||
2614 | sal_Int32 nBottomDockingAreaSize( implts_getTopBottomDockingAreaSizes().Height() ); | |||
2615 | bool bHorizontalDockArea(( eDockedArea == ui::DockingArea_DOCKINGAREA_TOP ) || | |||
2616 | ( eDockedArea == ui::DockingArea_DOCKINGAREA_BOTTOM )); | |||
2617 | sal_Int32 nMaxLeftRightDockAreaSize = aContainerWinSize.Height() - | |||
2618 | nTopDockingAreaSize - | |||
2619 | nBottomDockingAreaSize - | |||
2620 | aDockingAreaOffsets.Top() - | |||
2621 | aDockingAreaOffsets.Bottom(); | |||
2622 | ::tools::Rectangle aDockingAreaRect; | |||
2623 | ||||
2624 | aReadLock.reset(); | |||
2625 | xDockingAreaWindow = m_xDockAreaWindows[static_cast<int>(eDockedArea)]; | |||
2626 | aReadLock.clear(); | |||
2627 | ||||
2628 | { | |||
2629 | SolarMutexGuard aGuard; | |||
2630 | pDockingAreaWindow = VCLUnoHelper::GetWindow( xDockingAreaWindow ).get(); | |||
2631 | VclPtr<vcl::Window> pDockWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
2632 | ToolBox* pToolBox( nullptr ); | |||
2633 | if ( pDockWindow && pDockWindow->GetType() == WindowType::TOOLBOX ) | |||
2634 | pToolBox = static_cast<ToolBox *>(pDockWindow.get()); | |||
2635 | ||||
2636 | aDockingAreaRect = ::tools::Rectangle( pDockingAreaWindow->GetPosPixel(), pDockingAreaWindow->GetSizePixel() ); | |||
| ||||
2637 | if ( pToolBox ) | |||
2638 | { | |||
2639 | // docked toolbars always have one line | |||
2640 | ::Size aSize = pToolBox->CalcWindowSizePixel( 1, ImplConvertAlignment( eDockedArea ) ); | |||
2641 | aTrackingRect.SetSize( ::Size( aSize.Width(), aSize.Height() )); | |||
2642 | } | |||
2643 | } | |||
2644 | ||||
2645 | // default docking operation, dock on the given row/column | |||
2646 | bool bOpOutsideOfDockingArea( !aDockingAreaRect.IsInside( rMousePos )); | |||
2647 | ||||
2648 | std::vector< SingleRowColumnWindowData > aRowColumnsWindowData; | |||
2649 | ||||
2650 | rDockingOperation = DOCKOP_ON_COLROW; | |||
2651 | implts_getDockingAreaElementInfos( eDockedArea, aRowColumnsWindowData ); | |||
2652 | ||||
2653 | // determine current first row/column and last row/column | |||
2654 | sal_Int32 nMaxRowCol( -1 ); | |||
2655 | sal_Int32 nMinRowCol( SAL_MAX_INT32((sal_Int32) 0x7FFFFFFF) ); | |||
2656 | const sal_uInt32 nCount = aRowColumnsWindowData.size(); | |||
2657 | for ( sal_uInt32 i = 0; i < nCount; i++ ) | |||
2658 | { | |||
2659 | if ( aRowColumnsWindowData[i].nRowColumn > nMaxRowCol ) | |||
2660 | nMaxRowCol = aRowColumnsWindowData[i].nRowColumn; | |||
2661 | if ( aRowColumnsWindowData[i].nRowColumn < nMinRowCol ) | |||
2662 | nMinRowCol = aRowColumnsWindowData[i].nRowColumn; | |||
2663 | } | |||
2664 | ||||
2665 | if ( !bOpOutsideOfDockingArea ) | |||
2666 | { | |||
2667 | // docking inside our docking area | |||
2668 | sal_Int32 nIndex( -1 ); | |||
2669 | sal_Int32 nRowCol( -1 ); | |||
2670 | ::tools::Rectangle aWindowRect; | |||
2671 | ::tools::Rectangle aRowColumnRect; | |||
2672 | ||||
2673 | const sal_uInt32 nWindowDataCount = aRowColumnsWindowData.size(); | |||
2674 | for ( sal_uInt32 i = 0; i < nWindowDataCount; i++ ) | |||
2675 | { | |||
2676 | ::tools::Rectangle aRect( aRowColumnsWindowData[i].aRowColumnRect.X, | |||
2677 | aRowColumnsWindowData[i].aRowColumnRect.Y, | |||
2678 | aRowColumnsWindowData[i].aRowColumnRect.X + aRowColumnsWindowData[i].aRowColumnRect.Width, | |||
2679 | aRowColumnsWindowData[i].aRowColumnRect.Y + aRowColumnsWindowData[i].aRowColumnRect.Height ); | |||
2680 | ||||
2681 | { | |||
2682 | // Calc correct position of the column/row rectangle to be able to compare it with mouse pos/tracking rect | |||
2683 | SolarMutexGuard aGuard; | |||
2684 | aRect.SetPos( pContainerWindow->ScreenToOutputPixel( pDockingAreaWindow->OutputToScreenPixel( aRect.TopLeft() ))); | |||
2685 | } | |||
2686 | ||||
2687 | bool bIsInsideRowCol( aRect.IsInside( rMousePos ) ); | |||
2688 | if ( bIsInsideRowCol ) | |||
2689 | { | |||
2690 | nIndex = i; | |||
2691 | nRowCol = aRowColumnsWindowData[i].nRowColumn; | |||
2692 | rDockingOperation = implts_determineDockingOperation( eDockedArea, aRect, rMousePos ); | |||
2693 | aWindowRect = implts_getWindowRectFromRowColumn( eDockedArea, aRowColumnsWindowData[i], rMousePos, rUIElement.m_aName ); | |||
2694 | aRowColumnRect = aRect; | |||
2695 | break; | |||
2696 | } | |||
2697 | } | |||
2698 | ||||
2699 | OSL_ENSURE( ( nIndex >= 0 ) && ( nRowCol >= 0 ), "Impossible case - no row/column found but mouse pointer is inside our docking area" )do { if (true && (!(( nIndex >= 0 ) && ( nRowCol >= 0 )))) { sal_detail_logFormat((SAL_DETAIL_LOG_LEVEL_WARN ), ("legacy.osl"), ("/home/maarten/src/libreoffice/core/framework/source/layoutmanager/toolbarlayoutmanager.cxx" ":" "2699" ": "), "%s", "Impossible case - no row/column found but mouse pointer is inside our docking area" ); } } while (false); | |||
2700 | if (( nIndex >= 0 ) && ( nRowCol >= 0 )) | |||
2701 | { | |||
2702 | if ( rDockingOperation == DOCKOP_ON_COLROW ) | |||
2703 | { | |||
2704 | if ( !aWindowRect.IsEmpty()) | |||
2705 | { | |||
2706 | // Tracking rect is on a row/column and mouse is over a docked toolbar. | |||
2707 | // Determine if the tracking rect must be located before/after the docked toolbar. | |||
2708 | ||||
2709 | ::tools::Rectangle aUIElementRect( aWindowRect ); | |||
2710 | sal_Int32 nMiddle( bHorizontalDockArea ? ( aWindowRect.Left() + aWindowRect.getWidth() / 2 ) : | |||
2711 | ( aWindowRect.Top() + aWindowRect.getHeight() / 2 )); | |||
2712 | bool bInsertBefore( bHorizontalDockArea ? ( rMousePos.X() < nMiddle ) : ( rMousePos.Y() < nMiddle )); | |||
2713 | if ( bInsertBefore ) | |||
2714 | { | |||
2715 | if ( bHorizontalDockArea ) | |||
2716 | { | |||
2717 | sal_Int32 nSize = ::std::max( sal_Int32( 0 ), std::min( sal_Int32( aContainerWinSize.Width() - aWindowRect.Left() ), | |||
2718 | sal_Int32( aTrackingRect.getWidth() ))); | |||
2719 | if ( nSize == 0 ) | |||
2720 | nSize = aWindowRect.getWidth(); | |||
2721 | ||||
2722 | aUIElementRect.SetSize( ::Size( nSize, aWindowRect.getHeight() )); | |||
2723 | aWindowRect = implts_determineFrontDockingRect( eDockedArea, nRowCol, aWindowRect,rUIElement.m_aName, aUIElementRect ); | |||
2724 | ||||
2725 | // Set virtual position | |||
2726 | rUIElement.m_aDockedData.m_aPos.X = aWindowRect.Left(); | |||
2727 | rUIElement.m_aDockedData.m_aPos.Y = nRowCol; | |||
2728 | } | |||
2729 | else | |||
2730 | { | |||
2731 | sal_Int32 nSize = ::std::max( sal_Int32( 0 ), std::min( sal_Int32( | |||
2732 | nTopDockingAreaSize + nMaxLeftRightDockAreaSize - aWindowRect.Top() ), | |||
2733 | sal_Int32( aTrackingRect.getHeight() ))); | |||
2734 | if ( nSize == 0 ) | |||
2735 | nSize = aWindowRect.getHeight(); | |||
2736 | ||||
2737 | aUIElementRect.SetSize( ::Size( aWindowRect.getWidth(), nSize )); | |||
2738 | aWindowRect = implts_determineFrontDockingRect( eDockedArea, nRowCol, aWindowRect, rUIElement.m_aName, aUIElementRect ); | |||
2739 | ||||
2740 | // Set virtual position | |||
2741 | sal_Int32 nPosY = pDockingAreaWindow->ScreenToOutputPixel( | |||
2742 | pContainerWindow->OutputToScreenPixel( aWindowRect.TopLeft() )).Y(); | |||
2743 | rUIElement.m_aDockedData.m_aPos.X = nRowCol; | |||
2744 | rUIElement.m_aDockedData.m_aPos.Y = nPosY; | |||
2745 | } | |||
2746 | ||||
2747 | rTrackingRect = aWindowRect; | |||
2748 | return; | |||
2749 | } | |||
2750 | else | |||
2751 | { | |||
2752 | if ( bHorizontalDockArea ) | |||
2753 | { | |||
2754 | sal_Int32 nSize = ::std::max( sal_Int32( 0 ), std::min( sal_Int32(( aContainerWinSize.Width() ) - aWindowRect.Right() ), | |||
2755 | sal_Int32( aTrackingRect.getWidth() ))); | |||
2756 | if ( nSize == 0 ) | |||
2757 | { | |||
2758 | aUIElementRect.SetPos( ::Point( aContainerWinSize.Width() - aTrackingRect.getWidth(), aWindowRect.Top() )); | |||
2759 | aUIElementRect.SetSize( ::Size( aTrackingRect.getWidth(), aWindowRect.getHeight() )); | |||
2760 | rUIElement.m_aDockedData.m_aPos.X = aUIElementRect.Left(); | |||
2761 | ||||
2762 | } | |||
2763 | else | |||
2764 | { | |||
2765 | aUIElementRect.SetPos( ::Point( aWindowRect.Right(), aWindowRect.Top() )); | |||
2766 | aUIElementRect.SetSize( ::Size( nSize, aWindowRect.getHeight() )); | |||
2767 | rUIElement.m_aDockedData.m_aPos.X = aWindowRect.Right(); | |||
2768 | } | |||
2769 | ||||
2770 | // Set virtual position | |||
2771 | rUIElement.m_aDockedData.m_aPos.Y = nRowCol; | |||
2772 | } | |||
2773 | else | |||
2774 | { | |||
2775 | sal_Int32 nSize = ::std::max( sal_Int32( 0 ), std::min( sal_Int32( nTopDockingAreaSize + nMaxLeftRightDockAreaSize - aWindowRect.Bottom() ), | |||
2776 | sal_Int32( aTrackingRect.getHeight() ))); | |||
2777 | aUIElementRect.SetPos( ::Point( aWindowRect.Left(), aWindowRect.Bottom() )); | |||
2778 | aUIElementRect.SetSize( ::Size( aWindowRect.getWidth(), nSize )); | |||
2779 | ||||
2780 | // Set virtual position | |||
2781 | sal_Int32 nPosY( 0 ); | |||
2782 | { | |||
2783 | SolarMutexGuard aGuard; | |||
2784 | nPosY = pDockingAreaWindow->ScreenToOutputPixel( | |||
2785 | pContainerWindow->OutputToScreenPixel( aWindowRect.BottomRight() )).Y(); | |||
2786 | } | |||
2787 | rUIElement.m_aDockedData.m_aPos.X = nRowCol; | |||
2788 | rUIElement.m_aDockedData.m_aPos.Y = nPosY; | |||
2789 | } | |||
2790 | ||||
2791 | rTrackingRect = aUIElementRect; | |||
2792 | return; | |||
2793 | } | |||
2794 | } | |||
2795 | else | |||
2796 | { | |||
2797 | implts_setTrackingRect( eDockedArea, rMousePos, aTrackingRect ); | |||
2798 | rTrackingRect = implts_calcTrackingAndElementRect( | |||
2799 | eDockedArea, nRowCol, rUIElement, | |||
2800 | aTrackingRect, aRowColumnRect, aContainerWinSize ); | |||
2801 | return; | |||
2802 | } | |||
2803 | } | |||
2804 | else | |||
2805 | { | |||
2806 | if ((( nRowCol == nMinRowCol ) && ( rDockingOperation == DOCKOP_BEFORE_COLROW )) || | |||
2807 | (( nRowCol == nMaxRowCol ) && ( rDockingOperation == DOCKOP_AFTER_COLROW ))) | |||
2808 | bOpOutsideOfDockingArea = true; | |||
2809 | else | |||
2810 | { | |||
2811 | // handle docking before/after a row | |||
2812 | implts_setTrackingRect( eDockedArea, rMousePos, aTrackingRect ); | |||
2813 | rTrackingRect = implts_calcTrackingAndElementRect( | |||
2814 | eDockedArea, nRowCol, rUIElement, | |||
2815 | aTrackingRect, aRowColumnRect, aContainerWinSize ); | |||
2816 | ||||
2817 | sal_Int32 nOffsetX( 0 ); | |||
2818 | sal_Int32 nOffsetY( 0 ); | |||
2819 | if ( bHorizontalDockArea ) | |||
2820 | nOffsetY = sal_Int32( floor( aRowColumnRect.getHeight() / 2.0 + 0.5 )); | |||
2821 | else | |||
2822 | nOffsetX = sal_Int32( floor( aRowColumnRect.getWidth() / 2.0 + 0.5 )); | |||
2823 | ||||
2824 | if ( rDockingOperation == DOCKOP_BEFORE_COLROW ) | |||
2825 | { | |||
2826 | if (( eDockedArea == ui::DockingArea_DOCKINGAREA_TOP ) || ( eDockedArea == ui::DockingArea_DOCKINGAREA_LEFT )) | |||
2827 | { | |||
2828 | // Docking before/after means move track rectangle half column/row. | |||
2829 | // As left and top are ordered 0...n instead of right and bottom | |||
2830 | // which uses n...0, we have to use negative values for top/left. | |||
2831 | nOffsetX *= -1; | |||
2832 | nOffsetY *= -1; | |||
2833 | } | |||
2834 | } | |||
2835 | else | |||
2836 | { | |||
2837 | if (( eDockedArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) || ( eDockedArea == ui::DockingArea_DOCKINGAREA_RIGHT )) | |||
2838 | { | |||
2839 | // Docking before/after means move track rectangle half column/row. | |||
2840 | // As left and top are ordered 0...n instead of right and bottom | |||
2841 | // which uses n...0, we have to use negative values for top/left. | |||
2842 | nOffsetX *= -1; | |||
2843 | nOffsetY *= -1; | |||
2844 | } | |||
2845 | nRowCol++; | |||
2846 | } | |||
2847 | ||||
2848 | if ( bHorizontalDockArea ) | |||
2849 | rUIElement.m_aDockedData.m_aPos.Y = nRowCol; | |||
2850 | else | |||
2851 | rUIElement.m_aDockedData.m_aPos.X = nRowCol; | |||
2852 | ||||
2853 | rTrackingRect.Move( nOffsetX, nOffsetY ); | |||
2854 | rTrackingRect.SetSize( aTrackingRect.GetSize() ); | |||
2855 | } | |||
2856 | } | |||
2857 | } | |||
2858 | } | |||
2859 | ||||
2860 | // Docking outside of our docking window area => | |||
2861 | // Users want to dock before/after first/last docked element or to an empty docking area | |||
2862 | if ( !bOpOutsideOfDockingArea ) | |||
2863 | return; | |||
2864 | ||||
2865 | // set correct size for docking | |||
2866 | implts_setTrackingRect( eDockedArea, rMousePos, aTrackingRect ); | |||
2867 | rTrackingRect = aTrackingRect; | |||
2868 | ||||
2869 | if ( bHorizontalDockArea ) | |||
2870 | { | |||
2871 | sal_Int32 nPosX( std::max( sal_Int32( rTrackingRect.Left()), sal_Int32( 0 ))); | |||
2872 | if (( nPosX + rTrackingRect.getWidth()) > aContainerWinSize.Width() ) | |||
2873 | nPosX = std::min( nPosX, | |||
2874 | std::max( sal_Int32( aContainerWinSize.Width() - rTrackingRect.getWidth() ), | |||
2875 | sal_Int32( 0 ))); | |||
2876 | ||||
2877 | sal_Int32 nSize = std::min( aContainerWinSize.Width(), rTrackingRect.getWidth() ); | |||
2878 | sal_Int32 nDockHeight = std::max( static_cast<sal_Int32>(aDockingAreaRect.getHeight()), sal_Int32( 0 )); | |||
2879 | if ( nDockHeight == 0 ) | |||
2880 | { | |||
2881 | sal_Int32 nPosY( std::max( aDockingAreaRect.Top(), aDockingAreaRect.Bottom() )); | |||
2882 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) | |||
2883 | nPosY -= rTrackingRect.getHeight(); | |||
2884 | rTrackingRect.SetPos( Point( nPosX, nPosY )); | |||
2885 | rUIElement.m_aDockedData.m_aPos.Y = 0; | |||
2886 | } | |||
2887 | else if ( rMousePos.Y() < ( aDockingAreaRect.Top() + ( nDockHeight / 2 ))) | |||
2888 | { | |||
2889 | rTrackingRect.SetPos( Point( nPosX, aDockingAreaRect.Top() - rTrackingRect.getHeight() )); | |||
2890 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
2891 | rUIElement.m_aDockedData.m_aPos.Y = 0; | |||
2892 | else | |||
2893 | rUIElement.m_aDockedData.m_aPos.Y = ( nMaxRowCol >= 0 ) ? nMaxRowCol+1 : 0; | |||
2894 | rDockingOperation = DOCKOP_BEFORE_COLROW; | |||
2895 | } | |||
2896 | else | |||
2897 | { | |||
2898 | rTrackingRect.SetPos( Point( nPosX, aDockingAreaRect.Bottom() )); | |||
2899 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
2900 | rUIElement.m_aDockedData.m_aPos.Y = ( nMaxRowCol >= 0 ) ? nMaxRowCol+1 : 0; | |||
2901 | else | |||
2902 | rUIElement.m_aDockedData.m_aPos.Y = 0; | |||
2903 | rDockingOperation = DOCKOP_AFTER_COLROW; | |||
2904 | } | |||
2905 | rTrackingRect.setWidth( nSize ); | |||
2906 | ||||
2907 | { | |||
2908 | SolarMutexGuard aGuard; | |||
2909 | nPosX = pDockingAreaWindow->ScreenToOutputPixel( | |||
2910 | pContainerWindow->OutputToScreenPixel( rTrackingRect.TopLeft() )).X(); | |||
2911 | } | |||
2912 | rUIElement.m_aDockedData.m_aPos.X = nPosX; | |||
2913 | } | |||
2914 | else | |||
2915 | { | |||
2916 | sal_Int32 nMaxDockingAreaHeight = std::max<sal_Int32>( 0, nMaxLeftRightDockAreaSize ); | |||
2917 | sal_Int32 nPosY( std::max<sal_Int32>( aTrackingRect.Top(), nTopDockingAreaSize )); | |||
2918 | if (( nPosY + aTrackingRect.getHeight()) > ( nTopDockingAreaSize + nMaxDockingAreaHeight )) | |||
2919 | nPosY = std::min( nPosY, | |||
2920 | std::max<sal_Int32>( nTopDockingAreaSize + ( nMaxDockingAreaHeight - aTrackingRect.getHeight() ), | |||
2921 | nTopDockingAreaSize )); | |||
2922 | ||||
2923 | sal_Int32 nSize = std::min( nMaxDockingAreaHeight, static_cast<sal_Int32>(aTrackingRect.getHeight()) ); | |||
2924 | sal_Int32 nDockWidth = std::max( static_cast<sal_Int32>(aDockingAreaRect.getWidth()), sal_Int32( 0 )); | |||
2925 | if ( nDockWidth == 0 ) | |||
2926 | { | |||
2927 | sal_Int32 nPosX( std::max( aDockingAreaRect.Left(), aDockingAreaRect.Right() )); | |||
2928 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_RIGHT ) | |||
2929 | nPosX -= rTrackingRect.getWidth(); | |||
2930 | rTrackingRect.SetPos( Point( nPosX, nPosY )); | |||
2931 | rUIElement.m_aDockedData.m_aPos.X = 0; | |||
2932 | } | |||
2933 | else if ( rMousePos.X() < ( aDockingAreaRect.Left() + ( nDockWidth / 2 ))) | |||
2934 | { | |||
2935 | rTrackingRect.SetPos( Point( aDockingAreaRect.Left() - rTrackingRect.getWidth(), nPosY )); | |||
2936 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
2937 | rUIElement.m_aDockedData.m_aPos.X = 0; | |||
2938 | else | |||
2939 | rUIElement.m_aDockedData.m_aPos.X = ( nMaxRowCol >= 0 ) ? nMaxRowCol+1 : 0; | |||
2940 | rDockingOperation = DOCKOP_BEFORE_COLROW; | |||
2941 | } | |||
2942 | else | |||
2943 | { | |||
2944 | rTrackingRect.SetPos( Point( aDockingAreaRect.Right(), nPosY )); | |||
2945 | if ( eDockedArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
2946 | rUIElement.m_aDockedData.m_aPos.X = ( nMaxRowCol >= 0 ) ? nMaxRowCol+1 : 0; | |||
2947 | else | |||
2948 | rUIElement.m_aDockedData.m_aPos.X = 0; | |||
2949 | rDockingOperation = DOCKOP_AFTER_COLROW; | |||
2950 | } | |||
2951 | rTrackingRect.setHeight( nSize ); | |||
2952 | ||||
2953 | { | |||
2954 | SolarMutexGuard aGuard; | |||
2955 | nPosY = pDockingAreaWindow->ScreenToOutputPixel( | |||
2956 | pContainerWindow->OutputToScreenPixel( rTrackingRect.TopLeft() )).Y(); | |||
2957 | } | |||
2958 | rUIElement.m_aDockedData.m_aPos.Y = nPosY; | |||
2959 | } | |||
2960 | } | |||
2961 | ||||
2962 | framework::ToolbarLayoutManager::DockingOperation ToolbarLayoutManager::implts_determineDockingOperation( | |||
2963 | ui::DockingArea DockingArea, | |||
2964 | const ::tools::Rectangle& rRowColRect, | |||
2965 | const Point& rMousePos ) | |||
2966 | { | |||
2967 | constexpr sal_Int32 nHorzVerticalRegionSize = 6; | |||
2968 | constexpr sal_Int32 nHorzVerticalMoveRegion = 4; | |||
2969 | ||||
2970 | if ( rRowColRect.IsInside( rMousePos )) | |||
2971 | { | |||
2972 | if ( isHorizontalDockingArea( DockingArea )) | |||
2973 | { | |||
2974 | sal_Int32 nRegion = rRowColRect.getHeight() / nHorzVerticalRegionSize; | |||
2975 | sal_Int32 nPosY = rRowColRect.Top() + nRegion; | |||
2976 | ||||
2977 | if ( rMousePos.Y() < nPosY ) | |||
2978 | return ( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) ? DOCKOP_BEFORE_COLROW : DOCKOP_AFTER_COLROW; | |||
2979 | else if ( rMousePos.Y() < ( nPosY + nRegion*nHorzVerticalMoveRegion )) | |||
2980 | return DOCKOP_ON_COLROW; | |||
2981 | else | |||
2982 | return ( DockingArea == ui::DockingArea_DOCKINGAREA_TOP ) ? DOCKOP_AFTER_COLROW : DOCKOP_BEFORE_COLROW; | |||
2983 | } | |||
2984 | else | |||
2985 | { | |||
2986 | sal_Int32 nRegion = rRowColRect.getWidth() / nHorzVerticalRegionSize; | |||
2987 | sal_Int32 nPosX = rRowColRect.Left() + nRegion; | |||
2988 | ||||
2989 | if ( rMousePos.X() < nPosX ) | |||
2990 | return ( DockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) ? DOCKOP_BEFORE_COLROW : DOCKOP_AFTER_COLROW; | |||
2991 | else if ( rMousePos.X() < ( nPosX + nRegion*nHorzVerticalMoveRegion )) | |||
2992 | return DOCKOP_ON_COLROW; | |||
2993 | else | |||
2994 | return ( DockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) ? DOCKOP_AFTER_COLROW : DOCKOP_BEFORE_COLROW; | |||
2995 | } | |||
2996 | } | |||
2997 | else | |||
2998 | return DOCKOP_ON_COLROW; | |||
2999 | } | |||
3000 | ||||
3001 | ::tools::Rectangle ToolbarLayoutManager::implts_calcTrackingAndElementRect( | |||
3002 | ui::DockingArea eDockingArea, | |||
3003 | sal_Int32 nRowCol, | |||
3004 | UIElement& rUIElement, | |||
3005 | const ::tools::Rectangle& rTrackingRect, | |||
3006 | const ::tools::Rectangle& rRowColumnRect, | |||
3007 | const ::Size& rContainerWinSize ) | |||
3008 | { | |||
3009 | SolarMutexResettableGuard aReadGuard; | |||
3010 | ::tools::Rectangle aDockingAreaOffsets( m_aDockingAreaOffsets ); | |||
3011 | aReadGuard.clear(); | |||
3012 | ||||
3013 | bool bHorizontalDockArea( isHorizontalDockingArea( eDockingArea )); | |||
3014 | ||||
3015 | sal_Int32 nTopDockingAreaSize( implts_getTopBottomDockingAreaSizes().Width() ); | |||
3016 | sal_Int32 nBottomDockingAreaSize( implts_getTopBottomDockingAreaSizes().Height() ); | |||
3017 | ||||
3018 | sal_Int32 nMaxLeftRightDockAreaSize = rContainerWinSize.Height() - | |||
3019 | nTopDockingAreaSize - | |||
3020 | nBottomDockingAreaSize - | |||
3021 | aDockingAreaOffsets.Top() - | |||
3022 | aDockingAreaOffsets.Bottom(); | |||
3023 | ||||
3024 | ::tools::Rectangle aTrackingRect( rTrackingRect ); | |||
3025 | if ( bHorizontalDockArea ) | |||
3026 | { | |||
3027 | sal_Int32 nPosX( std::max( sal_Int32( rTrackingRect.Left()), sal_Int32( 0 ))); | |||
3028 | if (( nPosX + rTrackingRect.getWidth()) > rContainerWinSize.Width() ) | |||
3029 | nPosX = std::min( nPosX, | |||
3030 | std::max( sal_Int32( rContainerWinSize.Width() - rTrackingRect.getWidth() ), | |||
3031 | sal_Int32( 0 ))); | |||
3032 | ||||
3033 | sal_Int32 nSize = std::min( rContainerWinSize.Width(), rTrackingRect.getWidth() ); | |||
3034 | ||||
3035 | aTrackingRect.SetPos( ::Point( nPosX, rRowColumnRect.Top() )); | |||
3036 | aTrackingRect.setWidth( nSize ); | |||
3037 | aTrackingRect.setHeight( rRowColumnRect.getHeight() ); | |||
3038 | ||||
3039 | // Set virtual position | |||
3040 | rUIElement.m_aDockedData.m_aPos.X = nPosX; | |||
3041 | rUIElement.m_aDockedData.m_aPos.Y = nRowCol; | |||
3042 | } | |||
3043 | else | |||
3044 | { | |||
3045 | sal_Int32 nMaxDockingAreaHeight = std::max<sal_Int32>( 0, nMaxLeftRightDockAreaSize ); | |||
3046 | ||||
3047 | sal_Int32 nPosY( std::max<sal_Int32>( aTrackingRect.Top(), nTopDockingAreaSize )); | |||
3048 | if (( nPosY + aTrackingRect.getHeight()) > ( nTopDockingAreaSize + nMaxDockingAreaHeight )) | |||
3049 | nPosY = std::min( nPosY, | |||
3050 | std::max<sal_Int32>( nTopDockingAreaSize + ( nMaxDockingAreaHeight - aTrackingRect.getHeight() ), | |||
3051 | nTopDockingAreaSize )); | |||
3052 | ||||
3053 | sal_Int32 nSize = std::min( nMaxDockingAreaHeight, static_cast<sal_Int32>(aTrackingRect.getHeight()) ); | |||
3054 | ||||
3055 | aTrackingRect.SetPos( ::Point( rRowColumnRect.Left(), nPosY )); | |||
3056 | aTrackingRect.setWidth( rRowColumnRect.getWidth() ); | |||
3057 | aTrackingRect.setHeight( nSize ); | |||
3058 | ||||
3059 | aReadGuard.reset(); | |||
3060 | uno::Reference< awt::XWindow > xDockingAreaWindow( m_xDockAreaWindows[static_cast<int>(eDockingArea)] ); | |||
3061 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
3062 | aReadGuard.clear(); | |||
3063 | ||||
3064 | sal_Int32 nDockPosY( 0 ); | |||
3065 | { | |||
3066 | SolarMutexGuard aGuard; | |||
3067 | vcl::Window* pDockingAreaWindow = VCLUnoHelper::GetWindow( xDockingAreaWindow ).get(); | |||
3068 | VclPtr<vcl::Window> pContainerWindow = VCLUnoHelper::GetWindow( xContainerWindow ); | |||
3069 | nDockPosY = pDockingAreaWindow->ScreenToOutputPixel( pContainerWindow->OutputToScreenPixel( ::Point( 0, nPosY ))).Y(); | |||
3070 | } | |||
3071 | ||||
3072 | // Set virtual position | |||
3073 | rUIElement.m_aDockedData.m_aPos.X = nRowCol; | |||
3074 | rUIElement.m_aDockedData.m_aPos.Y = nDockPosY; | |||
3075 | } | |||
3076 | ||||
3077 | return aTrackingRect; | |||
3078 | } | |||
3079 | ||||
3080 | void ToolbarLayoutManager::implts_setTrackingRect( ui::DockingArea eDockingArea, const ::Point& rMousePos, ::tools::Rectangle& rTrackingRect ) | |||
3081 | { | |||
3082 | ::Point aPoint( rTrackingRect.TopLeft()); | |||
3083 | if ( isHorizontalDockingArea( eDockingArea )) | |||
3084 | aPoint.setX( rMousePos.X() ); | |||
3085 | else | |||
3086 | aPoint.setY( rMousePos.Y() ); | |||
3087 | rTrackingRect.SetPos( aPoint ); | |||
3088 | } | |||
3089 | ||||
3090 | void ToolbarLayoutManager::implts_renumberRowColumnData( | |||
3091 | ui::DockingArea eDockingArea, | |||
3092 | const UIElement& rUIElement ) | |||
3093 | { | |||
3094 | SolarMutexClearableGuard aReadLock; | |||
3095 | uno::Reference< container::XNameAccess > xPersistentWindowState( m_xPersistentWindowState ); | |||
3096 | aReadLock.clear(); | |||
3097 | ||||
3098 | bool bHorzDockingArea( isHorizontalDockingArea( eDockingArea )); | |||
3099 | sal_Int32 nRowCol( bHorzDockingArea ? rUIElement.m_aDockedData.m_aPos.Y : rUIElement.m_aDockedData.m_aPos.X ); | |||
3100 | ||||
3101 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
3102 | { | |||
3103 | SolarMutexGuard aWriteLock; | |||
3104 | for (auto& elem : m_aUIElements) | |||
3105 | { | |||
3106 | if ((elem.m_aDockedData.m_nDockedArea == eDockingArea) | |||
3107 | && (elem.m_aName != rUIElement.m_aName)) | |||
3108 | { | |||
3109 | // Don't change toolbars without a valid docking position! | |||
3110 | if (isDefaultPos(elem.m_aDockedData.m_aPos)) | |||
3111 | continue; | |||
3112 | ||||
3113 | sal_Int32 nWindowRowCol | |||
3114 | = bHorzDockingArea ? elem.m_aDockedData.m_aPos.Y : elem.m_aDockedData.m_aPos.X; | |||
3115 | if (nWindowRowCol >= nRowCol) | |||
3116 | { | |||
3117 | if (bHorzDockingArea) | |||
3118 | elem.m_aDockedData.m_aPos.Y += 1; | |||
3119 | else | |||
3120 | elem.m_aDockedData.m_aPos.X += 1; | |||
3121 | } | |||
3122 | } | |||
3123 | } | |||
3124 | } | |||
3125 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
3126 | ||||
3127 | // We have to change the persistent window state part | |||
3128 | if ( !xPersistentWindowState.is() ) | |||
3129 | return; | |||
3130 | ||||
3131 | try | |||
3132 | { | |||
3133 | const uno::Sequence< OUString > aWindowElements = xPersistentWindowState->getElementNames(); | |||
3134 | for ( OUString const & rWindowElementName : aWindowElements ) | |||
3135 | { | |||
3136 | if ( rUIElement.m_aName != rWindowElementName ) | |||
3137 | { | |||
3138 | try | |||
3139 | { | |||
3140 | uno::Sequence< beans::PropertyValue > aPropValueSeq; | |||
3141 | awt::Point aDockedPos; | |||
3142 | ui::DockingArea nDockedArea( ui::DockingArea_DOCKINGAREA_DEFAULT ); | |||
3143 | ||||
3144 | xPersistentWindowState->getByName( rWindowElementName ) >>= aPropValueSeq; | |||
3145 | for ( beans::PropertyValue const & rProp : std::as_const(aPropValueSeq) ) | |||
3146 | { | |||
3147 | if ( rProp.Name == WINDOWSTATE_PROPERTY_DOCKINGAREA"DockingArea" ) | |||
3148 | rProp.Value >>= nDockedArea; | |||
3149 | else if ( rProp.Name == WINDOWSTATE_PROPERTY_DOCKPOS"DockPos" ) | |||
3150 | rProp.Value >>= aDockedPos; | |||
3151 | } | |||
3152 | ||||
3153 | // Don't change toolbars without a valid docking position! | |||
3154 | if ( isDefaultPos( aDockedPos )) | |||
3155 | continue; | |||
3156 | ||||
3157 | sal_Int32 nWindowRowCol = bHorzDockingArea ? aDockedPos.Y : aDockedPos.X; | |||
3158 | if (( nDockedArea == eDockingArea ) && ( nWindowRowCol >= nRowCol )) | |||
3159 | { | |||
3160 | if ( bHorzDockingArea ) | |||
3161 | aDockedPos.Y += 1; | |||
3162 | else | |||
3163 | aDockedPos.X += 1; | |||
3164 | ||||
3165 | uno::Reference< container::XNameReplace > xReplace( xPersistentWindowState, uno::UNO_QUERY ); | |||
3166 | xReplace->replaceByName( rWindowElementName, makeAny( aPropValueSeq )); | |||
3167 | } | |||
3168 | } | |||
3169 | catch (const uno::Exception&) | |||
3170 | { | |||
3171 | } | |||
3172 | } | |||
3173 | } | |||
3174 | } | |||
3175 | catch (const uno::Exception&) | |||
3176 | { | |||
3177 | } | |||
3178 | } | |||
3179 | ||||
3180 | // XWindowListener | |||
3181 | ||||
3182 | void SAL_CALL ToolbarLayoutManager::windowResized( const awt::WindowEvent& aEvent ) | |||
3183 | { | |||
3184 | SolarMutexClearableGuard aWriteLock; | |||
3185 | bool bLocked( m_bDockingInProgress ); | |||
3186 | bool bLayoutInProgress( m_bLayoutInProgress ); | |||
3187 | aWriteLock.clear(); | |||
3188 | ||||
3189 | // Do not do anything if we are in the middle of a docking process. This would interfere all other | |||
3190 | // operations. We will store the new position and size in the docking handlers. | |||
3191 | // Do not do anything if we are in the middle of our layouting process. We will adapt the position | |||
3192 | // and size of the user interface elements. | |||
3193 | if ( bLocked || bLayoutInProgress ) | |||
3194 | return; | |||
3195 | ||||
3196 | bool bNotify( false ); | |||
3197 | uno::Reference< awt::XWindow > xWindow( aEvent.Source, uno::UNO_QUERY ); | |||
3198 | ||||
3199 | UIElement aUIElement = implts_findToolbar( aEvent.Source ); | |||
3200 | if ( aUIElement.m_xUIElement.is() ) | |||
3201 | { | |||
3202 | if ( aUIElement.m_bFloating ) | |||
3203 | { | |||
3204 | uno::Reference< awt::XWindow2 > xWindow2( xWindow, uno::UNO_QUERY ); | |||
3205 | ||||
3206 | if( xWindow2.is() ) | |||
3207 | { | |||
3208 | awt::Rectangle aPos = xWindow2->getPosSize(); | |||
3209 | awt::Size aSize = xWindow2->getOutputSize(); // always use output size for consistency | |||
3210 | bool bVisible = xWindow2->isVisible(); | |||
3211 | ||||
3212 | // update element data | |||
3213 | aUIElement.m_aFloatingData.m_aPos = awt::Point(aPos.X, aPos.Y); | |||
3214 | aUIElement.m_aFloatingData.m_aSize = aSize; | |||
3215 | aUIElement.m_bVisible = bVisible; | |||
3216 | } | |||
3217 | ||||
3218 | implts_writeWindowStateData( aUIElement ); | |||
3219 | } | |||
3220 | else | |||
3221 | { | |||
3222 | implts_setLayoutDirty(); | |||
3223 | bNotify = true; | |||
3224 | } | |||
3225 | } | |||
3226 | ||||
3227 | if ( bNotify ) | |||
3228 | m_pParentLayouter->requestLayout(); | |||
3229 | } | |||
3230 | ||||
3231 | void SAL_CALL ToolbarLayoutManager::windowMoved( const awt::WindowEvent& /*aEvent*/ ) | |||
3232 | { | |||
3233 | } | |||
3234 | ||||
3235 | void SAL_CALL ToolbarLayoutManager::windowShown( const lang::EventObject& /*aEvent*/ ) | |||
3236 | { | |||
3237 | } | |||
3238 | ||||
3239 | void SAL_CALL ToolbarLayoutManager::windowHidden( const lang::EventObject& /*aEvent*/ ) | |||
3240 | { | |||
3241 | } | |||
3242 | ||||
3243 | // XDockableWindowListener | |||
3244 | ||||
3245 | void SAL_CALL ToolbarLayoutManager::startDocking( const awt::DockingEvent& e ) | |||
3246 | { | |||
3247 | bool bWinFound( false ); | |||
3248 | ||||
3249 | SolarMutexClearableGuard aReadGuard; | |||
3250 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
3251 | uno::Reference< awt::XWindow2 > xWindow( e.Source, uno::UNO_QUERY ); | |||
3252 | aReadGuard.clear(); | |||
3253 | ||||
3254 | UIElement aUIElement = implts_findToolbar( e.Source ); | |||
3255 | ||||
3256 | if ( aUIElement.m_xUIElement.is() && xWindow.is() ) | |||
3257 | { | |||
3258 | bWinFound = true; | |||
3259 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
3260 | if ( xDockWindow->isFloating() ) | |||
3261 | { | |||
3262 | awt::Rectangle aPos = xWindow->getPosSize(); | |||
3263 | awt::Size aSize = xWindow->getOutputSize(); | |||
3264 | ||||
3265 | aUIElement.m_aFloatingData.m_aPos = awt::Point(aPos.X, aPos.Y); | |||
3266 | aUIElement.m_aFloatingData.m_aSize = aSize; | |||
3267 | ||||
3268 | SolarMutexGuard aGuard; | |||
3269 | ||||
3270 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
3271 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
3272 | { | |||
3273 | ToolBox* pToolBox = static_cast<ToolBox *>(pWindow.get()); | |||
3274 | aUIElement.m_aFloatingData.m_nLines = pToolBox->GetFloatingLines(); | |||
3275 | aUIElement.m_aFloatingData.m_bIsHorizontal = isToolboxHorizontalAligned( pToolBox ); | |||
3276 | } | |||
3277 | } | |||
3278 | } | |||
3279 | ||||
3280 | SolarMutexGuard g; | |||
3281 | m_bDockingInProgress = bWinFound; | |||
3282 | m_aDockUIElement = aUIElement; | |||
3283 | m_aDockUIElement.m_bUserActive = true; | |||
3284 | } | |||
3285 | ||||
3286 | awt::DockingData SAL_CALL ToolbarLayoutManager::docking( const awt::DockingEvent& e ) | |||
3287 | { | |||
3288 | constexpr sal_Int32 MAGNETIC_DISTANCE_UNDOCK = 25; | |||
3289 | constexpr sal_Int32 MAGNETIC_DISTANCE_DOCK = 20; | |||
3290 | ||||
3291 | SolarMutexClearableGuard aReadLock; | |||
3292 | awt::DockingData aDockingData; | |||
3293 | uno::Reference< awt::XDockableWindow > xDockWindow( e.Source, uno::UNO_QUERY ); | |||
3294 | uno::Reference< awt::XWindow > xWindow( e.Source, uno::UNO_QUERY ); | |||
3295 | uno::Reference< awt::XWindow > xTopDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_TOP)] ); | |||
3296 | uno::Reference< awt::XWindow > xLeftDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_LEFT)] ); | |||
3297 | uno::Reference< awt::XWindow > xRightDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_RIGHT)] ); | |||
3298 | uno::Reference< awt::XWindow > xBottomDockingWindow( m_xDockAreaWindows[int(ui::DockingArea_DOCKINGAREA_BOTTOM)] ); | |||
3299 | uno::Reference< awt::XWindow2 > xContainerWindow( m_xContainerWindow ); | |||
3300 | UIElement aUIDockingElement( m_aDockUIElement ); | |||
3301 | ||||
3302 | bool bDockingInProgress( m_bDockingInProgress ); | |||
3303 | aReadLock.clear(); | |||
3304 | ||||
3305 | if ( bDockingInProgress ) | |||
3306 | aDockingData.TrackingRectangle = e.TrackingRectangle; | |||
3307 | ||||
3308 | if ( bDockingInProgress && xDockWindow.is() && xWindow.is() ) | |||
3309 | { | |||
3310 | try | |||
3311 | { | |||
3312 | SolarMutexGuard aGuard; | |||
3313 | ||||
3314 | DockingOperation eDockingOperation( DOCKOP_ON_COLROW ); | |||
3315 | ui::DockingArea eDockingArea( ui::DockingArea(-1) ); // none | |||
3316 | sal_Int32 nMagneticZone( aUIDockingElement.m_bFloating ? MAGNETIC_DISTANCE_DOCK : MAGNETIC_DISTANCE_UNDOCK ); | |||
3317 | ::tools::Rectangle aTrackingRect( e.TrackingRectangle.X, e.TrackingRectangle.Y, | |||
3318 | ( e.TrackingRectangle.X + e.TrackingRectangle.Width ), | |||
3319 | ( e.TrackingRectangle.Y + e.TrackingRectangle.Height )); | |||
3320 | ||||
3321 | awt::Rectangle aTmpRect = xTopDockingWindow->getPosSize(); | |||
3322 | ::tools::Rectangle aTopDockRect( aTmpRect.X, aTmpRect.Y, aTmpRect.Width, aTmpRect.Height ); | |||
3323 | ::tools::Rectangle aHotZoneTopDockRect( implts_calcHotZoneRect( aTopDockRect, nMagneticZone )); | |||
3324 | ||||
3325 | aTmpRect = xBottomDockingWindow->getPosSize(); | |||
3326 | ::tools::Rectangle aBottomDockRect( aTmpRect.X, aTmpRect.Y, ( aTmpRect.X + aTmpRect.Width), ( aTmpRect.Y + aTmpRect.Height )); | |||
3327 | ::tools::Rectangle aHotZoneBottomDockRect( implts_calcHotZoneRect( aBottomDockRect, nMagneticZone )); | |||
3328 | ||||
3329 | aTmpRect = xLeftDockingWindow->getPosSize(); | |||
3330 | ::tools::Rectangle aLeftDockRect( aTmpRect.X, aTmpRect.Y, ( aTmpRect.X + aTmpRect.Width ), ( aTmpRect.Y + aTmpRect.Height )); | |||
3331 | ::tools::Rectangle aHotZoneLeftDockRect( implts_calcHotZoneRect( aLeftDockRect, nMagneticZone )); | |||
3332 | ||||
3333 | aTmpRect = xRightDockingWindow->getPosSize(); | |||
3334 | ::tools::Rectangle aRightDockRect( aTmpRect.X, aTmpRect.Y, ( aTmpRect.X + aTmpRect.Width ), ( aTmpRect.Y + aTmpRect.Height )); | |||
3335 | ::tools::Rectangle aHotZoneRightDockRect( implts_calcHotZoneRect( aRightDockRect, nMagneticZone )); | |||
3336 | ||||
3337 | VclPtr<vcl::Window> pContainerWindow( VCLUnoHelper::GetWindow( xContainerWindow ) ); | |||
3338 | ::Point aMousePos( pContainerWindow->ScreenToOutputPixel( ::Point( e.MousePos.X, e.MousePos.Y ))); | |||
3339 | ||||
3340 | if ( aHotZoneTopDockRect.IsInside( aMousePos )) | |||
3341 | eDockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
3342 | else if ( aHotZoneBottomDockRect.IsInside( aMousePos )) | |||
3343 | eDockingArea = ui::DockingArea_DOCKINGAREA_BOTTOM; | |||
3344 | else if ( aHotZoneLeftDockRect.IsInside( aMousePos )) | |||
3345 | eDockingArea = ui::DockingArea_DOCKINGAREA_LEFT; | |||
3346 | else if ( aHotZoneRightDockRect.IsInside( aMousePos )) | |||
3347 | eDockingArea = ui::DockingArea_DOCKINGAREA_RIGHT; | |||
3348 | ||||
3349 | // Higher priority for movements inside the real docking area | |||
3350 | if ( aTopDockRect.IsInside( aMousePos )) | |||
3351 | eDockingArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
3352 | else if ( aBottomDockRect.IsInside( aMousePos )) | |||
3353 | eDockingArea = ui::DockingArea_DOCKINGAREA_BOTTOM; | |||
3354 | else if ( aLeftDockRect.IsInside( aMousePos )) | |||
3355 | eDockingArea = ui::DockingArea_DOCKINGAREA_LEFT; | |||
3356 | else if ( aRightDockRect.IsInside( aMousePos )) | |||
3357 | eDockingArea = ui::DockingArea_DOCKINGAREA_RIGHT; | |||
3358 | ||||
3359 | // Determine if we have a toolbar and set alignment according to the docking area! | |||
3360 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
3361 | ToolBox* pToolBox = nullptr; | |||
3362 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
3363 | pToolBox = static_cast<ToolBox *>(pWindow.get()); | |||
3364 | ||||
3365 | if ( eDockingArea != ui::DockingArea(-1) ) | |||
3366 | { | |||
3367 | if ( eDockingArea == ui::DockingArea_DOCKINGAREA_TOP ) | |||
3368 | { | |||
3369 | aUIDockingElement.m_aDockedData.m_nDockedArea = ui::DockingArea_DOCKINGAREA_TOP; | |||
3370 | aUIDockingElement.m_bFloating = false; | |||
3371 | } | |||
3372 | else if ( eDockingArea == ui::DockingArea_DOCKINGAREA_BOTTOM ) | |||
3373 | { | |||
3374 | aUIDockingElement.m_aDockedData.m_nDockedArea = ui::DockingArea_DOCKINGAREA_BOTTOM; | |||
3375 | aUIDockingElement.m_bFloating = false; | |||
3376 | } | |||
3377 | else if ( eDockingArea == ui::DockingArea_DOCKINGAREA_LEFT ) | |||
3378 | { | |||
3379 | aUIDockingElement.m_aDockedData.m_nDockedArea = ui::DockingArea_DOCKINGAREA_LEFT; | |||
3380 | aUIDockingElement.m_bFloating = false; | |||
3381 | } | |||
3382 | else if ( eDockingArea == ui::DockingArea_DOCKINGAREA_RIGHT ) | |||
3383 | { | |||
3384 | aUIDockingElement.m_aDockedData.m_nDockedArea = ui::DockingArea_DOCKINGAREA_RIGHT; | |||
3385 | aUIDockingElement.m_bFloating = false; | |||
3386 | } | |||
3387 | ||||
3388 | ::Point aOutputPos = pContainerWindow->ScreenToOutputPixel( aTrackingRect.TopLeft() ); | |||
3389 | aTrackingRect.SetPos( aOutputPos ); | |||
3390 | ||||
3391 | ::tools::Rectangle aNewDockingRect( aTrackingRect ); | |||
3392 | ||||
3393 | implts_calcDockingPosSize( aUIDockingElement, eDockingOperation, aNewDockingRect, aMousePos ); | |||
3394 | ||||
3395 | ::Point aScreenPos = pContainerWindow->OutputToScreenPixel( aNewDockingRect.TopLeft() ); | |||
3396 | aDockingData.TrackingRectangle = awt::Rectangle( aScreenPos.X(), aScreenPos.Y(), | |||
3397 | aNewDockingRect.getWidth(), aNewDockingRect.getHeight() ); | |||
3398 | } | |||
3399 | else if (pToolBox) | |||
3400 | { | |||
3401 | bool bIsHorizontal = isToolboxHorizontalAligned( pToolBox ); | |||
3402 | awt::Size aFloatSize = aUIDockingElement.m_aFloatingData.m_aSize; | |||
3403 | if ( aFloatSize.Width > 0 && aFloatSize.Height > 0 ) | |||
3404 | { | |||
3405 | aUIDockingElement.m_aFloatingData.m_aPos = AWTPoint(pContainerWindow->ScreenToOutputPixel(VCLPoint(e.MousePos))); | |||
3406 | aDockingData.TrackingRectangle.Height = aFloatSize.Height; | |||
3407 | aDockingData.TrackingRectangle.Width = aFloatSize.Width; | |||
3408 | } | |||
3409 | else | |||
3410 | { | |||
3411 | aFloatSize = AWTSize(pToolBox->CalcWindowSizePixel()); | |||
3412 | if ( !bIsHorizontal ) | |||
3413 | { | |||
3414 | // Floating toolbars are always horizontal aligned! We have to swap | |||
3415 | // width/height if we have a vertical aligned toolbar. | |||
3416 | sal_Int32 nTemp = aFloatSize.Height; | |||
3417 | aFloatSize.Height = aFloatSize.Width; | |||
3418 | aFloatSize.Width = nTemp; | |||
3419 | } | |||
3420 | ||||
3421 | aDockingData.TrackingRectangle.Height = aFloatSize.Height; | |||
3422 | aDockingData.TrackingRectangle.Width = aFloatSize.Width; | |||
3423 | ||||
3424 | // For the first time we don't have any data about the floating size of a toolbar. | |||
3425 | // We calculate it and store it for later use. | |||
3426 | aUIDockingElement.m_aFloatingData.m_aPos = AWTPoint(pContainerWindow->ScreenToOutputPixel(VCLPoint(e.MousePos))); | |||
3427 | aUIDockingElement.m_aFloatingData.m_aSize = aFloatSize; | |||
3428 | aUIDockingElement.m_aFloatingData.m_nLines = pToolBox->GetFloatingLines(); | |||
3429 | aUIDockingElement.m_aFloatingData.m_bIsHorizontal = isToolboxHorizontalAligned( pToolBox ); | |||
3430 | } | |||
3431 | aDockingData.TrackingRectangle.X = e.MousePos.X; | |||
3432 | aDockingData.TrackingRectangle.Y = e.MousePos.Y; | |||
3433 | } | |||
3434 | ||||
3435 | aDockingData.bFloating = ( eDockingArea == ui::DockingArea(-1) ); | |||
3436 | ||||
3437 | // Write current data to the member docking progress data | |||
3438 | SolarMutexGuard g; | |||
3439 | m_aDockUIElement.m_bFloating = aDockingData.bFloating; | |||
3440 | if ( !aDockingData.bFloating ) | |||
3441 | { | |||
3442 | m_aDockUIElement.m_aDockedData = aUIDockingElement.m_aDockedData; | |||
3443 | ||||
3444 | m_eDockOperation = eDockingOperation; | |||
3445 | } | |||
3446 | else | |||
3447 | m_aDockUIElement.m_aFloatingData = aUIDockingElement.m_aFloatingData; | |||
3448 | } | |||
3449 | catch (const uno::Exception&) | |||
3450 | { | |||
3451 | } | |||
3452 | } | |||
3453 | ||||
3454 | return aDockingData; | |||
3455 | } | |||
3456 | ||||
3457 | void SAL_CALL ToolbarLayoutManager::endDocking( const awt::EndDockingEvent& e ) | |||
3458 | { | |||
3459 | if (e.bCancelled) | |||
3460 | return; | |||
3461 | ||||
3462 | bool bDockingInProgress( false ); | |||
3463 | bool bStartDockFloated( false ); | |||
3464 | bool bFloating( false ); | |||
3465 | UIElement aUIDockingElement; | |||
3466 | ||||
3467 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
3468 | SolarMutexResettableGuard aWriteLock; | |||
3469 | bDockingInProgress = m_bDockingInProgress; | |||
3470 | aUIDockingElement = m_aDockUIElement; | |||
3471 | bFloating = aUIDockingElement.m_bFloating; | |||
3472 | ||||
3473 | UIElement& rUIElement = impl_findToolbar( aUIDockingElement.m_aName ); | |||
3474 | if ( rUIElement.m_aName == aUIDockingElement.m_aName ) | |||
3475 | { | |||
3476 | if ( aUIDockingElement.m_bFloating ) | |||
3477 | { | |||
3478 | // Write last position into position data | |||
3479 | uno::Reference< awt::XWindow > xWindow( aUIDockingElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
3480 | rUIElement.m_aFloatingData = aUIDockingElement.m_aFloatingData; | |||
3481 | awt::Rectangle aTmpRect = xWindow->getPosSize(); | |||
3482 | rUIElement.m_aFloatingData.m_aPos = awt::Point(aTmpRect.X, aTmpRect.Y); | |||
3483 | // make changes also for our local data as we use it to make data persistent | |||
3484 | aUIDockingElement.m_aFloatingData = rUIElement.m_aFloatingData; | |||
3485 | } | |||
3486 | else | |||
3487 | { | |||
3488 | rUIElement.m_aDockedData = aUIDockingElement.m_aDockedData; | |||
3489 | rUIElement.m_aFloatingData.m_aSize = aUIDockingElement.m_aFloatingData.m_aSize; | |||
3490 | ||||
3491 | if ( m_eDockOperation != DOCKOP_ON_COLROW ) | |||
3492 | { | |||
3493 | // we have to renumber our row/column data to insert a new row/column | |||
3494 | implts_renumberRowColumnData(aUIDockingElement.m_aDockedData.m_nDockedArea, aUIDockingElement ); | |||
3495 | } | |||
3496 | } | |||
3497 | ||||
3498 | bStartDockFloated = rUIElement.m_bFloating; | |||
3499 | rUIElement.m_bFloating = m_aDockUIElement.m_bFloating; | |||
3500 | rUIElement.m_bUserActive = true; | |||
3501 | } | |||
3502 | ||||
3503 | // reset member for next docking operation | |||
3504 | m_aDockUIElement.m_xUIElement.clear(); | |||
3505 | m_eDockOperation = DOCKOP_ON_COLROW; | |||
3506 | aWriteLock.clear(); | |||
3507 | /* SAFE AREA ----------------------------------------------------------------------------------------------- */ | |||
3508 | ||||
3509 | implts_writeWindowStateData( aUIDockingElement ); | |||
3510 | ||||
3511 | if ( bDockingInProgress ) | |||
3512 | { | |||
3513 | SolarMutexGuard aGuard; | |||
3514 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( uno::Reference< awt::XWindow >( e.Source, uno::UNO_QUERY )); | |||
3515 | ToolBox* pToolBox = nullptr; | |||
3516 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
3517 | pToolBox = static_cast<ToolBox *>(pWindow.get()); | |||
3518 | ||||
3519 | if ( pToolBox ) | |||
3520 | { | |||
3521 | if( e.bFloating ) | |||
3522 | { | |||
3523 | if ( aUIDockingElement.m_aFloatingData.m_bIsHorizontal ) | |||
3524 | pToolBox->SetAlign(); | |||
3525 | else | |||
3526 | pToolBox->SetAlign( WindowAlign::Left ); | |||
3527 | } | |||
3528 | else | |||
3529 | { | |||
3530 | ::Size aSize; | |||
3531 | ||||
3532 | pToolBox->SetAlign( ImplConvertAlignment( aUIDockingElement.m_aDockedData.m_nDockedArea) ); | |||
3533 | ||||
3534 | // Docked toolbars have always one line | |||
3535 | aSize = pToolBox->CalcWindowSizePixel( 1 ); | |||
3536 | ||||
3537 | // Lock layouting updates as our listener would be called due to SetSizePixel | |||
3538 | pToolBox->SetOutputSizePixel( aSize ); | |||
3539 | } | |||
3540 | } | |||
3541 | } | |||
3542 | ||||
3543 | implts_sortUIElements(); | |||
3544 | ||||
3545 | aWriteLock.reset(); | |||
3546 | m_bDockingInProgress = false; | |||
3547 | m_bLayoutDirty = !bStartDockFloated || !bFloating; | |||
3548 | bool bNotify = m_bLayoutDirty; | |||
3549 | aWriteLock.clear(); | |||
3550 | ||||
3551 | if ( bNotify ) | |||
3552 | m_pParentLayouter->requestLayout(); | |||
3553 | } | |||
3554 | ||||
3555 | sal_Bool SAL_CALL ToolbarLayoutManager::prepareToggleFloatingMode( const lang::EventObject& e ) | |||
3556 | { | |||
3557 | SolarMutexClearableGuard aReadLock; | |||
3558 | bool bDockingInProgress = m_bDockingInProgress; | |||
3559 | aReadLock.clear(); | |||
3560 | ||||
3561 | UIElement aUIDockingElement = implts_findToolbar( e.Source ); | |||
3562 | bool bWinFound( !aUIDockingElement.m_aName.isEmpty() ); | |||
3563 | uno::Reference< awt::XWindow > xWindow( e.Source, uno::UNO_QUERY ); | |||
3564 | ||||
3565 | if ( bWinFound && xWindow.is() ) | |||
3566 | { | |||
3567 | if ( !bDockingInProgress ) | |||
3568 | { | |||
3569 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
3570 | if ( xDockWindow->isFloating() ) | |||
3571 | { | |||
3572 | { | |||
3573 | SolarMutexGuard aGuard; | |||
3574 | VclPtr<vcl::Window> pWindow = VCLUnoHelper::GetWindow( xWindow ); | |||
3575 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
3576 | { | |||
3577 | ToolBox* pToolBox = static_cast< ToolBox *>( pWindow.get() ); | |||
3578 | aUIDockingElement.m_aFloatingData.m_aPos = AWTPoint(pToolBox->GetPosPixel()); | |||
3579 | aUIDockingElement.m_aFloatingData.m_aSize = AWTSize(pToolBox->GetOutputSizePixel()); | |||
3580 | aUIDockingElement.m_aFloatingData.m_nLines = pToolBox->GetFloatingLines(); | |||
3581 | aUIDockingElement.m_aFloatingData.m_bIsHorizontal = isToolboxHorizontalAligned( pToolBox ); | |||
3582 | } | |||
3583 | } | |||
3584 | ||||
3585 | UIElement aUIElement = implts_findToolbar( aUIDockingElement.m_aName ); | |||
3586 | if ( aUIElement.m_aName == aUIDockingElement.m_aName ) | |||
3587 | implts_setToolbar( aUIDockingElement ); | |||
3588 | } | |||
3589 | } | |||
3590 | } | |||
3591 | ||||
3592 | return true; | |||
3593 | } | |||
3594 | ||||
3595 | void SAL_CALL ToolbarLayoutManager::toggleFloatingMode( const lang::EventObject& e ) | |||
3596 | { | |||
3597 | UIElement aUIDockingElement; | |||
3598 | ||||
3599 | SolarMutexResettableGuard aReadLock; | |||
3600 | bool bDockingInProgress( m_bDockingInProgress ); | |||
3601 | if ( bDockingInProgress ) | |||
3602 | aUIDockingElement = m_aDockUIElement; | |||
3603 | aReadLock.clear(); | |||
3604 | ||||
3605 | vcl::Window* pWindow( nullptr ); | |||
3606 | ToolBox* pToolBox( nullptr ); | |||
3607 | uno::Reference< awt::XWindow2 > xWindow; | |||
3608 | ||||
3609 | { | |||
3610 | SolarMutexGuard aGuard; | |||
3611 | xWindow.set( e.Source, uno::UNO_QUERY ); | |||
3612 | pWindow = VCLUnoHelper::GetWindow( xWindow ).get(); | |||
3613 | ||||
3614 | if ( pWindow && pWindow->GetType() == WindowType::TOOLBOX ) | |||
3615 | pToolBox = static_cast<ToolBox *>(pWindow); | |||
3616 | } | |||
3617 | ||||
3618 | if ( !bDockingInProgress ) | |||
3619 | { | |||
3620 | aUIDockingElement = implts_findToolbar( e.Source ); | |||
3621 | bool bWinFound = !aUIDockingElement.m_aName.isEmpty(); | |||
3622 | ||||
3623 | if ( bWinFound && xWindow.is() ) | |||
3624 | { | |||
3625 | aUIDockingElement.m_bFloating = !aUIDockingElement.m_bFloating; | |||
3626 | aUIDockingElement.m_bUserActive = true; | |||
3627 | ||||
3628 | implts_setLayoutInProgress(); | |||
3629 | if ( aUIDockingElement.m_bFloating ) | |||
3630 | { | |||
3631 | SolarMutexGuard aGuard; | |||
3632 | if ( pToolBox ) | |||
3633 | { | |||
3634 | pToolBox->SetLineCount( aUIDockingElement.m_aFloatingData.m_nLines ); | |||
3635 | if ( aUIDockingElement.m_aFloatingData.m_bIsHorizontal ) | |||
3636 | pToolBox->SetAlign(); | |||
3637 | else | |||
3638 | pToolBox->SetAlign( WindowAlign::Left ); | |||
3639 | } | |||
3640 | ||||
3641 | bool bUndefPos = hasDefaultPosValue( aUIDockingElement.m_aFloatingData.m_aPos ); | |||
3642 | bool bSetSize = !hasEmptySize( aUIDockingElement.m_aFloatingData.m_aSize ); | |||
3643 | ||||
3644 | if ( bUndefPos ) | |||
3645 | aUIDockingElement.m_aFloatingData.m_aPos = implts_findNextCascadeFloatingPos(); | |||
3646 | ||||
3647 | if ( !bSetSize ) | |||
3648 | { | |||
3649 | if ( pToolBox ) | |||
3650 | aUIDockingElement.m_aFloatingData.m_aSize = AWTSize(pToolBox->CalcFloatingWindowSizePixel()); | |||
3651 | else if ( pWindow ) | |||
3652 | aUIDockingElement.m_aFloatingData.m_aSize = AWTSize(pWindow->GetOutputSizePixel()); | |||
3653 | } | |||
3654 | ||||
3655 | xWindow->setPosSize( aUIDockingElement.m_aFloatingData.m_aPos.X, | |||
3656 | aUIDockingElement.m_aFloatingData.m_aPos.Y, | |||
3657 | 0, 0, awt::PosSize::POS ); | |||
3658 | xWindow->setOutputSize(aUIDockingElement.m_aFloatingData.m_aSize); | |||
3659 | } | |||
3660 | else | |||
3661 | { | |||
3662 | if ( isDefaultPos( aUIDockingElement.m_aDockedData.m_aPos )) | |||
3663 | { | |||
3664 | // Docking on its default position without a preset position - | |||
3665 | // we have to find a good place for it. | |||
3666 | ::Point aPixelPos; | |||
3667 | awt::Point aDockPos; | |||
3668 | ::Size aSize; | |||
3669 | ||||
3670 | { | |||
3671 | SolarMutexGuard aGuard; | |||
3672 | if ( pToolBox ) | |||
3673 | aSize = pToolBox->CalcWindowSizePixel( 1, ImplConvertAlignment( aUIDockingElement.m_aDockedData.m_nDockedArea ) ); | |||
3674 | else if ( pWindow ) | |||
3675 | aSize = pWindow->GetSizePixel(); | |||
3676 | } | |||
3677 | ||||
3678 | implts_findNextDockingPos(aUIDockingElement.m_aDockedData.m_nDockedArea, aSize, aDockPos, aPixelPos ); | |||
3679 | aUIDockingElement.m_aDockedData.m_aPos = aDockPos; | |||
3680 | } | |||
3681 | ||||
3682 | SolarMutexGuard aGuard; | |||
3683 | if ( pToolBox ) | |||
3684 | { | |||
3685 | pToolBox->SetAlign( ImplConvertAlignment( aUIDockingElement.m_aDockedData.m_nDockedArea) ); | |||
3686 | ::Size aSize = pToolBox->CalcWindowSizePixel( 1 ); | |||
3687 | awt::Rectangle aRect = xWindow->getPosSize(); | |||
3688 | xWindow->setPosSize( aRect.X, aRect.Y, 0, 0, awt::PosSize::POS ); | |||
3689 | xWindow->setOutputSize( AWTSize( aSize ) ); | |||
3690 | } | |||
3691 | } | |||
3692 | ||||
3693 | implts_setLayoutInProgress( false ); | |||
3694 | implts_setToolbar( aUIDockingElement ); | |||
3695 | implts_writeWindowStateData( aUIDockingElement ); | |||
3696 | implts_sortUIElements(); | |||
3697 | implts_setLayoutDirty(); | |||
3698 | ||||
3699 | aReadLock.reset(); | |||
3700 | LayoutManager* pParentLayouter( m_pParentLayouter ); | |||
3701 | aReadLock.clear(); | |||
3702 | ||||
3703 | if ( pParentLayouter ) | |||
3704 | pParentLayouter->requestLayout(); | |||
3705 | } | |||
3706 | } | |||
3707 | else | |||
3708 | { | |||
3709 | SolarMutexGuard aGuard; | |||
3710 | if ( pToolBox ) | |||
3711 | { | |||
3712 | if ( aUIDockingElement.m_bFloating ) | |||
3713 | { | |||
3714 | if ( aUIDockingElement.m_aFloatingData.m_bIsHorizontal ) | |||
3715 | pToolBox->SetAlign(); | |||
3716 | else | |||
3717 | pToolBox->SetAlign( WindowAlign::Left ); | |||
3718 | } | |||
3719 | else | |||
3720 | pToolBox->SetAlign( ImplConvertAlignment( aUIDockingElement.m_aDockedData.m_nDockedArea) ); | |||
3721 | } | |||
3722 | } | |||
3723 | } | |||
3724 | ||||
3725 | void SAL_CALL ToolbarLayoutManager::closed( const lang::EventObject& e ) | |||
3726 | { | |||
3727 | OUString aName; | |||
3728 | UIElement aUIElement; | |||
3729 | ||||
3730 | { | |||
3731 | SolarMutexGuard aWriteLock; | |||
3732 | for (auto& elem : m_aUIElements) | |||
3733 | { | |||
3734 | uno::Reference<ui::XUIElement> xUIElement(elem.m_xUIElement); | |||
3735 | if (xUIElement.is()) | |||
3736 | { | |||
3737 | uno::Reference<uno::XInterface> xIfac(xUIElement->getRealInterface(), | |||
3738 | uno::UNO_QUERY); | |||
3739 | if (xIfac == e.Source) | |||
3740 | { | |||
3741 | aName = elem.m_aName; | |||
3742 | ||||
3743 | // user closes a toolbar => | |||
3744 | // context sensitive toolbar: only destroy toolbar and store state. | |||
3745 | // non context sensitive toolbar: make it invisible, store state and destroy it. | |||
3746 | if (!elem.m_bContextSensitive) | |||
3747 | elem.m_bVisible = false; | |||
3748 | ||||
3749 | aUIElement = elem; | |||
3750 | break; | |||
3751 | } | |||
3752 | } | |||
3753 | } | |||
3754 | } | |||
3755 | ||||
3756 | // destroy element | |||
3757 | if ( aName.isEmpty() ) | |||
3758 | return; | |||
3759 | ||||
3760 | implts_writeWindowStateData( aUIElement ); | |||
3761 | destroyToolbar( aName ); | |||
3762 | ||||
3763 | SolarMutexClearableGuard aReadLock; | |||
3764 | bool bLayoutDirty = m_bLayoutDirty; | |||
3765 | LayoutManager* pParentLayouter( m_pParentLayouter ); | |||
3766 | aReadLock.clear(); | |||
3767 | ||||
3768 | if ( bLayoutDirty && pParentLayouter ) | |||
3769 | pParentLayouter->requestLayout(); | |||
3770 | } | |||
3771 | ||||
3772 | void SAL_CALL ToolbarLayoutManager::endPopupMode( const awt::EndPopupModeEvent& /*e*/ ) | |||
3773 | { | |||
3774 | } | |||
3775 | ||||
3776 | // XUIConfigurationListener | |||
3777 | ||||
3778 | void SAL_CALL ToolbarLayoutManager::elementInserted( const ui::ConfigurationEvent& rEvent ) | |||
3779 | { | |||
3780 | UIElement aUIElement = implts_findToolbar( rEvent.ResourceURL ); | |||
3781 | ||||
3782 | uno::Reference< ui::XUIElementSettings > xElementSettings( aUIElement.m_xUIElement, uno::UNO_QUERY ); | |||
3783 | if ( xElementSettings.is() ) | |||
3784 | { | |||
3785 | uno::Reference< beans::XPropertySet > xPropSet( xElementSettings, uno::UNO_QUERY ); | |||
3786 | if ( xPropSet.is() ) | |||
3787 | { | |||
3788 | if ( rEvent.Source == uno::Reference< uno::XInterface >( m_xDocCfgMgr, uno::UNO_QUERY )) | |||
3789 | xPropSet->setPropertyValue( "ConfigurationSource", makeAny( m_xDocCfgMgr )); | |||
3790 | } | |||
3791 | xElementSettings->updateSettings(); | |||
3792 | } | |||
3793 | else | |||
3794 | { | |||
3795 | OUString aElementType; | |||
3796 | OUString aElementName; | |||
3797 | parseResourceURL( rEvent.ResourceURL, aElementType, aElementName ); | |||
3798 | if ( aElementName.indexOf( "custom_" ) != -1 ) | |||
3799 | { | |||
3800 | // custom toolbar must be directly created, shown and layouted! | |||
3801 | createToolbar( rEvent.ResourceURL ); | |||
3802 | uno::Reference< ui::XUIElement > xUIElement = getToolbar( rEvent.ResourceURL ); | |||
3803 | if ( xUIElement.is() ) | |||
3804 | { | |||
3805 | OUString aUIName; | |||
3806 | uno::Reference< ui::XUIConfigurationManager > xCfgMgr; | |||
3807 | uno::Reference< beans::XPropertySet > xPropSet; | |||
3808 | ||||
3809 | try | |||
3810 | { | |||
3811 | xCfgMgr.set( rEvent.Source, uno::UNO_QUERY ); | |||
3812 | xPropSet.set( xCfgMgr->getSettings( rEvent.ResourceURL, false ), uno::UNO_QUERY ); | |||
3813 | ||||
3814 | if ( xPropSet.is() ) | |||
3815 | xPropSet->getPropertyValue("UIName") >>= aUIName; | |||
3816 | } | |||
3817 | catch (const container::NoSuchElementException&) | |||
3818 | { | |||
3819 | } | |||
3820 | catch (const beans::UnknownPropertyException&) | |||
3821 | { | |||
3822 | } | |||
3823 | catch (const lang::WrappedTargetException&) | |||
3824 | { | |||
3825 | } | |||
3826 | ||||
3827 | { | |||
3828 | SolarMutexGuard aGuard; | |||
3829 | vcl::Window* pWindow = getWindowFromXUIElement( xUIElement ); | |||
3830 | if ( pWindow ) | |||
3831 | pWindow->SetText( aUIName ); | |||
3832 | } | |||
3833 | ||||
3834 | showToolbar( rEvent.ResourceURL ); | |||
3835 | } | |||
3836 | } | |||
3837 | } | |||
3838 | } | |||
3839 | ||||
3840 | void SAL_CALL ToolbarLayoutManager::elementRemoved( const ui::ConfigurationEvent& rEvent ) | |||
3841 | { | |||
3842 | SolarMutexClearableGuard aReadLock; | |||
3843 | uno::Reference< awt::XWindow > xContainerWindow = m_xContainerWindow; | |||
3844 | uno::Reference< ui::XUIConfigurationManager > xModuleCfgMgr( m_xModuleCfgMgr ); | |||
3845 | uno::Reference< ui::XUIConfigurationManager > xDocCfgMgr( m_xDocCfgMgr ); | |||
3846 | aReadLock.clear(); | |||
3847 | ||||
3848 | UIElement aUIElement = implts_findToolbar( rEvent.ResourceURL ); | |||
3849 | uno::Reference< ui::XUIElementSettings > xElementSettings( aUIElement.m_xUIElement, uno::UNO_QUERY ); | |||
3850 | if ( !xElementSettings.is() ) | |||
3851 | return; | |||
3852 | ||||
3853 | bool bNoSettings( false ); | |||
3854 | OUString aConfigSourcePropName( "ConfigurationSource" ); | |||
3855 | uno::Reference< uno::XInterface > xElementCfgMgr; | |||
3856 | uno::Reference< beans::XPropertySet > xPropSet( xElementSettings, uno::UNO_QUERY ); | |||
3857 | ||||
3858 | if ( xPropSet.is() ) | |||
3859 | xPropSet->getPropertyValue( aConfigSourcePropName ) >>= xElementCfgMgr; | |||
3860 | ||||
3861 | if ( !xElementCfgMgr.is() ) | |||
3862 | return; | |||
3863 | ||||
3864 | // Check if the same UI configuration manager has changed => check further | |||
3865 | if ( rEvent.Source == xElementCfgMgr ) | |||
3866 | { | |||
3867 | // Same UI configuration manager where our element has its settings | |||
3868 | if ( rEvent.Source == uno::Reference< uno::XInterface >( xDocCfgMgr, uno::UNO_QUERY )) | |||
3869 | { | |||
3870 | // document settings removed | |||
3871 | if ( xModuleCfgMgr->hasSettings( rEvent.ResourceURL )) | |||
3872 | { | |||
3873 | xPropSet->setPropertyValue( aConfigSourcePropName, makeAny( xModuleCfgMgr )); | |||
3874 | xElementSettings->updateSettings(); | |||
3875 | return; | |||
3876 | } | |||
3877 | } | |||
3878 | ||||
3879 | bNoSettings = true; | |||
3880 | } | |||
3881 | ||||
3882 | // No settings anymore, element must be destroyed | |||
3883 | if ( xContainerWindow.is() && bNoSettings ) | |||
3884 | destroyToolbar( rEvent.ResourceURL ); | |||
3885 | } | |||
3886 | ||||
3887 | void SAL_CALL ToolbarLayoutManager::elementReplaced( const ui::ConfigurationEvent& rEvent ) | |||
3888 | { | |||
3889 | UIElement aUIElement = implts_findToolbar( rEvent.ResourceURL ); | |||
3890 | ||||
3891 | uno::Reference< ui::XUIElementSettings > xElementSettings( aUIElement.m_xUIElement, uno::UNO_QUERY ); | |||
3892 | if ( !xElementSettings.is() ) | |||
3893 | return; | |||
3894 | ||||
3895 | uno::Reference< uno::XInterface > xElementCfgMgr; | |||
3896 | uno::Reference< beans::XPropertySet > xPropSet( xElementSettings, uno::UNO_QUERY ); | |||
3897 | ||||
3898 | if ( xPropSet.is() ) | |||
3899 | xPropSet->getPropertyValue( "ConfigurationSource" ) >>= xElementCfgMgr; | |||
3900 | ||||
3901 | if ( !xElementCfgMgr.is() ) | |||
3902 | return; | |||
3903 | ||||
3904 | // Check if the same UI configuration manager has changed => update settings | |||
3905 | if ( rEvent.Source != xElementCfgMgr ) | |||
3906 | return; | |||
3907 | ||||
3908 | xElementSettings->updateSettings(); | |||
3909 | ||||
3910 | SolarMutexClearableGuard aWriteLock; | |||
3911 | bool bNotify = !aUIElement.m_bFloating; | |||
3912 | m_bLayoutDirty = bNotify; | |||
3913 | LayoutManager* pParentLayouter( m_pParentLayouter ); | |||
3914 | aWriteLock.clear(); | |||
3915 | ||||
3916 | if ( bNotify && pParentLayouter ) | |||
3917 | pParentLayouter->requestLayout(); | |||
3918 | } | |||
3919 | ||||
3920 | uno::Reference< ui::XUIElement > ToolbarLayoutManager::getToolbar( const OUString& aName ) | |||
3921 | { | |||
3922 | return implts_findToolbar( aName ).m_xUIElement; | |||
3923 | } | |||
3924 | ||||
3925 | uno::Sequence< uno::Reference< ui::XUIElement > > ToolbarLayoutManager::getToolbars() | |||
3926 | { | |||
3927 | uno::Sequence< uno::Reference< ui::XUIElement > > aSeq; | |||
3928 | ||||
3929 | SolarMutexGuard g; | |||
3930 | if ( !m_aUIElements.empty() ) | |||
3931 | { | |||
3932 | sal_uInt32 nCount(0); | |||
3933 | for (auto const& elem : m_aUIElements) | |||
3934 | { | |||
3935 | if ( elem.m_xUIElement.is() ) | |||
3936 | { | |||
3937 | ++nCount; | |||
3938 | aSeq.realloc( nCount ); | |||
3939 | aSeq[nCount-1] = elem.m_xUIElement; | |||
3940 | } | |||
3941 | } | |||
3942 | } | |||
3943 | ||||
3944 | return aSeq; | |||
3945 | } | |||
3946 | ||||
3947 | bool ToolbarLayoutManager::floatToolbar( const OUString& rResourceURL ) | |||
3948 | { | |||
3949 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
3950 | if ( aUIElement.m_xUIElement.is() ) | |||
3951 | { | |||
3952 | try | |||
3953 | { | |||
3954 | uno::Reference< awt::XDockableWindow > xDockWindow( aUIElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
3955 | if ( xDockWindow.is() && !xDockWindow->isFloating() ) | |||
3956 | { | |||
3957 | aUIElement.m_bFloating = true; | |||
3958 | implts_writeWindowStateData( aUIElement ); | |||
3959 | xDockWindow->setFloatingMode( true ); | |||
3960 | ||||
3961 | implts_setLayoutDirty(); | |||
3962 | implts_setToolbar( aUIElement ); | |||
3963 | return true; | |||
3964 | } | |||
3965 | } | |||
3966 | catch (const lang::DisposedException&) | |||
3967 | { | |||
3968 | } | |||
3969 | } | |||
3970 | ||||
3971 | return false; | |||
3972 | } | |||
3973 | ||||
3974 | bool ToolbarLayoutManager::lockToolbar( const OUString& rResourceURL ) | |||
3975 | { | |||
3976 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
3977 | if ( aUIElement.m_xUIElement.is() ) | |||
3978 | { | |||
3979 | try | |||
3980 | { | |||
3981 | uno::Reference< awt::XDockableWindow > xDockWindow( aUIElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
3982 | if ( xDockWindow.is() && !xDockWindow->isFloating() && !xDockWindow->isLocked() ) | |||
3983 | { | |||
3984 | aUIElement.m_aDockedData.m_bLocked = true; | |||
3985 | implts_writeWindowStateData( aUIElement ); | |||
3986 | xDockWindow->lock(); | |||
3987 | ||||
3988 | implts_setLayoutDirty(); | |||
3989 | implts_setToolbar( aUIElement ); | |||
3990 | return true; | |||
3991 | } | |||
3992 | } | |||
3993 | catch (const lang::DisposedException&) | |||
3994 | { | |||
3995 | } | |||
3996 | } | |||
3997 | ||||
3998 | return false; | |||
3999 | } | |||
4000 | ||||
4001 | bool ToolbarLayoutManager::unlockToolbar( const OUString& rResourceURL ) | |||
4002 | { | |||
4003 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
4004 | if ( aUIElement.m_xUIElement.is() ) | |||
4005 | { | |||
4006 | try | |||
4007 | { | |||
4008 | uno::Reference< awt::XDockableWindow > xDockWindow( aUIElement.m_xUIElement->getRealInterface(), uno::UNO_QUERY ); | |||
4009 | if ( xDockWindow.is() && !xDockWindow->isFloating() && xDockWindow->isLocked() ) | |||
4010 | { | |||
4011 | aUIElement.m_aDockedData.m_bLocked = false; | |||
4012 | implts_writeWindowStateData( aUIElement ); | |||
4013 | xDockWindow->unlock(); | |||
4014 | ||||
4015 | implts_setLayoutDirty(); | |||
4016 | implts_setToolbar( aUIElement ); | |||
4017 | return true; | |||
4018 | } | |||
4019 | } | |||
4020 | catch (const lang::DisposedException&) | |||
4021 | { | |||
4022 | } | |||
4023 | } | |||
4024 | ||||
4025 | return false; | |||
4026 | } | |||
4027 | ||||
4028 | bool ToolbarLayoutManager::isToolbarVisible( const OUString& rResourceURL ) | |||
4029 | { | |||
4030 | uno::Reference< awt::XWindow2 > xWindow2( implts_getXWindow( rResourceURL ), uno::UNO_QUERY ); | |||
4031 | return ( xWindow2.is() && xWindow2->isVisible() ); | |||
4032 | } | |||
4033 | ||||
4034 | bool ToolbarLayoutManager::isToolbarFloating( const OUString& rResourceURL ) | |||
4035 | { | |||
4036 | uno::Reference< awt::XDockableWindow > xDockWindow( implts_getXWindow( rResourceURL ), uno::UNO_QUERY ); | |||
4037 | return ( xDockWindow.is() && xDockWindow->isFloating() ); | |||
4038 | } | |||
4039 | ||||
4040 | bool ToolbarLayoutManager::isToolbarDocked( const OUString& rResourceURL ) | |||
4041 | { | |||
4042 | return !isToolbarFloating( rResourceURL ); | |||
4043 | } | |||
4044 | ||||
4045 | bool ToolbarLayoutManager::isToolbarLocked( const OUString& rResourceURL ) | |||
4046 | { | |||
4047 | uno::Reference< awt::XDockableWindow > xDockWindow( implts_getXWindow( rResourceURL ), uno::UNO_QUERY ); | |||
4048 | return ( xDockWindow.is() && xDockWindow->isLocked() ); | |||
4049 | } | |||
4050 | ||||
4051 | awt::Size ToolbarLayoutManager::getToolbarSize( const OUString& rResourceURL ) | |||
4052 | { | |||
4053 | vcl::Window* pWindow = implts_getWindow( rResourceURL ); | |||
4054 | ||||
4055 | SolarMutexGuard aGuard; | |||
4056 | if ( pWindow ) | |||
4057 | { | |||
4058 | ::Size aSize = pWindow->GetSizePixel(); | |||
4059 | awt::Size aWinSize; | |||
4060 | aWinSize.Width = aSize.Width(); | |||
4061 | aWinSize.Height = aSize.Height(); | |||
4062 | return aWinSize; | |||
4063 | } | |||
4064 | ||||
4065 | return awt::Size(); | |||
4066 | } | |||
4067 | ||||
4068 | awt::Point ToolbarLayoutManager::getToolbarPos( const OUString& rResourceURL ) | |||
4069 | { | |||
4070 | awt::Point aPos; | |||
4071 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
4072 | ||||
4073 | uno::Reference< awt::XWindow > xWindow( implts_getXWindow( rResourceURL )); | |||
4074 | if ( xWindow.is() ) | |||
4075 | { | |||
4076 | if ( aUIElement.m_bFloating ) | |||
4077 | { | |||
4078 | awt::Rectangle aRect = xWindow->getPosSize(); | |||
4079 | aPos.X = aRect.X; | |||
4080 | aPos.Y = aRect.Y; | |||
4081 | } | |||
4082 | else | |||
4083 | aPos = aUIElement.m_aDockedData.m_aPos; | |||
4084 | } | |||
4085 | ||||
4086 | return aPos; | |||
4087 | } | |||
4088 | ||||
4089 | void ToolbarLayoutManager::setToolbarSize( const OUString& rResourceURL, const awt::Size& aSize ) | |||
4090 | { | |||
4091 | uno::Reference< awt::XWindow2 > xWindow( implts_getXWindow( rResourceURL ), uno::UNO_QUERY ); | |||
4092 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
4093 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
4094 | ||||
4095 | if ( xWindow.is() && xDockWindow.is() && xDockWindow->isFloating() ) | |||
4096 | { | |||
4097 | xWindow->setOutputSize( aSize ); | |||
4098 | aUIElement.m_aFloatingData.m_aSize = aSize; | |||
4099 | implts_setToolbar( aUIElement ); | |||
4100 | implts_writeWindowStateData( aUIElement ); | |||
4101 | implts_sortUIElements(); | |||
4102 | } | |||
4103 | } | |||
4104 | ||||
4105 | void ToolbarLayoutManager::setToolbarPos( const OUString& rResourceURL, const awt::Point& aPos ) | |||
4106 | { | |||
4107 | uno::Reference< awt::XWindow > xWindow( implts_getXWindow( rResourceURL )); | |||
4108 | uno::Reference< awt::XDockableWindow > xDockWindow( xWindow, uno::UNO_QUERY ); | |||
4109 | UIElement aUIElement = implts_findToolbar( rResourceURL ); | |||
4110 | ||||
4111 | if ( xWindow.is() && xDockWindow.is() && xDockWindow->isFloating() ) | |||
4112 | { | |||
4113 | xWindow->setPosSize( aPos.X, aPos.Y, 0, 0, awt::PosSize::POS ); | |||
4114 | aUIElement.m_aFloatingData.m_aPos = aPos; | |||
4115 | implts_setToolbar( aUIElement ); | |||
4116 | implts_writeWindowStateData( aUIElement ); | |||
4117 | implts_sortUIElements(); | |||
4118 | } | |||
4119 | } | |||
4120 | ||||
4121 | void ToolbarLayoutManager::setToolbarPosSize( const OUString& rResourceURL, const awt::Point& aPos, const awt::Size& aSize ) | |||
4122 | { | |||
4123 | setToolbarPos( rResourceURL, aPos ); | |||
4124 | setToolbarSize( rResourceURL, aSize ); | |||
4125 | } | |||
4126 | ||||
4127 | } // namespace framework | |||
4128 | ||||
4129 | /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |
1 | /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ |
2 | /* |
3 | * This file is part of the LibreOffice project. |
4 | * |
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/. |
8 | * |
9 | * This file incorporates work covered by the following license notice: |
10 | * |
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 . |
18 | */ |
19 | |
20 | #ifndef INCLUDED_RTL_REF_HXX |
21 | #define INCLUDED_RTL_REF_HXX |
22 | |
23 | #include "sal/config.h" |
24 | |
25 | #include <cassert> |
26 | #include <cstddef> |
27 | #include <functional> |
28 | #ifdef LIBO_INTERNAL_ONLY1 |
29 | #include <type_traits> |
30 | #endif |
31 | |
32 | #include "sal/types.h" |
33 | |
34 | namespace rtl |
35 | { |
36 | |
37 | /** Template reference class for reference type. |
38 | */ |
39 | template <class reference_type> |
40 | class Reference |
41 | { |
42 | /** The <b>reference_type</b> body pointer. |
43 | */ |
44 | reference_type * m_pBody; |
45 | |
46 | |
47 | public: |
48 | /** Constructor... |
49 | */ |
50 | Reference() |
51 | : m_pBody (NULL__null) |
52 | {} |
53 | |
54 | |
55 | /** Constructor... |
56 | */ |
57 | Reference (reference_type * pBody, __sal_NoAcquire) |
58 | : m_pBody (pBody) |
59 | { |
60 | } |
61 | |
62 | /** Constructor... |
63 | */ |
64 | Reference (reference_type * pBody) |
65 | : m_pBody (pBody) |
66 | { |
67 | if (m_pBody) |
68 | m_pBody->acquire(); |
69 | } |
70 | |
71 | /** Copy constructor... |
72 | */ |
73 | Reference (const Reference<reference_type> & handle) |
74 | : m_pBody (handle.m_pBody) |
75 | { |
76 | if (m_pBody) |
77 | m_pBody->acquire(); |
78 | } |
79 | |
80 | #ifdef LIBO_INTERNAL_ONLY1 |
81 | /** Move constructor... |
82 | */ |
83 | Reference (Reference<reference_type> && handle) noexcept |
84 | : m_pBody (handle.m_pBody) |
85 | { |
86 | handle.m_pBody = nullptr; |
87 | } |
88 | #endif |
89 | |
90 | #if defined LIBO_INTERNAL_ONLY1 |
91 | /** Up-casting conversion constructor: Copies interface reference. |
92 | |
93 | Does not work for up-casts to ambiguous bases. |
94 | |
95 | @param rRef another reference |
96 | */ |
97 | template< class derived_type > |
98 | inline Reference( |
99 | const Reference< derived_type > & rRef, |
100 | std::enable_if_t<std::is_base_of_v<reference_type, derived_type>, int> = 0 ) |
101 | : m_pBody (rRef.get()) |
102 | { |
103 | if (m_pBody) |
104 | m_pBody->acquire(); |
105 | } |
106 | #endif |
107 | |
108 | /** Destructor... |
109 | */ |
110 | ~Reference() COVERITY_NOEXCEPT_FALSE |
111 | { |
112 | if (m_pBody) |
113 | m_pBody->release(); |
114 | } |
115 | |
116 | /** Set... |
117 | Similar to assignment. |
118 | */ |
119 | Reference<reference_type> & |
120 | SAL_CALL set (reference_type * pBody) |
121 | { |
122 | if (pBody) |
123 | pBody->acquire(); |
124 | reference_type * const pOld = m_pBody; |
125 | m_pBody = pBody; |
126 | if (pOld) |
127 | pOld->release(); |
128 | return *this; |
129 | } |
130 | |
131 | /** Assignment. |
132 | Unbinds this instance from its body (if bound) and |
133 | bind it to the body represented by the handle. |
134 | */ |
135 | Reference<reference_type> & |
136 | SAL_CALL operator= (const Reference<reference_type> & handle) |
137 | { |
138 | return set( handle.m_pBody ); |
139 | } |
140 | |
141 | #ifdef LIBO_INTERNAL_ONLY1 |
142 | /** Assignment. |
143 | * Unbinds this instance from its body (if bound), |
144 | * bind it to the body represented by the handle, and |
145 | * set the body represented by the handle to nullptr. |
146 | */ |
147 | Reference<reference_type> & |
148 | operator= (Reference<reference_type> && handle) |
149 | { |
150 | // self-movement guts ourself |
151 | if (m_pBody) |
152 | m_pBody->release(); |
153 | m_pBody = handle.m_pBody; |
154 | handle.m_pBody = nullptr; |
155 | return *this; |
156 | } |
157 | #endif |
158 | |
159 | /** Assignment... |
160 | */ |
161 | Reference<reference_type> & |
162 | SAL_CALL operator= (reference_type * pBody) |
163 | { |
164 | return set( pBody ); |
165 | } |
166 | |
167 | /** Unbind the body from this handle. |
168 | Note that for a handle representing a large body, |
169 | "handle.clear().set(new body());" _might_ |
170 | perform a little bit better than "handle.set(new body());", |
171 | since in the second case two large objects exist in memory |
172 | (the old body and the new body). |
173 | */ |
174 | Reference<reference_type> & SAL_CALL clear() |
175 | { |
176 | if (m_pBody) |
177 | { |
178 | reference_type * const pOld = m_pBody; |
179 | m_pBody = NULL__null; |
180 | pOld->release(); |
181 | } |
182 | return *this; |
183 | } |
184 | |
185 | |
186 | /** Get the body. Can be used instead of operator->(). |
187 | I.e. handle->someBodyOp() and handle.get()->someBodyOp() |
188 | are the same. |
189 | */ |
190 | reference_type * SAL_CALL get() const |
191 | { |
192 | return m_pBody; |
193 | } |
194 | |
195 | |
196 | /** Probably most common used: handle->someBodyOp(). |
197 | */ |
198 | reference_type * SAL_CALL operator->() const |
199 | { |
200 | assert(m_pBody != NULL)(static_cast <bool> (m_pBody != __null) ? void (0) : __assert_fail ("m_pBody != NULL", "/home/maarten/src/libreoffice/core/include/rtl/ref.hxx" , 200, __extension__ __PRETTY_FUNCTION__)); |
201 | return m_pBody; |
202 | } |
203 | |
204 | |
205 | /** Allows (*handle).someBodyOp(). |
206 | */ |
207 | reference_type & SAL_CALL operator*() const |
208 | { |
209 | assert(m_pBody != NULL)(static_cast <bool> (m_pBody != __null) ? void (0) : __assert_fail ("m_pBody != NULL", "/home/maarten/src/libreoffice/core/include/rtl/ref.hxx" , 209, __extension__ __PRETTY_FUNCTION__)); |
210 | return *m_pBody; |
211 | } |
212 | |
213 | |
214 | /** Returns True if the handle does point to a valid body. |
215 | */ |
216 | bool SAL_CALL is() const |
217 | { |
218 | return (m_pBody != NULL__null); |
219 | } |
220 | |
221 | #if defined LIBO_INTERNAL_ONLY1 |
222 | /** Returns True if the handle does point to a valid body. |
223 | */ |
224 | explicit operator bool() const |
225 | { |
226 | return is(); |
227 | } |
228 | #endif |
229 | |
230 | /** Returns True if this points to pBody. |
231 | */ |
232 | bool SAL_CALL operator== (const reference_type * pBody) const |
233 | { |
234 | return (m_pBody == pBody); |
235 | } |
236 | |
237 | |
238 | /** Returns True if handle points to the same body. |
239 | */ |
240 | bool |
241 | SAL_CALL operator== (const Reference<reference_type> & handle) const |
242 | { |
243 | return (m_pBody == handle.m_pBody); |
244 | } |
245 | |
246 | |
247 | /** Needed to place References into STL collection. |
248 | */ |
249 | bool |
250 | SAL_CALL operator!= (const Reference<reference_type> & handle) const |
251 | { |
252 | return (m_pBody != handle.m_pBody); |
253 | } |
254 | |
255 | |
256 | /** Needed to place References into STL collection. |
257 | */ |
258 | bool |
259 | SAL_CALL operator< (const Reference<reference_type> & handle) const |
260 | { |
261 | return (m_pBody < handle.m_pBody); |
262 | } |
263 | |
264 | |
265 | /** Needed to place References into STL collection. |
266 | */ |
267 | bool |
268 | SAL_CALL operator> (const Reference<reference_type> & handle) const |
269 | { |
270 | return (m_pBody > handle.m_pBody); |
271 | } |
272 | }; |
273 | |
274 | } // namespace rtl |
275 | |
276 | #if defined LIBO_INTERNAL_ONLY1 |
277 | namespace std |
278 | { |
279 | |
280 | /// @cond INTERNAL |
281 | /** |
282 | Make rtl::Reference hashable by default for use in STL containers. |
283 | |
284 | @since LibreOffice 6.3 |
285 | */ |
286 | template<typename T> |
287 | struct hash<::rtl::Reference<T>> |
288 | { |
289 | std::size_t operator()(::rtl::Reference<T> const & s) const |
290 | { return std::size_t(s.get()); } |
291 | }; |
292 | /// @endcond |
293 | |
294 | } |
295 | |
296 | #endif |
297 | |
298 | #endif /* ! INCLUDED_RTL_REF_HXX */ |
299 | |
300 | /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |
1 | /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ |
2 | /* |
3 | * This file is part of the LibreOffice project. |
4 | * |
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/. |
8 | * |
9 | * This file incorporates work covered by the following license notice: |
10 | * |
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 . |
18 | */ |
19 | #ifndef INCLUDED_VCL_Reference_HXX |
20 | #define INCLUDED_VCL_Reference_HXX |
21 | |
22 | #include <vcl/dllapi.h> |
23 | #include <osl/interlck.h> |
24 | |
25 | class VCL_DLLPUBLIC__attribute__ ((visibility("default"))) VclReferenceBase |
26 | { |
27 | mutable oslInterlockedCount mnRefCnt; |
28 | |
29 | template<typename T> friend class VclPtr; |
30 | |
31 | public: |
32 | void acquire() const |
33 | { |
34 | osl_atomic_increment(&mnRefCnt)__sync_add_and_fetch((&mnRefCnt), 1); |
35 | } |
36 | |
37 | void release() const |
38 | { |
39 | if (osl_atomic_decrement(&mnRefCnt)__sync_sub_and_fetch((&mnRefCnt), 1) == 0) |
40 | delete this; |
41 | } |
42 | #ifdef DBG_UTIL |
43 | #ifndef _WIN32 |
44 | sal_Int32 getRefCount() const { return mnRefCnt; } |
45 | #endif |
46 | #endif |
47 | |
48 | |
49 | private: |
50 | VclReferenceBase(const VclReferenceBase&) = delete; |
51 | VclReferenceBase& operator=(const VclReferenceBase&) = delete; |
52 | |
53 | bool mbDisposed : 1; |
54 | |
55 | protected: |
56 | VclReferenceBase(); |
57 | protected: |
58 | virtual ~VclReferenceBase(); |
59 | |
60 | protected: |
61 | virtual void dispose(); |
62 | |
63 | public: |
64 | void disposeOnce(); |
65 | bool isDisposed() const { return mbDisposed; } |
66 | |
67 | }; |
68 | #endif |