luametalatex/luametalatex-pdf-font.lua

275 lines
9.6 KiB
Lua

local mapping = require'luametalatex-pdf-font-map'
mapping.mapfile(kpse.find_file('pdftex.map', 'map', true))
local tounicode = {
[-3] = require'luametalatex-pdf-font-cmap3',
require'luametalatex-pdf-font-cmap1',
require'luametalatex-pdf-font-cmap2',
}
local function allcids(fontdir)
local cids = {}
local scale = 1000/fontdir.size
for i,v in pairs(fontdir.characters) do
v.used = true
cids[#cids+1] = {v.index or i, math.floor(v.width*scale+.5), v.tounicode}
end
return cids
end
local function buildW(f, usedcids)
local used = #usedcids
if used == 0 then return "" end
local result = {}
local index = 1
while index <= used do
local width = usedcids[index][2]
local last = index
while last ~= used and usedcids[last+1][2] == width do
last = last + 1
end
if index == last then
local span = {width}
width = (usedcids[last+1] or {})[2]
while (last + 2 <= used
and usedcids[last+2][2] ~= width
or last + 1 == used)
and usedcids[last+1][1]-usedcids[last][1] <= 2 do
for i=usedcids[last][1]+1,usedcids[last+1][1]-1 do
span[#span+1] = 0
end
last = last + 1
span[#span+1] = width
width = (usedcids[last + 1] or {})[2]
end
result[#result+1] = string.format("%i[%s]", usedcids[index][1], table.concat(span, ' '))
else
result[#result+1] = string.format("%i %i %i ", usedcids[index][1], usedcids[last][1], width)
end
index = last + 1
end
return table.concat(result)
end
local function fontdescriptor(pdf, basefont, fontdir, stream, kind)
local scale = 1000/fontdir.size
return string.format(
"<</Type/FontDescriptor/FontName/%s/Flags %i/FontBBox[%i %i %i %i]/ItalicAngle %i/Ascent %i/Descent %i/CapHeight %i/StemV %i/FontFile%s %i 0 R>>",
basefont,
4,-- FIXME: Flags ??? (4 means "symbolic")
fontdir.bbox[1], fontdir.bbox[2], fontdir.bbox[3], fontdir.bbox[4], -- FIXME: How to determine BBox?
math.floor(math.atan(fontdir.parameters.slant or fontdir.parameters[1] or 0, 0x10000)+.5),
fontdir.bbox[4], fontdir.bbox[2],
fontdir.parameters[8] and math.floor(fontdir.parameters[8]*scale+0.5) or fontdir.bbox[4],
fontdir.StemV or 100, -- FIXME: How to determine StemV?
kind, stream)
end
local function cidmap1byte(pdf)
if not pdf.cidmap1byte then
pdf.cidmap1byte = string.format(" %i 0 R", pdf:stream(nil, [[/Type/CMap/CMapName/Identity-8-H/CIDSystemInfo<</Registry(Adobe)/Ordering(Identity)/Supplement 0>>]],
[[%!PS-Adobe-3.0 Resource-CMap
%%DocumentNeededResources : ProcSet (CIDInit)
%%IncludeResource : ProcSet (CIDInit)
%%BeginResource : CMap (Identity-8-H)
%%Title: (Custom 8bit Identity CMap)
%%Version: 1.000
%%EndComments
/CIDInit /ProcSet findresource begin
12 dict begin
begincmap
/CIDSystemInfo
3 dict dup begin
/Registry (Adobe) def
/Ordering (Identity) def
/Supplement 0 def
end def
/CMapName /Identity-8-H def
/CMapVersion 1.000 def
/CMapType 1 def
/WMode 0 def
1 begincodespacerange
<00> <FF>
endcodespacerange
1 begincidrange
<00> <FF> 0
endcidrange
endcmap
CMapName currentdict /CMap defineresource pop
end
end
%%EndResource
%%EOF]]))
end
return pdf.cidmap1byte
end
local function cidmap3byte(pdf)
if not pdf.cidmap3byte then
pdf.cidmap3byte = string.format(" %i 0 R", pdf:stream(nil, [[/Type/CMap/CMapName/Identity-Var-H/CIDSystemInfo<</Registry(Adobe)/Ordering(Identity)/Supplement 0>>]],
[[%!PS-Adobe-3.0 Resource-CMap
%%DocumentNeededResources : ProcSet (CIDInit)
%%IncludeResource : ProcSet (CIDInit)
%%BeginResource : CMap (Identity-Var-H)
%%Title: (Custom 8-24bit variable size Identity CMap)
%%Version: 1.000
%%EndComments
/CIDInit /ProcSet findresource begin
12 dict begin
begincmap
/CIDSystemInfo
3 dict dup begin
/Registry (Adobe) def
/Ordering (Identity) def
/Supplement 0 def
end def
/CMapName /Identity-Var-H def
/CMapVersion 1.000 def
/CMapType 1 def
/WMode 0 def
3 begincodespacerange
<FF0000> <FF807F>
<00> <7F>
<8000> <FEFF>
endcodespacerange
3 begincidrange
<00> <7F> 0
<8000> <FEFF> 128
<FF0000> <FF807F> 32640
endcidrange
endcmap
CMapName currentdict /CMap defineresource pop
end
end
%%EndResource
%%EOF]]))
end
return pdf.cidmap3byte
end
local capitals = {string.byte("ABCDEFGHIJKLMNOPQRSTUVWXYZ", 1, -1)}
local function gen_subsettagchar(i, left, ...)
if left == 0 then return ... end
return gen_subsettagchar(i//#capitals, left-1, capitals[i%#capitals+1], ...)
end
local function gen_subsettag(ident)
local i = string.unpack("j", sha2.digest256(ident))
return string.char(gen_subsettagchar(i, 7, 43))
end
local function buildfont0cff(pdf, fontdir, usedcids)
local basefont = fontdir.psname or fontdir.fullname or fontdir.name -- FIXME: Subset-tag(?), Name-Escaping(?), fallback
if fontdir.cff then
cff = fontdir:cff(usedcids)
else
if fontdir.filename then
if fontdir.format == "type1" then
cff = require'luametalatex-pdf-font-t1'(fontdir.filename, fontdir.encoding)(fontdir, usedcids)
elseif fontdir.format == "opentype" then
cff = require'luametalatex-pdf-font-cff'(fontdir.filename, fontdir.encodingbytes == 1 and (fontdir.encoding or true))(fontdir, usedcids)
else
error[[Unsupported format]]
end
else
return string.format("<</Type/Font/Subtype/Type1/BaseFont/%s/FontDescriptor %i 0 R/FirstChar %i/LastChar %i>>", basefont, -42, usedcids[1][1], usedcids[#usedcids][1])
end
end
local widths = buildW(fontdir, usedcids) -- Do this after generating the CFF to allow for extracting the widths from the font file
basefont = gen_subsettag(widths)..basefont
local cidfont = pdf:indirect(nil, string.format(
"<</Type/Font/Subtype/CIDFontType0/BaseFont/%s/CIDSystemInfo<</Registry(Adobe)/Ordering(Identity)/Supplement 0>>/FontDescriptor %i 0 R/W[%s]>>",
basefont,
pdf:indirect(nil, fontdescriptor(pdf, basefont, fontdir, pdf:stream(nil, '/Subtype/CIDFontType0C', cff), 3)),
widths
))
return basefont, cidfont
end
local function buildfont0ttf(pdf, fontdir, usedcids)
local basefont = fontdir.psname or fontdir.fullname or fontdir.name -- FIXME: Subset-tag(?), Name-Escaping(?), fallback
local ttf
if fontdir.ttf then
ttf = fontdir:ttf(usedcids) -- WARNING: If you implement this: You always have to add a .notdef glyph at index 0. This one is *not* included in usedcids
else
ttf = require'luametalatex-pdf-font-ttf'(fontdir.filename, 1, fontdir.encoding)(fontdir, usedcids)
end
local lastcid = -1
local cidtogid = {}
for i=1,#usedcids do
cidtogid[2*i-1] = string.rep("\0\0", usedcids[i][1]-lastcid-1)
cidtogid[2*i] = string.pack(">I2", i)
lastcid = usedcids[i][1]
end
cidtogid = pdf:stream(nil, "", table.concat(cidtogid))
local widths = buildW(fontdir, usedcids)
basefont = gen_subsettag(widths)..basefont
local cidfont = pdf:indirect(nil, string.format(
"<</Type/Font/Subtype/CIDFontType2/BaseFont/%s/CIDSystemInfo<</Registry(Adobe)/Ordering(Identity)/Supplement 0>>/FontDescriptor %i 0 R/W[%s]/CIDToGIDMap %i 0 R>>",
basefont,
pdf:indirect(nil, fontdescriptor(pdf, basefont, fontdir, pdf:stream(nil, string.format('/Length1 %i', #ttf), ttf), 2)),
widths,
cidtogid
))
return basefont, cidfont
end
local function buildfont0(pdf, fontdir, usedcids)
usedcids = usedcids or allcids(fontdir)
table.sort(usedcids, function(a,b) return a[1]<b[1] end)
local enc
if fontdir.encodingbytes == 1 then
enc = cidmap1byte(pdf)
elseif false then -- FIXME: This should only be used for encodingbyzes == -3 (variable, max 3)
fontdir.encodingbytes = -3 -- FIXME
enc = cidmap3byte(pdf)
else
enc = "/Identity-H"
end
local basefont, cidfont = (fontdir.format == "truetype" and buildfont0ttf or buildfont0cff)(pdf, fontdir, usedcids)
local touni = pdf:stream(nil, "", tounicode[fontdir.encodingbytes](fontdir, usedcids)) -- Done late to allow for defaults set from the font file
return string.format(
"<</Type/Font/Subtype/Type0/BaseFont/%s/Encoding%s/ToUnicode %i 0 R/DescendantFonts[%i 0 R]>>",
basefont,
enc,
touni,
cidfont)
end
local fontextensions = {
ttf = {"truetype", "truetype fonts",},
otf = {"opentype", "opentype fonts",},
pfb = {"type1", "type1 fonts",},
}
fontextensions.cff = fontextensions.otf
local fontformats = {
fontextensions.pfb, fontextensions.otf, fontextensions.ttf,
}
return function(pdf, fontdir, usedcids)
if fontdir.encodingbytes == 0 then fontdir.encodingbytes = nil end
if fontdir.format == "unknown" or not fontdir.format or fontdir.encodingbytes == 1 then -- TODO: How to check this?
fontdir.encodingbytes = fontdir.encodingbytes or 1
local mapentry = mapping.fontmap[fontdir.name]
if mapentry then
local format = mapentry[3] and mapentry[3]:sub(-4, -4) == '.' and fontextensions[mapentry[3]:sub(-3, -1)]
if format then
fontdir.format = format[1]
fontdir.filename = kpse.find_file(mapentry[3], format[2])
if mapentry[4] then
fontdir.encoding = kpse.find_file(mapentry[4], 'enc files')
end
goto format_set
else
for _, format in ipairs(fontformats) do
local font = kpse.find_file(mapentry[3],format[2])
if font then
fontdir.format = "type1"
fontdir.filename = font
if mapentry[4] then
fontdir.encoding = kpse.find_file(mapentry[4], 'enc files')
end
goto format_set
end
end
end
end
fontdir.format = "type3"
::format_set::
else
fontdir.encodingbytes = fontdir.encodingbytes or 2
end
if fontdir.format == "type3" then
error[[Currently unsupported]] -- TODO
else
return buildfont0(pdf, fontdir, usedcids)
end
end