summaryrefslogtreecommitdiffstats
path: root/share/extensions/tests/test_inkex_svg.py
diff options
context:
space:
mode:
Diffstat (limited to 'share/extensions/tests/test_inkex_svg.py')
-rw-r--r--share/extensions/tests/test_inkex_svg.py461
1 files changed, 461 insertions, 0 deletions
diff --git a/share/extensions/tests/test_inkex_svg.py b/share/extensions/tests/test_inkex_svg.py
new file mode 100644
index 0000000..4f1e8b7
--- /dev/null
+++ b/share/extensions/tests/test_inkex_svg.py
@@ -0,0 +1,461 @@
+#!/usr/bin/env python
+# coding=utf-8
+#
+# Copyright (C) 2018 Martin Owens
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA.
+#
+"""
+Test the svg interface for inkscape extensions.
+"""
+from inkex.transforms import Vector2d
+from inkex.utils import addNS
+from inkex import Guide
+from inkex.tester import TestCase
+from inkex.tester.svg import svg, svg_file, uu_svg
+
+class BasicSvgTest(TestCase):
+ """Basic svg tests"""
+
+ def test_svg_load(self):
+ """Test loading an svg with the right parser"""
+ self.assertEqual(type(svg()).__name__, 'SvgDocumentElement')
+
+ def test_add_ns(self):
+ """Test adding a namespace to a tag"""
+ self.assertEqual(addNS('g', 'svg'), '{http://www.w3.org/2000/svg}g')
+ self.assertEqual(addNS('h', 'inkscape'), '{http://www.inkscape.org/namespaces/inkscape}h')
+ self.assertEqual(addNS('i', 'sodipodi'),
+ '{http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd}i')
+ self.assertEqual(addNS('{p}j'), '{p}j')
+
+ def test_svg_ids(self):
+ """Test a list of ids from an svg document"""
+ self.assertEqual(svg('id="apples"').get_ids(), {'apples'})
+
+ def test_svg_new_id(self):
+ """Test generatign a new id for a given tag"""
+ doc = svg('id="apples"')
+ usedids = set(['apples'])
+ for prefix in ['apples'] * 3:
+ newid = doc.get_unique_id(prefix)
+ self.assertTrue(newid.startswith(prefix))
+ self.assertTrue(newid not in usedids)
+ usedids.add(newid)
+
+ def test_svg_select_id(self):
+ """Select an id from the document"""
+ doc = svg('id="bananas"')
+ doc.selection.set('bananas')
+ self.assertEqual(doc.selection['bananas'], doc)
+ self.assertEqual(doc.selection.first(), doc)
+ doc = svg('id="apples"')
+ doc.selected.set(doc.getElementById('apples'))
+ self.assertEqual(doc.selection['apples'], doc)
+ self.assertEqual(doc.selection.first(), doc)
+
+ def test_svg_by_class(self):
+ """Select elements by class"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ elems = doc.getElementsByClass('frog')
+ self.assertEqual([elem.get_id() for elem in elems],
+ ['path3902', 'text3926', 'path3900', 'rect3898'])
+ elems = doc.getElementsByClass('apple')
+ self.assertEqual([elem.get_id() for elem in elems], ['text3926', 'rect3898'])
+
+ def test_svg_by_href(self):
+ """Select element by xlink href"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ elem = doc.getElementsByHref('path3900')[0]
+ self.assertEqual(elem.TAG, 'textPath')
+ self.assertEqual(elem.get_id(), 'textPath3923')
+ elem = doc.getElementsByHref('path3904')[0]
+ self.assertEqual(elem.TAG, 'textPath')
+ self.assertEqual(elem.get_id(), 'textPath3906')
+ self.assertEqual(doc.getElementsByHref('not-an-id'), [])
+
+ def test_svg_by_url_link(self):
+ """Select element by urls in styles"""
+ doc = svg_file(self.data_file('svg', 'markers.svg'))
+ elem = doc.getElementsByStyleUrl('Arrow1Lend')[0]
+ self.assertEqual(elem.get_id(), 'dimension')
+ elem = doc.getElementsByStyleUrl('Arrow1Lstart')[0]
+ self.assertEqual(elem.get_id(), 'dimension')
+ self.assertEqual(doc.getElementsByStyleUrl('not-an-id'), [])
+
+ def test_selected_bbox(self):
+ """Can we get a bounding box from the selected items"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ doc.selected.set('path3904', 'path3902')
+ from inkex.transforms import BoundingBox
+ x, y, w, h = 199.544, 156.412, 377.489, 199.972 # from inkscape --query-all
+ expected_3904 = BoundingBox((x, x + w), (y, y + h))
+ x, y, w, h = 145.358, 478.373, 439.135, 419.142 # from inkscape --query-all
+ expected_3902 = BoundingBox((x, x + w), (y, y + h))
+ expected = list(expected_3902 + expected_3904)
+
+ for x, y in zip(expected, doc.selection.bounding_box()):
+ self.assertDeepAlmostEqual(tuple(x), tuple(y), delta=1e-3)
+
+
+ def test_svg_name(self):
+ """Can get the sodipodi name attribute"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ self.assertEqual(doc.name, 'Nouveau document 1')
+
+ def test_svg_nameview(self):
+ """Can get the sodipodi nameview element"""
+ doc = svg()
+ self.assertEqual(doc.namedview.center.x, 0)
+ self.assertEqual(type(doc.namedview).__name__, 'NamedView')
+
+ def test_svg_layers(self):
+ """Selected layer is selected"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ self.assertEqual(doc.get_current_layer().get('id'), 'layer3')
+ doc = svg('id="empty"')
+ self.assertEqual(doc.get_current_layer(), doc)
+
+ def test_svg_center_position(self):
+ """SVG with namedview has a center position"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ self.assertTrue(doc.namedview.center.is_close((30.714286, 520.0)))
+ self.assertTrue(svg().namedview.center.is_close(Vector2d()))
+
+ def test_defs(self):
+ """Can get the defs from an svg file"""
+ doc = svg_file(self.data_file('svg', 'markers.svg'))
+ self.assertEqual(len(doc.defs), 2)
+ doc = svg('id="empty"')
+ self.assertEqual(len(doc.defs), 0)
+
+ def test_scale(self):
+ """Scale of a document"""
+ doc = svg('id="empty" viewBox="0 0 100 100" width="200" height="200"')
+ self.assertEqual(doc.width, 200.0)
+ self.assertEqual(doc.get_viewbox()[2], 100.0)
+ self.assertEqual(doc.scale, 2.0)
+ doc = svg('id="empty" viewBox="0 0 0 0" width="200" height="200"')
+ self.assertEqual(doc.scale, 1.0)
+
+class NamedViewTest(TestCase):
+ """Tests for the named view functionality"""
+
+ def test_create_guide(self):
+ """Test creating guides"""
+ doc = svg_file(self.data_file('svg', 'multilayered-test.svg'))
+ namedview = doc.namedview
+ self.assertEqual(len(namedview.get_guides()), 0)
+
+ namedview.add(Guide().move_to(50, 50, 45))
+ self.assertEqual(len(namedview.get_guides()), 1)
+ guide, = namedview.get_guides()
+ self.assertEqual(guide.get('position'), '50,50')
+ self.assertEqual(guide.get('orientation'), '0.707107,-0.707107')
+
+
+class GetDocumentWidthTest(TestCase):
+ """Tests for Effect.width."""
+
+ def test_no_dimensions(self):
+ """An empty width value should be default zero width"""
+ self.assertEqual(svg().width, 0)
+
+ def test_empty_width(self):
+ """An empty width value should be the same as a missing width."""
+ self.assertEqual(svg('width=""').width, 0)
+
+ def test_empty_viewbox(self):
+ """An empty viewBox value should be the same as a missing viewBox."""
+ self.assertEqual(svg('viewBox=""').width, 0)
+
+ def test_empty_width_and_viewbox(self):
+ """Empty values for both should be the same as both missing."""
+ self.assertEqual(svg('width="" viewBox=""').width, 0)
+
+ def test_width_only(self):
+ """Test a fixed width"""
+ self.assertAlmostEqual(svg('width="120mm"').width, 453.5433071)
+
+ def test_width_and_viewbox(self):
+ """If both are present, width overrides viewBox."""
+ self.assertAlmostEqual(svg('width="120mm" viewBox="0 0 22 99"').width, 453.5433071)
+
+ def test_viewbox_only(self):
+ """IF only the viewBox is present"""
+ self.assertEqual(svg('viewBox="0 0 22 99"').width, 22.0)
+
+ def test_only_valid_viewbox(self):
+ """An empty width value should be the same as a missing width."""
+ self.assertEqual(svg('width="" viewBox="0 0 22 99"').width, 22.0)
+
+ def test_non_zero_viewbox_x(self):
+ """Demonstrate that a non-zero x value (viewbox[0]) does not affect the width value."""
+ self.assertEqual(svg('width="" viewBox="5 7 22 99"').width, 22.0)
+
+
+class GetDocumentHeightTest(TestCase):
+ """Tests for Effect.height."""
+
+ def test_no_dimensions(self):
+ """Test height from blank svg"""
+ self.assertEqual(svg().height, 0)
+
+ def test_empty_height(self):
+ """An empty height value should be the same as a missing height."""
+ self.assertEqual(svg('height=""').height, 0)
+
+ def test_empty_viewbox(self):
+ """An empty viewBox value should be the same as a missing viewBox."""
+ self.assertEqual(svg('viewBox=""').height, 0)
+
+ def test_empty_height_viewbox(self):
+ """Empty values for both should be the same as both missing."""
+ self.assertEqual(svg('height="" viewBox=""').height, 0)
+
+ def test_height_only(self):
+ """A simple height only in px"""
+ self.assertEqual(svg('height="330px"').height, 330)
+
+ def test_height_and_viewbox(self):
+ """If both are present, height overrides viewBox."""
+ self.assertEqual(svg('height="330px" viewBox="0 0 22 99"').height, 330)
+
+ def test_viewbox_only(self):
+ """Height from viewBox only"""
+ self.assertEqual(svg('viewBox="0 0 22 99"').height, 99.0)
+
+ def test_no_height_valid_viewbox(self):
+ """An empty height value should be the same as a missing height."""
+ self.assertEqual(svg('height="" viewBox="0 0 22 99"').height, 99.0)
+
+ def test_non_zero_viewbox_y(self):
+ """Demonstrate that a non-zero y value (viewbox[1]) does not affect the height value."""
+ self.assertEqual(svg('height="" viewBox="5 7 22 99"').height, 99.0)
+
+
+class GetDocumentUnitTest(TestCase):
+ """Tests for Effect.unit."""
+
+ def test_no_dimensions(self):
+ """Default units with no arguments"""
+ self.assertEqual(svg().unit, 'px')
+
+ def test_width_only(self):
+ """"Units from document width only"""
+ # TODO: Determine whether returning 'px' in this case is the
+ # intended behavior.
+ self.assertEqual(svg('width="100m"').unit, 'px')
+
+ def test_height_only(self):
+ """Units from document height only"""
+ # TODO: Determine whether returning 'px' in this case is the
+ # intended behavior.
+ self.assertEqual(svg('height="100m"').unit, 'px')
+
+ def test_viewbox_only(self):
+ """Test viewbox only document units"""
+ self.assertEqual(svg('viewBox="0 0 377 565"').unit, 'px')
+
+ # Unit-ratio tests. Don't exhaustively test every unit conversion, just
+ # demonstrate that the logic works.
+
+ def test_width_and_viewbox_px(self):
+ """100mm is ~377px, so unit should be 'px'."""
+ self.assertEqual(svg('width="100mm" viewBox="0 0 377 565"').unit, 'px')
+
+ def test_width_and_viewbox_in(self):
+ """100mm is ~3.94in, so unit should be 'in'."""
+ self.assertEqual(svg('width="100mm" viewBox="0 0 3.94 5.90"').unit, 'in')
+
+ def test_unitless_width_and_viewbox(self):
+ """Unitless width should be treated as 'px'."""
+ # 3779px is ~1m, so unit should be 'm'.
+ self.assertEqual(svg('width="3779" viewBox="0 0 1 1.5"').unit, 'm')
+
+ def test_height_with_viewbox(self):
+ """150mm is ~5.90in, so unit should be 'in', but height is ignored"""
+ # TODO: Determine whether returning 'px' in this case is the intended
+ # behavior.
+ self.assertEqual(svg('height="150mm" viewBox="0 0 3.94 5.90"').unit, 'px')
+
+ def test_height_width_and_viewbox(self):
+ """100mm is ~23.6pc, so unit should be 'pc'."""
+ doc = svg('width="100mm" height="150mm" viewBox="0 0 23.6 35.4"')
+ self.assertEqual(doc.unit, 'pc')
+
+ def test_large_error_reverts_to_px(self):
+ """'px' instead of using the closest match 'pc'."""
+ # 100mm is ~23.6pc; 24.1 is ~2% off from that, so unit should fall back
+ self.assertEqual(svg('width="100mm" viewBox="0 0 24.1 35.4"').unit, 'px')
+
+ # TODO: Demonstrate that unknown width units are treated as px while
+ # determining the ratio.
+
+ # 100px fallback tests. Although that value is an arbitrary default, it's
+ # possible for users of inkex.Effect to depend on this behavior.
+ # NOTE: Do not treat the existence of these tests as a reason to preserve
+ # the 100px fallback logic.
+
+ def test_bad_width_number(self):
+ """Fallback test: Bad numbers default to 100"""
+ # First, demonstrate that 1in is 2.54cm, so unit should be 'cm'.
+ self.assertEqual(svg('width="1in" viewBox="0 0 2.54 1"').unit, 'cm')
+
+ # Corrupt the width to contain an invalid number component; note that
+ # the units change to 'px'. This is because the corrupt number part is
+ # replaced with 100px, producing a width of "100px";
+ self.assertEqual(svg('width="ABCDin" viewBox="0 0 2.54 1"').unit, 'px')
+
+ def test_bad_viewbox_entry(self):
+ """Fallback test: Bad viewBox default to 100"""
+ # First, demonstrate that 3779px is 1m, so unit should be 'm'.
+ self.assertEqual(svg('width="3779px" viewBox="0 0 1 1"').unit, 'm')
+
+ # Corrupt the viewBox to include a non-float value; will default to 'px'
+ self.assertEqual(svg('width="3779px" viewBox="x 0 1 1"').unit, 'px')
+
+
+class UserUnitTest(TestCase):
+ """Tests for methods that are based on the value of unit."""
+
+ def assertToUserUnit(self, user_unit, test_value, expected): # pylint: disable=invalid-name
+ """Checks a user unit and a test_value against the expected result"""
+ doc = uu_svg(user_unit)
+ self.assertEqual(doc.unit, user_unit, msg=svg)
+ self.assertAlmostEqual(doc.unittouu(test_value), expected)
+
+ def assertFromUserUnit(self, user_unit, value, unit, expected): # pylint: disable=invalid-name
+ """Check converting from a user unity for the test_value"""
+ self.assertAlmostEqual(uu_svg(user_unit).uutounit(value, unit), expected)
+
+ # Unit-ratio tests. Don't exhaustively test every unit conversion, just
+ # demonstrate that the logic works.
+
+ def test_unittouu_in_to_cm(self):
+ """1in is ~2.54cm"""
+ self.assertToUserUnit('cm', '1in', 2.54)
+
+ def test_unittouu_yd_to_m(self):
+ """1yd is ~0.9144m"""
+ self.assertToUserUnit('m', '1yd', 0.9144)
+
+ def test_unittouu_identity(self):
+ """If the input and output units are the same, the input and output
+ values should exactly be the same, too."""
+ self.assertToUserUnit('pc', '9.87654321pc', 9.87654321)
+
+ def test_unittouu_unitless_input(self):
+ """Passing a unitless value to unittouu() should treat the units as 'px'."""
+ self.assertToUserUnit('in', '96', 1) # 1in == 96px
+
+ def test_unittouu_empty_input(self):
+ """Passing an empty string to unittouu() should treat the value as zero."""
+ self.assertToUserUnit('in', '', 0)
+
+ def test_unittouu_parsing(self):
+ """Test user unit parsing forms"""
+ for value in (
+ '100pc',
+ '100 pc',
+ ' 100pc',
+ '100pc ',
+ '+100pc',
+ '100.0pc',
+ '100.0e0pc',
+ '10.0e1pc',
+ '10.0e+1pc',
+ '1000.0e-1pc',
+ '.1e+3pc',
+ '+.1e+3pc',
+ ):
+ # 100pc is ~3.937in
+ self.assertToUserUnit('px', value, 1600)
+
+ def test_unittouu_bad_input_number(self):
+ """Bad input number"""
+ self.assertToUserUnit('cm', '1in', 2.54)
+ # Demonstrate that 1in is ~2.54cm.
+
+ # Corrupt the input to contain an invalid number component; note that
+ # the result changes to zero.
+ self.assertToUserUnit('cm', 'ABCDin', 0)
+
+ def test_unittouu_bad_input_unit(self):
+ """Bad input unit"""
+ # Demonstrate that 1.0in passes through without change.
+ self.assertToUserUnit('in', '1.0in', 1.0)
+
+ # Corrupt the input to contain an invalid unit component; note that the
+ # result changes to 0.0, because corrupt parsing is zero px.
+ # it used to be the ratio between inches and pixels. This was
+ # because unittouu() treats unknown units as 'px'.
+ self.assertToUserUnit('in', '1.0ABCD', 0)
+
+ # Unit-ratio tests. Don't exhaustively test every unit conversion, just
+ # demonstrate that the logic works.
+
+ def test_uutounit_cm_to_in(self):
+ """Convert 1 user unit ('in') to 'cm'."""
+ self.assertFromUserUnit('in', 1, 'cm', 2.54) # 1in is ~2.54cm
+
+ def test_uutounit_m_to_yd(self):
+ """Convert 1 user unit ('yd') to 'm'."""
+ self.assertFromUserUnit('yd', 1, 'm', 0.9144) # 1yd is ~0.9144m
+
+ def test_uutounit_identity(self):
+ """If the input and output units are the same, the input and output
+ values should exactly be the same, too."""
+ self.assertFromUserUnit('pc', 9.87654321, 'pc', 9.87654)
+
+ def test_uutounit_unknown_unit(self):
+ """Demonstrate that passing an unknown unit string to uutounit()"""
+ self.assertEqual(uu_svg('in').uutounit(1, 'px'), 96.0)
+
+ def test_adddocumentunit_common(self):
+ """Test common add_unit results"""
+ # For valid float inputs, the output should be the input with the user unit appended.
+ doc = uu_svg('pt')
+ cases = (
+ # Input, expected output
+ (100, '100pt'),
+ ('100', '100pt'),
+ ('+100', '100pt'),
+ ('-100', '-100pt'),
+ ('100.0', '100pt'),
+ ('100.0e0', '100pt'),
+ ('10.0e1', '100pt'),
+ ('10.0e+1', '100pt'),
+ ('1000.0e-1', '100pt'),
+ ('.1e+3', '100pt'),
+ ('+.1e+3', '100pt'),
+ (' 100', '100pt'),
+ ('100 ', '100pt'),
+ (' 100 ', '100pt'),
+ )
+ for input_value, expected in cases:
+ self.assertEqual(doc.add_unit(input_value), expected)
+
+ def test_adddocumentunit_non_float(self):
+ """Strings that are invalid floats should pass through unchanged."""
+ doc = uu_svg('pt')
+ inputs = (
+ '',
+ 'ABCD',
+ '.',
+ ' ',
+ )
+ for value in inputs:
+ self.assertEqual(doc.add_unit(value), '')