1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #ifndef INCLUDED_SLIDESHOW_LAYERMANAGER_HXX
25 #define INCLUDED_SLIDESHOW_LAYERMANAGER_HXX
26 
27 #include <boost/shared_ptr.hpp>
28 #include <boost/noncopyable.hpp>
29 
30 #include <cppcanvas/spritecanvas.hxx>
31 
32 #include "unoview.hxx"
33 #include "unoviewcontainer.hxx"
34 #include "attributableshape.hxx"
35 #include "layer.hxx"
36 #include "tools.hxx"
37 
38 #include <vector>
39 #include <map>
40 #include <hash_map>
41 #include <algorithm>
42 #include <functional>
43 
44 namespace basegfx {
45     class B2DRange;
46 }
47 
48 namespace slideshow
49 {
50     namespace internal
51     {
52 		/* Definition of Layermanager class */
53 
54         /** This class manages all of a slide's layers (and shapes)
55 
56 			Since layer content changes when animations start or end,
57 			the layer manager keeps track of this and also handles
58 			starting/stopping of Shape animations. Note that none of
59 			the methods actually perform a screen update, this is
60 			always delayed until the ActivitiesQueue explicitely
61 			performs it.
62 
63             @see Layer
64             @see Shape
65          */
66         class LayerManager : private boost::noncopyable
67         {
68         public:
69             /** Create a new layer manager for the given page bounds
70 
71                 @param rViews
72                 Views currently registered
73 
74                 @param rPageBounds
75                 Overall page bounds, in user space coordinates
76 
77                 @param bDisableAnimationZOrder
78                 When true, all sprite animations run in the
79                 foreground.  That is, no extra layers are created, and
80                 the slideshow runs potentially faster.
81              */
82             LayerManager( const UnoViewContainer&    rViews,
83                           const ::basegfx::B2DRange& rPageBounds,
84                           bool                       bDisableAnimationZOrder );
85 
86             /** Activate the LayerManager
87 
88                 This method activates the LayerManager. Prior to
89                 activation, this instance will be passive, i.e. won't
90                 render anything to any view.
91 
92                 @param bSlideBackgoundPainted
93                 When true, the initial slide content on the background
94                 layer is already rendered (e.g. from a previous slide
95                 transition). When false, LayerManager also renders
96                 initial content of background layer on next update()
97                 call.
98              */
99             void activate( bool bSlideBackgoundPainted );
100 
101             /** Deactivate the LayerManager
102 
103                 This method deactivates the LayerManager. After
104                 deactivation, this instance will be passive,
105                 i.e. don't render anything to any view.  Furthermore,
106                 if there's currently more than one Layer active, this
107                 method also removes all but one.
108              */
109             void deactivate();
110 
111             // From ViewEventHandler, forwarded by SlideImpl
112             /// Notify new view added to UnoViewContainer
113             void viewAdded( const UnoViewSharedPtr& rView );
114             /// Notify view removed from UnoViewContainer
115             void viewRemoved( const UnoViewSharedPtr& rView );
116             void viewChanged( const UnoViewSharedPtr& rView );
117             void viewsChanged();
118 
119             /** Add the shape to this object
120 
121             	This method adds a shape to the page.
122              */
123             void addShape( const ShapeSharedPtr& rShape );
124 
125             /** Remove shape from this object
126 
127             	This method removes a shape from the shape.
128              */
129             bool removeShape( const ShapeSharedPtr& rShape );
130 
131             /** Lookup a Shape from an XShape model object
132 
133             	This method looks up the internal shape map for one
134             	representing the given XShape.
135 
136                 @param xShape
137                 The XShape object, for which the representing Shape
138                 should be looked up.
139              */
140             ShapeSharedPtr lookupShape( const ::com::sun::star::uno::Reference<
141                                            ::com::sun::star::drawing::XShape >& xShape ) const;
142 
143             /** Query a subset of the given original shape
144 
145 				This method queries a new (but not necessarily unique)
146 				shape, which displays only the given subset of the
147 				original one.
148              */
149             AttributableShapeSharedPtr getSubsetShape( const AttributableShapeSharedPtr& 	rOrigShape,
150                                                        const DocTreeNode&					rTreeNode );
151 
152             /** Revoke a previously queried subset shape.
153 
154             	With this method, a previously requested subset shape
155             	is revoked again. If the last client revokes a given
156             	subset, it will cease to be displayed, and the
157             	original shape will again show the subset data.
158 
159                 @param rOrigShape
160                 The shape the subset was created from
161 
162                 @param rSubsetShape
163                 The subset created from rOrigShape
164              */
165             void revokeSubset( const AttributableShapeSharedPtr& rOrigShape,
166                                const AttributableShapeSharedPtr& rSubsetShape );
167 
168             /** Notify the LayerManager that the given Shape starts an
169                 animation now.
170 
171 				This method enters animation mode for the Shape on all
172 				registered views.
173              */
174             void enterAnimationMode( const AnimatableShapeSharedPtr& rShape );
175 
176             /** Notify the LayerManager that the given Shape is no
177                 longer animated.
178 
179 				This methods ends animation mode for the given Shape
180 				on all registered views.
181              */
182             void leaveAnimationMode( const AnimatableShapeSharedPtr& rShape );
183 
184             /** Notify that a shape needs an update
185 
186             	This method notifies the layer manager that a shape
187             	update is necessary. This is useful if, during
188             	animation playback, changes occur to shapes which make
189             	an update necessary on an update() call. Otherwise,
190             	update() will not render anything, which is not
191             	triggered by calling one of the other LayerManager
192             	methods.
193 
194                 @param rShape
195                 Shape which needs an update
196              */
197             void notifyShapeUpdate( const ShapeSharedPtr& rShape);
198 
199             /** Check whether any update operations  are pending.
200 
201             	@return true, if this LayerManager has any updates
202             	pending, i.e. needs to repaint something for the next
203             	frame.
204              */
205             bool isUpdatePending() const;
206 
207             /** Update the content
208 
209 				This method updates the content on all layers on all
210 				registered views. It does not issues a
211 				View::updateScreen() call on registered views. Please
212 				note that this method only takes into account changes
213 				to shapes induced directly by calling methods of the
214 				LayerManager. If a shape needs an update, because of
215 				some external event unknown to the LayerManager (most
216 				notably running animations), you have to notify the
217 				LayerManager via notifyShapeUpdate().
218 
219                 @see LayerManager::updateScreen()
220 
221 	            @return whether the update finished successfully.
222             */
223             bool update();
224 
225             /** Render the content to given canvas
226 
227                 This is a one-shot operation, which simply draws all
228                 shapes onto the given canvas, without any caching or
229                 other fuzz. Don't use that for repeated output onto
230                 the same canvas, the View concept is more optimal
231                 then.
232 
233                 @param rTargetCanvas
234                 Target canvas to output onto.
235              */
236             bool renderTo( const ::cppcanvas::CanvasSharedPtr& rTargetCanvas ) const;
237 
238         private:
239             /** A hash map which maps the XShape to the corresponding Shape object.
240 
241                 Provides quicker lookup than ShapeSet for simple mappings
242              */
243             typedef ::std::hash_map<
244                 ::com::sun::star::uno::Reference<
245                     ::com::sun::star::drawing::XShape >,
246                 ShapeSharedPtr,
247                 hash< ::com::sun::star::uno::Reference<
248                       ::com::sun::star::drawing::XShape > > > XShapeHash;
249 
250             class ShapeComparator
251             {
252             public:
253                 bool operator() (const ShapeSharedPtr& rpS1, const ShapeSharedPtr& rpS2 ) const
254                 {
255                     return Shape::lessThanShape::compare(rpS1.get(), rpS2.get());
256                 }
257             };
258             /** Set of all shapes
259              */
260         private:
261             typedef ::std::map< ShapeSharedPtr, LayerWeakPtr, ShapeComparator > LayerShapeMap;
262             typedef ::std::set< ShapeSharedPtr > ShapeUpdateSet;
263 
264 
265             ////////////////////////////////////////////////////////////////////////
266 
267 
268             /// Adds shape area to containing layer's damage area
269             void addUpdateArea( ShapeSharedPtr const& rShape );
270 
271             LayerSharedPtr createForegroundLayer() const;
272 
273             /** Push changes from updateShapeLayerAssociations() to current layer
274 
275                 Factored-out method that resizes layer, if necessary,
276                 assigns correct layer priority, and repaints contained shapes.
277 
278                 @param nCurrLayerIndex
279                 Index of current layer in maLayers
280 
281                 @param aFirstLayerShape
282                 Valid iterator out of maAllShapes, denoting the first
283                 shape from nCurrLayerIndex
284 
285                 @param aEndLayerShapes
286                 Valid iterator or end iterator out of maAllShapes,
287                 denoting one-behind-the-last shape of nCurrLayerIndex
288              */
289             void           commitLayerChanges( std::size_t                    nCurrLayerIndex,
290                                                LayerShapeMap::const_iterator  aFirstLayerShape,
291                                                LayerShapeMap::const_iterator  aEndLayerShapes );
292 
293             /** Init Shape layers with background layer.
294              */
295             void          putShape2BackgroundLayer( LayerShapeMap::value_type& rShapeEntry );
296 
297             /** Commits any pending layer reorg, due to shapes either
298                 entering or leaving animation mode
299 
300                 @param bBackgroundLayerPainted
301                 When true, LayerManager does not render anything on
302                 the background layer. Use this, if background has been
303                 updated by other means (e.g. slide transition)
304             */
305             void          updateShapeLayers( bool bBackgroundLayerPainted );
306 
307             /** Common stuff when adding a shape
308              */
309             void          implAddShape( const ShapeSharedPtr& rShape );
310 
311             /** Common stuff when removing a shape
312              */
313             void          implRemoveShape( const ShapeSharedPtr& rShape );
314 
315             /** Add or remove views
316 
317                 Sharing duplicate code from viewAdded and viewRemoved
318                 method. The only point of variation at those places
319                 are removal vs. adding.
320              */
321             template<typename LayerFunc,
322                      typename ShapeFunc> void manageViews( LayerFunc layerFunc,
323                                                            ShapeFunc shapeFunc );
324 
325             bool updateSprites();
326 
327             /// Registered views
328             const UnoViewContainer&  mrViews;
329 
330             /// All layers of this object. Vector owns the layers
331             LayerVector              maLayers;
332 
333             /** Contains all shapes with their XShape reference as the key
334              */
335             XShapeHash				 maXShapeHash;
336 
337             /** Set of shapes this LayerManager own
338 
339                 Contains the same set of shapes as XShapeHash, but is
340                 sorted in z order, for painting and layer
341                 association. Set entries are enriched with two flags
342                 for buffering animation enable/disable changes, and
343                 shape update requests.
344             */
345             LayerShapeMap            maAllShapes;
346 
347             /** Set of shapes that have requested an update
348 
349                 When a shape is member of this set, its maShapes entry
350                 has bNeedsUpdate set to true. We maintain this
351                 redundant information for faster update processing.
352              */
353             ShapeUpdateSet           maUpdateShapes;
354 
355             /** Overall slide bounds (in user coordinate
356                 system). shapes that exceed this boundary are clipped,
357                 thus, layers only need to be of this size.
358              */
359             const basegfx::B2DRange  maPageBounds;
360 
361             /// Number of shape sprites currently active on this LayerManager
362             sal_Int32                mnActiveSprites;
363 
364             /// sal_True, if shapes might need to move to different layer
365             bool                     mbLayerAssociationDirty;
366 
367             /// sal_False when deactivated
368             bool                     mbActive;
369 
370             /** When true, all sprite animations run in the foreground.  That
371                 is, no extra layers are created, and the slideshow runs
372                 potentially faster.
373              */
374             bool                     mbDisableAnimationZOrder;
375         };
376 
377         typedef ::boost::shared_ptr< LayerManager > LayerManagerSharedPtr;
378     }
379 }
380 
381 #endif /* INCLUDED_SLIDESHOW_LAYERMANAGER_HXX */
382