about summary refs log tree commit diff stats
path: root/src/layout/engine.nim
diff options
context:
space:
mode:
authorbptato <nincsnevem662@gmail.com>2024-02-29 01:24:24 +0100
committerbptato <nincsnevem662@gmail.com>2024-02-29 01:24:24 +0100
commit558596b72c1c0fd355b0e7f1476651852e3cee5a (patch)
tree625194da2eb2b5b8c077369ae32d833ca503eaec /src/layout/engine.nim
parent074e2089d909e4f7ef85df89698875ec2b8935c3 (diff)
downloadchawan-558596b72c1c0fd355b0e7f1476651852e3cee5a.tar.gz
layout: reduce useless empty lines in inline boxes
If we are going to round things in layout, let's do it properly. Adding
fake height has negative utility (things get more annoying to read),
so now we don't.

TODO: finding the correct positioning of the baseline after adding
padding is an open question.  Probably have to revise the algorithm
somewhat to get it right.

(This should also fix the bug where error correction would make inline
box backgrounds shift into unrelated text, causing much confusion for
the reader.)
Diffstat (limited to 'src/layout/engine.nim')
-rw-r--r--src/layout/engine.nim27
1 files changed, 17 insertions, 10 deletions
diff --git a/src/layout/engine.nim b/src/layout/engine.nim
index 0db56704..c0b29e5e 100644
--- a/src/layout/engine.nim
+++ b/src/layout/engine.nim
@@ -196,7 +196,10 @@ type
     # Set at the end of layoutText. It helps determine the beginning of the
     # next inline fragment.
     widthAfterWhitespace: LayoutUnit
+    # offset of line in root fragment
     offsety: LayoutUnit
+    # minimum height to fit all inline atoms
+    minHeight: LayoutUnit
 
   LineBox = ref object
     atoms: seq[InlineAtom]
@@ -310,7 +313,7 @@ proc newWord(ictx: var InlineContext, state: var InlineState) =
   ictx.hasshy = false
 
 proc horizontalAlignLine(ictx: var InlineContext, state: InlineState,
-    line: var LineBox) =
+    line: LineBox) =
   let width = case ictx.space.w.t
   of MIN_CONTENT, MAX_CONTENT:
     ictx.size.w
@@ -350,7 +353,7 @@ proc verticalAlignLine(ictx: var InlineContext) =
   # Also, collect the maximum vertical margins of inline blocks.
   var marginTop: LayoutUnit = 0
   var bottomEdge = baseline
-  for i in 0 ..< ictx.currentLine.atoms.len:
+  for i, atom in ictx.currentLine.atoms:
     let atom = ictx.currentLine.atoms[i]
     let iastate = ictx.currentLine.atomstates[i]
     case iastate.vertalign.keyword
@@ -364,10 +367,13 @@ proc verticalAlignLine(ictx: var InlineContext) =
     else:
       baseline = max(baseline, iastate.baseline)
 
+  let ch = ictx.cellheight
+  baseline = baseline.round(ch)
+
   # Resize the line's height based on atoms' height and baseline.
   # The line height should be at least as high as the highest baseline used by
   # an atom plus that atom's height.
-  for i in 0 ..< ictx.currentLine.atoms.len:
+  for i, atom in ictx.currentLine.atoms:
     let atom = ictx.currentLine.atoms[i]
     let iastate = ictx.currentLine.atomstates[i]
     # In all cases, the line's height must at least equal the atom's height.
@@ -394,9 +400,8 @@ proc verticalAlignLine(ictx: var InlineContext) =
         atom.size.h, ictx.currentLine.size.h)
 
   # Now we can calculate the actual position of atoms inside the line.
-  for i in 0 ..< ictx.currentLine.atoms.len:
+  for i, atom in ictx.currentLine.atoms:
     let iastate = ictx.currentLine.atomstates[i]
-    let atom = addr ictx.currentLine.atoms[i]
     case iastate.vertalign.keyword
     of VERTICAL_ALIGN_BASELINE:
       # Atom is placed at (line baseline) - (atom baseline) - len
@@ -426,9 +431,10 @@ proc verticalAlignLine(ictx: var InlineContext) =
   # line box's top padding.
   let paddingTop = ictx.currentLine.paddingTop
   let offsety = ictx.currentLine.offsety
-  let ch = ictx.cellheight
   for atom in ictx.currentLine.atoms:
     atom.offset.y = (atom.offset.y + marginTop + paddingTop + offsety).round(ch)
+    ictx.currentLine.minHeight = max(ictx.currentLine.minHeight,
+      atom.offset.y - offsety + atom.size.h)
 
   # Set the line height to new top edge + old bottom edge, and set the
   # baseline.
@@ -437,7 +443,11 @@ proc verticalAlignLine(ictx: var InlineContext) =
   # Add padding.
   ictx.currentLine.size.h += ictx.currentLine.paddingTop
   ictx.currentLine.size.h += ictx.currentLine.paddingBottom
+  #TODO this does not really work with rounding :/
   ictx.currentLine.baseline += ictx.currentLine.paddingTop
+  # Round line
+  ictx.currentLine.size.h = max(ictx.currentLine.size.h.round(ch),
+    ictx.currentLine.minHeight)
 
 proc addSpacing(ictx: var InlineContext, width, height: LayoutUnit,
     hang = false) =
@@ -527,9 +537,6 @@ proc finishLine(ictx: var InlineContext, state: var InlineState, wrap: bool,
       state.fragment.size.w = max(lineWidth, state.fragment.size.w)
     ictx.size.w = max(ictx.size.w, lineWidth)
     ictx.lines.add(ictx.currentLine.line)
-    # round line height to real cell height, so that the next line will actually
-    # come after ours.
-    ictx.currentLine.size.h = ictx.currentLine.size.h.round(ictx.cellheight)
     ictx.currentLine = LineBoxState(
       offsety: y + ictx.currentLine.size.h,
       line: LineBox()
@@ -538,7 +545,7 @@ proc finishLine(ictx: var InlineContext, state: var InlineState, wrap: bool,
 
 proc finish(ictx: var InlineContext, state: var InlineState) =
   ictx.finishLine(state, wrap = false)
-  for line in ictx.lines.mitems:
+  for line in ictx.lines:
     ictx.horizontalAlignLine(state, line)
 
 func minwidth(atom: InlineAtom): LayoutUnit =