/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * 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 .
*/
namespace chart
{ usingnamespace ::com::sun::star; using ::basegfx::B2DRectangle; using ::basegfx::B2DTuple;
namespace{ /** @descr This is a supporting function for lcl_clip2d. It computes a new parametric value for an entering (dTE) or leaving (dTL) intersection point with one of the edges bounding the clipping area. For explanation of the parameters please refer to :
Liang-Biarsky parametric line-clipping algorithm as described in: Computer Graphics: principles and practice, 2nd ed., James D. Foley et al., Section 3.12.4 on page 117.
*/ bool lcl_CLIPt(double fDenom,double fNum, double & fTE, double & fTL)
{ double fT;
if (fDenom > 0) // Intersection enters: PE
{
fT = fNum / fDenom; // Parametric value at the intersection. if (fT > fTL) // fTE and fTL crossover returnfalse; // therefore reject the line. elseif (fT > fTE) // A new fTE has been found.
fTE = fT;
} elseif (fDenom < 0) // Intersection leaves: PL
{
fT = fNum / fDenom; // Parametric Value at the intersection. if (fT < fTE) // fTE and fTL crossover returnfalse; // therefore reject the line. elseif (fT < fTL) // A new fTL has been found.
fTL = fT;
} elseif (fNum > 0) returnfalse; // Line lies on the outside of the edge.
returntrue;
}
/** @descr The line given by its two endpoints rP0 and rP1 is clipped at the rectangle rRectangle. If there is at least a part of it visible then sal_True is returned and the endpoints of that part are stored in rP0 and rP1. The points rP0 and rP1 may have the same coordinates. @param rP0 Start point of the line to clip. Modified to contain a start point inside the clipping area if possible. @param rP1 End point of the line to clip. Modified to contain an end point inside the clipping area if possible. @param rRectangle Clipping area. @return If the line lies completely or partly inside the clipping area then TRUE is returned. If the line lies completely outside then sal_False is returned and rP0 and rP1 are left unmodified.
*/ bool lcl_clip2d(B2DTuple& rPoint0, B2DTuple& rPoint1, const B2DRectangle& rRectangle)
{ //Direction vector of the line.
B2DTuple aDirection = rPoint1 - rPoint0;
if( aDirection.getX()==0 && aDirection.getY()==0 && rRectangle.isInside(rPoint0) )
{ // Degenerate case of a zero length line. returntrue;
} else
{ // Values of the line parameter where the line enters resp. leaves the rectangle. double fTE = 0,
fTL = 1;
// Test whether at least a part lies in the four half-planes with respect to // the rectangles four edges. if( lcl_CLIPt(aDirection.getX(), rRectangle.getMinX() - rPoint0.getX(), fTE, fTL) ) if( lcl_CLIPt(-aDirection.getX(), rPoint0.getX() - rRectangle.getMaxX(), fTE, fTL) ) if( lcl_CLIPt(aDirection.getY(), rRectangle.getMinY() - rPoint0.getY(), fTE, fTL) ) if( lcl_CLIPt(-aDirection.getY(), rPoint0.getY() - rRectangle.getMaxY(), fTE, fTL) )
{ // At least a part is visible. if (fTL < 1)
{ // Compute the new end point.
rPoint1.setX( rPoint0.getX() + fTL * aDirection.getX() );
rPoint1.setY( rPoint0.getY() + fTL * aDirection.getY() );
} if (fTE > 0)
{ // Compute the new starting point.
rPoint0.setX( rPoint0.getX() + fTE * aDirection.getX() );
rPoint0.setY( rPoint0.getY() + fTE * aDirection.getY() );
} returntrue;
}
unsignedint round_up_nearest_pow2(unsignedint v)
{ // compute the next highest power of 2 of 32-bit v
--v;
v |= v >> 1;
v |= v >> 2;
v |= v >> 4;
v |= v >> 8;
v |= v >> 16;
++v; return v;
}
void lcl_addPointToPoly( drawing::PolyPolygonShape3D& rPoly
, const drawing::Position3D& rPos
, sal_Int32 nPolygonIndex
, std::vector< sal_Int32 >& rResultPointCount
, sal_Int32 nReservePointCount )
{ if(nPolygonIndex<0)
{
OSL_FAIL( "The polygon index needs to be > 0");
nPolygonIndex=0;
}
//make sure that we have enough polygons if(nPolygonIndex >= rPoly.SequenceX.getLength() )
{
rPoly.SequenceX.realloc(nPolygonIndex+1);
rPoly.SequenceY.realloc(nPolygonIndex+1);
rPoly.SequenceZ.realloc(nPolygonIndex+1);
rResultPointCount.resize(nPolygonIndex+1,0);
}
void lcl_addPointToPoly( std::vector<std::vector<css::drawing::Position3D>>& rPoly
, const drawing::Position3D& rPos
, sal_Int32 nPolygonIndex
, std::vector< sal_Int32 >& rResultPointCount
, sal_Int32 nReservePointCount )
{ if(nPolygonIndex<0)
{
OSL_FAIL( "The polygon index needs to be > 0");
nPolygonIndex=0;
}
//make sure that we have enough polygons if(o3tl::make_unsigned(nPolygonIndex) >= rPoly.size() )
{
rPoly.resize(nPolygonIndex+1);
rResultPointCount.resize(nPolygonIndex+1,0);
}
// set last point to a position outside the rectangle, such that the first // time lcl_clip2d returns true, the comparison to last will always yield false
drawing::Position3D aLast(rRectangle.getMinX()-1.0,rRectangle.getMinY()-1.0, 0.0 );
// set last point to a position outside the rectangle, such that the first // time lcl_clip2d returns true, the comparison to last will always yield false
drawing::Position3D aLast(rRectangle.getMinX()-1.0,rRectangle.getMinY()-1.0, 0.0 );
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.