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

sile-typesetter / sile / 9699312690

27 Jun 2024 03:22PM UTC coverage: 62.16% (-9.2%) from 71.339%
9699312690

push

github

web-flow
Merge ea2d501b6 into 8c5a2a2ba

10771 of 17328 relevant lines covered (62.16%)

2463.58 hits per line

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

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

4
local nodetypes = {}
90✔
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)
180✔
9

10
local function _maxnode (nodes, dim)
11
   local dims = SU.map(function (node)
18,274✔
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])
23,179✔
16
   end, nodes)
9,137✔
17
   return SU.max(SILE.types.length(0), pl.utils.unpack(dims))
27,411✔
18
end
19

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

22
nodetypes.box = pl.class()
180✔
23
nodetypes.box.type = "special"
90✔
24

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

34
function nodetypes.box:_init (spec)
180✔
35
   if
36
      type(spec) == "string"
16,870✔
37
      or type(spec) == "number"
16,219✔
38
      or SU.type(spec) == "measurement"
31,214✔
39
      or SU.type(spec) == "length"
30,996✔
40
   then
41
      self[self._default_length] = SU.cast("length", spec)
6,564✔
42
   elseif SU.type(spec) == "table" then
27,176✔
43
      if spec._tospec then
9,651✔
44
         spec = spec:_tospec()
×
45
      end
46
      for k, v in pairs(spec) do
51,360✔
47
         self[k] = _dims[k] and SU.cast("length", v) or v
54,045✔
48
      end
49
   elseif type(spec) ~= "nil" and SU.type(spec) ~= self.type then
3,937✔
50
      SU.error("Unimplemented, creating " .. self.type .. " node from " .. SU.type(spec), 1)
×
51
   end
52
   for dim in pairs(_dims) do
67,480✔
53
      if not self[dim] then
51,645✔
54
         self[dim] = SILE.types.length()
69,984✔
55
      end
56
   end
57
   self["is_" .. self.type] = true
16,870✔
58
   self.is_box = self.is_hbox or self.is_vbox or self.is_zerohbox or self.is_alternative or self.is_nnode
18,595✔
59
   self.is_zero = self.is_zerohbox or self.is_zerovglue
17,560✔
60
   if self.is_migrating then
17,215✔
61
      self.is_hbox, self.is_box = true, true
6✔
62
   end
63
end
64

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

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

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

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

82
function nodetypes.box:absolute ()
180✔
83
   local clone = nodetypes[self.type](self:_tospec())
546✔
84
   for dim in pairs(_dims) do
1,092✔
85
      clone[dim] = self[dim]:absolute()
1,638✔
86
   end
87
   if self.nodes then
273✔
88
      clone.nodes = pl.tablex.map_named_method("absolute", self.nodes)
×
89
   end
90
   return clone
273✔
91
end
92

93
function nodetypes.box:lineContribution ()
180✔
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
11,150✔
97
end
98

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

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

107
function nodetypes.box:isBox ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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 ()
180✔
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)
180✔
177
nodetypes.hbox.type = "hbox"
90✔
178

179
function nodetypes.hbox:_init (spec)
180✔
180
   nodetypes.box._init(self, spec)
11,618✔
181
end
182

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

187
function nodetypes.hbox:scaledWidth (line)
180✔
188
   return SU.rationWidth(self:lineContribution(), self.width, line.ratio)
6,480✔
189
end
190

191
function nodetypes.hbox:outputYourself (typesetter, line)
180✔
192
   local outputWidth = self:scaledWidth(line)
3,226✔
193
   if not self.value.glyphString then
3,226✔
194
      return
1,050✔
195
   end
196
   if typesetter.frame:writingDirection() == "RTL" then
4,352✔
197
      typesetter.frame:advanceWritingDirection(outputWidth)
115✔
198
   end
199
   SILE.outputter:setCursor(typesetter.frame.state.cursorX, typesetter.frame.state.cursorY)
2,176✔
200
   SILE.outputter:setFont(self.value.options)
2,176✔
201
   SILE.outputter:drawHbox(self.value, outputWidth)
2,176✔
202
   if typesetter.frame:writingDirection() ~= "RTL" then
4,352✔
203
      typesetter.frame:advanceWritingDirection(outputWidth)
2,061✔
204
   end
205
end
206

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

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

217
function nodetypes.nnode:_init (spec)
180✔
218
   self:super(spec)
4,119✔
219
   if 0 == self.depth:tonumber() then
8,238✔
220
      self.depth = _maxnode(self.nodes, "depth")
8,238✔
221
   end
222
   if 0 == self.height:tonumber() then
8,238✔
223
      self.height = _maxnode(self.nodes, "height")
8,238✔
224
   end
225
   if 0 == self.width:tonumber() then
8,238✔
226
      self.width = SU.sum(SU.map(function (node)
12,357✔
227
         return node.width
3,774✔
228
      end, self.nodes))
8,238✔
229
   end
230
end
231

232
function nodetypes.nnode:__tostring ()
180✔
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 ()
180✔
245
   return self
×
246
end
247

248
function nodetypes.nnode:outputYourself (typesetter, line)
180✔
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
2,921✔
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
745✔
260
         self.parent:outputYourself(typesetter, line)
313✔
261
      end
262
      self.parent.used = true
745✔
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
4,352✔
269
         node:outputYourself(typesetter, line)
2,176✔
270
      end
271
   end
272
end
273

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

278
nodetypes.unshaped = pl.class(nodetypes.nnode)
180✔
279
nodetypes.unshaped.type = "unshaped"
90✔
280

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

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

290
getmetatable(nodetypes.unshaped).__index = function (_, _)
90✔
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 ()
180✔
298
   local node = SILE.shaper:createNnodes(self.text, self.options)
318✔
299
   for i = 1, #node do
4,090✔
300
      node[i].parent = self.parent
7,544✔
301
   end
302
   return node
318✔
303
end
304

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

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

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

317
function nodetypes.discretionary:__tostring ()
180✔
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 ()
180✔
328
   return self.used and "-" or "_"
×
329
end
330

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

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

353
function nodetypes.discretionary:outputYourself (typesetter, line)
180✔
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
513✔
363
      return
432✔
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
81✔
370
      -- This is the actual hyphenation point.
371
      if self.is_prebreak then
56✔
372
         for _, node in ipairs(self.prebreak) do
56✔
373
            node:outputYourself(typesetter, line)
28✔
374
         end
375
      else
376
         for _, node in ipairs(self.postbreak) do
28✔
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
26✔
386
         node:outputYourself(typesetter, line)
1✔
387
      end
388
   end
389
end
390

391
function nodetypes.discretionary:prebreakWidth ()
180✔
392
   if self.prebw then
1,142✔
393
      return self.prebw
657✔
394
   end
395
   self.prebw = SILE.types.length()
970✔
396
   for _, node in ipairs(self.prebreak) do
970✔
397
      self.prebw:___add(node.width)
485✔
398
   end
399
   return self.prebw
485✔
400
end
401

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

413
function nodetypes.discretionary:replacementWidth ()
180✔
414
   if self.replacew then
622✔
415
      return self.replacew
137✔
416
   end
417
   self.replacew = SILE.types.length()
970✔
418
   for _, node in ipairs(self.replacement) do
488✔
419
      self.replacew:___add(node.width)
3✔
420
   end
421
   return self.replacew
485✔
422
end
423

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

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

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

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

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

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

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

466
function nodetypes.alternative:minWidth ()
180✔
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 ()
180✔
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)
180✔
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)
180✔
489
nodetypes.glue.type = "glue"
90✔
490
nodetypes.glue.discardable = true
90✔
491

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

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

500
function nodetypes.glue:outputYourself (typesetter, line)
180✔
501
   local outputWidth = SU.rationWidth(self.width:absolute(), self.width:absolute(), line.ratio)
8,913✔
502
   typesetter.frame:advanceWritingDirection(outputWidth)
2,971✔
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)
180✔
508
function nodetypes.hfillglue:_init (spec)
180✔
509
   self:super(spec)
160✔
510
   self.width = SILE.types.length(self.width.length, infinity, self.width.shrink)
320✔
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)
180✔
516
function nodetypes.hssglue:_init (spec)
180✔
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)
180✔
522
nodetypes.kern.type = "kern" -- Perhaps some smell here, see comment on vkern
90✔
523
nodetypes.kern.discardable = false
90✔
524

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

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

535
function nodetypes.vglue:_init (spec)
180✔
536
   self.adjustment = SILE.types.measurement()
1,922✔
537
   self:super(spec)
961✔
538
end
539

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

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

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

552
function nodetypes.vglue:unbox ()
180✔
553
   return { self }
4✔
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)
180✔
559
function nodetypes.vfillglue:_init (spec)
180✔
560
   self:super(spec)
107✔
561
   self.height = SILE.types.length(self.width.length, infinity, self.width.shrink)
214✔
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)
180✔
567
function nodetypes.vssglue:_init (spec)
180✔
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)
180✔
573

574
nodetypes.vkern = pl.class(nodetypes.vglue)
180✔
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
90✔
585

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

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

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

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

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

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

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

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

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

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

632
function nodetypes.vbox:toText ()
180✔
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)
180✔
644
   typesetter.frame:advancePageDirection(self.height)
389✔
645
   local initial = true
389✔
646
   for _, node in ipairs(self.nodes) do
7,887✔
647
      if not (initial and (node.is_glue or node.is_penalty)) then
7,498✔
648
         initial = false
7,498✔
649
         node:outputYourself(typesetter, line)
7,498✔
650
      end
651
   end
652
   typesetter.frame:advancePageDirection(self.depth)
389✔
653
   typesetter.frame:newLine()
389✔
654
end
655

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

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

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

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

699
local _deprecated_nodefactory = {
90✔
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, {
180✔
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
90✔
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