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

sile-typesetter / sile / 9324025270

31 May 2024 08:35PM UTC coverage: 66.168% (-8.0%) from 74.124%
9324025270

push

github

web-flow
Merge 235329972 into 70ff5c335

1753 of 2583 new or added lines in 108 files covered. (67.87%)

1498 existing lines in 74 files now uncovered.

11404 of 17235 relevant lines covered (66.17%)

2956.13 hits per line

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

83.67
/pagebuilders/base.lua
1
--- SILE pagebuilder class.
2
-- @interfaces pagebuilders
3

4
local pagebuilder = pl.class()
48✔
5
pagebuilder.type = "pagebuilder"
48✔
6
pagebuilder._name = "base"
48✔
7

8
function pagebuilder:_init ()
48✔
9
   self.awful_bad = 1073741823
49✔
10
   self.inf_bad = 10000
49✔
11
   self.eject_penalty = -self.inf_bad
49✔
12
   self.deplorable = 100000
49✔
13
end
14

15
function pagebuilder.collateVboxes (_, vboxlist)
48✔
16
   local output = SILE.types.node.vbox()
4✔
17
   output:append(vboxlist)
4✔
18
   return output
4✔
19
end
20

21
-- Note: Almost 1/3 of the time in a typical SILE in taken iterating through
22
-- this function. As a result there are some micro-optimizations here that
23
-- make it a-typical of preferred coding styles. In particular note that
24
-- we absolutize heavily iterated lengths as early as possible and make
25
-- make direct calls to their integer amounts, assumed to be in points by
26
-- the point they are called **without actually checking**!
27
function pagebuilder:findBestBreak (options)
48✔
28
   local vboxlist = SU.required(options, "vboxlist", "in findBestBreak")
530✔
29
   local target = SU.required(options, "target", "in findBestBreak", "length")
530✔
30
   local restart = options.restart or false
530✔
31
   local force = options.force or false
530✔
32
   local i = 0
530✔
33
   local totalHeight = SILE.types.length()
530✔
34
   local bestBreak = nil
530✔
35
   local started = false
530✔
36
   if restart and restart.target == target then
530✔
37
      totalHeight = restart.totalHeight
×
38
      i = restart.i
×
39
      started = restart.started
×
40
   end
41
   local leastC = self.inf_bad
530✔
42
   SU.debug("pagebuilder", function ()
1,060✔
43
      return "Page builder for frame "
×
44
         .. SILE.typesetter.frame.id
×
45
         .. " called with "
×
46
         .. #vboxlist
×
47
         .. " nodes, "
×
48
         .. tostring(target)
×
49
   end)
50
   if SU.debugging("vboxes") then
1,060✔
51
      for j, box in ipairs(vboxlist) do
×
52
         SU.debug("vboxes", function ()
×
53
            return (j == i and " >" or "  ") .. j .. ": " .. box
×
54
         end)
55
      end
56
   end
57
   while not started and i < #vboxlist do
1,345✔
58
      i = i + 1
1,335✔
59
      if not vboxlist[i].is_vglue then
1,335✔
60
         started = true
520✔
61
         i = i - 1
520✔
62
         break
520✔
63
      end
64
   end
65
   local pi
66
   while i < #vboxlist do
7,684✔
67
      i = i + 1
7,237✔
68
      local vbox = vboxlist[i]
7,237✔
69
      SU.debug("pagebuilder", "Dealing with VBox", vbox)
7,237✔
70
      if vbox.is_vbox then
7,237✔
71
         totalHeight:___add(vbox.height)
2,195✔
72
         totalHeight:___add(vbox.depth)
4,390✔
73
      elseif vbox.is_vglue then
5,042✔
74
         totalHeight:___add(vbox.height)
9,308✔
75
      elseif vbox.is_insertion then
388✔
76
         -- TODO: refactor as hook and without side effects!
77
         target = SILE.insertions.processInsertion(vboxlist, i, totalHeight, target)
24✔
78
         vbox = vboxlist[i]
12✔
79
      end
80
      local left = target - totalHeight
7,237✔
81
      SU.debug("pagebuilder", "I have", left, "left")
7,237✔
82
      -- if left < -20 then SU.error("\nCatastrophic page breaking failure!"); end
83
      pi = 0
7,237✔
84
      if vbox.is_penalty then
7,237✔
85
         pi = vbox.penalty
377✔
86
      end
87
      if
88
         vbox.is_penalty and vbox.penalty < self.inf_bad
7,237✔
89
         or (vbox.is_vglue and i > 1 and not vboxlist[i - 1].discardable)
6,894✔
90
      then
91
         local badness
92
         SU.debug("pagebuilder", "totalHeight", totalHeight, "with target", target)
2,951✔
93
         if totalHeight.length.amount < target.length.amount then -- TeX #1039
2,951✔
94
            -- Account for infinite stretch?
95
            badness = SU.rateBadness(self.inf_bad, left.length.amount, totalHeight.stretch.amount)
5,864✔
96
         elseif left.length.amount < totalHeight.shrink.amount then
19✔
97
            badness = self.awful_bad
18✔
98
         else
99
            badness = SU.rateBadness(self.inf_bad, -left.length.amount, totalHeight.shrink.amount)
2✔
100
         end
101

102
         local c
103
         if badness < self.awful_bad then
2,951✔
104
            if pi <= self.eject_penalty then
2,933✔
105
               c = pi
65✔
106
            elseif badness < self.inf_bad then
2,868✔
107
               c = badness + pi -- plus insert
189✔
108
            else
109
               c = self.deplorable
2,679✔
110
            end
111
         else
112
            c = badness
18✔
113
         end
114
         if c < leastC then
2,951✔
115
            leastC = c
224✔
116
            bestBreak = i
224✔
117
         else
118
            restart = { totalHeight = totalHeight, i = i, started = started, target = target }
2,727✔
119
         end
120

121
         SU.debug("pagebuilder", "Badness:", c)
2,951✔
122
         if c == self.awful_bad or pi <= self.eject_penalty then
2,951✔
123
            SU.debug("pagebuilder", "outputting")
83✔
124
            local onepage = {}
83✔
125
            if not bestBreak then
83✔
126
               bestBreak = i
16✔
127
            end
128
            for j = 1, bestBreak do
1,316✔
129
               onepage[j] = table.remove(vboxlist, 1)
2,466✔
130
            end
131
            while #onepage > 1 and onepage[#onepage].discardable do
173✔
132
               onepage[#onepage] = nil
90✔
133
            end
134
            return onepage, pi
83✔
135
         end
136
      end
137
   end
138
   SU.debug("pagebuilder", "No page break here")
447✔
139
   if force and bestBreak then
447✔
UNCOV
140
      local onepage = {}
×
UNCOV
141
      for j = 1, bestBreak do
×
UNCOV
142
         onepage[j] = table.remove(vboxlist, 1)
×
143
      end
UNCOV
144
      return onepage, pi
×
145
   end
146
   return false, restart
447✔
147
end
148

149
return pagebuilder
48✔
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