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

sile-typesetter / sile / 9409557472

07 Jun 2024 12:09AM UTC coverage: 69.448% (-4.5%) from 73.988%
9409557472

push

github

alerque
fix(build): Distribute vendored compat-5.3.c source file

12025 of 17315 relevant lines covered (69.45%)

6023.46 hits per line

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

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

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

10
local function _maxnode (nodes, dim)
11
   local dims = SU.map(function (node)
46,060✔
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])
53,430✔
16
   end, nodes)
23,030✔
17
   return SU.max(SILE.types.length(0), pl.utils.unpack(dims))
69,090✔
18
end
19

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

22
nodetypes.box = pl.class()
312✔
23
nodetypes.box.type = "special"
156✔
24

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

34
function nodetypes.box:_init (spec)
312✔
35
   if
36
      type(spec) == "string"
37,383✔
37
      or type(spec) == "number"
36,274✔
38
      or SU.type(spec) == "measurement"
70,236✔
39
      or SU.type(spec) == "length"
69,936✔
40
   then
41
      self[self._default_length] = SU.cast("length", spec)
13,370✔
42
   elseif SU.type(spec) == "table" then
61,396✔
43
      if spec._tospec then
24,785✔
44
         spec = spec:_tospec()
×
45
      end
46
      for k, v in pairs(spec) do
131,507✔
47
         self[k] = _dims[k] and SU.cast("length", v) or v
137,945✔
48
      end
49
   elseif type(spec) ~= "nil" and SU.type(spec) ~= self.type then
5,913✔
50
      SU.error("Unimplemented, creating " .. self.type .. " node from " .. SU.type(spec), 1)
×
51
   end
52
   for dim in pairs(_dims) do
149,532✔
53
      if not self[dim] then
114,720✔
54
         self[dim] = SILE.types.length()
148,482✔
55
      end
56
   end
57
   self["is_" .. self.type] = true
37,383✔
58
   self.is_box = self.is_hbox or self.is_vbox or self.is_zerohbox or self.is_alternative or self.is_nnode
41,668✔
59
   self.is_zero = self.is_zerohbox or self.is_zerovglue
39,097✔
60
   if self.is_migrating then
38,240✔
61
      self.is_hbox, self.is_box = true, true
24✔
62
   end
63
end
64

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

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

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

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

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

93
function nodetypes.box:lineContribution ()
312✔
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
24,351✔
97
end
98

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

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

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

179
function nodetypes.hbox:_init (spec)
312✔
180
   nodetypes.box._init(self, spec)
27,014✔
181
end
182

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

187
function nodetypes.hbox:scaledWidth (line)
312✔
188
   return SU.rationWidth(self:lineContribution(), self.width, line.ratio)
14,744✔
189
end
190

191
function nodetypes.hbox:outputYourself (typesetter, line)
312✔
192
   local outputWidth = self:scaledWidth(line)
7,125✔
193
   if not self.value.glyphString then
7,125✔
194
      return
2,068✔
195
   end
196
   if typesetter.frame:writingDirection() == "RTL" then
10,114✔
197
      typesetter.frame:advanceWritingDirection(outputWidth)
115✔
198
   end
199
   SILE.outputter:setCursor(typesetter.frame.state.cursorX, typesetter.frame.state.cursorY)
5,057✔
200
   SILE.outputter:setFont(self.value.options)
5,057✔
201
   SILE.outputter:drawHbox(self.value, outputWidth)
5,057✔
202
   if typesetter.frame:writingDirection() ~= "RTL" then
10,114✔
203
      typesetter.frame:advanceWritingDirection(outputWidth)
4,942✔
204
   end
205
end
206

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

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

217
function nodetypes.nnode:_init (spec)
312✔
218
   self:super(spec)
10,546✔
219
   if 0 == self.depth:tonumber() then
21,092✔
220
      self.depth = _maxnode(self.nodes, "depth")
21,092✔
221
   end
222
   if 0 == self.height:tonumber() then
21,092✔
223
      self.height = _maxnode(self.nodes, "height")
21,092✔
224
   end
225
   if 0 == self.width:tonumber() then
21,092✔
226
      self.width = SU.sum(SU.map(function (node)
31,638✔
227
         return node.width
9,687✔
228
      end, self.nodes))
21,092✔
229
   end
230
end
231

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

248
function nodetypes.nnode:outputYourself (typesetter, line)
312✔
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
7,326✔
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
2,269✔
260
         self.parent:outputYourself(typesetter, line)
944✔
261
      end
262
      self.parent.used = true
2,269✔
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
10,114✔
269
         node:outputYourself(typesetter, line)
5,057✔
270
      end
271
   end
272
end
273

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

278
nodetypes.unshaped = pl.class(nodetypes.nnode)
312✔
279
nodetypes.unshaped.type = "unshaped"
156✔
280

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

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

290
getmetatable(nodetypes.unshaped).__index = function (_, _)
156✔
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 ()
312✔
298
   local node = SILE.shaper:createNnodes(self.text, self.options)
832✔
299
   for i = 1, #node do
9,148✔
300
      node[i].parent = self.parent
16,609✔
301
   end
302
   return node
832✔
303
end
304

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

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

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

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

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

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

353
function nodetypes.discretionary:outputYourself (typesetter, line)
312✔
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,583✔
363
      return
1,325✔
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
258✔
370
      -- This is the actual hyphenation point.
371
      if self.is_prebreak then
182✔
372
         for _, node in ipairs(self.prebreak) do
183✔
373
            node:outputYourself(typesetter, line)
92✔
374
         end
375
      else
376
         for _, node in ipairs(self.postbreak) do
92✔
377
            node:outputYourself(typesetter, line)
1✔
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
83✔
386
         node:outputYourself(typesetter, line)
7✔
387
      end
388
   end
389
end
390

391
function nodetypes.discretionary:prebreakWidth ()
312✔
392
   if self.prebw then
3,546✔
393
      return self.prebw
2,054✔
394
   end
395
   self.prebw = SILE.types.length()
2,984✔
396
   for _, node in ipairs(self.prebreak) do
2,988✔
397
      self.prebw:___add(node.width)
1,496✔
398
   end
399
   return self.prebw
1,492✔
400
end
401

402
function nodetypes.discretionary:postbreakWidth ()
312✔
403
   if self.postbw then
328✔
404
      return self.postbw
6✔
405
   end
406
   self.postbw = SILE.types.length()
644✔
407
   for _, node in ipairs(self.postbreak) do
324✔
408
      self.postbw:___add(node.width)
2✔
409
   end
410
   return self.postbw
322✔
411
end
412

413
function nodetypes.discretionary:replacementWidth ()
312✔
414
   if self.replacew then
1,928✔
415
      return self.replacew
436✔
416
   end
417
   self.replacew = SILE.types.length()
2,984✔
418
   for _, node in ipairs(self.replacement) do
1,504✔
419
      self.replacew:___add(node.width)
12✔
420
   end
421
   return self.replacew
1,492✔
422
end
423

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

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

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

448
function nodetypes.discretionary:replacementDepth ()
312✔
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)
312✔
457

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

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

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

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

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

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

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

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

535
function nodetypes.vglue:_init (spec)
312✔
536
   self.adjustment = SILE.measurement()
3,546✔
537
   self:super(spec)
1,773✔
538
end
539

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

544
function nodetypes.vglue:adjustGlue (adjustment)
312✔
545
   self.adjustment = adjustment
1,195✔
546
end
547

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

699
local _deprecated_nodefactory = {
156✔
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, {
312✔
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
156✔
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