• 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

74.26
/types/node.lua
1
--- SILE node type.
2
-- @types node
3

4
local nodetypes = {}
118✔
5

6
-- This infinity needs to be smaller than an actual infinity but bigger than the infinite stretch
7
-- added by the typesetter. See https://github.com/sile-typesetter/sile/issues/227
8
local infinity = SILE.types.measurement(1e13)
236✔
9

10
local function _maxnode (nodes, dim)
11
   local dims = SU.map(function (node)
35,650✔
12
      -- TODO there is a bug here because we shouldn't need to cast to lengths,
13
      -- somebody is setting a height as a number value (test in Lua 5.1)
14
      -- return node[dim]
15
      return SU.cast("length", node[dim])
41,667✔
16
   end, nodes)
17,825✔
17
   return SU.max(SILE.types.length(0), pl.utils.unpack(dims))
53,475✔
18
end
19

20
local _dims = pl.Set({ "width", "height", "depth" })
236✔
21

22
nodetypes.box = pl.class()
236✔
23
nodetypes.box.type = "special"
118✔
24

25
nodetypes.box.height = nil
118✔
26
nodetypes.box.depth = nil
118✔
27
nodetypes.box.width = nil
118✔
28
nodetypes.box.misfit = false
118✔
29
nodetypes.box.explicit = false
118✔
30
nodetypes.box.discardable = false
118✔
31
nodetypes.box.value = nil
118✔
32
nodetypes.box._default_length = "width"
118✔
33

34
function nodetypes.box:_init (spec)
236✔
35
   if
36
      type(spec) == "string"
29,119✔
37
      or type(spec) == "number"
28,275✔
38
      or SU.type(spec) == "measurement"
54,936✔
39
      or SU.type(spec) == "length"
54,700✔
40
   then
41
      self[self._default_length] = SU.cast("length", spec)
10,096✔
42
   elseif SU.type(spec) == "table" then
48,142✔
43
      if spec._tospec then
19,461✔
44
         spec = spec:_tospec()
×
45
      end
46
      for k, v in pairs(spec) do
103,832✔
47
         self[k] = _dims[k] and SU.cast("length", v) or v
109,017✔
48
      end
49
   elseif type(spec) ~= "nil" and SU.type(spec) ~= self.type then
4,610✔
50
      SU.error("Unimplemented, creating " .. self.type .. " node from " .. SU.type(spec), 1)
×
51
   end
52
   for dim in pairs(_dims) do
116,476✔
53
      if not self[dim] then
88,704✔
54
         self[dim] = SILE.types.length()
115,326✔
55
      end
56
   end
57
   self["is_" .. self.type] = true
29,119✔
58
   self.is_box = self.is_hbox or self.is_vbox or self.is_zerohbox or self.is_alternative or self.is_nnode
31,364✔
59
   self.is_zero = self.is_zerohbox or self.is_zerovglue
30,017✔
60
   if self.is_migrating then
29,568✔
61
      self.is_hbox, self.is_box = true, true
22✔
62
   end
63
end
64

65
-- De-init instances by shallow copying properties and removing meta table
66
function nodetypes.box:_tospec ()
236✔
67
   return pl.tablex.copy(self)
364✔
68
end
69

70
function nodetypes.box:tostring ()
236✔
71
   return self:__tostring()
×
72
end
73

74
function nodetypes.box:__tostring ()
236✔
75
   return self.type
×
76
end
77

78
function nodetypes.box.__concat (a, b)
236✔
79
   return tostring(a) .. tostring(b)
×
80
end
81

82
function nodetypes.box:absolute ()
236✔
83
   local clone = nodetypes[self.type](self:_tospec())
728✔
84
   for dim in pairs(_dims) do
1,456✔
85
      clone[dim] = self[dim]:absolute()
2,184✔
86
   end
87
   if self.nodes then
364✔
88
      clone.nodes = pl.tablex.map_named_method("absolute", self.nodes)
8✔
89
   end
90
   return clone
364✔
91
end
92

93
function nodetypes.box:lineContribution ()
236✔
94
   -- Regardless of the orientations, "width" is always in the
95
   -- writingDirection, and "height" is always in the "pageDirection"
96
   return self.misfit and self.height or self.width
17,828✔
97
end
98

99
function nodetypes.box:outputYourself ()
236✔
100
   SU.error(self.type .. " with no output routine")
×
101
end
102

103
function nodetypes.box:toText ()
236✔
104
   return self.type
×
105
end
106

107
function nodetypes.box:isBox ()
236✔
108
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
109
   return self.type == "hbox"
×
110
      or self.type == "zerohbox"
×
111
      or self.type == "alternative"
×
112
      or self.type == "nnode"
×
113
      or self.type == "vbox"
×
114
end
115

116
function nodetypes.box:isNnode ()
236✔
117
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
118
   return self.type == "nnode"
×
119
end
120

121
function nodetypes.box:isGlue ()
236✔
122
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
123
   return self.type == "glue"
×
124
end
125

126
function nodetypes.box:isVglue ()
236✔
127
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
128
   return self.type == "vglue"
×
129
end
130

131
function nodetypes.box:isZero ()
236✔
132
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
133
   return self.type == "zerohbox" or self.type == "zerovglue"
×
134
end
135

136
function nodetypes.box:isUnshaped ()
236✔
137
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
138
   return self.type == "unshaped"
×
139
end
140

141
function nodetypes.box:isAlternative ()
236✔
142
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
143
   return self.type == "alternative"
×
144
end
145

146
function nodetypes.box:isVbox ()
236✔
147
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
148
   return self.type == "vbox"
×
149
end
150

151
function nodetypes.box:isInsertion ()
236✔
152
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
153
   return self.type == "insertion"
×
154
end
155

156
function nodetypes.box:isMigrating ()
236✔
157
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
158
   return self.migrating
×
159
end
160

161
function nodetypes.box:isPenalty ()
236✔
162
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
163
   return self.type == "penalty"
×
164
end
165

166
function nodetypes.box:isDiscretionary ()
236✔
167
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
168
   return self.type == "discretionary"
×
169
end
170

171
function nodetypes.box:isKern ()
236✔
172
   SU.warn("Deprecated function, please use boolean is_<type> property to check types", true)
×
173
   return self.type == "kern"
×
174
end
175

176
nodetypes.hbox = pl.class(nodetypes.box)
236✔
177
nodetypes.hbox.type = "hbox"
118✔
178

179
function nodetypes.hbox:_init (spec)
236✔
180
   nodetypes.box._init(self, spec)
21,493✔
181
end
182

183
function nodetypes.hbox:__tostring ()
236✔
184
   return "H<" .. tostring(self.width) .. ">^" .. tostring(self.height) .. "-" .. tostring(self.depth) .. "v"
×
185
end
186

187
function nodetypes.hbox:scaledWidth (line)
236✔
188
   return SU.rationWidth(self:lineContribution(), self.width, line.ratio)
10,256✔
189
end
190

191
function nodetypes.hbox:outputYourself (typesetter, line)
236✔
192
   local outputWidth = self:scaledWidth(line)
5,114✔
193
   if not self.value.glyphString then
5,114✔
194
      return
1,455✔
195
   end
196
   if typesetter.frame:writingDirection() == "RTL" then
7,318✔
197
      typesetter.frame:advanceWritingDirection(outputWidth)
115✔
198
   end
199
   SILE.outputter:setCursor(typesetter.frame.state.cursorX, typesetter.frame.state.cursorY)
3,659✔
200
   SILE.outputter:setFont(self.value.options)
3,659✔
201
   SILE.outputter:drawHbox(self.value, outputWidth)
3,659✔
202
   if typesetter.frame:writingDirection() ~= "RTL" then
7,318✔
203
      typesetter.frame:advanceWritingDirection(outputWidth)
3,544✔
204
   end
205
end
206

207
nodetypes.zerohbox = pl.class(nodetypes.hbox)
236✔
208
nodetypes.zerohbox.type = "zerohbox"
118✔
209
nodetypes.zerohbox.value = { glyph = 0 }
118✔
210

211
nodetypes.nnode = pl.class(nodetypes.hbox)
236✔
212
nodetypes.nnode.type = "nnode"
118✔
213
nodetypes.nnode.language = ""
118✔
214
nodetypes.nnode.pal = nil
118✔
215
nodetypes.nnode.nodes = {}
118✔
216

217
function nodetypes.nnode:_init (spec)
236✔
218
   self:super(spec)
8,214✔
219
   if 0 == self.depth:tonumber() then
16,428✔
220
      self.depth = _maxnode(self.nodes, "depth")
16,428✔
221
   end
222
   if 0 == self.height:tonumber() then
16,428✔
223
      self.height = _maxnode(self.nodes, "height")
16,428✔
224
   end
225
   if 0 == self.width:tonumber() then
16,428✔
226
      self.width = SU.sum(SU.map(function (node)
24,642✔
227
         return node.width
7,765✔
228
      end, self.nodes))
16,428✔
229
   end
230
end
231

232
function nodetypes.nnode:__tostring ()
236✔
233
   return "N<"
×
234
      .. tostring(self.width)
×
235
      .. ">^"
×
236
      .. tostring(self.height)
×
237
      .. "-"
×
238
      .. tostring(self.depth)
×
239
      .. "v("
×
240
      .. self:toText()
×
241
      .. ")"
×
242
end
243

244
function nodetypes.nnode:absolute ()
236✔
245
   return self
2✔
246
end
247

248
function nodetypes.nnode:outputYourself (typesetter, line)
236✔
249
   -- See typesetter:computeLineRatio() which implements the currently rather messy
250
   -- and probably slightly dubious 'hyphenated' logic.
251
   -- Example: consider the word "out-put".
252
   -- The node queue therefore contains N(out)D(-)N(put) all pointing to the same
253
   -- parent N(output).
254
   if self.parent and not self.parent.hyphenated then
5,607✔
255
      -- When we hit N(out) and are not hyphenated, we output N(output) directly
256
      -- and mark it as used, so as to skip D(-)N(put) afterwards.
257
      -- I guess this was done to ensure proper kerning (vs. outputting each of
258
      -- the nodes separately).
259
      if not self.parent.used then
1,948✔
260
         self.parent:outputYourself(typesetter, line)
816✔
261
      end
262
      self.parent.used = true
1,948✔
263
   else
264
      -- It's possible not to have a parent, e.g. N(word) without hyphenation points.
265
      -- Either that, or we have a hyphenated parent but are in the case we are
266
      -- outputting one of the elements e.g. N(out)D(-) [line break] N(put).
267
      -- (ignoring the G(margin) nodes and potentially zerohbox nodes also on either side of the line break)
268
      for _, node in ipairs(self.nodes) do
7,318✔
269
         node:outputYourself(typesetter, line)
3,659✔
270
      end
271
   end
272
end
273

274
function nodetypes.nnode:toText ()
236✔
275
   return self.text
2✔
276
end
277

278
nodetypes.unshaped = pl.class(nodetypes.nnode)
236✔
279
nodetypes.unshaped.type = "unshaped"
118✔
280

281
function nodetypes.unshaped:_init (spec)
236✔
282
   self:super(spec)
449✔
283
   self.width = nil
449✔
284
end
285

286
function nodetypes.unshaped:__tostring ()
236✔
287
   return "U(" .. self:toText() .. ")"
×
288
end
289

290
getmetatable(nodetypes.unshaped).__index = function (_, _)
118✔
291
   -- if k == "width" then SU.error("Can't get width of unshaped node", true) end
292
   -- TODO: No idea why porting to proper Penlight classes this ^^^^^^ started
293
   -- killing everything. Perhaps because this function started working and would
294
   -- actually need to return rawget(self, k) or something?
295
end
296

297
function nodetypes.unshaped:shape ()
236✔
298
   local node = SILE.shaper:createNnodes(self.text, self.options)
422✔
299
   for i = 1, #node do
6,824✔
300
      node[i].parent = self.parent
12,804✔
301
   end
302
   return node
422✔
303
end
304

305
function nodetypes.unshaped.outputYourself (_)
236✔
306
   SU.error("An unshaped node made it to output", true)
×
307
end
308

309
nodetypes.discretionary = pl.class(nodetypes.hbox)
236✔
310

311
nodetypes.discretionary.type = "discretionary"
118✔
312
nodetypes.discretionary.prebreak = {}
118✔
313
nodetypes.discretionary.postbreak = {}
118✔
314
nodetypes.discretionary.replacement = {}
118✔
315
nodetypes.discretionary.used = false
118✔
316

317
function nodetypes.discretionary:__tostring ()
236✔
318
   return "D("
×
319
      .. SU.concat(self.prebreak, "")
×
320
      .. "|"
×
321
      .. SU.concat(self.postbreak, "")
×
322
      .. "|"
×
323
      .. SU.concat(self.replacement, "")
×
324
      .. ")"
×
325
end
326

327
function nodetypes.discretionary:toText ()
236✔
328
   return self.used and "-" or "_"
×
329
end
330

331
function nodetypes.discretionary:markAsPrebreak ()
236✔
332
   self.used = true
68✔
333
   if self.parent then
68✔
334
      self.parent.hyphenated = true
68✔
335
   end
336
   self.is_prebreak = true
68✔
337
end
338

339
function nodetypes.discretionary:cloneAsPostbreak ()
236✔
340
   if not self.used then
68✔
341
      SU.error("Cannot clone a non-used discretionary (previously marked as prebreak)")
×
342
   end
343
   return SILE.types.node.discretionary({
68✔
344
      prebreak = self.prebreak,
68✔
345
      postbreak = self.postbreak,
68✔
346
      replacement = self.replacement,
68✔
347
      parent = self.parent,
68✔
348
      used = true,
349
      is_prebreak = false,
350
   })
68✔
351
end
352

353
function nodetypes.discretionary:outputYourself (typesetter, line)
236✔
354
   -- See typesetter:computeLineRatio() which implements the currently rather
355
   -- messy hyphenated checks.
356
   -- Example: consider the word "out-put-ter".
357
   -- The node queue contains N(out)D(-)N(put)D(-)N(ter) all pointing to the same
358
   -- parent N(output), and here we hit D(-)
359

360
   -- Non-hyphenated parent: when N(out) was hit, we went for outputting
361
   -- the whole parent, so all other elements must now be skipped.
362
   if self.parent and not self.parent.hyphenated then
1,315✔
363
      return
1,132✔
364
   end
365

366
   -- It's possible not to have a parent (e.g. on a discretionary directly
367
   -- added in the queue and not coming from the hyphenator logic).
368
   -- Eiher that, or we have a hyphenated parent.
369
   if self.used then
183✔
370
      -- This is the actual hyphenation point.
371
      if self.is_prebreak then
136✔
372
         for _, node in ipairs(self.prebreak) do
136✔
373
            node:outputYourself(typesetter, line)
68✔
374
         end
375
      else
376
         for _, node in ipairs(self.postbreak) do
68✔
UNCOV
377
            node:outputYourself(typesetter, line)
×
378
         end
379
      end
380
   else
381
      -- This is not the hyphenation point (but another discretionary in the queue)
382
      -- E.g. we were in the case where we have N(out)D(-) [line break] N(out)D(-)N(ter)
383
      -- and now hit the second D(-).
384
      -- Unused discretionaries are obviously replaced.
385
      for _, node in ipairs(self.replacement) do
48✔
386
         node:outputYourself(typesetter, line)
1✔
387
      end
388
   end
389
end
390

391
function nodetypes.discretionary:prebreakWidth ()
236✔
392
   if self.prebw then
2,830✔
393
      return self.prebw
1,583✔
394
   end
395
   self.prebw = SILE.types.length()
2,494✔
396
   for _, node in ipairs(self.prebreak) do
2,494✔
397
      self.prebw:___add(node.width)
1,247✔
398
   end
399
   return self.prebw
1,247✔
400
end
401

402
function nodetypes.discretionary:postbreakWidth ()
236✔
403
   if self.postbw then
272✔
404
      return self.postbw
1✔
405
   end
406
   self.postbw = SILE.types.length()
542✔
407
   for _, node in ipairs(self.postbreak) do
271✔
UNCOV
408
      self.postbw:___add(node.width)
×
409
   end
410
   return self.postbw
271✔
411
end
412

413
function nodetypes.discretionary:replacementWidth ()
236✔
414
   if self.replacew then
1,530✔
415
      return self.replacew
283✔
416
   end
417
   self.replacew = SILE.types.length()
2,494✔
418
   for _, node in ipairs(self.replacement) do
1,250✔
419
      self.replacew:___add(node.width)
3✔
420
   end
421
   return self.replacew
1,247✔
422
end
423

424
function nodetypes.discretionary:prebreakHeight ()
236✔
425
   if self.prebh then
68✔
426
      return self.prebh
×
427
   end
428
   self.prebh = _maxnode(self.prebreak, "height")
136✔
429
   return self.prebh
68✔
430
end
431

432
function nodetypes.discretionary:postbreakHeight ()
236✔
433
   if self.postbh then
68✔
434
      return self.postbh
×
435
   end
436
   self.postbh = _maxnode(self.postbreak, "height")
136✔
437
   return self.postbh
68✔
438
end
439

440
function nodetypes.discretionary:replacementHeight ()
236✔
441
   if self.replaceh then
47✔
442
      return self.replaceh
×
443
   end
444
   self.replaceh = _maxnode(self.replacement, "height")
94✔
445
   return self.replaceh
47✔
446
end
447

448
function nodetypes.discretionary:replacementDepth ()
236✔
UNCOV
449
   if self.replaced then
×
450
      return self.replaced
×
451
   end
UNCOV
452
   self.replaced = _maxnode(self.replacement, "depth")
×
UNCOV
453
   return self.replaced
×
454
end
455

456
nodetypes.alternative = pl.class(nodetypes.hbox)
236✔
457

458
nodetypes.alternative.type = "alternative"
118✔
459
nodetypes.alternative.options = {}
118✔
460
nodetypes.alternative.selected = nil
118✔
461

462
function nodetypes.alternative:__tostring ()
236✔
463
   return "A(" .. SU.concat(self.options, " / ") .. ")"
×
464
end
465

466
function nodetypes.alternative:minWidth ()
236✔
467
   local minW = function (a, b)
468
      return SU.min(a.width, b.width)
×
469
   end
470
   return pl.tablex.reduce(minW, self.options)
×
471
end
472

473
function nodetypes.alternative:deltas ()
236✔
474
   local minWidth = self:minWidth()
×
475
   local rv = {}
×
476
   for i = 1, #self.options do
×
477
      rv[#rv + 1] = self.options[i].width - minWidth
×
478
   end
479
   return rv
×
480
end
481

482
function nodetypes.alternative:outputYourself (typesetter, line)
236✔
483
   if self.selected then
×
484
      self.options[self.selected]:outputYourself(typesetter, line)
×
485
   end
486
end
487

488
nodetypes.glue = pl.class(nodetypes.box)
236✔
489
nodetypes.glue.type = "glue"
118✔
490
nodetypes.glue.discardable = true
118✔
491

492
function nodetypes.glue:__tostring ()
236✔
493
   return (self.explicit and "E:" or "") .. "G<" .. tostring(self.width) .. ">"
×
494
end
495

496
function nodetypes.glue.toText (_)
236✔
497
   return " "
×
498
end
499

500
function nodetypes.glue:outputYourself (typesetter, line)
236✔
501
   local outputWidth = SU.rationWidth(self.width:absolute(), self.width:absolute(), line.ratio)
14,073✔
502
   typesetter.frame:advanceWritingDirection(outputWidth)
4,691✔
503
end
504

505
-- A hfillglue is just a glue with infinite stretch.
506
-- (Convenience so callers do not have to know what infinity is.)
507
nodetypes.hfillglue = pl.class(nodetypes.glue)
236✔
508
function nodetypes.hfillglue:_init (spec)
236✔
509
   self:super(spec)
178✔
510
   self.width = SILE.types.length(self.width.length, infinity, self.width.shrink)
356✔
511
end
512

513
-- A hssglue is just a glue with infinite stretch and shrink.
514
-- (Convenience so callers do not have to know what infinity is.)
515
nodetypes.hssglue = pl.class(nodetypes.glue)
236✔
516
function nodetypes.hssglue:_init (spec)
236✔
517
   self:super(spec)
×
518
   self.width = SILE.types.length(self.width.length, infinity, infinity)
×
519
end
520

521
nodetypes.kern = pl.class(nodetypes.glue)
236✔
522
nodetypes.kern.type = "kern" -- Perhaps some smell here, see comment on vkern
118✔
523
nodetypes.kern.discardable = false
118✔
524

525
function nodetypes.kern:__tostring ()
236✔
526
   return "K<" .. tostring(self.width) .. ">"
×
527
end
528

529
nodetypes.vglue = pl.class(nodetypes.box)
236✔
530
nodetypes.vglue.type = "vglue"
118✔
531
nodetypes.vglue.discardable = true
118✔
532
nodetypes.vglue._default_length = "height"
118✔
533
nodetypes.vglue.adjustment = nil
118✔
534

535
function nodetypes.vglue:_init (spec)
236✔
536
   self.adjustment = SILE.types.measurement()
2,624✔
537
   self:super(spec)
1,312✔
538
end
539

540
function nodetypes.vglue:__tostring ()
236✔
541
   return (self.explicit and "E:" or "") .. "VG<" .. tostring(self.height) .. ">"
×
542
end
543

544
function nodetypes.vglue:adjustGlue (adjustment)
236✔
545
   self.adjustment = adjustment
748✔
546
end
547

548
function nodetypes.vglue:outputYourself (typesetter, line)
236✔
549
   typesetter.frame:advancePageDirection(line.height:absolute() + line.depth:absolute() + self.adjustment)
4,685✔
550
end
551

552
function nodetypes.vglue:unbox ()
236✔
553
   return { self }
11✔
554
end
555

556
-- A vfillglue is just a vglue with infinite stretch.
557
-- (Convenience so callers do not have to know what infinity is.)
558
nodetypes.vfillglue = pl.class(nodetypes.vglue)
236✔
559
function nodetypes.vfillglue:_init (spec)
236✔
560
   self:super(spec)
133✔
561
   self.height = SILE.types.length(self.width.length, infinity, self.width.shrink)
266✔
562
end
563

564
-- A vssglue is just a vglue with infinite stretch and shrink.
565
-- (Convenience so callers do not have to know what infinity is.)
566
nodetypes.vssglue = pl.class(nodetypes.vglue)
236✔
567
function nodetypes.vssglue:_init (spec)
236✔
568
   self:super(spec)
×
569
   self.height = SILE.types.length(self.width.length, infinity, infinity)
×
570
end
571

572
nodetypes.zerovglue = pl.class(nodetypes.vglue)
236✔
573

574
nodetypes.vkern = pl.class(nodetypes.vglue)
236✔
575
-- FIXME TODO
576
-- Here we cannot do:
577
--   nodetypes.vkern.type = "vkern"
578
-- It cannot be typed as "vkern" as the pagebuilder doesn't check is_vkern.
579
-- So it's just a vglue currrenty, marked as not discardable...
580
-- But on the other hand, nodetypes.kern is typed "kern" and is not a glue...
581
-- Frankly, the discardable/explicit flags and the types are too
582
-- entangled and point towards a more general design issue.
583
-- N.B. this vkern node is only used in the linespacing package so far.
584
nodetypes.vkern.discardable = false
118✔
585

586
function nodetypes.vkern:__tostring ()
236✔
587
   return "VK<" .. tostring(self.height) .. ">"
×
588
end
589

590
nodetypes.penalty = pl.class(nodetypes.box)
236✔
591
nodetypes.penalty.type = "penalty"
118✔
592
nodetypes.penalty.discardable = true
118✔
593
nodetypes.penalty.penalty = 0
118✔
594

595
function nodetypes.penalty:_init (spec)
236✔
596
   self:super(spec)
915✔
597
   if type(spec) ~= "table" then
915✔
598
      self.penalty = SU.cast("number", spec)
1,598✔
599
   end
600
end
601

602
function nodetypes.penalty:__tostring ()
236✔
603
   return "P(" .. tostring(self.penalty) .. ")"
×
604
end
605

606
function nodetypes.penalty.outputYourself (_) end
604✔
607

608
function nodetypes.penalty.toText (_)
236✔
609
   return "(!)"
×
610
end
611

612
function nodetypes.penalty:unbox ()
236✔
613
   return { self }
×
614
end
615

616
nodetypes.vbox = pl.class(nodetypes.box)
236✔
617
nodetypes.vbox.type = "vbox"
118✔
618
nodetypes.vbox.nodes = {}
118✔
619
nodetypes.vbox._default_length = "height"
118✔
620

621
function nodetypes.vbox:_init (spec)
236✔
622
   self.nodes = {}
607✔
623
   self:super(spec)
607✔
624
   self.depth = _maxnode(self.nodes, "depth")
1,214✔
625
   self.height = _maxnode(self.nodes, "height")
1,214✔
626
end
627

628
function nodetypes.vbox:__tostring ()
236✔
629
   return "VB<" .. tostring(self.height) .. "|" .. self:toText() .. "v" .. tostring(self.depth) .. ")"
×
630
end
631

632
function nodetypes.vbox:toText ()
236✔
633
   return "VB["
×
634
      .. SU.concat(
×
635
         SU.map(function (node)
×
636
            return node:toText()
×
637
         end, self.nodes),
×
638
         ""
639
      )
640
      .. "]"
×
641
end
642

643
function nodetypes.vbox:outputYourself (typesetter, line)
236✔
644
   typesetter.frame:advancePageDirection(self.height)
560✔
645
   local initial = true
560✔
646
   for _, node in ipairs(self.nodes) do
13,220✔
647
      if not (initial and (node.is_glue or node.is_penalty)) then
12,660✔
648
         initial = false
12,660✔
649
         node:outputYourself(typesetter, line)
12,660✔
650
      end
651
   end
652
   typesetter.frame:advancePageDirection(self.depth)
560✔
653
   typesetter.frame:newLine()
560✔
654
end
655

656
function nodetypes.vbox:unbox ()
236✔
657
   for i = 1, #self.nodes do
17✔
658
      if self.nodes[i].is_vbox or self.nodes[i].is_vglue then
17✔
659
         return self.nodes
17✔
660
      end
661
   end
662
   return { self }
×
663
end
664

665
function nodetypes.vbox:append (box)
236✔
666
   local nodes = box
42✔
667
   if not box then
42✔
668
      SU.error("nil box given", true)
×
669
   end
670
   if nodes.type then
42✔
671
      nodes = box:unbox()
52✔
672
   end
673
   self.height = self.height:absolute()
84✔
674
   self.height:___add(self.depth)
42✔
675
   local lastdepth = SILE.types.length()
42✔
676
   for i = 1, #nodes do
370✔
677
      table.insert(self.nodes, nodes[i])
328✔
678
      self.height:___add(nodes[i].height)
328✔
679
      self.height:___add(nodes[i].depth:absolute())
656✔
680
      if nodes[i].is_vbox then
328✔
681
         lastdepth = nodes[i].depth
135✔
682
      end
683
   end
684
   self.height:___sub(lastdepth)
42✔
685
   self.ratio = 1
42✔
686
   self.depth = lastdepth
42✔
687
end
688

689
nodetypes.migrating = pl.class(nodetypes.hbox)
236✔
690
nodetypes.migrating.type = "migrating"
118✔
691
nodetypes.migrating.material = {}
118✔
692
nodetypes.migrating.value = {}
118✔
693
nodetypes.migrating.nodes = {}
118✔
694

695
function nodetypes.migrating:__tostring ()
236✔
696
   return "<M: " .. tostring(self.material) .. ">"
×
697
end
698

699
local _deprecated_nodefactory = {
118✔
700
   newHbox = true,
701
   newNnode = true,
702
   newUnshaped = true,
703
   newDisc = true,
704
   disc = true,
705
   newAlternative = true,
706
   newGlue = true,
707
   newKern = true,
708
   newVglue = true,
709
   newVKern = true,
710
   newPenalty = true,
711
   newDiscretionary = true,
712
   newVbox = true,
713
   newMigrating = true,
714
   zeroGlue = true,
715
   hfillGlue = true,
716
   vfillGlue = true,
717
   hssGlue = true,
718
   vssGlue = true,
719
   zeroHbox = true,
720
   zeroVglue = true,
721
}
722

723
setmetatable(nodetypes, {
236✔
724
   __index = function (_, prop)
725
      if _deprecated_nodefactory[prop] then
×
726
         SU.deprecated(
×
727
            "SILE.types.node." .. prop,
×
728
            "SILE.types.node." .. prop:match("n?e?w?(.*)"):lower(),
×
729
            "0.10.0",
730
            "0.14.0"
731
         )
732
      elseif type(prop) == "number" then -- luacheck: ignore 542
×
733
      -- Likely an attempt to iterate, inspect, or dump the table, sort of safe to ignore
734
      else
735
         SU.error("Attempt to access non-existent SILE.types.node." .. prop)
×
736
      end
737
   end,
738
})
739

740
return nodetypes
118✔
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

© 2025 Coveralls, Inc