Added support for both linked and embedded images

This commit is contained in:
Mario Voigt 2020-08-18 14:36:13 +02:00
parent 807fa69a16
commit ab5c2e54fc

View File

@ -31,17 +31,13 @@ import os
import sys import sys
import base64 import base64
from io import StringIO, BytesIO from io import StringIO, BytesIO
import urllib.parse import urllib.request as urllib
import urllib.request
import inkex import inkex
from PIL import Image from PIL import Image
from lxml import etree from lxml import etree
inkex.localization.localize
DEBUG = False DEBUG = False
# int r = ( hexcolor >> 16 ) & 0xFF; # int r = ( hexcolor >> 16 ) & 0xFF;
# int g = ( hexcolor >> 8 ) & 0xFF; # int g = ( hexcolor >> 8 ) & 0xFF;
# int b = hexcolor & 0xFF; # int b = hexcolor & 0xFF;
@ -53,7 +49,6 @@ def hex_to_int_color(v):
assert(len(v) == 6) assert(len(v) == 6)
return int(v[:2], 16), int(v[2:4], 16), int(v[4:6], 16) return int(v[:2], 16), int(v[2:4], 16), int(v[4:6], 16)
class Pixel2SVG(inkex.Effect): class Pixel2SVG(inkex.Effect):
def __init__(self): def __init__(self):
inkex.Effect.__init__(self) inkex.Effect.__init__(self)
@ -69,55 +64,30 @@ class Pixel2SVG(inkex.Effect):
self.arg_parser.add_argument("--color", default="FFFFFF", help="Special color") self.arg_parser.add_argument("--color", default="FFFFFF", help="Special color")
self.arg_parser.add_argument("--tab") self.arg_parser.add_argument("--tab")
def getImagePath(self, node, xlink): def checkImagePath(self, node):
""" """Embed the data of the selected Image Tag element"""
Find image file, return path xlink = node.get('xlink:href')
""" if xlink and xlink[:5] == 'data:':
absref = node.get(inkex.addNS('absref', 'sodipodi')) # No need, data alread embedded
url = urlparse(xlink) return
href = urllib.request.url2pathname
path = '' url = urllib.urlparse(xlink)
#path selection strategy: href = urllib.url2pathname(url.path)
# 1. href if absolute
# 2. realpath-ified href
# 3. absref, only if the above does not point to a file
if href is not None:
path = os.path.realpath(href)
if (not os.path.isfile(path)):
if absref is not None:
path = absref
try: # Primary location always the filename itself.
path = unicode(path, "utf-8") path = self.absolute_href(href or '')
except TypeError:
path = path
if (not os.path.isfile(path)): # Backup directory where we can find the image
inkex.errormsg(_( if not os.path.isfile(path):
"No xlink:href or sodipodi:absref attributes found, " + path = node.get('sodipodi:absref', path)
"or they do not point to an existing file! Unable to find image file."))
if path: if not os.path.isfile(path):
inkex.errormsg(_("Sorry we could not locate %s") % str(path)) inkex.errormsg('File not found "{}". Unable to embed image.').format(path)
return False return
if (os.path.isfile(path)): if (os.path.isfile(path)):
return path return path
def getImageData(self, xlink):
"""
Read, decode and return data of embedded image
"""
comma = xlink.find(',')
data = ''
if comma > 0:
data = base64.decodebytes(xlink[comma:].encode('UTF-8'))
else:
inkex.errormsg(_("Failed to read embedded image data."))
return data
def getImage(self, node): def getImage(self, node):
image_element=self.svg.find('.//{http://www.w3.org/2000/svg}image') image_element=self.svg.find('.//{http://www.w3.org/2000/svg}image')
image_string=image_element.get('{http://www.w3.org/1999/xlink}href') image_string=image_element.get('{http://www.w3.org/1999/xlink}href')
@ -170,7 +140,18 @@ class Pixel2SVG(inkex.Effect):
Parse RGBA values of linked bitmap image, create a group and Parse RGBA values of linked bitmap image, create a group and
draw the rectangles (SVG pixels) inside the new group draw the rectangles (SVG pixels) inside the new group
""" """
image = self.getImage(node) self.path = self.checkImagePath(node) # This also ensures the file exists
if self.path is None: # check if image is embedded or linked
image_string = node.get('{http://www.w3.org/1999/xlink}href')
# find comma position
i = 0
while i < 40:
if image_string[i] == ',':
break
i = i + 1
image = Image.open(BytesIO(base64.b64decode(image_string[i + 1:len(image_string)])))
else:
image = Image.open(self.path)
if image: if image:
# init, set limit (default: 256) # init, set limit (default: 256)
@ -281,10 +262,10 @@ class Pixel2SVG(inkex.Effect):
else: else:
# bail out with larger images # bail out with larger images
inkex.errormsg(_( inkex.errormsg(
"Bailing out: this extension is not intended for large images.\n" + "Bailing out: this extension is not intended for large images.\n" +
"The current limit is %spx for either dimension of the bitmap image." "The current limit is %spx for either dimension of the bitmap image."
% pixel2svg_max)) % pixel2svg_max)
sys.exit(0) sys.exit(0)
# clean-up? # clean-up?
@ -292,7 +273,7 @@ class Pixel2SVG(inkex.Effect):
inkex.utils.debug("Done.") inkex.utils.debug("Done.")
else: else:
inkex.errormsg(_("Bailing out: No supported image file or data found")) inkex.errormsg("Bailing out: No supported image file or data found")
sys.exit(1) sys.exit(1)
def effect(self): def effect(self):
@ -307,8 +288,7 @@ class Pixel2SVG(inkex.Effect):
self.vectorizeImage(node) self.vectorizeImage(node)
if not found_image: if not found_image:
inkex.errormsg(_("Please select one or more bitmap image(s) for Pixel2SVG")) inkex.errormsg("Please select one or more bitmap image(s) for Pixel2SVG")
sys.exit(0) sys.exit(0)
if __name__ == '__main__':
Pixel2SVG().run() Pixel2SVG().run()