sw: layout: fix bad split table resulting in empty cell
[LibreOffice.git] / sd / source / console / PresenterSlideShowView.cxx
blob956aabb3745a17b1b46a9d6fc1ebac2a8abe3ad6
2 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 /*
4 * This file is part of the LibreOffice project.
6 * This Source Code Form is subject to the terms of the Mozilla Public
7 * License, v. 2.0. If a copy of the MPL was not distributed with this
8 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
10 * This file incorporates work covered by the following license notice:
12 * Licensed to the Apache Software Foundation (ASF) under one or more
13 * contributor license agreements. See the NOTICE file distributed
14 * with this work for additional information regarding copyright
15 * ownership. The ASF licenses this file to you under the Apache
16 * License, Version 2.0 (the "License"); you may not use this file
17 * except in compliance with the License. You may obtain a copy of
18 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include "PresenterSlideShowView.hxx"
22 #include "PresenterCanvasHelper.hxx"
23 #include "PresenterGeometryHelper.hxx"
24 #include "PresenterHelper.hxx"
25 #include "PresenterPaneContainer.hxx"
26 #include <DrawController.hxx>
27 #include <com/sun/star/awt/InvalidateStyle.hpp>
28 #include <com/sun/star/awt/PosSize.hpp>
29 #include <com/sun/star/awt/Pointer.hpp>
30 #include <com/sun/star/awt/Toolkit.hpp>
31 #include <com/sun/star/awt/WindowAttribute.hpp>
32 #include <com/sun/star/awt/XWindow.hpp>
33 #include <com/sun/star/awt/XWindowPeer.hpp>
34 #include <com/sun/star/drawing/XPresenterHelper.hpp>
35 #include <com/sun/star/drawing/framework/XConfigurationController.hpp>
36 #include <com/sun/star/rendering/CompositeOperation.hpp>
37 #include <com/sun/star/rendering/TextDirection.hpp>
38 #include <com/sun/star/rendering/TexturingMode.hpp>
39 #include <osl/mutex.hxx>
40 #include <utility>
42 using namespace ::com::sun::star;
43 using namespace ::com::sun::star::uno;
44 using namespace ::com::sun::star::drawing::framework;
46 namespace sdext::presenter {
48 //===== PresenterSlideShowView ================================================
50 PresenterSlideShowView::PresenterSlideShowView (
51 css::uno::Reference<css::uno::XComponentContext> xContext,
52 css::uno::Reference<css::drawing::framework::XResourceId> xViewId,
53 const rtl::Reference<::sd::DrawController>& rxController,
54 ::rtl::Reference<PresenterController> xPresenterController)
55 : PresenterSlideShowViewInterfaceBase(m_aMutex),
56 mxComponentContext(std::move(xContext)),
57 mpPresenterController(std::move(xPresenterController)),
58 mxViewId(std::move(xViewId)),
59 mxController(rxController),
60 mxSlideShowController(PresenterHelper::GetSlideShowController(rxController)),
61 mbIsViewAdded(false),
62 mnPageAspectRatio(28.0/21.0),
63 maBroadcaster(m_aMutex),
64 mbIsForcedPaintPending(false),
65 mbIsPaintPending(true),
66 mbIsEndSlideVisible(false)
68 if (mpPresenterController)
70 mnPageAspectRatio = mpPresenterController->GetSlideAspectRatio();
71 mpBackground = mpPresenterController->GetViewBackground(mxViewId->getResourceURL());
75 void PresenterSlideShowView::LateInit()
77 mxSlideShow.set( mxSlideShowController->getSlideShow(), UNO_SET_THROW);
78 Reference<lang::XComponent> xSlideShowComponent (mxSlideShow, UNO_QUERY);
79 xSlideShowComponent->addEventListener(static_cast<awt::XWindowListener*>(this));
81 Reference<lang::XMultiComponentFactory> xFactory (
82 mxComponentContext->getServiceManager(), UNO_SET_THROW);
83 mxPresenterHelper.set (xFactory->createInstanceWithContext(
84 u"com.sun.star.comp.Draw.PresenterHelper"_ustr,
85 mxComponentContext),
86 UNO_QUERY_THROW);
88 // Use view id and controller to retrieve window and canvas from
89 // configuration controller.
90 Reference<XConfigurationController> xCC (mxController->getConfigurationController());
92 if (xCC.is())
94 mxTopPane.set(xCC->getResource(mxViewId->getAnchor()->getAnchor()), UNO_QUERY);
96 Reference<XPane> xPane (xCC->getResource(mxViewId->getAnchor()), UNO_QUERY_THROW);
98 mxWindow = xPane->getWindow();
99 mxCanvas = xPane->getCanvas();
101 if (mxWindow.is())
103 mxWindow->addPaintListener(this);
104 mxWindow->addWindowListener(this);
107 // The window does not have to paint a background. We do
108 // that ourself.
109 Reference<awt::XWindowPeer> xPeer (mxWindow, UNO_QUERY);
110 if (xPeer.is())
111 xPeer->setBackground(util::Color(0xff000000));
114 // Create a window for the actual slide show view. It is places
115 // centered and with maximal size inside the pane.
116 mxViewWindow = CreateViewWindow(mxWindow);
118 mxViewCanvas = CreateViewCanvas(mxViewWindow);
120 if (mxViewWindow.is())
122 // Register listeners at window.
123 mxViewWindow->addPaintListener(this);
124 mxViewWindow->addMouseListener(this);
125 mxViewWindow->addMouseMotionListener(this);
128 if (mxViewWindow.is())
129 Resize();
131 if (mxWindow.is())
132 mxWindow->setVisible(true);
134 // Add the new slide show view to the slide show.
135 if (mxSlideShow.is() && ! mbIsViewAdded)
137 impl_addAndConfigureView();
138 mbIsViewAdded = true;
141 // Read text for one past last slide.
142 PresenterConfigurationAccess aConfiguration (
143 mxComponentContext,
144 PresenterConfigurationAccess::msPresenterScreenRootName,
145 PresenterConfigurationAccess::READ_ONLY);
146 aConfiguration.GetConfigurationNode(
147 u"Presenter/Views/CurrentSlidePreview/"
148 "Strings/ClickToExitPresentationText/String"_ustr)
149 >>= msClickToExitPresentationText;
150 aConfiguration.GetConfigurationNode(
151 u"Presenter/Views/CurrentSlidePreview/"
152 "Strings/ClickToExitPresentationTitle/String"_ustr)
153 >>= msClickToExitPresentationTitle;
156 PresenterSlideShowView::~PresenterSlideShowView()
160 void PresenterSlideShowView::disposing()
162 // Tell all listeners that we are disposed.
163 lang::EventObject aEvent;
164 aEvent.Source = static_cast<XWeak*>(this);
166 ::cppu::OInterfaceContainerHelper* pIterator
167 = maBroadcaster.getContainer(cppu::UnoType<lang::XEventListener>::get());
168 if (pIterator != nullptr)
169 pIterator->disposeAndClear(aEvent);
171 // Do this for
172 // XPaintListener, XModifyListener,XMouseListener,XMouseMotionListener,XWindowListener?
174 if (mxWindow.is())
176 mxWindow->removePaintListener(this);
177 mxWindow->removeMouseListener(this);
178 mxWindow->removeMouseMotionListener(this);
179 mxWindow->removeWindowListener(this);
180 mxWindow = nullptr;
182 mxSlideShowController = nullptr;
183 mxSlideShow = nullptr;
184 if (mxViewCanvas.is())
186 Reference<XComponent> xComponent (mxViewCanvas, UNO_QUERY);
187 mxViewCanvas = nullptr;
188 if (xComponent.is())
189 xComponent->dispose();
191 if (mxViewWindow.is())
193 Reference<XComponent> xComponent = mxViewWindow;
194 mxViewWindow = nullptr;
195 if (xComponent.is())
196 xComponent->dispose();
198 if (mxPointer.is())
200 Reference<XComponent> xComponent (mxPointer, UNO_QUERY);
201 mxPointer = nullptr;
202 if (xComponent.is())
203 xComponent->dispose();
205 if (mxBackgroundPolygon1.is())
207 Reference<XComponent> xComponent (mxBackgroundPolygon1, UNO_QUERY);
208 mxBackgroundPolygon1 = nullptr;
209 if (xComponent.is())
210 xComponent->dispose();
212 if (mxBackgroundPolygon2.is())
214 Reference<XComponent> xComponent (mxBackgroundPolygon2, UNO_QUERY);
215 mxBackgroundPolygon2 = nullptr;
216 if (xComponent.is())
217 xComponent->dispose();
220 mxComponentContext = nullptr;
221 mpPresenterController = nullptr;
222 mxViewId = nullptr;
223 mxController = nullptr;
224 mxCanvas = nullptr;
225 mpBackground.reset();
226 msClickToExitPresentationText.clear();
227 msClickToExitPresentationTitle.clear();
228 msTitleTemplate.clear();
229 mxCurrentSlide = nullptr;
232 //----- XDrawView -------------------------------------------------------------
234 void SAL_CALL PresenterSlideShowView::setCurrentPage (
235 const css::uno::Reference<css::drawing::XDrawPage>& rxSlide)
237 mxCurrentSlide = rxSlide;
238 if (mpPresenterController
239 && mxSlideShowController.is()
240 && ! mpPresenterController->GetCurrentSlide().is()
241 && ! mxSlideShowController->isPaused())
243 mbIsEndSlideVisible = true;
244 Reference<awt::XWindowPeer> xPeer (mxViewWindow, UNO_QUERY);
245 if (xPeer.is())
246 xPeer->invalidate(awt::InvalidateStyle::NOTRANSPARENT);
248 // For the end slide we use a special title, without the (n of m)
249 // part. Save the title template for the case that the user goes
250 // backwards.
251 PresenterPaneContainer::SharedPaneDescriptor pDescriptor (
252 mpPresenterController->GetPaneContainer()->FindViewURL(mxViewId->getResourceURL()));
253 if (pDescriptor)
255 msTitleTemplate = pDescriptor->msTitleTemplate;
256 pDescriptor->msTitleTemplate = msClickToExitPresentationTitle;
257 mpPresenterController->UpdatePaneTitles();
260 else if (mbIsEndSlideVisible)
262 mbIsEndSlideVisible = false;
264 // Restore the title template.
265 PresenterPaneContainer::SharedPaneDescriptor pDescriptor (
266 mpPresenterController->GetPaneContainer()->FindViewURL(mxViewId->getResourceURL()));
267 if (pDescriptor)
269 pDescriptor->msTitleTemplate = msTitleTemplate;
270 pDescriptor->msTitle.clear();
271 mpPresenterController->UpdatePaneTitles();
276 css::uno::Reference<css::drawing::XDrawPage> SAL_CALL PresenterSlideShowView::getCurrentPage()
278 return mxCurrentSlide;
281 //----- CachablePresenterView -------------------------------------------------
283 void PresenterSlideShowView::ReleaseView()
285 if (mxSlideShow.is() && mbIsViewAdded)
287 mxSlideShow->removeView(this);
288 mbIsViewAdded = false;
292 //----- XSlideShowView --------------------------------------------------------
294 Reference<rendering::XSpriteCanvas> SAL_CALL PresenterSlideShowView::getCanvas()
296 ThrowIfDisposed();
298 return Reference<rendering::XSpriteCanvas>(mxViewCanvas, UNO_QUERY);
301 void SAL_CALL PresenterSlideShowView::clear()
303 ThrowIfDisposed();
304 mbIsForcedPaintPending = false;
305 mbIsPaintPending = false;
307 if (!(mxViewCanvas.is() && mxViewWindow.is()))
308 return;
310 // Create a polygon for the window outline.
311 awt::Rectangle aViewWindowBox (mxViewWindow->getPosSize());
312 Reference<rendering::XPolyPolygon2D> xPolygon (PresenterGeometryHelper::CreatePolygon(
313 awt::Rectangle(0,0, aViewWindowBox.Width,aViewWindowBox.Height),
314 mxViewCanvas->getDevice()));
316 rendering::ViewState aViewState (
317 geometry::AffineMatrix2D(1,0,0, 0,1,0),
318 nullptr);
319 double const aColor[4] = {0,0,0,0};
320 rendering::RenderState aRenderState(
321 geometry::AffineMatrix2D(1,0,0, 0,1,0),
322 nullptr,
323 Sequence<double>(aColor,4),
324 rendering::CompositeOperation::SOURCE);
325 mxViewCanvas->fillPolyPolygon(xPolygon, aViewState, aRenderState);
328 geometry::AffineMatrix2D SAL_CALL PresenterSlideShowView::getTransformation()
330 ThrowIfDisposed();
332 if (mxViewWindow.is())
334 // When the mbIsInModifyNotification is set then a slightly modified
335 // version of the transformation is returned in order to get past
336 // optimizations the avoid updates when the transformation is
337 // unchanged (when the window size changes then due to the constant
338 // aspect ratio the size of the preview may remain the same while
339 // the position changes. The position, however, is represented by
340 // the position of the view window. This transformation is given
341 // relative to the view window and therefore does not contain the
342 // position.)
343 const awt::Rectangle aWindowBox = mxViewWindow->getPosSize();
344 return geometry::AffineMatrix2D(
345 aWindowBox.Width-1, 0, 0,
346 0, aWindowBox.Height-1, 0);
348 else
350 return geometry::AffineMatrix2D(1,0,0, 0,1,0);
354 geometry::IntegerSize2D SAL_CALL PresenterSlideShowView::getTranslationOffset()
356 ThrowIfDisposed();
357 return geometry::IntegerSize2D(0,0);
360 void SAL_CALL PresenterSlideShowView::addTransformationChangedListener(
361 const Reference<util::XModifyListener>& rxListener)
363 ThrowIfDisposed();
364 maBroadcaster.addListener(
365 cppu::UnoType<util::XModifyListener>::get(),
366 rxListener);
369 void SAL_CALL PresenterSlideShowView::removeTransformationChangedListener(
370 const Reference<util::XModifyListener>& rxListener)
372 ThrowIfDisposed();
373 maBroadcaster.removeListener(
374 cppu::UnoType<util::XModifyListener>::get(),
375 rxListener);
378 void SAL_CALL PresenterSlideShowView::addPaintListener(
379 const Reference<awt::XPaintListener>& rxListener)
381 ThrowIfDisposed();
382 maBroadcaster.addListener(
383 cppu::UnoType<awt::XPaintListener>::get(),
384 rxListener);
387 void SAL_CALL PresenterSlideShowView::removePaintListener(
388 const Reference<awt::XPaintListener>& rxListener)
390 ThrowIfDisposed();
391 maBroadcaster.removeListener(
392 cppu::UnoType<awt::XPaintListener>::get(),
393 rxListener);
396 void SAL_CALL PresenterSlideShowView::addMouseListener(
397 const Reference<awt::XMouseListener>& rxListener)
399 ThrowIfDisposed();
400 maBroadcaster.addListener(
401 cppu::UnoType<awt::XMouseListener>::get(),
402 rxListener);
405 void SAL_CALL PresenterSlideShowView::removeMouseListener(
406 const Reference<awt::XMouseListener>& rxListener)
408 ThrowIfDisposed();
409 maBroadcaster.removeListener(
410 cppu::UnoType<awt::XMouseListener>::get(),
411 rxListener);
414 void SAL_CALL PresenterSlideShowView::addMouseMotionListener(
415 const Reference<awt::XMouseMotionListener>& rxListener)
417 ThrowIfDisposed();
418 maBroadcaster.addListener(
419 cppu::UnoType<awt::XMouseMotionListener>::get(),
420 rxListener);
423 void SAL_CALL PresenterSlideShowView::removeMouseMotionListener(
424 const Reference<awt::XMouseMotionListener>& rxListener)
426 ThrowIfDisposed();
427 maBroadcaster.removeListener(
428 cppu::UnoType<awt::XMouseMotionListener>::get(),
429 rxListener);
432 void SAL_CALL PresenterSlideShowView::setMouseCursor(::sal_Int16 nPointerShape)
434 ThrowIfDisposed();
436 // Create a pointer when it does not yet exist.
437 if ( ! mxPointer.is())
439 mxPointer = awt::Pointer::create(mxComponentContext);
442 // Set the pointer to the given shape and the window(peer) to the
443 // pointer.
444 Reference<awt::XWindowPeer> xPeer (mxViewWindow, UNO_QUERY);
445 if (mxPointer.is() && xPeer.is())
447 mxPointer->setType(nPointerShape);
448 xPeer->setPointer(mxPointer);
452 awt::Rectangle SAL_CALL PresenterSlideShowView::getCanvasArea( )
454 if( mxViewWindow.is() && mxTopPane.is() )
455 return mxPresenterHelper->getWindowExtentsRelative( mxViewWindow, mxTopPane->getWindow() );
457 awt::Rectangle aRectangle;
459 aRectangle.X = aRectangle.Y = aRectangle.Width = aRectangle.Height = 0;
461 return aRectangle;
464 //----- lang::XEventListener --------------------------------------------------
466 void SAL_CALL PresenterSlideShowView::disposing (const lang::EventObject& rEvent)
468 if (rEvent.Source == mxViewWindow)
469 mxViewWindow = nullptr;
470 else if (rEvent.Source == mxSlideShow)
471 mxSlideShow = nullptr;
474 //----- XPaintListener --------------------------------------------------------
476 void SAL_CALL PresenterSlideShowView::windowPaint (const awt::PaintEvent& rEvent)
478 // Deactivated views must not be painted.
479 if ( ! mbIsPresenterViewActive)
480 return;
482 awt::Rectangle aViewWindowBox (mxViewWindow->getPosSize());
483 if (aViewWindowBox.Width <= 0 || aViewWindowBox.Height <= 0)
484 return;
486 if (rEvent.Source == mxWindow)
487 PaintOuterWindow(rEvent.UpdateRect);
488 else if (mbIsEndSlideVisible)
489 PaintEndSlide(rEvent.UpdateRect);
490 else
491 PaintInnerWindow(rEvent);
494 //----- XMouseListener --------------------------------------------------------
496 void SAL_CALL PresenterSlideShowView::mousePressed (const awt::MouseEvent& rEvent)
498 awt::MouseEvent aEvent (rEvent);
499 aEvent.Source = static_cast<XWeak*>(this);
500 ::cppu::OInterfaceContainerHelper* pIterator
501 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseListener>::get());
502 if (pIterator != nullptr)
504 pIterator->notifyEach(&awt::XMouseListener::mousePressed, aEvent);
507 // Only when the end slide is displayed we forward the mouse event to
508 // the PresenterController so that it switches to the next slide and
509 // ends the presentation.
510 if (mbIsEndSlideVisible)
511 if (mpPresenterController)
512 mpPresenterController->HandleMouseClick(rEvent);
515 void SAL_CALL PresenterSlideShowView::mouseReleased (const awt::MouseEvent& rEvent)
517 awt::MouseEvent aEvent (rEvent);
518 aEvent.Source = static_cast<XWeak*>(this);
519 ::cppu::OInterfaceContainerHelper* pIterator
520 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseListener>::get());
521 if (pIterator != nullptr)
523 pIterator->notifyEach(&awt::XMouseListener::mouseReleased, aEvent);
527 void SAL_CALL PresenterSlideShowView::mouseEntered (const awt::MouseEvent& rEvent)
529 awt::MouseEvent aEvent (rEvent);
530 aEvent.Source = static_cast<XWeak*>(this);
531 ::cppu::OInterfaceContainerHelper* pIterator
532 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseListener>::get());
533 if (pIterator != nullptr)
535 pIterator->notifyEach(&awt::XMouseListener::mouseEntered, aEvent);
539 void SAL_CALL PresenterSlideShowView::mouseExited (const awt::MouseEvent& rEvent)
541 awt::MouseEvent aEvent (rEvent);
542 aEvent.Source = static_cast<XWeak*>(this);
543 ::cppu::OInterfaceContainerHelper* pIterator
544 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseListener>::get());
545 if (pIterator != nullptr)
547 pIterator->notifyEach(&awt::XMouseListener::mouseExited, aEvent);
551 //----- XMouseMotionListener --------------------------------------------------
553 void SAL_CALL PresenterSlideShowView::mouseDragged (const awt::MouseEvent& rEvent)
555 awt::MouseEvent aEvent (rEvent);
556 aEvent.Source = static_cast<XWeak*>(this);
557 ::cppu::OInterfaceContainerHelper* pIterator
558 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseMotionListener>::get());
559 if (pIterator != nullptr)
561 pIterator->notifyEach(&awt::XMouseMotionListener::mouseDragged, aEvent);
565 void SAL_CALL PresenterSlideShowView::mouseMoved (const awt::MouseEvent& rEvent)
567 awt::MouseEvent aEvent (rEvent);
568 aEvent.Source = static_cast<XWeak*>(this);
569 ::cppu::OInterfaceContainerHelper* pIterator
570 = maBroadcaster.getContainer(cppu::UnoType<awt::XMouseMotionListener>::get());
571 if (pIterator != nullptr)
573 pIterator->notifyEach(&awt::XMouseMotionListener::mouseMoved, aEvent);
577 //----- XWindowListener -------------------------------------------------------
579 void SAL_CALL PresenterSlideShowView::windowResized (const awt::WindowEvent&)
581 ThrowIfDisposed();
582 ::osl::MutexGuard aGuard (::osl::Mutex::getGlobalMutex());
584 Resize();
587 void SAL_CALL PresenterSlideShowView::windowMoved (const awt::WindowEvent&)
589 if ( ! mbIsPaintPending)
590 mbIsForcedPaintPending = true;
593 void SAL_CALL PresenterSlideShowView::windowShown (const lang::EventObject&)
595 Resize();
598 void SAL_CALL PresenterSlideShowView::windowHidden (const lang::EventObject&) {}
600 //----- XView -----------------------------------------------------------------
602 Reference<XResourceId> SAL_CALL PresenterSlideShowView::getResourceId()
604 return mxViewId;
607 sal_Bool SAL_CALL PresenterSlideShowView::isAnchorOnly()
609 return false;
612 //----- CachablePresenterView -------------------------------------------------
614 void PresenterSlideShowView::ActivatePresenterView()
616 if (mxSlideShow.is() && ! mbIsViewAdded)
618 impl_addAndConfigureView();
619 mbIsViewAdded = true;
623 void PresenterSlideShowView::DeactivatePresenterView()
625 if (mxSlideShow.is() && mbIsViewAdded)
627 mxSlideShow->removeView(this);
628 mbIsViewAdded = false;
633 void PresenterSlideShowView::PaintOuterWindow (const awt::Rectangle& rRepaintBox)
635 if ( ! mxCanvas.is())
636 return;
638 if (!mpBackground)
639 return;
641 const rendering::ViewState aViewState(
642 geometry::AffineMatrix2D(1,0,0, 0,1,0),
643 PresenterGeometryHelper::CreatePolygon(rRepaintBox, mxCanvas->getDevice()));
645 rendering::RenderState aRenderState (
646 geometry::AffineMatrix2D(1,0,0, 0,1,0),
647 nullptr,
648 Sequence<double>(4),
649 rendering::CompositeOperation::SOURCE);
651 Reference<rendering::XBitmap> xBackgroundBitmap (mpBackground->GetNormalBitmap());
652 if (xBackgroundBitmap.is())
654 const geometry::IntegerSize2D aBitmapSize(xBackgroundBitmap->getSize());
655 Sequence<rendering::Texture> aTextures
658 geometry::AffineMatrix2D( aBitmapSize.Width,0,0, 0,aBitmapSize.Height,0),
661 xBackgroundBitmap,
662 nullptr,
663 nullptr,
664 rendering::StrokeAttributes(),
665 rendering::TexturingMode::REPEAT,
666 rendering::TexturingMode::REPEAT
670 if (mxBackgroundPolygon1.is())
671 mxCanvas->fillTexturedPolyPolygon(
672 mxBackgroundPolygon1,
673 aViewState,
674 aRenderState,
675 aTextures);
676 if (mxBackgroundPolygon2.is())
677 mxCanvas->fillTexturedPolyPolygon(
678 mxBackgroundPolygon2,
679 aViewState,
680 aRenderState,
681 aTextures);
683 else
685 PresenterCanvasHelper::SetDeviceColor(aRenderState, mpBackground->maReplacementColor);
687 if (mxBackgroundPolygon1.is())
688 mxCanvas->fillPolyPolygon(mxBackgroundPolygon1, aViewState, aRenderState);
689 if (mxBackgroundPolygon2.is())
690 mxCanvas->fillPolyPolygon(mxBackgroundPolygon2, aViewState, aRenderState);
694 void PresenterSlideShowView::PaintEndSlide (const awt::Rectangle& rRepaintBox)
696 if ( ! mxCanvas.is())
697 return;
699 const rendering::ViewState aViewState(
700 geometry::AffineMatrix2D(1,0,0, 0,1,0),
701 PresenterGeometryHelper::CreatePolygon(rRepaintBox, mxCanvas->getDevice()));
703 rendering::RenderState aRenderState (
704 geometry::AffineMatrix2D(1,0,0, 0,1,0),
705 nullptr,
706 Sequence<double>(4),
707 rendering::CompositeOperation::SOURCE);
708 PresenterCanvasHelper::SetDeviceColor(aRenderState, util::Color(0x00000000));
709 mxCanvas->fillPolyPolygon(
710 PresenterGeometryHelper::CreatePolygon(mxViewWindow->getPosSize(), mxCanvas->getDevice()),
711 aViewState,
712 aRenderState);
716 if (!mpPresenterController)
717 break;
718 std::shared_ptr<PresenterTheme> pTheme (mpPresenterController->GetTheme());
719 if (pTheme == nullptr)
720 break;
722 const OUString sViewStyle (pTheme->GetStyleName(mxViewId->getResourceURL()));
723 PresenterTheme::SharedFontDescriptor pFont (pTheme->GetFont(sViewStyle));
724 if (!pFont)
725 break;
727 /// this is responsible of the " presentation exit " text inside the slide windows
728 PresenterCanvasHelper::SetDeviceColor(aRenderState, util::Color(0x00ffffff));
729 aRenderState.AffineTransform.m02 = 20;
730 aRenderState.AffineTransform.m12 = 40;
731 const rendering::StringContext aContext (
732 msClickToExitPresentationText, 0, msClickToExitPresentationText.getLength());
733 pFont->PrepareFont(mxCanvas);
734 const Reference<rendering::XTextLayout> xLayout (
735 pFont->mxFont->createTextLayout(aContext,rendering::TextDirection::WEAK_LEFT_TO_RIGHT,0));
736 mxCanvas->drawTextLayout(
737 xLayout,
738 aViewState,
739 aRenderState);
741 while (false);
743 // Finally, in double buffered environments, request the changes to be
744 // made visible.
745 Reference<rendering::XSpriteCanvas> xSpriteCanvas (mxCanvas, UNO_QUERY);
746 if (xSpriteCanvas.is())
747 xSpriteCanvas->updateScreen(true);
750 void PresenterSlideShowView::PaintInnerWindow (const awt::PaintEvent& rEvent)
752 // Forward window paint to listeners.
753 awt::PaintEvent aEvent (rEvent);
754 aEvent.Source = static_cast<XWeak*>(this);
755 ::cppu::OInterfaceContainerHelper* pIterator
756 = maBroadcaster.getContainer(cppu::UnoType<awt::XPaintListener>::get());
757 if (pIterator != nullptr)
759 pIterator->notifyEach(&awt::XPaintListener::windowPaint, aEvent);
762 /** The slide show relies on the back buffer of the canvas not being
763 modified. With a shared canvas there are times when that can not be
764 guaranteed.
766 if (mbIsForcedPaintPending && mxSlideShow.is() && mbIsViewAdded)
768 mxSlideShow->removeView(this);
769 impl_addAndConfigureView();
772 // Finally, in double buffered environments, request the changes to be
773 // made visible.
774 Reference<rendering::XSpriteCanvas> xSpriteCanvas (mxCanvas, UNO_QUERY);
775 if (xSpriteCanvas.is())
776 xSpriteCanvas->updateScreen(true);
779 Reference<awt::XWindow> PresenterSlideShowView::CreateViewWindow (
780 const Reference<awt::XWindow>& rxParentWindow) const
782 Reference<awt::XWindow> xViewWindow;
785 Reference<lang::XMultiComponentFactory> xFactory (mxComponentContext->getServiceManager());
786 if ( ! xFactory.is())
787 return xViewWindow;
789 Reference<awt::XToolkit2> xToolkit = awt::Toolkit::create(mxComponentContext);
790 awt::WindowDescriptor aWindowDescriptor (
791 awt::WindowClass_CONTAINER,
792 OUString(),
793 Reference<awt::XWindowPeer>(rxParentWindow,UNO_QUERY_THROW),
794 -1, // parent index not available
795 awt::Rectangle(0,0,10,10),
796 awt::WindowAttribute::SIZEABLE
797 | awt::WindowAttribute::MOVEABLE
798 | awt::WindowAttribute::NODECORATION);
799 xViewWindow.set( xToolkit->createWindow(aWindowDescriptor),UNO_QUERY_THROW);
801 // Make the background transparent. The slide show paints its own background.
802 Reference<awt::XWindowPeer> xPeer (xViewWindow, UNO_QUERY_THROW);
803 xPeer->setBackground(0xff000000);
805 xViewWindow->setVisible(true);
807 catch (RuntimeException&)
810 return xViewWindow;
813 Reference<rendering::XCanvas> PresenterSlideShowView::CreateViewCanvas (
814 const Reference<awt::XWindow>& rxViewWindow) const
816 // Create a canvas for the view window.
817 return mxPresenterHelper->createSharedCanvas(
818 Reference<rendering::XSpriteCanvas>(mxTopPane->getCanvas(), UNO_QUERY),
819 mxTopPane->getWindow(),
820 mxTopPane->getCanvas(),
821 mxTopPane->getWindow(),
822 rxViewWindow);
825 void PresenterSlideShowView::Resize()
827 if ( ! mxWindow.is() || ! mxViewWindow.is())
828 return;
830 const awt::Rectangle aWindowBox (mxWindow->getPosSize());
831 if (aWindowBox.Height > 0)
833 awt::Rectangle aViewWindowBox;
834 const double nWindowAspectRatio (
835 double(aWindowBox.Width) / double(aWindowBox.Height));
836 if (nWindowAspectRatio > mnPageAspectRatio)
838 // Slides will be painted with the full parent window height.
839 aViewWindowBox.Width = sal_Int32(aWindowBox.Height * mnPageAspectRatio + 0.5);
840 aViewWindowBox.Height = aWindowBox.Height;
841 aViewWindowBox.X = (aWindowBox.Width - aViewWindowBox.Width) / 2;
842 aViewWindowBox.Y = 0;
844 else
846 // Slides will be painted with the full parent window width.
847 aViewWindowBox.Width = aWindowBox.Width;
848 aViewWindowBox.Height = sal_Int32(aWindowBox.Width / mnPageAspectRatio + 0.5);
849 aViewWindowBox.X = 0;
850 aViewWindowBox.Y = (aWindowBox.Height - aViewWindowBox.Height) / 2;
852 mxViewWindow->setPosSize(
853 aViewWindowBox.X,
854 aViewWindowBox.Y,
855 aViewWindowBox.Width,
856 aViewWindowBox.Height,
857 awt::PosSize::POSSIZE);
860 // Clear the background polygon so that on the next paint it is created
861 // for the new size.
862 CreateBackgroundPolygons();
864 // Notify listeners that the transformation that maps the view into the
865 // window has changed.
866 lang::EventObject aEvent (static_cast<XWeak*>(this));
867 ::cppu::OInterfaceContainerHelper* pIterator
868 = maBroadcaster.getContainer(cppu::UnoType<util::XModifyListener>::get());
869 if (pIterator != nullptr)
871 pIterator->notifyEach(&util::XModifyListener::modified, aEvent);
874 // Due to constant aspect ratio resizing may lead a preview that changes
875 // its position but not its size. This invalidates the back buffer and
876 // we have to enforce a complete repaint.
877 if ( ! mbIsPaintPending)
878 mbIsForcedPaintPending = true;
881 void PresenterSlideShowView::CreateBackgroundPolygons()
883 const awt::Rectangle aWindowBox (mxWindow->getPosSize());
884 const awt::Rectangle aViewWindowBox (mxViewWindow->getPosSize());
885 if (aWindowBox.Height == aViewWindowBox.Height && aWindowBox.Width == aViewWindowBox.Width)
887 mxBackgroundPolygon1 = nullptr;
888 mxBackgroundPolygon2 = nullptr;
890 else if (aWindowBox.Height == aViewWindowBox.Height)
892 // Paint two boxes to the left and right of the view window.
893 mxBackgroundPolygon1 = PresenterGeometryHelper::CreatePolygon(
894 awt::Rectangle(
897 aViewWindowBox.X,
898 aWindowBox.Height),
899 mxCanvas->getDevice());
900 mxBackgroundPolygon2 = PresenterGeometryHelper::CreatePolygon(
901 awt::Rectangle(
902 aViewWindowBox.X + aViewWindowBox.Width,
904 aWindowBox.Width - aViewWindowBox.X - aViewWindowBox.Width,
905 aWindowBox.Height),
906 mxCanvas->getDevice());
908 else
910 // Paint two boxes above and below the view window.
911 mxBackgroundPolygon1 = PresenterGeometryHelper::CreatePolygon(
912 awt::Rectangle(
915 aWindowBox.Width,
916 aViewWindowBox.Y),
917 mxCanvas->getDevice());
918 mxBackgroundPolygon2 = PresenterGeometryHelper::CreatePolygon(
919 awt::Rectangle(
921 aViewWindowBox.Y + aViewWindowBox.Height,
922 aWindowBox.Width,
923 aWindowBox.Height - aViewWindowBox.Y - aViewWindowBox.Height),
924 mxCanvas->getDevice());
928 void PresenterSlideShowView::ThrowIfDisposed()
930 if (rBHelper.bDisposed || rBHelper.bInDispose)
932 throw lang::DisposedException (
933 u"PresenterSlideShowView object has already been disposed"_ustr,
934 static_cast<uno::XWeak*>(this));
938 void PresenterSlideShowView::impl_addAndConfigureView()
940 Reference<presentation::XSlideShowView> xView (this);
941 mxSlideShow->addView(xView);
942 // Prevent embedded sounds being played twice at the same time by
943 // disabling sound for the new slide show view.
944 beans::PropertyValue aProperty;
945 aProperty.Name = "IsSoundEnabled";
946 Sequence<Any> aValues{ Any(xView), Any(false) };
947 aProperty.Value <<= aValues;
948 mxSlideShow->setProperty(aProperty);
951 } // end of namespace ::sd::presenter
953 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */