/* * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include namespace svgio::svgreader { SvgFeCompositeNode::SvgFeCompositeNode(SvgDocument& rDocument, SvgNode* pParent) : SvgFilterNode(SVGToken::FeComposite, rDocument, pParent) , maOperator(Operator::Over) { } SvgFeCompositeNode::~SvgFeCompositeNode() {} void SvgFeCompositeNode::parseAttribute(SVGToken aSVGToken, const OUString& aContent) { // parse own switch (aSVGToken) { case SVGToken::Style: { readLocalCssStyle(aContent); break; } case SVGToken::In: { maIn = aContent.trim(); break; } case SVGToken::In2: { maIn2 = aContent.trim(); break; } case SVGToken::Result: { maResult = aContent.trim(); break; } case SVGToken::Operator: { if (!aContent.isEmpty()) { if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"over")) { maOperator = Operator::Over; } else if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"in")) { maOperator = Operator::In; } else if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"out")) { maOperator = Operator::Out; } else if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"xor")) { maOperator = Operator::Xor; } else if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"atop")) { maOperator = Operator::Atop; } else if (o3tl::equalsIgnoreAsciiCase(o3tl::trim(aContent), u"arithmetic")) { maOperator = Operator::Arithmetic; } } break; } case SVGToken::K1: { SvgNumber aNum; if (readSingleNumber(aContent, aNum)) { maK1 = aNum; } break; } case SVGToken::K2: { SvgNumber aNum; if (readSingleNumber(aContent, aNum)) { maK2 = aNum; } break; } case SVGToken::K3: { SvgNumber aNum; if (readSingleNumber(aContent, aNum)) { maK3 = aNum; } break; } case SVGToken::K4: { SvgNumber aNum; if (readSingleNumber(aContent, aNum)) { maK4 = aNum; } break; } default: { break; } } } void SvgFeCompositeNode::apply(drawinglayer::primitive2d::Primitive2DContainer& rTarget, const SvgFilterNode* pParent) const { const drawinglayer::primitive2d::Primitive2DContainer* pSource = pParent->findGraphicSource(maIn); const drawinglayer::primitive2d::Primitive2DContainer* pSource2 = pParent->findGraphicSource(maIn2); if (maOperator != Operator::Arithmetic) { basegfx::B2DPolyPolygon aPolyPolygon, aPolyPolygon2; // Process maIn2 first if (pSource2) { rTarget.append(*pSource2); drawinglayer::processor2d::ContourExtractor2D aExtractor( drawinglayer::geometry::ViewInformation2D(), true); aExtractor.process(*pSource2); const basegfx::B2DPolyPolygonVector& rResult(aExtractor.getExtractedContour()); aPolyPolygon2 = basegfx::utils::mergeToSinglePolyPolygon(rResult); } if (pSource) { rTarget.append(*pSource); drawinglayer::processor2d::ContourExtractor2D aExtractor( drawinglayer::geometry::ViewInformation2D(), true); aExtractor.process(*pSource); const basegfx::B2DPolyPolygonVector& rResult(aExtractor.getExtractedContour()); aPolyPolygon = basegfx::utils::mergeToSinglePolyPolygon(rResult); } basegfx::B2DPolyPolygon aResult; if (maOperator == Operator::Over) { aResult = basegfx::utils::solvePolygonOperationOr(aPolyPolygon, aPolyPolygon2); } else if (maOperator == Operator::Out) { aResult = basegfx::utils::solvePolygonOperationDiff(aPolyPolygon, aPolyPolygon2); } else if (maOperator == Operator::In) { aResult = basegfx::utils::solvePolygonOperationAnd(aPolyPolygon, aPolyPolygon2); } else if (maOperator == Operator::Xor) { aResult = basegfx::utils::solvePolygonOperationXor(aPolyPolygon, aPolyPolygon2); } else if (maOperator == Operator::Atop) { // Atop is the union of In and Out. // The parts of in2 graphic that do not overlap with the in graphic stay untouched. aResult = basegfx::utils::solvePolygonOperationDiff(aPolyPolygon2, aPolyPolygon); aResult.append(basegfx::utils::solvePolygonOperationAnd(aPolyPolygon, aPolyPolygon2)); } rTarget = drawinglayer::primitive2d::Primitive2DContainer{ new drawinglayer::primitive2d::MaskPrimitive2D(std::move(aResult), std::move(rTarget)) }; pParent->addGraphicSourceToMapper(maResult, rTarget); } else { basegfx::B2DRange aRange, aRange2; const drawinglayer::geometry::ViewInformation2D aViewInformation2D; if (pSource) { aRange = pSource->getB2DRange(aViewInformation2D); } if (pSource2) { aRange2 = pSource2->getB2DRange(aViewInformation2D); } const basegfx::B2DRange aBaseRange(0, 0, std::max(aRange.getMaxX(), aRange2.getMaxX()), std::max(aRange.getMaxY(), aRange2.getMaxY())); BitmapEx aBmpEx, aBmpEx2; if (pSource) { drawinglayer::primitive2d::Primitive2DContainer aSource(*pSource); aBmpEx = drawinglayer::convertToBitmapEx( std::move(aSource), aViewInformation2D, aBaseRange.getWidth(), aBaseRange.getHeight(), aBaseRange.getWidth() * aBaseRange.getHeight()); } else { aBmpEx = drawinglayer::convertToBitmapEx( std::move(rTarget), aViewInformation2D, aBaseRange.getWidth(), aBaseRange.getHeight(), aBaseRange.getWidth() * aBaseRange.getHeight()); } if (pSource2) { drawinglayer::primitive2d::Primitive2DContainer aSource(*pSource2); aBmpEx2 = drawinglayer::convertToBitmapEx( std::move(aSource), aViewInformation2D, aBaseRange.getWidth(), aBaseRange.getHeight(), aBaseRange.getWidth() * aBaseRange.getHeight()); } else { aBmpEx2 = drawinglayer::convertToBitmapEx( std::move(rTarget), aViewInformation2D, aBaseRange.getWidth(), aBaseRange.getHeight(), aBaseRange.getWidth() * aBaseRange.getHeight()); } BitmapArithmeticBlendFilter aArithmeticFilter(aBmpEx2, maK1.getNumber(), maK2.getNumber(), maK3.getNumber(), maK4.getNumber()); BitmapEx aResBmpEx = aArithmeticFilter.execute(aBmpEx); const drawinglayer::primitive2d::Primitive2DReference xRef( new drawinglayer::primitive2d::BitmapPrimitive2D( aResBmpEx, basegfx::utils::createScaleTranslateB2DHomMatrix( aBaseRange.getRange(), aBaseRange.getMinimum()))); rTarget = drawinglayer::primitive2d::Primitive2DContainer{ xRef }; } } } // end of namespace svgio::svgreader /* vim:set shiftwidth=4 softtabstop=4 expandtab: */