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 // MARKER(update_precomp.py): autogen include statement, do not remove
23 #include "precompiled_drawinglayer.hxx"
24 
25 #include <drawinglayer/primitive2d/graphicprimitive2d.hxx>
26 #include <drawinglayer/primitive2d/cropprimitive2d.hxx>
27 #include <drawinglayer/primitive2d/drawinglayer_primitivetypes2d.hxx>
28 #include <drawinglayer/primitive2d/maskprimitive2d.hxx>
29 #include <drawinglayer/primitive2d/graphicprimitivehelper2d.hxx>
30 #include <drawinglayer/primitive2d/unifiedtransparenceprimitive2d.hxx>
31 #include <drawinglayer/primitive2d/modifiedcolorprimitive2d.hxx>
32 #include <basegfx/matrix/b2dhommatrixtools.hxx>
33 #include <vcl/svapp.hxx>
34 #include <vcl/outdev.hxx>
35 
36 //////////////////////////////////////////////////////////////////////////////
37 
38 namespace drawinglayer
39 {
40     namespace primitive2d
41     {
create2DDecomposition(const geometry::ViewInformation2D & rViewInformation) const42         Primitive2DSequence GraphicPrimitive2D::create2DDecomposition(const geometry::ViewInformation2D&
43 #ifdef USE_DEBUG_CODE_TO_TEST_METAFILE_DECOMPOSE
44             rViewInformation
45 #else
46             /*rViewInformation*/
47 #endif // USE_DEBUG_CODE_TO_TEST_METAFILE_DECOMPOSE
48             ) const
49         {
50             Primitive2DSequence aRetval;
51 
52             if(255L == getGraphicAttr().GetTransparency())
53             {
54                 // content is invisible, done
55                 return aRetval;
56             }
57 
58             // do not apply mirroring from GraphicAttr to the Metafile by calling
59             // GetTransformedGraphic, this will try to mirror the Metafile using Scale()
60             // at the Metafile. This again calls Scale at the single MetaFile actions,
61             // but this implementation never worked. I reworked that implementations,
62             // but for security reasons i will try not to use it.
63             basegfx::B2DHomMatrix aTransform(getTransform());
64 
65             if(getGraphicAttr().IsMirrored())
66             {
67                 // content needs mirroring
68                 const bool bHMirr(getGraphicAttr().GetMirrorFlags() & BMP_MIRROR_HORZ);
69                 const bool bVMirr(getGraphicAttr().GetMirrorFlags() & BMP_MIRROR_VERT);
70 
71                 // mirror by applying negative scale to the unit primitive and
72                 // applying the object transformation on it.
73                 aTransform = basegfx::tools::createScaleB2DHomMatrix(
74                     bHMirr ? -1.0 : 1.0,
75                     bVMirr ? -1.0 : 1.0);
76                 aTransform.translate(
77                     bHMirr ? 1.0 : 0.0,
78                     bVMirr ? 1.0 : 0.0);
79                 aTransform = getTransform() * aTransform;
80             }
81 
82             // Get transformed graphic. Suppress rotation and cropping, only filtering is needed
83             // here (and may be replaced later on). Cropping is handled below as mask primitive (if set).
84             // Also need to suppress mirroring, it is part of the transformation now (see above).
85             // Also move transparency handling to embedding to a UnifiedTransparencePrimitive2D; do
86             // that by remembering original transparency and applying that later if needed
87             GraphicAttr aSuppressGraphicAttr(getGraphicAttr());
88 
89             aSuppressGraphicAttr.SetCrop(0, 0, 0, 0);
90             aSuppressGraphicAttr.SetRotation(0);
91             aSuppressGraphicAttr.SetMirrorFlags(0);
92             aSuppressGraphicAttr.SetTransparency(0);
93 
94             const GraphicObject& rGraphicObject = getGraphicObject();
95             Graphic aTransformedGraphic(rGraphicObject.GetGraphic());
96             const bool isBitmap(GRAPHIC_BITMAP == aTransformedGraphic.GetType() && !aTransformedGraphic.getSvgData().get());
97             const bool isAdjusted(getGraphicAttr().IsAdjusted());
98             const bool isDrawMode(GRAPHICDRAWMODE_STANDARD != getGraphicAttr().GetDrawMode());
99 
100             if(isBitmap && (isAdjusted || isDrawMode))
101             {
102                 // the pure primitive solution with the color modifiers works well, too, but when
103                 // it is a bitmap graphic the old modification currently is faster; so use it here
104                 // instead of creating all as in create2DColorModifierEmbeddingsAsNeeded (see below).
105                 // Still, crop, rotation, mirroring and transparency is handled by primitives already
106                 // (see above).
107                 // This could even be done when vector graphic, but we explicitely want to have the
108                 // pure primitive solution for this; this will allow vector graphics to stay vector
109                 // geraphics, independent from the color filtering stuff. This will enhance e.g.
110                 // SVG and print quality while reducing data size at the same time.
111                 // The other way around the old modifications when only used on already bitmap objects
112                 // will not loose any quality.
113                 aTransformedGraphic = rGraphicObject.GetTransformedGraphic(&aSuppressGraphicAttr);
114 
115                 // reset GraphicAttr after use to not apply double
116                 aSuppressGraphicAttr = GraphicAttr();
117             }
118 
119             // create sub-content; helper takes care of correct handling of
120             // bitmap, svg or metafile content
121             aRetval = create2DDecompositionOfGraphic(
122                 aTransformedGraphic,
123                 aTransform);
124 
125             if(!aRetval.getLength())
126             {
127                 // content is invisible, done
128                 return aRetval;
129             }
130 
131             if(isAdjusted || isDrawMode)
132             {
133                 // embed to needed ModifiedColorPrimitive2D's if necessary. Do this for
134                 // adjustments and draw mode specials
135                 aRetval = create2DColorModifierEmbeddingsAsNeeded(
136                     aRetval,
137                     aSuppressGraphicAttr.GetDrawMode(),
138                     basegfx::clamp(aSuppressGraphicAttr.GetLuminance() * 0.01, -1.0, 1.0),
139                     basegfx::clamp(aSuppressGraphicAttr.GetContrast() * 0.01, -1.0, 1.0),
140                     basegfx::clamp(aSuppressGraphicAttr.GetChannelR() * 0.01, -1.0, 1.0),
141                     basegfx::clamp(aSuppressGraphicAttr.GetChannelG() * 0.01, -1.0, 1.0),
142                     basegfx::clamp(aSuppressGraphicAttr.GetChannelB() * 0.01, -1.0, 1.0),
143                     basegfx::clamp(aSuppressGraphicAttr.GetGamma(), 0.0, 10.0),
144                     aSuppressGraphicAttr.IsInvert());
145 
146                 if(!aRetval.getLength())
147                 {
148                     // content is invisible, done
149                     return aRetval;
150                 }
151             }
152 
153             if(getGraphicAttr().IsTransparent())
154             {
155                 // check for transparency
156                 const double fTransparency(basegfx::clamp(getGraphicAttr().GetTransparency() * (1.0 / 255.0), 0.0, 1.0));
157 
158                 if(!basegfx::fTools::equalZero(fTransparency))
159                 {
160                     const Primitive2DReference aUnifiedTransparence(
161                         new UnifiedTransparencePrimitive2D(
162                             aRetval,
163                             fTransparency));
164 
165                     aRetval = Primitive2DSequence(&aUnifiedTransparence, 1);
166                 }
167             }
168 
169             if(getGraphicAttr().IsCropped())
170             {
171                 // check for cropping
172                 // calculate scalings between real image size and logic object size. This
173                 // is necessary since the crop values are relative to original bitmap size
174                 const basegfx::B2DVector aObjectScale(aTransform * basegfx::B2DVector(1.0, 1.0));
175                 const basegfx::B2DVector aCropScaleFactor(
176                     rGraphicObject.calculateCropScaling(
177                         aObjectScale.getX(),
178                         aObjectScale.getY(),
179                         getGraphicAttr().GetLeftCrop(),
180                         getGraphicAttr().GetTopCrop(),
181                         getGraphicAttr().GetRightCrop(),
182                         getGraphicAttr().GetBottomCrop()));
183 
184                 // embed content in cropPrimitive
185                 Primitive2DReference xPrimitive(
186                     new CropPrimitive2D(
187                         aRetval,
188                         aTransform,
189                         getGraphicAttr().GetLeftCrop() * aCropScaleFactor.getX(),
190                         getGraphicAttr().GetTopCrop() * aCropScaleFactor.getY(),
191                         getGraphicAttr().GetRightCrop() * aCropScaleFactor.getX(),
192                         getGraphicAttr().GetBottomCrop() * aCropScaleFactor.getY()));
193 
194                 aRetval = Primitive2DSequence(&xPrimitive, 1);
195             }
196 
197             return aRetval;
198         }
199 
GraphicPrimitive2D(const basegfx::B2DHomMatrix & rTransform,const GraphicObject & rGraphicObject,const GraphicAttr & rGraphicAttr)200         GraphicPrimitive2D::GraphicPrimitive2D(
201             const basegfx::B2DHomMatrix& rTransform,
202             const GraphicObject& rGraphicObject,
203             const GraphicAttr& rGraphicAttr)
204         :	BufferedDecompositionPrimitive2D(),
205             maTransform(rTransform),
206             maGraphicObject(rGraphicObject),
207             maGraphicAttr(rGraphicAttr)
208         {
209         }
210 
GraphicPrimitive2D(const basegfx::B2DHomMatrix & rTransform,const GraphicObject & rGraphicObject)211         GraphicPrimitive2D::GraphicPrimitive2D(
212             const basegfx::B2DHomMatrix& rTransform,
213             const GraphicObject& rGraphicObject)
214         :	BufferedDecompositionPrimitive2D(),
215             maTransform(rTransform),
216             maGraphicObject(rGraphicObject),
217             maGraphicAttr()
218         {
219         }
220 
operator ==(const BasePrimitive2D & rPrimitive) const221         bool GraphicPrimitive2D::operator==(const BasePrimitive2D& rPrimitive) const
222         {
223             if(BufferedDecompositionPrimitive2D::operator==(rPrimitive))
224             {
225                 const GraphicPrimitive2D& rCompare = (GraphicPrimitive2D&)rPrimitive;
226 
227                 return (getTransform() == rCompare.getTransform()
228                     && getGraphicObject() == rCompare.getGraphicObject()
229                     && getGraphicAttr() == rCompare.getGraphicAttr());
230             }
231 
232             return false;
233         }
234 
getB2DRange(const geometry::ViewInformation2D &) const235         basegfx::B2DRange GraphicPrimitive2D::getB2DRange(const geometry::ViewInformation2D& /*rViewInformation*/) const
236         {
237             basegfx::B2DRange aRetval(0.0, 0.0, 1.0, 1.0);
238             aRetval.transform(getTransform());
239             return aRetval;
240         }
241 
242         // provide unique ID
243         ImplPrimitrive2DIDBlock(GraphicPrimitive2D, PRIMITIVE2D_ID_GRAPHICPRIMITIVE2D)
244 
245     } // end of namespace primitive2d
246 } // end of namespace drawinglayer
247 
248 //////////////////////////////////////////////////////////////////////////////
249 // eof
250