1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
import inkex
import cubicsuperpath
from .svg import apply_transforms
from .svg.tags import SVG_USE_TAG, SVG_SYMBOL_TAG, CONNECTION_START, CONNECTION_END, XLINK_HREF
class CommandParseError(Exception):
pass
def get_node_by_url(svg, url):
# url will be #path12345. Find the object at the other end.
if url is None:
raise CommandParseError("url is None")
if not url.startswith('#'):
raise CommandParseError("invalid connection url: %s" % url)
id = url[1:]
try:
return svg.xpath(".//*[@id='%s']" % id)[0]
except (IndexError, AttributeError):
raise CommandParseError("could not find node by url %s" % id)
class Command(object):
def __init__(self, connector):
self.connector = connector
self.svg = self.connector.getroottree().getroot()
self.parse_command()
def parse_connector_path(self):
path = cubicsuperpath.parsePath(self.connector.get('d'))
return apply_transforms(path, self.connector)
def parse_command(self):
path = self.parse_connector_path()
neighbors = [
(get_node_by_url(self.svg, self.connector.get(CONNECTION_START)), path[0][0][1]),
(get_node_by_url(self.svg, self.connector.get(CONNECTION_END)), path[0][-1][1])
]
if neighbors[0][0].tag != SVG_USE_TAG:
neighbors.reverse()
if neighbors[0][0].tag != SVG_USE_TAG:
raise CommandParseError("connector does not point to a use tag")
self.symbol = self.get_node_by_url(self.svg, neighbors[0][0].get(XLINK_HREF))
if self.symbol.tag != SVG_SYMBOL_TAG:
raise CommandParseError("use points to non-symbol")
self.command = self.symbol.get('id')
if self.command.startswith('inkstitch_'):
self.command = self.command[10:]
else:
raise CommandParseError("symbol is not an Ink/Stitch command")
self.target = neighbors[1][0]
self.target_point = neighbors[1][1]
def __repr__(self):
return "Command('%s', %s)" % (self.command, self.target_point)
class StandaloneCommand(object):
def __init__(self, symbol):
self.symbol = symbol
self.svg = self.connector.getroottree().getroot()
self.parse_command()
def parse_command(self):
pass
def find_commands(node):
"""Find the symbols this node is connected to and return them as Commands"""
# find all paths that have this object as a connection
xpath = ".//*[@inkscape:connection-start='#%(id)s' or @inkscape:connection-end='#%(id)s']" % dict(id=node.get('id'))
connectors = node.getroottree().getroot().xpath(xpath, namespaces=inkex.NSS)
# try to turn them into commands
commands = []
for connector in connectors:
try:
commands.append(Command(connector))
except ValueError:
# Parsing the connector failed, meaning it's not actually an Ink/Stitch command.
pass
return commands
def layer_commands(layer, command):
"""Find standalone (unconnected) command symbols in this layer."""
pass
def global_commands(svg):
"""Find all unconnected command symbols in the SVG."""
xpath = ".//svg:use[starts-with(@xlink:href, '#inkstitch_')]"
symbols = svg.xpath(xpath, namespace=inkex.NSS)
commands = []
for symbol in symbols:
try:
commands.append(StandaloneCommand(symbol))
except CommandParseError:
pass
return commands
def is_command(node):
return CONNECTION_START in node.attrib or CONNECTION_END in node.attrib
|