/************************************************************** * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_canvas.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "cairo_canvascustomsprite.hxx" #include "cairo_spritehelper.hxx" #include using namespace ::cairo; using namespace ::com::sun::star; namespace cairocanvas { SpriteHelper::SpriteHelper() : mpSpriteCanvas(), mpBufferSurface(), mbTextureDirty(true) {} void SpriteHelper::init( const geometry::RealSize2D& rSpriteSize, const SpriteCanvasRef& rSpriteCanvas ) { ENSURE_OR_THROW( rSpriteCanvas.get(), "SpriteHelper::init(): Invalid device, sprite canvas or surface" ); mpSpriteCanvas = rSpriteCanvas; mbTextureDirty = true; // also init base class CanvasCustomSpriteHelper::init( rSpriteSize, rSpriteCanvas.get() ); } void SpriteHelper::setSurface( const SurfaceSharedPtr& pBufferSurface ) { mpBufferSurface = pBufferSurface; } void SpriteHelper::disposing() { mpBufferSurface.reset(); mpSpriteCanvas.clear(); // forward to parent CanvasCustomSpriteHelper::disposing(); } void SpriteHelper::redraw( const CairoSharedPtr& pCairo, const ::basegfx::B2DPoint& rPos, bool& /*io_bSurfacesDirty*/, bool /*bBufferedUpdate*/ ) const { #ifdef CAIRO_CANVAS_PERF_TRACE struct timespec aTimer; mxDevice->startPerfTrace( &aTimer ); #endif const double fAlpha( getAlpha() ); const ::basegfx::B2DHomMatrix aTransform( getTransformation() ); if( isActive() && !::basegfx::fTools::equalZero( fAlpha ) ) { OSL_TRACE ("CanvasCustomSprite::redraw called"); if( pCairo ) { basegfx::B2DVector aSize = getSizePixel(); cairo_save( pCairo.get() ); double fX, fY; fX = rPos.getX(); fY = rPos.getY(); if( !aTransform.isIdentity() ) { cairo_matrix_t aMatrix, aInverseMatrix; cairo_matrix_init( &aMatrix, aTransform.get( 0, 0 ), aTransform.get( 1, 0 ), aTransform.get( 0, 1 ), aTransform.get( 1, 1 ), aTransform.get( 0, 2 ), aTransform.get( 1, 2 ) ); aMatrix.x0 = basegfx::fround( aMatrix.x0 ); aMatrix.y0 = basegfx::fround( aMatrix.y0 ); cairo_matrix_init( &aInverseMatrix, aMatrix.xx, aMatrix.yx, aMatrix.xy, aMatrix.yy, aMatrix.x0, aMatrix.y0 ); cairo_matrix_invert( &aInverseMatrix ); cairo_matrix_transform_distance( &aInverseMatrix, &fX, &fY ); cairo_set_matrix( pCairo.get(), &aMatrix ); } fX = basegfx::fround( fX ); fY = basegfx::fround( fY ); cairo_matrix_t aOrigMatrix; cairo_get_matrix( pCairo.get(), &aOrigMatrix ); cairo_translate( pCairo.get(), fX, fY ); if( getClip().is() ) { const uno::Reference& rClip( getClip() ); ::basegfx::B2DPolyPolygon aClipPoly( ::basegfx::unotools::b2DPolyPolygonFromXPolyPolygon2D( rClip )); doPolyPolygonImplementation( aClipPoly, Clip, pCairo.get(), NULL, SurfaceProviderRef(mpSpriteCanvas.get()), rClip->getFillRule() ); } OSL_TRACE ("aSize %f x %f position: %f,%f", aSize.getX(), aSize.getY(), fX, fY ); cairo_rectangle( pCairo.get(), 0, 0, floor( aSize.getX() ), floor( aSize.getY() ) ); cairo_clip( pCairo.get() ); cairo_set_matrix( pCairo.get(), &aOrigMatrix ); if( isContentFullyOpaque() ) cairo_set_operator( pCairo.get(), CAIRO_OPERATOR_SOURCE ); cairo_set_source_surface( pCairo.get(), mpBufferSurface->getCairoSurface().get(), fX, fY ); if( ::rtl::math::approxEqual( fAlpha, 1.0 ) ) cairo_paint( pCairo.get() ); else cairo_paint_with_alpha( pCairo.get(), fAlpha ); cairo_restore( pCairo.get() ); } } #ifdef CAIRO_CANVAS_PERF_TRACE mxDevice->stopPerfTrace( &aTimer, "sprite redraw" ); #endif } ::basegfx::B2DPolyPolygon SpriteHelper::polyPolygonFromXPolyPolygon2D( uno::Reference< rendering::XPolyPolygon2D >& xPoly ) const { return ::basegfx::unotools::b2DPolyPolygonFromXPolyPolygon2D(xPoly); } }