Files
libreoffice/basegfx/source/tools/b2dclipstate.cxx

478 lines
16 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
re-base on ALv2 code. Includes (at least) relevant parts of: linecap: Reintegrating finished LineCap feature Patch contributed by Regina Henschel http://svn.apache.org/viewvc?view=revision&revision=1232507 Patches contributed by Sven Jacobi impress212: #i81610# fixed animation export http://svn.apache.org/viewvc?view=revision&revision=1167620 impress212: drawinglayer gbuild environment changes http://svn.apache.org/viewvc?view=revision&revision=1167627 http://svn.apache.org/viewvc?view=revision&revision=1167628 impress212: DffPropSet -> minor code improvements, removing table http://svn.apache.org/viewvc?view=revision&revision=1167634 impress212: #158494# fixed excel import (text rotation) http://svn.apache.org/viewvc?view=revision&revision=1167638 Patches contributed by Armin Le Grand Svg: Reintegrated Svg replacement from /branches/alg/svgreplavement http://svn.apache.org/viewvc?view=revision&revision=1220836 #118728# changed indentifying definitions for Svg file detection http://svn.apache.org/viewvc?view=revision&revision=1229961 #118838# LineGeometry creation for complicated cases optimized to create single Polygons http://svn.apache.org/viewvc?view=revision&revision=1236232 #119176# corrected file type detection for SVG for svg files without xml header http://svn.apache.org/viewvc?view=revision&revision=1309445 #118728# Extended Svg file detection http://svn.apache.org/viewvc?view=revision&revision=1230531 #118529# solve break converters and convert commands for OLEs and images http://svn.apache.org/viewvc?view=revision&revision=1186168 svg: added WaE changes from branch svgreplacement to trunc http://svn.apache.org/viewvc?view=revision&revision=1222974 svg: corrected missing member initialization http://svn.apache.org/viewvc?view=revision&revision=1226134 fix for #118525#: Using primitives for chart sub-geometry visualisation http://svn.apache.org/viewvc?view=revision&revision=1226879 #118898# Adapted ImpGraphic::ImplGetBitmap to correctly convert metafiles to bitmapEx ... http://svn.apache.org/viewvc?view=revision&revision=1293316 fix for #118525#: removed no longer used variable maOriginalMapMode, one more exception eliminated http://svn.apache.org/viewvc?view=revision&revision=1227097 #16758# Added buffering to the VDev usages of the VclProcessor2D derivates... http://svn.apache.org/viewvc?view=revision&revision=1229521 #116758# Secured VDev buffer device to Vcl deinit http://svn.apache.org/viewvc?view=revision&revision=1230574 #116758# added remembering allocated VDevs for VDevBuffer to be able to also delete these when vcl goes down; it should never happen, but You never know http://svn.apache.org/viewvc?view=revision&revision=1230927 #118730# Changed SvgClipPathNode to use MaskPrimitive2D for primitive representation instead of TransparencePrimitive2D http://svn.apache.org/viewvc?view=revision&revision=1231198 #118822# secured 3D geometry creation (slices) by subdividing the 2D source polyPolygon early http://svn.apache.org/viewvc?view=revision&revision=1234749 #118829# enhanced Svg gradient quality, obstacles avoided http://svn.apache.org/viewvc?view=revision&revision=1235361 #118834# Unified usage of TextBreakupHelper as single tooling class for i18n text primitive breakup http://svn.apache.org/viewvc?view=revision&revision=1236110 #118853# added square pixel size limit to conversion of TransparencePrimitive2D to Metafile action http://svn.apache.org/viewvc?view=revision&revision=1237656 #118824# coreccted mirroring and boundrect when the graphicmanager is used for bitmap output http://svn.apache.org/viewvc?view=revision&revision=1240097 #115092# Corrected VclProcessor2D::RenderPolygonStrokePrimitive2D for various optimization scenarios http://svn.apache.org/viewvc?view=revision&revision=1241434 #118783# Corrected errors in ID strings, corrected Svg line/fill export, corrected polygon close state http://svn.apache.org/viewvc?view=revision&revision=1232006 #118796# corrected null-pointer usage in SVG text exporter http://svn.apache.org/viewvc?view=revision&revision=1240262 #118729# Use GraphicStreamUrl and GraphicUrl to allow multi image import with linked graphics, too http://svn.apache.org/viewvc?view=revision&revision=1229962 #118898# corrected error in GDIMetaFile::GetBoundRect in handling MetaFloatTransparentAction http://svn.apache.org/viewvc?view=revision&revision=1293349 #118855# Corrected handling of possibly created empty clipRegions after PolyPolygon clipping http://svn.apache.org/viewvc?view=revision&revision=1237725 #115962# Better (but not yet optimal, see comments in task) handling of MetaFloatTransparentAction in PDF export http://svn.apache.org/viewvc?view=revision&revision=1241078 IP clearance: #118466# This patch removes librsvg, libcroco, libgsf, ... http://svn.apache.org/viewvc?view=revision&revision=1200879 118779# Added svg content streaming in/out to ImpGraphic stream operators http://svn.apache.org/viewvc?view=revision&revision=1231908 linecap: correctons for WaE and mac drawing http://svn.apache.org/viewvc?view=revision&revision=1232793 svg: uses current system Dpi for Svg replacement image creation http://svn.apache.org/viewvc?view=revision&revision=1233948 Patches contributed by Mathias Bauer (and others) gnumake4 work variously http://svn.apache.org/viewvc?view=revision&revision=1394326 http://svn.apache.org/viewvc?view=revision&revision=1396797 http://svn.apache.org/viewvc?view=revision&revision=1397315 http://svn.apache.org/viewvc?view=revision&revision=1394326 Remove duplicate header includes. cws mba34issues01: #i117720#: convert assertion into warning http://svn.apache.org/viewvc?view=revision&revision=1172352 118485 - Styles for OLEs are not saved. Submitted by Armin Le Grand. http://svn.apache.org/viewvc?view=revision&revision=1182166 cws mba34issues01: #i117714#: remove assertion http://svn.apache.org/viewvc?view=revision&revision=1172357 Patch contributed by Jurgen Schmidt add some additional checks to ensure proper reading operations http://svn.apache.org/viewvc?view=revision&revision=1209022 mostly prefer our stream / bounds checking work. Patches contributed by Herbert Duerr #i118816# add clarifying comment regarding Font::*Color*() methods http://svn.apache.org/viewvc?view=revision&revision=1233833 extend macro->string handling for empty strings http://svn.apache.org/viewvc?view=revision&revision=1175801 avoid magic constants for SALCOLOR_NONE http://svn.apache.org/viewvc?view=revision&revision=1177543 initialize slant properly in ImplFontMetricData constructor (author=iorsh) http://svn.apache.org/viewvc?view=revision&revision=1177551 #i118675# make check for extension updates more stable http://svn.apache.org/viewvc?view=revision&revision=1214797 #a118617# remove VBasicEventListener.dll binary There are no known users depending on its CLSID http://svn.apache.org/viewvc?view=revision&revision=1203697 Patches contributed by Ariel Constenla-Haile Fix build breaker on Linux/gcc http://svn.apache.org/viewvc?view=revision&revision=1221104 Fix crash when trying to instantiate css.graphic.GraphicRasterizer_RSVG http://svn.apache.org/viewvc?view=revision&revision=1215559 Patches contributed by Oliver-Rainer Wittmann sw34bf06: #i117962# - method <SwFlyFrm::IsPaint(..)> - consider instances of <SwFlyDrawObj> http://svn.apache.org/viewvc?view=revision&revision=1172120 sw34bf06: #i117783# - Writer's implementation of XPagePrintable - apply print settings to new printing routines http://svn.apache.org/viewvc?view=revision&revision=1172115 gnumake4 work variously from Hans-Joachim Lankenau http://svn.apache.org/viewvc?view=revision&revision=1397315 http://svn.apache.org/viewvc?view=revision&revision=1396797 http://svn.apache.org/viewvc?view=revision&revision=1396782 http://svn.apache.org/viewvc?view=revision&revision=1394707 plus some amount of re-splitting of legacy headers. Patch contributed by Pavel Janik WaE: Remove unused variables. http://svn.apache.org/viewvc?view=revision&revision=1230697 Patches contributed by Takashi Ono mingwport35: i#117795: MinGW port fix for vcl2gnumake http://svn.apache.org/viewvc?view=revision&revision=1172091 mingwport35: i#117795: MinGW port fix for vcl2gnumake http://svn.apache.org/viewvc?view=revision&revision=1172091 Patch contributed by Christian Lippka impress212: #i98044# re enable Text menu for outline and title shapes http://svn.apache.org/viewvc?view=revision&revision=1167639 Patch contributed by Andre Fischer 118674: Made category B code optional and disabled by default. http://svn.apache.org/viewvc?view=revision&revision=1215131 118881: Ignore empty paragraphs after bullets. http://svn.apache.org/viewvc?view=revision&revision=1296205 Patches contributed by Philipp Lohmann ooo340fixes: #i117780# use rtl allocator http://svn.apache.org/viewvc?view=revision&revision=1172087 ooo34gsl02: #i117807# fix an off by one error (index actually inside the pfb section header) http://svn.apache.org/viewvc?view=revision&revision=1167576 various cleanups, related compilation fixes, warning cleanups, re-working of obsolete stl template pieces to use boost instead, changed string classes, re-adapt KDE about data, about dialog, fixing warnings, and other fixes & improvements. Disable svg import / render for about/ branding code-paths for now. Restore full icon theme set. Remove OS/2 conditionals and sources. Remove conflicting gtk/full-screen monitors support. Retain existing svg rasterizer files - temporarily disabled. Standardize stringificaiton and fixup dllpostfix issues. Rename SvgGradientHelper::== to equalTo to avoid overloading issues. Use the flat GdiPlus API for LineCaps calls.
2012-10-09 12:22:23 +01:00
/*
* 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 .
*/
#include <basegfx/tools/b2dclipstate.hxx>
#include <basegfx/range/b2drange.hxx>
#include <basegfx/range/b2dpolyrange.hxx>
#include <basegfx/range/b2drangeclipper.hxx>
#include <basegfx/polygon/b2dpolygon.hxx>
#include <basegfx/polygon/b2dpolygontools.hxx>
#include <basegfx/polygon/b2dpolypolygon.hxx>
#include <basegfx/polygon/b2dpolypolygontools.hxx>
#include <basegfx/polygon/b2dpolypolygoncutter.hxx>
#include <osl/diagnose.h>
namespace basegfx
{
namespace tools
{
2011-02-16 08:50:51 +00:00
class ImplB2DClipState
{
public:
enum Operation {UNION, INTERSECT, XOR, SUBTRACT};
ImplB2DClipState() :
maPendingPolygons(),
maPendingRanges(),
maClipPoly(),
mePendingOps(UNION)
{}
explicit ImplB2DClipState( const B2DPolyPolygon& rPoly ) :
maPendingPolygons(),
maPendingRanges(),
maClipPoly(rPoly),
mePendingOps(UNION)
{}
bool isCleared() const
{
return !maClipPoly.count()
&& !maPendingPolygons.count()
&& !maPendingRanges.count();
}
bool isNullClipPoly() const
{
return maClipPoly.count() == 1
&& !maClipPoly.getB2DPolygon(0).count();
}
bool isNull() const
{
return !maPendingPolygons.count()
&& !maPendingRanges.count()
&& isNullClipPoly();
}
void makeNull()
{
maPendingPolygons.clear();
maPendingRanges.clear();
maClipPoly.clear();
maClipPoly.append(B2DPolygon());
mePendingOps = UNION;
}
bool operator==(const ImplB2DClipState& rRHS) const
{
return maPendingPolygons == rRHS.maPendingPolygons
&& maPendingRanges == rRHS.maPendingRanges
&& maClipPoly == rRHS.maClipPoly
&& mePendingOps == rRHS.mePendingOps;
}
void addRange(const B2DRange& rRange, Operation eOp)
{
if( rRange.isEmpty() )
return;
commitPendingPolygons();
if( mePendingOps != eOp )
commitPendingRanges();
mePendingOps = eOp;
maPendingRanges.appendElement(
rRange,
ORIENTATION_POSITIVE);
}
void addPolyPolygon(B2DPolyPolygon aPoly, Operation eOp)
{
commitPendingRanges();
if( mePendingOps != eOp )
commitPendingPolygons();
mePendingOps = eOp;
maPendingPolygons.append(aPoly);
}
void unionRange(const B2DRange& rRange)
{
if( isCleared() )
return;
addRange(rRange,UNION);
}
void unionPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
if( isCleared() )
return;
addPolyPolygon(rPolyPoly,UNION);
}
void intersectRange(const B2DRange& rRange)
{
if( isNull() )
return;
addRange(rRange,INTERSECT);
}
void intersectPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
if( isNull() )
return;
addPolyPolygon(rPolyPoly,INTERSECT);
}
void subtractRange(const B2DRange& rRange )
{
if( isNull() )
return;
addRange(rRange,SUBTRACT);
}
void subtractPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
if( isNull() )
return;
addPolyPolygon(rPolyPoly,SUBTRACT);
}
void xorRange(const B2DRange& rRange)
{
addRange(rRange,XOR);
}
void xorPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
addPolyPolygon(rPolyPoly,XOR);
}
B2DPolyPolygon getClipPoly() const
{
commitPendingRanges();
commitPendingPolygons();
return maClipPoly;
}
private:
void commitPendingPolygons() const
{
if( !maPendingPolygons.count() )
return;
// assumption: maClipPoly has kept polygons prepared for
// clipping; i.e. no neutral polygons & correct
// orientation
maPendingPolygons = tools::prepareForPolygonOperation(maPendingPolygons);
const bool bIsEmpty=isNullClipPoly();
const bool bIsCleared=!maClipPoly.count();
switch(mePendingOps)
{
case UNION:
OSL_ASSERT( !bIsCleared );
if( bIsEmpty )
maClipPoly = maPendingPolygons;
else
maClipPoly = tools::solvePolygonOperationOr(
maClipPoly,
maPendingPolygons);
break;
case INTERSECT:
OSL_ASSERT( !bIsEmpty );
if( bIsCleared )
maClipPoly = maPendingPolygons;
else
maClipPoly = tools::solvePolygonOperationAnd(
maClipPoly,
maPendingPolygons);
break;
case XOR:
if( bIsEmpty )
maClipPoly = maPendingPolygons;
else if( bIsCleared )
{
// not representable, strictly speaking,
// using polygons with the common even/odd
// or nonzero winding number fill rule. If
// we'd want to represent it, fill rule
// would need to be "non-negative winding
// number" (and we then would return
// 'holes' here)
// going for an ugly hack meanwhile
maClipPoly = tools::solvePolygonOperationXor(
B2DPolyPolygon(
tools::createPolygonFromRect(B2DRange(-1E20,-1E20,1E20,1E20))),
maPendingPolygons);
}
else
maClipPoly = tools::solvePolygonOperationXor(
maClipPoly,
maPendingPolygons);
break;
case SUBTRACT:
OSL_ASSERT( !bIsEmpty );
// first union all pending ones, subtract en bloc then
maPendingPolygons = solveCrossovers(maPendingPolygons);
maPendingPolygons = stripNeutralPolygons(maPendingPolygons);
maPendingPolygons = stripDispensablePolygons(maPendingPolygons, false);
if( bIsCleared )
{
// not representable, strictly speaking,
// using polygons with the common even/odd
// or nonzero winding number fill rule. If
// we'd want to represent it, fill rule
// would need to be "non-negative winding
// number" (and we then would return
// 'holes' here)
// going for an ugly hack meanwhile
maClipPoly = tools::solvePolygonOperationDiff(
B2DPolyPolygon(
tools::createPolygonFromRect(B2DRange(-1E20,-1E20,1E20,1E20))),
maPendingPolygons);
}
else
maClipPoly = tools::solvePolygonOperationDiff(
maClipPoly,
maPendingPolygons);
break;
}
maPendingPolygons.clear();
mePendingOps = UNION;
}
void commitPendingRanges() const
{
if( !maPendingRanges.count() )
return;
// use the specialized range clipper for the win
B2DPolyPolygon aCollectedRanges;
const bool bIsEmpty=isNullClipPoly();
const bool bIsCleared=!maClipPoly.count();
switch(mePendingOps)
{
case UNION:
OSL_ASSERT( !bIsCleared );
aCollectedRanges = maPendingRanges.solveCrossovers();
aCollectedRanges = stripNeutralPolygons(aCollectedRanges);
aCollectedRanges = stripDispensablePolygons(aCollectedRanges, false);
if( bIsEmpty )
maClipPoly = aCollectedRanges;
else
maClipPoly = tools::solvePolygonOperationOr(
maClipPoly,
aCollectedRanges);
break;
case INTERSECT:
OSL_ASSERT( !bIsEmpty );
aCollectedRanges = maPendingRanges.solveCrossovers();
aCollectedRanges = stripNeutralPolygons(aCollectedRanges);
if( maPendingRanges.count() > 1 )
aCollectedRanges = stripDispensablePolygons(aCollectedRanges, true);
if( bIsCleared )
maClipPoly = aCollectedRanges;
else
maClipPoly = tools::solvePolygonOperationAnd(
maClipPoly,
aCollectedRanges);
break;
case XOR:
aCollectedRanges = maPendingRanges.solveCrossovers();
aCollectedRanges = stripNeutralPolygons(aCollectedRanges);
aCollectedRanges = correctOrientations(aCollectedRanges);
if( bIsEmpty )
maClipPoly = aCollectedRanges;
else if( bIsCleared )
{
// not representable, strictly speaking,
// using polygons with the common even/odd
// or nonzero winding number fill rule. If
// we'd want to represent it, fill rule
// would need to be "non-negative winding
// number" (and we then would return
// 'holes' here)
// going for an ugly hack meanwhile
maClipPoly = tools::solvePolygonOperationXor(
B2DPolyPolygon(
tools::createPolygonFromRect(B2DRange(-1E20,-1E20,1E20,1E20))),
aCollectedRanges);
}
else
maClipPoly = tools::solvePolygonOperationXor(
maClipPoly,
aCollectedRanges);
break;
case SUBTRACT:
OSL_ASSERT( !bIsEmpty );
// first union all pending ranges, subtract en bloc then
aCollectedRanges = maPendingRanges.solveCrossovers();
aCollectedRanges = stripNeutralPolygons(aCollectedRanges);
aCollectedRanges = stripDispensablePolygons(aCollectedRanges, false);
if( bIsCleared )
{
// not representable, strictly speaking,
// using polygons with the common even/odd
// or nonzero winding number fill rule. If
// we'd want to represent it, fill rule
// would need to be "non-negative winding
// number" (and we then would return
// 'holes' here)
// going for an ugly hack meanwhile
maClipPoly = tools::solvePolygonOperationDiff(
B2DPolyPolygon(
tools::createPolygonFromRect(B2DRange(-1E20,-1E20,1E20,1E20))),
aCollectedRanges);
}
else
maClipPoly = tools::solvePolygonOperationDiff(
maClipPoly,
aCollectedRanges);
break;
}
maPendingRanges.clear();
mePendingOps = UNION;
}
mutable B2DPolyPolygon maPendingPolygons;
mutable B2DPolyRange maPendingRanges;
mutable B2DPolyPolygon maClipPoly;
mutable Operation mePendingOps;
};
B2DClipState::B2DClipState() :
mpImpl()
{}
B2DClipState::~B2DClipState()
{}
B2DClipState::B2DClipState( const B2DClipState& rOrig ) :
mpImpl(rOrig.mpImpl)
{}
B2DClipState::B2DClipState( const B2DPolyPolygon& rPolyPoly ) :
mpImpl( ImplB2DClipState(rPolyPoly) )
{}
B2DClipState& B2DClipState::operator=( const B2DClipState& rRHS )
{
mpImpl = rRHS.mpImpl;
return *this;
}
void B2DClipState::makeNull()
{
mpImpl->makeNull();
}
bool B2DClipState::isCleared() const
{
return mpImpl->isCleared();
}
bool B2DClipState::operator==(const B2DClipState& rRHS) const
{
if(mpImpl.same_object(rRHS.mpImpl))
return true;
return ((*mpImpl) == (*rRHS.mpImpl));
}
bool B2DClipState::operator!=(const B2DClipState& rRHS) const
{
return !(*this == rRHS);
}
void B2DClipState::unionRange(const B2DRange& rRange)
{
mpImpl->unionRange(rRange);
}
void B2DClipState::unionPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
mpImpl->unionPolyPolygon(rPolyPoly);
}
void B2DClipState::intersectRange(const B2DRange& rRange)
{
mpImpl->intersectRange(rRange);
}
void B2DClipState::intersectPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
mpImpl->intersectPolyPolygon(rPolyPoly);
}
void B2DClipState::subtractRange(const B2DRange& rRange)
{
mpImpl->subtractRange(rRange);
}
void B2DClipState::subtractPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
mpImpl->subtractPolyPolygon(rPolyPoly);
}
void B2DClipState::xorRange(const B2DRange& rRange)
{
mpImpl->xorRange(rRange);
}
void B2DClipState::xorPolyPolygon(const B2DPolyPolygon& rPolyPoly)
{
mpImpl->xorPolyPolygon(rPolyPoly);
}
B2DPolyPolygon B2DClipState::getClipPoly() const
{
return mpImpl->getClipPoly();
}
} // end of namespace tools
} // end of namespace basegfx
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */