2022-12-11 12:58:10 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
import cairo
|
|
|
|
import numpy as np
|
2023-12-17 14:10:00 +01:00
|
|
|
from HersheyFonts import HersheyFonts
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Paper:
|
|
|
|
def __init__(self, width, height, margins=None):
|
|
|
|
self.width = width
|
|
|
|
self.height = height
|
|
|
|
self.margins = {
|
|
|
|
'top': 0,
|
|
|
|
'right': 0,
|
|
|
|
'bottom': 0,
|
|
|
|
'left': 0
|
|
|
|
}
|
2022-12-17 23:11:39 +01:00
|
|
|
self.set_margins(margins)
|
2023-12-24 13:53:56 +01:00
|
|
|
self.background_colour = None
|
2022-12-17 23:11:39 +01:00
|
|
|
|
2023-12-16 09:53:37 +01:00
|
|
|
def set_margins(self, margins=None, top=None, right=None, bottom=None,
|
|
|
|
left=None):
|
2022-12-11 12:58:10 +01:00
|
|
|
if isinstance(margins, int) or isinstance(margins, float):
|
|
|
|
self.margins['top'] = margins
|
|
|
|
self.margins['right'] = margins
|
|
|
|
self.margins['bottom'] = margins
|
|
|
|
self.margins['left'] = margins
|
|
|
|
elif isinstance(margins, list) and (isinstance(margins[0], int)
|
|
|
|
or isinstance(margins[0], float)):
|
|
|
|
if len(margins) == 2:
|
|
|
|
self.margins['top'] = margins[0]
|
|
|
|
self.margins['bottom'] = margins[0]
|
|
|
|
self.margins['left'] = margins[1]
|
|
|
|
self.margins['right'] = margins[1]
|
|
|
|
elif len(margins) == 3:
|
|
|
|
self.margins['top'] = margins[0]
|
|
|
|
self.margins['left'] = margins[1]
|
|
|
|
self.margins['right'] = margins[1]
|
|
|
|
self.margins['bottom'] = margins[2]
|
|
|
|
elif len(margins) == 4:
|
|
|
|
self.margins['top'] = margins[0]
|
|
|
|
self.margins['right'] = margins[1]
|
|
|
|
self.margins['bottom'] = margins[2]
|
|
|
|
self.margins['left'] = margins[3]
|
2023-12-16 09:53:37 +01:00
|
|
|
elif margins is not None:
|
2022-12-11 12:58:10 +01:00
|
|
|
self.margins = margins
|
|
|
|
|
2023-12-16 09:53:37 +01:00
|
|
|
if top is not None:
|
|
|
|
self.margins['top'] = top
|
|
|
|
if right is not None:
|
|
|
|
self.margins['right'] = right
|
|
|
|
if bottom is not None:
|
|
|
|
self.margins['bottom'] = bottom
|
|
|
|
if left is not None:
|
|
|
|
self.margins['left'] = left
|
|
|
|
|
2022-12-11 12:58:10 +01:00
|
|
|
@property
|
|
|
|
def content_width(self):
|
|
|
|
return self.width - self.margins['left'] - self.margins['right']
|
|
|
|
|
|
|
|
@property
|
|
|
|
def content_height(self):
|
|
|
|
return self.height - self.margins['top'] - self.margins['bottom']
|
|
|
|
|
|
|
|
def size(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return np.array([self.content_width, self.content_height])
|
|
|
|
else:
|
|
|
|
return np.array([self.width, self.height])
|
|
|
|
|
|
|
|
def is_within(self, point, honour_margins=True):
|
|
|
|
return ((self.left(honour_margins) < point[0])
|
|
|
|
and (point[0] < self.right(honour_margins))
|
|
|
|
and (self.top(honour_margins) < point[1])
|
|
|
|
and (point[1] < self.bottom(honour_margins)))
|
|
|
|
|
|
|
|
def top_left(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return np.array([self.margins['left'],
|
2023-12-17 14:09:20 +01:00
|
|
|
self.margins['top']])
|
2022-12-11 12:58:10 +01:00
|
|
|
else:
|
2023-12-17 14:09:20 +01:00
|
|
|
return np.array([0, 0])
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
def bottom_left(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return np.array([self.margins['left'],
|
2023-12-17 14:09:20 +01:00
|
|
|
self.height - self.margins['bottom']])
|
2022-12-11 12:58:10 +01:00
|
|
|
else:
|
2023-12-17 14:09:20 +01:00
|
|
|
return np.array([0, self.height])
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
def top_right(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return np.array([self.width - self.margins['right'],
|
2023-12-17 14:09:20 +01:00
|
|
|
self.margins['top']])
|
2022-12-11 12:58:10 +01:00
|
|
|
else:
|
2023-12-17 14:09:20 +01:00
|
|
|
return np.array([self.width, 0])
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
def bottom_right(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return np.array([self.width - self.margins['right'],
|
2023-12-17 14:09:20 +01:00
|
|
|
self.height - self.margins['bottom']])
|
2022-12-11 12:58:10 +01:00
|
|
|
else:
|
2023-12-17 14:09:20 +01:00
|
|
|
return np.array([self.width, self.height])
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
def centre(self, honour_margins=True):
|
|
|
|
return (self.bottom_left(honour_margins)
|
|
|
|
+ self.top_right(honour_margins)) / 2
|
|
|
|
|
|
|
|
def top(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return self.margins['top']
|
|
|
|
else:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def left(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return self.margins['left']
|
|
|
|
else:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def right(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return self.width - self.margins['right']
|
|
|
|
else:
|
|
|
|
return self.width
|
|
|
|
|
|
|
|
def bottom(self, honour_margins=True):
|
|
|
|
if honour_margins:
|
|
|
|
return self.height - self.margins['bottom']
|
|
|
|
else:
|
|
|
|
return self.height
|
|
|
|
|
|
|
|
|
|
|
|
A4_PORTRAIT = Paper(210, 297, 20)
|
2022-12-18 12:36:06 +01:00
|
|
|
A4_LANDSCAPE = Paper(297, 210, 20)
|
2022-12-11 12:58:10 +01:00
|
|
|
A6_PORTRAIT = Paper(105, 148, 8)
|
2022-12-18 12:36:06 +01:00
|
|
|
A6_LANDSCAPE = Paper(148, 105, 8)
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
|
2023-12-17 14:10:00 +01:00
|
|
|
def _measure_text(hf, text):
|
|
|
|
points = np.array([point for stroke in hf.strokes_for_text(text)
|
|
|
|
for point in stroke])
|
|
|
|
return {
|
|
|
|
'x': points[:, 0].min(),
|
|
|
|
'y': points[:, 1].min(),
|
|
|
|
'width': points[:, 0].max() - points[:, 0].min(),
|
|
|
|
'height': points[:, 1].max() - points[:, 1].min(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-12-11 12:58:10 +01:00
|
|
|
class Plotter:
|
|
|
|
|
|
|
|
def __init__(self, paper, line_width=0, colour=[0, 0, 0, 1]):
|
|
|
|
self.paper = paper
|
|
|
|
self.pen_down = False
|
|
|
|
self.position = np.array([0, 0])
|
|
|
|
|
|
|
|
self.layers = []
|
|
|
|
self.add_layer(colour, line_width)
|
|
|
|
|
|
|
|
def add_layer(self, colour=[0, 0, 0, 1], line_width=0.5, name='',
|
|
|
|
switch=True):
|
|
|
|
self.layers.append({
|
|
|
|
'line_width': line_width,
|
|
|
|
'colour': colour,
|
|
|
|
'name': name
|
|
|
|
})
|
|
|
|
if switch:
|
|
|
|
self.switch_layer(len(self.layers) - 1)
|
|
|
|
return len(self.layers) - 1
|
|
|
|
|
|
|
|
def switch_layer(self, layer):
|
|
|
|
self.current_layer = layer
|
|
|
|
|
|
|
|
def move_to(self, point):
|
|
|
|
self.pen_down = False
|
|
|
|
self.position = point
|
|
|
|
|
|
|
|
def line_to(self, point):
|
|
|
|
self.pen_down = True
|
|
|
|
self.position = point
|
|
|
|
|
2022-12-17 20:20:17 +01:00
|
|
|
def draw_frame(self):
|
|
|
|
self.move_to(self.paper.bottom_left())
|
|
|
|
self.line_to(self.paper.bottom_right())
|
|
|
|
self.line_to(self.paper.top_right())
|
|
|
|
self.line_to(self.paper.top_left())
|
|
|
|
self.line_to(self.paper.bottom_left())
|
|
|
|
|
2023-12-17 14:10:00 +01:00
|
|
|
def draw_text(self, text, position, font='futural', height=10, x_scale=1,
|
|
|
|
fit_width=None, fixed_height=True, spacing=None,
|
|
|
|
v_align='baseline', h_align='left'):
|
|
|
|
x, y = position
|
|
|
|
hf = HersheyFonts()
|
|
|
|
hf.load_default_font(font)
|
|
|
|
hf.normalize_rendering(height)
|
|
|
|
hf.render_options['scaley'] *= -1
|
|
|
|
hf.render_options['scalex'] *= x_scale
|
|
|
|
if spacing is not None:
|
|
|
|
hf.render_options['spacing'] = spacing
|
|
|
|
|
|
|
|
hf.render_options['yofs'] -= hf.render_options['base_line']
|
|
|
|
|
|
|
|
metrics = _measure_text(hf, text)
|
|
|
|
|
|
|
|
if fit_width is not None:
|
|
|
|
hf.render_options['scalex'] *= fit_width / metrics['width']
|
|
|
|
hf.render_options['spacing'] *= fit_width / metrics['width']
|
|
|
|
metrics['x'] *= fit_width / metrics['width']
|
|
|
|
if not fixed_height:
|
|
|
|
hf.render_options['scaley'] *= fit_width / metrics['width']
|
|
|
|
metrics['y'] *= fit_width / text_width
|
|
|
|
metrics['height'] *= fit_width / metrics['width']
|
|
|
|
metrics['width'] = fit_width
|
|
|
|
|
|
|
|
if h_align == 'centre':
|
|
|
|
x -= metrics['width'] / 2
|
|
|
|
elif h_align == 'right':
|
|
|
|
x -= metrics['width']
|
|
|
|
|
|
|
|
x -= metrics['x']
|
|
|
|
|
|
|
|
if v_align == 'bottom':
|
|
|
|
y -= metrics['y'] + metrics['height']
|
|
|
|
elif v_align == 'top':
|
|
|
|
y -= metrics['y']
|
|
|
|
|
|
|
|
for s in hf.strokes_for_text(text):
|
|
|
|
self.move_to((s[0][0] + x, s[0][1] + y))
|
|
|
|
for p in s[1:]:
|
|
|
|
self.line_to((p[0] + x, p[1] + y))
|
|
|
|
|
2022-12-11 12:58:10 +01:00
|
|
|
|
|
|
|
class SVGPlotter(Plotter):
|
|
|
|
|
|
|
|
def __init__(self, file_name, paper, line_width=0.5, colour=[0, 0, 0, 1]):
|
|
|
|
dpi = 72
|
2023-12-24 13:53:56 +01:00
|
|
|
width = paper.width / 25.4 * dpi
|
|
|
|
height = paper.height / 25.4 * dpi
|
2022-12-11 12:58:10 +01:00
|
|
|
self.file_name = file_name
|
2023-12-24 13:53:56 +01:00
|
|
|
self.surface = cairo.SVGSurface(file_name, width, height)
|
2022-12-11 12:58:10 +01:00
|
|
|
self.context = cairo.Context(self.surface)
|
|
|
|
self.context.scale(dpi / 25.4, dpi / 25.4)
|
|
|
|
self.context.set_line_cap(cairo.LINE_CAP_ROUND)
|
2022-12-18 12:40:11 +01:00
|
|
|
self.context.set_line_join(cairo.LINE_JOIN_ROUND)
|
2022-12-11 12:58:10 +01:00
|
|
|
|
2023-12-24 13:53:56 +01:00
|
|
|
if paper.background_colour is not None:
|
|
|
|
self.context.set_source_rgba(*paper.background_colour)
|
|
|
|
self.context.rectangle(0, 0, width, height)
|
|
|
|
self.context.fill()
|
|
|
|
|
2022-12-11 12:58:10 +01:00
|
|
|
super().__init__(paper, line_width, colour)
|
|
|
|
|
|
|
|
def move_to(self, point):
|
|
|
|
if self.pen_down:
|
|
|
|
self.context.stroke()
|
|
|
|
self.context.move_to(point[0], point[1])
|
|
|
|
super().move_to(point)
|
|
|
|
|
|
|
|
def line_to(self, point):
|
|
|
|
self.context.line_to(point[0], point[1])
|
|
|
|
super().line_to(point)
|
|
|
|
|
|
|
|
def finalise(self):
|
|
|
|
if self.pen_down:
|
|
|
|
self.context.stroke()
|
|
|
|
self.pen_down = False
|
|
|
|
self.surface.finish()
|
|
|
|
|
|
|
|
def switch_layer(self, layer):
|
|
|
|
if self.pen_down:
|
|
|
|
self.context.stroke()
|
|
|
|
|
|
|
|
super().switch_layer(layer)
|
|
|
|
self.context.set_source_rgba(
|
|
|
|
*self.layers[self.current_layer]['colour'])
|
|
|
|
self.context.set_line_width(
|
|
|
|
self.layers[self.current_layer]['line_width'])
|
|
|
|
|
|
|
|
return len(self.layers) - 1
|
|
|
|
|
|
|
|
|
|
|
|
class HPGLPlotter(Plotter):
|
|
|
|
|
|
|
|
def __init__(self, paper, file_name_pattern):
|
|
|
|
self.file_name_pattern = file_name_pattern
|
|
|
|
self.resolution = (4000 / 81.5, 4000 / 80.5)
|
|
|
|
self.file = None
|
|
|
|
super().__init__(paper)
|
|
|
|
|
|
|
|
def _layer_file_name(self, index):
|
|
|
|
return self.file_name_pattern.format(
|
|
|
|
index=index,
|
|
|
|
name=self.layers[index]['name'],
|
|
|
|
colour=self.layers[index]['colour']
|
|
|
|
)
|
|
|
|
|
|
|
|
def _mm_to_plotter(self, mm):
|
|
|
|
p = (mm[0] * self.resolution[0],
|
|
|
|
(self.paper.height - mm[1]) * self.resolution[1])
|
|
|
|
return f'{p[0]:.0f},{p[1]:.0f}'
|
|
|
|
|
|
|
|
def move_to(self, point):
|
|
|
|
if self.pen_down:
|
|
|
|
self.file.write(';PU')
|
|
|
|
else:
|
|
|
|
self.file.write(',')
|
|
|
|
self.file.write(self._mm_to_plotter(point))
|
|
|
|
return super().move_to(point)
|
|
|
|
|
|
|
|
def line_to(self, point):
|
|
|
|
if not self.pen_down:
|
|
|
|
self.file.write(';PD')
|
|
|
|
else:
|
|
|
|
self.file.write(',')
|
|
|
|
self.file.write(self._mm_to_plotter(point))
|
|
|
|
return super().line_to(point)
|
|
|
|
|
|
|
|
def add_layer(self, colour='black', line_width=0.5, name='', switch=True):
|
|
|
|
new_index = super().add_layer(colour=colour, line_width=line_width,
|
|
|
|
name=name, switch=False)
|
|
|
|
layer_file = open(self._layer_file_name(new_index), 'wt')
|
|
|
|
|
|
|
|
layer_file.write('IN;PU0,0')
|
|
|
|
|
|
|
|
if switch:
|
|
|
|
if self.file:
|
|
|
|
self.file.close()
|
|
|
|
self.file = layer_file
|
|
|
|
self.current_layer = len(self.layers) - 1
|
|
|
|
else:
|
|
|
|
layer_file.close()
|
|
|
|
|
|
|
|
return len(self.layers) - 1
|
|
|
|
|
|
|
|
def switch_layer(self, layer):
|
|
|
|
if self.pen_down:
|
|
|
|
self.file.write(';PU')
|
|
|
|
self.pen_down = False
|
|
|
|
self.file.close()
|
|
|
|
super().switch_layer(layer)
|
|
|
|
self.file = open(self._layer_file_name(self.current_layer), 'at')
|
|
|
|
|
|
|
|
def finalise(self):
|
|
|
|
self.file.close()
|
|
|
|
for i in range(len(self.layers)):
|
|
|
|
with open(self._layer_file_name(i), 'at') as file:
|
|
|
|
file.write(';PU0,0;IN')
|
2022-12-17 20:12:21 +01:00
|
|
|
|
|
|
|
|
|
|
|
class MultiPlotter(Plotter):
|
|
|
|
def __init__(self):
|
|
|
|
self.plotters = []
|
|
|
|
|
|
|
|
def move_to(self, point):
|
|
|
|
for plotter in self.plotters:
|
|
|
|
plotter.move_to(point)
|
|
|
|
|
|
|
|
def line_to(self, point):
|
|
|
|
for plotter in self.plotters:
|
|
|
|
plotter.line_to(point)
|
|
|
|
|
|
|
|
def add_layer(self, *args, **kwargs):
|
|
|
|
for plotter in self.plotters:
|
|
|
|
plotter.add_layer(*args, **kwargs)
|
|
|
|
|
2023-12-17 14:09:38 +01:00
|
|
|
def switch_layer(self, *args, **kwargs):
|
2022-12-17 20:12:21 +01:00
|
|
|
for plotter in self.plotters:
|
2023-12-17 14:09:38 +01:00
|
|
|
plotter.switch_layer(*args, **kwargs)
|
2022-12-17 20:12:21 +01:00
|
|
|
|
|
|
|
def finalise(self):
|
|
|
|
for plotter in self.plotters:
|
|
|
|
plotter.finalise()
|
|
|
|
|
|
|
|
def register_plotter(self, plotter):
|
2022-12-17 20:20:17 +01:00
|
|
|
self.plotters.append(plotter)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def paper(self):
|
2023-12-16 09:53:37 +01:00
|
|
|
return self.plotters[0].paper
|