1
0
Fork 0
inkscape/share/extensions/ink2canvas_lib/canvas.py
Daniel Baumann 02d935e272
Adding upstream version 1.4.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 23:40:13 +02:00

199 lines
6.5 KiB
Python

# coding=utf-8
#
# Copyright (C) 2011 Karlisson Bezerra <contact@hacktoon.com>
#
# 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-1301, USA.
#
"""
Canvas module for ink2canvas extension
"""
from textwrap import dedent
from inkex import Color, Style
# pylint: disable=too-many-public-methods, disable=missing-function-docstring
class Canvas:
"""Canvas API helper class"""
def __init__(self, parent, width, height, context="ctx"):
self.obj = context
self.code = [] # stores the code
self.style = Style()
self.style_cache = Style() # stores the previous style applied
self.parent = parent
self.width = width
self.height = height
def write(self, text):
self.code.append("\t" + text.replace("ctx", self.obj) + "\n")
def output(self):
html = """
<!DOCTYPE html>
<html>
<head>
<title>Inkscape Output</title>
</head>
<body>
<canvas id='canvas' width='%d' height='%d'></canvas>
<script>
var %s = document.getElementById("canvas").getContext("2d");
%s
</script>
</body>
</html>
"""
return dedent(html) % (self.width, self.height, self.obj, "".join(self.code))
def equal_style(self, style, key):
"""Checks if the last style used is the same or there's no style yet"""
if key in self.style_cache:
return True
if key not in style:
return True
return style[key] == self.style_cache[key]
def begin_path(self):
self.write("ctx.beginPath();")
def create_linear_gradient(self, href, x1, y1, x2, y2):
data = (href, x1, y1, x2, y2)
self.write(
"var %s = \
ctx.createLinearGradient(%f,%f,%f,%f);"
% data
)
def create_radial_gradient(self, href, cx1, cy1, rx, cx2, cy2, ry):
data = (href, cx1, cy1, rx, cx2, cy2, ry)
self.write(
"var %s = ctx.createRadialGradient\
(%f,%f,%f,%f,%f,%f);"
% data
)
def add_color_stop(self, href, pos, color):
self.write("%s.addColorStop(%f, %s);" % (href, pos, color))
@staticmethod
def get_color(rgb, alpha):
return "'{}'".format(str(Color(rgb).to_rgba(alpha)))
def set_gradient(self, href):
"""
for stop in gstops:
style = simplestyle.parseStyle(stop.get("style"))
stop_color = style["stop-color"]
opacity = style["stop-opacity"]
color = self.get_color(stop_color, opacity)
pos = float(stop.get("offset"))
self.add_color_stop(href, pos, color)
"""
return None # href
def set_opacity(self, value):
self.write("ctx.globalAlpha = %.1f;" % float(value))
def set_fill(self, value):
alpha = self.style("fill-opacity")
if not value.startswith("url("):
fill = self.get_color(value, alpha)
self.write("ctx.fillStyle = %s;" % fill)
def set_stroke(self, value):
alpha = self.style("stroke-opacity")
self.write("ctx.strokeStyle = %s;" % self.get_color(value, alpha))
def set_stroke_width(self, value):
self.write("ctx.lineWidth = %f;" % self.parent.svg.unittouu(value))
def set_stroke_linecap(self, value):
self.write("ctx.lineCap = '%s';" % value)
def set_stroke_linejoin(self, value):
self.write("ctx.lineJoin = '%s';" % value)
def set_stroke_miterlimit(self, value):
self.write("ctx.miterLimit = %s;" % value)
def set_font(self, value):
self.write('ctx.font = "%s";' % value)
def move_to(self, x, y):
self.write("ctx.moveTo(%f, %f);" % (x, y))
def line_to(self, x, y):
self.write("ctx.lineTo(%f, %f);" % (x, y))
def close_path(self):
self.write("ctx.closePath();")
def quadratic_curve_to(self, cpx, cpy, x, y):
data = (cpx, cpy, x, y)
self.write("ctx.quadraticCurveTo(%f, %f, %f, %f);" % data)
def bezier_curve_to(self, x1, y1, x2, y2, x, y):
data = (x1, y1, x2, y2, x, y)
self.write("ctx.bezierCurveTo(%f, %f, %f, %f, %f, %f);" % data)
def rect(self, x, y, width, height, rx=0, ry=0):
if rx or ry:
# rounded rectangle, starts top-left anticlockwise
self.move_to(x, y + ry)
self.line_to(x, y + height - ry)
self.quadratic_curve_to(x, y + height, x + rx, y + height)
self.line_to(x + width - rx, y + height)
self.quadratic_curve_to(x + width, y + height, x + width, y + height - ry)
self.line_to(x + width, y + ry)
self.quadratic_curve_to(x + width, y, x + width - rx, y)
self.line_to(x + rx, y)
self.quadratic_curve_to(x, y, x, y + ry)
else:
self.write("ctx.rect(%f, %f, %f, %f);" % (x, y, width, height))
def arc(self, x, y, r, a1, a2, flag):
data = (x, y, r, a1, a2, flag)
self.write("ctx.arc(%f, %f, %f, %f, %.8f, %d);" % data)
def fill_text(self, text, x, y):
self.write('ctx.fillText("%s", %f, %f);' % (text, x, y))
def translate(self, cx, cy):
self.write("ctx.translate(%f, %f);" % (cx, cy))
def rotate(self, angle):
self.write("ctx.rotate(%f);" % angle)
def scale(self, rx, ry):
self.write("ctx.scale(%f, %f);" % (rx, ry))
def transform(self, m11, m12, m21, m22, dx, dy):
data = (m11, m12, m21, m22, dx, dy)
self.write("ctx.transform(%f, %f, %f, %f, %f, %f);" % data)
def save(self):
self.write("ctx.save();")
def restore(self):
self.write("ctx.restore();")
def finish_path(self):
if self.style("fill") is not None:
self.write("ctx.fill();")
if self.style("stroke") is not None:
self.write("ctx.stroke();")