mathematical 0.1.0 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/ext/mathematical/extconf.rb +14 -0
- data/ext/mathematical/itex2MML.h +63 -0
- data/ext/mathematical/lasemrender.c +257 -0
- data/ext/mathematical/lex.yy.c +6548 -0
- data/ext/mathematical/lsm.c +31 -0
- data/ext/mathematical/lsm.h +39 -0
- data/ext/mathematical/lsmattributes.c +279 -0
- data/ext/mathematical/lsmattributes.h +75 -0
- data/ext/mathematical/lsmcairo.c +598 -0
- data/ext/mathematical/lsmcairo.h +51 -0
- data/ext/mathematical/lsmdebug.c +183 -0
- data/ext/mathematical/lsmdebug.h +73 -0
- data/ext/mathematical/lsmdom.h +43 -0
- data/ext/mathematical/lsmdomcharacterdata.c +114 -0
- data/ext/mathematical/lsmdomcharacterdata.h +59 -0
- data/ext/mathematical/lsmdomdocument.c +255 -0
- data/ext/mathematical/lsmdomdocument.h +75 -0
- data/ext/mathematical/lsmdomdocumentfragment.c +81 -0
- data/ext/mathematical/lsmdomdocumentfragment.h +55 -0
- data/ext/mathematical/lsmdomelement.c +148 -0
- data/ext/mathematical/lsmdomelement.h +62 -0
- data/ext/mathematical/lsmdomentities.c +2166 -0
- data/ext/mathematical/lsmdomentities.h +35 -0
- data/ext/mathematical/lsmdomenumtypes.c +99 -0
- data/ext/mathematical/lsmdomenumtypes.h +26 -0
- data/ext/mathematical/lsmdomimplementation.c +82 -0
- data/ext/mathematical/lsmdomimplementation.h +41 -0
- data/ext/mathematical/lsmdomnamednodemap.c +118 -0
- data/ext/mathematical/lsmdomnamednodemap.h +64 -0
- data/ext/mathematical/lsmdomnode.c +737 -0
- data/ext/mathematical/lsmdomnode.h +122 -0
- data/ext/mathematical/lsmdomnodelist.c +70 -0
- data/ext/mathematical/lsmdomnodelist.h +58 -0
- data/ext/mathematical/lsmdomparser.c +461 -0
- data/ext/mathematical/lsmdomparser.h +54 -0
- data/ext/mathematical/lsmdomtext.c +82 -0
- data/ext/mathematical/lsmdomtext.h +55 -0
- data/ext/mathematical/lsmdomtypes.h +44 -0
- data/ext/mathematical/lsmdomview.c +422 -0
- data/ext/mathematical/lsmdomview.h +94 -0
- data/ext/mathematical/lsmitex.c +76 -0
- data/ext/mathematical/lsmitex.h +36 -0
- data/ext/mathematical/lsmmathml.h +66 -0
- data/ext/mathematical/lsmmathmlactionelement.c +93 -0
- data/ext/mathematical/lsmmathmlactionelement.h +57 -0
- data/ext/mathematical/lsmmathmlaligngroupelement.c +102 -0
- data/ext/mathematical/lsmmathmlaligngroupelement.h +56 -0
- data/ext/mathematical/lsmmathmlalignmarkelement.c +102 -0
- data/ext/mathematical/lsmmathmlalignmarkelement.h +56 -0
- data/ext/mathematical/lsmmathmlattributes.c +197 -0
- data/ext/mathematical/lsmmathmlattributes.h +126 -0
- data/ext/mathematical/lsmmathmldocument.c +304 -0
- data/ext/mathematical/lsmmathmldocument.h +61 -0
- data/ext/mathematical/lsmmathmlelement.c +491 -0
- data/ext/mathematical/lsmmathmlelement.h +107 -0
- data/ext/mathematical/lsmmathmlenums.c +429 -0
- data/ext/mathematical/lsmmathmlenums.h +182 -0
- data/ext/mathematical/lsmmathmlenumtypes.c +666 -0
- data/ext/mathematical/lsmmathmlenumtypes.h +90 -0
- data/ext/mathematical/lsmmathmlerrorelement.c +58 -0
- data/ext/mathematical/lsmmathmlerrorelement.h +56 -0
- data/ext/mathematical/lsmmathmlfencedelement.c +178 -0
- data/ext/mathematical/lsmmathmlfencedelement.h +65 -0
- data/ext/mathematical/lsmmathmlfractionelement.c +253 -0
- data/ext/mathematical/lsmmathmlfractionelement.h +62 -0
- data/ext/mathematical/lsmmathmlglyphtableams.c +597 -0
- data/ext/mathematical/lsmmathmlglyphtableams.h +45 -0
- data/ext/mathematical/lsmmathmlitexelement.c +187 -0
- data/ext/mathematical/lsmmathmlitexelement.h +60 -0
- data/ext/mathematical/lsmmathmllayoututils.c +191 -0
- data/ext/mathematical/lsmmathmllayoututils.h +58 -0
- data/ext/mathematical/lsmmathmlmathelement.c +204 -0
- data/ext/mathematical/lsmmathmlmathelement.h +81 -0
- data/ext/mathematical/lsmmathmloperatordictionary.c +3332 -0
- data/ext/mathematical/lsmmathmloperatordictionary.h +54 -0
- data/ext/mathematical/lsmmathmloperatorelement.c +307 -0
- data/ext/mathematical/lsmmathmloperatorelement.h +73 -0
- data/ext/mathematical/lsmmathmlpaddedelement.c +58 -0
- data/ext/mathematical/lsmmathmlpaddedelement.h +56 -0
- data/ext/mathematical/lsmmathmlphantomelement.c +71 -0
- data/ext/mathematical/lsmmathmlphantomelement.h +56 -0
- data/ext/mathematical/lsmmathmlpresentationcontainer.c +43 -0
- data/ext/mathematical/lsmmathmlpresentationcontainer.h +54 -0
- data/ext/mathematical/lsmmathmlpresentationtoken.c +303 -0
- data/ext/mathematical/lsmmathmlpresentationtoken.h +83 -0
- data/ext/mathematical/lsmmathmlradicalelement.c +266 -0
- data/ext/mathematical/lsmmathmlradicalelement.h +71 -0
- data/ext/mathematical/lsmmathmlrowelement.c +58 -0
- data/ext/mathematical/lsmmathmlrowelement.h +56 -0
- data/ext/mathematical/lsmmathmlscriptelement.c +282 -0
- data/ext/mathematical/lsmmathmlscriptelement.h +78 -0
- data/ext/mathematical/lsmmathmlsemanticselement.c +84 -0
- data/ext/mathematical/lsmmathmlsemanticselement.h +56 -0
- data/ext/mathematical/lsmmathmlspaceelement.c +142 -0
- data/ext/mathematical/lsmmathmlspaceelement.h +60 -0
- data/ext/mathematical/lsmmathmlstringelement.c +123 -0
- data/ext/mathematical/lsmmathmlstringelement.h +58 -0
- data/ext/mathematical/lsmmathmlstyle.c +130 -0
- data/ext/mathematical/lsmmathmlstyle.h +81 -0
- data/ext/mathematical/lsmmathmlstyleelement.c +307 -0
- data/ext/mathematical/lsmmathmlstyleelement.h +87 -0
- data/ext/mathematical/lsmmathmltablecellelement.c +122 -0
- data/ext/mathematical/lsmmathmltablecellelement.h +62 -0
- data/ext/mathematical/lsmmathmltableelement.c +545 -0
- data/ext/mathematical/lsmmathmltableelement.h +78 -0
- data/ext/mathematical/lsmmathmltablerowelement.c +120 -0
- data/ext/mathematical/lsmmathmltablerowelement.h +64 -0
- data/ext/mathematical/lsmmathmltraits.c +819 -0
- data/ext/mathematical/lsmmathmltraits.h +119 -0
- data/ext/mathematical/lsmmathmltypes.h +66 -0
- data/ext/mathematical/lsmmathmlunderoverelement.c +485 -0
- data/ext/mathematical/lsmmathmlunderoverelement.h +82 -0
- data/ext/mathematical/lsmmathmlutils.c +170 -0
- data/ext/mathematical/lsmmathmlutils.h +50 -0
- data/ext/mathematical/lsmmathmlview.c +1048 -0
- data/ext/mathematical/lsmmathmlview.h +164 -0
- data/ext/mathematical/lsmproperties.c +418 -0
- data/ext/mathematical/lsmproperties.h +85 -0
- data/ext/mathematical/lsmstr.c +231 -0
- data/ext/mathematical/lsmstr.h +114 -0
- data/ext/mathematical/lsmsvg.h +67 -0
- data/ext/mathematical/lsmsvgaelement.c +73 -0
- data/ext/mathematical/lsmsvgaelement.h +55 -0
- data/ext/mathematical/lsmsvgattributes.h +118 -0
- data/ext/mathematical/lsmsvgcircleelement.c +153 -0
- data/ext/mathematical/lsmsvgcircleelement.h +59 -0
- data/ext/mathematical/lsmsvgclippathelement.c +134 -0
- data/ext/mathematical/lsmsvgclippathelement.h +59 -0
- data/ext/mathematical/lsmsvgcolors.c +212 -0
- data/ext/mathematical/lsmsvgcolors.h +39 -0
- data/ext/mathematical/lsmsvgdefselement.c +74 -0
- data/ext/mathematical/lsmsvgdefselement.h +55 -0
- data/ext/mathematical/lsmsvgdocument.c +288 -0
- data/ext/mathematical/lsmsvgdocument.h +64 -0
- data/ext/mathematical/lsmsvgelement.c +373 -0
- data/ext/mathematical/lsmsvgelement.h +81 -0
- data/ext/mathematical/lsmsvgellipseelement.c +158 -0
- data/ext/mathematical/lsmsvgellipseelement.h +60 -0
- data/ext/mathematical/lsmsvgenums.c +544 -0
- data/ext/mathematical/lsmsvgenums.h +357 -0
- data/ext/mathematical/lsmsvgenumtypes.c +1083 -0
- data/ext/mathematical/lsmsvgenumtypes.h +111 -0
- data/ext/mathematical/lsmsvgfilterblend.c +105 -0
- data/ext/mathematical/lsmsvgfilterblend.h +58 -0
- data/ext/mathematical/lsmsvgfiltercomposite.c +109 -0
- data/ext/mathematical/lsmsvgfiltercomposite.h +58 -0
- data/ext/mathematical/lsmsvgfilterelement.c +266 -0
- data/ext/mathematical/lsmsvgfilterelement.h +66 -0
- data/ext/mathematical/lsmsvgfilterflood.c +86 -0
- data/ext/mathematical/lsmsvgfilterflood.h +55 -0
- data/ext/mathematical/lsmsvgfiltergaussianblur.c +114 -0
- data/ext/mathematical/lsmsvgfiltergaussianblur.h +57 -0
- data/ext/mathematical/lsmsvgfiltermerge.c +98 -0
- data/ext/mathematical/lsmsvgfiltermerge.h +55 -0
- data/ext/mathematical/lsmsvgfiltermergenode.c +87 -0
- data/ext/mathematical/lsmsvgfiltermergenode.h +57 -0
- data/ext/mathematical/lsmsvgfilteroffset.c +112 -0
- data/ext/mathematical/lsmsvgfilteroffset.h +58 -0
- data/ext/mathematical/lsmsvgfilterprimitive.c +168 -0
- data/ext/mathematical/lsmsvgfilterprimitive.h +66 -0
- data/ext/mathematical/lsmsvgfilterspecularlighting.c +127 -0
- data/ext/mathematical/lsmsvgfilterspecularlighting.h +60 -0
- data/ext/mathematical/lsmsvgfiltersurface.c +455 -0
- data/ext/mathematical/lsmsvgfiltersurface.h +66 -0
- data/ext/mathematical/lsmsvgfiltertile.c +102 -0
- data/ext/mathematical/lsmsvgfiltertile.h +57 -0
- data/ext/mathematical/lsmsvggelement.c +73 -0
- data/ext/mathematical/lsmsvggelement.h +55 -0
- data/ext/mathematical/lsmsvggradientelement.c +151 -0
- data/ext/mathematical/lsmsvggradientelement.h +68 -0
- data/ext/mathematical/lsmsvgimageelement.c +261 -0
- data/ext/mathematical/lsmsvgimageelement.h +67 -0
- data/ext/mathematical/lsmsvglength.c +152 -0
- data/ext/mathematical/lsmsvglength.h +65 -0
- data/ext/mathematical/lsmsvglineargradientelement.c +271 -0
- data/ext/mathematical/lsmsvglineargradientelement.h +60 -0
- data/ext/mathematical/lsmsvglineelement.c +153 -0
- data/ext/mathematical/lsmsvglineelement.h +60 -0
- data/ext/mathematical/lsmsvgmarkerelement.c +266 -0
- data/ext/mathematical/lsmsvgmarkerelement.h +74 -0
- data/ext/mathematical/lsmsvgmaskelement.c +232 -0
- data/ext/mathematical/lsmsvgmaskelement.h +64 -0
- data/ext/mathematical/lsmsvgmatrix.c +205 -0
- data/ext/mathematical/lsmsvgmatrix.h +59 -0
- data/ext/mathematical/lsmsvgpathelement.c +115 -0
- data/ext/mathematical/lsmsvgpathelement.h +59 -0
- data/ext/mathematical/lsmsvgpatternelement.c +398 -0
- data/ext/mathematical/lsmsvgpatternelement.h +69 -0
- data/ext/mathematical/lsmsvgpolygonelement.c +106 -0
- data/ext/mathematical/lsmsvgpolygonelement.h +57 -0
- data/ext/mathematical/lsmsvgpolylineelement.c +106 -0
- data/ext/mathematical/lsmsvgpolylineelement.h +57 -0
- data/ext/mathematical/lsmsvgradialgradientelement.c +323 -0
- data/ext/mathematical/lsmsvgradialgradientelement.h +61 -0
- data/ext/mathematical/lsmsvgrectelement.c +184 -0
- data/ext/mathematical/lsmsvgrectelement.h +62 -0
- data/ext/mathematical/lsmsvgstopelement.c +106 -0
- data/ext/mathematical/lsmsvgstopelement.h +57 -0
- data/ext/mathematical/lsmsvgstyle.c +560 -0
- data/ext/mathematical/lsmsvgstyle.h +217 -0
- data/ext/mathematical/lsmsvgsvgelement.c +260 -0
- data/ext/mathematical/lsmsvgsvgelement.h +71 -0
- data/ext/mathematical/lsmsvgswitchelement.c +103 -0
- data/ext/mathematical/lsmsvgswitchelement.h +55 -0
- data/ext/mathematical/lsmsvgsymbolelement.c +74 -0
- data/ext/mathematical/lsmsvgsymbolelement.h +55 -0
- data/ext/mathematical/lsmsvgtextelement.c +203 -0
- data/ext/mathematical/lsmsvgtextelement.h +60 -0
- data/ext/mathematical/lsmsvgtraits.c +1232 -0
- data/ext/mathematical/lsmsvgtraits.h +104 -0
- data/ext/mathematical/lsmsvgtransformable.c +106 -0
- data/ext/mathematical/lsmsvgtransformable.h +54 -0
- data/ext/mathematical/lsmsvgtspanelement.c +174 -0
- data/ext/mathematical/lsmsvgtspanelement.h +60 -0
- data/ext/mathematical/lsmsvgtypes.h +77 -0
- data/ext/mathematical/lsmsvguseelement.c +237 -0
- data/ext/mathematical/lsmsvguseelement.h +69 -0
- data/ext/mathematical/lsmsvgview.c +2583 -0
- data/ext/mathematical/lsmsvgview.h +179 -0
- data/ext/mathematical/lsmtraits.c +119 -0
- data/ext/mathematical/lsmtraits.h +49 -0
- data/ext/mathematical/lsmtypes.h +36 -0
- data/ext/mathematical/lsmutils.c +54 -0
- data/ext/mathematical/lsmutils.h +56 -0
- data/ext/mathematical/mathematical.c +145 -0
- data/ext/mathematical/y.tab.c +6179 -0
- data/ext/mathematical/y.tab.h +389 -0
- data/lib/mathematical/version.rb +1 -1
- data/mathematical.gemspec +8 -6
- data/test/mathematical/basic_test.rb +9 -0
- data/test/mathematical/fixtures/after/brackets_display.html +1 -0
- data/test/mathematical/fixtures/after/compliance_accents.html +17 -0
- data/test/mathematical/fixtures/after/compliance_arrows.html +71 -0
- data/test/mathematical/fixtures/after/compliance_colors.html +4 -0
- data/test/mathematical/fixtures/after/compliance_greek_letters.html +42 -0
- data/test/mathematical/fixtures/after/compliance_large_math.html +19 -0
- data/test/mathematical/fixtures/after/compliance_log_symbols.html +34 -0
- data/test/mathematical/fixtures/after/compliance_operators.html +262 -0
- data/test/mathematical/fixtures/after/dollar_sign_inline.html +1 -0
- data/test/mathematical/fixtures/after/equation_display.html +1 -0
- data/test/mathematical/fixtures/after/multiple_dollar_inline.html +1 -0
- data/test/mathematical/fixtures/after/parens_inline.html +1 -0
- data/test/mathematical/fixtures/before/brackets_display.text +1 -0
- data/test/mathematical/fixtures/before/compliance_accents.text +17 -0
- data/test/mathematical/fixtures/before/compliance_arrows.text +71 -0
- data/test/mathematical/fixtures/before/compliance_colors.text +4 -0
- data/test/mathematical/fixtures/before/compliance_greek_letters.text +42 -0
- data/test/mathematical/fixtures/before/compliance_large_math.text +19 -0
- data/test/mathematical/fixtures/before/compliance_log_symbols.text +34 -0
- data/test/mathematical/fixtures/before/compliance_operators.text +262 -0
- data/test/mathematical/fixtures/before/dollar_sign_inline.text +1 -0
- data/test/mathematical/fixtures/before/equation_display.text +1 -0
- data/test/mathematical/fixtures/before/multiple_dollar_inline.text +1 -0
- data/test/mathematical/fixtures/before/parens_inline.text +1 -0
- data/test/mathematical/fixtures/performance/big_file.text +1767 -0
- data/test/mathematical/fixtures_test.rb +45 -0
- data/test/mathematical/maliciousness_test.rb +45 -0
- data/test/mathematical/performance_test.rb +15 -0
- data/test/test_helper.rb +5 -0
- metadata +301 -14
- data/lib/mathematical.bundle +0 -0
- data/lib/mathematical/mathematical.bundle +0 -0
@@ -0,0 +1,77 @@
|
|
1
|
+
/* Lasem - SVG and Mathml library
|
2
|
+
*
|
3
|
+
* Copyright © 2010 Emmanuel Pacaud
|
4
|
+
*
|
5
|
+
* This library is free software; you can redistribute it and/or
|
6
|
+
* modify it under the terms of the GNU Lesser General Public
|
7
|
+
* License as published by the Free Software Foundation; either
|
8
|
+
* version 2 of the License, or (at your option) any later version.
|
9
|
+
*
|
10
|
+
* This library is distributed in the hope that it will be useful,
|
11
|
+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
12
|
+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
13
|
+
* Lesser General Public License for more details.
|
14
|
+
*
|
15
|
+
* You should have received a copy of the GNU Lesser General
|
16
|
+
* Public License along with this library; if not, write to the
|
17
|
+
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
18
|
+
* Boston, MA 02111-1307, USA.
|
19
|
+
*
|
20
|
+
* Author:
|
21
|
+
* Emmanuel Pacaud <emmanuel@gnome.org>
|
22
|
+
*/
|
23
|
+
|
24
|
+
#ifndef LSM_SVG_TYPES_H
|
25
|
+
#define LSM_SVG_TYPES_H
|
26
|
+
|
27
|
+
#include <lsmsvgenums.h>
|
28
|
+
|
29
|
+
G_BEGIN_DECLS
|
30
|
+
|
31
|
+
typedef struct _LsmSvgDocument LsmSvgDocument;
|
32
|
+
typedef struct _LsmSvgElement LsmSvgElement;
|
33
|
+
typedef struct _LsmSvgTransformable LsmSvgTransformable;
|
34
|
+
typedef struct _LsmSvgGraphic LsmSvgGraphic;
|
35
|
+
typedef struct _LsmSvgClipPathElement LsmSvgClipPathElement;
|
36
|
+
typedef struct _LsmSvgSvgElement LsmSvgSvgElement;
|
37
|
+
typedef struct _LsmSvgAElement LsmSvgAElement;
|
38
|
+
typedef struct _LsmSvgGElement LsmSvgGElement;
|
39
|
+
typedef struct _LsmSvgDefsElement LsmSvgDefsElement;
|
40
|
+
typedef struct _LsmSvgUseElement LsmSvgUseElement;
|
41
|
+
typedef struct _LsmSvgImageElement LsmSvgImageElement;
|
42
|
+
typedef struct _LsmSvgSymbolElement LsmSvgSymbolElement;
|
43
|
+
typedef struct _LsmSvgMarkerElement LsmSvgMarkerElement;
|
44
|
+
typedef struct _LsmSvgRectElement LsmSvgRectElement;
|
45
|
+
typedef struct _LsmSvgCircleElement LsmSvgCircleElement;
|
46
|
+
typedef struct _LsmSvgEllipseElement LsmSvgEllipseElement;
|
47
|
+
typedef struct _LsmSvgFilterElement LsmSvgFilterElement;
|
48
|
+
typedef struct _LsmSvgFilterPrimitive LsmSvgFilterPrimitive;
|
49
|
+
typedef struct _LsmSvgFilterBlend LsmSvgFilterBlend;
|
50
|
+
typedef struct _LsmSvgFilterComposite LsmSvgFilterComposite;
|
51
|
+
typedef struct _LsmSvgFilterFlood LsmSvgFilterFlood;
|
52
|
+
typedef struct _LsmSvgFilterGaussianBlur LsmSvgFilterGaussianBlur;
|
53
|
+
typedef struct _LsmSvgFilterMerge LsmSvgFilterMerge;
|
54
|
+
typedef struct _LsmSvgFilterMergeNode LsmSvgFilterMergeNode;
|
55
|
+
typedef struct _LsmSvgFilterOffset LsmSvgFilterOffset;
|
56
|
+
typedef struct _LsmSvgFilterSpecularLighting LsmSvgFilterSpecularLighting;
|
57
|
+
typedef struct _LsmSvgFilterTile LsmSvgFilterTile;
|
58
|
+
typedef struct _LsmSvgLineElement LsmSvgLineElement;
|
59
|
+
typedef struct _LsmSvgPolylineElement LsmSvgPolylineElement;
|
60
|
+
typedef struct _LsmSvgPolygonElement LsmSvgPolygonElement;
|
61
|
+
typedef struct _LsmSvgPathElement LsmSvgPathElement;
|
62
|
+
typedef struct _LsmSvgTextElement LsmSvgTextElement;
|
63
|
+
typedef struct _LsmSvgTspanElement LsmSvgTspanElement;
|
64
|
+
typedef struct _LsmSvgGradientElement LsmSvgGradientElement;
|
65
|
+
typedef struct _LsmSvgLinearGradientElement LsmSvgLinearGradientElement;
|
66
|
+
typedef struct _LsmSvgRadialGradientElement LsmSvgRadialGradientElement;
|
67
|
+
typedef struct _LsmSvgStopElement LsmSvgStopElement;
|
68
|
+
typedef struct _LsmSvgSwitchElement LsmSvgSwitchElement;
|
69
|
+
typedef struct _LsmSvgPatternElement LsmSvgPatternElement;
|
70
|
+
typedef struct _LsmSvgMaskElement LsmSvgMaskElement;
|
71
|
+
|
72
|
+
typedef struct _LsmSvgView LsmSvgView;
|
73
|
+
typedef struct _LsmSvgStyle LsmSvgStyle;
|
74
|
+
|
75
|
+
G_END_DECLS
|
76
|
+
|
77
|
+
#endif
|
@@ -0,0 +1,237 @@
|
|
1
|
+
/* Lasem
|
2
|
+
*
|
3
|
+
* Copyright © 2009 Emmanuel Pacaud
|
4
|
+
*
|
5
|
+
* This library is free software; you can redistribute it and/or
|
6
|
+
* modify it under the terms of the GNU Lesser General Public
|
7
|
+
* License as published by the Free Software Foundation; either
|
8
|
+
* version 2 of the License, or (at your option) any later version.
|
9
|
+
*
|
10
|
+
* This library is distributed in the hope that it will be useful,
|
11
|
+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
12
|
+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
13
|
+
* Lesser General Public License for more details.
|
14
|
+
*
|
15
|
+
* You should have received a copy of the GNU Lesser General
|
16
|
+
* Public License along with this library; if not, write to the
|
17
|
+
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
18
|
+
* Boston, MA 02111-1307, USA.
|
19
|
+
*
|
20
|
+
* Author:
|
21
|
+
* Emmanuel Pacaud <emmanuel@gnome.org>
|
22
|
+
*/
|
23
|
+
|
24
|
+
#include <lsmsvguseelement.h>
|
25
|
+
#include <lsmsvgview.h>
|
26
|
+
#include <lsmdebug.h>
|
27
|
+
#include <lsmsvgdocument.h>
|
28
|
+
#include <stdio.h>
|
29
|
+
|
30
|
+
static GObjectClass *parent_class;
|
31
|
+
|
32
|
+
/* GdomNode implementation */
|
33
|
+
|
34
|
+
static const char *
|
35
|
+
lsm_svg_use_element_get_node_name (LsmDomNode *node)
|
36
|
+
{
|
37
|
+
return "use";
|
38
|
+
}
|
39
|
+
|
40
|
+
static gboolean
|
41
|
+
lsm_svg_use_can_append_child (LsmDomNode *node, LsmDomNode *child)
|
42
|
+
{
|
43
|
+
return FALSE;
|
44
|
+
}
|
45
|
+
|
46
|
+
/* LsmSvgElement implementation */
|
47
|
+
|
48
|
+
/* LsmSvgGraphic implementation */
|
49
|
+
|
50
|
+
static LsmDomElement *
|
51
|
+
_get_used_element (LsmSvgUseElement *use_element)
|
52
|
+
{
|
53
|
+
LsmDomDocument *document;
|
54
|
+
LsmDomElement *element;
|
55
|
+
const char *id;
|
56
|
+
|
57
|
+
document = lsm_dom_node_get_owner_document (LSM_DOM_NODE (use_element));
|
58
|
+
if (document == NULL) {
|
59
|
+
lsm_debug_dom ("[LsmSvgUseElement::_get_used_element] Owner document not found");
|
60
|
+
return NULL;
|
61
|
+
}
|
62
|
+
|
63
|
+
id = use_element->href.value;
|
64
|
+
if (id == NULL)
|
65
|
+
return NULL;
|
66
|
+
|
67
|
+
if (*id == '#')
|
68
|
+
id++;
|
69
|
+
|
70
|
+
element = LSM_DOM_ELEMENT (lsm_svg_document_get_element_by_id (LSM_SVG_DOCUMENT (document), id));
|
71
|
+
if (!LSM_IS_SVG_ELEMENT (element)) {
|
72
|
+
lsm_debug_dom ("[LsmSvgUseElement::_get_used_element] Target '%s' not found", id);
|
73
|
+
return NULL;
|
74
|
+
}
|
75
|
+
|
76
|
+
return element;
|
77
|
+
}
|
78
|
+
|
79
|
+
static void
|
80
|
+
lsm_svg_use_element_render (LsmSvgElement *self, LsmSvgView *view)
|
81
|
+
{
|
82
|
+
LsmSvgUseElement *use_element = LSM_SVG_USE_ELEMENT (self);
|
83
|
+
LsmDomElement *element;
|
84
|
+
LsmSvgMatrix matrix;
|
85
|
+
double x, y;
|
86
|
+
|
87
|
+
if (use_element->flags & LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_RENDER) {
|
88
|
+
lsm_debug_render ("[LsmSvgUseElement::render] Circular reference");
|
89
|
+
return;
|
90
|
+
}
|
91
|
+
|
92
|
+
element = _get_used_element (use_element);
|
93
|
+
if (element == NULL)
|
94
|
+
return;
|
95
|
+
|
96
|
+
use_element->flags |= LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_RENDER;
|
97
|
+
|
98
|
+
x = lsm_svg_view_normalize_length (view, &use_element->x.length,
|
99
|
+
LSM_SVG_LENGTH_DIRECTION_HORIZONTAL);
|
100
|
+
y = lsm_svg_view_normalize_length (view, &use_element->y.length,
|
101
|
+
LSM_SVG_LENGTH_DIRECTION_VERTICAL);
|
102
|
+
|
103
|
+
lsm_svg_matrix_init_translate (&matrix, x, y);
|
104
|
+
|
105
|
+
if (lsm_svg_view_push_matrix (view, &matrix))
|
106
|
+
lsm_svg_element_render (LSM_SVG_ELEMENT (element), view);
|
107
|
+
|
108
|
+
lsm_svg_view_pop_matrix (view);
|
109
|
+
|
110
|
+
use_element->flags &= ~LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_RENDER;
|
111
|
+
}
|
112
|
+
|
113
|
+
static void
|
114
|
+
lsm_svg_use_element_get_extents (LsmSvgElement *self, LsmSvgView *view, LsmExtents *extents)
|
115
|
+
{
|
116
|
+
LsmSvgUseElement *use_element = LSM_SVG_USE_ELEMENT (self);
|
117
|
+
LsmDomElement *element;
|
118
|
+
LsmSvgMatrix matrix;
|
119
|
+
double x, y;
|
120
|
+
|
121
|
+
if (use_element->flags & LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_GET_EXTENTS) {
|
122
|
+
lsm_debug_render ("[LsmSvgUseElement::get_extents] Circular reference");
|
123
|
+
extents->x1 = 0;
|
124
|
+
extents->y1 = 0;
|
125
|
+
extents->x2 = 0;
|
126
|
+
extents->y2 = 0;
|
127
|
+
return;
|
128
|
+
}
|
129
|
+
|
130
|
+
element = _get_used_element (use_element);
|
131
|
+
if (element == NULL) {
|
132
|
+
extents->x1 = 0;
|
133
|
+
extents->y1 = 0;
|
134
|
+
extents->x2 = 0;
|
135
|
+
extents->y2 = 0;
|
136
|
+
return;
|
137
|
+
}
|
138
|
+
|
139
|
+
use_element->flags |= LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_GET_EXTENTS;
|
140
|
+
|
141
|
+
x = lsm_svg_view_normalize_length (view, &use_element->x.length,
|
142
|
+
LSM_SVG_LENGTH_DIRECTION_HORIZONTAL);
|
143
|
+
y = lsm_svg_view_normalize_length (view, &use_element->y.length,
|
144
|
+
LSM_SVG_LENGTH_DIRECTION_VERTICAL);
|
145
|
+
|
146
|
+
lsm_svg_element_transformed_get_extents (LSM_SVG_ELEMENT (element), view, extents);
|
147
|
+
|
148
|
+
lsm_svg_matrix_init_translate (&matrix, x, y);
|
149
|
+
|
150
|
+
if (lsm_svg_view_push_matrix (view, &matrix))
|
151
|
+
lsm_svg_matrix_transform_bounding_box (&matrix,
|
152
|
+
&extents->x1, &extents->y1,
|
153
|
+
&extents->x2, &extents->y2);
|
154
|
+
lsm_svg_view_pop_matrix (view);
|
155
|
+
|
156
|
+
use_element->flags &= ~LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_GET_EXTENTS;
|
157
|
+
}
|
158
|
+
|
159
|
+
/* LsmSvgUseElement implementation */
|
160
|
+
|
161
|
+
LsmDomNode *
|
162
|
+
lsm_svg_use_element_new (void)
|
163
|
+
{
|
164
|
+
return g_object_new (LSM_TYPE_SVG_USE_ELEMENT, NULL);
|
165
|
+
}
|
166
|
+
|
167
|
+
static const LsmSvgLength length_default = { .value_unit = 0.0, .type = LSM_SVG_LENGTH_TYPE_PX};
|
168
|
+
|
169
|
+
static void
|
170
|
+
lsm_svg_use_element_init (LsmSvgUseElement *self)
|
171
|
+
{
|
172
|
+
self->x.length = length_default;
|
173
|
+
self->y.length = length_default;
|
174
|
+
self->width.length = length_default;
|
175
|
+
self->height.length = length_default;
|
176
|
+
}
|
177
|
+
|
178
|
+
/* LsmSvgUseElement class */
|
179
|
+
|
180
|
+
static const LsmAttributeInfos lsm_svg_use_element_attribute_infos[] = {
|
181
|
+
{
|
182
|
+
.name = "x",
|
183
|
+
.attribute_offset = offsetof (LsmSvgUseElement, x),
|
184
|
+
.trait_class = &lsm_svg_length_trait_class,
|
185
|
+
.trait_default = &length_default
|
186
|
+
},
|
187
|
+
{
|
188
|
+
.name = "y",
|
189
|
+
.attribute_offset = offsetof (LsmSvgUseElement, y),
|
190
|
+
.trait_class = &lsm_svg_length_trait_class,
|
191
|
+
.trait_default = &length_default
|
192
|
+
},
|
193
|
+
{
|
194
|
+
.name = "width",
|
195
|
+
.attribute_offset = offsetof (LsmSvgUseElement, width),
|
196
|
+
.trait_class = &lsm_svg_length_trait_class,
|
197
|
+
.trait_default = &length_default
|
198
|
+
},
|
199
|
+
{
|
200
|
+
.name = "height",
|
201
|
+
.attribute_offset = offsetof (LsmSvgUseElement, height),
|
202
|
+
.trait_class = &lsm_svg_length_trait_class,
|
203
|
+
.trait_default = &length_default
|
204
|
+
},
|
205
|
+
{
|
206
|
+
.name = "xlink:href",
|
207
|
+
.attribute_offset = offsetof (LsmSvgUseElement, href),
|
208
|
+
.trait_class = &lsm_null_trait_class
|
209
|
+
}
|
210
|
+
};
|
211
|
+
|
212
|
+
static void
|
213
|
+
lsm_svg_use_element_class_init (LsmSvgUseElementClass *klass)
|
214
|
+
{
|
215
|
+
LsmDomNodeClass *d_node_class = LSM_DOM_NODE_CLASS (klass);
|
216
|
+
LsmSvgElementClass *s_element_class = LSM_SVG_ELEMENT_CLASS (klass);
|
217
|
+
|
218
|
+
parent_class = g_type_class_peek_parent (klass);
|
219
|
+
|
220
|
+
d_node_class->get_node_name = lsm_svg_use_element_get_node_name;
|
221
|
+
d_node_class->can_append_child = lsm_svg_use_can_append_child;
|
222
|
+
|
223
|
+
s_element_class->category =
|
224
|
+
LSM_SVG_ELEMENT_CATEGORY_GRAPHICS |
|
225
|
+
LSM_SVG_ELEMENT_CATEGORY_GRAPHICS_REFERENCING |
|
226
|
+
LSM_SVG_ELEMENT_CATEGORY_STRUCTURAL;
|
227
|
+
|
228
|
+
s_element_class->render = lsm_svg_use_element_render;
|
229
|
+
s_element_class->get_extents = lsm_svg_use_element_get_extents;
|
230
|
+
s_element_class->attribute_manager = lsm_attribute_manager_duplicate (s_element_class->attribute_manager);
|
231
|
+
|
232
|
+
lsm_attribute_manager_add_attributes (s_element_class->attribute_manager,
|
233
|
+
G_N_ELEMENTS (lsm_svg_use_element_attribute_infos),
|
234
|
+
lsm_svg_use_element_attribute_infos);
|
235
|
+
}
|
236
|
+
|
237
|
+
G_DEFINE_TYPE (LsmSvgUseElement, lsm_svg_use_element, LSM_TYPE_SVG_TRANSFORMABLE)
|
@@ -0,0 +1,69 @@
|
|
1
|
+
/* Lasem
|
2
|
+
*
|
3
|
+
* Copyright © 2009-2012 Emmanuel Pacaud
|
4
|
+
*
|
5
|
+
* This library is free software; you can redistribute it and/or
|
6
|
+
* modify it under the terms of the GNU Lesser General Public
|
7
|
+
* License as published by the Free Software Foundation; either
|
8
|
+
* version 2 of the License, or (at your option) any later version.
|
9
|
+
*
|
10
|
+
* This library is distributed in the hope that it will be useful,
|
11
|
+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
12
|
+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
13
|
+
* Lesser General Public License for more details.
|
14
|
+
*
|
15
|
+
* You should have received a copy of the GNU Lesser General
|
16
|
+
* Public License along with this library; if not, write to the
|
17
|
+
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
18
|
+
* Boston, MA 02111-1307, USA.
|
19
|
+
*
|
20
|
+
* Author:
|
21
|
+
* Emmanuel Pacaud <emmanuel@gnome.org>
|
22
|
+
*/
|
23
|
+
|
24
|
+
#ifndef LSM_SVG_USE_ELEMENT_H
|
25
|
+
#define LSM_SVG_USE_ELEMENT_H
|
26
|
+
|
27
|
+
#include <lsmsvgtypes.h>
|
28
|
+
#include <lsmsvgtransformable.h>
|
29
|
+
|
30
|
+
G_BEGIN_DECLS
|
31
|
+
|
32
|
+
typedef enum {
|
33
|
+
LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_RENDER = 1 << 0,
|
34
|
+
LSM_SVG_USE_ELEMENT_FLAGS_IN_USE_FOR_GET_EXTENTS = 1 << 1
|
35
|
+
} LsmSvgUseElementFlags;
|
36
|
+
|
37
|
+
#define LSM_TYPE_SVG_USE_ELEMENT (lsm_svg_use_element_get_type ())
|
38
|
+
#define LSM_SVG_USE_ELEMENT(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), LSM_TYPE_SVG_USE_ELEMENT, LsmSvgUseElement))
|
39
|
+
#define LSM_SVG_USE_ELEMENT_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), LSM_TYPE_SVG_USE_ELEMENT, LsmSvgUseElementClass))
|
40
|
+
#define LSM_IS_SVG_USE_ELEMENT(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), LSM_TYPE_SVG_USE_ELEMENT))
|
41
|
+
#define LSM_IS_SVG_USE_ELEMENT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), LSM_TYPE_SVG_USE_ELEMENT))
|
42
|
+
#define LSM_SVG_USE_ELEMENT_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj), LSM_TYPE_SVG_USE_ELEMENT, LsmSvgUseElementClass))
|
43
|
+
|
44
|
+
typedef struct _LsmSvgUseElementClass LsmSvgUseElementClass;
|
45
|
+
|
46
|
+
struct _LsmSvgUseElement {
|
47
|
+
LsmSvgTransformable base;
|
48
|
+
|
49
|
+
LsmSvgLengthAttribute x;
|
50
|
+
LsmSvgLengthAttribute y;
|
51
|
+
LsmSvgLengthAttribute width;
|
52
|
+
LsmSvgLengthAttribute height;
|
53
|
+
|
54
|
+
LsmAttribute href;
|
55
|
+
|
56
|
+
LsmSvgUseElementFlags flags;
|
57
|
+
};
|
58
|
+
|
59
|
+
struct _LsmSvgUseElementClass {
|
60
|
+
LsmSvgTransformableClass base_class;
|
61
|
+
};
|
62
|
+
|
63
|
+
GType lsm_svg_use_element_get_type (void);
|
64
|
+
|
65
|
+
LsmDomNode * lsm_svg_use_element_new (void);
|
66
|
+
|
67
|
+
G_END_DECLS
|
68
|
+
|
69
|
+
#endif
|
@@ -0,0 +1,2583 @@
|
|
1
|
+
/* Lasem
|
2
|
+
*
|
3
|
+
* Copyright © 2009 Emmanuel Pacaud
|
4
|
+
*
|
5
|
+
* This library is free software; you can redistribute it and/or
|
6
|
+
* modify it under the terms of the GNU Lesser General Public
|
7
|
+
* License as published by the Free Software Foundation; either
|
8
|
+
* version 2 of the License, or (at your option) any later version.
|
9
|
+
*
|
10
|
+
* This library is distributed in the hope that it will be useful,
|
11
|
+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
12
|
+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
13
|
+
* Lesser General Public License for more details.
|
14
|
+
*
|
15
|
+
* You should have received a copy of the GNU Lesser General
|
16
|
+
* Public License along with this library; if not, write to the
|
17
|
+
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
18
|
+
* Boston, MA 02111-1307, USA.
|
19
|
+
*
|
20
|
+
* Author:
|
21
|
+
* Emmanuel Pacaud <emmanuel@gnome.org>
|
22
|
+
*/
|
23
|
+
|
24
|
+
#include <lsmdebug.h>
|
25
|
+
#include <lsmsvgview.h>
|
26
|
+
#include <lsmsvgdocument.h>
|
27
|
+
#include <lsmsvgelement.h>
|
28
|
+
#include <lsmsvgsvgelement.h>
|
29
|
+
#include <lsmsvgradialgradientelement.h>
|
30
|
+
#include <lsmsvgfilterelement.h>
|
31
|
+
#include <lsmsvglineargradientelement.h>
|
32
|
+
#include <lsmsvgpatternelement.h>
|
33
|
+
#include <lsmsvgmarkerelement.h>
|
34
|
+
#include <lsmsvgclippathelement.h>
|
35
|
+
#include <lsmsvgmaskelement.h>
|
36
|
+
#include <lsmsvgfiltersurface.h>
|
37
|
+
#include <lsmcairo.h>
|
38
|
+
#include <lsmstr.h>
|
39
|
+
#include <gdk-pixbuf/gdk-pixbuf.h>
|
40
|
+
#include <glib/gprintf.h>
|
41
|
+
|
42
|
+
#include <math.h>
|
43
|
+
#include <string.h>
|
44
|
+
|
45
|
+
static gboolean lsm_svg_view_circular_reference_check (LsmSvgView *view, LsmSvgElement *element);
|
46
|
+
|
47
|
+
static GObjectClass *parent_class;
|
48
|
+
|
49
|
+
typedef struct {
|
50
|
+
gboolean is_text_path;
|
51
|
+
gboolean is_extents_defined;
|
52
|
+
LsmExtents extents;
|
53
|
+
PangoLayout *pango_layout;
|
54
|
+
} LsmSvgViewPathInfos;
|
55
|
+
|
56
|
+
static LsmSvgViewPathInfos default_path_infos = {
|
57
|
+
.is_text_path = FALSE,
|
58
|
+
.is_extents_defined = FALSE,
|
59
|
+
.extents = {0.0, 0.0, 0.0, 0.0},
|
60
|
+
.pango_layout = NULL
|
61
|
+
};
|
62
|
+
|
63
|
+
struct _LsmSvgViewPatternData {
|
64
|
+
cairo_t *old_cairo;
|
65
|
+
|
66
|
+
cairo_pattern_t *pattern;
|
67
|
+
|
68
|
+
LsmBox extents;
|
69
|
+
LsmBox object_extents;
|
70
|
+
|
71
|
+
double opacity;
|
72
|
+
};
|
73
|
+
|
74
|
+
typedef struct {
|
75
|
+
cairo_surface_t *surface;
|
76
|
+
double group_opacity;
|
77
|
+
gboolean enable_background;
|
78
|
+
} LsmSvgViewBackground;
|
79
|
+
|
80
|
+
cairo_operator_t cairo_operators[] = {
|
81
|
+
CAIRO_OPERATOR_CLEAR,
|
82
|
+
CAIRO_OPERATOR_SOURCE,
|
83
|
+
CAIRO_OPERATOR_DEST,
|
84
|
+
CAIRO_OPERATOR_OVER,
|
85
|
+
CAIRO_OPERATOR_DEST_OVER,
|
86
|
+
CAIRO_OPERATOR_IN,
|
87
|
+
CAIRO_OPERATOR_DEST_IN,
|
88
|
+
CAIRO_OPERATOR_OUT,
|
89
|
+
CAIRO_OPERATOR_DEST_OUT,
|
90
|
+
CAIRO_OPERATOR_ATOP,
|
91
|
+
CAIRO_OPERATOR_DEST_ATOP,
|
92
|
+
CAIRO_OPERATOR_XOR,
|
93
|
+
CAIRO_OPERATOR_ADD,
|
94
|
+
CAIRO_OPERATOR_MULTIPLY,
|
95
|
+
CAIRO_OPERATOR_SCREEN,
|
96
|
+
CAIRO_OPERATOR_OVERLAY,
|
97
|
+
CAIRO_OPERATOR_DARKEN,
|
98
|
+
CAIRO_OPERATOR_LIGHTEN,
|
99
|
+
CAIRO_OPERATOR_COLOR_DODGE,
|
100
|
+
CAIRO_OPERATOR_COLOR_BURN,
|
101
|
+
CAIRO_OPERATOR_HARD_LIGHT,
|
102
|
+
CAIRO_OPERATOR_SOFT_LIGHT,
|
103
|
+
CAIRO_OPERATOR_DIFFERENCE,
|
104
|
+
CAIRO_OPERATOR_EXCLUSION
|
105
|
+
};
|
106
|
+
|
107
|
+
static void
|
108
|
+
lsm_cairo_set_comp_op (cairo_t *cairo, LsmSvgCompOp comp_op)
|
109
|
+
{
|
110
|
+
if (G_LIKELY (cairo != NULL && comp_op >= LSM_SVG_COMP_OP_CLEAR && comp_op <= LSM_SVG_COMP_OP_EXCLUSION)) {
|
111
|
+
lsm_log_render ("[lsmSvgView::set_comp_op] Set comp-op to %s", lsm_svg_comp_op_to_string (comp_op));
|
112
|
+
cairo_set_operator (cairo, cairo_operators[comp_op]);
|
113
|
+
}
|
114
|
+
}
|
115
|
+
|
116
|
+
double
|
117
|
+
lsm_svg_view_normalize_length (LsmSvgView *view, const LsmSvgLength *length, LsmSvgLengthDirection direction)
|
118
|
+
{
|
119
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), 0.0);
|
120
|
+
|
121
|
+
return lsm_svg_length_normalize (length, view->viewbox_stack->data, view->style->font_size_px, direction);
|
122
|
+
}
|
123
|
+
|
124
|
+
double *
|
125
|
+
lsm_svg_view_normalize_length_list (LsmSvgView *view, const LsmSvgLengthList *list, LsmSvgLengthDirection direction, unsigned int *n_data)
|
126
|
+
{
|
127
|
+
double *data;
|
128
|
+
unsigned int i;
|
129
|
+
|
130
|
+
g_return_val_if_fail (n_data != NULL, NULL);
|
131
|
+
*n_data = 0;
|
132
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), NULL);
|
133
|
+
|
134
|
+
if (list->n_lengths == 0)
|
135
|
+
return NULL;
|
136
|
+
|
137
|
+
*n_data = list->n_lengths;
|
138
|
+
data = g_new (double, list->n_lengths);
|
139
|
+
for (i = 0; i < list->n_lengths; i++)
|
140
|
+
data[i] = lsm_svg_view_normalize_length (view, &list->lengths[i], direction);
|
141
|
+
|
142
|
+
return data;
|
143
|
+
}
|
144
|
+
|
145
|
+
static void
|
146
|
+
_start_pattern (LsmSvgView *view, const LsmBox *extents, const LsmBox *object_extents, double opacity)
|
147
|
+
{
|
148
|
+
lsm_debug_render ("[LsmSvgView::start_pattern]");
|
149
|
+
|
150
|
+
view->pattern_stack = g_slist_prepend (view->pattern_stack, view->pattern_data);
|
151
|
+
|
152
|
+
view->pattern_data = g_new (LsmSvgViewPatternData, 1);
|
153
|
+
view->pattern_data->old_cairo = view->dom_view.cairo;
|
154
|
+
view->pattern_data->pattern = NULL;
|
155
|
+
view->pattern_data->extents = *extents;
|
156
|
+
view->pattern_data->opacity = opacity;
|
157
|
+
view->pattern_data->object_extents = *object_extents;
|
158
|
+
|
159
|
+
view->dom_view.cairo = NULL;
|
160
|
+
}
|
161
|
+
|
162
|
+
static void
|
163
|
+
_end_pattern (LsmSvgView *view)
|
164
|
+
{
|
165
|
+
g_return_if_fail (view->pattern_data != NULL);
|
166
|
+
|
167
|
+
if (view->pattern_data->pattern != NULL)
|
168
|
+
cairo_pattern_destroy (view->pattern_data->pattern);
|
169
|
+
if (view->dom_view.cairo != NULL)
|
170
|
+
cairo_destroy (view->dom_view.cairo);
|
171
|
+
|
172
|
+
view->dom_view.cairo = view->pattern_data->old_cairo;
|
173
|
+
|
174
|
+
g_free (view->pattern_data);
|
175
|
+
|
176
|
+
if (view->pattern_stack != NULL) {
|
177
|
+
view->pattern_data = view->pattern_stack->data;
|
178
|
+
view->pattern_stack = g_slist_delete_link (view->pattern_stack, view->pattern_stack);
|
179
|
+
} else
|
180
|
+
view->pattern_data = NULL;
|
181
|
+
|
182
|
+
lsm_debug_render ("[LsmSvgView::end_pattern]");
|
183
|
+
}
|
184
|
+
|
185
|
+
static void
|
186
|
+
_set_pattern (LsmSvgView *view, cairo_pattern_t *pattern)
|
187
|
+
{
|
188
|
+
g_return_if_fail (view->pattern_data != NULL);
|
189
|
+
g_return_if_fail (view->pattern_data->pattern == NULL);
|
190
|
+
|
191
|
+
view->pattern_data->pattern = pattern;
|
192
|
+
view->last_stop_offset = 0.0;
|
193
|
+
}
|
194
|
+
|
195
|
+
void
|
196
|
+
lsm_svg_view_create_radial_gradient (LsmSvgView *view,
|
197
|
+
double cx, double cy,
|
198
|
+
double r,
|
199
|
+
double fx, double fy)
|
200
|
+
{
|
201
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
202
|
+
|
203
|
+
_set_pattern (view, cairo_pattern_create_radial (fx, fy, 0, cx, cy, r));
|
204
|
+
}
|
205
|
+
|
206
|
+
void
|
207
|
+
lsm_svg_view_create_linear_gradient (LsmSvgView *view,
|
208
|
+
double x1, double y1,
|
209
|
+
double x2, double y2)
|
210
|
+
{
|
211
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
212
|
+
|
213
|
+
_set_pattern (view, cairo_pattern_create_linear (x1, y1, x2, y2));
|
214
|
+
}
|
215
|
+
|
216
|
+
void
|
217
|
+
lsm_svg_view_add_gradient_color_stop (LsmSvgView *view, double offset)
|
218
|
+
{
|
219
|
+
const LsmSvgStyle *style;
|
220
|
+
const LsmSvgColor *color;
|
221
|
+
|
222
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
223
|
+
g_return_if_fail (view->pattern_data != NULL);
|
224
|
+
g_return_if_fail (view->pattern_data->pattern != NULL);
|
225
|
+
|
226
|
+
if (offset > 1.0)
|
227
|
+
offset = 1.0;
|
228
|
+
|
229
|
+
if (offset < view->last_stop_offset)
|
230
|
+
offset = view->last_stop_offset;
|
231
|
+
else
|
232
|
+
view->last_stop_offset = offset;
|
233
|
+
|
234
|
+
style = view->style;
|
235
|
+
|
236
|
+
lsm_debug_render ("[LsmSvgView::add_gradient_color_stop] opacity = %g", style->stop_opacity->value);
|
237
|
+
|
238
|
+
color = &style->stop_color->value;
|
239
|
+
|
240
|
+
if (color->red < 0.0 || color->blue < 0.0 || color->green < 0.0)
|
241
|
+
color = &style->color->value;
|
242
|
+
|
243
|
+
lsm_debug_render ("[LsmSvgView::add_gradient_color_stop] color = %2x%2x%2x",
|
244
|
+
(int) (255.0 * color->red),
|
245
|
+
(int) (255.0 * color->green),
|
246
|
+
(int) (255.0 * color->blue));
|
247
|
+
|
248
|
+
cairo_pattern_add_color_stop_rgba (view->pattern_data->pattern, offset,
|
249
|
+
color->red,
|
250
|
+
color->green,
|
251
|
+
color->blue,
|
252
|
+
style->stop_opacity->value * view->pattern_data->opacity);
|
253
|
+
}
|
254
|
+
|
255
|
+
gboolean
|
256
|
+
lsm_svg_view_set_gradient_properties (LsmSvgView *view,
|
257
|
+
LsmSvgSpreadMethod method,
|
258
|
+
LsmSvgPatternUnits units,
|
259
|
+
const LsmSvgMatrix *gradient_matrix)
|
260
|
+
{
|
261
|
+
cairo_matrix_t matrix;
|
262
|
+
cairo_matrix_t inv_matrix;
|
263
|
+
cairo_status_t status;
|
264
|
+
|
265
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), FALSE);
|
266
|
+
g_return_val_if_fail (view->pattern_data != NULL, FALSE);
|
267
|
+
g_return_val_if_fail (view->pattern_data->pattern != NULL, FALSE);
|
268
|
+
|
269
|
+
switch (method) {
|
270
|
+
case LSM_SVG_SPREAD_METHOD_REFLECT:
|
271
|
+
cairo_pattern_set_extend (view->pattern_data->pattern, CAIRO_EXTEND_REFLECT);
|
272
|
+
break;
|
273
|
+
case LSM_SVG_SPREAD_METHOD_REPEAT:
|
274
|
+
cairo_pattern_set_extend (view->pattern_data->pattern, CAIRO_EXTEND_REPEAT);
|
275
|
+
break;
|
276
|
+
default:
|
277
|
+
cairo_pattern_set_extend (view->pattern_data->pattern, CAIRO_EXTEND_PAD);
|
278
|
+
}
|
279
|
+
|
280
|
+
if (gradient_matrix != NULL) {
|
281
|
+
cairo_matrix_init (&matrix,
|
282
|
+
gradient_matrix->a, gradient_matrix->b,
|
283
|
+
gradient_matrix->c, gradient_matrix->d,
|
284
|
+
gradient_matrix->e, gradient_matrix->f);
|
285
|
+
cairo_matrix_invert (&matrix);
|
286
|
+
} else
|
287
|
+
cairo_matrix_init_identity (&matrix);
|
288
|
+
|
289
|
+
if (units == LSM_SVG_PATTERN_UNITS_OBJECT_BOUNDING_BOX &&
|
290
|
+
view->pattern_data->extents.width > 0.0 &&
|
291
|
+
view->pattern_data->extents.height > 0.0) {
|
292
|
+
cairo_matrix_scale (&matrix,
|
293
|
+
1.0 / view->pattern_data->extents.width,
|
294
|
+
1.0 / view->pattern_data->extents.height);
|
295
|
+
cairo_matrix_translate (&matrix,
|
296
|
+
-view->pattern_data->extents.x,
|
297
|
+
-view->pattern_data->extents.y);
|
298
|
+
|
299
|
+
}
|
300
|
+
|
301
|
+
inv_matrix = matrix;
|
302
|
+
status = cairo_matrix_invert (&inv_matrix);
|
303
|
+
|
304
|
+
if (status != CAIRO_STATUS_SUCCESS) {
|
305
|
+
lsm_debug_render ("[LsmSvgView::set_gradient_properties] Not invertible matrix");
|
306
|
+
return FALSE;
|
307
|
+
}
|
308
|
+
|
309
|
+
cairo_pattern_set_matrix (view->pattern_data->pattern, &matrix);
|
310
|
+
|
311
|
+
return TRUE;
|
312
|
+
}
|
313
|
+
|
314
|
+
gboolean
|
315
|
+
lsm_svg_view_create_surface_pattern (LsmSvgView *view,
|
316
|
+
const LsmBox *viewport,
|
317
|
+
const LsmSvgMatrix *pattern_matrix,
|
318
|
+
LsmSvgViewSurfaceType surface_type)
|
319
|
+
{
|
320
|
+
cairo_surface_t *surface;
|
321
|
+
cairo_pattern_t *pattern;
|
322
|
+
cairo_matrix_t matrix;
|
323
|
+
cairo_matrix_t inv_matrix;
|
324
|
+
cairo_status_t status;
|
325
|
+
double x1, y1, x2, y2;
|
326
|
+
double device_width, device_height;
|
327
|
+
double x_scale, y_scale;
|
328
|
+
|
329
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), FALSE);
|
330
|
+
g_return_val_if_fail (viewport != NULL, FALSE);
|
331
|
+
g_return_val_if_fail (view->pattern_data != NULL, FALSE);
|
332
|
+
g_return_val_if_fail (view->dom_view.cairo == NULL, FALSE);
|
333
|
+
|
334
|
+
x1 = viewport->x;
|
335
|
+
y1 = viewport->y;
|
336
|
+
x2 = viewport->x + viewport->width;
|
337
|
+
y2 = viewport->y;
|
338
|
+
|
339
|
+
cairo_user_to_device (view->pattern_data->old_cairo, &x1, &y1);
|
340
|
+
cairo_user_to_device (view->pattern_data->old_cairo, &x2, &y2);
|
341
|
+
|
342
|
+
device_width = sqrt ((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
|
343
|
+
|
344
|
+
x2 = viewport->x;
|
345
|
+
y2 = viewport->y + viewport->height;
|
346
|
+
|
347
|
+
cairo_user_to_device (view->pattern_data->old_cairo, &x2, &y2);
|
348
|
+
|
349
|
+
device_height = sqrt ((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
|
350
|
+
|
351
|
+
if (surface_type == LSM_SVG_VIEW_SURFACE_TYPE_IMAGE) {
|
352
|
+
device_height = ceil (device_height);
|
353
|
+
device_width = ceil (device_width);
|
354
|
+
}
|
355
|
+
|
356
|
+
x_scale = device_width / viewport->width;
|
357
|
+
y_scale = device_height / viewport->height;
|
358
|
+
|
359
|
+
lsm_debug_render ("[LsmSvgView::create_surface_pattern] pattern size = %g ,%g at %g, %g (scale %g x %g)",
|
360
|
+
device_width, device_height, viewport->x, viewport->y, x_scale, y_scale);
|
361
|
+
|
362
|
+
switch (surface_type) {
|
363
|
+
case LSM_SVG_VIEW_SURFACE_TYPE_AUTO:
|
364
|
+
surface = cairo_surface_create_similar (cairo_get_target (view->pattern_data->old_cairo),
|
365
|
+
CAIRO_CONTENT_COLOR_ALPHA,
|
366
|
+
device_width, device_height);
|
367
|
+
break;
|
368
|
+
default:
|
369
|
+
case LSM_SVG_VIEW_SURFACE_TYPE_IMAGE:
|
370
|
+
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, device_width, device_height);
|
371
|
+
break;
|
372
|
+
}
|
373
|
+
|
374
|
+
pattern = cairo_pattern_create_for_surface (surface);
|
375
|
+
view->dom_view.cairo = cairo_create (surface);
|
376
|
+
cairo_surface_destroy (surface);
|
377
|
+
|
378
|
+
cairo_scale (view->dom_view.cairo, x_scale, y_scale);
|
379
|
+
cairo_translate (view->dom_view.cairo, -viewport->x, -viewport->y);
|
380
|
+
|
381
|
+
_set_pattern (view, pattern);
|
382
|
+
|
383
|
+
if (pattern_matrix != NULL) {
|
384
|
+
cairo_matrix_init (&matrix,
|
385
|
+
pattern_matrix->a, pattern_matrix->b,
|
386
|
+
pattern_matrix->c, pattern_matrix->d,
|
387
|
+
pattern_matrix->e + viewport->x, pattern_matrix->f + viewport->y);
|
388
|
+
cairo_matrix_scale (&matrix, 1.0 / x_scale, 1.0 / y_scale);
|
389
|
+
cairo_matrix_invert (&matrix);
|
390
|
+
} else {
|
391
|
+
cairo_matrix_init_scale (&matrix, x_scale, y_scale);
|
392
|
+
cairo_matrix_translate (&matrix, -viewport->x, -viewport->y);
|
393
|
+
}
|
394
|
+
|
395
|
+
inv_matrix = matrix;
|
396
|
+
status = cairo_matrix_invert (&inv_matrix);
|
397
|
+
|
398
|
+
if (status != CAIRO_STATUS_SUCCESS) {
|
399
|
+
lsm_debug_render ("[LsmSvgView::create_surface_pattern] Not invertible matrix");
|
400
|
+
return FALSE;
|
401
|
+
}
|
402
|
+
|
403
|
+
cairo_pattern_set_matrix (view->pattern_data->pattern, &matrix);
|
404
|
+
cairo_pattern_set_extend (view->pattern_data->pattern, CAIRO_EXTEND_REPEAT);
|
405
|
+
|
406
|
+
lsm_debug_render ("[LsmSvgView::create_surface_pattern] Pattern matrix %g, %g, %g, %g, %g, %g",
|
407
|
+
matrix.xx, matrix.xy, matrix.yx, matrix.yy,
|
408
|
+
matrix.x0, matrix.y0);
|
409
|
+
|
410
|
+
return TRUE;
|
411
|
+
}
|
412
|
+
|
413
|
+
typedef enum {
|
414
|
+
LSM_SVG_VIEW_PAINT_OPERATION_FILL,
|
415
|
+
LSM_SVG_VIEW_PAINT_OPERATION_STROKE
|
416
|
+
} LsmSvgViewPaintOperation;
|
417
|
+
|
418
|
+
static void
|
419
|
+
_paint_url (LsmSvgView *view,
|
420
|
+
LsmSvgViewPathInfos *path_infos,
|
421
|
+
LsmSvgViewPaintOperation operation,
|
422
|
+
const char *url, double opacity)
|
423
|
+
{
|
424
|
+
cairo_t *cairo;
|
425
|
+
LsmSvgElement *element;
|
426
|
+
LsmBox extents;
|
427
|
+
|
428
|
+
element = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document), url);
|
429
|
+
if ((!LSM_IS_SVG_RADIAL_GRADIENT_ELEMENT (element) &&
|
430
|
+
!LSM_IS_SVG_LINEAR_GRADIENT_ELEMENT (element) &&
|
431
|
+
!LSM_IS_SVG_PATTERN_ELEMENT (element)) ||
|
432
|
+
lsm_svg_view_circular_reference_check (view, element)) {
|
433
|
+
|
434
|
+
cairo_set_source_rgba (view->dom_view.cairo, 0, 0, 0, 0);
|
435
|
+
|
436
|
+
lsm_warning_render ("[LsmSvgView::_paint_url] Paint url not found: %s", url);
|
437
|
+
|
438
|
+
return;
|
439
|
+
}
|
440
|
+
|
441
|
+
lsm_debug_render ("[LsmSvgView::_paint_url] Paint using '%s'", url);
|
442
|
+
|
443
|
+
if (!path_infos->is_extents_defined) {
|
444
|
+
cairo_path_extents (view->dom_view.cairo,
|
445
|
+
&path_infos->extents.x1,
|
446
|
+
&path_infos->extents.y1,
|
447
|
+
&path_infos->extents.x2,
|
448
|
+
&path_infos->extents.y2);
|
449
|
+
path_infos->is_extents_defined = TRUE;
|
450
|
+
}
|
451
|
+
|
452
|
+
extents.x = path_infos->extents.x1;
|
453
|
+
extents.y = path_infos->extents.y1;
|
454
|
+
extents.width = path_infos->extents.x2 - extents.x;
|
455
|
+
extents.height = path_infos->extents.y2 - extents.y;
|
456
|
+
|
457
|
+
lsm_debug_render ("[LsmSvgView::_paint_url] Pattern extents x = %g, y = %g, w = %g, h = %g",
|
458
|
+
extents.x, extents.y, extents.width, extents.height);
|
459
|
+
|
460
|
+
_start_pattern (view, &extents, &extents, opacity);
|
461
|
+
|
462
|
+
lsm_svg_element_force_render (LSM_SVG_ELEMENT (element), view);
|
463
|
+
|
464
|
+
cairo = view->pattern_data->old_cairo;
|
465
|
+
|
466
|
+
if (view->pattern_data->pattern) {
|
467
|
+
if (view->debug_pattern && view->dom_view.cairo) {
|
468
|
+
char *filename;
|
469
|
+
|
470
|
+
filename = g_strdup_printf ("pattern-%s).png", url);
|
471
|
+
cairo_surface_write_to_png (cairo_get_target (view->dom_view.cairo), filename);
|
472
|
+
g_free (filename);
|
473
|
+
}
|
474
|
+
|
475
|
+
cairo_set_source (cairo, view->pattern_data->pattern);
|
476
|
+
} else
|
477
|
+
cairo_set_source_rgba (cairo, 0.0, 0.0, 0.0, 0.0);
|
478
|
+
|
479
|
+
_end_pattern (view);
|
480
|
+
}
|
481
|
+
|
482
|
+
static gboolean
|
483
|
+
_set_color (LsmSvgView *view,
|
484
|
+
LsmSvgViewPathInfos *path_infos,
|
485
|
+
LsmSvgViewPaintOperation operation,
|
486
|
+
const LsmSvgPaint *paint, double opacity)
|
487
|
+
{
|
488
|
+
cairo_t *cairo = view->dom_view.cairo;
|
489
|
+
|
490
|
+
switch (paint->type) {
|
491
|
+
case LSM_SVG_PAINT_TYPE_NONE:
|
492
|
+
return FALSE;
|
493
|
+
case LSM_SVG_PAINT_TYPE_RGB_COLOR:
|
494
|
+
cairo_set_source_rgba (cairo,
|
495
|
+
paint->color.red,
|
496
|
+
paint->color.green,
|
497
|
+
paint->color.blue,
|
498
|
+
opacity);
|
499
|
+
break;
|
500
|
+
case LSM_SVG_PAINT_TYPE_CURRENT_COLOR:
|
501
|
+
cairo_set_source_rgba (cairo,
|
502
|
+
view->style->color->value.red,
|
503
|
+
view->style->color->value.green,
|
504
|
+
view->style->color->value.blue,
|
505
|
+
opacity);
|
506
|
+
break;
|
507
|
+
case LSM_SVG_PAINT_TYPE_URI:
|
508
|
+
case LSM_SVG_PAINT_TYPE_URI_RGB_COLOR:
|
509
|
+
case LSM_SVG_PAINT_TYPE_URI_CURRENT_COLOR:
|
510
|
+
case LSM_SVG_PAINT_TYPE_URI_NONE:
|
511
|
+
_paint_url (view, path_infos, operation, paint->url, opacity);
|
512
|
+
break;
|
513
|
+
default:
|
514
|
+
return FALSE;
|
515
|
+
}
|
516
|
+
|
517
|
+
return TRUE;
|
518
|
+
}
|
519
|
+
|
520
|
+
static void
|
521
|
+
paint_markers (LsmSvgView *view)
|
522
|
+
{
|
523
|
+
const LsmSvgStyle *style;
|
524
|
+
LsmSvgElement *marker;
|
525
|
+
LsmSvgElement *marker_start;
|
526
|
+
LsmSvgElement *marker_mid;
|
527
|
+
LsmSvgElement *marker_end;
|
528
|
+
cairo_t *cairo;
|
529
|
+
cairo_path_t *path;
|
530
|
+
cairo_path_data_t *data;
|
531
|
+
cairo_path_data_t *next_data;
|
532
|
+
double stroke_width;
|
533
|
+
double prev_x, prev_y;
|
534
|
+
double x = 0;
|
535
|
+
double y = 0;
|
536
|
+
double next_x, next_y;
|
537
|
+
cairo_path_data_type_t type;
|
538
|
+
cairo_path_data_type_t next_type;
|
539
|
+
double angle;
|
540
|
+
int i;
|
541
|
+
|
542
|
+
style = view->style;
|
543
|
+
|
544
|
+
if ((style->marker->value == NULL || strcmp (style->marker->value, "none") == 0) &&
|
545
|
+
(style->marker_mid->value == NULL || strcmp (style->marker_mid->value, "none") == 0) &&
|
546
|
+
(style->marker_end->value == NULL || strcmp (style->marker_end->value, "none") == 0) &&
|
547
|
+
(style->marker_start->value == NULL || strcmp (style->marker_start->value, "none") == 0))
|
548
|
+
return;
|
549
|
+
|
550
|
+
cairo = view->dom_view.cairo;
|
551
|
+
|
552
|
+
marker = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
553
|
+
style->marker->value);
|
554
|
+
marker_start = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
555
|
+
style->marker_start->value);
|
556
|
+
marker_mid = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
557
|
+
style->marker_mid->value);
|
558
|
+
marker_end = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
559
|
+
style->marker_end->value);
|
560
|
+
stroke_width = lsm_svg_view_normalize_length (view, &view->style->stroke_width->length,
|
561
|
+
LSM_SVG_LENGTH_DIRECTION_DIAGONAL);
|
562
|
+
|
563
|
+
if (marker != NULL && lsm_svg_view_circular_reference_check (view, marker))
|
564
|
+
return;
|
565
|
+
if (marker_start != NULL && lsm_svg_view_circular_reference_check (view, marker_start))
|
566
|
+
return;
|
567
|
+
if (marker_mid != NULL && lsm_svg_view_circular_reference_check (view, marker_mid))
|
568
|
+
return;
|
569
|
+
if (marker_end != NULL && lsm_svg_view_circular_reference_check (view, marker_end))
|
570
|
+
return;
|
571
|
+
|
572
|
+
if (marker_start == NULL)
|
573
|
+
marker_start = marker;
|
574
|
+
if (marker_mid == NULL)
|
575
|
+
marker_mid = marker;
|
576
|
+
if (marker_end == NULL)
|
577
|
+
marker_end = marker;
|
578
|
+
|
579
|
+
path = cairo_copy_path (cairo);
|
580
|
+
cairo_new_path (cairo);
|
581
|
+
|
582
|
+
if (path->num_data > 0) {
|
583
|
+
next_data = &path->data[0];
|
584
|
+
next_type = next_data->header.type;
|
585
|
+
|
586
|
+
if (next_type == CAIRO_PATH_CURVE_TO) {
|
587
|
+
next_x = next_data[3].point.x;
|
588
|
+
next_y = next_data[3].point.y;
|
589
|
+
} else {
|
590
|
+
next_x = next_data[1].point.x;
|
591
|
+
next_y = next_data[1].point.y;
|
592
|
+
}
|
593
|
+
|
594
|
+
for (i = 0; i < path->num_data; i += path->data[i].header.length) {
|
595
|
+
data = next_data;
|
596
|
+
|
597
|
+
prev_x = x;
|
598
|
+
prev_y = y;
|
599
|
+
x = next_x;
|
600
|
+
y = next_y;
|
601
|
+
type = next_type;
|
602
|
+
|
603
|
+
if (i + path->data[i].header.length < path->num_data) {
|
604
|
+
next_data = &path->data[i + path->data[i].header.length];
|
605
|
+
next_type = next_data->header.type;
|
606
|
+
|
607
|
+
if (next_type == CAIRO_PATH_CURVE_TO) {
|
608
|
+
next_x = next_data[3].point.x;
|
609
|
+
next_y = next_data[3].point.y;
|
610
|
+
} else {
|
611
|
+
next_x = next_data[1].point.x;
|
612
|
+
next_y = next_data[1].point.y;
|
613
|
+
}
|
614
|
+
} else {
|
615
|
+
next_data = NULL;
|
616
|
+
next_type = CAIRO_PATH_MOVE_TO;
|
617
|
+
next_x = 0.0;
|
618
|
+
next_y = 0.0;
|
619
|
+
}
|
620
|
+
|
621
|
+
if (data->header.type == CAIRO_PATH_CLOSE_PATH) {
|
622
|
+
marker = NULL;
|
623
|
+
} else if (next_data == NULL ||
|
624
|
+
next_data->header.type == CAIRO_PATH_MOVE_TO) {
|
625
|
+
marker = marker_end;
|
626
|
+
if (type == CAIRO_PATH_CURVE_TO) {
|
627
|
+
if (y != data[2].point.y ||
|
628
|
+
x != data[2].point.x)
|
629
|
+
angle = atan2 (y - data[2].point.y,
|
630
|
+
x - data[2].point.x);
|
631
|
+
else
|
632
|
+
angle = atan2 (y - data[1].point.y,
|
633
|
+
x - data[1].point.x);
|
634
|
+
} else
|
635
|
+
angle = atan2 (y - prev_y, x - prev_x);
|
636
|
+
} else if (data->header.type == CAIRO_PATH_MOVE_TO) {
|
637
|
+
marker = marker_start;
|
638
|
+
if (next_type == CAIRO_PATH_CURVE_TO) {
|
639
|
+
if (next_data[1].point.y != y ||
|
640
|
+
next_data[1].point.x != x)
|
641
|
+
angle = atan2 (next_data[1].point.y - y,
|
642
|
+
next_data[1].point.x - x);
|
643
|
+
else
|
644
|
+
angle = atan2 (next_data[2].point.y - y,
|
645
|
+
next_data[2].point.x - x);
|
646
|
+
} else
|
647
|
+
angle = atan2 (next_y - y, next_x - x);
|
648
|
+
} else {
|
649
|
+
double xdifin, ydifin, xdifout, ydifout, intot, outtot;
|
650
|
+
|
651
|
+
marker = marker_mid;
|
652
|
+
|
653
|
+
if (type == CAIRO_PATH_CURVE_TO) {
|
654
|
+
if (x != data[2].point.x ||
|
655
|
+
y != data[2].point.y) {
|
656
|
+
xdifin = x - data[2].point.x;
|
657
|
+
ydifin = y - data[2].point.y;
|
658
|
+
} else {
|
659
|
+
xdifin = x - data[1].point.x;
|
660
|
+
ydifin = y - data[1].point.y;
|
661
|
+
}
|
662
|
+
} else {
|
663
|
+
xdifin = x - prev_x;
|
664
|
+
ydifin = y - prev_y;
|
665
|
+
}
|
666
|
+
if (next_type == CAIRO_PATH_CURVE_TO) {
|
667
|
+
if (next_data[1].point.x != x ||
|
668
|
+
next_data[1].point.y != y) {
|
669
|
+
xdifout = next_data[1].point.x - x;
|
670
|
+
ydifout = next_data[1].point.y - y;
|
671
|
+
} else {
|
672
|
+
xdifout = next_data[2].point.x - x;
|
673
|
+
ydifout = next_data[2].point.y - y;
|
674
|
+
}
|
675
|
+
} else {
|
676
|
+
xdifout = next_x - x;
|
677
|
+
ydifout = next_y - y;
|
678
|
+
}
|
679
|
+
|
680
|
+
intot = sqrt (xdifin * xdifin + ydifin * ydifin);
|
681
|
+
outtot = sqrt (xdifout * xdifout + ydifout * ydifout);
|
682
|
+
|
683
|
+
xdifin /= intot;
|
684
|
+
ydifin /= intot;
|
685
|
+
xdifout /= outtot;
|
686
|
+
ydifout /= outtot;
|
687
|
+
|
688
|
+
angle = atan2 ((ydifin + ydifout) / 2, (xdifin + xdifout) / 2);
|
689
|
+
}
|
690
|
+
|
691
|
+
if (marker != NULL) {
|
692
|
+
cairo_save (cairo);
|
693
|
+
cairo_translate (cairo, x, y);
|
694
|
+
lsm_svg_marker_element_render (LSM_SVG_MARKER_ELEMENT (marker), view,
|
695
|
+
stroke_width, angle);
|
696
|
+
cairo_restore (cairo);
|
697
|
+
}
|
698
|
+
}
|
699
|
+
|
700
|
+
cairo_path_destroy (path);
|
701
|
+
}
|
702
|
+
}
|
703
|
+
|
704
|
+
static void
|
705
|
+
paint (LsmSvgView *view, LsmSvgViewPathInfos *path_infos)
|
706
|
+
{
|
707
|
+
const LsmSvgStyle *style;
|
708
|
+
LsmSvgElement *element;
|
709
|
+
cairo_t *cairo;
|
710
|
+
gboolean use_group;
|
711
|
+
double group_opacity;
|
712
|
+
|
713
|
+
element = view->element_stack->data;
|
714
|
+
|
715
|
+
g_return_if_fail (LSM_IS_SVG_ELEMENT (element));
|
716
|
+
|
717
|
+
cairo = view->dom_view.cairo;
|
718
|
+
style = view->style;
|
719
|
+
|
720
|
+
if ((style->opacity != NULL ||
|
721
|
+
style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER ) &&
|
722
|
+
style->ignore_group_opacity &&
|
723
|
+
g_strcmp0 (style->filter->value, "none") == 0) {
|
724
|
+
group_opacity = style->opacity->value;
|
725
|
+
|
726
|
+
use_group = ((style->fill->paint.type != LSM_SVG_PAINT_TYPE_NONE &&
|
727
|
+
style->stroke->paint.type != LSM_SVG_PAINT_TYPE_NONE) ||
|
728
|
+
style->stroke->paint.type == LSM_SVG_PAINT_TYPE_URI ||
|
729
|
+
style->stroke->paint.type == LSM_SVG_PAINT_TYPE_URI_RGB_COLOR ||
|
730
|
+
style->fill->paint.type == LSM_SVG_PAINT_TYPE_URI_RGB_COLOR ||
|
731
|
+
style->fill->paint.type == LSM_SVG_PAINT_TYPE_URI) &&
|
732
|
+
(group_opacity < 1.0 || style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER );
|
733
|
+
} else {
|
734
|
+
use_group = FALSE;
|
735
|
+
group_opacity = 1.0;
|
736
|
+
}
|
737
|
+
|
738
|
+
/* Instead of push_group, we should restrict to the current path bounding box */
|
739
|
+
if (use_group) {
|
740
|
+
cairo_push_group (cairo);
|
741
|
+
} else if (style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER)
|
742
|
+
lsm_cairo_set_comp_op (cairo, style->comp_op->value);
|
743
|
+
|
744
|
+
if (_set_color (view,
|
745
|
+
path_infos,
|
746
|
+
LSM_SVG_VIEW_PAINT_OPERATION_FILL,
|
747
|
+
&style->fill->paint,
|
748
|
+
style->fill_opacity->value * (use_group ? 1.0 : group_opacity))) {
|
749
|
+
|
750
|
+
if (path_infos->is_text_path) {
|
751
|
+
pango_cairo_show_layout (cairo, path_infos->pango_layout);
|
752
|
+
} else {
|
753
|
+
cairo_set_fill_rule (cairo, style->fill_rule->value == LSM_SVG_FILL_RULE_EVEN_ODD ?
|
754
|
+
CAIRO_FILL_RULE_EVEN_ODD : CAIRO_FILL_RULE_WINDING);
|
755
|
+
cairo_fill_preserve (cairo);
|
756
|
+
}
|
757
|
+
}
|
758
|
+
|
759
|
+
if (_set_color (view,
|
760
|
+
path_infos,
|
761
|
+
LSM_SVG_VIEW_PAINT_OPERATION_STROKE,
|
762
|
+
&style->stroke->paint,
|
763
|
+
style->stroke_opacity->value * (use_group ? 1.0 : group_opacity))) {
|
764
|
+
double line_width;
|
765
|
+
|
766
|
+
if (path_infos->is_text_path) {
|
767
|
+
pango_cairo_layout_path (cairo, path_infos->pango_layout);
|
768
|
+
}
|
769
|
+
|
770
|
+
switch (style->stroke_line_join->value) {
|
771
|
+
case LSM_SVG_LINE_JOIN_MITER:
|
772
|
+
cairo_set_line_join (cairo, CAIRO_LINE_JOIN_MITER);
|
773
|
+
break;
|
774
|
+
case LSM_SVG_LINE_JOIN_ROUND:
|
775
|
+
cairo_set_line_join (cairo,CAIRO_LINE_JOIN_ROUND);
|
776
|
+
break;
|
777
|
+
default:
|
778
|
+
cairo_set_line_join (cairo,CAIRO_LINE_JOIN_BEVEL);
|
779
|
+
}
|
780
|
+
|
781
|
+
switch (style->stroke_line_cap->value) {
|
782
|
+
case LSM_SVG_LINE_CAP_BUTT:
|
783
|
+
cairo_set_line_cap (cairo, CAIRO_LINE_CAP_BUTT);
|
784
|
+
break;
|
785
|
+
case LSM_SVG_LINE_CAP_ROUND:
|
786
|
+
cairo_set_line_cap (cairo, CAIRO_LINE_CAP_ROUND);
|
787
|
+
break;
|
788
|
+
default:
|
789
|
+
cairo_set_line_cap (cairo, CAIRO_LINE_CAP_SQUARE);
|
790
|
+
}
|
791
|
+
|
792
|
+
line_width = lsm_svg_view_normalize_length (view, &style->stroke_width->length,
|
793
|
+
LSM_SVG_LENGTH_DIRECTION_DIAGONAL);
|
794
|
+
|
795
|
+
cairo_set_miter_limit (cairo, style->stroke_miter_limit->value);
|
796
|
+
cairo_set_line_width (cairo, line_width);
|
797
|
+
|
798
|
+
if (style->stroke_dash_array->value.n_dashes > 0) {
|
799
|
+
double dash_offset;
|
800
|
+
double *dashes;
|
801
|
+
unsigned int i;
|
802
|
+
|
803
|
+
dash_offset = lsm_svg_view_normalize_length (view, &style->stroke_dash_offset->length,
|
804
|
+
LSM_SVG_LENGTH_DIRECTION_DIAGONAL);
|
805
|
+
dashes = g_new (double, style->stroke_dash_array->value.n_dashes);
|
806
|
+
for (i = 0; i < style->stroke_dash_array->value.n_dashes; i++)
|
807
|
+
dashes[i] = lsm_svg_view_normalize_length (view,
|
808
|
+
&style->stroke_dash_array->value.dashes[i],
|
809
|
+
LSM_SVG_LENGTH_DIRECTION_DIAGONAL);
|
810
|
+
|
811
|
+
cairo_set_dash (cairo, dashes, style->stroke_dash_array->value.n_dashes, dash_offset);
|
812
|
+
g_free (dashes);
|
813
|
+
} else
|
814
|
+
cairo_set_dash (cairo, NULL, 0, 0.0);
|
815
|
+
|
816
|
+
cairo_stroke_preserve (cairo);
|
817
|
+
}
|
818
|
+
|
819
|
+
paint_markers (view);
|
820
|
+
|
821
|
+
cairo_new_path (cairo);
|
822
|
+
|
823
|
+
if (use_group) {
|
824
|
+
cairo_pop_group_to_source (cairo);
|
825
|
+
if (G_UNLIKELY (style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER))
|
826
|
+
lsm_cairo_set_comp_op (cairo, style->comp_op->value);
|
827
|
+
cairo_paint_with_alpha (cairo, group_opacity);
|
828
|
+
}
|
829
|
+
|
830
|
+
if (view->style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER)
|
831
|
+
lsm_cairo_set_comp_op (cairo, LSM_SVG_COMP_OP_SRC_OVER);
|
832
|
+
}
|
833
|
+
|
834
|
+
static void
|
835
|
+
process_path (LsmSvgView *view, LsmSvgViewPathInfos *path_infos)
|
836
|
+
{
|
837
|
+
g_return_if_fail (view->style != NULL);
|
838
|
+
|
839
|
+
if (view->is_clipping) {
|
840
|
+
if (path_infos->is_text_path)
|
841
|
+
pango_cairo_layout_path (view->dom_view.cairo, path_infos->pango_layout);
|
842
|
+
cairo_set_fill_rule (view->dom_view.cairo, view->style->clip_rule->value);
|
843
|
+
} else
|
844
|
+
paint (view, path_infos);
|
845
|
+
}
|
846
|
+
|
847
|
+
void
|
848
|
+
lsm_svg_view_show_viewport (LsmSvgView*view, const LsmBox *viewport)
|
849
|
+
{
|
850
|
+
LsmSvgPaint *paint;
|
851
|
+
|
852
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
853
|
+
g_return_if_fail (viewport != NULL);
|
854
|
+
|
855
|
+
paint = &view->style->viewport_fill->paint;
|
856
|
+
|
857
|
+
switch (paint->type) {
|
858
|
+
case LSM_SVG_PAINT_TYPE_RGB_COLOR:
|
859
|
+
cairo_set_source_rgba (view->dom_view.cairo,
|
860
|
+
paint->color.red,
|
861
|
+
paint->color.green,
|
862
|
+
paint->color.blue,
|
863
|
+
view->style->viewport_fill_opacity->value);
|
864
|
+
break;
|
865
|
+
case LSM_SVG_PAINT_TYPE_CURRENT_COLOR:
|
866
|
+
cairo_set_source_rgba (view->dom_view.cairo,
|
867
|
+
view->style->color->value.red,
|
868
|
+
view->style->color->value.green,
|
869
|
+
view->style->color->value.blue,
|
870
|
+
view->style->viewport_fill_opacity->value);
|
871
|
+
default:
|
872
|
+
return;
|
873
|
+
}
|
874
|
+
|
875
|
+
cairo_rectangle (view->dom_view.cairo, viewport->x, viewport->y, 0, 0);
|
876
|
+
cairo_paint (view->dom_view.cairo);
|
877
|
+
}
|
878
|
+
|
879
|
+
/*
|
880
|
+
* Code for show_rectangle and show ellipse is inspired from
|
881
|
+
* the librsvg library (rsvg-shapes.c)
|
882
|
+
*
|
883
|
+
* Copyright (C) 2000 Eazel, Inc.
|
884
|
+
* Copyright (C) 2002 Dom Lachowicz <cinamod@hotmail.com>
|
885
|
+
*
|
886
|
+
* Authors: Raph Levien <raph@artofcode.com>,
|
887
|
+
* Dom Lachowicz <cinamod@hotmail.com>,
|
888
|
+
* Caleb Moore <c.moore@student.unsw.edu.au>
|
889
|
+
*/
|
890
|
+
|
891
|
+
void
|
892
|
+
lsm_svg_view_show_rectangle (LsmSvgView *view,
|
893
|
+
double x, double y,
|
894
|
+
double w, double h,
|
895
|
+
double rx, double ry)
|
896
|
+
{
|
897
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
898
|
+
cairo_t *cairo;
|
899
|
+
|
900
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
901
|
+
|
902
|
+
cairo = view->dom_view.cairo;
|
903
|
+
|
904
|
+
if (rx <= 0.0 || ry <= 0.0)
|
905
|
+
cairo_rectangle (cairo, x, y, w, h);
|
906
|
+
else {
|
907
|
+
|
908
|
+
if (rx > fabs (w / 2.))
|
909
|
+
rx = fabs (w / 2.);
|
910
|
+
if (ry > fabs (h / 2.))
|
911
|
+
ry = fabs (h / 2.);
|
912
|
+
|
913
|
+
cairo_move_to (cairo, x + rx, y);
|
914
|
+
cairo_line_to (cairo, x + w - rx, y);
|
915
|
+
lsm_cairo_elliptical_arc (cairo, rx, ry, 0.0, FALSE, TRUE, x + w, y + ry);
|
916
|
+
cairo_line_to (cairo, x + w, y + h -ry);
|
917
|
+
lsm_cairo_elliptical_arc (cairo, rx, ry, 0.0, FALSE, TRUE, x + w - rx, y + h);
|
918
|
+
cairo_line_to (cairo, x + rx, y + h);
|
919
|
+
lsm_cairo_elliptical_arc (cairo, rx, ry, 0.0, FALSE, TRUE, x, y + h - ry);
|
920
|
+
cairo_line_to (cairo, x, y + ry);
|
921
|
+
lsm_cairo_elliptical_arc (cairo, rx, ry, 0.0, FALSE, TRUE, x + rx, y);
|
922
|
+
cairo_close_path (cairo);
|
923
|
+
}
|
924
|
+
|
925
|
+
process_path (view, &path_infos);
|
926
|
+
}
|
927
|
+
|
928
|
+
void
|
929
|
+
lsm_svg_view_show_circle (LsmSvgView *view, double cx, double cy, double r)
|
930
|
+
{
|
931
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
932
|
+
|
933
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
934
|
+
|
935
|
+
cairo_arc (view->dom_view.cairo, cx, cy, r, 0, 2 * M_PI);
|
936
|
+
|
937
|
+
process_path (view, &path_infos);
|
938
|
+
}
|
939
|
+
|
940
|
+
#define LSM_SVG_ARC_MAGIC ((double) 0.5522847498) /* 4/3 * (1-cos 45)/sin 45 = 4/3 * sqrt(2) - 1 */
|
941
|
+
|
942
|
+
void
|
943
|
+
lsm_svg_view_show_ellipse (LsmSvgView *view, double cx, double cy, double rx, double ry)
|
944
|
+
{
|
945
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
946
|
+
cairo_t *cairo;
|
947
|
+
|
948
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
949
|
+
|
950
|
+
if (rx <= 0 || ry <= 0)
|
951
|
+
return;
|
952
|
+
|
953
|
+
cairo = view->dom_view.cairo;
|
954
|
+
|
955
|
+
cairo_move_to (cairo, cx + rx, cy);
|
956
|
+
cairo_curve_to (cairo, cx + rx, cy + LSM_SVG_ARC_MAGIC * ry, cx + LSM_SVG_ARC_MAGIC * rx, cy + ry, cx, cy + ry);
|
957
|
+
cairo_curve_to (cairo, cx - LSM_SVG_ARC_MAGIC * rx, cy + ry, cx - rx, cy + LSM_SVG_ARC_MAGIC * ry, cx - rx, cy);
|
958
|
+
cairo_curve_to (cairo, cx - rx, cy - LSM_SVG_ARC_MAGIC * ry, cx - LSM_SVG_ARC_MAGIC * rx, cy - ry, cx, cy - ry);
|
959
|
+
cairo_curve_to (cairo, cx + LSM_SVG_ARC_MAGIC * rx, cy - ry, cx + rx, cy - LSM_SVG_ARC_MAGIC * ry, cx + rx, cy);
|
960
|
+
cairo_close_path (cairo);
|
961
|
+
|
962
|
+
process_path (view, &path_infos);
|
963
|
+
}
|
964
|
+
|
965
|
+
void
|
966
|
+
lsm_svg_view_show_path (LsmSvgView *view,
|
967
|
+
const char *d)
|
968
|
+
{
|
969
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
970
|
+
|
971
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
972
|
+
|
973
|
+
lsm_cairo_emit_svg_path (view->dom_view.cairo, d);
|
974
|
+
|
975
|
+
process_path (view, &path_infos);
|
976
|
+
}
|
977
|
+
|
978
|
+
void
|
979
|
+
lsm_svg_view_path_extents (LsmSvgView *view,
|
980
|
+
const char *path,
|
981
|
+
LsmExtents *extents)
|
982
|
+
{
|
983
|
+
double x1, y1, x2, y2;
|
984
|
+
|
985
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
986
|
+
g_return_if_fail (extents != NULL);
|
987
|
+
|
988
|
+
cairo_new_path (view->dom_view.cairo);
|
989
|
+
lsm_cairo_emit_svg_path (view->dom_view.cairo, path);
|
990
|
+
cairo_path_extents (view->dom_view.cairo, &x1, &y1, &x2, &y2);
|
991
|
+
cairo_new_path (view->dom_view.cairo);
|
992
|
+
|
993
|
+
extents->x1 = x1;
|
994
|
+
extents->x2 = x2;
|
995
|
+
extents->y1 = y1;
|
996
|
+
extents->y2 = y2;
|
997
|
+
}
|
998
|
+
|
999
|
+
void
|
1000
|
+
lsm_svg_view_show_line (LsmSvgView *view, double x1, double y1, double x2, double y2)
|
1001
|
+
{
|
1002
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
1003
|
+
|
1004
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1005
|
+
|
1006
|
+
cairo_move_to (view->dom_view.cairo, x1, y1);
|
1007
|
+
cairo_line_to (view->dom_view.cairo, x2, y2);
|
1008
|
+
|
1009
|
+
process_path (view, &path_infos);
|
1010
|
+
}
|
1011
|
+
|
1012
|
+
static void
|
1013
|
+
_show_points (LsmSvgView *view, const char *points, gboolean close_path)
|
1014
|
+
{
|
1015
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
1016
|
+
char *str;
|
1017
|
+
double values[2];
|
1018
|
+
unsigned int n_values;
|
1019
|
+
unsigned int count = 0;
|
1020
|
+
|
1021
|
+
if (points == NULL)
|
1022
|
+
return;
|
1023
|
+
|
1024
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1025
|
+
|
1026
|
+
str = (char *) points;
|
1027
|
+
|
1028
|
+
do {
|
1029
|
+
n_values = lsm_str_parse_double_list (&str, 2, values);
|
1030
|
+
if (n_values == 2) {
|
1031
|
+
if (count == 0)
|
1032
|
+
cairo_move_to (view->dom_view.cairo, values[0], values[1]);
|
1033
|
+
else
|
1034
|
+
cairo_line_to (view->dom_view.cairo, values[0], values[1]);
|
1035
|
+
} else if (n_values != 0) {
|
1036
|
+
cairo_new_path (view->dom_view.cairo);
|
1037
|
+
return;
|
1038
|
+
}
|
1039
|
+
count++;
|
1040
|
+
} while (n_values == 2);
|
1041
|
+
|
1042
|
+
if (close_path)
|
1043
|
+
cairo_close_path (view->dom_view.cairo);
|
1044
|
+
|
1045
|
+
process_path (view, &path_infos);
|
1046
|
+
}
|
1047
|
+
|
1048
|
+
void
|
1049
|
+
lsm_svg_view_show_polyline (LsmSvgView *view, const char *points)
|
1050
|
+
{
|
1051
|
+
_show_points (view, points, FALSE);
|
1052
|
+
}
|
1053
|
+
|
1054
|
+
void
|
1055
|
+
lsm_svg_view_show_polygon (LsmSvgView *view, const char *points)
|
1056
|
+
{
|
1057
|
+
_show_points (view, points, TRUE);
|
1058
|
+
}
|
1059
|
+
|
1060
|
+
static void
|
1061
|
+
_update_pango_layout (LsmSvgView *view, unsigned int n, char const *string, double x, double y,
|
1062
|
+
unsigned int n_dx, const double *dx,
|
1063
|
+
unsigned int n_dy, const double *dy,
|
1064
|
+
LsmSvgViewPathInfos *path_infos)
|
1065
|
+
{
|
1066
|
+
const LsmSvgStyle *style;
|
1067
|
+
PangoLayout *pango_layout;
|
1068
|
+
PangoFontDescription *font_description;
|
1069
|
+
PangoStretch font_stretch;
|
1070
|
+
PangoStyle font_style;
|
1071
|
+
PangoLayoutIter *iter;
|
1072
|
+
PangoRectangle rectangle;
|
1073
|
+
PangoAttrList *attrs;
|
1074
|
+
PangoAttribute *attr;
|
1075
|
+
int baseline;
|
1076
|
+
int i;
|
1077
|
+
double x1, y1;
|
1078
|
+
|
1079
|
+
style = view->style;
|
1080
|
+
|
1081
|
+
pango_layout = view->pango_layout;
|
1082
|
+
font_description = view->dom_view.font_description;
|
1083
|
+
|
1084
|
+
pango_font_description_set_family (font_description, style->font_family->value);
|
1085
|
+
pango_font_description_set_size (font_description, PANGO_SCALE * style->font_size_px);
|
1086
|
+
pango_font_description_set_weight (font_description, style->font_weight->value);
|
1087
|
+
|
1088
|
+
switch (style->font_stretch->value) {
|
1089
|
+
case LSM_SVG_FONT_STRETCH_ULTRA_CONDENSED:
|
1090
|
+
font_stretch = PANGO_STRETCH_ULTRA_CONDENSED;
|
1091
|
+
break;
|
1092
|
+
case LSM_SVG_FONT_STRETCH_EXTRA_CONDENSED:
|
1093
|
+
font_stretch = PANGO_STRETCH_EXTRA_CONDENSED;
|
1094
|
+
break;
|
1095
|
+
case LSM_SVG_FONT_STRETCH_CONDENSED:
|
1096
|
+
font_stretch = PANGO_STRETCH_CONDENSED;
|
1097
|
+
break;
|
1098
|
+
case LSM_SVG_FONT_STRETCH_SEMI_CONDENSED:
|
1099
|
+
font_stretch = PANGO_STRETCH_SEMI_CONDENSED;
|
1100
|
+
break;
|
1101
|
+
case LSM_SVG_FONT_STRETCH_SEMI_EXPANDED:
|
1102
|
+
font_stretch = PANGO_STRETCH_SEMI_EXPANDED;
|
1103
|
+
break;
|
1104
|
+
case LSM_SVG_FONT_STRETCH_EXPANDED:
|
1105
|
+
font_stretch = PANGO_STRETCH_EXPANDED;
|
1106
|
+
break;
|
1107
|
+
case LSM_SVG_FONT_STRETCH_EXTRA_EXPANDED:
|
1108
|
+
font_stretch = PANGO_STRETCH_EXTRA_EXPANDED;
|
1109
|
+
break;
|
1110
|
+
case LSM_SVG_FONT_STRETCH_ULTRA_EXPANDED:
|
1111
|
+
font_stretch = PANGO_STRETCH_ULTRA_EXPANDED;
|
1112
|
+
break;
|
1113
|
+
case LSM_SVG_FONT_STRETCH_NORMAL:
|
1114
|
+
default:
|
1115
|
+
font_stretch = PANGO_STRETCH_NORMAL;
|
1116
|
+
break;
|
1117
|
+
}
|
1118
|
+
pango_font_description_set_stretch (font_description, font_stretch);
|
1119
|
+
|
1120
|
+
switch (style->font_style->value) {
|
1121
|
+
case LSM_SVG_FONT_STYLE_OBLIQUE:
|
1122
|
+
font_style = PANGO_STYLE_OBLIQUE;
|
1123
|
+
break;
|
1124
|
+
case LSM_SVG_FONT_STYLE_ITALIC:
|
1125
|
+
font_style = PANGO_STYLE_ITALIC;
|
1126
|
+
break;
|
1127
|
+
case LSM_SVG_FONT_STYLE_NORMAL:
|
1128
|
+
default:
|
1129
|
+
font_style = PANGO_STYLE_NORMAL;
|
1130
|
+
break;
|
1131
|
+
}
|
1132
|
+
pango_font_description_set_style (font_description, font_style);
|
1133
|
+
|
1134
|
+
pango_layout_set_text (pango_layout, string, n);
|
1135
|
+
|
1136
|
+
attrs = pango_attr_list_new ();
|
1137
|
+
#if 0
|
1138
|
+
for (i = 0; i < n_dx; i++) {
|
1139
|
+
|
1140
|
+
attr = pango_attr_letter_spacing_new (pango_units_from_double (dx[i]));
|
1141
|
+
attr->start_index = i;
|
1142
|
+
if (i < n_dx - 1)
|
1143
|
+
attr->end_index = i + 1;
|
1144
|
+
else
|
1145
|
+
attr->end_index = PANGO_ATTR_INDEX_TO_TEXT_END;
|
1146
|
+
|
1147
|
+
pango_attr_list_insert (attrs, attr);
|
1148
|
+
}
|
1149
|
+
#endif
|
1150
|
+
for (i = 0; i < n_dy; i++) {
|
1151
|
+
|
1152
|
+
attr = pango_attr_rise_new (-pango_units_from_double (dy[i]));
|
1153
|
+
attr->start_index = i;
|
1154
|
+
if (i < n_dy - 1)
|
1155
|
+
attr->end_index = i + 1;
|
1156
|
+
else
|
1157
|
+
attr->end_index = PANGO_ATTR_INDEX_TO_TEXT_END;
|
1158
|
+
|
1159
|
+
pango_attr_list_insert (attrs, attr);
|
1160
|
+
}
|
1161
|
+
pango_layout_set_attributes (pango_layout, attrs);
|
1162
|
+
pango_attr_list_unref (attrs);
|
1163
|
+
|
1164
|
+
pango_layout_set_font_description (pango_layout, font_description);
|
1165
|
+
pango_layout_get_extents (pango_layout, &rectangle, NULL);
|
1166
|
+
|
1167
|
+
iter = pango_layout_get_iter (pango_layout);
|
1168
|
+
baseline = pango_layout_iter_get_baseline (iter);
|
1169
|
+
pango_layout_iter_free (iter);
|
1170
|
+
|
1171
|
+
x1 = x - pango_units_to_double (rectangle.x);
|
1172
|
+
y1 = y - pango_units_to_double (baseline);
|
1173
|
+
|
1174
|
+
switch (style->text_anchor->value) {
|
1175
|
+
case LSM_SVG_TEXT_ANCHOR_END:
|
1176
|
+
x1 -= pango_units_to_double (rectangle.width);
|
1177
|
+
break;
|
1178
|
+
case LSM_SVG_TEXT_ANCHOR_MIDDLE:
|
1179
|
+
x1 -= pango_units_to_double (rectangle.width) / 2.0;
|
1180
|
+
break;
|
1181
|
+
case LSM_SVG_TEXT_ANCHOR_START:
|
1182
|
+
default:
|
1183
|
+
break;
|
1184
|
+
}
|
1185
|
+
|
1186
|
+
path_infos->is_text_path = TRUE;
|
1187
|
+
path_infos->is_extents_defined = TRUE;
|
1188
|
+
path_infos->extents.x1 = x1;
|
1189
|
+
path_infos->extents.y1 = y1;
|
1190
|
+
path_infos->extents.x2 = x1 + pango_units_to_double (rectangle.width);
|
1191
|
+
path_infos->extents.y2 = y1 + pango_units_to_double (rectangle.height);
|
1192
|
+
path_infos->pango_layout = pango_layout;
|
1193
|
+
}
|
1194
|
+
|
1195
|
+
static gboolean
|
1196
|
+
_lock_pango_layout (LsmSvgView *view)
|
1197
|
+
{
|
1198
|
+
/* A text may be painted with a text based pattern. In this case,
|
1199
|
+
* we take care to create a new pango layout if the current one is in use. */
|
1200
|
+
if (view->is_pango_layout_in_use) {
|
1201
|
+
PangoContext *pango_context;
|
1202
|
+
|
1203
|
+
pango_context = pango_layout_get_context (view->pango_layout);
|
1204
|
+
view->pango_layout_stack = g_slist_prepend (view->pango_layout_stack, view->pango_layout);
|
1205
|
+
view->pango_layout = pango_layout_new (pango_context);
|
1206
|
+
|
1207
|
+
lsm_debug_render ("[LsmSvgView::_lock_pango_layout] Create a new pango layout");
|
1208
|
+
|
1209
|
+
return TRUE;
|
1210
|
+
}
|
1211
|
+
|
1212
|
+
view->is_pango_layout_in_use = TRUE;
|
1213
|
+
|
1214
|
+
return FALSE;
|
1215
|
+
}
|
1216
|
+
|
1217
|
+
static void
|
1218
|
+
_unlock_pango_layout (LsmSvgView *view, gboolean need_pop)
|
1219
|
+
{
|
1220
|
+
|
1221
|
+
if (need_pop) {
|
1222
|
+
lsm_debug_render ("[LsmSvgView::_unlock_pango_layout] Free the child pango layout");
|
1223
|
+
|
1224
|
+
if (view->pango_layout != NULL) {
|
1225
|
+
g_object_unref (view->pango_layout);
|
1226
|
+
|
1227
|
+
view->pango_layout = view->pango_layout_stack->data;
|
1228
|
+
view->pango_layout_stack = g_slist_delete_link (view->pango_layout_stack,
|
1229
|
+
view->pango_layout_stack);
|
1230
|
+
} else
|
1231
|
+
g_warning ("[LsmSvgView::_unlock_pango_layout] Pango layout stack empty");
|
1232
|
+
}
|
1233
|
+
|
1234
|
+
view->is_pango_layout_in_use = FALSE;
|
1235
|
+
}
|
1236
|
+
|
1237
|
+
static void
|
1238
|
+
_show_text (LsmSvgView *view,
|
1239
|
+
unsigned int n, char const *string,
|
1240
|
+
unsigned int n_x, double *x, unsigned int n_y, double *y,
|
1241
|
+
unsigned int n_dx, double *dx, unsigned int n_dy, double *dy)
|
1242
|
+
{
|
1243
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
1244
|
+
PangoRectangle extents;
|
1245
|
+
PangoLayoutIter *layout_iter;
|
1246
|
+
const LsmSvgStyle *style;
|
1247
|
+
gboolean need_pop;
|
1248
|
+
double x_text;
|
1249
|
+
double y_text;
|
1250
|
+
double x_end, y_end;
|
1251
|
+
double baseline;
|
1252
|
+
cairo_t *cairo;
|
1253
|
+
|
1254
|
+
|
1255
|
+
cairo = view->dom_view.cairo;
|
1256
|
+
|
1257
|
+
style = view->style;
|
1258
|
+
|
1259
|
+
lsm_debug_render ("[LsmSvgView::show_text] Show '%s' at %g,%g (%g px)", string, x, y, style->font_size_px);
|
1260
|
+
|
1261
|
+
need_pop = _lock_pango_layout (view);
|
1262
|
+
|
1263
|
+
cairo_get_current_point (cairo, &x_text, &y_text);
|
1264
|
+
if (x != NULL)
|
1265
|
+
x_text = x[0];
|
1266
|
+
if (y != NULL)
|
1267
|
+
y_text = y[0];
|
1268
|
+
|
1269
|
+
|
1270
|
+
_update_pango_layout (view, n, string, x_text, y_text, n_dx, dx, n_dy, dy, &path_infos);
|
1271
|
+
|
1272
|
+
if (style->writing_mode->value == LSM_SVG_WRITING_MODE_TB ||
|
1273
|
+
style->writing_mode->value == LSM_SVG_WRITING_MODE_TB_RL) {
|
1274
|
+
|
1275
|
+
cairo_save (cairo);
|
1276
|
+
cairo_rotate (cairo, M_PI / 2.0);
|
1277
|
+
cairo_move_to (cairo, path_infos.extents.x1, path_infos.extents.y1);
|
1278
|
+
|
1279
|
+
process_path (view, &path_infos);
|
1280
|
+
|
1281
|
+
cairo_restore (cairo);
|
1282
|
+
} else {
|
1283
|
+
cairo_move_to (cairo, path_infos.extents.x1, path_infos.extents.y1);
|
1284
|
+
process_path (view, &path_infos);
|
1285
|
+
}
|
1286
|
+
|
1287
|
+
layout_iter = pango_layout_get_iter (view->pango_layout);
|
1288
|
+
pango_layout_iter_get_line_extents (layout_iter, NULL, &extents);
|
1289
|
+
baseline = pango_units_to_double (pango_layout_iter_get_baseline (layout_iter));
|
1290
|
+
pango_layout_iter_free (layout_iter);
|
1291
|
+
|
1292
|
+
x_end = pango_units_to_double (extents.x + extents.width) + path_infos.extents.x1;
|
1293
|
+
y_end = pango_units_to_double (extents.y) + baseline + path_infos.extents.y1;
|
1294
|
+
|
1295
|
+
if (view->debug_text) {
|
1296
|
+
double x1, y1;
|
1297
|
+
|
1298
|
+
lsm_debug_render ("[LsmSvgView::_show_text] Logical extents %gx%g at %g,%g\n",
|
1299
|
+
pango_units_to_double (extents.width),
|
1300
|
+
pango_units_to_double (extents.height),
|
1301
|
+
pango_units_to_double (extents.x),
|
1302
|
+
pango_units_to_double (extents.y));
|
1303
|
+
|
1304
|
+
lsm_debug_render ("[LsmSvgView::_show_text] End point is %g, %g\n", x_end, y_end);
|
1305
|
+
lsm_debug_render ("[LsmSvgView::_show_text] Baseline = %g\n", baseline);
|
1306
|
+
|
1307
|
+
x1 = 1;
|
1308
|
+
y1 = 1;
|
1309
|
+
cairo_device_to_user_distance (cairo, &x1, &y1);
|
1310
|
+
|
1311
|
+
cairo_set_source_rgba (cairo, 1.0, 0.0, 0.0, 0.2);
|
1312
|
+
cairo_set_line_width (cairo, x1);
|
1313
|
+
cairo_rectangle (cairo, path_infos.extents.x1, path_infos.extents.y1,
|
1314
|
+
pango_units_to_double (extents.x + extents.width),
|
1315
|
+
pango_units_to_double (extents.y + extents.height));
|
1316
|
+
cairo_stroke (cairo);
|
1317
|
+
cairo_set_source_rgba (cairo, 1.0, 0.0, 0.0, 0.2);
|
1318
|
+
cairo_arc (cairo, x_end, y_end, 4.0 * x1, 0.0, 2.0 * M_PI);
|
1319
|
+
cairo_arc (cairo, path_infos.extents.x1, path_infos.extents.y1, 4.0 * x1, 0.0, 2.0 * M_PI);
|
1320
|
+
cairo_fill (cairo);
|
1321
|
+
}
|
1322
|
+
|
1323
|
+
cairo_move_to (cairo, x_end, y_end);
|
1324
|
+
|
1325
|
+
_unlock_pango_layout (view, need_pop);
|
1326
|
+
}
|
1327
|
+
|
1328
|
+
void
|
1329
|
+
lsm_svg_view_start_text (LsmSvgView *view)
|
1330
|
+
{
|
1331
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1332
|
+
|
1333
|
+
cairo_move_to (view->dom_view.cairo, 0, 0);
|
1334
|
+
}
|
1335
|
+
|
1336
|
+
void
|
1337
|
+
lsm_svg_view_end_text (LsmSvgView *view)
|
1338
|
+
{
|
1339
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1340
|
+
|
1341
|
+
cairo_new_path (view->dom_view.cairo);
|
1342
|
+
}
|
1343
|
+
|
1344
|
+
void
|
1345
|
+
lsm_svg_view_show_text (LsmSvgView *view, char const *string,
|
1346
|
+
unsigned int n_x, double *x, unsigned int n_y, double *y,
|
1347
|
+
unsigned int n_dx, double *dx, unsigned int n_dy, double *dy)
|
1348
|
+
{
|
1349
|
+
unsigned int n, i;
|
1350
|
+
char *iter = (char *) string;
|
1351
|
+
|
1352
|
+
if (string == NULL || string[0] == '\0')
|
1353
|
+
return;
|
1354
|
+
|
1355
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1356
|
+
g_return_if_fail (n_x > 0 || x == NULL);
|
1357
|
+
g_return_if_fail (n_y > 0 || y == NULL);
|
1358
|
+
g_return_if_fail (n_dx > 0 || dx == NULL);
|
1359
|
+
g_return_if_fail (n_dy > 0 || dy == NULL);
|
1360
|
+
|
1361
|
+
n = MAX (n_x, n_y);
|
1362
|
+
if (n <= 1) {
|
1363
|
+
_show_text (view, strlen (string), string, n_x, x, n_y, y, n_dx, dx, n_dy, dy);
|
1364
|
+
return;
|
1365
|
+
}
|
1366
|
+
|
1367
|
+
for (i = 0; (i < n - 1) && iter[0] != '\0'; i++) {
|
1368
|
+
char *next_char;
|
1369
|
+
|
1370
|
+
next_char = g_utf8_next_char (iter);
|
1371
|
+
_show_text (view, next_char - iter, iter,
|
1372
|
+
i < n_x ? n_x - i : 0,
|
1373
|
+
i < n_x ? &x[i] : NULL,
|
1374
|
+
i < n_y ? n_y - i : 0,
|
1375
|
+
i < n_y ? &y[i] : NULL,
|
1376
|
+
i < n_dx ? n_dx - i : 0,
|
1377
|
+
i < n_dx ? &dx[i] : NULL,
|
1378
|
+
i < n_dy ? n_dy - i : 0,
|
1379
|
+
i < n_dy ? &dy[i] : NULL);
|
1380
|
+
iter = next_char;
|
1381
|
+
}
|
1382
|
+
|
1383
|
+
if (iter[0] != '\0')
|
1384
|
+
_show_text (view, strlen (iter), iter,
|
1385
|
+
i < n_x ? n_x - i : 0,
|
1386
|
+
i < n_x ? &x[i] : NULL,
|
1387
|
+
i < n_y ? n_y - i : 0,
|
1388
|
+
i < n_y ? &y[i] : NULL,
|
1389
|
+
i < n_dx ? n_dx - i : 0,
|
1390
|
+
i < n_dx ? &dx[i] : NULL,
|
1391
|
+
i < n_dy ? n_dy - i : 0,
|
1392
|
+
i < n_dy ? &dy[i] : NULL);
|
1393
|
+
}
|
1394
|
+
|
1395
|
+
void
|
1396
|
+
lsm_svg_view_text_extents (LsmSvgView *view, char const *string,
|
1397
|
+
double x, double y,
|
1398
|
+
unsigned int n_dx, double *dx, unsigned int n_dy, double *dy,
|
1399
|
+
LsmExtents *extents)
|
1400
|
+
{
|
1401
|
+
LsmSvgViewPathInfos path_infos = default_path_infos;
|
1402
|
+
gboolean need_pop;
|
1403
|
+
|
1404
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1405
|
+
g_return_if_fail (extents != NULL);
|
1406
|
+
|
1407
|
+
if (string == NULL ||
|
1408
|
+
string[0] == '\0') {
|
1409
|
+
extents->x1 = 0;
|
1410
|
+
extents->y1 = 0;
|
1411
|
+
extents->y1 = 0;
|
1412
|
+
extents->y2 = 0;
|
1413
|
+
}
|
1414
|
+
|
1415
|
+
need_pop = _lock_pango_layout (view);
|
1416
|
+
|
1417
|
+
_update_pango_layout (view, strlen (string), string, x, y, n_dx, dx, n_dy, dy, &path_infos);
|
1418
|
+
|
1419
|
+
_unlock_pango_layout (view, need_pop);
|
1420
|
+
|
1421
|
+
*extents = path_infos.extents;
|
1422
|
+
}
|
1423
|
+
|
1424
|
+
void
|
1425
|
+
lsm_svg_view_show_pixbuf (LsmSvgView *view, GdkPixbuf *pixbuf)
|
1426
|
+
{
|
1427
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1428
|
+
g_return_if_fail (GDK_IS_PIXBUF (pixbuf));
|
1429
|
+
|
1430
|
+
lsm_cairo_set_source_pixbuf (view->dom_view.cairo, pixbuf, 0, 0);
|
1431
|
+
cairo_paint (view->dom_view.cairo);
|
1432
|
+
}
|
1433
|
+
|
1434
|
+
void
|
1435
|
+
lsm_svg_view_push_viewbox (LsmSvgView *view, const LsmBox *viewbox)
|
1436
|
+
{
|
1437
|
+
LsmSvgViewbox *svg_viewbox;
|
1438
|
+
|
1439
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1440
|
+
|
1441
|
+
lsm_debug_render ("[LsmSvgView::push_viewbox] viewbox = %g, %g, %g, %g",
|
1442
|
+
viewbox->x, viewbox->y, viewbox->width, viewbox->height);
|
1443
|
+
|
1444
|
+
svg_viewbox = lsm_svg_viewbox_new (view->resolution_ppi, viewbox);
|
1445
|
+
|
1446
|
+
view->viewbox_stack = g_slist_prepend (view->viewbox_stack, svg_viewbox);
|
1447
|
+
}
|
1448
|
+
|
1449
|
+
void
|
1450
|
+
lsm_svg_view_pop_viewbox (LsmSvgView *view)
|
1451
|
+
{
|
1452
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1453
|
+
g_return_if_fail (view->viewbox_stack != NULL);
|
1454
|
+
|
1455
|
+
lsm_debug_render ("[LsmSvgView::pop_viewbox]");
|
1456
|
+
|
1457
|
+
lsm_svg_viewbox_free (view->viewbox_stack->data);
|
1458
|
+
view->viewbox_stack = g_slist_delete_link (view->viewbox_stack, view->viewbox_stack);
|
1459
|
+
}
|
1460
|
+
|
1461
|
+
static const LsmBox *
|
1462
|
+
_compute_viewbox_scale (const LsmBox *viewport, const LsmBox *viewbox,
|
1463
|
+
const LsmSvgPreserveAspectRatio *aspect_ratio,
|
1464
|
+
double *x_offset, double *y_offset,
|
1465
|
+
double *x_scale, double *y_scale)
|
1466
|
+
{
|
1467
|
+
if (viewbox != NULL) {
|
1468
|
+
double x, y;
|
1469
|
+
double x_ratio;
|
1470
|
+
double y_ratio;
|
1471
|
+
|
1472
|
+
x_ratio = viewbox->width > 0.0 ? viewport->width / viewbox->width : 0.0;
|
1473
|
+
y_ratio = viewbox->height > 0.0 ? viewport->height / viewbox->height : 0.0;
|
1474
|
+
|
1475
|
+
if (aspect_ratio != NULL && aspect_ratio->align > LSM_SVG_ALIGN_NONE) {
|
1476
|
+
if (aspect_ratio->meet_or_slice == LSM_SVG_MEET_OR_SLICE_MEET) {
|
1477
|
+
*x_scale = MIN (x_ratio, y_ratio);
|
1478
|
+
*y_scale = *x_scale;
|
1479
|
+
} else {
|
1480
|
+
*x_scale = MAX (x_ratio, y_ratio);
|
1481
|
+
*y_scale = *x_scale;
|
1482
|
+
}
|
1483
|
+
|
1484
|
+
x = -viewbox->x * *x_scale;
|
1485
|
+
y = -viewbox->y * *y_scale;
|
1486
|
+
|
1487
|
+
switch (aspect_ratio->align) {
|
1488
|
+
case LSM_SVG_ALIGN_X_MIN_Y_MIN:
|
1489
|
+
break;
|
1490
|
+
case LSM_SVG_ALIGN_X_MIN_Y_MID:
|
1491
|
+
y += (viewport->height- viewbox->height * *y_scale) * 0.5;
|
1492
|
+
break;
|
1493
|
+
case LSM_SVG_ALIGN_X_MIN_Y_MAX:
|
1494
|
+
y += (viewport->height - viewbox->height * *y_scale);
|
1495
|
+
break;
|
1496
|
+
case LSM_SVG_ALIGN_X_MID_Y_MIN:
|
1497
|
+
x += (viewport->width - viewbox->width * *x_scale) * 0.5;
|
1498
|
+
break;
|
1499
|
+
case LSM_SVG_ALIGN_X_MID_Y_MID:
|
1500
|
+
x += (viewport->width - viewbox->width * *x_scale) * 0.5;
|
1501
|
+
y += (viewport->height- viewbox->height * *y_scale) * 0.5;
|
1502
|
+
break;
|
1503
|
+
case LSM_SVG_ALIGN_X_MID_Y_MAX:
|
1504
|
+
x += (viewport->width - viewbox->width * *x_scale) * 0.5;
|
1505
|
+
y += (viewport->height - viewbox->height * *y_scale);
|
1506
|
+
break;
|
1507
|
+
case LSM_SVG_ALIGN_X_MAX_Y_MIN:
|
1508
|
+
x += (viewport->width - viewbox->width * *x_scale);
|
1509
|
+
break;
|
1510
|
+
case LSM_SVG_ALIGN_X_MAX_Y_MID:
|
1511
|
+
x += (viewport->width - viewbox->width * *x_scale);
|
1512
|
+
y += (viewport->height- viewbox->height * *y_scale) * 0.5;
|
1513
|
+
break;
|
1514
|
+
case LSM_SVG_ALIGN_X_MAX_Y_MAX:
|
1515
|
+
x += (viewport->width - viewbox->width * *x_scale);
|
1516
|
+
y += (viewport->height - viewbox->height * *y_scale);
|
1517
|
+
break;
|
1518
|
+
default:
|
1519
|
+
break;
|
1520
|
+
}
|
1521
|
+
|
1522
|
+
*x_offset = x;
|
1523
|
+
*y_offset = y;
|
1524
|
+
|
1525
|
+
} else {
|
1526
|
+
*x_scale = x_ratio;
|
1527
|
+
*y_scale = y_ratio;
|
1528
|
+
|
1529
|
+
*x_offset = -viewbox->x * *x_scale;
|
1530
|
+
*y_offset = -viewbox->y * *y_scale;
|
1531
|
+
}
|
1532
|
+
|
1533
|
+
lsm_debug_render ("[LsmSvgView::_compute_viewbox_scale] scale = %g, %g", *x_scale, *y_scale);
|
1534
|
+
|
1535
|
+
return viewbox;
|
1536
|
+
}
|
1537
|
+
|
1538
|
+
*x_scale = *y_scale = 1.0;
|
1539
|
+
*x_offset = 0.0;
|
1540
|
+
*y_offset = 0.0;
|
1541
|
+
|
1542
|
+
return viewport;
|
1543
|
+
}
|
1544
|
+
|
1545
|
+
void
|
1546
|
+
lsm_svg_view_push_viewport (LsmSvgView *view, const LsmBox *viewport, const LsmBox *viewbox,
|
1547
|
+
const LsmSvgPreserveAspectRatio *aspect_ratio,
|
1548
|
+
LsmSvgOverflow overflow)
|
1549
|
+
{
|
1550
|
+
cairo_t *cairo;
|
1551
|
+
const LsmBox *actual_viewbox;
|
1552
|
+
double x_offset, y_offset;
|
1553
|
+
double x_scale, y_scale;
|
1554
|
+
|
1555
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1556
|
+
g_return_if_fail (viewport != NULL);
|
1557
|
+
|
1558
|
+
actual_viewbox = _compute_viewbox_scale (viewport, viewbox, aspect_ratio,
|
1559
|
+
&x_offset, &y_offset, &x_scale, &y_scale);
|
1560
|
+
lsm_svg_view_push_viewbox (view, actual_viewbox);
|
1561
|
+
|
1562
|
+
cairo = view->dom_view.cairo;
|
1563
|
+
|
1564
|
+
cairo_save (cairo);
|
1565
|
+
|
1566
|
+
if (lsm_debug_check (&lsm_debug_category_viewport, LSM_DEBUG_LEVEL_LOG)) {
|
1567
|
+
cairo_save (cairo);
|
1568
|
+
cairo_set_line_width (cairo, 1.0);
|
1569
|
+
cairo_set_source_rgb (cairo, 0.0, 0.0, 0.0);
|
1570
|
+
cairo_rectangle (cairo, viewport->x, viewport->y, viewport->width, viewport->height);
|
1571
|
+
cairo_stroke (cairo);
|
1572
|
+
cairo_restore (cairo);
|
1573
|
+
}
|
1574
|
+
|
1575
|
+
if (overflow == LSM_SVG_OVERFLOW_HIDDEN) {
|
1576
|
+
cairo_rectangle (cairo, viewport->x, viewport->y, viewport->width, viewport->height);
|
1577
|
+
cairo_clip (cairo);
|
1578
|
+
}
|
1579
|
+
|
1580
|
+
cairo_translate (cairo, viewport->x + x_offset, viewport->y + y_offset);
|
1581
|
+
cairo_scale (cairo, x_scale, y_scale);
|
1582
|
+
}
|
1583
|
+
|
1584
|
+
void
|
1585
|
+
lsm_svg_view_pop_viewport (LsmSvgView *view)
|
1586
|
+
{
|
1587
|
+
cairo_restore (view->dom_view.cairo);
|
1588
|
+
|
1589
|
+
lsm_svg_view_pop_viewbox (view);
|
1590
|
+
}
|
1591
|
+
|
1592
|
+
void
|
1593
|
+
lsm_svg_view_viewbox_to_viewport (LsmSvgView *view, const LsmBox *viewport, const LsmBox *viewbox,
|
1594
|
+
const LsmSvgPreserveAspectRatio *aspect_ratio,
|
1595
|
+
double *x, double *y)
|
1596
|
+
{
|
1597
|
+
double x_offset, y_offset;
|
1598
|
+
double x_scale, y_scale;
|
1599
|
+
|
1600
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1601
|
+
g_return_if_fail (viewport != NULL);
|
1602
|
+
|
1603
|
+
_compute_viewbox_scale (viewport, viewbox, aspect_ratio,
|
1604
|
+
&x_offset, &y_offset, &x_scale, &y_scale);
|
1605
|
+
|
1606
|
+
if (x != NULL)
|
1607
|
+
*x = *x * x_scale + x_offset;
|
1608
|
+
|
1609
|
+
if (y != NULL)
|
1610
|
+
*y = *y * y_scale + y_offset;
|
1611
|
+
}
|
1612
|
+
|
1613
|
+
gboolean
|
1614
|
+
lsm_svg_view_push_matrix (LsmSvgView *view, const LsmSvgMatrix *matrix)
|
1615
|
+
{
|
1616
|
+
cairo_matrix_t cr_matrix;
|
1617
|
+
cairo_matrix_t cr_inv_matrix;
|
1618
|
+
cairo_matrix_t *ctm;
|
1619
|
+
cairo_status_t status;
|
1620
|
+
|
1621
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), FALSE);
|
1622
|
+
|
1623
|
+
ctm = g_new (cairo_matrix_t, 1);
|
1624
|
+
cairo_get_matrix (view->dom_view.cairo, ctm);
|
1625
|
+
|
1626
|
+
view->matrix_stack = g_slist_prepend (view->matrix_stack, ctm);
|
1627
|
+
|
1628
|
+
lsm_debug_render ("[LsmSvgView::push_matrix] New transform %g, %g, %g, %g, %g, %g",
|
1629
|
+
matrix->a, matrix->b, matrix->c, matrix->d, matrix->e, matrix->f);
|
1630
|
+
|
1631
|
+
cairo_matrix_init (&cr_matrix, matrix->a, matrix->b, matrix->c, matrix->d, matrix->e, matrix->f);
|
1632
|
+
cr_inv_matrix = cr_matrix;
|
1633
|
+
status = cairo_matrix_invert (&cr_inv_matrix) == CAIRO_STATUS_SUCCESS;
|
1634
|
+
|
1635
|
+
if (status == CAIRO_STATUS_SUCCESS) {
|
1636
|
+
lsm_debug_render ("[LsmSvgView::push_matrix] Not invertible matrix");
|
1637
|
+
return FALSE;
|
1638
|
+
}
|
1639
|
+
|
1640
|
+
cairo_transform (view->dom_view.cairo, &cr_matrix);
|
1641
|
+
|
1642
|
+
{
|
1643
|
+
cairo_matrix_t current_ctm;
|
1644
|
+
cairo_get_matrix (view->dom_view.cairo, ¤t_ctm);
|
1645
|
+
|
1646
|
+
lsm_debug_render ("[LsmSvgView::push_matrix] Current ctm %g, %g, %g, %g, %g, %g",
|
1647
|
+
current_ctm.xx, current_ctm.xy, current_ctm.yx, current_ctm.yy,
|
1648
|
+
current_ctm.x0, current_ctm.y0);
|
1649
|
+
}
|
1650
|
+
|
1651
|
+
return TRUE;
|
1652
|
+
}
|
1653
|
+
|
1654
|
+
void
|
1655
|
+
lsm_svg_view_pop_matrix (LsmSvgView *view)
|
1656
|
+
{
|
1657
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1658
|
+
|
1659
|
+
if (view->matrix_stack != NULL) {
|
1660
|
+
cairo_matrix_t *ctm;
|
1661
|
+
|
1662
|
+
ctm = view->matrix_stack->data;
|
1663
|
+
|
1664
|
+
cairo_set_matrix (view->dom_view.cairo, ctm);
|
1665
|
+
|
1666
|
+
lsm_debug_render ("[LsmSvgView::pop_matrix] Restore ctm %g, %g, %g, %g, %g, %g",
|
1667
|
+
ctm->xx, ctm->xy, ctm->yx, ctm->yy,
|
1668
|
+
ctm->x0, ctm->y0);
|
1669
|
+
|
1670
|
+
g_free (ctm);
|
1671
|
+
view->matrix_stack = g_slist_delete_link (view->matrix_stack, view->matrix_stack);
|
1672
|
+
}
|
1673
|
+
}
|
1674
|
+
|
1675
|
+
static void
|
1676
|
+
lsm_svg_view_push_clip (LsmSvgView *view)
|
1677
|
+
{
|
1678
|
+
LsmSvgElement *element;
|
1679
|
+
LsmExtents extents;
|
1680
|
+
char *url;
|
1681
|
+
|
1682
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1683
|
+
g_return_if_fail (!view->is_clipping);
|
1684
|
+
|
1685
|
+
lsm_svg_element_get_extents (view->element_stack->data, view, &extents);
|
1686
|
+
|
1687
|
+
url = view->style->clip_path->value;
|
1688
|
+
|
1689
|
+
lsm_debug_render ("[LsmSvgView::push_clip] Using '%s'", url);
|
1690
|
+
|
1691
|
+
cairo_save (view->dom_view.cairo);
|
1692
|
+
|
1693
|
+
view->clip_extents.x = extents.x1;
|
1694
|
+
view->clip_extents.y = extents.y1;
|
1695
|
+
view->clip_extents.width = extents.x2 - extents.x1;
|
1696
|
+
view->clip_extents.height = extents.y2 - extents.y1;
|
1697
|
+
|
1698
|
+
lsm_debug_render ("[LsmSvgView::push_clip] x=%g y=%g w=%g h=%g",
|
1699
|
+
view->clip_extents.x,
|
1700
|
+
view->clip_extents.y,
|
1701
|
+
view->clip_extents.width,
|
1702
|
+
view->clip_extents.height);
|
1703
|
+
|
1704
|
+
element = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document), url);
|
1705
|
+
if (LSM_IS_SVG_CLIP_PATH_ELEMENT (element) &&
|
1706
|
+
!lsm_svg_view_circular_reference_check (view, element)) {
|
1707
|
+
view->is_clipping = TRUE;
|
1708
|
+
lsm_svg_element_force_render (LSM_SVG_ELEMENT (element), view);
|
1709
|
+
cairo_clip (view->dom_view.cairo);
|
1710
|
+
view->is_clipping = FALSE;
|
1711
|
+
} else
|
1712
|
+
lsm_warning_render ("[LsmSvgView::push_clip] Clip path not found: %s", view->style->clip_path->value);
|
1713
|
+
}
|
1714
|
+
|
1715
|
+
static void
|
1716
|
+
lsm_svg_view_pop_clip (LsmSvgView *view)
|
1717
|
+
{
|
1718
|
+
lsm_debug_render ("[LsmSvgView::pop_clip");
|
1719
|
+
|
1720
|
+
cairo_restore (view->dom_view.cairo);
|
1721
|
+
}
|
1722
|
+
|
1723
|
+
static void
|
1724
|
+
lsm_svg_view_push_mask (LsmSvgView *view)
|
1725
|
+
{
|
1726
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1727
|
+
|
1728
|
+
cairo_push_group (view->dom_view.cairo);
|
1729
|
+
}
|
1730
|
+
|
1731
|
+
static void
|
1732
|
+
lsm_svg_view_pop_mask (LsmSvgView *view)
|
1733
|
+
{
|
1734
|
+
LsmSvgElement *mask_element;
|
1735
|
+
|
1736
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1737
|
+
|
1738
|
+
mask_element = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
1739
|
+
view->style->mask->value);
|
1740
|
+
|
1741
|
+
if (LSM_IS_SVG_MASK_ELEMENT (mask_element) &&
|
1742
|
+
!lsm_svg_view_circular_reference_check (view, mask_element)) {
|
1743
|
+
LsmExtents extents;
|
1744
|
+
LsmBox mask_extents;
|
1745
|
+
cairo_t *cairo;
|
1746
|
+
|
1747
|
+
lsm_svg_element_get_extents (view->element_stack->data, view, &extents);
|
1748
|
+
|
1749
|
+
mask_extents.x = extents.x1;
|
1750
|
+
mask_extents.y = extents.y1;
|
1751
|
+
mask_extents.width = extents.x2 - extents.x1;
|
1752
|
+
mask_extents.height = extents.y2 - extents.y1;
|
1753
|
+
|
1754
|
+
cairo = view->dom_view.cairo;
|
1755
|
+
|
1756
|
+
_start_pattern (view, &mask_extents, &mask_extents, 1.0);
|
1757
|
+
|
1758
|
+
lsm_svg_element_force_render (LSM_SVG_ELEMENT (mask_element), view);
|
1759
|
+
|
1760
|
+
cairo_pop_group_to_source (cairo);
|
1761
|
+
if (view->pattern_data->pattern != NULL) {
|
1762
|
+
cairo_surface_t *surface;
|
1763
|
+
int width, height, row, i, stride;
|
1764
|
+
unsigned char *pixels;
|
1765
|
+
|
1766
|
+
cairo_pattern_get_surface (view->pattern_data->pattern, &surface);
|
1767
|
+
pixels = cairo_image_surface_get_data (surface);
|
1768
|
+
height = cairo_image_surface_get_height (surface);
|
1769
|
+
width = cairo_image_surface_get_width (surface);
|
1770
|
+
stride = cairo_image_surface_get_stride (surface);
|
1771
|
+
|
1772
|
+
for (row = 0; row < height; row++) {
|
1773
|
+
guint8 *row_data = (pixels + (row * stride));
|
1774
|
+
for (i = 0; i < width; i++) {
|
1775
|
+
guint32 *pixel = (guint32 *) row_data + i;
|
1776
|
+
*pixel = ((((*pixel & 0x00ff0000) >> 16) * 13817 +
|
1777
|
+
((*pixel & 0x0000ff00) >> 8) * 46518 +
|
1778
|
+
((*pixel & 0x000000ff)) * 4688) * 0xff
|
1779
|
+
/* * opacity */);
|
1780
|
+
}
|
1781
|
+
}
|
1782
|
+
|
1783
|
+
cairo_pattern_set_extend (view->pattern_data->pattern, CAIRO_EXTEND_NONE);
|
1784
|
+
|
1785
|
+
if (view->debug_mask && view->dom_view.cairo != NULL) {
|
1786
|
+
char *filename;
|
1787
|
+
|
1788
|
+
filename = g_strdup_printf ("mask-%s.png", view->style->mask->value);
|
1789
|
+
cairo_surface_write_to_png (cairo_get_target (view->dom_view.cairo), filename);
|
1790
|
+
g_free (filename);
|
1791
|
+
}
|
1792
|
+
|
1793
|
+
cairo_mask (cairo, view->pattern_data->pattern);
|
1794
|
+
} else {
|
1795
|
+
cairo_paint (cairo);
|
1796
|
+
}
|
1797
|
+
|
1798
|
+
_end_pattern (view);
|
1799
|
+
} else {
|
1800
|
+
lsm_warning_render ("[LsmSvgView::pop_mask] Mask url nout found: %s", view->style->mask->value);
|
1801
|
+
|
1802
|
+
cairo_pop_group_to_source (view->dom_view.cairo);
|
1803
|
+
cairo_paint (view->dom_view.cairo);
|
1804
|
+
}
|
1805
|
+
}
|
1806
|
+
|
1807
|
+
static void
|
1808
|
+
lsm_svg_view_push_filter (LsmSvgView *view)
|
1809
|
+
{
|
1810
|
+
LsmExtents extents;
|
1811
|
+
LsmBox object_extents;
|
1812
|
+
LsmBox effect_viewport;
|
1813
|
+
LsmSvgElement *filter_element;
|
1814
|
+
gboolean success;
|
1815
|
+
|
1816
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1817
|
+
g_return_if_fail (view->element_stack != NULL);
|
1818
|
+
|
1819
|
+
lsm_svg_element_get_extents (view->element_stack->data, view, &extents);
|
1820
|
+
|
1821
|
+
object_extents.x = extents.x1;
|
1822
|
+
object_extents.y = extents.y1;
|
1823
|
+
object_extents.width = extents.x2 - extents.x1;
|
1824
|
+
object_extents.height = extents.y2 - extents.y1;
|
1825
|
+
|
1826
|
+
filter_element = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
1827
|
+
view->style->filter->value);
|
1828
|
+
|
1829
|
+
if (LSM_IS_SVG_FILTER_ELEMENT (filter_element)) {
|
1830
|
+
effect_viewport = lsm_svg_filter_element_get_effect_viewport (LSM_SVG_FILTER_ELEMENT (filter_element),
|
1831
|
+
&object_extents, view);
|
1832
|
+
|
1833
|
+
_start_pattern (view, &effect_viewport, &object_extents,
|
1834
|
+
view->style->opacity != NULL ? view->style->opacity->value : 1.0);
|
1835
|
+
|
1836
|
+
success = lsm_svg_view_create_surface_pattern (view,
|
1837
|
+
&effect_viewport,
|
1838
|
+
NULL,
|
1839
|
+
LSM_SVG_VIEW_SURFACE_TYPE_IMAGE);
|
1840
|
+
} else {
|
1841
|
+
lsm_warning_render ("LsmSvgView::push_filter] Filter not found: %s", view->style->filter->value);
|
1842
|
+
|
1843
|
+
_start_pattern (view, &object_extents, &object_extents, 0.0);
|
1844
|
+
|
1845
|
+
success = lsm_svg_view_create_surface_pattern (view,
|
1846
|
+
&object_extents,
|
1847
|
+
NULL,
|
1848
|
+
LSM_SVG_VIEW_SURFACE_TYPE_IMAGE);
|
1849
|
+
}
|
1850
|
+
|
1851
|
+
if (!success)
|
1852
|
+
lsm_warning_render ("LsmSvgView::push_filter] Failed to create subsurface");
|
1853
|
+
}
|
1854
|
+
|
1855
|
+
static void
|
1856
|
+
lsm_svg_view_pop_filter (LsmSvgView *view)
|
1857
|
+
{
|
1858
|
+
LsmSvgElement *filter_element;
|
1859
|
+
LsmSvgFilterSurface *filter_surface;
|
1860
|
+
cairo_surface_t *surface;
|
1861
|
+
GSList *iter;
|
1862
|
+
|
1863
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
1864
|
+
|
1865
|
+
filter_element = lsm_svg_document_get_element_by_url (LSM_SVG_DOCUMENT (view->dom_view.document),
|
1866
|
+
view->style->filter->value);
|
1867
|
+
|
1868
|
+
if (LSM_IS_SVG_FILTER_ELEMENT (filter_element) &&
|
1869
|
+
view->pattern_data->pattern != NULL) {
|
1870
|
+
cairo_matrix_t matrix;
|
1871
|
+
LsmBox subregion;
|
1872
|
+
|
1873
|
+
view->filter_surfaces = NULL;
|
1874
|
+
|
1875
|
+
cairo_pattern_get_surface (view->pattern_data->pattern, &surface);
|
1876
|
+
|
1877
|
+
subregion.x = 0;
|
1878
|
+
subregion.y = 0;
|
1879
|
+
subregion.width = cairo_image_surface_get_width (surface);
|
1880
|
+
subregion.height = cairo_image_surface_get_height (surface);
|
1881
|
+
|
1882
|
+
filter_surface = lsm_svg_filter_surface_new_with_content ("SourceGraphic", surface, &subregion);
|
1883
|
+
cairo_pattern_get_matrix (view->pattern_data->pattern, &matrix);
|
1884
|
+
|
1885
|
+
view->filter_surfaces = g_slist_prepend (view->filter_surfaces, filter_surface);
|
1886
|
+
|
1887
|
+
lsm_svg_element_force_render (filter_element, view);
|
1888
|
+
|
1889
|
+
if (view->debug_filter) {
|
1890
|
+
GSList *iter;
|
1891
|
+
char *filename;
|
1892
|
+
static int count = 0;
|
1893
|
+
|
1894
|
+
for (iter = view->filter_surfaces; iter != NULL; iter = iter->next) {
|
1895
|
+
LsmSvgFilterSurface *surface = iter->data;
|
1896
|
+
|
1897
|
+
filename = g_strdup_printf ("filter-%04d-%s-%s.png", count++,
|
1898
|
+
view->style->filter->value,
|
1899
|
+
lsm_svg_filter_surface_get_name (surface));
|
1900
|
+
cairo_surface_write_to_png (lsm_svg_filter_surface_get_cairo_surface (surface), filename);
|
1901
|
+
g_free (filename);
|
1902
|
+
}
|
1903
|
+
}
|
1904
|
+
|
1905
|
+
if (view->filter_surfaces->next != NULL) {
|
1906
|
+
cairo_pattern_t *pattern;
|
1907
|
+
cairo_surface_t *surface;
|
1908
|
+
|
1909
|
+
surface = lsm_svg_filter_surface_get_cairo_surface (view->filter_surfaces->data);
|
1910
|
+
pattern = cairo_pattern_create_for_surface (surface);
|
1911
|
+
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_NONE);
|
1912
|
+
cairo_pattern_set_matrix (pattern, &matrix);
|
1913
|
+
cairo_set_source (view->pattern_data->old_cairo, pattern);
|
1914
|
+
cairo_pattern_destroy (pattern);
|
1915
|
+
cairo_paint_with_alpha (view->pattern_data->old_cairo, view->style->opacity->value);
|
1916
|
+
}
|
1917
|
+
|
1918
|
+
for (iter = view->filter_surfaces; iter != NULL; iter = iter->next)
|
1919
|
+
lsm_svg_filter_surface_unref (iter->data);
|
1920
|
+
g_slist_free (view->filter_surfaces);
|
1921
|
+
view->filter_surfaces = NULL;
|
1922
|
+
}
|
1923
|
+
|
1924
|
+
_end_pattern (view);
|
1925
|
+
}
|
1926
|
+
|
1927
|
+
static LsmSvgFilterSurface *
|
1928
|
+
_get_filter_surface (LsmSvgView *view, const char *input)
|
1929
|
+
{
|
1930
|
+
GSList *iter;
|
1931
|
+
LsmSvgFilterSurface *source_surface = NULL;
|
1932
|
+
|
1933
|
+
if (input == NULL)
|
1934
|
+
return view->filter_surfaces->data;
|
1935
|
+
|
1936
|
+
for (iter = view->filter_surfaces; iter != NULL; iter = iter->next) {
|
1937
|
+
LsmSvgFilterSurface *surface = iter->data;
|
1938
|
+
|
1939
|
+
if (g_strcmp0 (input, lsm_svg_filter_surface_get_name (surface)) == 0)
|
1940
|
+
return surface;
|
1941
|
+
|
1942
|
+
source_surface = surface;
|
1943
|
+
}
|
1944
|
+
|
1945
|
+
if (g_strcmp0 (input, "SourceAlpha") == 0 && source_surface != NULL) {
|
1946
|
+
LsmSvgFilterSurface *surface;
|
1947
|
+
|
1948
|
+
surface = lsm_svg_filter_surface_new_similar ("SourceAlpha", source_surface, NULL);
|
1949
|
+
lsm_svg_filter_surface_alpha (source_surface, surface);
|
1950
|
+
view->filter_surfaces = g_slist_prepend (view->filter_surfaces, surface);
|
1951
|
+
|
1952
|
+
return surface;
|
1953
|
+
} else if (g_strcmp0 (input, "BackgroundImage") == 0) {
|
1954
|
+
LsmSvgFilterSurface *surface;
|
1955
|
+
LsmSvgViewBackground *background;
|
1956
|
+
gboolean background_processing = FALSE;
|
1957
|
+
cairo_matrix_t matrix;
|
1958
|
+
cairo_matrix_t pattern_matrix;
|
1959
|
+
cairo_t *cairo;
|
1960
|
+
GList *iter;
|
1961
|
+
|
1962
|
+
for (iter = view->background_stack; iter != NULL; iter = iter->next) {
|
1963
|
+
background = iter->data;
|
1964
|
+
|
1965
|
+
if (background->enable_background) {
|
1966
|
+
background_processing = TRUE;
|
1967
|
+
break;
|
1968
|
+
}
|
1969
|
+
}
|
1970
|
+
|
1971
|
+
if (!background_processing) {
|
1972
|
+
lsm_debug_render ("[LsmSvgView::_get_filter_surface] Background processing not enabled");
|
1973
|
+
return NULL;
|
1974
|
+
}
|
1975
|
+
|
1976
|
+
surface = lsm_svg_filter_surface_new_similar ("BackgroundImage", source_surface, NULL);
|
1977
|
+
view->filter_surfaces = g_slist_prepend (view->filter_surfaces, surface);
|
1978
|
+
|
1979
|
+
cairo_get_matrix (view->pattern_data->old_cairo, &matrix);
|
1980
|
+
cairo_pattern_get_matrix (view->pattern_data->pattern, &pattern_matrix);
|
1981
|
+
|
1982
|
+
cairo_matrix_invert (&matrix);
|
1983
|
+
cairo_matrix_multiply (&matrix, &matrix, &pattern_matrix);
|
1984
|
+
|
1985
|
+
lsm_debug_render ("[LsmSvgView::_get_filter_surface] Background image matrix %g, %g, %g, %g, %g, %g",
|
1986
|
+
matrix.xx, matrix.xy, matrix.yx, matrix.yy,
|
1987
|
+
matrix.x0, matrix.y0);
|
1988
|
+
|
1989
|
+
cairo = cairo_create (lsm_svg_filter_surface_get_cairo_surface (surface));
|
1990
|
+
cairo_set_matrix (cairo, &matrix);
|
1991
|
+
|
1992
|
+
for (; iter != NULL; iter = iter->prev) {
|
1993
|
+
background = iter->data;
|
1994
|
+
|
1995
|
+
cairo_set_source_surface (cairo, background->surface, 0, 0);
|
1996
|
+
cairo_paint_with_alpha (cairo, background->group_opacity);
|
1997
|
+
}
|
1998
|
+
|
1999
|
+
cairo_destroy (cairo);
|
2000
|
+
|
2001
|
+
return surface;
|
2002
|
+
} else if (g_strcmp0 (input, "BackgroundAlpha") == 0) {
|
2003
|
+
LsmSvgFilterSurface *surface;
|
2004
|
+
LsmSvgFilterSurface *background_surface;
|
2005
|
+
|
2006
|
+
if (view->background_stack == NULL)
|
2007
|
+
return NULL;
|
2008
|
+
|
2009
|
+
background_surface = _get_filter_surface (view, "BackgroundImage");
|
2010
|
+
|
2011
|
+
surface = lsm_svg_filter_surface_new_similar ("BackgroundAlpha", background_surface, NULL);
|
2012
|
+
lsm_svg_filter_surface_alpha (background_surface, surface);
|
2013
|
+
view->filter_surfaces = g_slist_prepend (view->filter_surfaces, surface);
|
2014
|
+
|
2015
|
+
return surface;
|
2016
|
+
}
|
2017
|
+
|
2018
|
+
return NULL;
|
2019
|
+
}
|
2020
|
+
|
2021
|
+
static LsmSvgFilterSurface *
|
2022
|
+
_create_filter_surface (LsmSvgView *view, const char *output, LsmSvgFilterSurface *input_surface, const LsmBox *subregion)
|
2023
|
+
{
|
2024
|
+
LsmSvgFilterSurface *surface;
|
2025
|
+
|
2026
|
+
surface = lsm_svg_filter_surface_new_similar (output, input_surface, subregion);
|
2027
|
+
|
2028
|
+
view->filter_surfaces = g_slist_prepend (view->filter_surfaces, surface);
|
2029
|
+
|
2030
|
+
return surface;
|
2031
|
+
}
|
2032
|
+
|
2033
|
+
LsmBox
|
2034
|
+
lsm_svg_view_get_filter_surface_extents (LsmSvgView *view, const char *name)
|
2035
|
+
{
|
2036
|
+
static LsmBox null_extents = {.x = 0.0, .y = 0.0, .width = 0.0, .height = 0.0};
|
2037
|
+
LsmSvgFilterSurface *surface;
|
2038
|
+
LsmBox extents;
|
2039
|
+
|
2040
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), null_extents);
|
2041
|
+
|
2042
|
+
surface = _get_filter_surface (view, name);
|
2043
|
+
if (surface == NULL)
|
2044
|
+
return null_extents;
|
2045
|
+
|
2046
|
+
lsm_cairo_box_device_to_user (view->dom_view.cairo, &extents, lsm_svg_filter_surface_get_subregion (surface));
|
2047
|
+
|
2048
|
+
return extents;
|
2049
|
+
}
|
2050
|
+
|
2051
|
+
void
|
2052
|
+
lsm_svg_view_apply_blend (LsmSvgView *view, const char *input_1, const char*input_2, const char *output,
|
2053
|
+
const LsmBox *subregion, LsmSvgBlendingMode mode)
|
2054
|
+
{
|
2055
|
+
LsmSvgFilterSurface *output_surface;
|
2056
|
+
LsmSvgFilterSurface *input_1_surface;
|
2057
|
+
LsmSvgFilterSurface *input_2_surface;
|
2058
|
+
LsmBox subregion_px;
|
2059
|
+
|
2060
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2061
|
+
|
2062
|
+
input_1_surface = _get_filter_surface (view, input_1);
|
2063
|
+
input_2_surface = _get_filter_surface (view, input_2);
|
2064
|
+
|
2065
|
+
if (input_1_surface == NULL || input_2_surface == NULL) {
|
2066
|
+
lsm_warning_render ("[SvgView::apply_blend] Inputs '%s' or '%s' not found", input_1, input_2);
|
2067
|
+
return;
|
2068
|
+
}
|
2069
|
+
|
2070
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2071
|
+
output_surface = _create_filter_surface (view, output, input_1_surface, &subregion_px);
|
2072
|
+
|
2073
|
+
lsm_log_render ("[SvgView::blend] mode = %s", lsm_svg_blending_mode_to_string (mode));
|
2074
|
+
|
2075
|
+
lsm_svg_filter_surface_blend (input_1_surface, input_2_surface, output_surface, mode);
|
2076
|
+
}
|
2077
|
+
|
2078
|
+
void
|
2079
|
+
lsm_svg_view_apply_flood (LsmSvgView *view, const char *output, const LsmBox *subregion)
|
2080
|
+
{
|
2081
|
+
LsmSvgFilterSurface *output_surface;
|
2082
|
+
LsmSvgFilterSurface *input_surface;
|
2083
|
+
LsmBox subregion_px;
|
2084
|
+
|
2085
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2086
|
+
|
2087
|
+
input_surface = _get_filter_surface (view, NULL);
|
2088
|
+
|
2089
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2090
|
+
output_surface = _create_filter_surface (view, output, input_surface, &subregion_px);
|
2091
|
+
|
2092
|
+
lsm_log_render ("[SvgView::apply_flood] subregion %gx%g px at %g,%g px",
|
2093
|
+
subregion_px.width, subregion_px.height,
|
2094
|
+
subregion_px.x, subregion_px.y);
|
2095
|
+
|
2096
|
+
lsm_svg_filter_surface_flood (output_surface,
|
2097
|
+
view->style->flood_color->value.red,
|
2098
|
+
view->style->flood_color->value.green,
|
2099
|
+
view->style->flood_color->value.blue,
|
2100
|
+
view->style->flood_opacity->value);
|
2101
|
+
}
|
2102
|
+
|
2103
|
+
void
|
2104
|
+
lsm_svg_view_apply_gaussian_blur (LsmSvgView *view, const char *input, const char *output,
|
2105
|
+
const LsmBox *subregion, double std_x, double std_y)
|
2106
|
+
{
|
2107
|
+
LsmSvgFilterSurface *input_surface;
|
2108
|
+
LsmSvgFilterSurface *output_surface;
|
2109
|
+
LsmBox subregion_px;
|
2110
|
+
|
2111
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2112
|
+
|
2113
|
+
input_surface = _get_filter_surface (view, input);
|
2114
|
+
|
2115
|
+
if (input_surface == NULL) {
|
2116
|
+
lsm_debug_render ("[SvgView::apply_gaussian_blur] Input '%s' not found", input);
|
2117
|
+
return;
|
2118
|
+
}
|
2119
|
+
|
2120
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2121
|
+
output_surface = _create_filter_surface (view, output, input_surface, &subregion_px);
|
2122
|
+
|
2123
|
+
lsm_log_render ("[SvgView::apply_gaussian_blur] %s -> %s (%g,%g)",
|
2124
|
+
input != NULL ? input : "previous",
|
2125
|
+
output != NULL ? output : "next",
|
2126
|
+
std_x, std_y);
|
2127
|
+
|
2128
|
+
cairo_user_to_device_distance (view->dom_view.cairo, &std_x, &std_y);
|
2129
|
+
|
2130
|
+
lsm_log_render ("[SvgView::apply_gaussian_blur] %g px,%g px",
|
2131
|
+
std_x, std_y);
|
2132
|
+
|
2133
|
+
lsm_svg_filter_surface_fast_blur (input_surface, output_surface, std_x, std_y);
|
2134
|
+
}
|
2135
|
+
|
2136
|
+
void
|
2137
|
+
lsm_svg_view_apply_offset (LsmSvgView *view, const char *input, const char *output,
|
2138
|
+
const LsmBox *subregion, double dx, double dy)
|
2139
|
+
{
|
2140
|
+
LsmSvgFilterSurface *input_surface;
|
2141
|
+
LsmSvgFilterSurface *output_surface;
|
2142
|
+
LsmBox subregion_px;
|
2143
|
+
|
2144
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2145
|
+
|
2146
|
+
input_surface = _get_filter_surface (view, input);
|
2147
|
+
|
2148
|
+
if (input_surface == NULL) {
|
2149
|
+
lsm_debug_render ("[SvgView::apply_offset] Input '%s' not found", input);
|
2150
|
+
return;
|
2151
|
+
}
|
2152
|
+
|
2153
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2154
|
+
output_surface = _create_filter_surface (view, output, input_surface, &subregion_px);
|
2155
|
+
|
2156
|
+
lsm_log_render ("[SvgView::apply_offset] %s -> %s (dx:%g,dy:%g)", input, output, dx, dy);
|
2157
|
+
|
2158
|
+
cairo_user_to_device_distance (view->dom_view.cairo, &dx, &dy);
|
2159
|
+
|
2160
|
+
lsm_log_render ("[SvgView::apply_offset] %g px,%g px", dx, dy);
|
2161
|
+
|
2162
|
+
lsm_svg_filter_surface_offset (input_surface, output_surface, dx, dy);
|
2163
|
+
}
|
2164
|
+
|
2165
|
+
void
|
2166
|
+
lsm_svg_view_apply_merge (LsmSvgView *view, const char *input, const char *output, const LsmBox *subregion)
|
2167
|
+
{
|
2168
|
+
LsmSvgFilterSurface *input_surface;
|
2169
|
+
LsmSvgFilterSurface *output_surface;
|
2170
|
+
LsmBox subregion_px;
|
2171
|
+
|
2172
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2173
|
+
|
2174
|
+
input_surface = _get_filter_surface (view, input);
|
2175
|
+
|
2176
|
+
if (input_surface == NULL) {
|
2177
|
+
lsm_debug_render ("[SvgView::apply_offset] Input '%s' not found", input);
|
2178
|
+
return;
|
2179
|
+
}
|
2180
|
+
|
2181
|
+
output_surface = _get_filter_surface (view, output);
|
2182
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2183
|
+
if (output_surface == NULL)
|
2184
|
+
output_surface = _create_filter_surface (view, output, input_surface, &subregion_px);
|
2185
|
+
|
2186
|
+
if (output_surface != NULL)
|
2187
|
+
lsm_svg_filter_surface_merge (input_surface, output_surface);
|
2188
|
+
}
|
2189
|
+
|
2190
|
+
void
|
2191
|
+
lsm_svg_view_apply_tile (LsmSvgView *view, const char *input, const char *output, const LsmBox *subregion)
|
2192
|
+
{
|
2193
|
+
LsmSvgFilterSurface *input_surface;
|
2194
|
+
LsmSvgFilterSurface *output_surface;
|
2195
|
+
LsmBox subregion_px;
|
2196
|
+
|
2197
|
+
input_surface = _get_filter_surface (view, input);
|
2198
|
+
|
2199
|
+
if (input_surface == NULL) {
|
2200
|
+
lsm_debug_render ("[SvgView::apply_offset] Input '%s' not found", input);
|
2201
|
+
return;
|
2202
|
+
}
|
2203
|
+
|
2204
|
+
lsm_cairo_box_user_to_device (view->dom_view.cairo, &subregion_px, subregion);
|
2205
|
+
output_surface = _create_filter_surface (view, output, input_surface, &subregion_px);
|
2206
|
+
|
2207
|
+
lsm_svg_filter_surface_tile (input_surface, output_surface);
|
2208
|
+
}
|
2209
|
+
|
2210
|
+
void
|
2211
|
+
lsm_svg_view_push_element (LsmSvgView *view, const LsmSvgElement *element)
|
2212
|
+
{
|
2213
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2214
|
+
g_return_if_fail (LSM_IS_SVG_ELEMENT (element));
|
2215
|
+
|
2216
|
+
view->element_stack = g_slist_prepend (view->element_stack, (void *) element);
|
2217
|
+
}
|
2218
|
+
|
2219
|
+
void
|
2220
|
+
lsm_svg_view_pop_element (LsmSvgView *view)
|
2221
|
+
{
|
2222
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2223
|
+
g_return_if_fail (view->element_stack != NULL);
|
2224
|
+
|
2225
|
+
view->element_stack = g_slist_delete_link (view->element_stack, view->element_stack);
|
2226
|
+
}
|
2227
|
+
|
2228
|
+
static gboolean
|
2229
|
+
lsm_svg_view_circular_reference_check (LsmSvgView *view, LsmSvgElement *element)
|
2230
|
+
{
|
2231
|
+
GSList *iter;
|
2232
|
+
|
2233
|
+
for (iter = view->element_stack; iter != NULL; iter = iter->next)
|
2234
|
+
if (iter->data == element) {
|
2235
|
+
lsm_debug_render ("[LsmSvgView::circular_reference_check] "
|
2236
|
+
"Circular reference to %s (id = %s)",
|
2237
|
+
lsm_dom_element_get_tag_name (LSM_DOM_ELEMENT (element)),
|
2238
|
+
lsm_dom_element_get_attribute (LSM_DOM_ELEMENT (element), "id"));
|
2239
|
+
return TRUE;
|
2240
|
+
}
|
2241
|
+
|
2242
|
+
return FALSE;
|
2243
|
+
}
|
2244
|
+
|
2245
|
+
void
|
2246
|
+
lsm_svg_view_push_style (LsmSvgView *view, LsmSvgStyle *style)
|
2247
|
+
{
|
2248
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2249
|
+
g_return_if_fail (style != NULL);
|
2250
|
+
|
2251
|
+
lsm_log_render ("[SvgView::push_style]");
|
2252
|
+
|
2253
|
+
if (view->style == NULL || (style->font_size != view->style->font_size)) {
|
2254
|
+
LsmSvgViewbox font_viewbox;
|
2255
|
+
LsmSvgViewbox *viewbox;
|
2256
|
+
double current_font_size_px;
|
2257
|
+
|
2258
|
+
if (view->style != NULL)
|
2259
|
+
current_font_size_px = view->style->font_size_px;
|
2260
|
+
else
|
2261
|
+
current_font_size_px = 0.0;
|
2262
|
+
|
2263
|
+
viewbox = view->viewbox_stack->data;
|
2264
|
+
font_viewbox.resolution_ppi = viewbox->resolution_ppi;
|
2265
|
+
font_viewbox.viewbox.x = 0;
|
2266
|
+
font_viewbox.viewbox.y = 0;
|
2267
|
+
font_viewbox.viewbox.width = current_font_size_px;
|
2268
|
+
font_viewbox.viewbox.height = current_font_size_px;
|
2269
|
+
|
2270
|
+
style->font_size_px = lsm_svg_length_normalize (&style->font_size->length, &font_viewbox,
|
2271
|
+
current_font_size_px, LSM_SVG_LENGTH_DIRECTION_VERTICAL);
|
2272
|
+
|
2273
|
+
if (style->font_size_px < 0.0)
|
2274
|
+
style->font_size_px = 0.0;
|
2275
|
+
lsm_log_render ("[SvgView::push_style] Font size = %g pixels", style->font_size_px);
|
2276
|
+
} else
|
2277
|
+
style->font_size_px = view->style->font_size_px;
|
2278
|
+
|
2279
|
+
view->style_stack = g_slist_prepend (view->style_stack, (void *) style);
|
2280
|
+
view->style = style;
|
2281
|
+
|
2282
|
+
}
|
2283
|
+
|
2284
|
+
void
|
2285
|
+
lsm_svg_view_push_composition (LsmSvgView *view, LsmSvgStyle *style)
|
2286
|
+
{
|
2287
|
+
gboolean do_filter;
|
2288
|
+
gboolean do_mask;
|
2289
|
+
gboolean do_clip;
|
2290
|
+
|
2291
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2292
|
+
g_return_if_fail (style != NULL);
|
2293
|
+
|
2294
|
+
lsm_svg_view_push_style (view, style);
|
2295
|
+
|
2296
|
+
lsm_log_render ("[SvgView::push_composition]");
|
2297
|
+
|
2298
|
+
do_clip = (g_strcmp0 (style->clip_path->value, "none") != 0);
|
2299
|
+
do_mask = (g_strcmp0 (style->mask->value, "none") != 0);
|
2300
|
+
do_filter = (g_strcmp0 (style->filter->value, "none") != 0);
|
2301
|
+
|
2302
|
+
if (G_UNLIKELY((view->style->opacity->value < 1.0 ||
|
2303
|
+
view->style->enable_background->value == LSM_SVG_ENABLE_BACKGROUND_NEW ||
|
2304
|
+
view->style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER) &&
|
2305
|
+
!do_filter &&
|
2306
|
+
!view->is_clipping &&
|
2307
|
+
!view->style->ignore_group_opacity &&
|
2308
|
+
view->dom_view.cairo != NULL)) {
|
2309
|
+
LsmSvgViewBackground *background;
|
2310
|
+
|
2311
|
+
lsm_debug_render ("[LsmSvgView::push_composition] Push group");
|
2312
|
+
cairo_push_group (view->dom_view.cairo);
|
2313
|
+
|
2314
|
+
background = g_slice_new (LsmSvgViewBackground);
|
2315
|
+
background->surface = cairo_get_group_target (view->dom_view.cairo);
|
2316
|
+
background->group_opacity = view->style->opacity->value;
|
2317
|
+
background->enable_background = view->style->enable_background->value == LSM_SVG_ENABLE_BACKGROUND_NEW;
|
2318
|
+
|
2319
|
+
view->background_stack = g_list_prepend (view->background_stack, background);
|
2320
|
+
}
|
2321
|
+
|
2322
|
+
if (G_UNLIKELY (do_clip)) {
|
2323
|
+
lsm_debug_render ("[LsmSvgView::push_style] Start clip '%s'", style->clip_path->value);
|
2324
|
+
lsm_svg_view_push_clip (view);
|
2325
|
+
}
|
2326
|
+
|
2327
|
+
if (G_UNLIKELY (do_mask)) {
|
2328
|
+
lsm_debug_render ("[LsmSvgView::push_style] Start mask '%s'", style->mask->value);
|
2329
|
+
lsm_svg_view_push_mask (view);
|
2330
|
+
}
|
2331
|
+
|
2332
|
+
/* Don't do filtering during a clipping operation, as filter will
|
2333
|
+
* create a new subsurface, where clipping should occur with the path
|
2334
|
+
* of the clip-path element. */
|
2335
|
+
if (G_UNLIKELY (do_filter && !view->is_clipping)) {
|
2336
|
+
lsm_debug_render ("[LsmSvgView::push_style] Start filter '%s'", style->filter->value);
|
2337
|
+
lsm_svg_view_push_filter (view);
|
2338
|
+
}
|
2339
|
+
}
|
2340
|
+
|
2341
|
+
void lsm_svg_view_pop_style (LsmSvgView *view)
|
2342
|
+
{
|
2343
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2344
|
+
g_return_if_fail (view->style_stack != NULL);
|
2345
|
+
|
2346
|
+
view->style_stack = g_slist_delete_link (view->style_stack, view->style_stack);
|
2347
|
+
view->style = view->style_stack != NULL ? view->style_stack->data : NULL;
|
2348
|
+
|
2349
|
+
lsm_log_render ("[SvgView::pop_style]");
|
2350
|
+
}
|
2351
|
+
|
2352
|
+
void lsm_svg_view_pop_composition (LsmSvgView *view)
|
2353
|
+
{
|
2354
|
+
gboolean do_filter;
|
2355
|
+
gboolean do_mask;
|
2356
|
+
gboolean do_clip;
|
2357
|
+
cairo_t *cairo;
|
2358
|
+
|
2359
|
+
g_return_if_fail (LSM_IS_SVG_VIEW (view));
|
2360
|
+
g_return_if_fail (view->style != NULL);
|
2361
|
+
|
2362
|
+
lsm_log_render ("[SvgView::pop_composition]");
|
2363
|
+
|
2364
|
+
do_clip = (g_strcmp0 (view->style->clip_path->value, "none") != 0);
|
2365
|
+
do_mask = (g_strcmp0 (view->style->mask->value, "none") != 0);
|
2366
|
+
do_filter = (g_strcmp0 (view->style->filter->value, "none") != 0);
|
2367
|
+
|
2368
|
+
/* Don't do filtering during a clipping operation, as filter will
|
2369
|
+
* create a new subsurface, where clipping should occur with the path
|
2370
|
+
* of the clip-path element. */
|
2371
|
+
if (G_UNLIKELY (do_filter && !view->is_clipping))
|
2372
|
+
lsm_svg_view_pop_filter (view);
|
2373
|
+
|
2374
|
+
if (G_UNLIKELY (do_mask))
|
2375
|
+
lsm_svg_view_pop_mask (view);
|
2376
|
+
|
2377
|
+
if (G_UNLIKELY (do_clip))
|
2378
|
+
lsm_svg_view_pop_clip (view);
|
2379
|
+
|
2380
|
+
cairo = view->dom_view.cairo;
|
2381
|
+
|
2382
|
+
if (G_UNLIKELY ((view->style->opacity->value < 1.0 ||
|
2383
|
+
view->style->enable_background->value == LSM_SVG_ENABLE_BACKGROUND_NEW ||
|
2384
|
+
view->style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER) &&
|
2385
|
+
!do_filter &&
|
2386
|
+
!view->is_clipping &&
|
2387
|
+
!view->style->ignore_group_opacity &&
|
2388
|
+
cairo != NULL)) {
|
2389
|
+
g_slice_free (LsmSvgViewBackground, view->background_stack->data);
|
2390
|
+
view->background_stack = g_list_delete_link (view->background_stack, view->background_stack);
|
2391
|
+
|
2392
|
+
cairo_pop_group_to_source (view->dom_view.cairo);
|
2393
|
+
if (G_UNLIKELY (view->style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER))
|
2394
|
+
lsm_cairo_set_comp_op (cairo, view->style->comp_op->value);
|
2395
|
+
cairo_paint_with_alpha (cairo, view->style->opacity->value);
|
2396
|
+
if (G_UNLIKELY (view->style->comp_op->value != LSM_SVG_COMP_OP_SRC_OVER))
|
2397
|
+
lsm_cairo_set_comp_op (cairo, LSM_SVG_COMP_OP_SRC_OVER);
|
2398
|
+
lsm_debug_render ("[LsmSvgView::pop_composition] Pop group");
|
2399
|
+
}
|
2400
|
+
|
2401
|
+
lsm_svg_view_pop_style (view);
|
2402
|
+
}
|
2403
|
+
|
2404
|
+
LsmSvgStyle *
|
2405
|
+
lsm_svg_view_get_current_style (LsmSvgView *view)
|
2406
|
+
{
|
2407
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), NULL);
|
2408
|
+
|
2409
|
+
return (LsmSvgStyle *) view->style;
|
2410
|
+
}
|
2411
|
+
|
2412
|
+
const LsmBox *
|
2413
|
+
lsm_svg_view_get_pattern_extents (LsmSvgView *view)
|
2414
|
+
{
|
2415
|
+
static LsmBox null_extents = {.x = 0.0, .y = 0.0, .width = 0.0, .height = 0.0};
|
2416
|
+
|
2417
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), &null_extents);
|
2418
|
+
g_return_val_if_fail (view->pattern_data != NULL, &null_extents);
|
2419
|
+
|
2420
|
+
return &view->pattern_data->extents;
|
2421
|
+
}
|
2422
|
+
|
2423
|
+
const LsmBox *
|
2424
|
+
lsm_svg_view_get_object_extents (LsmSvgView *view)
|
2425
|
+
{
|
2426
|
+
static LsmBox null_extents = {.x = 0.0, .y = 0.0, .width = 0.0, .height = 0.0};
|
2427
|
+
|
2428
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), &null_extents);
|
2429
|
+
g_return_val_if_fail (view->pattern_data != NULL, &null_extents);
|
2430
|
+
|
2431
|
+
return &view->pattern_data->object_extents;
|
2432
|
+
}
|
2433
|
+
|
2434
|
+
const LsmBox *
|
2435
|
+
lsm_svg_view_get_clip_extents (LsmSvgView *view)
|
2436
|
+
{
|
2437
|
+
static LsmBox null_extents = {.x = 0.0, .y = 0.0, .width = 0.0, .height = 0.0};
|
2438
|
+
|
2439
|
+
g_return_val_if_fail (LSM_IS_SVG_VIEW (view), &null_extents);
|
2440
|
+
g_return_val_if_fail (view->is_clipping, &null_extents);
|
2441
|
+
|
2442
|
+
return &view->clip_extents;
|
2443
|
+
}
|
2444
|
+
|
2445
|
+
static void
|
2446
|
+
lsm_svg_view_measure (LsmDomView *view, double *width, double *height, double *baseline)
|
2447
|
+
{
|
2448
|
+
LsmSvgSvgElement *svg_element;
|
2449
|
+
|
2450
|
+
svg_element = lsm_svg_document_get_root_element (LSM_SVG_DOCUMENT (view->document));
|
2451
|
+
if (svg_element == NULL)
|
2452
|
+
return;
|
2453
|
+
|
2454
|
+
lsm_svg_svg_element_measure (svg_element, LSM_SVG_VIEW (view), width, height);
|
2455
|
+
if (baseline)
|
2456
|
+
*baseline = *height;
|
2457
|
+
}
|
2458
|
+
|
2459
|
+
static void
|
2460
|
+
lsm_svg_view_render (LsmDomView *view)
|
2461
|
+
{
|
2462
|
+
LsmSvgView *svg_view;
|
2463
|
+
LsmSvgSvgElement *svg_element;
|
2464
|
+
|
2465
|
+
svg_view = LSM_SVG_VIEW (view);
|
2466
|
+
|
2467
|
+
svg_element = lsm_svg_document_get_root_element (LSM_SVG_DOCUMENT (view->document));
|
2468
|
+
if (svg_element == NULL)
|
2469
|
+
return;
|
2470
|
+
|
2471
|
+
svg_view->style_stack = NULL;
|
2472
|
+
svg_view->element_stack = NULL;
|
2473
|
+
svg_view->viewbox_stack = NULL;
|
2474
|
+
svg_view->matrix_stack = NULL;
|
2475
|
+
svg_view->pango_layout_stack = NULL;
|
2476
|
+
svg_view->background_stack = NULL;
|
2477
|
+
|
2478
|
+
svg_view->is_clipping = FALSE;
|
2479
|
+
svg_view->is_pango_layout_in_use = FALSE;
|
2480
|
+
svg_view->pango_layout = view->pango_layout;
|
2481
|
+
|
2482
|
+
svg_view->resolution_ppi = lsm_dom_view_get_resolution (view);
|
2483
|
+
|
2484
|
+
lsm_svg_svg_element_render (svg_element, svg_view);
|
2485
|
+
|
2486
|
+
if (svg_view->is_pango_layout_in_use)
|
2487
|
+
g_warning ("[LsmSvgView::render] Unfinished text redenring");
|
2488
|
+
|
2489
|
+
if (svg_view->is_clipping)
|
2490
|
+
g_warning ("[LsmSvgView::render] Unfinished clipping");
|
2491
|
+
|
2492
|
+
if (svg_view->pango_layout_stack != NULL) {
|
2493
|
+
g_warning ("[LsmSvgView::render] Dangling pango_layout in stack");
|
2494
|
+
g_slist_free (svg_view->pango_layout_stack);
|
2495
|
+
svg_view->pango_layout_stack = NULL;
|
2496
|
+
}
|
2497
|
+
|
2498
|
+
if (svg_view->matrix_stack != NULL) {
|
2499
|
+
g_warning ("[LsmSvgView::render] Dangling matrix in stack");
|
2500
|
+
g_slist_free (svg_view->matrix_stack);
|
2501
|
+
svg_view->matrix_stack = NULL;
|
2502
|
+
}
|
2503
|
+
if (svg_view->viewbox_stack != NULL) {
|
2504
|
+
g_warning ("[LsmSvgView::render] Dangling viewport in stack");
|
2505
|
+
g_slist_free (svg_view->viewbox_stack);
|
2506
|
+
svg_view->viewbox_stack = NULL;
|
2507
|
+
}
|
2508
|
+
if (svg_view->element_stack != NULL) {
|
2509
|
+
g_warning ("[LsmSvgView::render] Dangling element in stack");
|
2510
|
+
g_slist_free (svg_view->element_stack);
|
2511
|
+
svg_view->element_stack = NULL;
|
2512
|
+
}
|
2513
|
+
if (svg_view->style_stack != NULL) {
|
2514
|
+
g_warning ("[LsmSvgView::render] Dangling style in stack");
|
2515
|
+
g_slist_free (svg_view->style_stack);
|
2516
|
+
svg_view->style_stack = NULL;
|
2517
|
+
}
|
2518
|
+
if (svg_view->background_stack != NULL) {
|
2519
|
+
g_warning ("[LsmSvgView::render] Dangling background in stack");
|
2520
|
+
g_list_free (svg_view->background_stack);
|
2521
|
+
svg_view->background_stack = NULL;
|
2522
|
+
}
|
2523
|
+
}
|
2524
|
+
|
2525
|
+
static void
|
2526
|
+
lsm_svg_view_set_debug (LsmDomView *view, const char *feature, gboolean enable)
|
2527
|
+
{
|
2528
|
+
LsmSvgView *svg_view = LSM_SVG_VIEW (view);
|
2529
|
+
|
2530
|
+
if (g_strcmp0 (feature, "filter") == 0)
|
2531
|
+
svg_view->debug_filter = enable;
|
2532
|
+
else if (g_strcmp0 (feature, "mask") == 0)
|
2533
|
+
svg_view->debug_mask = enable;
|
2534
|
+
else if (g_strcmp0 (feature, "pattern") == 0)
|
2535
|
+
svg_view->debug_pattern = enable;
|
2536
|
+
else if (g_strcmp0 (feature, "group") == 0)
|
2537
|
+
svg_view->debug_group = enable;
|
2538
|
+
else if (g_strcmp0 (feature, "text") == 0)
|
2539
|
+
svg_view->debug_text = enable;
|
2540
|
+
}
|
2541
|
+
|
2542
|
+
LsmSvgView *
|
2543
|
+
lsm_svg_view_new (LsmSvgDocument *document)
|
2544
|
+
{
|
2545
|
+
LsmSvgView *view;
|
2546
|
+
|
2547
|
+
view = g_object_new (LSM_TYPE_SVG_VIEW, NULL);
|
2548
|
+
|
2549
|
+
lsm_dom_view_set_document (LSM_DOM_VIEW (view), LSM_DOM_DOCUMENT (document));
|
2550
|
+
|
2551
|
+
return view;
|
2552
|
+
}
|
2553
|
+
|
2554
|
+
static void
|
2555
|
+
lsm_svg_view_init (LsmSvgView *view)
|
2556
|
+
{
|
2557
|
+
view->debug_mask = FALSE;
|
2558
|
+
view->debug_filter = FALSE;
|
2559
|
+
view->debug_pattern = FALSE;
|
2560
|
+
}
|
2561
|
+
|
2562
|
+
static void
|
2563
|
+
lsm_svg_view_finalize (GObject *object)
|
2564
|
+
{
|
2565
|
+
parent_class->finalize (object);
|
2566
|
+
}
|
2567
|
+
|
2568
|
+
static void
|
2569
|
+
lsm_svg_view_class_init (LsmSvgViewClass *view_class)
|
2570
|
+
{
|
2571
|
+
GObjectClass *object_class = G_OBJECT_CLASS (view_class);
|
2572
|
+
LsmDomViewClass *d_view_class = LSM_DOM_VIEW_CLASS (view_class);
|
2573
|
+
|
2574
|
+
parent_class = g_type_class_peek_parent (view_class);
|
2575
|
+
|
2576
|
+
object_class->finalize = lsm_svg_view_finalize;
|
2577
|
+
|
2578
|
+
d_view_class->measure = lsm_svg_view_measure;
|
2579
|
+
d_view_class->render = lsm_svg_view_render;
|
2580
|
+
d_view_class->set_debug = lsm_svg_view_set_debug;
|
2581
|
+
}
|
2582
|
+
|
2583
|
+
G_DEFINE_TYPE (LsmSvgView, lsm_svg_view, LSM_TYPE_DOM_VIEW)
|