inkstitch/lib/commands.py

170 wiersze
5.1 KiB
Python
Czysty Zwykły widok Historia

2018-06-21 19:41:06 +00:00
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
from .utils import cache
2018-08-20 19:49:19 +00:00
from .i18n import _, N_
2018-06-21 19:41:06 +00:00
COMMANDS = {
2018-08-20 19:49:19 +00:00
# L10N command attached to an object
"fill_start": N_("Fill stitch starting position"),
2018-08-20 19:49:19 +00:00
# L10N command attached to an object
"fill_end": N_("Fill stitch ending position"),
2018-08-20 19:49:19 +00:00
# L10N command attached to an object
"stop": N_("Stop (pause machine) after sewing this object"),
2018-08-20 19:49:19 +00:00
# L10N command attached to an object
"trim": N_("Trim thread after sewing this object"),
2018-08-20 19:49:19 +00:00
# L10N command attached to an object
"ignore_object": N_("Ignore this object (do not stitch)"),
2018-08-20 19:49:19 +00:00
# L10N command that affects entire layer
"ignore_layer": N_("Ignore layer (do not stitch any objects in this layer)")
}
OBJECT_COMMANDS = [ "fill_start", "fill_end", "stop", "trim", "ignore_object" ]
LAYER_COMMANDS = [ "ignore_layer" ]
2018-06-21 19:41:06 +00:00
2018-08-03 00:04:08 +00:00
class CommandParseError(Exception):
pass
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
class BaseCommand(object):
@property
@cache
def description(self):
return get_command_description(self.command)
2018-08-17 00:30:37 +00:00
def parse_symbol(self):
if self.symbol.tag != SVG_SYMBOL_TAG:
raise CommandParseError("use points to non-symbol")
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
self.command = self.symbol.get('id')
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
if self.command.startswith('inkstitch_'):
self.command = self.command[10:]
else:
raise CommandParseError("symbol is not an Ink/Stitch command")
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
def get_node_by_url(self,url):
# url will be #path12345. Find the corresponding object.
if url is None:
raise CommandParseError("url is None")
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
if not url.startswith('#'):
raise CommandParseError("invalid connection url: %s" % url)
2018-08-03 00:04:08 +00:00
2018-08-17 00:30:37 +00:00
id = url[1:]
try:
return self.svg.xpath(".//*[@id='%s']" % id)[0]
except (IndexError, AttributeError):
raise CommandParseError("could not find node by url %s" % id)
2018-08-03 00:04:08 +00:00
2018-08-17 00:30:37 +00:00
class Command(BaseCommand):
2018-08-03 00:04:08 +00:00
def __init__(self, connector):
self.connector = connector
self.svg = self.connector.getroottree().getroot()
self.parse_command()
2018-06-21 19:41:06 +00:00
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 = [
2018-08-17 00:30:37 +00:00
(self.get_node_by_url(self.connector.get(CONNECTION_START)), path[0][0][1]),
(self.get_node_by_url(self.connector.get(CONNECTION_END)), path[0][-1][1])
2018-06-21 19:41:06 +00:00
]
if neighbors[0][0].tag != SVG_USE_TAG:
neighbors.reverse()
if neighbors[0][0].tag != SVG_USE_TAG:
2018-08-03 00:04:08 +00:00
raise CommandParseError("connector does not point to a use tag")
2018-06-21 19:41:06 +00:00
2018-08-17 00:30:37 +00:00
self.symbol = self.get_node_by_url(neighbors[0][0].get(XLINK_HREF))
self.parse_symbol()
2018-06-21 19:41:06 +00:00
self.target = neighbors[1][0]
self.target_point = neighbors[1][1]
def __repr__(self):
return "Command('%s', %s)" % (self.command, self.target_point)
2018-08-17 00:30:37 +00:00
class StandaloneCommand(BaseCommand):
def __init__(self, use):
self.node = use
2018-08-17 02:50:34 +00:00
self.svg = self.node.getroottree().getroot()
2018-08-03 00:04:08 +00:00
self.parse_command()
def parse_command(self):
2018-08-17 00:30:37 +00:00
self.symbol = self.get_node_by_url(self.node.get(XLINK_HREF))
if self.symbol.tag != SVG_SYMBOL_TAG:
raise CommandParseError("use points to non-symbol")
self.parse_symbol()
2018-08-03 00:04:08 +00:00
def get_command_description(command):
2018-08-20 19:49:19 +00:00
return _(COMMANDS[command])
2018-08-03 00:04:08 +00:00
2018-06-21 19:41:06 +00:00
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))
2018-08-17 00:30:37 +00:00
except CommandParseError:
2018-06-21 19:41:06 +00:00
# Parsing the connector failed, meaning it's not actually an Ink/Stitch command.
pass
return commands
2018-08-03 00:04:08 +00:00
def layer_commands(layer, command):
"""Find standalone (unconnected) command symbols in this layer."""
2018-08-17 00:30:37 +00:00
commands = []
for standalone_command in standalone_commands(layer.getroottree().getroot()):
if standalone_command.command == command:
2018-08-17 02:50:34 +00:00
if layer in standalone_command.node.iterancestors():
2018-08-17 00:30:37 +00:00
commands.append(command)
return commands
def standalone_commands(svg):
2018-08-03 00:04:08 +00:00
"""Find all unconnected command symbols in the SVG."""
xpath = ".//svg:use[starts-with(@xlink:href, '#inkstitch_')]"
2018-08-17 02:50:34 +00:00
symbols = svg.xpath(xpath, namespaces=inkex.NSS)
2018-08-03 00:04:08 +00:00
commands = []
for symbol in symbols:
try:
commands.append(StandaloneCommand(symbol))
except CommandParseError:
pass
return commands
2018-06-21 19:41:06 +00:00
def is_command(node):
return CONNECTION_START in node.attrib or CONNECTION_END in node.attrib