• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

sile-typesetter / sile / 9428435077

08 Jun 2024 11:35AM UTC coverage: 64.56% (-9.9%) from 74.46%
9428435077

push

github

web-flow
Merge pull request #2047 from alerque/end-pars

23 of 46 new or added lines in 5 files covered. (50.0%)

1684 existing lines in 60 files now uncovered.

11145 of 17263 relevant lines covered (64.56%)

4562.45 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

83.33
/shapers/base.lua
1
--- SILE shaper class.
2
-- @interfaces shapers
3

4
-- local smallTokenSize = 20 -- Small words will be cached
5
-- local shapeCache = {}
6
-- local _key = function (options)
7
--   return table.concat({ options.family;options.language;options.script;options.size;("%d"):format(options.weight);options.style;options.variant;options.features;options.variations;options.direction;options.filename }, ";")
8
-- end
9

10
SILE.settings:declare({ parameter = "shaper.variablespaces", type = "boolean", default = true })
60✔
11
SILE.settings:declare({ parameter = "shaper.spaceenlargementfactor", type = "number or integer", default = 1 })
60✔
12
SILE.settings:declare({ parameter = "shaper.spacestretchfactor", type = "number or integer", default = 1 / 2 })
60✔
13
SILE.settings:declare({ parameter = "shaper.spaceshrinkfactor", type = "number or integer", default = 1 / 3 })
60✔
14

15
SILE.settings:declare({
60✔
16
   parameter = "shaper.tracking",
17
   type = "number or nil",
18
   default = nil,
19
})
20

21
-- Function for testing shaping in the repl
22
-- luacheck: ignore makenodes
23
-- TODO, figure out a way to explicitly register things in the repl env
24
makenodes = function (token, options)
25
   return SILE.shaper:createNnodes(token, SILE.font.loadDefaults(options or {}))
×
26
end
27

28
local function shapespace (spacewidth)
29
   spacewidth = SU.cast("measurement", spacewidth)
5,714✔
30
   -- In some scripts with word-level kerning, glue can be negative.
31
   -- Use absolute value to ensure stretch and shrink work as expected.
32
   local absoluteSpaceWidth = math.abs(spacewidth:tonumber())
5,714✔
33
   local length = spacewidth * SILE.settings:get("shaper.spaceenlargementfactor")
5,714✔
34
   local stretch = absoluteSpaceWidth * SILE.settings:get("shaper.spacestretchfactor")
5,714✔
35
   local shrink = absoluteSpaceWidth * SILE.settings:get("shaper.spaceshrinkfactor")
5,714✔
36
   return SILE.types.length(length, stretch, shrink)
2,857✔
37
end
38

39
local shaper = pl.class()
60✔
40
shaper.type = "shaper"
60✔
41
shaper._name = "base"
60✔
42

43
-- Return the length of a space character
44
-- with a particular set of font options,
45
-- giving preference to document.spaceskip
46
-- Caching this has no significant speedup
47
function shaper:measureSpace (options)
60✔
48
   local ss = SILE.settings:get("document.spaceskip")
9✔
49
   if ss then
9✔
UNCOV
50
      SILE.settings:temporarily(function ()
×
UNCOV
51
         SILE.settings:set("font.size", options.size)
×
UNCOV
52
         SILE.settings:set("font.family", options.family)
×
UNCOV
53
         SILE.settings:set("font.filename", options.filename)
×
UNCOV
54
         ss = ss:absolute()
×
55
      end)
UNCOV
56
      return ss
×
57
   end
58
   local items, width = self:shapeToken(" ", options)
9✔
59
   if not width and not items[1] then
9✔
60
      SU.warn("Could not measure the width of a space")
×
61
      return SILE.types.length()
×
62
   end
63
   return shapespace(width and width.length or items[1].width)
9✔
64
end
65

66
function shaper:measureChar (char)
60✔
67
   local options = SILE.font.loadDefaults({})
1,638✔
68
   options.tracking = SILE.settings:get("shaper.tracking")
3,276✔
69
   local items = self:shapeToken(char, options)
1,638✔
70
   if #items > 0 then
1,638✔
71
      return { height = items[1].height, width = items[1].width }
1,638✔
72
   else
73
      SU.error("Unable to measure character", char)
×
74
   end
75
end
76

77
-- Given a text and some font options, return a bunch of boxes
78
function shaper.shapeToken (_, _, _)
60✔
79
   SU.error("Abstract function shapeToken called", true)
×
80
end
81

82
-- Given font options, select a font. We will handle
83
-- caching here. Returns an arbitrary, implementation-specific
84
-- object (ie a PAL for Pango, font number for libtexpdf, ...)
85
function shaper.getFace (_)
60✔
86
   SU.error("Abstract function getFace called", true)
×
87
end
88

89
function shaper.addShapedGlyphToNnodeValue (_, _, _)
60✔
90
   SU.error("Abstract function addShapedGlyphToNnodeValue called", true)
×
91
end
92

93
function shaper.preAddNodes (_, _, _) end
60✔
94

95
function shaper:createNnodes (token, options)
60✔
96
   options.tracking = SILE.settings:get("shaper.tracking")
9,426✔
97
   local items, _ = self:shapeToken(token, options)
4,713✔
98
   if #items < 1 then
4,713✔
99
      return {}
×
100
   end
101
   local lang = options.language
4,713✔
102
   SILE.languageSupport.loadLanguage(lang)
4,713✔
103
   local nodeMaker = SILE.nodeMakers[lang] or SILE.nodeMakers.unicode
4,713✔
104
   local nodes = {}
4,713✔
105
   for node in nodeMaker(options):iterator(items, token) do
24,832✔
106
      table.insert(nodes, node)
10,693✔
107
   end
108
   return nodes
4,713✔
109
end
110

111
function shaper:formNnode (contents, token, options)
60✔
112
   local nnodeContents = {}
7,765✔
113
   -- local glyphs = {}
114
   local totalWidth = 0
7,765✔
115
   local totalDepth = 0
7,765✔
116
   local totalHeight = 0
7,765✔
117
   -- local glyphNames = {}
118
   local nnodeValue = { text = token, options = options, glyphString = {} }
7,765✔
119
   SILE.shaper:preAddNodes(contents, nnodeValue)
7,765✔
120
   local misfit = false
7,765✔
121
   if SILE.typesetter.frame and SILE.typesetter.frame:writingDirection() == "TTB" then
15,530✔
122
      if options.direction == "LTR" then
2✔
123
         misfit = true
×
124
      end
125
   else
126
      if options.direction == "TTB" then
7,763✔
127
         misfit = true
×
128
      end
129
   end
130
   for i = 1, #contents do
31,433✔
131
      local glyph = contents[i]
23,668✔
132
      if (options.direction == "TTB") ~= misfit then
23,668✔
133
         if glyph.width > totalHeight then
2✔
134
            totalHeight = glyph.width
2✔
135
         end
136
         totalWidth = totalWidth + glyph.height
2✔
137
      else
138
         if glyph.depth > totalDepth then
23,666✔
139
            totalDepth = glyph.depth
6,616✔
140
         end
141
         if glyph.height > totalHeight then
23,666✔
142
            totalHeight = glyph.height
10,186✔
143
         end
144
         totalWidth = totalWidth + glyph.width
23,666✔
145
      end
146
      self:addShapedGlyphToNnodeValue(nnodeValue, glyph)
23,668✔
147
   end
148
   table.insert(
15,530✔
149
      nnodeContents,
7,765✔
150
      SILE.types.node.hbox({
15,530✔
151
         depth = totalDepth,
7,765✔
152
         height = totalHeight,
7,765✔
153
         misfit = misfit,
7,765✔
154
         width = SILE.types.length(totalWidth),
15,530✔
155
         value = nnodeValue,
7,765✔
156
      })
157
   )
158
   return SILE.types.node.nnode({
7,765✔
159
      nodes = nnodeContents,
7,765✔
160
      text = token,
7,765✔
161
      misfit = misfit,
7,765✔
162
      options = options,
7,765✔
163
      language = options.language,
7,765✔
164
   })
7,765✔
165
end
166

167
function shaper.makeSpaceNode (_, options, item)
60✔
168
   local width
169
   if SILE.settings:get("shaper.variablespaces") then
5,696✔
170
      width = shapespace(item.width)
5,696✔
171
   else
UNCOV
172
      width = SILE.shaper:measureSpace(options)
×
173
   end
174
   return SILE.types.node.glue(width)
2,848✔
175
end
176

177
function shaper.debugVersions (_) end
60✔
178

179
return shaper
60✔
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc