1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_canvas.hxx"
26 
27 #include <ctype.h> // don't ask. msdev breaks otherwise...
28 #include <canvas/debug.hxx>
29 #include <canvas/verbosetrace.hxx>
30 #include <tools/diagnose_ex.h>
31 
32 #include <rtl/logfile.hxx>
33 #include <rtl/math.hxx>
34 
35 #include <canvas/canvastools.hxx>
36 
37 #include <basegfx/matrix/b2dhommatrix.hxx>
38 #include <basegfx/point/b2dpoint.hxx>
39 #include <basegfx/tools/canvastools.hxx>
40 #include <basegfx/numeric/ftools.hxx>
41 #include <basegfx/polygon/b2dpolypolygontools.hxx>
42 #include <basegfx/polygon/b2dpolygontools.hxx>
43 #include <basegfx/polygon/b2dpolypolygonrasterconverter.hxx>
44 #include <basegfx/polygon/b2dpolygontriangulator.hxx>
45 #include <basegfx/polygon/b2dpolygoncutandtouch.hxx>
46 
47 #include "dx_canvascustomsprite.hxx"
48 #include "dx_spritehelper.hxx"
49 #include "dx_impltools.hxx"
50 
51 #include <memory>
52 
53 using namespace ::com::sun::star;
54 
55 namespace dxcanvas
56 {
57     SpriteHelper::SpriteHelper() :
58         mpSpriteCanvas(),
59         mpBitmap(),
60         mbTextureDirty( true ),
61         mbShowSpriteBounds( false )
62     {
63     }
64 
65     void SpriteHelper::init( const geometry::RealSize2D&	 rSpriteSize,
66                              const SpriteCanvasRef&			 rSpriteCanvas,
67                              const IDXRenderModuleSharedPtr& rRenderModule,
68                              const DXSurfaceBitmapSharedPtr	 rBitmap,
69                              bool							 bShowSpriteBounds )
70     {
71         ENSURE_OR_THROW( rSpriteCanvas.get() &&
72                           rRenderModule &&
73                           rBitmap,
74                           "SpriteHelper::init(): Invalid device, sprite canvas or surface" );
75 
76         mpSpriteCanvas     = rSpriteCanvas;
77         mpBitmap           = rBitmap;
78         mbTextureDirty     = true;
79         mbShowSpriteBounds = bShowSpriteBounds;
80 
81         // also init base class
82         CanvasCustomSpriteHelper::init( rSpriteSize,
83                                         rSpriteCanvas.get() );
84     }
85 
86     void SpriteHelper::disposing()
87     {
88         mpBitmap.reset();
89         mpSpriteCanvas.clear();
90 
91         // forward to parent
92         CanvasCustomSpriteHelper::disposing();
93     }
94 
95     ::basegfx::B2DPolyPolygon SpriteHelper::polyPolygonFromXPolyPolygon2D( uno::Reference< rendering::XPolyPolygon2D >& xPoly ) const
96     {
97         return tools::polyPolygonFromXPolyPolygon2D( xPoly );
98     }
99 
100     bool SpriteHelper::needRedraw() const
101     {
102         if( !mpBitmap ||
103             !mpSpriteCanvas.get() )
104         {
105             return false; // we're disposed, no redraw necessary
106         }
107 
108         if( !isActive() ||
109             ::basegfx::fTools::equalZero( getAlpha() ) )
110         {
111             return false; // sprite is invisible
112         }
113 
114         return true;
115     }
116 
117     void SpriteHelper::redraw( bool& io_bSurfaceDirty ) const
118     {
119         if( !mpBitmap ||
120             !mpSpriteCanvas.get() )
121         {
122             return; // we're disposed
123         }
124 
125         const ::basegfx::B2DPoint& rPos( getPosPixel() );
126         const double               fAlpha( getAlpha() );
127 
128 		if( isActive() &&
129             !::basegfx::fTools::equalZero( fAlpha ) )
130         {
131 
132 			// TODO(Q2): For the time being, Device does not take a target
133 			// surface, but always unconditionally renders to the
134 			// background buffer.
135 
136 			// log output pos in device pixel
137 			VERBOSE_TRACE( "SpriteHelper::redraw(): output pos is (%f, %f)",
138                            rPos.getX(),
139                            rPos.getY() );
140 
141 			const double                                       fAlpha( getAlpha() );
142 			const ::basegfx::B2DVector&                        rSize( getSizePixel() );
143 			const ::basegfx::B2DHomMatrix&                     rTransform( getTransformation() );
144 			const uno::Reference< rendering::XPolyPolygon2D >& xClip( getClip() );
145 
146 			mbTextureDirty   = false;
147 			io_bSurfaceDirty = false; // state taken, and processed.
148 
149 			::basegfx::B2DPolyPolygon	aClipPath; // empty for no clip
150 			bool						bIsClipRectangular( false ); // false, if no
151 																	// clip, or clip
152 																	// is complex
153 
154 			// setup and apply clip (if any)
155 			// =================================
156 
157 			if( xClip.is() )
158 			{
159 				aClipPath = tools::polyPolygonFromXPolyPolygon2D( xClip );
160 
161 				const sal_Int32 nNumClipPolygons( aClipPath.count() );
162 				if( nNumClipPolygons )
163 				{
164 					// TODO(P2): hold rectangle attribute directly
165 					// at the XPolyPolygon2D
166 
167 					// check whether the clip is rectangular
168 					if( nNumClipPolygons == 1 )
169 						if( ::basegfx::tools::isRectangle( aClipPath.getB2DPolygon( 0 ) ) )
170 							bIsClipRectangular = true;
171 				}
172 			}
173 
174 			const ::basegfx::B2DRectangle aSourceRect( 0.0,
175                                                        0.0,
176                                                        rSize.getX(),
177                                                        rSize.getY() );
178 
179 			// draw simple rectangular area if no clip is set.
180 			if( !aClipPath.count() )
181 			{
182 				mpBitmap->draw(fAlpha,rPos,rTransform);
183 			}
184 			else if( bIsClipRectangular )
185 			{
186 				// apply a simple rect clip
187 				// ========================
188 
189 				::basegfx::B2DRectangle aClipBounds(
190 					::basegfx::tools::getRange( aClipPath ) );
191 				aClipBounds.intersect( aSourceRect );
192 
193 				mpBitmap->draw(fAlpha,rPos,aClipBounds,rTransform);
194 			}
195 			else
196 			{
197 				// apply clip the hard way
198 				// =======================
199 
200 				mpBitmap->draw(fAlpha,rPos,aClipPath,rTransform);
201 			}
202 
203 			if( mbShowSpriteBounds )
204 			{
205 				if( aClipPath.count() )
206 				{
207 					// TODO(F2): Re-enable debug output
208                 }
209 			}
210 		}
211 	}
212 }
213