-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathluametalatex-pdf-savedbox.lua
105 lines (96 loc) · 3.81 KB
/
luametalatex-pdf-savedbox.lua
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
local writer -- = require'luametalatex-nodewriter' -- This would introduce some cyclic dependency
local pdfvariable = pdf.variable
-- XForms currently have the form {width, height, depth, objnum, attributes, list, margin}
local xforms = {}
local utils = require'luametalatex-pdf-utils'
local strip_floats = utils.strip_floats
local to_bp = utils.to_bp
local function shipout(pfile, xform, fontdirs, usedglyphs)
local list, margin = xform.list, xform.margin
if not list then return end -- Already shipped out
local last_page = cur_page cur_page = nil
local out, resources, annots = writer(pfile, list, fontdirs, usedglyphs)
cur_page = last_page
assert(annots == '')
if pdfvariable.xformattr ~= '' or pdfvariable.xformresources ~= '' then
texio.write_nl('term and log', 'WARNING (savedboxresource shipout): Ignoring unsupported PDF variables xformattr and xformresources. Specify resources and attributes for specific XForms instead.')
end
local bbox = strip_floats(string.format('/BBox[%f %f %f %f]', -to_bp(margin), -to_bp(list.depth+margin), to_bp(list.width+margin), to_bp(list.height+margin)))
local dict = string.format('/Subtype/Form%s/Resources%s%s', bbox, resources(xform.resources), xform.attributes or '')
node.flush_list(list)
xform.list = nil
pfile:stream(xform.objnum, dict, out)
end
local function save(pfile, n, attr, resources, immediate, type, margin, fontdirs, usedglyphs)
local objnum = pfile:getobj()
local xform = {
list = assert(n, 'List required for saveboxresource'),
width = n.width,
height = n.height,
depth = n.depth,
attributes = attr,
resources = resources,
margin = margin,
objnum = objnum,
-- type = type, -- TODO: Not yet used. Do we need this at all?
}
xforms[objnum] = xform
if immediate then
shipout(pfile, xform, fontdirs, usedglyphs)
end
return objnum
end
local function adjust_sizes(width, height, depth, real_width, real_height, real_depth)
if not depth then
if height then
local scale = height/real_height
depth = (real_depth*scale + .5)//1
width = width or (real_width*scale + .5)//1
elseif width then
local scale = width/real_width
depth = (real_depth*scale + .5)//1
height = (real_height*scale + .5)//1
else
width, height, depth = real_width, real_height, real_depth
end
elseif height then
width = width or (real_width*(height+depth)/(real_height+real_depth) + .5)//1
else
width = width or real_width
local scale = width/real_width
height = ((real_depth+real_height)*scale + .5)//1 - depth
end
return width, height, depth
end
local ruleid = node.id'rule'
local ruletypes = node.subtypes'rule'
local boxrule
for n, name in next, ruletypes do
if name == 'box' then boxrule = n break end
end
local function use(objnum, width, height, depth)
local xform = xforms[objnum]
if not xform then return nil, nil, nil, nil end
width, height, depth = adjust_sizes(width, height, depth, xform.width, xform.height, xform.depth)
local n = node.direct.new(ruleid, boxrule)
node.direct.setdata(n, objnum)
node.direct.setwhd(n, width, height, depth)
return node.direct.tonode(n), width, height, depth
end
local function do_box(objnum, p, n, x, y)
local xform = assert(xforms[objnum], 'Invalid XForm')
shipout(p.file, xform, p.fontdirs, p.usedglyphs)
local width, height, depth = node.direct.getwhd(n)
local xscale, yscale = width / xform.width, (height+depth) / (xform.height+xform.depth)
p.resources.XObject['Fm' .. tostring(objnum)] = objnum
pdf.write('page', strip_floats(string.format('q %f 0 0 %f %f %f cm /Fm%i Do Q',
xscale, yscale,
to_bp(x), to_bp(y-depth+yscale*xform.depth),
objnum)), nil, nil, p)
end
return {
save = save,
use = use,
ship = do_box,
init_nodewriter = function(t, nodewriter) writer, t.init_nodewriter = nodewriter, nil end,
}