1
//! Formatting context layout (block, inline, table, and flex/grid via Taffy)
2

            
3
use std::{
4
    collections::{BTreeMap, HashMap},
5
    sync::Arc,
6
};
7

            
8
use azul_core::{
9
    dom::{FormattingContext, NodeId, NodeType},
10
    geom::{LogicalPosition, LogicalRect, LogicalSize},
11
    resources::RendererResources,
12
    styled_dom::{StyledDom, StyledNodeState},
13
};
14
use azul_css::{
15
    css::CssPropertyValue,
16
    props::{
17
        basic::{
18
            font::{StyleFontStyle, StyleFontWeight},
19
            pixel::{DEFAULT_FONT_SIZE, PT_TO_PX},
20
            ColorU, PhysicalSize, PropertyContext, ResolutionContext, SizeMetric,
21
        },
22
        layout::{
23
            ColumnCount, ColumnWidth, LayoutBorderSpacing, LayoutClear, LayoutDisplay, LayoutFloat,
24
            LayoutHeight, LayoutJustifyContent, LayoutOverflow, LayoutPosition, LayoutTableLayout,
25
            LayoutTextJustify, LayoutWidth, LayoutWritingMode, ShapeInside, ShapeOutside,
26
            StyleBorderCollapse, StyleCaptionSide, StyleEmptyCells,
27
        },
28
        property::CssProperty,
29
        style::{
30
            BorderStyle, StyleDirection, StyleHyphens, StyleLineBreak, StyleListStylePosition,
31
            StyleListStyleType, StyleOverflowWrap, StyleTextAlign, StyleTextAlignLast,
32
            StyleTextBoxTrim, StyleTextCombineUpright, StyleTextOrientation, StyleUnicodeBidi,
33
            StyleVerticalAlign, StyleVisibility, StyleWhiteSpace, StyleWordBreak,
34
        },
35
    },
36
};
37
use rust_fontconfig::FcWeight;
38
use taffy::{AvailableSpace, LayoutInput, Line, Size as TaffySize};
39

            
40
#[cfg(feature = "text_layout")]
41
use crate::text3;
42
use crate::{
43
    debug_ifc_layout, debug_info, debug_log, debug_table_layout, debug_warning,
44
    font_traits::{
45
        ContentIndex, FontLoaderTrait, ImageSource, InlineContent, InlineImage, InlineShape,
46
        LayoutFragment, ObjectFit, ParsedFontTrait, SegmentAlignment, ShapeBoundary,
47
        ShapeDefinition, ShapedItem, Size, StyleProperties, StyledRun, TextLayoutCache,
48
        UnifiedConstraints,
49
    },
50
    solver3::{
51
        geometry::{BoxProps, EdgeSizes, IntrinsicSizes},
52
        getters::{
53
            get_css_border_bottom_width, get_css_border_top_width,
54
            get_css_height, get_css_padding_bottom, get_css_padding_top,
55
            get_css_width, get_direction_property, get_unicode_bidi_property,
56
            get_display_property, get_element_font_size, get_float, get_clear,
57
            get_list_style_position, get_list_style_type, get_overflow_x, get_overflow_y,
58
            get_parent_font_size, get_root_font_size, get_style_properties,
59
            get_text_align, get_text_box_trim_property, get_text_orientation_property,
60
            get_vertical_align_property, get_visibility, get_white_space_property,
61
            get_writing_mode, MultiValue,
62
        },
63
        layout_tree::{
64
            AnonymousBoxType, CachedInlineLayout, LayoutNode, LayoutNodeHot, LayoutNodeWarm, LayoutNodeCold, LayoutTree, PseudoElement,
65
        },
66
        positioning::get_position_type,
67
        scrollbar::ScrollbarRequirements,
68
        sizing::extract_text_from_node,
69
        taffy_bridge, LayoutContext, LayoutDebugMessage, LayoutError, Result,
70
    },
71
    text3::cache::{AvailableSpace as Text3AvailableSpace, TextAlign as Text3TextAlign},
72
};
73

            
74
/// Default scrollbar width in pixels (CSS `scrollbar-width: auto`).
75
/// This is only used as a fallback when per-node CSS cannot be queried.
76
/// Prefer `getters::get_layout_scrollbar_width_px()` for per-node resolution.
77
pub const DEFAULT_SCROLLBAR_WIDTH_PX: f32 = 16.0;
78

            
79
// Note: DEFAULT_FONT_SIZE and PT_TO_PX are imported from pixel
80

            
81
/// Result of BFC layout with margin escape information
82
#[derive(Debug, Clone)]
83
pub(crate) struct BfcLayoutResult {
84
    /// Standard layout output (positions, overflow size, baseline)
85
    pub output: LayoutOutput,
86
    /// Top margin that escaped the BFC (for parent-child collapse)
87
    /// If Some, this margin should be used by parent instead of positioning this BFC
88
    pub escaped_top_margin: Option<f32>,
89
    /// Bottom margin that escaped the BFC (for parent-child collapse)
90
    /// If Some, this margin should collapse with next sibling
91
    pub escaped_bottom_margin: Option<f32>,
92
}
93

            
94
impl BfcLayoutResult {
95
71676
    pub fn from_output(output: LayoutOutput) -> Self {
96
71676
        Self {
97
71676
            output,
98
71676
            escaped_top_margin: None,
99
71676
            escaped_bottom_margin: None,
100
71676
        }
101
71676
    }
102
}
103

            
104
/// The CSS `overflow` property behavior.
105
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
106
pub enum OverflowBehavior {
107
    Visible,
108
    Hidden,
109
    Clip,
110
    Scroll,
111
    Auto,
112
}
113

            
114
impl OverflowBehavior {
115
5
    pub fn is_clipped(&self) -> bool {
116
5
        matches!(self, Self::Hidden | Self::Clip | Self::Scroll | Self::Auto)
117
5
    }
118

            
119
5
    pub fn is_scroll(&self) -> bool {
120
5
        matches!(self, Self::Scroll | Self::Auto)
121
5
    }
122
}
123

            
124
/// Input constraints for a layout function.
125
#[derive(Debug)]
126
pub struct LayoutConstraints<'a> {
127
    /// The available space for the content, excluding padding and borders.
128
    pub available_size: LogicalSize,
129
    /// The CSS writing-mode of the context.
130
    pub writing_mode: LayoutWritingMode,
131
    /// Full writing mode context (writing-mode + direction + text-orientation).
132
    /// Used by writing-mode-aware layout code to correctly map inline/block
133
    /// dimensions to physical x/y coordinates.
134
    pub writing_mode_ctx: super::geometry::WritingModeContext,
135
    /// The state of the parent Block Formatting Context, if applicable.
136
    /// This is how state (like floats) is passed down.
137
    pub bfc_state: Option<&'a mut BfcState>,
138
    // Other properties like text-align would go here.
139
    pub text_align: TextAlign,
140
    /// The size of the containing block (parent's content box).
141
    /// This is used for resolving percentage-based sizes and as parent_size for Taffy.
142
    pub containing_block_size: LogicalSize,
143
    /// The semantic type of the available width constraint.
144
    ///
145
    /// This field is crucial for correct inline layout caching:
146
    /// - `Definite(w)`: Normal layout with a specific available width
147
    /// - `MinContent`: Intrinsic minimum width measurement (maximum wrapping)
148
    /// - `MaxContent`: Intrinsic maximum width measurement (no wrapping)
149
    ///
150
    /// When caching inline layouts, we must track which constraint type was used
151
    /// to compute the cached result. A layout computed with `MinContent` (width=0)
152
    /// must not be reused when the actual available width is known.
153
    pub available_width_type: Text3AvailableSpace,
154
}
155

            
156
/// Manages all layout state for a single Block Formatting Context.
157
/// This struct is created by the BFC root and lives for the duration of its layout.
158
#[derive(Debug, Clone)]
159
pub struct BfcState {
160
    /// The current position for the next in-flow block element.
161
    pub pen: LogicalPosition,
162
    /// The state of all floated elements within this BFC.
163
    pub floats: FloatingContext,
164
    /// The state of margin collapsing within this BFC.
165
    pub margins: MarginCollapseContext,
166
}
167

            
168
impl BfcState {
169
    pub fn new() -> Self {
170
        Self {
171
            pen: LogicalPosition::zero(),
172
            floats: FloatingContext::default(),
173
            margins: MarginCollapseContext::default(),
174
        }
175
    }
176
}
177

            
178
/// Manages vertical margin collapsing within a BFC.
179
#[derive(Debug, Default, Clone)]
180
pub struct MarginCollapseContext {
181
    /// The bottom margin of the last in-flow, block-level element.
182
    /// Can be positive or negative.
183
    pub last_in_flow_margin_bottom: f32,
184
}
185

            
186
/// The result of laying out a formatting context.
187
#[derive(Debug, Default, Clone)]
188
pub struct LayoutOutput {
189
    /// The final positions of child nodes, relative to the container's content-box origin.
190
    pub positions: BTreeMap<usize, LogicalPosition>,
191
    /// The total size occupied by the content, which may exceed `available_size`.
192
    pub overflow_size: LogicalSize,
193
    // +spec:inline-formatting-context:f7eebb - baseline along inline axis for glyph alignment
194
    /// The baseline of the context, if applicable, measured from the top of its content box.
195
    pub baseline: Option<f32>,
196
}
197

            
198
/// Text alignment options
199
#[derive(Debug, Clone, Copy, Default)]
200
pub enum TextAlign {
201
    #[default]
202
    Start,
203
    End,
204
    Center,
205
    Justify,
206
}
207

            
208
/// Represents a single floated element within a BFC.
209
#[derive(Debug, Clone, Copy)]
210
struct FloatBox {
211
    /// The type of float (Left or Right).
212
    kind: LayoutFloat,
213
    /// The rectangle of the float's content box (origin includes top/left margin offset).
214
    rect: LogicalRect,
215
    /// The margin sizes (needed to calculate true margin-box bounds).
216
    margin: EdgeSizes,
217
}
218

            
219
/// Manages the state of all floated elements within a Block Formatting Context.
220
// +spec:block-formatting-context:a4e6f9 - float rules reference only elements in the same BFC (scoped via BfcState)
221
// +spec:floats:2fa329 - Float positioning (left/right shift), content flow along sides, and clear property
222
/// +spec:floats:970b4c - Implements CSS2§9.5 float positioning and flow interaction
223
#[derive(Debug, Default, Clone)]
224
pub struct FloatingContext {
225
    /// All currently positioned floats within the BFC.
226
    pub floats: Vec<FloatBox>,
227
}
228

            
229
impl FloatingContext {
230
    /// Add a newly positioned float to the context
231
2464
    pub fn add_float(&mut self, kind: LayoutFloat, rect: LogicalRect, margin: EdgeSizes) {
232
2464
        self.floats.push(FloatBox { kind, rect, margin });
233
2464
    }
234

            
235
    // +spec:box-model:0c9b13 - line boxes next to floats are shortened to make room
236
    // +spec:floats:148fcd - floating boxes reduce available line box width between containing block edges
237
    // +spec:floats:49a491 - Line boxes stacked with no separation except float clearance, never overlap
238
    // +spec:floats:8974e6 - text flows into vacated space by narrowing line boxes around floats
239
    // +spec:floats:af94f2 - content displaced by float: line boxes shrink to avoid float margin boxes
240
    // +spec:floats:e5961b - remaining text flows into vacated space via available_line_box_space
241
    // +spec:inline-formatting-context:7cbe58 - shortened line boxes due to floats; shift down if too small
242
    /// Finds the available space on the cross-axis for a line box at a given main-axis range.
243
    // +spec:containing-block:4b0c44 - line boxes shortened by floats resume containing block width after float
244
    ///
245
    /// Returns a tuple of (`cross_start_offset`, `cross_end_offset`) relative to the
246
    /// BFC content box, defining the available space for an in-flow element.
247
    // +spec:inline-formatting-context:e70328 - line box width reduced by floats between containing block edges
248
2420
    pub fn available_line_box_space(
249
2420
        &self,
250
2420
        main_start: f32,
251
2420
        main_end: f32,
252
2420
        bfc_cross_size: f32,
253
2420
        wm: LayoutWritingMode,
254
2420
    ) -> (f32, f32) {
255
2420
        let mut available_cross_start = 0.0_f32;
256
2420
        let mut available_cross_end = bfc_cross_size;
257

            
258
3564
        for float in &self.floats {
259
            // Get the logical main-axis span of the existing float's MARGIN BOX.
260
1144
            let float_main_start = float.rect.origin.main(wm) - float.margin.main_start(wm);
261
1144
            let float_main_end = float_main_start + float.rect.size.main(wm)
262
1144
                + float.margin.main_start(wm) + float.margin.main_end(wm);
263

            
264
            // Check for overlap on the main axis.
265
1144
            if main_end > float_main_start && main_start < float_main_end {
266
                // CSS 2.2 § 9.5: border box must not overlap MARGIN BOX of floats,
267
                // so we include the float's margins in the cross-axis bounds.
268
1012
                let float_cross_start = float.rect.origin.cross(wm) - float.margin.cross_start(wm);
269
1012
                let float_cross_end = float_cross_start + float.rect.size.cross(wm)
270
1012
                    + float.margin.cross_start(wm) + float.margin.cross_end(wm);
271

            
272
                // +spec:floats:17a63f - float left/right map to line-left/line-right via logical coords
273
                // +spec:writing-modes:e55820 - line-relative mappings: left/right interpreted as line-left/line-right per writing mode
274
1012
                if float.kind == LayoutFloat::Left {
275
792
                    // "line-left", i.e., cross-start
276
792
                    available_cross_start = available_cross_start.max(float_cross_end);
277
792
                } else {
278
220
                    // Float::Right, i.e., cross-end
279
220
                    available_cross_end = available_cross_end.min(float_cross_start);
280
220
                }
281
132
            }
282
        }
283
2420
        (available_cross_start, available_cross_end)
284
2420
    }
285

            
286
    // +spec:block-formatting-context:d06e6e - clearance computation for clear property on blocks and floats (CSS 2.2 § 9.5.2)
287
    // +spec:floats:31a3d5 - Clearance computation: places border edge even with bottom outer edge of lowest float to be cleared
288
    // +spec:floats:f9bef1 - clear property moves element below preceding floats
289
    /// Returns the main-axis offset needed to be clear of floats of the given type.
290
    // +spec:block-formatting-context:7f6bde - CSS 2.2 § 9.5.2 clear property: clearance places border edge below bottom outer edge of cleared floats
291
    // +spec:block-formatting-context:ef493f - clearance computation: places border edge even with bottom outer edge of lowest float to be cleared; inhibits margin collapsing
292
    // +spec:box-model:b118fe - top border edge must be below bottom outer edge of earlier floats
293
    // +spec:floats:415066 - Clear property: top border edge below bottom outer edge of cleared floats
294
    // +spec:floats:7e4ad6 - clear property: element box may not be adjacent to earlier floats; only considers floats in same BFC
295
    // +spec:floats:32e45d - clear:right causes sibling to flow below right floats
296
    // +spec:floats:7f417a - clear property prevents content from flowing next to floats
297
    // +spec:floats:d06304 - clear property moves element below floats, leaving blank space
298
    // +spec:overflow:1a7aff - clearance calculation (incl. negative clearance) and clear on floats (constraint #10)
299
    // +spec:positioning:1c2508 - clearance calculation: places border edge even with bottom outer edge of lowest cleared float (CSS 2.2 § 9.5.2)
300
    // +spec:positioning:fe0912 - clearance computation: places border edge below bottom outer edge of cleared floats
301
    // (clearance = amount to place border edge even with bottom outer edge of lowest
302
    // float to be cleared); clearance can be negative per spec example 2
303
    // +spec:floats:054a1e - Clearance computation: positions border edge below bottom outer edge of cleared floats
304
    // +spec:floats:cb984c - Clearance can be negative per spec example 2; inhibits margin collapsing
305
660
    pub fn clearance_offset(
306
660
        &self,
307
660
        clear: LayoutClear,
308
660
        current_main_offset: f32,
309
660
        wm: LayoutWritingMode,
310
660
    ) -> f32 {
311
660
        let mut max_end_offset = 0.0_f32;
312

            
313
660
        let check_left = clear == LayoutClear::Left || clear == LayoutClear::Both;
314
660
        let check_right = clear == LayoutClear::Right || clear == LayoutClear::Both;
315

            
316
1452
        for float in &self.floats {
317
792
            let should_clear_this_float = (check_left && float.kind == LayoutFloat::Left)
318
396
                || (check_right && float.kind == LayoutFloat::Right);
319

            
320
792
            if should_clear_this_float {
321
616
                // CSS 2.2 § 9.5.2: "the top border edge of the box be below the bottom outer edge"
322
616
                // Outer edge = margin-box boundary (content + padding + border + margin)
323
616
                let float_margin_box_end = float.rect.origin.main(wm)
324
616
                    + float.rect.size.main(wm)
325
616
                    + float.margin.main_end(wm);
326
616
                max_end_offset = max_end_offset.max(float_margin_box_end);
327
616
            }
328
        }
329

            
330
660
        if max_end_offset > current_main_offset {
331
396
            max_end_offset
332
        } else {
333
264
            current_main_offset
334
        }
335
660
    }
336
}
337

            
338
/// Encapsulates all state needed to lay out a single Block Formatting Context.
339
struct BfcLayoutState {
340
    /// The current position for the next in-flow block element.
341
    pen: LogicalPosition,
342
    floats: FloatingContext,
343
    margins: MarginCollapseContext,
344
    /// The writing mode of the BFC root.
345
    writing_mode: LayoutWritingMode,
346
}
347

            
348
/// Result of a formatting context layout operation
349
#[derive(Debug, Default)]
350
pub struct LayoutResult {
351
    pub positions: Vec<(usize, LogicalPosition)>,
352
    pub overflow_size: Option<LogicalSize>,
353
    pub baseline_offset: f32,
354
}
355

            
356
// Entry Point & Dispatcher
357

            
358
/// Main dispatcher for formatting context layout.
359
///
360
/// Routes layout to the appropriate formatting context handler based on the node's
361
/// `formatting_context` property. This is the main entry point for all layout operations.
362
///
363
/// # CSS Spec References
364
/// - CSS 2.2 § 9.4: Formatting contexts
365
/// - CSS Flexbox § 3: Flex formatting contexts
366
/// - CSS Grid § 5: Grid formatting contexts
367
// +spec:block-formatting-context:b04653 - dispatches layout by formatting context type (BFC, IFC, Table, Flex, Grid)
368
// +spec:block-formatting-context:e46499 - inner display type determines formatting context (BFC, IFC, table, flex, grid)
369
85739
pub fn layout_formatting_context<T: ParsedFontTrait>(
370
85739
    ctx: &mut LayoutContext<'_, T>,
371
85739
    tree: &mut LayoutTree,
372
85739
    text_cache: &mut crate::font_traits::TextLayoutCache,
373
85739
    node_index: usize,
374
85739
    constraints: &LayoutConstraints,
375
85739
    float_cache: &mut HashMap<usize, FloatingContext>,
376
85739
) -> Result<BfcLayoutResult> {
377
    // [g147e az-web-lift DIAG] PURE-CONSTANT entry marker (0x609E0+slot) — fires before any node read,
378
    // so it reliably shows whether layout_formatting_context is ENTERED for the nested div nodes 1,2.
379
    #[cfg(feature = "web_lift")]
380
    unsafe { crate::az_mark(((0x609E0 + (node_index & 7) * 4)) as u32, (0xC0DE0042) as u32); }
381
85739
    let node = tree.get(node_index).ok_or(LayoutError::InvalidTree)?;
382
    // [g147i az-web-lift DIAG] node REFERENCE address (0x60B80+slot) — NOT a field deref, so reliable.
383
    // If nodes 0,1,2 aren't spaced by sizeof(LayoutNodeHot) → tree.get(index>0) mis-lifts the Vec stride,
384
    // making nodes 1,2 garbage references (which would explain FC reading garbage + reads destabilizing).
385
    #[cfg(feature = "web_lift")]
386
    unsafe { crate::az_mark(((0x60B80 + (node_index & 7) * 4)) as u32, ((node as *const _ as usize) as u32) as u32); }
387

            
388
    // [g147 az-web-lift] Recompute the IFC decision from the DOM: on the lift, the stored
389
    // `node.formatting_context` reads GARBAGE for nested inline divs (2026-06-10 re-test WITHOUT
390
    // this bypass: nodes 1/2 dispatch to the `_` arm + determine_formatting_context_for_display's
391
    // markers never fire → the FC ASSIGNMENT path itself mis-lifts upstream — NOT fixed by the
392
    // repr(C,u8) guard, NOT fixed by the leak-gated SP restore; same family as the enum/jump-table
393
    // devirt class). The styled_dom IS reliable, so a block container whose children are all
394
    // inline-level establishes an IFC (CSS 2.2 §9.2.1) — semantically valid recomputation, not a
395
    // hack on top of garbage. web_lift-gated → native untouched. Remove when the FC-assignment
396
    // mis-lift is root-caused (follow-up: bisect LayoutTreeBuilder's determine_/display match).
397
    #[cfg(feature = "web_lift")]
398
    {
399
        let force_ifc = node
400
            .dom_node_id
401
            .map_or(false, |dom_id| {
402
                crate::solver3::layout_tree::has_only_inline_children(ctx.styled_dom, dom_id)
403
            });
404
        if force_ifc {
405
            unsafe { crate::az_mark(((0x60BA0 + (node_index & 7) * 4)) as u32, (0xC0DE1FC0) as u32); }
406
            return layout_ifc(ctx, text_cache, tree, node_index, constraints)
407
                .map(BfcLayoutResult::from_output);
408
        }
409
    }
410

            
411
    // [g147b az-web-lift DIAG] per-node FormattingContext discriminant at layout_formatting_context
412
    // entry (0x609A0+slot). Pairs with the dispatch-arm marker (0x609C0+slot) inside each match arm:
413
    // if a text-div's FC reads Inline(2) but the arm marker shows Block(1) → match dispatch mis-lifts;
414
    // if FC reads Block(1) → tree-construction FC assignment is wrong; if 0x609A0 stays unset for the
415
    // div node → layout_formatting_context is never called for it (cache-hit short-circuit upstream).
416
    #[cfg(feature = "web_lift")]
417
    unsafe {
418
        let fc_disc = match node.formatting_context {
419
            FormattingContext::Block { .. } => 1u32,
420
            FormattingContext::Inline => 2,
421
            FormattingContext::InlineBlock => 3,
422
            FormattingContext::Flex => 4,
423
            FormattingContext::Grid => 5,
424
            FormattingContext::Table => 6,
425
            FormattingContext::TableCell => 7,
426
            FormattingContext::TableCaption => 8,
427
            _ => 0,
428
        };
429
        crate::az_mark(((0x609A0 + (node_index & 7) * 4)) as u32, (fc_disc | 0xC0DE0000) as u32);
430
    }
431

            
432
85739
    debug_info!(
433
84023
        ctx,
434
84023
        "[layout_formatting_context] node_index={}, fc={:?}, available_size={:?}",
435
        node_index,
436
        node.formatting_context,
437
        constraints.available_size
438
    );
439

            
440
    // +spec:block-formatting-context:06a24f - CSS 2.2 § 9.4: block-level boxes → BFC, inline-level → IFC
441
    // +spec:block-formatting-context:9428cf - block container can establish both BFC and IFC simultaneously
442
    // +spec:inline-formatting-context:8bfe73 - display:flow generates inline box (Inline) or block container (Block) based on outer display type
443
85739
    match node.formatting_context {
444
        FormattingContext::Block { .. } => {
445
            #[cfg(feature = "web_lift")]
446
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0001) as u32); }
447
8757
            layout_bfc(ctx, tree, text_cache, node_index, constraints, float_cache)
448
        }
449
        // +spec:inline-formatting-context:a180ed - IFC establishment: inline-level boxes fragmented into line boxes with baseline alignment
450
        FormattingContext::Inline => {
451
            #[cfg(feature = "web_lift")]
452
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0002) as u32); }
453
64437
            layout_ifc(ctx, text_cache, tree, node_index, constraints)
454
64437
                .map(BfcLayoutResult::from_output)
455
        }
456
        FormattingContext::InlineBlock => {
457
            #[cfg(feature = "web_lift")]
458
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0003) as u32); }
459
            // +spec:display-property:1f5ddf - inline-level boxes with non-flow inner display establish new formatting context
460
            // +spec:inline-formatting-context:1ad004 - atomic inline (inline-block) establishes new formatting context
461
            // CSS 2.2 § 9.4.1: "inline-blocks... establish new block formatting contexts"
462
            // +spec:inline-block:8d21f6 - inline-block generates inline-level block container (BFC inside, atomic inline outside)
463
            // InlineBlock ALWAYS establishes a BFC for its contents.
464
            // The element itself participates as an atomic inline in its parent's IFC,
465
            // but its children are laid out in a BFC, not an IFC.
466
176
            let mut temp_float_cache = HashMap::new();
467
176
            layout_bfc(ctx, tree, text_cache, node_index, constraints, &mut temp_float_cache)
468
        }
469
        // +spec:table-layout:753687 - CSS 2.2 §17.2 table model: display values map to FormattingContext variants and dispatch table layout
470
        FormattingContext::Table => {
471
            #[cfg(feature = "web_lift")]
472
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0006) as u32); }
473
1452
            layout_table_fc(ctx, tree, text_cache, node_index, constraints)
474
1452
                .map(BfcLayoutResult::from_output)
475
        }
476
        // Table-internal flex items are blockified during tree construction
477
        // (blockify_flex_item_if_table_internal in layout_tree.rs), so they arrive
478
        // here as Block, not TableCell etc.
479
        FormattingContext::Flex | FormattingContext::Grid => {
480
            #[cfg(feature = "web_lift")]
481
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0004) as u32); }
482
885
            layout_flex_grid(ctx, tree, text_cache, node_index, constraints)
483
        }
484
        // that are not block boxes, so they establish new BFCs for their contents
485
        FormattingContext::TableCell | FormattingContext::TableCaption => {
486
            #[cfg(feature = "web_lift")]
487
            unsafe { crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0007) as u32); }
488
10032
            let mut temp_float_cache = HashMap::new();
489
10032
            layout_bfc(ctx, tree, text_cache, node_index, constraints, &mut temp_float_cache)
490
        }
491
        _ => {
492
            // [g147g az-web-lift DIAG] read the RAW discriminant byte (offset 0 under repr(C,u8)) of the
493
            // node that fell through to `_`. node 0 won't hit `_`; nodes 1,2 (divs) write their disc to
494
            // 0x60B40+slot. disc=1 ⇒ value IS Inline but the dispatch match mis-branched (match/jump-table
495
            // lift bug); disc≠1 ⇒ tree-construction stored the wrong/garbage FC for the nested div.
496
            #[cfg(feature = "web_lift")]
497
            unsafe {
498
                crate::az_mark(((0x609C0 + (node_index & 7) * 4)) as u32, (0xC0DE0009) as u32);
499
                let disc: u8 = core::ptr::read_volatile((&node.formatting_context) as *const FormattingContext as *const u8);
500
                crate::az_mark(((0x60B40 + (node_index & 7) * 4)) as u32, (0xC0DE0000 | (disc as u32)) as u32);
501
            }
502
            // Unknown formatting context - fall back to BFC
503
            let mut temp_float_cache = HashMap::new();
504
            layout_bfc(
505
                ctx,
506
                tree,
507
                text_cache,
508
                node_index,
509
                constraints,
510
                &mut temp_float_cache,
511
            )
512
        }
513
    }
514
85739
}
515

            
516
// Flex / grid layout (taffy Bridge)
517
// containing block determined by grid-placement properties; Taffy handles this internally
518
// (grid auto-placement §8.5 and abspos grid items use grid-area CB, not just padding box)
519

            
520
/// Lays out a Flex or Grid formatting context using the Taffy layout engine.
521
///
522
/// # CSS Spec References
523
///
524
/// - CSS Flexbox § 9: Flex Layout Algorithm
525
/// - CSS Grid § 12: Grid Layout Algorithm
526
// gutters on either side of collapsed tracks collapse including distributed alignment space,
527
// minimum contribution = outer size from min-width/min-height if specified size is auto else
528
// min-content contribution) — all handled by Taffy grid implementation
529
///
530
/// # Implementation Notes
531
///
532
/// - Resolves explicit CSS dimensions to pixel values for `known_dimensions`
533
/// - Uses `InherentSize` mode when explicit dimensions are set
534
/// - Uses `ContentSize` mode for auto-sizing (shrink-to-fit)
535
885
fn layout_flex_grid<T: ParsedFontTrait>(
536
885
    ctx: &mut LayoutContext<'_, T>,
537
885
    tree: &mut LayoutTree,
538
885
    text_cache: &mut crate::font_traits::TextLayoutCache,
539
885
    node_index: usize,
540
885
    constraints: &LayoutConstraints,
541
885
) -> Result<BfcLayoutResult> {
542
    // Available space comes directly from constraints - margins are handled by Taffy
543
885
    let available_space = TaffySize {
544
885
        width: AvailableSpace::Definite(constraints.available_size.width),
545
885
        height: AvailableSpace::Definite(constraints.available_size.height),
546
885
    };
547

            
548
885
    let node = tree.get(node_index).ok_or(LayoutError::InvalidTree)?;
549

            
550
    // from flex line's cross size (clamped by min/max) when align-self:stretch, cross-size:auto,
551
    // and neither cross-axis margin is auto. Otherwise uses hypothetical cross size.
552
    // NOTE: visibility:collapse strut size for flex items is handled internally by Taffy.
553
    //
554
    // Resolve explicit CSS dimensions to pixel values.
555
    // This is CRITICAL for align-items: stretch to work correctly!
556
    // Taffy uses known_dimensions to calculate cross_axis_available_space for children.
557
885
    let (explicit_width, has_explicit_width) =
558
885
        resolve_explicit_dimension_width(ctx, node, constraints);
559
885
    let (explicit_height, has_explicit_height) =
560
885
        resolve_explicit_dimension_height(ctx, node, constraints);
561

            
562
    // FIX: For root nodes or nodes where the parent provides a definite size,
563
    // use the available_size as known_dimensions if no explicit CSS width/height is set.
564
    // This is critical for `align-self: stretch` to work - Taffy needs to know the
565
    // cross-axis size of the container to stretch children to fill it.
566
885
    let is_root = node.parent.is_none();
567

            
568
885
    let bp = node.box_props.unpack();
569
885
    let width_adjustment = bp.border.left
570
885
        + bp.border.right
571
885
        + bp.padding.left
572
885
        + bp.padding.right;
573
885
    let height_adjustment = bp.border.top
574
885
        + bp.border.bottom
575
885
        + bp.padding.top
576
885
        + bp.padding.bottom;
577

            
578
    // `constraints.available_size` is the root's CONTENT-BOX (produced by
579
    // `prepare_layout_context::inner_size(final_used_size)`), not the viewport
580
    // border-box. Previously, the code used it as if it were border-box,
581
    // causing taffy to subtract padding a second time and shrink the content
582
    // area by 2x padding. For the root, pull the actual border-box from
583
    // `node.used_size` (set by `calculate_used_size_for_node` before this call).
584
885
    let root_border_box = node.used_size;
585

            
586
885
    let effective_width = if has_explicit_width {
587
354
        explicit_width
588
531
    } else if is_root {
589
485
        root_border_box.as_ref().map(|s| s.width).or_else(|| {
590
            if constraints.available_size.width.is_finite() {
591
                // Fallback: convert content-box to border-box.
592
                Some(constraints.available_size.width + width_adjustment)
593
            } else {
594
                None
595
            }
596
        })
597
    } else {
598
        // Non-root flex/grid container with `width: auto`: for a block-level
599
        // child the parent's block layout has ALREADY resolved the used width
600
        // (auto → fill containing block) before descending into this FC — pass
601
        // it through as the definite border-box width, exactly like the root
602
        // branch does with its used_size. Without this, known_dimensions.width
603
        // stays None and taffy treats a column container's cross axis as
604
        // INDEFINITE, so `align-items: stretch` items get the flex line's
605
        // max-content width instead of the container width (live bug: under
606
        // the injected Html menubar wrapper, AzulPaint's body laid out its
607
        // header AND canvas at 315.776px — the header text's max-content —
608
        // instead of the body's 624px).
609
46
        node.used_size.as_ref().map(|s| s.width)
610
    };
611
885
    let effective_height = if has_explicit_height {
612
883
        explicit_height
613
2
    } else if is_root {
614
        root_border_box.as_ref().map(|s| s.height).or_else(|| {
615
            if constraints.available_size.height.is_finite() {
616
                Some(constraints.available_size.height + height_adjustment)
617
            } else {
618
                None
619
            }
620
        })
621
    } else {
622
2
        None
623
    };
624
885
    let has_effective_width = effective_width.is_some();
625
885
    let has_effective_height = effective_height.is_some();
626

            
627
    // Taffy interprets known_dimensions as border-box. CSS width/height default
628
    // to content-box, so explicit values need +padding+border added. For the
629
    // ROOT element, however, we auto-apply box-sizing: border-box — the common
630
    // CSS reset pattern — so `height:100%` + padding fits the viewport instead
631
    // of overflowing by padding (which the default content-box interpretation
632
    // would produce, since 100% of ICB is viewport-sized content, with padding
633
    // added outside pushing border-box past the viewport).
634
885
    let adjusted_width = if has_explicit_width && !is_root {
635
89
        explicit_width.map(|w| w + width_adjustment)
636
796
    } else if has_explicit_width && is_root {
637
265
        explicit_width
638
    } else {
639
531
        effective_width
640
    };
641
885
    let adjusted_height = if has_explicit_height && !is_root {
642
133
        explicit_height.map(|h| h + height_adjustment)
643
752
    } else if has_explicit_height && is_root {
644
750
        explicit_height
645
    } else {
646
2
        effective_height
647
    };
648

            
649
    // CSS Flexbox § 9.2: Use InherentSize when explicit dimensions are set,
650
    // ContentSize for auto-sizing (shrink-to-fit behavior).
651
885
    let sizing_mode = if has_effective_width || has_effective_height {
652
885
        taffy::SizingMode::InherentSize
653
    } else {
654
        taffy::SizingMode::ContentSize
655
    };
656

            
657
885
    let known_dimensions = TaffySize {
658
885
        width: adjusted_width,
659
885
        height: adjusted_height,
660
885
    };
661

            
662
    // parent_size tells Taffy the size of the container's parent.
663
    // For root nodes, the "parent" is the viewport, but since margins are already
664
    // handled by calculate_used_size_for_node(), we use containing_block_size directly.
665
    // For non-root nodes, containing_block_size is already the parent's content-box.
666
885
    let parent_size = translate_taffy_size(constraints.containing_block_size);
667

            
668
885
    let taffy_inputs = LayoutInput {
669
885
        known_dimensions,
670
885
        parent_size,
671
885
        available_space,
672
885
        run_mode: taffy::RunMode::PerformLayout,
673
885
        sizing_mode,
674
885
        axis: taffy::RequestedAxis::Both,
675
885
        // Flex and Grid containers establish a new BFC, preventing margin collapse.
676
885
        vertical_margins_are_collapsible: Line::FALSE,
677
885
    };
678

            
679
885
    debug_info!(
680
753
        ctx,
681
753
        "CALLING LAYOUT_TAFFY FOR FLEX/GRID FC node_index={:?}",
682
        node_index
683
    );
684

            
685
    // For the root with auto-applied border-box: sync node.used_size so
686
    // display-list rendering matches the border-box we handed taffy.
687
    // Without this, the root's background/border would paint at the
688
    // inflated size from calculate_used_size_for_node while taffy placed
689
    // children inside a smaller content-box.
690
885
    if is_root {
691
750
        if let (Some(aw), Some(ah)) = (adjusted_width, adjusted_height) {
692
750
            if let Some(node_mut) = tree.get_mut(node_index) {
693
750
                node_mut.used_size = Some(LogicalSize::new(aw, ah));
694
750
            }
695
        }
696
135
    }
697

            
698
    // Cache border values before the mutable borrow in layout_taffy_subtree
699
885
    let border_left = bp.border.left;
700
885
    let border_top = bp.border.top;
701

            
702
885
    let taffy_output =
703
885
        taffy_bridge::layout_taffy_subtree(ctx, tree, text_cache, node_index, taffy_inputs);
704

            
705
    // Collect child positions from the tree (Taffy stores results directly on nodes).
706
885
    let mut output = LayoutOutput::default();
707
    // Use content_size for overflow detection, not container size.
708
    // content_size represents the actual size of all children, which may exceed the container.
709
    //
710
    // Taffy's content_size is measured from (0,0) of the border-box, so it includes
711
    // border.top/left as a leading offset.  The scrollbar geometry and scroll clamp
712
    // both measure inside the padding-box (border stripped).  Subtract the start
713
    // border so that overflow_size is in the same coordinate space as the viewport
714
    // (padding-box), preventing extra scroll range equal to the border width.
715
885
    let raw = translate_taffy_size_back(taffy_output.content_size);
716
885
    output.overflow_size = LogicalSize::new(
717
885
        (raw.width - border_left).max(0.0),
718
885
        (raw.height - border_top).max(0.0),
719
885
    );
720

            
721
885
    let children: Vec<usize> = tree.children(node_index).to_vec();
722
2346
    for &child_idx in &children {
723
1461
        if let Some(warm_node) = tree.warm(child_idx) {
724
1461
            if let Some(pos) = warm_node.relative_position {
725
1461
                output.positions.insert(child_idx, pos);
726
1461
            }
727
        }
728
    }
729

            
730
885
    Ok(BfcLayoutResult::from_output(output))
731
885
}
732

            
733
/// Resolves explicit CSS width to pixel value for Taffy layout.
734
885
fn resolve_explicit_dimension_width<T: ParsedFontTrait>(
735
885
    ctx: &LayoutContext<'_, T>,
736
885
    node: &LayoutNodeHot,
737
885
    constraints: &LayoutConstraints,
738
885
) -> (Option<f32>, bool) {
739
885
    node.dom_node_id
740
885
        .map(|id| {
741
885
            let width = get_css_width(
742
885
                ctx.styled_dom,
743
885
                id,
744
885
                &ctx.styled_dom.styled_nodes.as_container()[id].styled_node_state,
745
            );
746
885
            match width.unwrap_or_default() {
747
531
                LayoutWidth::Auto => (None, false),
748
354
                LayoutWidth::Px(px) => {
749
354
                    let pixels = resolve_size_metric(
750
354
                        px.metric,
751
354
                        px.number.get(),
752
354
                        constraints.available_size.width,
753
354
                        ctx.viewport_size,
754
                    );
755
354
                    (Some(pixels), true)
756
                }
757
                LayoutWidth::MinContent | LayoutWidth::MaxContent | LayoutWidth::FitContent(_) => (None, false),
758
                LayoutWidth::Calc(items) => {
759
                    let node_state = &ctx.styled_dom.styled_nodes.as_container()[id].styled_node_state;
760
                    let em = get_element_font_size(ctx.styled_dom, id, node_state);
761
                    let calc_ctx = super::calc::CalcResolveContext {
762
                        items, em_size: em, rem_size: DEFAULT_FONT_SIZE,
763
                    };
764
                    let px = super::calc::evaluate_calc(&calc_ctx, constraints.available_size.width);
765
                    (Some(px), true)
766
                }
767
            }
768
885
        })
769
885
        .unwrap_or((None, false))
770
885
}
771

            
772
/// Resolves explicit CSS height to pixel value for Taffy layout.
773
885
fn resolve_explicit_dimension_height<T: ParsedFontTrait>(
774
885
    ctx: &LayoutContext<'_, T>,
775
885
    node: &LayoutNodeHot,
776
885
    constraints: &LayoutConstraints,
777
885
) -> (Option<f32>, bool) {
778
885
    node.dom_node_id
779
885
        .map(|id| {
780
885
            let height = get_css_height(
781
885
                ctx.styled_dom,
782
885
                id,
783
885
                &ctx.styled_dom.styled_nodes.as_container()[id].styled_node_state,
784
            );
785
885
            match height.unwrap_or_default() {
786
2
                LayoutHeight::Auto => (None, false),
787
883
                LayoutHeight::Px(px) => {
788
883
                    let pixels = resolve_size_metric(
789
883
                        px.metric,
790
883
                        px.number.get(),
791
883
                        constraints.available_size.height,
792
883
                        ctx.viewport_size,
793
                    );
794
883
                    (Some(pixels), true)
795
                }
796
                LayoutHeight::MinContent | LayoutHeight::MaxContent | LayoutHeight::FitContent(_) => (None, false),
797
                LayoutHeight::Calc(items) => {
798
                    let node_state = &ctx.styled_dom.styled_nodes.as_container()[id].styled_node_state;
799
                    let em = get_element_font_size(ctx.styled_dom, id, node_state);
800
                    let calc_ctx = super::calc::CalcResolveContext {
801
                        items, em_size: em, rem_size: DEFAULT_FONT_SIZE,
802
                    };
803
                    let px = super::calc::evaluate_calc(&calc_ctx, constraints.available_size.height);
804
                    (Some(px), true)
805
                }
806
            }
807
885
        })
808
885
        .unwrap_or((None, false))
809
885
}
810

            
811
// +spec:floats:167a2c - Float positioning rules (CSS 2.2 § 9.5.1): left/right/none, precise placement constraints
812
// +spec:floats:6a1769 - Float shortens line boxes, margins never collapse, stacking order
813
// +spec:floats:15bfd9 - float:right positions element at line-right edge within BFC
814
// +spec:floats:afc8e2 - Float positioning rules (CSS 2.2 § 9.5 rules 1-8): left/right edge containment, earlier-float stacking, outer-top constraints, and "move down" when insufficient space
815
/// Position a float within a BFC, considering existing floats.
816
/// Returns the LogicalRect (margin box) for the float.
817
// +spec:box-model:db0f02 - Float positioning: line boxes shortened by floats, floats shift down if no space, BFC elements must not overlap float margin boxes
818
// +spec:containing-block:136e45 - Float shifted left/right until outer edge touches containing block edge or another float
819
// +spec:containing-block:3ebb4e - Content moves below floats when containing block too narrow
820
// +spec:floats:45fce7 - Float positioning: pulled out of flow, line boxes shortened around float
821
// +spec:floats:f6c218 - float pulled out of flow, line boxes shorten around it
822
// +spec:height-calculation:86142a - CSS 2.2 §9.5 float positioning, clearance, and margin non-collapsing
823
// +spec:width-calculation:761677 - float positioning: content flows around floats, line boxes shortened by float presence
824
1144
fn position_float(
825
1144
    float_ctx: &FloatingContext,
826
1144
    float_type: LayoutFloat,
827
1144
    size: LogicalSize,
828
1144
    margin: &EdgeSizes,
829
1144
    current_main_offset: f32,
830
1144
    bfc_cross_size: f32,
831
1144
    wm: LayoutWritingMode,
832
1144
) -> LogicalRect {
833
    // Start at the current main-axis position (Y in horizontal-tb)
834
1144
    let mut main_start = current_main_offset;
835

            
836
    // Calculate total size including margins
837
1144
    let total_main = size.main(wm) + margin.main_start(wm) + margin.main_end(wm);
838
1144
    let total_cross = size.cross(wm) + margin.cross_start(wm) + margin.cross_end(wm);
839

            
840
    // +spec:floats:3d89d8 - shift float downward when not enough horizontal room
841
    // Find a position where the float fits
842
1144
    let cross_start = loop {
843
1188
        let (avail_start, avail_end) = float_ctx.available_line_box_space(
844
1188
            main_start,
845
1188
            main_start + total_main,
846
1188
            bfc_cross_size,
847
1188
            wm,
848
1188
        );
849

            
850
1188
        let available_width = avail_end - avail_start;
851

            
852
1188
        if available_width >= total_cross {
853
            // +spec:floats:449158 - left float positioned at line-left, content flows on right
854
            // Found space that fits
855
1144
            if float_type == LayoutFloat::Left {
856
                // +spec:writing-modes:84bcba - floats positioned at line-left / line-right
857
                // Position at line-left (avail_start)
858
792
                break avail_start + margin.cross_start(wm);
859
            } else {
860
                // Position at line-right (avail_end - size)
861
352
                break avail_end - total_cross + margin.cross_start(wm);
862
            }
863
44
        }
864

            
865
        // top is moved lower than earlier float's bottom (outer edge / margin box bottom)
866
        // Not enough space at this Y, move down past the lowest overlapping float's margin box bottom
867
44
        let next_main = float_ctx
868
44
            .floats
869
44
            .iter()
870
44
            .filter(|f| {
871
44
                let f_main_start = f.rect.origin.main(wm) - f.margin.main_start(wm);
872
44
                let f_main_end = f_main_start + f.rect.size.main(wm)
873
44
                    + f.margin.main_start(wm) + f.margin.main_end(wm);
874
44
                f_main_end > main_start && f_main_start < main_start + total_main
875
44
            })
876
44
            .map(|f| f.rect.origin.main(wm) + f.rect.size.main(wm) + f.margin.main_end(wm))
877
44
            .max_by(|a, b| a.partial_cmp(b).unwrap_or(std::cmp::Ordering::Equal));
878

            
879
44
        if let Some(next) = next_main {
880
44
            main_start = next;
881
44
        } else {
882
            // No overlapping floats found, use current position anyway
883
            if float_type == LayoutFloat::Left {
884
                break avail_start + margin.cross_start(wm);
885
            } else {
886
                break avail_end - total_cross + margin.cross_start(wm);
887
            }
888
        }
889
    };
890

            
891
1144
    LogicalRect {
892
1144
        origin: LogicalPosition::from_main_cross(
893
1144
            main_start + margin.main_start(wm),
894
1144
            cross_start,
895
1144
            wm,
896
1144
        ),
897
1144
        size,
898
1144
    }
899
1144
}
900

            
901
// Block Formatting Context (CSS 2.2 § 9.4.1)
902

            
903
/// Lays out a Block Formatting Context (BFC).
904
///
905
/// This is the corrected, architecturally-sound implementation. It solves the
906
/// "chicken-and-egg" problem by performing its own two-pass layout:
907
///
908
/// 1. **Sizing Pass:** It first iterates through its children and triggers their layout recursively
909
///    by calling `calculate_layout_for_subtree`. This ensures that the `used_size` property of each
910
///    child is correctly populated.
911
///
912
/// 2. **Positioning Pass:** It then iterates through the children again. Now that each child has a
913
///    valid size, it can apply the standard block-flow logic: stacking them vertically and
914
///    advancing a "pen" by each child's outer height.
915
///
916
/// # Margin Collapsing Architecture
917
///
918
/// CSS 2.1 Section 8.3.1 compliant margin collapsing:
919
///
920
/// ```text
921
/// layout_bfc()
922
///   ├─ Check parent border/padding blockers
923
///   ├─ For each child:
924
///   │   ├─ Check child border/padding blockers
925
///   │   ├─ is_first_child?
926
///   │   │   └─ Check parent-child top collapse
927
///   │   ├─ Sibling collapse?
928
///   │   │   └─ advance_pen_with_margin_collapse()
929
///   │   │       └─ collapse_margins(prev_bottom, curr_top)
930
///   │   ├─ Position child
931
///   │   ├─ is_empty_block()?
932
///   │   │   └─ Collapse own top+bottom margins (collapse through)
933
///   │   └─ Save bottom margin for next sibling
934
///   └─ Check parent-child bottom collapse
935
/// ```
936
///
937
/// **Collapsing Rules:**
938
///
939
/// - Sibling margins: Adjacent vertical margins collapse to max (or sum if mixed signs)
940
/// - Parent-child: First child's top margin can escape parent (if no border/padding)
941
/// - Parent-child: Last child's bottom margin can escape parent (if no border/padding/height)
942
/// - Empty blocks: Top+bottom margins collapse with each other, then with siblings
943
/// - Blockers: Border, padding, inline content, or new BFC prevents collapsing
944
///
945
/// This approach is compliant with the CSS visual formatting model and works within
946
/// the constraints of the existing layout engine architecture.
947
// +spec:display-property:f38f52 - BFC handles normal flow, relative positioning offsets, and float extraction (CSS 2.2 § 9.8)
948
18965
fn layout_bfc<T: ParsedFontTrait>(
949
18965
    ctx: &mut LayoutContext<'_, T>,
950
18965
    tree: &mut LayoutTree,
951
18965
    text_cache: &mut crate::font_traits::TextLayoutCache,
952
18965
    node_index: usize,
953
18965
    constraints: &LayoutConstraints,
954
18965
    float_cache: &mut HashMap<usize, FloatingContext>,
955
18965
) -> Result<BfcLayoutResult> {
956
18965
    let node = tree
957
18965
        .get(node_index)
958
18965
        .ok_or(LayoutError::InvalidTree)?
959
18965
        .clone();
960
    // +spec:block-formatting-context:4f4ff6 - writing-mode determines block flow direction (main axis) for ordering block-level boxes in BFC
961
18965
    let writing_mode = constraints.writing_mode;
962
18965
    let mut output = LayoutOutput::default();
963

            
964
18965
    debug_info!(
965
17381
        ctx,
966
17381
        "\n[layout_bfc] ENTERED for node_index={}, children.len()={}, incoming_bfc_state={}",
967
        node_index,
968
17381
        tree.children(node_index).len(),
969
17381
        constraints.bfc_state.is_some()
970
    );
971

            
972
    // Initialize FloatingContext for this BFC
973
    //
974
    // We always recalculate float positions in this pass, but we'll store them in the cache
975
    // so that subsequent layout passes (for auto-sizing) have access to the positioned floats
976
18965
    let mut float_context = FloatingContext::default();
977

            
978
    // +spec:containing-block:42b75f - Block element establishes containing block for inline content (IFC)
979
    // Calculate this node's content-box size for use as containing block for children
980
    // CSS 2.2 § 10.1: The containing block for in-flow children is formed by the
981
    // content edge of the parent's content box.
982
    //
983
    // We use constraints.available_size directly as this already represents the
984
    // content-box available to this node (set by parent). For nodes with explicit
985
    // sizes, used_size contains the border-box which we convert to content-box.
986
    //
987
    // NOTE(writing-modes): The containing block size uses physical width/height.
988
    // In vertical writing modes, the block progression direction is horizontal,
989
    // so the "available width" for children maps to the physical height of
990
    // the containing block. The main_pen variable below tracks block progression
991
    // using logical main-axis coordinates; the WritingModeContext in constraints
992
    // determines how main/cross map to physical x/y via from_main_cross().
993
    // +spec:inline-block:17944a - orthogonal flow roots get infinite available inline space here (not yet detected)
994
    // +spec:inline-block:a60e22 - other layout models pass through infinite inline space to contained block containers
995
18965
    let mut children_containing_block_size = if let Some(used_size) = node.used_size {
996
        // Node has used_size (border-box) - convert to content-box.
997
        // For auto-height containers, the pre-layout `used_size.height` is a
998
        // placeholder (calculate_used_size_for_node returns 0 for block-level
999
        // auto-height; apply_content_based_height resolves it after children lay out).
        // In that window, `constraints.available_size.height` holds the containing
        // block's height — the value children should use as their own containing
        // block for percentage-height resolution and indefinite-height semantics.
14299
        let inner = node.box_props.inner_size(used_size, writing_mode);
14299
        let height_is_auto = tree
14299
            .warm(node_index)
14299
            .map(|w| w.computed_style.height.is_none())
14299
            .unwrap_or(true);
14299
        if height_is_auto {
11021
            LogicalSize::new(inner.width, constraints.available_size.height)
        } else {
3278
            inner
        }
    } else {
        // No used_size yet - use available_size directly (this is already content-box
        // when coming from parent's layout constraints)
4666
        constraints.available_size
    };
    // +spec:overflow:ffe6f7 - scrollbar space subtracted from containing block per spec §11.1.1
    // Reserve space for vertical scrollbar when appropriate.
    //
    // - overflow: scroll  → ALWAYS reserve (CSS spec: scrollbar always shown)
    // - overflow: auto    → Reserve ONLY when a previous pass / the anti-jitter
    //   merge (`merge_scrollbar_info`) already determined a scrollbar is needed.
    //   On the very first pass the node has no scrollbar_info yet, so no space
    //   is reserved.  After `compute_scrollbar_info` detects overflow it sets
    //   `reflow_needed_for_scrollbars = true`, triggering a second pass where
    //   `node.scrollbar_info.needs_vertical == true` and space IS reserved.
    //   The merge uses `||` (keep once detected), preventing cross-frame jitter.
18965
    let scrollbar_reservation = node
18965
        .dom_node_id
18965
        .map(|dom_id| {
18961
            let styled_node_state = ctx
18961
                .styled_dom
18961
                .styled_nodes
18961
                .as_container()
18961
                .get(dom_id)
18961
                .map(|s| s.styled_node_state.clone())
18961
                .unwrap_or_default();
18961
            let overflow_y =
18961
                crate::solver3::getters::get_overflow_y(ctx.styled_dom, dom_id, &styled_node_state);
            use azul_css::props::layout::LayoutOverflow;
18961
            match overflow_y.unwrap_or_default() {
                LayoutOverflow::Scroll => {
                    crate::solver3::getters::get_layout_scrollbar_width_px(ctx, dom_id, &styled_node_state)
                }
                LayoutOverflow::Auto => {
                    let already_needs = tree.warm(node_index)
                        .and_then(|w| w.scrollbar_info.as_ref())
                        .map(|s| s.needs_vertical)
                        .unwrap_or(false);
                    if already_needs {
                        crate::solver3::getters::get_layout_scrollbar_width_px(ctx, dom_id, &styled_node_state)
                    } else {
                        0.0
                    }
                }
18961
                _ => 0.0,
            }
18961
        })
18965
        .unwrap_or(0.0);
18965
    if scrollbar_reservation > 0.0 {
        children_containing_block_size.width =
            (children_containing_block_size.width - scrollbar_reservation).max(0.0);
18965
    }
    // === Pass 1: Pre-compute child sizes (restored two-pass BFC) ===
    //
    // Inspired by Taffy's two-pass approach: first measure, then position.
    //
    // This was removed in commit 1a3e5850 and replaced with a single-pass approach
    // that computed sizes just-in-time during positioning. The single-pass approach
    // caused regression 8e092a2e because positioning decisions (margin collapsing,
    // float clearance, available width after floats) depend on knowing ALL sibling
    // sizes upfront, not just the ones visited so far.
    //
    // With the per-node cache (§9.1-§9.2), the re-added Pass 1 is efficient:
    // - Each child subtree is computed once and stored in NodeCache
    // - Pass 2 positioning reads sizes from tree nodes (used_size set by Pass 1)
    // - When calculate_layout_for_subtree recurses into children after layout_bfc
    //   returns, it hits the per-node cache (same available_size) — O(1) per child.
    //
    // Performance: O(n) for the tree. No double-computation thanks to caching.
    {
18965
        let mut temp_positions: super::PositionVec = Vec::new();
18965
        let mut temp_scrollbar_reflow = false;
18965
        let bfc_children = tree.children(node_index).to_vec();
        // [g147c az-web-lift DIAG] layout_bfc Pass-1 child-sizing loop: record bfc_children.len per parent
        // node (0x60A00+slot). If body shows len=2 but the divs never get the per-child "sized" marker
        // (0x60A40+childslot) below → the loop skips them; if they DO get it but layout_formatting_context
        // (0x609A0) stays unset → calculate(child,ComputeSize) cache-hit (vs 0x60A60 miss-flag in cache.rs).
        #[cfg(feature = "web_lift")]
        unsafe { crate::az_mark(((0x60A00 + (node_index & 7) * 4)) as u32, (bfc_children.len() as u32 | 0xC0DE0000) as u32); }
40808
        for &child_index in &bfc_children {
21843
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
21843
            let child_dom_id = child_node.dom_node_id;
            // +spec:positioning:447b06 - Absolute positioning pulls element out of flow, skip from normal layout
            // +spec:positioning:77a2d2 - Absolutely positioned children are ignored for auto height
            // +spec:positioning:b47ac2 - Only normal flow children taken into account for auto height
            // Skip absolutely/fixed positioned children — they're laid out separately
            // +spec:positioning:c7e5c5 - out-of-flow elements ignored for word boundary / hyphenation
            // +spec:positioning:7dd6d1 - Absolutely positioned boxes are taken out of the normal flow (no impact on later siblings, no margin collapsing)
21843
            let position_type = get_position_type(ctx.styled_dom, child_dom_id);
21843
            if position_type == LayoutPosition::Absolute || position_type == LayoutPosition::Fixed {
6561
                continue;
15282
            }
            // Compute the child's full subtree layout with temporary positions.
            // Position (0,0) is intentionally wrong — Pass 1 only cares about sizing.
            // The correct positions are determined in Pass 2 below.
            // [g147c] this child IS reached by Pass-1 sizing (per-child slot).
            #[cfg(feature = "web_lift")]
            unsafe { crate::az_mark(((0x60A40 + (child_index & 7) * 4)) as u32, (0xC0DE0000 | (child_index as u32 & 0xffff)) as u32); }
15282
            crate::solver3::cache::calculate_layout_for_subtree(
15282
                ctx,
15282
                tree,
15282
                text_cache,
15282
                child_index,
15282
                LogicalPosition::zero(),
15282
                children_containing_block_size,
15282
                &mut temp_positions,
15282
                &mut temp_scrollbar_reflow,
15282
                float_cache,
15282
                crate::solver3::cache::ComputeMode::ComputeSize,
            )?;
        }
    }
    // +spec:block-formatting-context:98b633 - CSS 2.2 § 9.4.1: boxes laid out vertically, margins collapse
    // === Pass 2: Position children using known sizes ===
    //
    // All children now have used_size set from Pass 1. This pass handles:
    // - Margin collapsing (parent-child + sibling-sibling)
    // - Float positioning and clearance
    // - Normal flow block positioning
18965
    let mut main_pen = 0.0f32;
18965
    let mut max_cross_size = 0.0f32;
    // Track escaped margins separately from content-box height
    // CSS 2.2 § 8.3.1: Escaped margins don't contribute to parent's content-box height,
    // but DO affect sibling positioning within the parent
18965
    let mut total_escaped_top_margin = 0.0f32;
    // Track all inter-sibling margins (collapsed) - these are also not part of content height
18965
    let mut total_sibling_margins = 0.0f32;
    // Margin collapsing state
18965
    let mut last_margin_bottom = 0.0f32;
18965
    let mut is_first_child = true;
18965
    let mut first_child_index: Option<usize> = None;
18965
    let mut last_child_index: Option<usize> = None;
    // Parent's own margins (for escape calculation)
18965
    let node_bp = node.box_props.unpack();
18965
    let parent_margin_top = node_bp.margin.main_start(writing_mode);
18965
    let parent_margin_bottom = node_bp.margin.main_end(writing_mode);
    // margins do not collapse across formatting context boundaries: an independent
    // BFC (float, overflow != visible, display: flex/grid, etc.) isolates its
    // children's margins. The DOM root is NOT a BFC boundary for this purpose —
    // its first child's margin still collapses through it (then gets absorbed at
    // the root, since there's no grandparent to escape to).
18965
    let establishes_own_bfc = establishes_new_bfc(ctx, &node, tree.cold(node_index));
18965
    let is_bfc_root = node.parent.is_none() || establishes_own_bfc;
    // parent_has_*_blocker inhibits parent-child margin collapse per CSS 2.2 §8.3.1.
    // An explicit border/padding blocks, and an independent BFC blocks, but the
    // root on its own does not.
18965
    let parent_has_top_blocker = establishes_own_bfc
6830
        || has_margin_collapse_blocker(&node_bp, writing_mode, true);
18965
    let parent_has_bottom_blocker = establishes_own_bfc
6830
        || has_margin_collapse_blocker(&node_bp, writing_mode, false);
    // Track accumulated top margin for first-child escape
18965
    let mut accumulated_top_margin = 0.0f32;
18965
    let mut top_margin_resolved = false;
    // Track if first child's margin escaped (for return value)
18965
    let mut top_margin_escaped = false;
    // Track if we have any actual content (non-empty blocks)
18965
    let mut has_content = false;
    // +spec:display-property:9f6e18 - BFC dispatches normal flow, floats, and relative positioning (CSS 2.2 §9.8)
18965
    let pos_children = tree.children(node_index).to_vec();
40808
    for &child_index in &pos_children {
21843
        let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
21843
        let child_dom_id = child_node.dom_node_id;
        // +spec:floats:2cec1b - 'position' and 'float' determine the positioning algorithm
        // +spec:positioning:dccad6 - floats only apply to non-absolutely-positioned boxes
21843
        let position_type = get_position_type(ctx.styled_dom, child_dom_id);
21843
        if position_type == LayoutPosition::Absolute || position_type == LayoutPosition::Fixed {
6561
            continue;
15282
        }
        // +spec:floats:2cec1b - float property determines positioning algorithm (float path)
        // +spec:floats:f6c0b2 - floats only processed in BFC; other formatting contexts (flex/grid) inhibit floating
        // Check if this child is a float - if so, position it at current main_pen
15282
        let is_float = if let Some(node_id) = child_dom_id {
15190
            let float_type = get_float_property(ctx.styled_dom, Some(node_id));
15190
            if float_type != LayoutFloat::None {
                // Calculate float size just-in-time if not already computed
26
                let float_size = match child_node.used_size {
26
                    Some(size) => size,
                    None => {
                        let intrinsic = tree.warm(child_index).and_then(|w| w.intrinsic_sizes).unwrap_or_default();
                        let child_bp = child_node.box_props.unpack();
                        let computed_size = crate::solver3::sizing::calculate_used_size_for_node(
                            ctx.styled_dom,
                            child_dom_id,
                            &children_containing_block_size,
                            intrinsic,
                            &child_bp,
                            &ctx.viewport_size,
                        )?;
                        if let Some(node_mut) = tree.get_mut(child_index) {
                            node_mut.used_size = Some(computed_size);
                        }
                        computed_size
                    }
                };
                // Re-borrow after potential mutation
26
                let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
26
                let child_bp2 = child_node.box_props.unpack();
26
                let float_margin = &child_bp2.margin;
                // +spec:floats:d0d163 - clear on floats adds constraint #10: float top below cleared floats' bottom
                // +spec:floats:7adb9d - Clear on floats: constraint #10, top outer edge must be below earlier cleared floats
26
                let float_clear = get_clear_property(ctx.styled_dom, Some(node_id));
26
                let float_y = if float_clear != LayoutClear::None {
                    float_context.clearance_offset(float_clear, main_pen + last_margin_bottom, writing_mode)
                } else {
                    // +spec:floats:ef96cb - Float margins never collapse with adjacent margins
                    // CSS 2.2 § 9.5: Float margins don't collapse with any other margins.
26
                    main_pen + last_margin_bottom
                };
26
                debug_info!(
26
                    ctx,
26
                    "[layout_bfc] Positioning float: index={}, type={:?}, size={:?}, at Y={} \
26
                     (main_pen={} + last_margin={})",
                    child_index,
                    float_type,
                    float_size,
                    float_y,
                    main_pen,
                    last_margin_bottom
                );
                // Position the float at the CURRENT main_pen + last margin (respects DOM order!)
26
                let float_rect = position_float(
26
                    &float_context,
26
                    float_type,
26
                    float_size,
26
                    float_margin,
                    // Include last_margin_bottom since float margins don't collapse!
26
                    float_y,
26
                    constraints.available_size.cross(writing_mode),
26
                    writing_mode,
                );
26
                debug_info!(ctx, "[layout_bfc] Float positioned at: {:?}", float_rect);
                // Add to float context BEFORE positioning next element
26
                float_context.add_float(float_type, float_rect, *float_margin);
                // Store position in output
26
                output.positions.insert(child_index, float_rect.origin);
26
                debug_info!(
26
                    ctx,
26
                    "[layout_bfc] *** FLOAT POSITIONED: child={}, main_pen={} (unchanged - floats \
26
                     don't advance pen)",
                    child_index,
                    main_pen
                );
                // Floats are taken out of normal flow - DON'T advance main_pen
                // Continue to next child
26
                continue;
15164
            }
15164
            false
        } else {
92
            false
        };
        // Early exit for floats (already handled above)
15256
        if is_float {
            continue;
15256
        }
        // From here: normal flow (non-float) children only
        // Track first and last in-flow children for parent-child collapse
15256
        if first_child_index.is_none() {
13957
            first_child_index = Some(child_index);
13957
        }
15256
        last_child_index = Some(child_index);
        // Calculate child's used_size just-in-time if not already computed
        // This replaces the old "Pass 1" that recursively laid out grandchildren with wrong positions
15256
        let child_size = match child_node.used_size {
15256
            Some(size) => size,
            None => {
                // Calculate size without recursive layout
                let intrinsic = tree.warm(child_index).and_then(|w| w.intrinsic_sizes).unwrap_or_default();
                let child_used_size = crate::solver3::sizing::calculate_used_size_for_node(
                    ctx.styled_dom,
                    child_dom_id,
                    &children_containing_block_size,
                    intrinsic,
                    &child_node.box_props.unpack(),
                    &ctx.viewport_size,
                )?;
                // Update the node with computed size (we need to re-borrow mutably)
                if let Some(node_mut) = tree.get_mut(child_index) {
                    node_mut.used_size = Some(child_used_size);
                }
                child_used_size
            }
        };
        // Re-borrow child_node after potential mutation
15256
        let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
15256
        let child_bp = child_node.box_props.unpack();
15256
        let child_margin = &child_bp.margin;
15256
        debug_info!(
15212
            ctx,
15212
            "[layout_bfc] Child {} margin from box_props: top={}, right={}, bottom={}, left={}",
            child_index,
            child_margin.top,
            child_margin.right,
            child_margin.bottom,
            child_margin.left
        );
        // +spec:block-formatting-context:0f802c - margins use containing block's writing mode for collapsing/auto expansion in orthogonal flows
15256
        let child_own_margin_top = child_margin.main_start(writing_mode);
15256
        let child_own_margin_bottom = child_margin.main_end(writing_mode);
        // CSS 2.2 § 8.3.1: If a child has no top blocker (no padding/border) and its
        // own BFC layout produced an escaped_top_margin, that margin represents the
        // collapsed value of (child's margin, child's first child's margin, ...).
        // Use it for sibling collapse instead of the child's own margin.
15256
        let child_escaped_top = if !has_margin_collapse_blocker(&child_bp, writing_mode, true) {
14768
            tree.warm(child_index).and_then(|w| w.escaped_top_margin)
488
        } else { None };
15256
        let child_escaped_bottom = if !has_margin_collapse_blocker(&child_bp, writing_mode, false) {
14771
            tree.warm(child_index).and_then(|w| w.escaped_bottom_margin)
485
        } else { None };
15256
        let child_margin_top = child_escaped_top.unwrap_or(child_own_margin_top);
15256
        let child_margin_bottom = child_escaped_bottom.unwrap_or(child_own_margin_bottom);
15256
        debug_info!(
15212
            ctx,
15212
            "[layout_bfc] Child {} final margins: margin_top={}, margin_bottom={}",
            child_index,
            child_margin_top,
            child_margin_bottom
        );
        // Check if this child has border/padding that prevents margin collapsing
15256
        let child_has_top_blocker =
15256
            has_margin_collapse_blocker(&child_bp, writing_mode, true);
15256
        let child_has_bottom_blocker =
15256
            has_margin_collapse_blocker(&child_bp, writing_mode, false);
        // +spec:floats:dc195a - Clear property only applies to block-level elements (CSS 2.2 § 9.5.2)
        // Check for clear property FIRST - clearance affects whether element is considered empty
        // CSS 2.2 § 9.5.2: "Clearance inhibits margin collapsing"
        // An element with clearance is NOT empty even if it has no content
15256
        let child_clear = if let Some(node_id) = child_dom_id {
15164
            get_clear_property(ctx.styled_dom, Some(node_id))
        } else {
92
            LayoutClear::None
        };
15256
        debug_info!(
15212
            ctx,
15212
            "[layout_bfc] Child {} clear property: {:?}",
            child_index,
            child_clear
        );
        // PHASE 1: Empty Block Detection & Self-Collapse
15256
        let is_empty = is_empty_block(tree, child_index);
        // Handle empty blocks FIRST (they collapse through and don't participate in layout)
        // EXCEPTION: Elements with clear property are NOT skipped even if empty!
        // CSS 2.2 § 9.5.2: Clear property affects positioning even for empty elements
15256
        if is_empty
2
            && !child_has_top_blocker
2
            && !child_has_bottom_blocker
2
            && child_clear == LayoutClear::None
        {
            // Empty block: collapse its own top and bottom margins FIRST
2
            let self_collapsed = collapse_margins(child_margin_top, child_margin_bottom);
            // Then collapse with previous margin (sibling or parent)
2
            if is_first_child {
1
                is_first_child = false;
                // Empty first child: its collapsed margin can escape with parent's
1
                if !parent_has_top_blocker {
1
                    accumulated_top_margin = collapse_margins(parent_margin_top, self_collapsed);
1
                } else {
                    // Parent has blocker: add margins
                    if accumulated_top_margin == 0.0 {
                        accumulated_top_margin = parent_margin_top;
                    }
                    main_pen += accumulated_top_margin + self_collapsed;
                    top_margin_resolved = true;
                    accumulated_top_margin = 0.0;
                }
1
                last_margin_bottom = self_collapsed;
1
            } else {
1
                // Empty sibling: collapse with previous sibling's bottom margin
1
                last_margin_bottom = collapse_margins(last_margin_bottom, self_collapsed);
1
            }
            // Skip positioning and pen advance (empty has no visual presence)
2
            continue;
15254
        }
        // From here on: non-empty blocks only (or empty blocks with clear property)
        // Apply clearance if needed
        // +spec:floats:148ee6 - clear:left pushes element below float; clearance added above top margin
        // CSS 2.2 § 9.5.2: Clearance inhibits margin collapsing.
        //
        // Per CSS 2.2 § 9.5.2, the clearance computation works as follows:
        // 1. Compute the "hypothetical position" — where the border edge would be
        //    with normal margin collapsing (as if clear:none).
        // 2. If the hypothetical position is NOT past the relevant floats,
        //    clearance is introduced and the border edge is placed at float bottom.
        // 3. The final border edge = max(float_bottom, hypothetical_position).
        //
        // This means child_margin_top is already accounted for in the hypothetical
        // position and must NOT be added again after clearance positions main_pen.
15254
        let clearance_applied = if child_clear != LayoutClear::None {
5
            let hypothetical = main_pen + collapse_margins(last_margin_bottom, child_margin_top);
5
            let cleared_position =
5
                float_context.clearance_offset(child_clear, hypothetical, writing_mode);
5
            debug_info!(
5
                ctx,
5
                "[layout_bfc] Child {} clearance check: cleared_position={}, hypothetical={} (main_pen={} + collapse({}, {}))",
                child_index,
                cleared_position,
                hypothetical,
                main_pen,
                last_margin_bottom,
                child_margin_top
            );
5
            if cleared_position > hypothetical {
4
                debug_info!(
4
                    ctx,
4
                    "[layout_bfc] Applying clearance: child={}, clear={:?}, old_pen={}, new_pen={}",
                    child_index,
                    child_clear,
                    main_pen,
                    cleared_position
                );
4
                main_pen = cleared_position;
4
                true // Signal that clearance was applied
            } else {
1
                false
            }
        } else {
15249
            false
        };
        // PHASE 2: Parent-Child Top Margin Escape (First Child)
        //
        // CSS 2.2 § 8.3.1: "The top margin of a box is adjacent to the top margin of its first
        // in-flow child if the box has no top border, no top padding, and the child has no
        // clearance." CSS 2.2 § 9.5.2: "Clearance inhibits margin collapsing"
15254
        if is_first_child {
13956
            is_first_child = false;
            // Clearance prevents collapse (acts as invisible blocker)
13956
            if clearance_applied {
                // Clearance inhibits all margin collapsing for this element
                // The clearance has already positioned main_pen at the correct
                // border-edge position (= max(float_bottom, hypothetical)).
                // The hypothetical already includes child_margin_top via
                // collapse_margins, so we must NOT add it again here.
3
                debug_info!(
3
                    ctx,
3
                    "[layout_bfc] First child {} with CLEARANCE: no collapse, child_margin={}, \
3
                     main_pen={}",
                    child_index,
                    child_margin_top,
                    main_pen
                );
13953
            } else if !parent_has_top_blocker {
                // Margin Escape Case
                //
                // CSS 2.2 § 8.3.1: "The top margin of an in-flow block element collapses with
                // its first in-flow block-level child's top margin if the element has no top
                // border, no top padding, and the child has no clearance."
                //
                // When margins collapse, they "escape" upward through the parent to be resolved
                // in the grandparent's coordinate space. This is critical for understanding the
                // coordinate system separation:
                //
                // Example:
                // <body padding=20>
                //  <div margin=0>
                //      <div margin=30></div>
                //  </div>
                // </body>
                //
                //   - Middle div (our parent) has no padding → margins can escape
                //   - Inner div's 30px margin collapses with middle div's 0px margin = 30px
                //   - This 30px margin "escapes" to be handled by body's BFC
                //   - Body positions middle div at Y=30 (relative to body's content-box)
                //   - Middle div's content-box height does NOT include the escaped 30px
                //   - Inner div is positioned at Y=0 in middle div's content-box
                //
                // **NOTE**: This is a subtle but critical distinction in coordinate systems:
                //
                //   - Parent's margin belongs to grandparent's coordinate space
                //   - Child's margin (when escaped) also belongs to grandparent's coordinate space
                //   - They collapse BEFORE entering this BFC's coordinate space
                //   - We return the collapsed margin so grandparent can position parent correctly
                //
                // **NOTE**: Child's own blocker status (padding/border) is IRRELEVANT for
                // parent-child  collapse. The child may have padding that prevents
                // collapse with ITS OWN  children, but this doesn't prevent its
                // margin from escaping  through its parent.
                //
                // **NOTE**: Previously, we incorrectly added parent_margin_top to main_pen in
                //  the blocked case, which double-counted the margin by mixing
                //  coordinate systems. The parent's margin is NEVER in our (the
                //  parent's content-box) coordinate system!
                //
                // We collapse the parent's margin with the child's margin.
                // This combined margin is what "escapes" to the grandparent.
                // The grandparent uses this to position the parent.
                //
                // Effectively, we are saying "The parent starts here, but its effective
                // top margin is now max(parent_margin, child_margin)".
2507
                accumulated_top_margin = collapse_margins(parent_margin_top, child_margin_top);
2507
                top_margin_resolved = true;
2507
                top_margin_escaped = true;
                // Track escaped margin so it gets subtracted from content-box height
                // The escaped margin is NOT part of our content-box - it belongs to our
                // parent's parent
2507
                total_escaped_top_margin = accumulated_top_margin;
                // Position child at pen (no margin applied - it escaped!)
2507
                debug_info!(
2507
                    ctx,
2507
                    "[layout_bfc] First child {} margin ESCAPES: parent_margin={}, \
2507
                     child_margin={}, collapsed={}, total_escaped={}",
                    child_index,
                    parent_margin_top,
                    child_margin_top,
                    accumulated_top_margin,
                    total_escaped_top_margin
                );
            } else {
                // Margin Blocked Case
                //
                // CSS 2.2 § 8.3.1: "no top padding and no top border" required for collapse.
                // When padding or border exists, margins do NOT collapse and exist in different
                // coordinate spaces.
                //
                // CRITICAL COORDINATE SYSTEM SEPARATION:
                //
                //   This is where the architecture becomes subtle. When layout_bfc() is called:
                //   1. We are INSIDE the parent's content-box coordinate space (main_pen starts at
                //      0)
                //   2. The parent's own margin was ALREADY RESOLVED by the grandparent's BFC
                //   3. The parent's margin is in the grandparent's coordinate space, not ours
                //   4. We NEVER reference the parent's margin in this BFC - it's outside our scope
                //
                // Example:
                //
                // <body padding=20>
                //   <div margin=30 padding=20>
                //      <div margin=30></div>
                //   </div>
                // </body>
                //
                //   - Middle div has padding=20 → blocker exists, margins don't collapse
                //   - Body's BFC positions middle div at Y=30 (middle div's margin, in body's
                //     space)
                //   - Middle div's BFC starts at its content-box (after the padding)
                //   - main_pen=0 at the top of middle div's content-box
                //   - Inner div has margin=30 → we add 30 to main_pen (in OUR coordinate space)
                //   - Inner div positioned at Y=30 (relative to middle div's content-box)
                //   - Absolute position: 20 (body padding) + 30 (middle margin) + 20 (middle
                //     padding) + 30 (inner margin) = 100px
                //
                // **NOTE**: Previous code incorrectly added parent_margin_top to main_pen here:
                //
                //     - main_pen += parent_margin_top;  // WRONG! Mixes coordinate systems
                //     - main_pen += child_margin_top;
                //
                //   This caused the "double margin" bug where margins were applied twice:
                //
                //   - Once by grandparent positioning parent (correct)
                //   - Again inside parent's BFC (INCORRECT - wrong coordinate system)
                //
                //   The parent's margin belongs to GRANDPARENT's coordinate space and was already
                //   used to position the parent. Adding it again here is like adding feet to
                //   meters.
                //
                //   We ONLY add the child's margin in our (parent's content-box) coordinate space.
                //   The parent's margin is irrelevant to us - it's outside our scope.
11446
                main_pen += child_margin_top;
11446
                debug_info!(
11402
                    ctx,
11402
                    "[layout_bfc] First child {} BLOCKED: parent_has_blocker={}, advanced by \
11402
                     child_margin={}, main_pen={}",
                    child_index,
                    parent_has_top_blocker,
                    child_margin_top,
                    main_pen
                );
            }
        } else {
            // Not first child: handle sibling collapse
            // CSS 2.2 § 8.3.1 Rule 1: "Vertical margins of adjacent block boxes in the normal flow
            // collapse" CSS 2.2 § 9.5.2: "Clearance inhibits margin collapsing"
            // Resolve accumulated top margin if not yet done (for parent's first in-flow child)
1298
            if !top_margin_resolved {
132
                main_pen += accumulated_top_margin;
132
                top_margin_resolved = true;
132
                debug_info!(
132
                    ctx,
132
                    "[layout_bfc] RESOLVED top margin for node {} at sibling {}: accumulated={}, \
132
                     main_pen={}",
                    node_index,
                    child_index,
                    accumulated_top_margin,
                    main_pen
                );
1166
            }
1298
            if clearance_applied {
                // Clearance has already positioned main_pen at the correct
                // border-edge = max(float_bottom, hypothetical). The hypothetical
                // already includes collapse_margins(last_margin_bottom, child_margin_top),
                // so we must NOT add child_margin_top again here.
1
                debug_info!(
1
                    ctx,
1
                    "[layout_bfc] Child {} with CLEARANCE: no collapse with sibling, \
1
                     child_margin_top={}, main_pen={}",
                    child_index,
                    child_margin_top,
                    main_pen
                );
            } else {
                // Sibling Margin Collapse
                //
                // CSS 2.2 § 8.3.1: "Vertical margins of adjacent block boxes in the normal
                // flow collapse." The collapsed margin is the maximum of the two margins.
                //
                // IMPORTANT: Sibling margins ARE part of the parent's content-box height!
                //
                // Unlike escaped margins (which belong to grandparent's space), sibling margins
                // are the space BETWEEN children within our content-box.
                //
                // Example:
                //
                // <div>
                //  <div margin-bottom=30></div>
                //  <div margin-top=40></div>
                // </div>
                //
                //   - First child ends at Y=100 (including its content + margins)
                //   - Collapsed margin = max(30, 40) = 40px
                //   - Second child starts at Y=140 (100 + 40)
                //   - Parent's content-box height includes this 40px gap
                //
                // We track total_sibling_margins for debugging, but NOTE: we do **not**
                // subtract these from content-box height! They are part of the layout space.
                //
                // Previously we subtracted total_sibling_margins from content-box height:
                //
                //   content_box_height = main_pen - total_escaped_top_margin -
                // total_sibling_margins;
                //
                // This was wrong because sibling margins are between boxes (part of content),
                // not outside boxes (like escaped margins).
1297
                let collapsed = collapse_margins(last_margin_bottom, child_margin_top);
1297
                main_pen += collapsed;
1297
                total_sibling_margins += collapsed;
1297
                debug_info!(
1297
                    ctx,
1297
                    "[layout_bfc] Sibling collapse for child {}: last_margin_bottom={}, \
1297
                     child_margin_top={}, collapsed={}, main_pen={}, total_sibling_margins={}",
                    child_index,
                    last_margin_bottom,
                    child_margin_top,
                    collapsed,
                    main_pen,
                    total_sibling_margins
                );
            }
        }
        // Position child (non-empty blocks only reach here)
        //
        // +spec:block-formatting-context:1dada5 - Normal flow boxes in BFC touch containing block edge
        // +spec:block-formatting-context:9f56cb - each box's left outer edge touches containing block left edge; new BFC may shrink due to floats
        // CSS 2.2 § 9.4.1: "In a block formatting context, each box's left outer edge touches
        // the left edge of the containing block (for right-to-left formatting, right edges touch).
        // This is true even in the presence of floats (although a box's line boxes may shrink
        // due to the floats), unless the box establishes a new block formatting context
        // (in which case the box itself may become narrower due to the floats)."
        //
        // +spec:block-formatting-context:3d2811 - Float overlap with normal flow element borders
        // +spec:display-property:796059 - BFC/replaced/table border box must not overlap float margin boxes; line boxes shorten around floats
        // +spec:floats:5214a6 - BFC/replaced/table border box must not overlap float margin boxes; shrink or clear below
        // CSS 2.2 § 9.5: "The border box of a table, a block-level replaced element, or an element
        // in the normal flow that establishes a new block formatting context (such as an element
        // with 'overflow' other than 'visible') must not overlap any floats in the same block
        // formatting context as the element itself."
        // +spec:floats:a29f70 - BFC roots, tables, and block-level replaced elements must not overlap float margin boxes
15254
        let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
15254
        let avoids_floats = establishes_new_bfc(ctx, child_node, tree.cold(child_index))
14894
            || is_block_level_replaced(ctx, child_node);
        // Query available space considering floats ONLY if child avoids floats
15254
        let (cross_start, cross_end, available_cross) = if avoids_floats {
            // New BFC / replaced / table: Must shrink or move down to avoid overlapping floats
360
            let child_cross_needed = child_size.cross(writing_mode);
360
            let bfc_cross = constraints.available_size.cross(writing_mode);
360
            let (mut start, mut end) = float_context.available_line_box_space(
360
                main_pen,
360
                main_pen + child_size.main(writing_mode),
360
                bfc_cross,
360
                writing_mode,
360
            );
360
            let mut available = end - start;
            // CSS 2.2 § 9.5: "If necessary, implementations should clear the said element
            // by placing it below any preceding floats, but may place it adjacent to such
            // floats if there is sufficient space."
360
            if available < child_cross_needed && !float_context.floats.is_empty() {
1
                let clear_to = float_context.floats.iter()
1
                    .filter(|f| {
1
                        let f_main_start = f.rect.origin.main(writing_mode) - f.margin.main_start(writing_mode);
1
                        let f_main_end = f_main_start + f.rect.size.main(writing_mode)
1
                            + f.margin.main_start(writing_mode) + f.margin.main_end(writing_mode);
1
                        f_main_end > main_pen && f_main_start < main_pen + child_size.main(writing_mode)
1
                    })
1
                    .map(|f| {
1
                        f.rect.origin.main(writing_mode) + f.rect.size.main(writing_mode)
1
                            + f.margin.main_end(writing_mode)
1
                    })
1
                    .fold(main_pen, f32::max);
1
                if clear_to > main_pen {
1
                    main_pen = clear_to;
1
                    let (s, e) = float_context.available_line_box_space(
1
                        main_pen,
1
                        main_pen + child_size.main(writing_mode),
1
                        bfc_cross,
1
                        writing_mode,
1
                    );
1
                    start = s;
1
                    end = e;
1
                    available = end - start;
1
                }
359
            }
360
            debug_info!(
316
                ctx,
316
                "[layout_bfc] Child {} avoids floats: shrinking to avoid floats, \
316
                 cross_range={}..{}, available_cross={}",
                child_index,
                start,
                end,
                available
            );
360
            (start, end, available)
        } else {
            // Normal flow: Overlaps floats, positioned at full width
            // Only the child's INLINE CONTENT (if any) wraps around floats
14894
            let start = 0.0;
14894
            let end = constraints.available_size.cross(writing_mode);
14894
            let available = end - start;
14894
            debug_info!(
14894
                ctx,
14894
                "[layout_bfc] Child {} is normal flow: overlapping floats at full width, \
14894
                 available_cross={}",
                child_index,
                available
            );
14894
            (start, end, available)
        };
        // Get child's margin, margin_auto, size, and formatting context
15254
        let (child_margin_cloned, child_margin_auto, child_used_size, is_inline_fc, child_dom_id_for_debug) = {
15254
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
15254
            let cbp = child_node.box_props.unpack();
15254
            (
15254
                cbp.margin.clone(),
15254
                cbp.margin_auto,
15254
                child_node.used_size.unwrap_or_default(),
15254
                child_node.formatting_context == FormattingContext::Inline,
15254
                child_node.dom_node_id,
15254
            )
        };
15254
        let child_margin = &child_margin_cloned;
15254
        debug_info!(
15210
            ctx,
15210
            "[layout_bfc] Child {} margin_auto: left={}, right={}, top={}, bottom={}",
            child_index,
            child_margin_auto.left,
            child_margin_auto.right,
            child_margin_auto.top,
            child_margin_auto.bottom
        );
15254
        debug_info!(
15210
            ctx,
15210
            "[layout_bfc] Child {} used_size: width={}, height={}",
            child_index,
            child_used_size.width,
            child_used_size.height
        );
        // Position child
        // For normal flow blocks (including IFCs): position at full width (cross_start = 0)
        // For BFC-establishing blocks: position in available space between floats
        //
        // CSS 2.2 § 10.3.3: If margin-left and margin-right are both auto,
        // their used values are equal, centering the element horizontally.
15254
        let (child_cross_pos, mut child_main_pos) = if avoids_floats {
            // BFC: Position in float-free space, but also check margin:auto centering.
            // A flex container or overflow:hidden box establishes a BFC (must avoid floats)
            // but can still be centered via margin:auto — these are independent concepts.
360
            let cross_pos = if child_margin_auto.left && child_margin_auto.right {
                let remaining = (available_cross - child_used_size.cross(writing_mode)).max(0.0);
                debug_info!(
                    ctx,
                    "[layout_bfc] Child {} BFC + margin:auto centering: available={}, size={}, offset={}",
                    child_index, available_cross, child_used_size.cross(writing_mode), remaining / 2.0
                );
                cross_start + remaining / 2.0
360
            } else if child_margin_auto.left {
                let remaining = (available_cross - child_used_size.cross(writing_mode) - child_margin.right).max(0.0);
                cross_start + remaining
            } else {
360
                cross_start + child_margin.cross_start(writing_mode)
            };
360
            (cross_pos, main_pen)
        } else {
            // Normal flow: Check for margin: auto centering
14894
            let available_cross = constraints.available_size.cross(writing_mode);
14894
            let child_cross_size = child_used_size.cross(writing_mode);
14894
            debug_info!(
14894
                ctx,
14894
                "[layout_bfc] Child {} centering check: available_cross={}, child_cross_size={}, margin_auto.left={}, margin_auto.right={}",
                child_index,
                available_cross,
                child_cross_size,
                child_margin_auto.left,
                child_margin_auto.right
            );
            // +spec:block-formatting-context:d52ce5 - auto margins resolved per containing block's writing mode for centering
            // +spec:width-calculation:0c5044 - auto margins center element on cross axis (respects writing mode)
            // +spec:width-calculation:25c2fc - §10.3.3: block-level margin auto centering and over-constrained resolution
            // +spec:width-calculation:ba691f - auto margins treated as zero when element overflows containing block (via .max(0.0) on remaining_space)
            // +spec:width-calculation:324e7e - both margin-left and margin-right auto => equal used values (centering)
            // CSS 2.2 § 10.3.3: If both margin-left and margin-right are auto,
            // center the element within the available space
14894
            let cross_pos = if child_margin_auto.left && child_margin_auto.right {
                // Center: (available - child_width) / 2
1
                let remaining_space = (available_cross - child_cross_size).max(0.0);
1
                debug_info!(
1
                    ctx,
1
                    "[layout_bfc] Child {} CENTERING: remaining_space={}, cross_pos={}",
                    child_index,
                    remaining_space,
1
                    remaining_space / 2.0
                );
1
                remaining_space / 2.0
14893
            } else if child_margin_auto.left {
                // Only left is auto: push element to the right
                let remaining_space = (available_cross - child_cross_size - child_margin.right).max(0.0);
                debug_info!(
                    ctx,
                    "[layout_bfc] Child {} margin-left:auto only, pushing right: remaining_space={}",
                    child_index,
                    remaining_space
                );
                remaining_space
14893
            } else if child_margin_auto.right {
                // Only right is auto: element stays at left with its margin
                debug_info!(
                    ctx,
                    "[layout_bfc] Child {} margin-right:auto only, using left margin={}",
                    child_index,
                    child_margin.cross_start(writing_mode)
                );
                child_margin.cross_start(writing_mode)
            } else {
                // +spec:box-model:218643 - over-constrained: drop end margin per containing block writing mode
                // +spec:width-calculation:d172a4 - over-constrained: LTR ignores margin-right, RTL ignores margin-left
                // in LTR, margin-right is ignored (element positioned at margin-left);
                // in RTL, margin-left is ignored (element positioned from right edge)
14893
                let is_rtl = tree.get(node_index)
14893
                    .and_then(|n| n.dom_node_id)
14893
                    .map_or(false, |cb_dom_id| {
14889
                        let node_state = ctx.styled_dom.styled_nodes.as_container()
14889
                            .get(cb_dom_id)
14889
                            .map(|s| s.styled_node_state.clone())
14889
                            .unwrap_or_default();
14889
                        matches!(
14889
                            get_direction_property(ctx.styled_dom, cb_dom_id, &node_state),
                            MultiValue::Exact(StyleDirection::Rtl)
                        )
14889
                    });
14893
                let cross_pos = if is_rtl {
                    // RTL: ignore margin-left, position from right edge
                    available_cross - child_cross_size - child_margin.cross_end(writing_mode)
                } else {
                    // LTR (default): ignore margin-right, position at margin-left
14893
                    child_margin.cross_start(writing_mode)
                };
14893
                debug_info!(
14893
                    ctx,
14893
                    "[layout_bfc] Child {} NO auto margins (over-constrained), is_rtl={}, cross_pos={}",
                    child_index,
                    is_rtl,
                    cross_pos
                );
14893
                cross_pos
            };
14894
            (cross_pos, main_pen)
        };
        // NOTE: We do NOT adjust child_main_pos based on child's escaped_top_margin here!
        // The escaped_top_margin represents margins that escaped FROM the child's own children.
        // The child's position in THIS BFC is determined by main_pen and the child's own margin
        // (which was already handled in the margin collapse logic above).
        //
        // Previously, this code incorrectly added child_escaped_margin to child_main_pos,
        // which caused double-application of margins because:
        // 1. The child's margin was used to calculate its position in THIS BFC
        // 2. Then its escaped_top_margin (which included its own margin) was added again
        //
        // The correct behavior per CSS 2.2 § 8.3.1 is:
        // - The child's escaped_top_margin is used by THIS node's parent to position THIS node
        // - It does NOT affect how we position the child within our content-box
        // final_pos is [CoordinateSpace::Parent] - relative to this BFC's content-box
15254
        let final_pos =
15254
            LogicalPosition::from_main_cross(child_main_pos, child_cross_pos, writing_mode);
15254
        debug_info!(
15210
            ctx,
15210
            "[layout_bfc] *** NORMAL FLOW BLOCK POSITIONED: child={}, final_pos={:?}, \
15210
             main_pen={}, avoids_floats={}",
            child_index,
            final_pos,
            main_pen,
            avoids_floats
        );
        // Re-layout IFC children with float context for correct text wrapping
        // Normal flow blocks WITH inline content need float context propagated
15254
        if is_inline_fc && !avoids_floats {
            // Use cached floats if available (from previous layout passes),
            // otherwise use the floats positioned in this pass
11735
            let floats_for_ifc = float_cache.get(&node_index).unwrap_or(&float_context);
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc] Re-layouting IFC child {} (normal flow) with parent's float context \
11735
                 at Y={}, child_cross_pos={}",
                child_index,
                main_pen,
                child_cross_pos
            );
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc]   Using {} floats (from cache: {})",
11735
                floats_for_ifc.floats.len(),
11735
                float_cache.contains_key(&node_index)
            );
            // Translate float coordinates from BFC-relative to IFC-relative
            // The IFC child is positioned at (child_cross_pos, main_pen) in BFC coordinates
            // Floats need to be relative to the IFC's CONTENT-BOX origin (inside padding/border)
11735
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
11735
            let cbp = child_node.box_props.unpack();
11735
            let padding_border_cross = cbp.padding.cross_start(writing_mode)
11735
                + cbp.border.cross_start(writing_mode);
11735
            let padding_border_main = cbp.padding.main_start(writing_mode)
11735
                + cbp.border.main_start(writing_mode);
            // Content-box origin in BFC coordinates
11735
            let content_box_cross = child_cross_pos + padding_border_cross;
11735
            let content_box_main = main_pen + padding_border_main;
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc]   Border-box at ({}, {}), Content-box at ({}, {}), \
11735
                 padding+border=({}, {})",
                child_cross_pos,
                main_pen,
                content_box_cross,
                content_box_main,
                padding_border_cross,
                padding_border_main
            );
11735
            let mut ifc_floats = FloatingContext::default();
11741
            for float_box in &floats_for_ifc.floats {
                // Convert float position from BFC coords to IFC CONTENT-BOX relative coords
6
                let float_rel_to_ifc = LogicalRect {
6
                    origin: LogicalPosition {
6
                        x: float_box.rect.origin.x - content_box_cross,
6
                        y: float_box.rect.origin.y - content_box_main,
6
                    },
6
                    size: float_box.rect.size,
6
                };
6
                debug_info!(
6
                    ctx,
6
                    "[layout_bfc] Float {:?}: BFC coords = {:?}, IFC-content-relative = {:?}",
                    float_box.kind,
                    float_box.rect,
                    float_rel_to_ifc
                );
6
                ifc_floats.add_float(float_box.kind, float_rel_to_ifc, float_box.margin);
            }
            // Create a BfcState with IFC-relative float coordinates
11735
            let mut bfc_state = BfcState {
11735
                pen: LogicalPosition::zero(), // IFC starts at its own origin
11735
                floats: ifc_floats.clone(),
11735
                margins: MarginCollapseContext::default(),
11735
            };
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc]   Created IFC-relative FloatingContext with {} floats",
11735
                ifc_floats.floats.len()
            );
            // Get the IFC child's content-box size (after padding/border)
11735
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
11735
            let child_dom_id = child_node.dom_node_id;
            // +spec:containing-block:a8ada9 - line box width determined by containing block and floats
            // For inline elements (display: inline), use containing block width as available
            // width. Inline elements flow within the containing block and wrap at its width.
            // CSS 2.2 § 10.3.1: For inline elements, available width = containing block width.
11735
            let display = get_display_property(ctx.styled_dom, child_dom_id).unwrap_or_default();
11735
            let child_content_size = if display == LayoutDisplay::Inline {
                // Inline elements use the containing block's content-box width
9640
                LogicalSize::new(
9640
                    children_containing_block_size.width,
9640
                    children_containing_block_size.height,
                )
            } else {
                // Block-level elements use their own content-box
2095
                child_node.box_props.inner_size(child_size, writing_mode)
            };
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc]   IFC child size: border-box={:?}, content-box={:?}",
                child_size,
                child_content_size
            );
            // Create new constraints with float context
            // IMPORTANT: Use the child's CONTENT-BOX width, not the BFC width!
11735
            let ifc_constraints = LayoutConstraints {
11735
                available_size: child_content_size,
11735
                bfc_state: Some(&mut bfc_state),
11735
                writing_mode,
11735
                writing_mode_ctx: constraints.writing_mode_ctx,
11735
                text_align: constraints.text_align,
11735
                containing_block_size: constraints.containing_block_size,
11735
                available_width_type: Text3AvailableSpace::Definite(child_content_size.width),
11735
            };
            // Re-layout the IFC with float awareness
            // This will pass floats as exclusion zones to text3 for line wrapping
11735
            let ifc_result = layout_formatting_context(
11735
                ctx,
11735
                tree,
11735
                text_cache,
11735
                child_index,
11735
                &ifc_constraints,
11735
                float_cache,
            )?;
            // DON'T update used_size - the box keeps its full width!
            // Only the text layout inside changes to wrap around floats
11735
            debug_info!(
11735
                ctx,
11735
                "[layout_bfc] IFC child {} re-layouted with float context (text will wrap, box \
11735
                 stays full width)",
                child_index
            );
            // NOTE: We do NOT merge inline-block positions from the IFC's output.positions here!
            // The IFC's inline-block children will be correctly positioned when 
            // calculate_layout_for_subtree recursively processes the IFC node (child_index).
            // At that point, layout_ifc will be called again, and the inline-block positions
            // will be relative to the IFC's content-box, which is what we want.
            //
            // Merging them here would cause them to be processed by process_inflow_child
            // with the BFC's content-box position (self_content_box_pos of the BFC), 
            // resulting in incorrect absolute positions.
3519
        }
15254
        output.positions.insert(child_index, final_pos);
        // CSS margin collapse: escaped margins are handled via accumulated_top_margin
        // at the START of layout, not by adjusting positions after layout.
        // We simply advance by the child's actual size.
15254
        main_pen += child_size.main(writing_mode);
15254
        has_content = true;
        // Update last margin for next sibling
        // CSS 2.2 § 8.3.1: The bottom margin of this box will collapse with the top margin
        // of the next sibling (if no clearance or blockers intervene)
        // element (between prev sibling's bottom and this element's top margin). The cleared
        // element's bottom margin is still available for normal collapsing with the next sibling.
        // CSS 2.2 § 9.5.2: "Clearance inhibits margin collapsing and acts as spacing above
        // the margin-top of an element."
15254
        last_margin_bottom = child_margin_bottom;
15254
        debug_info!(
15210
            ctx,
15210
            "[layout_bfc] Child {} positioned at final_pos={:?}, size={:?}, advanced main_pen to \
15210
             {}, last_margin_bottom={}, clearance_applied={}",
            child_index,
            final_pos,
            child_size,
            main_pen,
            last_margin_bottom,
            clearance_applied
        );
        // Track the maximum cross-axis size to determine the BFC's overflow size.
15254
        let child_cross_extent =
15254
            child_cross_pos + child_size.cross(writing_mode) + child_margin.cross_end(writing_mode);
15254
        max_cross_size = max_cross_size.max(child_cross_extent);
    }
    // Store the float context in cache for future layout passes
    // This happens after ALL children (floats and normal) have been positioned
18965
    debug_info!(
17381
        ctx,
17381
        "[layout_bfc] Storing {} floats in cache for node {}",
17381
        float_context.floats.len(),
        node_index
    );
18965
    float_cache.insert(node_index, float_context.clone());
    // PHASE 3: Parent-Child Bottom Margin Escape
18965
    let mut escaped_top_margin = None;
18965
    let mut escaped_bottom_margin = None;
    // Handle top margin escape
18965
    if top_margin_escaped {
        // First child's margin escaped through parent
2507
        escaped_top_margin = Some(accumulated_top_margin);
2507
        debug_info!(
2507
            ctx,
2507
            "[layout_bfc] Returning escaped top margin: accumulated={}, node={}",
            accumulated_top_margin,
            node_index
        );
16458
    } else if !top_margin_resolved && accumulated_top_margin > 0.0 {
        // No content was positioned, all margins accumulated (empty blocks)
1
        escaped_top_margin = Some(accumulated_top_margin);
1
        debug_info!(
1
            ctx,
1
            "[layout_bfc] Escaping top margin (no content): accumulated={}, node={}",
            accumulated_top_margin,
            node_index
        );
    } else {
        // Don't set escaped_top_margin = Some(0) — that would override the child's
        // own margin (e.g., 30px) with 0 during sibling collapse.
16457
        debug_info!(
14873
            ctx,
14873
            "[layout_bfc] NOT escaping top margin: top_margin_resolved={}, escaped={}, \
14873
             accumulated={}, node={}",
            top_margin_resolved,
            top_margin_escaped,
            accumulated_top_margin,
            node_index
        );
    }
    // Handle bottom margin escape
18965
    if let Some(last_idx) = last_child_index {
13957
        let last_child = tree.get(last_idx).ok_or(LayoutError::InvalidTree)?;
13957
        let last_child_bp = last_child.box_props.unpack();
13957
        let last_has_bottom_blocker =
13957
            has_margin_collapse_blocker(&last_child_bp, writing_mode, false);
13957
        debug_info!(
13913
            ctx,
13913
            "[layout_bfc] Bottom margin for node {}: parent_has_bottom_blocker={}, \
13913
             last_has_bottom_blocker={}, last_margin_bottom={}, main_pen_before={}",
            node_index,
            parent_has_bottom_blocker,
            last_has_bottom_blocker,
            last_margin_bottom,
            main_pen
        );
13957
        if !parent_has_bottom_blocker && !last_has_bottom_blocker && has_content {
            // Last child's bottom margin can escape
2115
            let collapsed_bottom = collapse_margins(parent_margin_bottom, last_margin_bottom);
2115
            escaped_bottom_margin = Some(collapsed_bottom);
2115
            debug_info!(
2115
                ctx,
2115
                "[layout_bfc] Bottom margin ESCAPED for node {}: collapsed={}",
                node_index,
                collapsed_bottom
            );
            // Don't add last_margin_bottom to pen (it escaped)
        } else {
            // Can't escape: add to pen
11842
            main_pen += last_margin_bottom;
            // NOTE: We do NOT add parent_margin_bottom to main_pen here!
            // parent_margin_bottom is added OUTSIDE the content-box (in the margin-box)
            // The content-box height should only include children's content and margins
11842
            debug_info!(
11798
                ctx,
11798
                "[layout_bfc] Bottom margin BLOCKED for node {}: added last_margin_bottom={}, \
11798
                 main_pen_after={}",
                node_index,
                last_margin_bottom,
                main_pen
            );
        }
    } else {
        // No children: just use parent's margins
5008
        if !top_margin_resolved {
5008
            main_pen += parent_margin_top;
5008
        }
5008
        main_pen += parent_margin_bottom;
    }
    // CRITICAL: If this is a root node (no parent), apply escaped margins directly
    // instead of propagating them upward (since there's no parent to receive them)
18965
    let is_root_node = node.parent.is_none();
18965
    if is_root_node {
2477
        if let Some(top) = escaped_top_margin {
            // Adjust all child positions downward by the escaped top margin
1905
            for (_, pos) in output.positions.iter_mut() {
1905
                let current_main = pos.main(writing_mode);
1905
                *pos = LogicalPosition::from_main_cross(
1905
                    current_main + top,
1905
                    pos.cross(writing_mode),
1905
                    writing_mode,
1905
                );
1905
            }
1333
            main_pen += top;
1144
        }
2477
        if let Some(bottom) = escaped_bottom_margin {
936
            main_pen += bottom;
1548
        }
        // For root nodes, don't propagate margins further
2477
        escaped_top_margin = None;
2477
        escaped_bottom_margin = None;
16488
    }
    // CSS 2.2 § 9.5: Floats don't contribute to container height with overflow:visible
    //
    // However, browsers DO expand containers to contain floats in specific cases:
    //
    // 1. If there's NO in-flow content (main_pen == 0), floats determine height
    // 2. If container establishes a BFC (overflow != visible)
    //
    // In this case, we have in-flow content (main_pen > 0) and overflow:visible,
    // so floats should NOT expand the container. Their margins can "bleed" beyond
    // the container boundaries into the parent.
    //
    // This matches Chrome/Firefox behavior where float margins escape through
    // the container's padding when there's existing in-flow content.
    // +spec:block-formatting-context:7954a2 - 10.6.3: auto height for block-level non-replaced elements in normal flow
    // Content-box Height Calculation
    //
    // CSS 2.2 § 8.3.1: "The top border edge of the box is defined to coincide with
    // the top border edge of the [first] child" when margins collapse/escape.
    //
    // This means escaped margins do NOT contribute to the parent's content-box height.
    //
    // Calculation:
    //
    //   main_pen = total vertical space used by all children and margins
    //
    //   Components of main_pen:
    //
    //   1. Children's border-boxes (always included)
    //   2. Sibling collapsed margins (space BETWEEN children - part of content)
    //   3. First child's position (0 if margin escaped, margin_top if blocked)
    //
    //   What to subtract:
    //
    //   - total_escaped_top_margin: First child's margin that went to grandparent's space This
    //     margin is OUTSIDE our content-box, so we must subtract it.
    //
    //   What NOT to subtract:
    //
    //   - total_sibling_margins: These are the gaps BETWEEN children, which are
    //    legitimately part of our content area's layout space.
    //
    // Example with escaped margin:
    //   <div class="parent" padding=0>              <!-- Node 2 -->
    //     <div class="child1" margin=30></div>      <!-- Node 3, margin escapes -->
    //     <div class="child2" margin=40></div>      <!-- Node 5 -->
    //   </div>
    //
    //   Layout process:
    //
    //   - Node 3 positioned at main_pen=0 (margin escaped)
    //   - Node 3 size=140px → main_pen advances to 140
    //   - Sibling collapse: max(30 child1 bottom, 40 child2 top) = 40px
    //   - main_pen advances to 180
    //   - Node 5 size=130px → main_pen advances to 310
    //   - total_escaped_top_margin = 30
    //   - total_sibling_margins = 40 (tracked but NOT subtracted)
    //   - content_box_height = 310 - 30 = 280px ✓
    //
    // Previously, we calculated:
    //
    //   content_box_height = main_pen - total_escaped_top_margin - total_sibling_margins
    //
    // This incorrectly subtracted sibling margins, making parent too small.
    // Sibling margins are *between* boxes (part of layout), not *outside* boxes
    // (like escaped margins).
    // +spec:box-model:4eebed - auto height for BFC = top margin-edge of topmost child to bottom margin-edge of bottommost child
    // +spec:box-model:4eebed - auto height = top margin-edge of topmost child to bottom margin-edge of bottommost child
    // +spec:height-calculation:d65226 - §10.6.7 auto heights for BFC roots: block children use
    // margin-edge of topmost/bottommost, floats extend height if below content edge
    // +spec:positioning:1a05bb - 10.6.7 auto height for BFC roots: block children use margin edges,
    // abspos ignored (skipped in Pass 1/2), relative considered without offset (applied after layout),
    // floats whose bottom margin edge exceeds content edge expand height (below)
    // +spec:positioning:e6712c - Auto height for BFC: distance between top/bottom margin-edges of
    // block children (minus escaped margins), ignoring absolutely positioned children (skipped at
    // line ~966), considering relatively positioned boxes without offset (applied after layout),
    // and extending to include floats whose bottom margin edge exceeds content edge
    // +spec:positioning:f94d22 - 10.6.3: block-level non-replaced auto height = distance from top content edge to last in-flow child bottom margin edge (or zero)
    // CSS 2.2 §8.3.1: escaped margins (both top and bottom) don't contribute to parent height
18965
    let mut content_box_height = main_pen - total_escaped_top_margin
18965
        - escaped_bottom_margin.unwrap_or(0.0);
    // +spec:block-formatting-context:f73d3e - BFC root grows to fully contain its floats; floats from outside cannot protrude in
    // whose bottom margin edge exceeds bottom content edge; only floats participating
    // in this BFC are counted (not floats inside abspos descendants or nested BFCs)
    // +spec:box-model:1d4798 - auto height includes floats whose bottom margin edge exceeds content edge
    // only floats participating in this BFC are counted (not floats inside abspos descendants or nested BFCs)
18965
    if is_bfc_root {
14172
        for float_box in &float_context.floats {
            let float_bottom_margin_edge = float_box.rect.origin.main(writing_mode)
                + float_box.rect.size.main(writing_mode)
                + float_box.margin.main_end(writing_mode);
            if float_bottom_margin_edge > content_box_height {
                content_box_height = float_bottom_margin_edge;
            }
        }
4793
    }
    // +spec:display-contents:f6de1a - content height overflow tracked via overflow_size
    // +spec:overflow:043182 - overflow computed from box bounds + children overflow
18965
    output.overflow_size =
18965
        LogicalSize::from_main_cross(content_box_height, max_cross_size, writing_mode);
18965
    debug_info!(
17381
        ctx,
17381
        "[layout_bfc] FINAL for node {}: main_pen={}, total_escaped_top={}, \
17381
         total_sibling_margins={}, content_box_height={}",
        node_index,
        main_pen,
        total_escaped_top_margin,
        total_sibling_margins,
        content_box_height
    );
    // +spec:inline-formatting-context:2227a4 - atomic inline baseline for inline-block/inline-table
    // Baseline calculation would happen here in a full implementation.
    // CSS2 §10.8.1: For inline-block, baseline is the baseline of the last
    // line box in normal flow, or the bottom margin edge if no line boxes.
18965
    output.baseline = None;
    // Store escaped margins in the LayoutNode for use by parent
18965
    if let Some(warm_mut) = tree.warm_mut(node_index) {
18965
        warm_mut.escaped_top_margin = escaped_top_margin;
18965
        warm_mut.escaped_bottom_margin = escaped_bottom_margin;
18965
    }
18965
    if let Some(warm_mut) = tree.warm_mut(node_index) {
18965
        warm_mut.baseline = output.baseline;
18965
    }
18965
    Ok(BfcLayoutResult {
18965
        output,
18965
        escaped_top_margin,
18965
        escaped_bottom_margin,
18965
    })
18965
}
// Inline Formatting Context (CSS 2.2 § 9.4.2)
// +spec:display-property:ede6f4 - inline layout: mixed stream of text and inline-level boxes
/// Lays out an Inline Formatting Context (IFC) by delegating to the `text3` engine.
///
/// This function acts as a bridge between the box-tree world of `solver3` and the
/// rich text layout world of `text3`. Its responsibilities are:
///
/// 1. **Collect Content**: Traverse the direct children of the IFC root and convert them into a
///    `Vec<InlineContent>`, the input format for `text3`. This involves:
///
///     - Recursively laying out `inline-block` children to determine their final size and baseline,
///       which are then passed to `text3` as opaque objects.
///     - Extracting raw text runs from inline text nodes.
///
/// 2. **Translate Constraints**: Convert the `LayoutConstraints` (available space, floats) from
///    `solver3` into the more detailed `UnifiedConstraints` that `text3` requires.
///
/// 3. **Invoke Text Layout**: Call the `text3` cache's `layout_flow` method to perform the complex
///    tasks of BIDI analysis, shaping, line breaking, justification, and vertical alignment.
/// +spec:display-property:e96c82 - inline formatting context: flow of elements/text wrapped into lines
///
/// 4. **Integrate Results**: Process the `UnifiedLayout` returned by `text3`:
///
///     - Store the rich layout result on the IFC root `LayoutNode` for the display list generation
///       pass.
///     - Update the `positions` map for all `inline-block` children based on the positions
///       calculated by `text3`.
///     - Extract the final overflow size and baseline for the IFC root itself
// NOTE(writing-modes): The IFC currently assumes inline direction = horizontal
// and block direction = vertical. In vertical writing modes, line boxes would
// stack horizontally and inline content would flow vertically. The writing mode
// is now available via constraints.writing_mode_ctx for agents to use when
// implementing vertical text layout in the text3 engine.
// +spec:display-property:574e7b - text-box-trim for inline boxes trims block-end to content edge (TODO: implement trimming per text-box-edge metric)
// +spec:display-property:da284a - IFC: flow inline-level boxes into line boxes, size/position each fragment
// +spec:inline-formatting-context:275f64 - IFC: boxes laid out horizontally into line boxes, respecting margins/borders/padding
65625
fn layout_ifc<T: ParsedFontTrait>(
65625
    ctx: &mut LayoutContext<'_, T>,
65625
    text_cache: &mut crate::font_traits::TextLayoutCache,
65625
    tree: &mut LayoutTree,
65625
    node_index: usize,
65625
    constraints: &LayoutConstraints,
65625
) -> Result<LayoutOutput> {
65625
    unsafe { crate::az_mark((0x60704) as u32, (0x20u32) as u32); }
    // [g147 az-web-lift DIAG] CALLER-side tree validity at layout_ifc entry, indexed by node_index
    // (0x60900+ = nodes.len, 0x60920+ = tree ptr) to dodge marker-overwrite across multiple IFCs.
    // Compare vs _impl's CALLEE-side (0x60940+/0x60960+): ptr differs ⇒ &mut tree mis-passes across
    // the call; ptr same but len differs ⇒ the tree's `nodes` Vec is emptied in place.
    #[cfg(feature = "web_lift")]
    unsafe {
        let slot = (node_index & 7) * 4;
        crate::az_mark(((0x60900 + slot)) as u32, (tree.nodes.len() as u32) as u32);
        crate::az_mark(((0x60920 + slot)) as u32, ((&*tree as *const LayoutTree as usize) as u32) as u32);
    }
65625
    let ifc_start = (ctx.get_system_time_fn.cb)();
65625
    let float_count = constraints
65625
        .bfc_state
65625
        .as_ref()
65625
        .map(|s| s.floats.floats.len())
65625
        .unwrap_or(0);
65625
    debug_info!(
65625
        ctx,
65625
        "[layout_ifc] ENTRY: node_index={}, has_bfc_state={}, float_count={}",
        node_index,
65625
        constraints.bfc_state.is_some(),
        float_count
    );
65625
    debug_ifc_layout!(ctx, "CALLED for node_index={}", node_index);
    // +spec:display-property:7f3c1d - Anonymous inline boxes: text directly in block containers treated as anonymous inline elements in IFC
    // +spec:display-property:5a795c - root inline box: block container generates anonymous inline box holding all inline-level contents, inheriting from parent
    // For anonymous boxes, we need to find the DOM ID from a parent or child
    // CSS 2.2 § 9.2.1.1: Anonymous boxes inherit properties from their enclosing box
65625
    let node = tree.get(node_index).ok_or(LayoutError::InvalidTree)?;
65625
    let ifc_root_dom_id = match node.dom_node_id {
65449
        Some(id) => id,
        None => {
            // Anonymous box - get DOM ID from parent or first child with DOM ID
176
            let parent_dom_id = node
176
                .parent
176
                .and_then(|p| tree.get(p))
176
                .and_then(|n| n.dom_node_id);
176
            if let Some(id) = parent_dom_id {
176
                id
            } else {
                // Try to find DOM ID from first child
                tree.children(node_index)
                    .iter()
                    .filter_map(|&child_idx| tree.get(child_idx))
                    .filter_map(|n| n.dom_node_id)
                    .next()
                    .ok_or(LayoutError::InvalidTree)?
            }
        }
    };
65625
    debug_ifc_layout!(ctx, "ifc_root_dom_id={:?}", ifc_root_dom_id);
    // +spec:display-property:a469a6 - line boxes created as needed for inline-level content in IFC
    // +spec:display-property:f3c875 - calculate layout bounds (size contributions) of each inline-level box
    // Phase 1: Collect and measure all inline-level children.
65625
    let phase1_start = (ctx.get_system_time_fn.cb)();
65625
    let collect_result = collect_and_measure_inline_content(
65625
        ctx,
65625
        text_cache,
65625
        tree,
65625
        node_index,
65625
        constraints,
    );
    // [g133 az-web-lift DIAG] which early-return fires in POSITIONING's layout_ifc.
    #[cfg(feature = "web_lift")]
    unsafe {
        crate::az_mark((0x60680) as u32, (collect_result.as_ref().map(|(c, _)| c.len()).unwrap_or(0) as u32) as u32);
        crate::az_mark((0x60684) as u32, (if collect_result.is_ok() { 0xC0DE0680u32 } else { 0x000000EEu32 }) as u32);
    }
65625
    let (inline_content, child_map) = collect_result?;
65625
    let _phase1_time = (ctx.get_system_time_fn.cb)().duration_since(&phase1_start);
    // #11 fix: hash the inline content once. Used to (a) skip stale Phase 2d
    // fast-path reuse and (b) force a cache REPLACE when content changed even
    // though available width is unchanged — the display-list generator paints
    // text from the cached `inline_layout_result` (display_list.rs), so a
    // content change at a same-width constraint MUST overwrite it or the old
    // glyphs keep rendering (#11 stale display list).
65625
    let current_content_hash = {
        use std::hash::{Hash, Hasher};
65625
        let mut h = std::collections::hash_map::DefaultHasher::new();
65625
        inline_content.hash(&mut h);
65625
        h.finish()
    };
65625
    debug_info!(
65625
        ctx,
65625
        "[layout_ifc] Collected {} inline content items for node {}",
65625
        inline_content.len(),
        node_index
    );
65625
    if inline_content.len() > 10 {
        let _text_count = inline_content.iter().filter(|i| matches!(i, InlineContent::Text(_))).count();
        let _shape_count = inline_content.iter().filter(|i| matches!(i, InlineContent::Shape(_))).count();
65625
    }
65663
    for (i, item) in inline_content.iter().enumerate() {
65663
        match item {
65467
            InlineContent::Text(run) => debug_info!(ctx, "  [{}] Text: '{}'", i, run.text),
            InlineContent::Marker {
                run,
                position_outside,
            } => debug_info!(
                ctx,
                "  [{}] Marker: '{}' (outside={})",
                i,
                run.text,
                position_outside
            ),
176
            InlineContent::Shape(_) => debug_info!(ctx, "  [{}] Shape", i),
            InlineContent::Image(_) => debug_info!(ctx, "  [{}] Image", i),
20
            _ => debug_info!(ctx, "  [{}] Other", i),
        }
    }
65625
    debug_ifc_layout!(
65625
        ctx,
65625
        "Collected {} inline content items",
65625
        inline_content.len()
    );
65625
    if inline_content.is_empty() {
        debug_warning!(ctx, "inline_content is empty, returning default output!");
        // The node has no inline-level content this pass (e.g. its only
        // inline child — a text run or an inline image — was removed by a
        // relayout). Any `inline_layout_result` left over from a previous
        // frame is now stale: the display-list generator paints inline
        // objects (images, inline-block shapes) straight out of this cached
        // layout (see display_list.rs `paint_inline_*`), so a leftover entry
        // would re-emit the removed content AND index `styled_nodes` with a
        // `source_node_id` that no longer exists in the new DOM (OOB panic).
        // Clear it so the empty IFC renders nothing.
        if let Some(warm_node) = tree.warm_mut(node_index) {
            warm_node.inline_layout_result = None;
        }
        return Ok(LayoutOutput::default());
65625
    }
    // === Phase 2d: IFC incremental relayout decision tree ===
    //
    // Check if a cached layout exists with matching constraints. If so,
    // try incremental relayout (GlyphSwap or LineShift) before falling
    // back to full layout_flow().
    {
65625
        let cached_ifc = tree
65625
            .warm(node_index)
65625
            .and_then(|n| n.inline_layout_result.as_ref());
        // Only reuse the cached inline layout when the available WIDTH is unchanged.
        // This fast path was built for text edits (content changes, width constant); on a
        // viewport/container resize the width differs and the text must RE-WRAP, so the
        // cached old-width layout must NOT be reused — fall through to full layout_flow()
        // below. Without this guard, resizing kept the stale line breaks (#45). Real
        // text-edit incremental relayout (with dirty items) lives in
        // LayoutWindow::try_incremental_text_relayout.
65625
        let resize_has_floats = constraints
65625
            .bfc_state
65625
            .as_ref()
65625
            .map(|s| !s.floats.floats.is_empty())
65625
            .unwrap_or(false);
        // #11 fix: cache validity is keyed on WIDTH only, so a same-width
        // RefreshDom whose text CHANGED would otherwise reuse the stale shaped
        // layout. Require the inline content hash to match too.
65625
        let cached_ifc = cached_ifc
65625
            .filter(|c| c.is_valid_for(constraints.available_width_type, resize_has_floats))
65625
            .filter(|c| c.inline_content_hash == current_content_hash);
65625
        if let Some(cached) = cached_ifc {
19748
            if let Some(ref line_breaks) = cached.line_breaks {
                // Collect per-item advance widths from cached metrics
19748
                let old_advances: Vec<f32> = cached.item_metrics.iter()
19748
                    .map(|m| m.advance_width)
19748
                    .collect();
                // Cache-reuse fast path. Real incremental relayout for text
                // edits lives in LayoutWindow::try_incremental_text_relayout
                // (window.rs) — it has the newly-shaped items and the edited
                // node id, so it can compute real dirty_item_indices and
                // take the GlyphSwap / LineShift branches. Here we only
                // know the IFC is being re-entered (e.g. viewport resize on
                // a static IFC); with nothing re-shaped yet, the best we can
                // do is "no items changed at this level" → trivial GlyphSwap
                // to return the cached layout unchanged.
19748
                let result = crate::text3::cache::try_incremental_relayout(
19748
                    &[], // empty = no dirty items detected at this level
19748
                    &old_advances,
19748
                    &old_advances, // same advances since we haven't reshaped yet
19748
                    line_breaks,
                );
19748
                match result {
                    crate::text3::cache::IncrementalRelayoutResult::GlyphSwap => {
                        // No items changed — return cached layout directly
19748
                        debug_info!(ctx, "[layout_ifc] Phase 2d: GlyphSwap — reusing cached layout");
19748
                        let main_frag = &cached.layout;
19748
                        let frag_bounds = main_frag.bounds();
19748
                        let mut output = LayoutOutput::default();
19748
                        output.overflow_size = LogicalSize::new(frag_bounds.width, frag_bounds.height);
19748
                        output.baseline = main_frag.last_baseline();
                        // Re-position inline-block children from cached layout
175282
                        for positioned_item in &main_frag.items {
175282
                            if let ShapedItem::Object { source, .. } = &positioned_item.item {
88
                                if let Some(&child_node_index) = child_map.get(source) {
88
                                    output.positions.insert(child_node_index, LogicalPosition {
88
                                        x: positioned_item.position.x,
88
                                        y: positioned_item.position.y,
88
                                    });
88
                                }
175194
                            }
                        }
19748
                        return Ok(output);
                    }
                    _ => {
                        // Fall through to full layout_flow
                    }
                }
            }
45877
        }
    }
    // Phase 2: Translate constraints and define a single layout fragment for text3.
45877
    let text3_constraints =
45877
        translate_to_text3_constraints(ctx, constraints, ctx.styled_dom, ifc_root_dom_id);
    // Clone constraints for caching (before they're moved into fragments)
45877
    let cached_constraints = text3_constraints.clone();
45877
    debug_info!(
45877
        ctx,
45877
        "[layout_ifc] CALLING text_cache.layout_flow for node {} with {} exclusions",
        node_index,
45877
        text3_constraints.shape_exclusions.len()
    );
45877
    let fragments = vec![LayoutFragment {
45877
        id: "main".to_string(),
45877
        constraints: text3_constraints,
45877
    }];
    // Phase 3: Invoke the text layout engine.
    // Get pre-loaded fonts from font manager (fonts should be loaded before layout)
45877
    let phase3_start = (ctx.get_system_time_fn.cb)();
45877
    let loaded_fonts = ctx.font_manager.get_loaded_fonts();
45877
    let text_layout_result = match text_cache.layout_flow(
45877
        &inline_content,
45877
        &[],
45877
        &fragments,
45877
        &ctx.font_manager.font_chain_cache,
45877
        &ctx.font_manager.fc_cache,
45877
        &loaded_fonts,
45877
        ctx.debug_messages,
45877
    ) {
45877
        Ok(result) => {
            // [g133 az-web-lift DIAG] layout_flow returned Ok.
            #[cfg(feature = "web_lift")]
            unsafe { crate::az_mark((0x60688) as u32, (0xC0DE0688u32) as u32); }
45877
            result
        }
        Err(e) => {
            // [g133 az-web-lift DIAG] layout_flow returned Err → zero-sized (text not positioned).
            #[cfg(feature = "web_lift")]
            unsafe {
                crate::az_mark((0x60688) as u32, (0x000000EEu32) as u32);
                // Read the error's first byte (discriminant) for the marker — a
                // `*const u8` read is always aligned + in-bounds; the old
                // `*const u32` read was UB on a 1-aligned / <4-byte enum.
                crate::az_mark((0x6068C) as u32, (*(&e as *const _ as *const u8)) as u32);
            }
            // Font errors should not stop layout of other elements.
            // Log the error and return a zero-sized layout.
            debug_warning!(ctx, "Text layout failed: {:?}", e);
            debug_warning!(
                ctx,
                "Continuing with zero-sized layout for node {}",
                node_index
            );
            let mut output = LayoutOutput::default();
            output.overflow_size = LogicalSize::new(0.0, 0.0);
            return Ok(output);
        }
    };
45877
    let _phase3_time = (ctx.get_system_time_fn.cb)().duration_since(&phase3_start);
45877
    let _total_ifc_time = (ctx.get_system_time_fn.cb)().duration_since(&ifc_start);
    // Phase 4: Integrate results back into the solver3 layout tree.
45877
    let mut output = LayoutOutput::default();
45877
    debug_ifc_layout!(
45877
        ctx,
45877
        "text_layout_result has {} fragment_layouts",
45877
        text_layout_result.fragment_layouts.len()
    );
45877
    if let Some(main_frag) = text_layout_result.fragment_layouts.get("main") {
45877
        let frag_bounds = main_frag.bounds();
45877
        debug_ifc_layout!(
45877
            ctx,
45877
            "Found 'main' fragment with {} items, bounds={}x{}",
45877
            main_frag.items.len(),
            frag_bounds.width,
            frag_bounds.height
        );
45877
        debug_ifc_layout!(ctx, "Storing inline_layout_result on node {}", node_index);
        // Determine if we should store this layout result using the new
        // CachedInlineLayout system. The key insight is that inline layouts
        // depend on available width:
        //
        // - Min-content measurement uses width ≈ 0 (maximum line wrapping)
        // - Max-content measurement uses width = ∞ (no line wrapping)
        // - Final layout uses the actual column/container width
        //
        // We must track which constraint type was used, otherwise a min-content
        // measurement would incorrectly be reused for final rendering.
45877
        let has_floats = constraints
45877
            .bfc_state
45877
            .as_ref()
45877
            .map(|s| !s.floats.floats.is_empty())
45877
            .unwrap_or(false);
45877
        let current_width_type = constraints.available_width_type;
45877
        let warm_node = tree.warm_mut(node_index).ok_or(LayoutError::InvalidTree)?;
45877
        let should_store = match &warm_node.inline_layout_result {
            None => {
                // No cached result - always store
21907
                debug_info!(
21907
                    ctx,
21907
                    "[layout_ifc] Storing NEW inline_layout_result for node {} (width_type={:?}, \
21907
                     has_floats={})",
                    node_index,
                    current_width_type,
                    has_floats
                );
21907
                true
            }
23970
            Some(cached) => {
                // Check if the new result should replace the cached one
23970
                if cached.should_replace_with(current_width_type, has_floats)
                    || cached.inline_content_hash != current_content_hash
                {
                    // #11 fix: the cached layout is what the display-list
                    // generator paints from; replace it when the inline content
                    // changed, even if the width constraint is unchanged.
23970
                    debug_info!(
23970
                        ctx,
23970
                        "[layout_ifc] REPLACING inline_layout_result for node {} (old: \
23970
                         width={:?}, floats={}) with (new: width={:?}, floats={})",
                        node_index,
                        cached.available_width,
                        cached.has_floats,
                        current_width_type,
                        has_floats
                    );
23970
                    true
                } else {
                    debug_info!(
                        ctx,
                        "[layout_ifc] KEEPING cached inline_layout_result for node {} (cached: \
                         width={:?}, floats={}, new: width={:?}, floats={})",
                        node_index,
                        cached.available_width,
                        cached.has_floats,
                        current_width_type,
                        has_floats
                    );
                    false
                }
            }
        };
45877
        if should_store {
45877
            let mut cil = CachedInlineLayout::new_with_constraints(
45877
                main_frag.clone(),
45877
                current_width_type,
45877
                has_floats,
45877
                cached_constraints.clone(),
45877
            );
45877
            // #11 fix: record the content hash so Phase 2d only fast-path-reuses
45877
            // this layout when the inline content is genuinely unchanged, and so
45877
            // the store decision above can detect content changes.
45877
            cil.inline_content_hash = current_content_hash;
45877
            warm_node.inline_layout_result = Some(cil);
45877
        }
        // Extract the overall size and baseline for the IFC root.
        // +spec:display-property:a0d0ab - IFC height = top of topmost line box to bottom of bottommost line box
        // +spec:display-property:a63b8f - baseline-source defaults to auto (last baseline for inline-block/IFC)
45877
        output.overflow_size = LogicalSize::new(frag_bounds.width, frag_bounds.height);
45877
        output.baseline = main_frag.last_baseline();
45877
        warm_node.baseline = output.baseline;
        // +spec:box-model:929f42 - text-box-trim: trim half-leading from first/last formatted line
        // +spec:box-model:02e0f9 - text-box-trim: trim-end and trim-both, no effect with non-zero padding/border
        //
        // CSS Inline 3 § 6.2: For block containers, trim the block-start/block-end side
        // of the first/last formatted line. If there is intervening non-zero padding or
        // borders, there is no effect. Does not apply to flex, grid, or table contexts.
45877
        let ifc_node_state = &ctx.styled_dom.styled_nodes.as_container()[ifc_root_dom_id].styled_node_state;
        // Fast path: if no node in the DOM declared text-box-trim, the cascade
        // walk would always return None → skip it.
45877
        let text_box_trim = {
45877
            let skip = ctx.styled_dom
45877
                .css_property_cache
45877
                .ptr
45877
                .compact_cache
45877
                .as_ref()
45877
                .map(|cc| cc.dom_declared_flags & azul_css::compact_cache::DOM_HAS_TEXT_BOX_TRIM == 0)
45877
                .unwrap_or(false);
45877
            if skip {
45877
                StyleTextBoxTrim::None
            } else {
                get_text_box_trim_property(ctx.styled_dom, ifc_root_dom_id, ifc_node_state)
                    .unwrap_or(StyleTextBoxTrim::None)
            }
        };
45877
        if text_box_trim != StyleTextBoxTrim::None && !main_frag.items.is_empty() {
            // Half-leading = (line-height - (ascent + descent)) / 2
            let half_leading = (cached_constraints.resolved_line_height()
                - (cached_constraints.strut_ascent + cached_constraints.strut_descent))
                / 2.0;
            let half_leading = half_leading.max(0.0);
            // Check for intervening non-zero padding/border on block-start (top)
            let has_pad_or_border_top = match get_css_padding_top(ctx.styled_dom, ifc_root_dom_id, ifc_node_state) {
                MultiValue::Exact(pv) => pv.number.get() != 0.0,
                _ => false,
            } || match get_css_border_top_width(ctx.styled_dom, ifc_root_dom_id, ifc_node_state) {
                MultiValue::Exact(pv) => pv.number.get() != 0.0,
                _ => false,
            };
            // Check for intervening non-zero padding/border on block-end (bottom)
            let has_pad_or_border_bottom = match get_css_padding_bottom(ctx.styled_dom, ifc_root_dom_id, ifc_node_state) {
                MultiValue::Exact(pv) => pv.number.get() != 0.0,
                _ => false,
            } || match get_css_border_bottom_width(ctx.styled_dom, ifc_root_dom_id, ifc_node_state) {
                MultiValue::Exact(pv) => pv.number.get() != 0.0,
                _ => false,
            };
            let trim_start = matches!(text_box_trim, StyleTextBoxTrim::TrimStart | StyleTextBoxTrim::TrimBoth)
                && !has_pad_or_border_top;
            let trim_end = matches!(text_box_trim, StyleTextBoxTrim::TrimEnd | StyleTextBoxTrim::TrimBoth)
                && !has_pad_or_border_bottom;
            let mut height_reduction = 0.0;
            if trim_start && half_leading > 0.0 {
                height_reduction += half_leading;
            }
            if trim_end && half_leading > 0.0 {
                height_reduction += half_leading;
            }
            if height_reduction > 0.0 {
                output.overflow_size.height = (output.overflow_size.height - height_reduction).max(0.0);
            }
45877
        }
        // Position all the inline-block children based on text3's calculations.
        // [CoordinateSpace::Parent] - positions are relative to IFC's content-box (0,0)
327443
        for positioned_item in &main_frag.items {
327443
            if let ShapedItem::Object { source, content, .. } = &positioned_item.item {
88
                if let Some(&child_node_index) = child_map.get(source) {
88
                    // new_relative_pos is [CoordinateSpace::Parent] - relative to this IFC's content-box
88
                    let new_relative_pos = LogicalPosition {
88
                        x: positioned_item.position.x,
88
                        y: positioned_item.position.y,
88
                    };
88
                    output.positions.insert(child_node_index, new_relative_pos);
88
                }
327355
            }
        }
    }
    // [g132 az-web-lift VERIFY] Capture the IFC content geometry (the line-box bounds from
    // main_frag.bounds(), set above as output.overflow_size). height>0 proves the text LAID OUT
    // (not just shaped). Free-band addrs, f32 bits. REVERT at cleanup.
    #[cfg(feature = "web_lift")]
    unsafe {
        crate::az_mark((0x60670) as u32, (output.overflow_size.width.to_bits()) as u32);
        crate::az_mark((0x60674) as u32, (output.overflow_size.height.to_bits()) as u32);
        crate::az_mark((0x60678) as u32, (output.positions.len() as u32) as u32);
        crate::az_mark((0x6067C) as u32, (0xC0DE0132u32) as u32);
    }
45877
    Ok(output)
65625
}
1100
fn translate_taffy_size(size: LogicalSize) -> TaffySize<Option<f32>> {
1100
    TaffySize {
1100
        width: Some(size.width),
1100
        height: Some(size.height),
1100
    }
1100
}
/// Helper: Convert StyleFontStyle to text3::cache::FontStyle
97812
pub fn convert_font_style(style: StyleFontStyle) -> crate::font_traits::FontStyle {
97812
    match style {
97812
        StyleFontStyle::Normal => crate::font_traits::FontStyle::Normal,
        StyleFontStyle::Italic => crate::font_traits::FontStyle::Italic,
        StyleFontStyle::Oblique => crate::font_traits::FontStyle::Oblique,
    }
97812
}
/// Helper: Convert StyleFontWeight to FcWeight
97812
pub fn convert_font_weight(weight: StyleFontWeight) -> FcWeight {
97812
    match weight {
        StyleFontWeight::W100 => FcWeight::Thin,
        StyleFontWeight::W200 => FcWeight::ExtraLight,
        StyleFontWeight::W300 | StyleFontWeight::Lighter => FcWeight::Light,
97592
        StyleFontWeight::Normal => FcWeight::Normal,
        StyleFontWeight::W500 => FcWeight::Medium,
        StyleFontWeight::W600 => FcWeight::SemiBold,
220
        StyleFontWeight::Bold => FcWeight::Bold,
        StyleFontWeight::W800 => FcWeight::ExtraBold,
        StyleFontWeight::W900 | StyleFontWeight::Bolder => FcWeight::Black,
    }
97812
}
/// Resolves a CSS size metric to pixels.
///
/// - `metric`: The CSS unit (px, pt, em, vw, etc.)
/// - `value`: The numeric value
/// - `containing_block_size`: Size of containing block (for percentage)
/// - `viewport_size`: Viewport dimensions (for vw, vh, vmin, vmax)
#[inline]
1452
fn resolve_size_metric(
1452
    metric: SizeMetric,
1452
    value: f32,
1452
    containing_block_size: f32,
1452
    viewport_size: LogicalSize,
1452
) -> f32 {
1452
    match metric {
352
        SizeMetric::Px => value,
        SizeMetric::Pt => value * PT_TO_PX,
1100
        SizeMetric::Percent => value / 100.0 * containing_block_size,
        SizeMetric::Em | SizeMetric::Rem => value * DEFAULT_FONT_SIZE,
        SizeMetric::Vw => value / 100.0 * viewport_size.width,
        SizeMetric::Vh => value / 100.0 * viewport_size.height,
        SizeMetric::Vmin => value / 100.0 * viewport_size.width.min(viewport_size.height),
        SizeMetric::Vmax => value / 100.0 * viewport_size.width.max(viewport_size.height),
        // In, Cm, Mm: convert to pixels using standard DPI (96)
        SizeMetric::In => value * 96.0,
        SizeMetric::Cm => value * 96.0 / 2.54,
        SizeMetric::Mm => value * 96.0 / 25.4,
    }
1452
}
93016
pub fn translate_taffy_size_back(size: TaffySize<f32>) -> LogicalSize {
93016
    LogicalSize {
93016
        width: size.width,
93016
        height: size.height,
93016
    }
93016
}
11792
pub fn translate_taffy_point_back(point: taffy::Point<f32>) -> LogicalPosition {
11792
    LogicalPosition {
11792
        x: point.x,
11792
        y: point.y,
11792
    }
11792
}
// +spec:block-formatting-context:40e03e - BFC root: block container establishing new BFC (contains floats, excludes external floats, suppresses margin collapsing)
/// Checks if a node establishes a new Block Formatting Context (BFC).
///
/// Per CSS 2.2 § 9.4.1, a BFC is established by:
/// - Floats (elements with float other than 'none')
/// - Absolutely positioned elements (position: absolute or fixed)
/// - Block containers that are not block boxes (e.g., inline-blocks, table-cells)
/// - Block boxes with 'overflow' other than 'visible' and 'clip'
/// - Elements with 'display: flow-root'
/// - Table cells, table captions, and inline-blocks
///
/// Normal flow block-level boxes do NOT establish a new BFC.
///
/// This is critical for correct float interaction: normal blocks should overlap floats
/// (not shrink around them), while their inline content wraps around floats.
// +spec:block-formatting-context:241d22 - block container establishes new BFC or continues parent's, based on overflow/position/float/display
// +spec:block-formatting-context:9fe441 - BFC establishment based on position, float, overflow, and display properties
// +spec:display-property:3c7369 - block boxes establishing independent FC create new BFC; flex containers already do; non-replaced inlines cannot
// +spec:positioning:1e94f6 - floats, abspos, inline-blocks/table-cells/table-captions, overflow!=visible establish new BFC
34219
fn establishes_new_bfc<T: ParsedFontTrait>(ctx: &LayoutContext<'_, T>, node: &LayoutNodeHot, cold: Option<&LayoutNodeCold>) -> bool {
    // +spec:block-formatting-context:f39cd3 - table wrapper box establishes a BFC (CSS 2.2 §17.4)
    // Anonymous table wrapper boxes have no dom_node_id but must still establish BFC
    // +spec:height-calculation:e20498 - table wrapper box establishes BFC (CSS 2.2 §17.4)
    // +spec:positioning:b780d3 - Table wrapper box establishes BFC (CSS 2.2 § 17.4)
34219
    if cold.and_then(|c| c.anonymous_type) == Some(AnonymousBoxType::TableWrapper) {
        return true;
34219
    }
34219
    let Some(dom_id) = node.dom_node_id else {
96
        return false;
    };
34123
    let node_state = &ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state;
    // 1. Floats establish BFC
34123
    let float_val = get_float(ctx.styled_dom, dom_id, node_state);
34098
    if matches!(
34123
        float_val,
        MultiValue::Exact(LayoutFloat::Left | LayoutFloat::Right)
    ) {
25
        return true;
34098
    }
    // +spec:positioning:69468c - absolute/fixed forces independent formatting context
34098
    let position = crate::solver3::positioning::get_position_type(ctx.styled_dom, Some(dom_id));
34098
    if matches!(position, LayoutPosition::Absolute | LayoutPosition::Fixed) {
354
        return true;
33744
    }
    // 3. Inline-blocks, table-cells, table-captions establish BFC
33744
    let display = get_display_property(ctx.styled_dom, Some(dom_id));
23052
    if matches!(
33744
        display,
        MultiValue::Exact(
            LayoutDisplay::InlineBlock | LayoutDisplay::TableCell | LayoutDisplay::TableCaption
        )
    ) {
10692
        return true;
23052
    }
    // 4. display: flow-root establishes BFC
    // +spec:display-property:14bae6 - flow-root establishes a formatting context that contains/excludes floats
23052
    if matches!(display, MultiValue::Exact(LayoutDisplay::FlowRoot)) {
        return true;
23052
    }
    // +spec:overflow:0a944d - clip does NOT establish BFC; hidden/scroll/auto do establish BFC
    // +spec:overflow:631a4c - scroll containers establish independent formatting context (BFC)
    // +spec:overflow:f6a186 - overflow:clip does NOT establish BFC; use display:flow-root for that
    // +spec:overflow:717de1 - overflow != visible/clip establishes BFC per CSS 2.2 §9.4.1
    // +spec:positioning:6feb32 - overflow:clip does NOT establish new formatting context; hidden/scroll/auto do
    // 5. Block boxes with overflow other than 'visible' or 'clip' establish BFC
    // +spec:overflow:b34aef - Block boxes with overflow other than 'visible' or 'clip' establish BFC
    // Note: 'clip' does NOT establish BFC per CSS Overflow Module Level 3
23052
    let overflow_x = get_overflow_x(ctx.styled_dom, dom_id, node_state);
23052
    let overflow_y = get_overflow_y(ctx.styled_dom, dom_id, node_state);
44991
    let creates_bfc_via_overflow = |ov: &MultiValue<LayoutOverflow>| {
43878
        matches!(
44991
            ov,
            &MultiValue::Exact(
                LayoutOverflow::Hidden | LayoutOverflow::Scroll | LayoutOverflow::Auto
            )
        )
44991
    };
23052
    if creates_bfc_via_overflow(&overflow_x) || creates_bfc_via_overflow(&overflow_y) {
1113
        return true;
21939
    }
    // 6. Table, Flex, and Grid containers establish BFC (via FormattingContext)
    // +spec:block-formatting-context:f15b87 - display:table participates in a BFC
21628
    if matches!(
21939
        node.formatting_context,
        FormattingContext::Table | FormattingContext::Flex | FormattingContext::Grid
    ) {
311
        return true;
21628
    }
    // +spec:block-formatting-context:33e6cd - block container with different writing-mode than parent establishes independent BFC
    // CSS Writing Modes 4 § 3.2: if a block container has a different writing-mode
    // than its parent, its inner display type computes to flow-root (i.e., it establishes BFC).
    {
21628
        let hierarchy = ctx.styled_dom.node_hierarchy.as_container();
21628
        if let Some(parent_dom_id) = hierarchy[dom_id].parent_id() {
19591
            let parent_state = &ctx.styled_dom.styled_nodes.as_container()[parent_dom_id].styled_node_state;
19591
            let child_wm = get_writing_mode(ctx.styled_dom, dom_id, node_state).unwrap_or_default();
19591
            let parent_wm = get_writing_mode(ctx.styled_dom, parent_dom_id, parent_state).unwrap_or_default();
19591
            if child_wm != parent_wm {
                return true;
19591
            }
2037
        }
    }
    // Normal flow block boxes do NOT establish BFC
    // NOTE: align-content != normal should also establish BFC per CSS-DISPLAY-3, but align-content is not yet implemented for block containers
21628
    false
34219
}
// +spec:display-property:5e5420 - replaced element identification (glossary: replaced elements have natural dimensions, establish independent formatting context)
/// CSS 2.2 § 9.5: "The border box of a table, a block-level replaced element, or an element
/// in the normal flow that establishes a new block formatting context [...] must not overlap
/// the margin box of any floats in the same block formatting context as the element itself."
14894
fn is_block_level_replaced<T: ParsedFontTrait>(ctx: &LayoutContext<'_, T>, node: &LayoutNodeHot) -> bool {
14894
    let Some(dom_id) = node.dom_node_id else {
92
        return false;
    };
    // Check display is block-level
14802
    let display = get_display_property(ctx.styled_dom, Some(dom_id));
14802
    let is_block_level = matches!(
14802
        display,
        MultiValue::Exact(LayoutDisplay::Block | LayoutDisplay::ListItem | LayoutDisplay::FlowRoot)
    );
14802
    if !is_block_level {
9552
        return false;
5250
    }
    // Check if the element is a replaced element (image, video, etc.)
5250
    let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
5250
    matches!(
5250
        node_data.get_node_type(),
        NodeType::Image(_)
    )
14894
}
/// Translates solver3 layout constraints into the text3 engine's unified constraints.
45877
fn translate_to_text3_constraints<'a, T: ParsedFontTrait>(
45877
    ctx: &mut LayoutContext<'_, T>,
45877
    constraints: &'a LayoutConstraints<'a>,
45877
    styled_dom: &StyledDom,
45877
    dom_id: NodeId,
45877
) -> UnifiedConstraints {
45877
    unsafe { crate::az_mark((0x60704) as u32, (0x30u32) as u32); }
    // DOM-level declared flags: if a bit is clear, no node in this DOM
    // declared the corresponding property → cascade walks always return
    // None, and we use the default value directly. All flags default to
    // "set" when there is no compact cache (paranoid fallback).
    use azul_css::compact_cache::{
        DOM_HAS_SHAPE_INSIDE, DOM_HAS_SHAPE_OUTSIDE, DOM_HAS_TEXT_JUSTIFY,
        DOM_HAS_TEXT_INDENT, DOM_HAS_COLUMN_COUNT, DOM_HAS_COLUMN_GAP,
        DOM_HAS_COLUMN_WIDTH,
        DOM_HAS_INITIAL_LETTER, DOM_HAS_INITIAL_LETTER_ALIGN,
        DOM_HAS_LINE_CLAMP, DOM_HAS_HANGING_PUNCTUATION,
        DOM_HAS_TEXT_COMBINE_UPRIGHT, DOM_HAS_EXCLUSION_MARGIN,
        DOM_HAS_SHAPE_MARGIN,
        DOM_HAS_HYPHENATION_LANGUAGE, DOM_HAS_UNICODE_BIDI,
        DOM_HAS_HYPHENS, DOM_HAS_WORD_BREAK, DOM_HAS_OVERFLOW_WRAP,
        DOM_HAS_LINE_BREAK, DOM_HAS_TEXT_ALIGN_LAST, DOM_HAS_LINE_HEIGHT,
    };
45877
    let dom_declared = styled_dom
45877
        .css_property_cache
45877
        .ptr
45877
        .compact_cache
45877
        .as_ref()
45877
        .map(|cc| cc.dom_declared_flags)
45877
        .unwrap_or(!0u32);
    // Convert floats into exclusion zones for text3 to flow around.
45877
    let mut shape_exclusions = if let Some(ref bfc_state) = constraints.bfc_state {
        debug_info!(
            ctx,
            "[translate_to_text3] dom_id={:?}, converting {} floats to exclusions",
            dom_id,
            bfc_state.floats.floats.len()
        );
        bfc_state
            .floats
            .floats
            .iter()
            .enumerate()
            .map(|(i, float_box)| {
                let rect = crate::text3::cache::Rect {
                    x: float_box.rect.origin.x,
                    y: float_box.rect.origin.y,
                    width: float_box.rect.size.width,
                    height: float_box.rect.size.height,
                };
                debug_info!(
                    ctx,
                    "[translate_to_text3]   Exclusion #{}: {:?} at ({}, {}) size {}x{}",
                    i,
                    float_box.kind,
                    rect.x,
                    rect.y,
                    rect.width,
                    rect.height
                );
                ShapeBoundary::Rectangle(rect)
            })
            .collect()
    } else {
45877
        debug_info!(
45877
            ctx,
45877
            "[translate_to_text3] dom_id={:?}, NO bfc_state - no float exclusions",
            dom_id
        );
45877
        Vec::new()
    };
45877
    debug_info!(
45877
        ctx,
45877
        "[translate_to_text3] dom_id={:?}, available_size={}x{}, shape_exclusions.len()={}",
        dom_id,
        constraints.available_size.width,
        constraints.available_size.height,
45877
        shape_exclusions.len()
    );
    // Map text-align and justify-content from CSS to text3 enums.
45877
    let id = dom_id;
45877
    let node_data = &styled_dom.node_data.as_container()[id];
45877
    let node_state = &styled_dom.styled_nodes.as_container()[id].styled_node_state;
    // Read CSS Shapes properties
    // For reference box, use the element's CSS height if available, otherwise available_size
    // This is important because available_size.height might be infinite during auto height
    // calculation
45877
    let ref_box_height = if constraints.available_size.height.is_finite() {
33196
        constraints.available_size.height
    } else {
        // Try to get explicit CSS height
        // NOTE: If height is infinite, we can't properly resolve % heights
        // This is a limitation - shape-inside with % heights requires finite containing block
12681
        styled_dom
12681
            .css_property_cache
12681
            .ptr
12681
            .get_height(node_data, &id, node_state)
12681
            .and_then(|v| v.get_property())
12681
            .and_then(|h| match h {
                LayoutHeight::Px(v) => {
                    // Only accept absolute units (px, pt, in, cm, mm) - no %, em, rem
                    // since we can't resolve relative units without proper context
                    match v.metric {
                        SizeMetric::Px => Some(v.number.get()),
                        SizeMetric::Pt => Some(v.number.get() * PT_TO_PX),
                        SizeMetric::In => Some(v.number.get() * 96.0),
                        SizeMetric::Cm => Some(v.number.get() * 96.0 / 2.54),
                        SizeMetric::Mm => Some(v.number.get() * 96.0 / 25.4),
                        _ => None, // Ignore %, em, rem
                    }
                }
                _ => None,
            })
12681
            .unwrap_or(constraints.available_size.width) // Fallback: use width as height (square)
    };
45877
    let reference_box = crate::text3::cache::Rect {
45877
        x: 0.0,
45877
        y: 0.0,
45877
        width: constraints.available_size.width,
45877
        height: ref_box_height,
45877
    };
    // shape-inside: Text flows within the shape boundary
45877
    debug_info!(ctx, "Checking shape-inside for node {:?}", id);
45877
    debug_info!(
45877
        ctx,
45877
        "Reference box: {:?} (available_size height was: {})",
        reference_box,
        constraints.available_size.height
    );
45877
    let shape_boundaries = if dom_declared & DOM_HAS_SHAPE_INSIDE != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_shape_inside(node_data, &id, node_state)
            .and_then(|v| {
                debug_info!(ctx, "Got shape-inside value: {:?}", v);
                v.get_property()
            })
            .and_then(|shape_inside| {
                debug_info!(ctx, "shape-inside property: {:?}", shape_inside);
                if let ShapeInside::Shape(css_shape) = shape_inside {
                    debug_info!(
                        ctx,
                        "Converting CSS shape to ShapeBoundary: {:?}",
                        css_shape
                    );
                    let boundary =
                        ShapeBoundary::from_css_shape(css_shape, reference_box, ctx.debug_messages);
                    debug_info!(ctx, "Created ShapeBoundary: {:?}", boundary);
                    Some(vec![boundary])
                } else {
                    debug_info!(ctx, "shape-inside is None");
                    None
                }
            })
            .unwrap_or_default()
    } else {
45877
        Vec::new()
    };
45877
    debug_info!(
45877
        ctx,
45877
        "Final shape_boundaries count: {}",
45877
        shape_boundaries.len()
    );
    // shape-outside: Text wraps around the shape (adds to exclusions)
45877
    debug_info!(ctx, "Checking shape-outside for node {:?}", id);
45877
    if dom_declared & DOM_HAS_SHAPE_OUTSIDE != 0 {
        if let Some(shape_outside_value) = styled_dom
            .css_property_cache
            .ptr
            .get_shape_outside(node_data, &id, node_state)
        {
            debug_info!(ctx, "Got shape-outside value: {:?}", shape_outside_value);
            if let Some(shape_outside) = shape_outside_value.get_property() {
                debug_info!(ctx, "shape-outside property: {:?}", shape_outside);
                if let ShapeOutside::Shape(css_shape) = shape_outside {
                    debug_info!(
                        ctx,
                        "Converting CSS shape-outside to ShapeBoundary: {:?}",
                        css_shape
                    );
                    let boundary =
                        ShapeBoundary::from_css_shape(css_shape, reference_box, ctx.debug_messages);
                    debug_info!(ctx, "Created ShapeBoundary (exclusion): {:?}", boundary);
                    shape_exclusions.push(boundary);
                }
            }
        } else {
            debug_info!(ctx, "No shape-outside value found");
        }
45877
    }
    // TODO: clip-path will be used for rendering clipping (not text layout)
45877
    let writing_mode = get_writing_mode(styled_dom, id, node_state).unwrap_or_default();
45877
    let text_align = get_text_align(styled_dom, id, node_state).unwrap_or_default();
45877
    let text_justify = if dom_declared & DOM_HAS_TEXT_JUSTIFY != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_text_justify(node_data, &id, node_state)
            .and_then(|s| s.get_property().copied())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
    // Get font-size for resolving line-height
    // Use helper function which checks dependency chain first
45877
    let font_size = get_element_font_size(styled_dom, id, node_state);
45877
    let line_height_value = if dom_declared & DOM_HAS_LINE_HEIGHT != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_line_height(node_data, &id, node_state)
            .and_then(|s| s.get_property().cloned())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
45877
    let hyphenation = if dom_declared & DOM_HAS_HYPHENS != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_hyphens(node_data, &id, node_state)
            .and_then(|s| s.get_property().copied())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
45877
    let word_break_css = if dom_declared & DOM_HAS_WORD_BREAK != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_word_break(node_data, &id, node_state)
            .and_then(|s| s.get_property().copied())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
45877
    let overflow_wrap_css = if dom_declared & DOM_HAS_OVERFLOW_WRAP != 0 {
44
        styled_dom
44
            .css_property_cache
44
            .ptr
44
            .get_overflow_wrap(node_data, &id, node_state)
44
            .and_then(|s| s.get_property().copied())
44
            .unwrap_or_default()
    } else {
45833
        Default::default()
    };
45877
    let line_break_css = if dom_declared & DOM_HAS_LINE_BREAK != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_line_break(node_data, &id, node_state)
            .and_then(|s| s.get_property().copied())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
45877
    let text_align_last_css = if dom_declared & DOM_HAS_TEXT_ALIGN_LAST != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_text_align_last(node_data, &id, node_state)
            .and_then(|s| s.get_property().copied())
            .unwrap_or_default()
    } else {
45877
        Default::default()
    };
45877
    let overflow_behaviour = get_overflow_x(styled_dom, id, node_state).unwrap_or_default();
    // +spec:display-property:21f728 - vertical-align shorthand resolves inline-level box alignment
    // +spec:display-property:98fa8e - alignment-baseline values for inline-level boxes in IFC (implemented via vertical-align shorthand)
    // +spec:display-property:1f71ad - baseline-shift + alignment-baseline longhands mapped through vertical-align
    // +spec:display-property:89dd7b - line-relative shift values (top/center/bottom) and aligned subtree alignment
    // +spec:inline-formatting-context:21da06 - vertical-align uses line-over/line-under sides via writing_mode logical mapping
    // +spec:inline-formatting-context:295603 - baseline alignment: vertical-align determines how inline boxes align (baseline, super, sub, etc.)
    // +spec:inline-formatting-context:7351bf - default alignment baseline is alphabetic in horizontal typographic mode
    // +spec:inline-formatting-context:85de3d - vertical-align shorthand: alignment within line box
    // +spec:inline-formatting-context:aa8af0 - alignment baseline chosen by vertical-align, defaults to parent's dominant baseline
    // +spec:inline-formatting-context:e475d2 - baseline and vertical-align control transverse alignment of inline content on line boxes
    // +spec:overflow:d44eac - vertical-align inline box alignment (CSS 2.2 model covers baseline/top/middle/bottom/sub/super/text-top/text-bottom)
    // +spec:writing-modes:313575 - alignment-baseline: inline-level boxes align baselines within parent inline box's alignment context along inline axis
    // +spec:writing-modes:60ad67 - inline layout aligns boxes in block axis via baselines
    // +spec:writing-modes:0127e5 - line-relative directions: line-over/under map to vertical-align top/bottom
    // Get vertical-align from CSS property cache (defaults to Baseline per CSS spec)
    // +spec:inline-formatting-context:686f8b - vertical-align shorthand: alignment-baseline + baseline-shift for inline boxes
    // +spec:inline-formatting-context:e579b6 - vertical-align / baseline alignment in inline context
    // +spec:inline-formatting-context:a01a75 - dominant baseline alignment for atomic inlines
45877
    let vertical_align = match get_vertical_align_property(styled_dom, id, node_state) {
1188
        MultiValue::Exact(v) => v,
44689
        _ => StyleVerticalAlign::default(),
    };
    // +spec:display-property:c03a6b - baseline-shift (sub/super/length/percentage) and line-relative (top/center/bottom) shifts handled via vertical-align
45877
    let vertical_align = match vertical_align {
44689
        StyleVerticalAlign::Baseline => text3::cache::VerticalAlign::Baseline,
        StyleVerticalAlign::Top => text3::cache::VerticalAlign::Top,
1188
        StyleVerticalAlign::Middle => text3::cache::VerticalAlign::Middle,
        StyleVerticalAlign::Bottom => text3::cache::VerticalAlign::Bottom,
        StyleVerticalAlign::Sub => text3::cache::VerticalAlign::Sub,
        // +spec:inline-formatting-context:fe563c - vertical-align: super shifts inline to superscript position
        // +spec:inline-formatting-context:fe563c - vertical-align:super shifts child to superscript position
        StyleVerticalAlign::Superscript => text3::cache::VerticalAlign::Super,
        StyleVerticalAlign::TextTop => text3::cache::VerticalAlign::TextTop,
        StyleVerticalAlign::TextBottom => text3::cache::VerticalAlign::TextBottom,
        // §10.8.1: <percentage> refers to line-height of the element itself
        StyleVerticalAlign::Percentage(p) => {
            let lh_n = line_height_value.inner.normalized();
            let resolved_lh = if lh_n < 0.0 { -lh_n } else { lh_n * font_size };
            let offset = p.normalized() * resolved_lh;
            text3::cache::VerticalAlign::Offset(offset)
        }
        // §10.8.1: <length> is absolute offset from baseline
        StyleVerticalAlign::Length(l) => {
            let offset = super::calc::resolve_pixel_value(&l, 0.0, font_size, font_size);
            text3::cache::VerticalAlign::Offset(offset)
        }
    };
    // +spec:block-formatting-context:987746 - text-orientation property (mixed/upright/sideways) for vertical writing modes
    // +spec:inline-formatting-context:cbe738 - text-orientation (mixed/upright/sideways) bi-orientational transform for vertical text
    // +spec:writing-modes:09a1bb - vertical typesetting orientation (upright/sideways) for vertical-rl/vertical-lr
    // +spec:writing-modes:2eb1b2 - text-orientation (mixed/upright/sideways) applied to vertical text layout
45877
    let text_orientation = match get_text_orientation_property(styled_dom, id, node_state) {
        MultiValue::Exact(o) => match o {
            StyleTextOrientation::Mixed => text3::cache::TextOrientation::Mixed,
            StyleTextOrientation::Upright => text3::cache::TextOrientation::Upright,
            // +spec:block-formatting-context:a606e6 - sideways text typeset rotated 90° CW in vertical modes
            StyleTextOrientation::Sideways => text3::cache::TextOrientation::Sideways,
        },
45877
        _ => text3::cache::TextOrientation::default(),
    };
    // +spec:display-property:8364c0 - direction property (ltr/rtl) sets paragraph embedding level for bidi algorithm
    // +spec:text-alignment-spacing:97b93a - direction property affects text-align:justify last-line alignment
    // +spec:writing-modes:73aaff - block elements inherit base direction from parent via CSS direction property
    // +spec:writing-modes:8a888b - line box inline base direction from containing block's direction
    // Get the direction property from the CSS cache (defaults to LTR if not set)
    // +spec:display-property:da3b59 - direction property specifies inline base direction for ordering inline-level content
    // +spec:inline-formatting-context:97af40 - direction property sets inline base direction for bidi, text alignment, overflow
    // +spec:writing-modes:2deb38 - bidirectional reordering via CSS direction property
    // +spec:writing-modes:fbb332 - in vertical writing modes, text-orientation:upright forces used direction to ltr
45877
    let direction = match constraints.writing_mode {
        LayoutWritingMode::VerticalRl | LayoutWritingMode::VerticalLr
            if matches!(text_orientation, text3::cache::TextOrientation::Upright) =>
        {
            Some(text3::cache::BidiDirection::Ltr)
        }
45877
        _ => match get_direction_property(styled_dom, id, node_state) {
45877
            MultiValue::Exact(d) => Some(match d {
45877
                StyleDirection::Ltr => text3::cache::BidiDirection::Ltr,
                StyleDirection::Rtl => text3::cache::BidiDirection::Rtl,
            }),
            _ => None,
        },
    };
    // Get unicode-bidi property for bidi algorithm configuration
    // +spec:containing-block:0d4914 - unicode-bidi: plaintext causes P2/P3 heuristics instead of HL1 override
45877
    let unicode_bidi_val = if dom_declared & DOM_HAS_UNICODE_BIDI != 0 {
        match get_unicode_bidi_property(styled_dom, id, node_state) {
            MultiValue::Exact(u) => match u {
                StyleUnicodeBidi::Normal => text3::cache::UnicodeBidi::Normal,
                StyleUnicodeBidi::Embed => text3::cache::UnicodeBidi::Embed,
                StyleUnicodeBidi::Isolate => text3::cache::UnicodeBidi::Isolate,
                StyleUnicodeBidi::BidiOverride => text3::cache::UnicodeBidi::BidiOverride,
                StyleUnicodeBidi::IsolateOverride => text3::cache::UnicodeBidi::IsolateOverride,
                StyleUnicodeBidi::Plaintext => text3::cache::UnicodeBidi::Plaintext,
            },
            _ => text3::cache::UnicodeBidi::Normal,
        }
    } else {
45877
        text3::cache::UnicodeBidi::Normal
    };
45877
    debug_info!(
45877
        ctx,
45877
        "dom_id={:?}, available_size={}x{}, setting available_width={}",
        dom_id,
        constraints.available_size.width,
        constraints.available_size.height,
        constraints.available_size.width
    );
    // +spec:box-model:8113d7 - text-indent treated as margin on start edge of line box
    // +spec:display-contents:5f95ac - text-indent: percentage=0 for intrinsic sizing, each-line and hanging keywords
    // +spec:floats:17c74a - text-indent applied to first line (5em indentation with no floats)
    // +spec:positioning:1e32b1 - text-indent with hanging/each-line keywords resolved and passed to text layout
45877
    let text_indent_prop = if dom_declared & DOM_HAS_TEXT_INDENT != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_text_indent(node_data, &id, node_state)
            .and_then(|s| s.get_property().cloned())
    } else {
45877
        None
    };
45877
    let is_intrinsic_sizing = matches!(
45877
        constraints.available_width_type,
        Text3AvailableSpace::MinContent | Text3AvailableSpace::MaxContent
    );
    // +spec:intrinsic-sizing:0e8625 - percentage text-indent treated as 0 for intrinsic size contributions
45877
    let text_indent = text_indent_prop
45877
        .map(|ti| {
            // CSS Text 3 §8.1: "Percentages must be treated as 0 for the purpose
            // of calculating intrinsic size contributions"
            if is_intrinsic_sizing && ti.inner.to_percent().is_some() {
                return 0.0;
            }
            let context = ResolutionContext {
                element_font_size: get_element_font_size(styled_dom, id, node_state),
                parent_font_size: get_parent_font_size(styled_dom, id, node_state),
                root_font_size: get_root_font_size(styled_dom, node_state),
                containing_block_size: PhysicalSize::new(constraints.available_size.width, 0.0),
                element_size: None,
                viewport_size: PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height),
            };
            ti.inner
                .resolve_with_context(&context, PropertyContext::Other)
        })
45877
        .unwrap_or(0.0);
45877
    let text_indent_each_line = text_indent_prop.map(|ti| ti.each_line).unwrap_or(false);
45877
    let text_indent_hanging = text_indent_prop.map(|ti| ti.hanging).unwrap_or(false);
    // ResolutionContext shared by column-gap and column-width (both resolve
    // lengths against the same font/viewport, with no containing-block size).
45877
    let column_resolve_ctx = ResolutionContext {
45877
        element_font_size: get_element_font_size(styled_dom, id, node_state),
45877
        parent_font_size: get_parent_font_size(styled_dom, id, node_state),
45877
        root_font_size: get_root_font_size(styled_dom, node_state),
45877
        containing_block_size: PhysicalSize::new(0.0, 0.0),
45877
        element_size: None,
45877
        viewport_size: PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height),
45877
    };
    // Read a declared CSS property from the cache, returning None when the
    // DOM-level declared bit is clear (no node sets the property).
    macro_rules! declared_prop {
        ($bit:expr, $getter:ident) => {
            if dom_declared & $bit != 0 {
                styled_dom
                    .css_property_cache
                    .ptr
                    .$getter(node_data, &id, node_state)
                    .and_then(|s| s.get_property())
            } else {
                None
            }
        };
    }
    // Get column-gap for multi-column layout (default: normal = 1em)
45877
    let column_gap = declared_prop!(DOM_HAS_COLUMN_GAP, get_column_gap)
45877
        .map(|cg| {
            cg.inner
                .resolve_with_context(&column_resolve_ctx, PropertyContext::Other)
        })
45877
        .unwrap_or_else(|| get_element_font_size(styled_dom, id, node_state));
    // Get column-width for multi-column layout (None = auto)
45877
    let column_width =
45877
        declared_prop!(DOM_HAS_COLUMN_WIDTH, get_column_width).and_then(|cw| match cw {
            ColumnWidth::Auto => None,
            ColumnWidth::Length(px) => {
                Some(px.resolve_with_context(&column_resolve_ctx, PropertyContext::Other))
            }
        });
    // Get column-count for multi-column layout (default: 1 = no columns)
45877
    let explicit_column_count =
45877
        declared_prop!(DOM_HAS_COLUMN_COUNT, get_column_count).copied();
    // CSS multi-column: derive column count from column-width when column-count is auto.
    // Per spec: N = max(1, floor((available-width + column-gap) / (column-width + column-gap)))
45877
    let columns = match (explicit_column_count, column_width) {
        (Some(ColumnCount::Integer(n)), _) => n,
        (_, Some(cw)) if cw > 0.0 => {
            let avail = constraints.available_size.width;
            ((avail + column_gap) / (cw + column_gap)).floor().max(1.0) as u32
        }
45877
        _ => 1,
    };
    // +spec:line-breaking:b4928e - white-space values mapped to wrap/whitespace processing rules
    // Map white-space CSS property to TextWrap
45877
    let resolved_ws = match get_white_space_property(styled_dom, id, node_state) {
45877
        MultiValue::Exact(ws) => ws,
        _ => StyleWhiteSpace::Normal,
    };
45877
    let text_wrap = match resolved_ws {
45822
        StyleWhiteSpace::Normal => text3::cache::TextWrap::Wrap,
45
        StyleWhiteSpace::Nowrap => text3::cache::TextWrap::NoWrap,
6
        StyleWhiteSpace::Pre => text3::cache::TextWrap::NoWrap,
2
        StyleWhiteSpace::PreWrap => text3::cache::TextWrap::Wrap,
2
        StyleWhiteSpace::PreLine => text3::cache::TextWrap::Wrap,
        StyleWhiteSpace::BreakSpaces => text3::cache::TextWrap::Wrap,
    };
45877
    let white_space_mode = match resolved_ws {
45822
        StyleWhiteSpace::Normal => text3::cache::WhiteSpaceMode::Normal,
45
        StyleWhiteSpace::Nowrap => text3::cache::WhiteSpaceMode::Nowrap,
6
        StyleWhiteSpace::Pre => text3::cache::WhiteSpaceMode::Pre,
2
        StyleWhiteSpace::PreWrap => text3::cache::WhiteSpaceMode::PreWrap,
2
        StyleWhiteSpace::PreLine => text3::cache::WhiteSpaceMode::PreLine,
        StyleWhiteSpace::BreakSpaces => text3::cache::WhiteSpaceMode::BreakSpaces,
    };
    // +spec:block-formatting-context:fd60a8 - initial letter box is in-flow in its BFC, originating line box
    // +spec:block-formatting-context:c5ba02 - initial letter inline flow layout (alignment, white space collapsing)
    // +spec:block-formatting-context:83f8a7 - initial letter wrapping modes (none, all, first)
    // +spec:block-formatting-context:fef28d - initial letter box is in-flow in its BFC, part of originating line box
    // +spec:box-model:c3ce58 - initial letter block-start margin edge must be below containing block content edge
    // +spec:display-contents:568fe2 - initial letter participates in same IFC as its line
    // +spec:display-property:a89adb - initial letter boxes from non-replaced inline boxes and atomic inlines
    // +spec:display-property:4b59ce - initial-letter applies to inline-level boxes at start of first line
    // +spec:display-property:756cad - initial-letter sizing: drop/raise/sunken initial computation
    // +spec:display-property:8b08f4 - initial-letter applied to first inline-level child of block container
    // +spec:display-property:8c1dce - initial-letter property: size/sink for drop caps on inline-level boxes
    // +spec:display-property:b453a3 - initial-letter applies to inline-level boxes in IFC
    // +spec:display-property:b5e149 - initial letters are in-flow inline-level content, not floats
    // +spec:display-property:fa044e - initial-letter applies to first-child inline-level boxes
    // +spec:line-height:306d87 - initial-letter sizing must use containing block's line-height, not spanned lines' heights
    // +spec:writing-modes:903310 - atomic initial letters use normal sizing; only positioning is special
    // Get initial-letter for drop caps
    // +spec:display-property:4c69bf - read initial-letter-align for alignment points
45877
    let initial_letter_align = if dom_declared & DOM_HAS_INITIAL_LETTER_ALIGN != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_initial_letter_align(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .map(|a| match a {
                azul_css::props::style::text::StyleInitialLetterAlign::Auto => text3::cache::InitialLetterAlign::Auto,
                azul_css::props::style::text::StyleInitialLetterAlign::Alphabetic => text3::cache::InitialLetterAlign::Alphabetic,
                azul_css::props::style::text::StyleInitialLetterAlign::Hanging => text3::cache::InitialLetterAlign::Hanging,
                azul_css::props::style::text::StyleInitialLetterAlign::Ideographic => text3::cache::InitialLetterAlign::Ideographic,
            })
            .unwrap_or(text3::cache::InitialLetterAlign::Auto)
    } else {
45877
        text3::cache::InitialLetterAlign::Auto
    };
    // +spec:display-property:5af252 - initial-letter on inline-level box not at line start uses normal
    // +spec:text-alignment-spacing:a17609 - sunken initial letters suppress letter-spacing and justification (not word-spacing) with adjacent content
    // +spec:display-property:68ab22 - initial-letter only applies in IFC (inline-level);
    // float!=none or position!=static causes display to compute to block (BFC), so
    // initial-letter naturally does not apply to those elements
    // +spec:writing-modes:c89d19 - initial-letter block-axis positioning: sink determines block offset
    // +spec:display-property:b67500 - initial-letter size/sink: values other than normal make box an initial letter box (inline-level, in-flow)
    // +spec:display-property:416f27 - initial-letter sink defaults to "drop" (sink = size floored) when omitted
45877
    let initial_letter = if dom_declared & DOM_HAS_INITIAL_LETTER != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_initial_letter(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .map(|il| {
                use std::num::NonZeroUsize;
                let sink = match il.sink {
                    azul_css::corety::OptionU32::Some(s) => s,
                    azul_css::corety::OptionU32::None => il.size, // "drop" assumed: sink = size
                };
                text3::cache::InitialLetter {
                    size: il.size as f32,
                    sink,
                    count: NonZeroUsize::new(1).unwrap(),
                    align: initial_letter_align,
                }
            })
    } else {
45877
        None
    };
    // If initial-letter is set, compute the drop cap exclusion area and add it
    // to the shape exclusions so that text wraps around the enlarged letter.
    // +spec:box-model:d4adf6 - ancestor inline boundaries excluded via geometric exclusion
    // +spec:floats:c5e23f - floats in subsequent lines adjacent to a sunk initial letter must clear it
45877
    if let Some(ref il) = initial_letter {
        let lh_n = line_height_value.inner.normalized();
        let computed_line_height = if lh_n < 0.0 { -lh_n } else { lh_n * font_size };
        let (letter_w, letter_h) = layout_initial_letter(
            il.size,
            il.sink,
            constraints.available_size.width,
            computed_line_height,
        );
        if letter_w > 0.0 && letter_h > 0.0 {
            // Place the exclusion at the inline-start (x=0, y=0 relative to the IFC).
            // This creates a rectangular exclusion that text flows around.
            shape_exclusions.push(ShapeBoundary::Rectangle(crate::text3::cache::Rect {
                x: 0.0,
                y: 0.0,
                width: letter_w,
                height: letter_h,
            }));
        }
45877
    }
    // Get line-clamp for limiting visible lines
45877
    let line_clamp = if dom_declared & DOM_HAS_LINE_CLAMP != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_line_clamp(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .and_then(|lc| std::num::NonZeroUsize::new(lc.max_lines))
    } else {
45877
        None
    };
    // Get hanging-punctuation for hanging punctuation marks
45877
    let hanging_punctuation = if dom_declared & DOM_HAS_HANGING_PUNCTUATION != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_hanging_punctuation(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .map(|hp| hp.is_enabled())
            .unwrap_or(false)
    } else {
45877
        false
    };
    // Get text-combine-upright for vertical text combination
    // +spec:line-breaking:9f150a - text-combine-upright:all composes glyphs horizontally, ignoring letter-spacing and forced line breaks
    // +spec:line-breaking:1b88cd - text-combine-upright:all layout: inline-block with 1em square, ignoring forced line breaks
    // +spec:inline-formatting-context:c8d8d9 - text-combine-upright compression passed to text shaping engine
    // +spec:inline-formatting-context:f4ef7d - text-combine-upright layout rules (1em square composition)
45877
    let text_combine_upright = if dom_declared & DOM_HAS_TEXT_COMBINE_UPRIGHT != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_text_combine_upright(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            // +spec:display-property:6f174d - text-combine-upright horizontal-in-vertical composition
            .map(|tcu| match tcu {
                StyleTextCombineUpright::None => text3::cache::TextCombineUpright::None,
                StyleTextCombineUpright::All => text3::cache::TextCombineUpright::All,
                StyleTextCombineUpright::Digits(n) => text3::cache::TextCombineUpright::Digits(*n),
            })
    } else {
45877
        None
    };
    // Get exclusion-margin (CSS Exclusions L1) and shape-margin (CSS Shapes L1)
    // for shape exclusions. We sum both into a single margin knob — strictly,
    // they apply to different sources (exclusion-margin → CSS Exclusions,
    // shape-margin → shape-outside), but the layout solver currently keeps
    // a single per-IFC margin value, so the two get added.
45877
    let exclusion_margin_base = if dom_declared & DOM_HAS_EXCLUSION_MARGIN != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_exclusion_margin(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .map(|em| em.inner.get() as f32)
            .unwrap_or(0.0)
    } else {
45877
        0.0
    };
45877
    let shape_margin = if dom_declared & DOM_HAS_SHAPE_MARGIN != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_shape_margin(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .map(|sm| sm.inner.number.get() as f32)
            .unwrap_or(0.0)
    } else {
45877
        0.0
    };
45877
    let exclusion_margin = exclusion_margin_base + shape_margin;
    // Get hyphenation-language for language-specific hyphenation
45877
    let hyphenation_language = if dom_declared & DOM_HAS_HYPHENATION_LANGUAGE != 0 {
        styled_dom
            .css_property_cache
            .ptr
            .get_hyphenation_language(node_data, &id, node_state)
            .and_then(|s| s.get_property())
            .and_then(|hl| {
                #[cfg(feature = "text_layout_hyphenation")]
                {
                    use hyphenation::{Language, Load};
                    // Parse BCP 47 language code to hyphenation::Language
                    match hl.inner.as_str() {
                        "en-US" | "en" => Some(Language::EnglishUS),
                        "de-DE" | "de" => Some(Language::German1996),
                        "fr-FR" | "fr" => Some(Language::French),
                        "es-ES" | "es" => Some(Language::Spanish),
                        "it-IT" | "it" => Some(Language::Italian),
                        "pt-PT" | "pt" => Some(Language::Portuguese),
                        "nl-NL" | "nl" => Some(Language::Dutch),
                        "pl-PL" | "pl" => Some(Language::Polish),
                        "ru-RU" | "ru" => Some(Language::Russian),
                        "zh-CN" | "zh" => Some(Language::Chinese),
                        _ => None, // Unsupported language
                    }
                }
                #[cfg(not(feature = "text_layout_hyphenation"))]
                {
                    None::<crate::text3::script::Language>
                }
            })
    } else {
45877
        None
    };
    UnifiedConstraints {
45877
        exclusion_margin,
45877
        hyphenation_language,
45877
        text_indent,
45877
        text_indent_each_line,
45877
        text_indent_hanging,
45877
        initial_letter,
45877
        line_clamp,
45877
        columns,
45877
        column_gap,
45877
        hanging_punctuation,
45877
        text_wrap,
45877
        white_space_mode,
45877
        text_combine_upright,
45877
        segment_alignment: SegmentAlignment::Total,
45877
        overflow: match overflow_behaviour {
45832
            LayoutOverflow::Visible => text3::cache::OverflowBehavior::Visible,
45
            LayoutOverflow::Hidden | LayoutOverflow::Clip => text3::cache::OverflowBehavior::Hidden,
            LayoutOverflow::Scroll => text3::cache::OverflowBehavior::Scroll,
            LayoutOverflow::Auto => text3::cache::OverflowBehavior::Auto,
        },
        // Use the semantic available_width_type directly instead of converting from float.
        // This preserves MinContent/MaxContent semantics for intrinsic sizing.
45877
        available_width: constraints.available_width_type,
        // For scrollable containers (overflow: scroll/auto), don't constrain height
        // so that the full content is laid out and content_size is calculated correctly.
45877
        available_height: match overflow_behaviour {
            LayoutOverflow::Scroll | LayoutOverflow::Auto => None,
45877
            _ => Some(constraints.available_size.height),
        },
45877
        shape_boundaries, // CSS shape-inside: text flows within shape
45877
        shape_exclusions, // CSS shape-outside + floats: text wraps around shapes
45877
        writing_mode: Some(match writing_mode {
45877
            LayoutWritingMode::HorizontalTb => text3::cache::WritingMode::HorizontalTb,
            LayoutWritingMode::VerticalRl => text3::cache::WritingMode::VerticalRl,
            LayoutWritingMode::VerticalLr => text3::cache::WritingMode::VerticalLr,
        }),
45877
        direction, // Use the CSS direction property (currently defaulting to LTR)
45877
        unicode_bidi: unicode_bidi_val,
        // +spec:overflow:7ff7d1 - hyphens property: none/manual/auto hyphenation control
45877
        hyphenation: match hyphenation {
            StyleHyphens::None => text3::cache::Hyphens::None,
45877
            StyleHyphens::Manual => text3::cache::Hyphens::Manual,
            StyleHyphens::Auto => text3::cache::Hyphens::Auto,
        },
45877
        text_orientation,
        // +spec:text-alignment-spacing:6cb965 - text-align shorthand sets text-align-all (mapped here from computed value)
        // +spec:text-alignment-spacing:838967 - map text-align values (start/end/left/right/center/justify) to inline alignment
        // +spec:text-alignment-spacing:d9ea45 - property index: text-align, text-justify, letter-spacing mapped to layout
        // +spec:text-alignment-spacing:600fda - text-align values (left/right/center/justify) mapped per CSS Text §6.1
45877
        text_align: match text_align {
            StyleTextAlign::Start => text3::cache::TextAlign::Start,
            StyleTextAlign::End => text3::cache::TextAlign::End,
45877
            StyleTextAlign::Left => text3::cache::TextAlign::Left,
            StyleTextAlign::Right => text3::cache::TextAlign::Right,
            StyleTextAlign::Center => text3::cache::TextAlign::Center,
            StyleTextAlign::Justify => text3::cache::TextAlign::Justify,
        },
        // +spec:text-alignment-spacing:0ea31d - text-justify inter-word/inter-character/distribute mapped per §6.4
        // +spec:text-alignment-spacing:01244f - text-justify: none disables justification, auto uses inter-word as universal default
45877
        text_justify: match text_justify {
            LayoutTextJustify::None => text3::cache::JustifyContent::None,
45877
            LayoutTextJustify::Auto => text3::cache::JustifyContent::InterWord,
            LayoutTextJustify::InterWord => text3::cache::JustifyContent::InterWord,
            LayoutTextJustify::InterCharacter => text3::cache::JustifyContent::InterCharacter,
            LayoutTextJustify::Distribute => text3::cache::JustifyContent::InterCharacter, // distribute computes to inter-character
        },
        // +spec:line-height:79f3aa - line-height resolved: normal defaults to 1.2, <number>/<percentage> × font-size
        // Negative normalized() = absolute px value (convention from parser for "50px" etc.)
        line_height: text3::cache::LineHeight::Px({
45877
            let n = line_height_value.inner.normalized();
45877
            if n < 0.0 { -n } else { n * font_size }
        }),
        // container's first available font. Approximated as 80%/20% of font_size (typical
        // for Latin fonts). TODO: resolve actual font and use its OS/2 metrics.
45877
        strut_ascent: font_size * 0.8,
45877
        strut_descent: font_size * 0.2,
45877
        strut_x_height: font_size * 0.5, // 0.5em fallback per CSS Inline 3 Appendix A
        // ch unit width: try to get actual space width from font, fall back to 0.5 * font_size
45877
        ch_width: font_size * 0.5, // TODO: resolve from ParsedFontTrait::get_space_width()
45877
        vertical_align,
        // +spec:inline-formatting-context:48ce44 - overflow-wrap property: break at otherwise disallowed points to prevent overflow
        // +spec:line-breaking:bbb5f7 - overflow-wrap: anywhere vs break-word distinction for min-content
45877
        overflow_wrap: if word_break_css == StyleWordBreak::BreakWord {
            // +spec:line-breaking:815882 - break-word forces overflow-wrap: anywhere
            text3::cache::OverflowWrap::Anywhere
        } else {
45877
            match overflow_wrap_css {
45833
                StyleOverflowWrap::Normal => text3::cache::OverflowWrap::Normal,
                StyleOverflowWrap::Anywhere => text3::cache::OverflowWrap::Anywhere,
44
                StyleOverflowWrap::BreakWord => text3::cache::OverflowWrap::BreakWord,
            }
        },
45877
        text_align_last: match text_align_last_css {
45877
            StyleTextAlignLast::Auto => text3::cache::TextAlign::default(),
            StyleTextAlignLast::Start => text3::cache::TextAlign::Start,
            StyleTextAlignLast::End => text3::cache::TextAlign::End,
            StyleTextAlignLast::Left => text3::cache::TextAlign::Left,
            StyleTextAlignLast::Right => text3::cache::TextAlign::Right,
            StyleTextAlignLast::Center => text3::cache::TextAlign::Center,
            StyleTextAlignLast::Justify => text3::cache::TextAlign::Justify,
        },
        // +spec:line-breaking:815882 - word-break: break-word => normal + overflow-wrap: anywhere
45877
        word_break: match word_break_css {
45877
            StyleWordBreak::Normal | StyleWordBreak::BreakWord => text3::cache::WordBreak::Normal,
            StyleWordBreak::BreakAll => text3::cache::WordBreak::BreakAll,
            StyleWordBreak::KeepAll => text3::cache::WordBreak::KeepAll,
        },
        // +spec:white-space-processing:bc5f7b - line-break with break-spaces allows breaking before first space
        // CSS Text Level 3 §5.3: The line-break property affects preserved white space behavior:
        // - normal/pre-line: preserved white space at end/start of line is discarded
        // - nowrap/pre: wrapping is forbidden altogether
        // - pre-wrap: preserved white space hangs
        // - break-spaces: allows breaking before first space of a sequence
        // break-spaces allows wrapping preserved spaces to next line; for other white-space values,
        // preserved spaces at line ends are either discarded (normal, pre-line), wrapping is
        // forbidden (nowrap, pre), or they hang (pre-wrap).
45877
        line_break: match line_break_css {
45877
            StyleLineBreak::Auto => text3::cache::LineBreakStrictness::Auto,
            StyleLineBreak::Loose => text3::cache::LineBreakStrictness::Loose,
            StyleLineBreak::Normal => text3::cache::LineBreakStrictness::Normal,
            StyleLineBreak::Strict => text3::cache::LineBreakStrictness::Strict,
            StyleLineBreak::Anywhere => text3::cache::LineBreakStrictness::Anywhere,
        },
    }
45877
}
// Table Formatting Context (CSS 2.2 § 17)
// +spec:display-property:d887c0 - Table wrapper box BFC, caption-side, table grid layout (§17.4-17.5)
// +spec:positioning:930891 - Table formatting context implementation (CSS 2.2 § 17 introduction)
// +spec:inline-formatting-context:9c272d - CSS table model: row-primary structure, display-to-table-element mapping, visual formatting as rectangular grid
/// Lays out a Table Formatting Context.
/// Table column information for layout calculations
#[derive(Debug, Clone)]
pub struct TableColumnInfo {
    /// Minimum width required for this column
    pub min_width: f32,
    /// Maximum width desired for this column
    pub max_width: f32,
    /// Computed final width for this column
    pub computed_width: Option<f32>,
}
/// Information about a table cell for layout
#[derive(Debug, Clone)]
pub struct TableCellInfo {
    /// Node index in the layout tree
    pub node_index: usize,
    /// Column index (0-based)
    pub column: usize,
    /// Number of columns this cell spans
    pub colspan: usize,
    /// Row index (0-based)
    pub row: usize,
    /// Number of rows this cell spans
    pub rowspan: usize,
}
/// Table layout context - holds all information needed for table layout
#[derive(Debug)]
struct TableLayoutContext {
    /// Information about each column
    columns: Vec<TableColumnInfo>,
    /// Information about each cell
    cells: Vec<TableCellInfo>,
    /// Number of rows in the table
    num_rows: usize,
    /// Whether to use fixed or auto layout algorithm
    use_fixed_layout: bool,
    /// Computed height for each row
    row_heights: Vec<f32>,
    /// Computed baseline offset for each row (distance from row top to row baseline)
    row_baselines: Vec<f32>,
    // +spec:inline-formatting-context:440ca9 - border-collapse/border-spacing/visibility:collapse table properties (CSS 2.2 §17.5-17.6)
    /// Border collapse mode
    border_collapse: StyleBorderCollapse,
    /// Border spacing (only used when border_collapse is Separate)
    border_spacing: LayoutBorderSpacing,
    /// CSS 2.2 Section 17.4: Index of table-caption child, if any
    caption_index: Option<usize>,
    //   from display without forcing table re-layout
    /// CSS 2.2 Section 17.6: Rows with visibility:collapse (dynamic effects)
    /// Set of row indices that have visibility:collapse
    collapsed_rows: std::collections::HashSet<usize>,
    /// CSS 2.2 Section 17.6: Columns with visibility:collapse (dynamic effects)
    /// Set of column indices that have visibility:collapse
    collapsed_columns: std::collections::HashSet<usize>,
    /// Rows that are hidden-empty (zero height, border-spacing on only one side)
    hidden_empty_rows: std::collections::HashSet<usize>,
    /// Layout tree indices for each row (row index → layout node index)
    row_node_indices: Vec<usize>,
}
impl TableLayoutContext {
1452
    fn new() -> Self {
1452
        Self {
1452
            columns: Vec::new(),
1452
            cells: Vec::new(),
1452
            num_rows: 0,
1452
            use_fixed_layout: false,
1452
            row_heights: Vec::new(),
1452
            row_baselines: Vec::new(),
1452
            border_collapse: StyleBorderCollapse::Separate,
1452
            border_spacing: LayoutBorderSpacing::default(),
1452
            caption_index: None,
1452
            collapsed_rows: std::collections::HashSet::new(),
1452
            collapsed_columns: std::collections::HashSet::new(),
1452
            hidden_empty_rows: std::collections::HashSet::new(),
1452
            row_node_indices: Vec::new(),
1452
        }
1452
    }
}
// +spec:table-layout:485791 - Six superimposed table layers: table, column-group, column, row-group, row, cell (bottom to top)
// +spec:table-layout:dcdf1b - Collapsing border model: border conflict resolution uses layer priority (cell > row > row-group > column > column-group > table)
/// Source of a border in the border conflict resolution algorithm
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub enum BorderSource {
    Table = 0,
    ColumnGroup = 1,
    Column = 2,
    RowGroup = 3,
    Row = 4,
    Cell = 5,
}
/// Information about a border for conflict resolution
#[derive(Debug, Clone)]
pub struct BorderInfo {
    pub width: f32,
    pub style: BorderStyle,
    pub color: ColorU,
    pub source: BorderSource,
}
impl BorderInfo {
    pub fn new(width: f32, style: BorderStyle, color: ColorU, source: BorderSource) -> Self {
        Self {
            width,
            style,
            color,
            source,
        }
    }
    // +spec:block-formatting-context:f772ae - border style priority for table border conflict resolution
    /// Get the priority of a border style for conflict resolution
    /// Higher number = higher priority
1760
    pub fn style_priority(style: &BorderStyle) -> u8 {
1760
        match style {
            BorderStyle::Hidden => 255, // Highest - suppresses all borders
            BorderStyle::None => 0,     // Lowest - loses to everything
352
            BorderStyle::Double => 8,
836
            BorderStyle::Solid => 7,
132
            BorderStyle::Dashed => 6,
88
            BorderStyle::Dotted => 5,
88
            BorderStyle::Ridge => 4,
88
            BorderStyle::Outset => 3,
88
            BorderStyle::Groove => 2,
88
            BorderStyle::Inset => 1,
        }
1760
    }
    // +spec:box-model:2255c2 - Collapsing border conflict resolution (hidden wins, then none loses, then wider wins, then style priority)
    // +spec:box-model:b42c79 - border conflict resolution: hidden wins, then wider, then style priority, then source
    // +spec:box-model:503e9e - border conflict resolution: hidden wins, then wider, then style priority, then source priority
    // +spec:box-model:7eb217 - Border conflict resolution: hidden > none < wider > style priority > source priority > left/top
    // +spec:overflow:1fb482 - Border conflict resolution per CSS 2.2 §17.6.2.1 (hidden wins, then wider, then style priority, then source priority)
    // +spec:table-layout:882560 - Border conflict resolution (17.6.2.1): hidden wins, none loses, wider wins, style priority, source priority
    /// Compare two borders for conflict resolution per CSS 2.2 Section 17.6.2.1
    /// Returns the winning border
    // +spec:table-layout:21053b - border conflict resolution: hidden suppresses all, style priorities
    // +spec:table-layout:076617 - border conflict resolution algorithm and border style semantics in collapsing model
1276
    pub fn resolve_conflict(a: &BorderInfo, b: &BorderInfo) -> Option<BorderInfo> {
        // 1. 'hidden' wins and suppresses all borders
1276
        if a.style == BorderStyle::Hidden || b.style == BorderStyle::Hidden {
132
            return None;
1144
        }
        // 2. Filter out 'none' - if both are none, no border
1144
        let a_is_none = a.style == BorderStyle::None;
1144
        let b_is_none = b.style == BorderStyle::None;
1144
        if a_is_none && b_is_none {
            return None;
1144
        }
1144
        if a_is_none {
44
            return Some(b.clone());
1100
        }
1100
        if b_is_none {
            return Some(a.clone());
1100
        }
        // 3. Wider border wins
1100
        if a.width > b.width {
            return Some(a.clone());
1100
        }
1100
        if b.width > a.width {
220
            return Some(b.clone());
880
        }
        // 4. If same width, compare style priority
880
        let a_priority = Self::style_priority(&a.style);
880
        let b_priority = Self::style_priority(&b.style);
880
        if a_priority > b_priority {
484
            return Some(a.clone());
396
        }
396
        if b_priority > a_priority {
44
            return Some(b.clone());
352
        }
        // 5. If same style, source priority:
        // Cell > Row > RowGroup > Column > ColumnGroup > Table
352
        if a.source > b.source {
308
            return Some(a.clone());
44
        }
44
        if b.source > a.source {
            return Some(b.clone());
44
        }
        // 6. Same priority - prefer first one (left/top in LTR)
44
        Some(a.clone())
1276
    }
}
/// Get border information for a node
fn get_border_info<T: ParsedFontTrait>(
    ctx: &LayoutContext<'_, T>,
    node: &LayoutNodeHot,
    source: BorderSource,
) -> (BorderInfo, BorderInfo, BorderInfo, BorderInfo) {
    use azul_css::props::{
        basic::{
            pixel::{PhysicalSize, PropertyContext, ResolutionContext},
            ColorU,
        },
        style::BorderStyle,
    };
    use get_element_font_size;
    use get_parent_font_size;
    use get_root_font_size;
    let default_border = BorderInfo::new(
        0.0,
        BorderStyle::None,
        ColorU {
            r: 0,
            g: 0,
            b: 0,
            a: 0,
        },
        source,
    );
    let Some(dom_id) = node.dom_node_id else {
        return (
            default_border.clone(),
            default_border.clone(),
            default_border.clone(),
            default_border.clone(),
        );
    };
    let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
    let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
    let cache = &ctx.styled_dom.css_property_cache.ptr;
    // FAST PATH: compact cache for normal state
    if let Some(ref cc) = cache.compact_cache {
        let idx = dom_id.index();
        // Border styles from packed u16
        let bts = cc.get_border_top_style(idx);
        let brs = cc.get_border_right_style(idx);
        let bbs = cc.get_border_bottom_style(idx);
        let bls = cc.get_border_left_style(idx);
        // Border colors from u32 RGBA
        let make_color = |raw: u32| -> ColorU {
            if raw == 0 {
                ColorU { r: 0, g: 0, b: 0, a: 0 }
            } else {
                ColorU {
                    r: ((raw >> 24) & 0xFF) as u8,
                    g: ((raw >> 16) & 0xFF) as u8,
                    b: ((raw >> 8) & 0xFF) as u8,
                    a: (raw & 0xFF) as u8,
                }
            }
        };
        let btc = make_color(cc.get_border_top_color_raw(idx));
        let brc = make_color(cc.get_border_right_color_raw(idx));
        let bbc = make_color(cc.get_border_bottom_color_raw(idx));
        let blc = make_color(cc.get_border_left_color_raw(idx));
        // Border widths from i16 × 10
        let decode_width = |raw: i16| -> f32 {
            if raw >= azul_css::compact_cache::I16_SENTINEL_THRESHOLD {
                0.0 // sentinel → fall back to 0
            } else {
                raw as f32 / 10.0
            }
        };
        let btw = decode_width(cc.get_border_top_width_raw(idx));
        let brw = decode_width(cc.get_border_right_width_raw(idx));
        let bbw = decode_width(cc.get_border_bottom_width_raw(idx));
        let blw = decode_width(cc.get_border_left_width_raw(idx));
        let top = if bts == BorderStyle::None { default_border.clone() }
            else { BorderInfo::new(btw, bts, btc, source) };
        let right = if brs == BorderStyle::None { default_border.clone() }
            else { BorderInfo::new(brw, brs, brc, source) };
        let bottom = if bbs == BorderStyle::None { default_border.clone() }
            else { BorderInfo::new(bbw, bbs, bbc, source) };
        let left = if bls == BorderStyle::None { default_border.clone() }
            else { BorderInfo::new(blw, bls, blc, source) };
        return (top, right, bottom, left);
    }
    // SLOW PATH: full cascade resolution
    let cache = &ctx.styled_dom.css_property_cache.ptr;
    // Create resolution context for border-width (em/rem support, no % support)
    let element_font_size = get_element_font_size(ctx.styled_dom, dom_id, &node_state);
    let parent_font_size = get_parent_font_size(ctx.styled_dom, dom_id, &node_state);
    let root_font_size = get_root_font_size(ctx.styled_dom, &node_state);
    let resolution_context = ResolutionContext {
        element_font_size,
        parent_font_size,
        root_font_size,
        // Not used for border-width
        containing_block_size: PhysicalSize::new(0.0, 0.0),
        // Not used for border-width
        element_size: None,
        viewport_size: PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height),
    };
    // Top border
    let top = cache
        .get_border_top_style(node_data, &dom_id, &node_state)
        .and_then(|s| s.get_property())
        .map(|style_val| {
            let width = cache
                .get_border_top_width(node_data, &dom_id, &node_state)
                .and_then(|w| w.get_property())
                .map(|w| {
                    w.inner
                        .resolve_with_context(&resolution_context, PropertyContext::BorderWidth)
                })
                .unwrap_or(0.0);
            let color = cache
                .get_border_top_color(node_data, &dom_id, &node_state)
                .and_then(|c| c.get_property())
                .map(|c| c.inner)
                .unwrap_or(ColorU {
                    r: 0,
                    g: 0,
                    b: 0,
                    a: 255,
                });
            BorderInfo::new(width, style_val.inner, color, source)
        })
        .unwrap_or_else(|| default_border.clone());
    // Right border
    let right = cache
        .get_border_right_style(node_data, &dom_id, &node_state)
        .and_then(|s| s.get_property())
        .map(|style_val| {
            let width = cache
                .get_border_right_width(node_data, &dom_id, &node_state)
                .and_then(|w| w.get_property())
                .map(|w| {
                    w.inner
                        .resolve_with_context(&resolution_context, PropertyContext::BorderWidth)
                })
                .unwrap_or(0.0);
            let color = cache
                .get_border_right_color(node_data, &dom_id, &node_state)
                .and_then(|c| c.get_property())
                .map(|c| c.inner)
                .unwrap_or(ColorU {
                    r: 0,
                    g: 0,
                    b: 0,
                    a: 255,
                });
            BorderInfo::new(width, style_val.inner, color, source)
        })
        .unwrap_or_else(|| default_border.clone());
    // Bottom border
    let bottom = cache
        .get_border_bottom_style(node_data, &dom_id, &node_state)
        .and_then(|s| s.get_property())
        .map(|style_val| {
            let width = cache
                .get_border_bottom_width(node_data, &dom_id, &node_state)
                .and_then(|w| w.get_property())
                .map(|w| {
                    w.inner
                        .resolve_with_context(&resolution_context, PropertyContext::BorderWidth)
                })
                .unwrap_or(0.0);
            let color = cache
                .get_border_bottom_color(node_data, &dom_id, &node_state)
                .and_then(|c| c.get_property())
                .map(|c| c.inner)
                .unwrap_or(ColorU {
                    r: 0,
                    g: 0,
                    b: 0,
                    a: 255,
                });
            BorderInfo::new(width, style_val.inner, color, source)
        })
        .unwrap_or_else(|| default_border.clone());
    // Left border
    let left = cache
        .get_border_left_style(node_data, &dom_id, &node_state)
        .and_then(|s| s.get_property())
        .map(|style_val| {
            let width = cache
                .get_border_left_width(node_data, &dom_id, &node_state)
                .and_then(|w| w.get_property())
                .map(|w| {
                    w.inner
                        .resolve_with_context(&resolution_context, PropertyContext::BorderWidth)
                })
                .unwrap_or(0.0);
            let color = cache
                .get_border_left_color(node_data, &dom_id, &node_state)
                .and_then(|c| c.get_property())
                .map(|c| c.inner)
                .unwrap_or(ColorU {
                    r: 0,
                    g: 0,
                    b: 0,
                    a: 255,
                });
            BorderInfo::new(width, style_val.inner, color, source)
        })
        .unwrap_or_else(|| default_border.clone());
    (top, right, bottom, left)
}
// +spec:table-layout:c5e446 - table-layout property (auto|fixed) controls layout algorithm selection
/// Get the table-layout property for a table node
1452
fn get_table_layout_property<T: ParsedFontTrait>(
1452
    ctx: &LayoutContext<'_, T>,
1452
    node: &LayoutNodeHot,
1452
) -> LayoutTableLayout {
1452
    let Some(dom_id) = node.dom_node_id else {
        return LayoutTableLayout::Auto;
    };
1452
    let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
1452
    let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
1452
    ctx.styled_dom
1452
        .css_property_cache
1452
        .ptr
1452
        .get_table_layout(node_data, &dom_id, &node_state)
1452
        .and_then(|prop| prop.get_property().copied())
1452
        .unwrap_or(LayoutTableLayout::Auto)
1452
}
/// Get the border-collapse property for a table node
1452
fn get_border_collapse_property<T: ParsedFontTrait>(
1452
    ctx: &LayoutContext<'_, T>,
1452
    node: &LayoutNodeHot,
1452
) -> StyleBorderCollapse {
1452
    let Some(dom_id) = node.dom_node_id else {
        return StyleBorderCollapse::Separate;
    };
    // FAST PATH: compact cache
1452
    if let Some(ref cc) = ctx.styled_dom.css_property_cache.ptr.compact_cache {
1452
        return cc.get_border_collapse(dom_id.index());
    }
    let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
    let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
    ctx.styled_dom
        .css_property_cache
        .ptr
        .get_border_collapse(node_data, &dom_id, &node_state)
        .and_then(|prop| prop.get_property().copied())
        .unwrap_or(StyleBorderCollapse::Separate)
1452
}
/// Get the border-spacing property for a table node
1452
fn get_border_spacing_property<T: ParsedFontTrait>(
1452
    ctx: &LayoutContext<'_, T>,
1452
    node: &LayoutNodeHot,
1452
) -> LayoutBorderSpacing {
1452
    if let Some(dom_id) = node.dom_node_id {
        // FAST PATH: compact cache
1452
        if let Some(ref cc) = ctx.styled_dom.css_property_cache.ptr.compact_cache {
1452
            let idx = dom_id.index();
1452
            let h_raw = cc.get_border_spacing_h_raw(idx);
1452
            let v_raw = cc.get_border_spacing_v_raw(idx);
            // If both are non-sentinel, use compact values
1452
            if h_raw < azul_css::compact_cache::I16_SENTINEL_THRESHOLD
1452
                && v_raw < azul_css::compact_cache::I16_SENTINEL_THRESHOLD
            {
1452
                return LayoutBorderSpacing::new_separate(
1452
                    azul_css::props::basic::pixel::PixelValue::px(h_raw as f32 / 10.0),
1452
                    azul_css::props::basic::pixel::PixelValue::px(v_raw as f32 / 10.0),
                );
            }
            // sentinel → fall through to slow path
        }
        let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
        let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
        if let Some(prop) = ctx.styled_dom.css_property_cache.ptr.get_border_spacing(
            node_data,
            &dom_id,
            &node_state,
        ) {
            if let Some(value) = prop.get_property() {
                return *value;
            }
        }
    }
    LayoutBorderSpacing::default() // Default: 0
1452
}
/// Get the empty-cells property for a table-cell node.
/// Returns Show (default) or Hide.
1496
fn get_empty_cells_property<T: ParsedFontTrait>(
1496
    ctx: &LayoutContext<'_, T>,
1496
    node: &LayoutNodeHot,
1496
) -> StyleEmptyCells {
1496
    let Some(dom_id) = node.dom_node_id else {
        return StyleEmptyCells::Show;
    };
1496
    let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
1496
    let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
1496
    ctx.styled_dom
1496
        .css_property_cache
1496
        .ptr
1496
        .get_empty_cells(node_data, &dom_id, &node_state)
1496
        .and_then(|prop| prop.get_property().copied())
1496
        .unwrap_or(StyleEmptyCells::Show)
1496
}
/// CSS 2.2 Section 17.4 - Tables in the visual formatting model:
///
/// "The caption box is a block box that retains its own content, padding,
/// border, and margin areas. The caption-side property specifies the position
/// of the caption box with respect to the table box."
///
/// Get the caption-side property for a table node.
/// Returns Top (default) or Bottom.
1452
fn get_caption_side_property<T: ParsedFontTrait>(
1452
    ctx: &LayoutContext<'_, T>,
1452
    node: &LayoutNodeHot,
1452
) -> StyleCaptionSide {
1452
    if let Some(dom_id) = node.dom_node_id {
1452
        let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
1452
        let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
        if let Some(prop) =
1452
            ctx.styled_dom
1452
                .css_property_cache
1452
                .ptr
1452
                .get_caption_side(node_data, &dom_id, &node_state)
        {
            if let Some(value) = prop.get_property() {
                return *value;
            }
1452
        }
    }
1452
    StyleCaptionSide::Top // Default per CSS 2.2
1452
}
//   removes entire row or column from display; space made available for other content;
//   spanned content clipped; does not otherwise affect table layout
// +spec:inline-formatting-context:9f5f31 - visibility:collapse for table rows/columns, border-collapse and border-spacing
/// CSS 2.2 Section 17.6 - Dynamic row and column effects:
///
// +spec:box-model:547563 - visibility:collapse removes table rows/columns; elsewhere same as hidden
/// "The 'visibility' value 'collapse' removes a row or column from display,
/// but it has a different effect than 'visibility: hidden' on other elements.
/// When a row or column is collapsed, the space normally occupied by the row
/// or column is removed."
///
/// Check if a node has visibility:collapse set.
///
/// This is used for table rows and columns to optimize dynamic hiding.
/// // +spec:overflow:ebb1f9 - For non-table elements, collapse == hidden (no special handling needed)
1496
fn is_visibility_collapsed<T: ParsedFontTrait>(
1496
    ctx: &LayoutContext<'_, T>,
1496
    node: &LayoutNodeHot,
1496
) -> bool {
1496
    if let Some(dom_id) = node.dom_node_id {
1496
        let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
1496
        if let MultiValue::Exact(value) = get_visibility(ctx.styled_dom, dom_id, &node_state) {
1496
            return matches!(value, StyleVisibility::Collapse);
        }
    }
    false
1496
}
// +spec:overflow:af97a8 - empty-cells in separated borders model; collapsing border overflow
// +spec:table-layout:dcdf1b - empty-cells property controls rendering of borders/backgrounds around empty cells in separated borders model
/// CSS 2.2 Section 17.6.1.1 - Borders and Backgrounds around empty cells
///
/// In the separated borders model, the 'empty-cells' property controls the rendering of
/// borders and backgrounds around cells that have no visible content. Empty means it has no
/// children, or has children that are only collapsed whitespace."
///
/// Check if a table cell is empty (has no visible content).
///
/// This is used by the rendering pipeline to decide whether to paint borders/backgrounds
/// when empty-cells: hide is set in separated border model.
///
//   in-flow content (including empty elements) other than collapsed whitespace
/// A cell is considered empty if:
///
/// - It has no children, OR
/// - It has children but no inline_layout_result (no rendered content)
///
/// Note: Full whitespace detection would require checking text content during rendering.
/// This function provides a basic check suitable for layout phase.
fn is_cell_empty(tree: &LayoutTree, cell_index: usize) -> bool {
    if tree.get(cell_index).is_none() {
        return true; // Invalid cell is considered empty
    }
    // No children = empty
    if tree.children(cell_index).is_empty() {
        return true;
    }
    // If cell has an inline layout result, check if it's empty
    if let Some(warm_node) = tree.warm(cell_index) {
        if let Some(ref cached_layout) = warm_node.inline_layout_result {
            // Check if inline layout has any rendered content
            // Empty inline layouts have no items (glyphs/fragments)
            // Note: This is a heuristic - full detection requires text content analysis
            return cached_layout.layout.items.is_empty();
        }
    }
    // Check if all children have no content
    // A more thorough check would recursively examine all descendants
    //
    // For now, we use a simple heuristic: if there are children, assume not empty
    // unless proven otherwise by inline_layout_result
    // Cell with children but no inline layout = likely has block-level content = not empty
    false
}
/// Main function to layout a table formatting context
// +spec:table-layout:235e8e - CSS 2.2 §17.1-17.2 table model: fixed/auto algorithms, row/column/cell/caption structure
// +spec:table-layout:a6422d - CSS table model: table structure analysis, row/column/cell layout, caption, border-collapse
1452
pub fn layout_table_fc<T: ParsedFontTrait>(
1452
    ctx: &mut LayoutContext<'_, T>,
1452
    tree: &mut LayoutTree,
1452
    text_cache: &mut crate::font_traits::TextLayoutCache,
1452
    node_index: usize,
1452
    constraints: &LayoutConstraints,
1452
) -> Result<LayoutOutput> {
1452
    debug_log!(ctx, "Laying out table");
1452
    debug_table_layout!(
1452
        ctx,
1452
        "node_index={}, available_size={:?}, writing_mode={:?}",
        node_index,
        constraints.available_size,
        constraints.writing_mode
    );
    // Multi-pass table layout algorithm:
    //
    // 1. Analyze table structure - identify rows, cells, columns
    // 2. Determine table-layout property (fixed vs auto)
    // 3. Calculate column widths
    // 4. Layout cells and calculate row heights
    // 5. Position cells in final grid
    // Get the table node to read CSS properties
1452
    let table_node = tree
1452
        .get(node_index)
1452
        .ok_or(LayoutError::InvalidTree)?
1452
        .clone();
    // Calculate the table's border-box width for column distribution
    // This accounts for the table's own width property (e.g., width: 100%)
1452
    let table_border_box_width = if let Some(dom_id) = table_node.dom_node_id {
        // Use calculate_used_size_for_node to resolve table width (respects width:100%)
1452
        let intrinsic = tree.warm(node_index).and_then(|w| w.intrinsic_sizes).unwrap_or_default();
1452
        let containing_block_size = LogicalSize {
1452
            width: constraints.available_size.width,
1452
            height: constraints.available_size.height,
1452
        };
1452
        let table_bp = table_node.box_props.unpack();
1452
        let table_size = crate::solver3::sizing::calculate_used_size_for_node(
1452
            ctx.styled_dom,
1452
            Some(dom_id),
1452
            &containing_block_size,
1452
            intrinsic,
1452
            &table_bp,
1452
            &ctx.viewport_size,
        )?;
1452
        table_size.width
    } else {
        constraints.available_size.width
    };
    // Subtract padding and border to get content-box width for column distribution
1452
    let tbp = table_node.box_props.unpack();
1452
    let table_content_box_width = {
1452
        let padding_width = tbp.padding.left + tbp.padding.right;
1452
        let border_width = tbp.border.left + tbp.border.right;
1452
        (table_border_box_width - padding_width - border_width).max(0.0)
    };
1452
    debug_table_layout!(ctx, "Table Layout Debug");
1452
    debug_table_layout!(ctx, "Node index: {}", node_index);
1452
    debug_table_layout!(
1452
        ctx,
1452
        "Available size from parent: {:.2} x {:.2}",
        constraints.available_size.width,
        constraints.available_size.height
    );
1452
    debug_table_layout!(ctx, "Table border-box width: {:.2}", table_border_box_width);
1452
    debug_table_layout!(
1452
        ctx,
1452
        "Table content-box width: {:.2}",
        table_content_box_width
    );
1452
    debug_table_layout!(
1452
        ctx,
1452
        "Table padding: L={:.2} R={:.2}",
        tbp.padding.left,
        tbp.padding.right
    );
1452
    debug_table_layout!(
1452
        ctx,
1452
        "Table border: L={:.2} R={:.2}",
        tbp.border.left,
        tbp.border.right
    );
1452
    debug_table_layout!(ctx, "=");
    // Phase 1: Analyze table structure
1452
    let mut table_ctx = analyze_table_structure(tree, node_index, ctx)?;
    // +spec:table-layout:ff5671 - table-layout property (fixed vs auto) controls column width algorithm
    // +spec:width-calculation:7a5b23 - table-layout property determines fixed vs auto algorithm (CSS 2.2 §17.5.2)
    // Phase 2: Read CSS properties and determine layout algorithm
1452
    let table_layout = get_table_layout_property(ctx, &table_node);
1452
    table_ctx.use_fixed_layout = matches!(table_layout, LayoutTableLayout::Fixed);
    // +spec:containing-block:cc1453 - collapsing border model: border-collapse property drives table border handling
    // Read border properties
1452
    table_ctx.border_collapse = get_border_collapse_property(ctx, &table_node);
1452
    table_ctx.border_spacing = get_border_spacing_property(ctx, &table_node);
1452
    debug_log!(
1452
        ctx,
1452
        "Table layout: {:?}, border-collapse: {:?}, border-spacing: {:?}",
        table_layout,
        table_ctx.border_collapse,
        table_ctx.border_spacing
    );
    // +spec:width-calculation:431d60 - fixed vs auto table layout column width algorithms (CSS 2.2 §17.5.2.1, §17.5.2.2)
    // Phase 3: Calculate column widths
1452
    if table_ctx.use_fixed_layout {
        // DEBUG: Log available width passed into fixed column calculation
        debug_table_layout!(
            ctx,
            "FIXED layout: table_content_box_width={:.2}",
            table_content_box_width
        );
        calculate_column_widths_fixed(ctx, tree, &mut table_ctx, table_content_box_width);
    } else {
        // Pass table_content_box_width for column distribution in auto layout
1452
        calculate_column_widths_auto_with_width(
1452
            &mut table_ctx,
1452
            tree,
1452
            text_cache,
1452
            ctx,
1452
            constraints,
1452
            table_content_box_width,
        )?;
    }
1452
    debug_table_layout!(ctx, "After column width calculation:");
1452
    debug_table_layout!(ctx, "  Number of columns: {}", table_ctx.columns.len());
3696
    for (i, col) in table_ctx.columns.iter().enumerate() {
3696
        debug_table_layout!(
3696
            ctx,
3696
            "  Column {}: width={:.2}",
            i,
3696
            col.computed_width.unwrap_or(0.0)
        );
    }
1452
    let total_col_width: f32 = table_ctx
1452
        .columns
1452
        .iter()
1452
        .filter_map(|c| c.computed_width)
1452
        .sum();
1452
    debug_table_layout!(ctx, "  Total column width: {:.2}", total_col_width);
    // Phase 4: Calculate row heights based on cell content
1452
    calculate_row_heights(&mut table_ctx, tree, text_cache, ctx, constraints)?;
    // Phase 5: Position cells in final grid and collect positions
1452
    let mut cell_positions =
1452
        position_table_cells(&mut table_ctx, tree, ctx, node_index, constraints)?;
    // Calculate final table size including border-spacing
1452
    let mut table_width: f32 = table_ctx
1452
        .columns
1452
        .iter()
1452
        .filter_map(|col| col.computed_width)
1452
        .sum();
1452
    let mut table_height: f32 = table_ctx.row_heights.iter().sum();
1452
    debug_table_layout!(
1452
        ctx,
1452
        "After calculate_row_heights: table_height={:.2}, row_heights={:?}",
        table_height,
        table_ctx.row_heights
    );
    // +spec:box-model:494f6b - collapsing border model: row-width formula and table border width computation
    // +spec:box-model:e7d0a3 - Separated borders model: border-spacing, empty-cells, collapsing border width calculation
    // +spec:box-sizing:ee702c - separated borders model: border-spacing between adjoining cells
    // Add border-spacing to table size if border-collapse is separate
    // +spec:box-model:acb81f - separated borders model: border-spacing between adjoining cell borders
    // +spec:box-model:e480b1 - table width = left inner padding edge to right inner padding edge (including border-spacing)
1452
    if table_ctx.border_collapse == StyleBorderCollapse::Separate {
        use get_element_font_size;
        use get_parent_font_size;
        use get_root_font_size;
        use PhysicalSize;
        use PropertyContext;
        use ResolutionContext;
1452
        let styled_dom = ctx.styled_dom;
1452
        let table_id = tree.nodes[node_index].dom_node_id.unwrap();
1452
        let table_state = &styled_dom.styled_nodes.as_container()[table_id].styled_node_state;
1452
        let spacing_context = ResolutionContext {
1452
            element_font_size: get_element_font_size(styled_dom, table_id, table_state),
1452
            parent_font_size: get_parent_font_size(styled_dom, table_id, table_state),
1452
            root_font_size: get_root_font_size(styled_dom, table_state),
1452
            containing_block_size: PhysicalSize::new(0.0, 0.0),
1452
            element_size: None,
1452
            viewport_size: PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height),
1452
        };
1452
        let h_spacing = table_ctx
1452
            .border_spacing
1452
            .horizontal
1452
            .resolve_with_context(&spacing_context, PropertyContext::Other)
1452
            .max(0.0);
1452
        let v_spacing = table_ctx
1452
            .border_spacing
1452
            .vertical
1452
            .resolve_with_context(&spacing_context, PropertyContext::Other)
1452
            .max(0.0);
        // Add spacing: left + (n-1 between columns) + right = n+1 spacings
1452
        let num_cols = table_ctx.columns.len();
1452
        if num_cols > 0 {
1452
            table_width += h_spacing * (num_cols + 1) as f32;
1452
        }
        // Add spacing: top + (n-1 between rows) + bottom = n+1 spacings
1452
        if table_ctx.num_rows > 0 {
1452
            let full_spacings = (table_ctx.num_rows + 1) as f32;
1452
            // Each hidden-empty row loses one side of border-spacing
1452
            let hidden_empty_count = table_ctx.hidden_empty_rows.len() as f32;
1452
            table_height += v_spacing * (full_spacings - hidden_empty_count);
1452
        }
    }
    // +spec:table-layout:24dbf9 - §17.4 table wrapper box model: caption positioning, BFC establishment
    // +spec:width-calculation:600f98 - caption-side positions caption above/below table box (CSS 2.2 §17.4)
    // CSS 2.2 Section 17.4: Layout and position the caption if present
    //
    // "The caption box is a block box that retains its own content,
    // padding, border, and margin areas."
1452
    let caption_side = get_caption_side_property(ctx, &table_node);
1452
    let mut caption_height = 0.0;
1452
    let mut table_y_offset = 0.0;
1452
    if let Some(caption_idx) = table_ctx.caption_index {
        debug_log!(
            ctx,
            "Laying out caption with caption-side: {:?}",
            caption_side
        );
        // Layout caption as a block with the table's width as available width
        let caption_constraints = LayoutConstraints {
            available_size: LogicalSize {
                width: table_width,
                height: constraints.available_size.height,
            },
            writing_mode: constraints.writing_mode,
            writing_mode_ctx: constraints.writing_mode_ctx,
            bfc_state: None, // Caption creates its own BFC
            text_align: constraints.text_align,
            containing_block_size: constraints.containing_block_size,
            available_width_type: Text3AvailableSpace::Definite(table_width),
        };
        // Layout the caption node
        let mut empty_float_cache = HashMap::new();
        let caption_result = layout_formatting_context(
            ctx,
            tree,
            text_cache,
            caption_idx,
            &caption_constraints,
            &mut empty_float_cache,
        )?;
        caption_height = caption_result.output.overflow_size.height;
        let caption_position = match caption_side {
            StyleCaptionSide::Top => {
                // Caption on top: position at y=0, table starts below caption
                table_y_offset = caption_height;
                LogicalPosition { x: 0.0, y: 0.0 }
            }
            StyleCaptionSide::Bottom => {
                // Caption on bottom: table starts at y=0, caption below table
                LogicalPosition {
                    x: 0.0,
                    y: table_height,
                }
            }
        };
        // Add caption position to the positions map
        cell_positions.insert(caption_idx, caption_position);
        debug_log!(
            ctx,
            "Caption positioned at x={:.2}, y={:.2}, height={:.2}",
            caption_position.x,
            caption_position.y,
            caption_height
        );
1452
    }
    // Adjust all table cell positions if caption is on top
1452
    if table_y_offset > 0.0 {
        debug_log!(
            ctx,
            "Adjusting table cells by y offset: {:.2}",
            table_y_offset
        );
        // Adjust cell positions in the map
        for cell_info in &table_ctx.cells {
            if let Some(pos) = cell_positions.get_mut(&cell_info.node_index) {
                pos.y += table_y_offset;
            }
        }
1452
    }
1452
    let total_height = table_height + caption_height;
1452
    debug_table_layout!(ctx, "Final table dimensions:");
1452
    debug_table_layout!(ctx, "  Content width (columns): {:.2}", table_width);
1452
    debug_table_layout!(ctx, "  Content height (rows): {:.2}", table_height);
1452
    debug_table_layout!(ctx, "  Caption height: {:.2}", caption_height);
1452
    debug_table_layout!(ctx, "  Total height: {:.2}", total_height);
1452
    debug_table_layout!(ctx, "End Table Debug");
    // Create output with the table's final size and cell positions
    // +spec:box-model:52fcfe - overflow_size must include borders that spill into margin in collapsing border model
1452
    let output = LayoutOutput {
1452
        overflow_size: LogicalSize {
1452
            width: table_width,
1452
            height: total_height,
1452
        },
1452
        // Cell positions calculated in position_table_cells
1452
        positions: cell_positions,
1452
        // line box or first in-flow table-row; if none, bottom of content edge
1452
        // TODO: implement proper table baseline propagation
1452
        baseline: None,
1452
    };
1452
    Ok(output)
1452
}
// +spec:display-property:f47f8a - Table structure analysis: caption positioning, row/column/row-group traversal per CSS 2.2 §17.4-17.5
/// Analyze the table structure to identify rows, cells, and columns
1452
fn analyze_table_structure<T: ParsedFontTrait>(
1452
    tree: &LayoutTree,
1452
    table_index: usize,
1452
    ctx: &mut LayoutContext<'_, T>,
1452
) -> Result<TableLayoutContext> {
1452
    let mut table_ctx = TableLayoutContext::new();
1452
    let table_node = tree.get(table_index).ok_or(LayoutError::InvalidTree)?;
    // +spec:width-calculation:0a2766 - table internal elements form rectangular grid of rows/columns (CSS 2.2 §17.5)
    // CSS 2.2 Section 17.4: A table may have one table-caption child.
    // Traverse children to find caption, columns/colgroups, rows, and row groups
1496
    for &child_idx in tree.children(table_index) {
1496
        if let Some(child) = tree.get(child_idx) {
            // Check if this is a table caption
1496
            if matches!(child.formatting_context, FormattingContext::TableCaption) {
                debug_log!(ctx, "Found table caption at index {}", child_idx);
                table_ctx.caption_index = Some(child_idx);
                continue;
1496
            }
            // CSS 2.2 Section 17.2: Check for column groups
1496
            if matches!(
1496
                child.formatting_context,
                FormattingContext::TableColumnGroup
            ) {
                analyze_table_colgroup(tree, child_idx, &mut table_ctx, ctx)?;
                continue;
1496
            }
            // Check if this is a table row or row group
1496
            match child.formatting_context {
                FormattingContext::TableRow => {
1496
                    analyze_table_row(tree, child_idx, &mut table_ctx, ctx)?;
                }
                FormattingContext::TableRowGroup => {
                    // Process rows within the row group
                    for &row_idx in tree.children(child_idx) {
                        if let Some(row) = tree.get(row_idx) {
                            if matches!(row.formatting_context, FormattingContext::TableRow) {
                                analyze_table_row(tree, row_idx, &mut table_ctx, ctx)?;
                            }
                        }
                    }
                }
                _ => {}
            }
        }
    }
1452
    debug_log!(
1452
        ctx,
1452
        "Table structure: {} rows, {} columns, {} cells{}",
        table_ctx.num_rows,
1452
        table_ctx.columns.len(),
1452
        table_ctx.cells.len(),
1452
        if table_ctx.caption_index.is_some() {
            ", has caption"
        } else {
1452
            ""
        }
    );
1452
    Ok(table_ctx)
1452
}
/// Analyze a table column group to identify columns and track collapsed columns
///
/// - CSS 2.2 Section 17.2: Column groups contain columns
/// - CSS 2.2 Section 17.6: Columns can have visibility:collapse
fn analyze_table_colgroup<T: ParsedFontTrait>(
    tree: &LayoutTree,
    colgroup_index: usize,
    table_ctx: &mut TableLayoutContext,
    ctx: &mut LayoutContext<'_, T>,
) -> Result<()> {
    let colgroup_node = tree.get(colgroup_index).ok_or(LayoutError::InvalidTree)?;
    // Check if the colgroup itself has visibility:collapse
    if is_visibility_collapsed(ctx, colgroup_node) {
        // All columns in this group should be collapsed
        // TODO: For now, just mark the group (actual column indices will be determined later)
        debug_log!(
            ctx,
            "Column group at index {} has visibility:collapse",
            colgroup_index
        );
    }
    // Check for individual column elements within the group
    for &col_idx in tree.children(colgroup_index) {
        if let Some(col_node) = tree.get(col_idx) {
            // Note: Individual columns don't have a FormattingContext::TableColumn
            // They are represented as children of TableColumnGroup
            // Check visibility:collapse on each column
            if is_visibility_collapsed(ctx, col_node) {
                // We need to determine the actual column index this represents
                // For now, we'll track it during cell analysis
                debug_log!(ctx, "Column at index {} has visibility:collapse", col_idx);
            }
        }
    }
    Ok(())
}
// +spec:display-property:7f167c - Table grid cell placement: rows fill table top-to-bottom, cells placed left-to-right with colspan/rowspan
/// Analyze a table row to identify cells and update column count
1496
fn analyze_table_row<T: ParsedFontTrait>(
1496
    tree: &LayoutTree,
1496
    row_index: usize,
1496
    table_ctx: &mut TableLayoutContext,
1496
    ctx: &mut LayoutContext<'_, T>,
1496
) -> Result<()> {
    // +spec:inline-formatting-context:3f8091 - table visual layout: cells occupy grid cells, row/column spanning
1496
    let row_node = tree.get(row_index).ok_or(LayoutError::InvalidTree)?;
1496
    let row_num = table_ctx.num_rows;
1496
    table_ctx.num_rows += 1;
    // Track the layout tree index for this row (for positioning/painting)
1496
    if table_ctx.row_node_indices.len() <= row_num {
1496
        table_ctx.row_node_indices.resize(row_num + 1, 0);
1496
    }
1496
    table_ctx.row_node_indices[row_num] = row_index;
    // CSS 2.2 Section 17.6: Check if this row has visibility:collapse
1496
    if is_visibility_collapsed(ctx, row_node) {
        debug_log!(ctx, "Row {} has visibility:collapse", row_num);
        table_ctx.collapsed_rows.insert(row_num);
1496
    }
1496
    let mut col_index = 0;
3740
    for &cell_idx in tree.children(row_index) {
3740
        if let Some(cell) = tree.get(cell_idx) {
3740
            if matches!(cell.formatting_context, FormattingContext::TableCell) {
                // Get colspan and rowspan (TODO: from CSS properties)
3740
                let colspan = 1; // TODO: Get from CSS
3740
                let rowspan = 1; // TODO: Get from CSS
3740
                let cell_info = TableCellInfo {
3740
                    node_index: cell_idx,
3740
                    column: col_index,
3740
                    colspan,
3740
                    row: row_num,
3740
                    rowspan,
3740
                };
3740
                table_ctx.cells.push(cell_info);
                // Update column count
3740
                let max_col = col_index + colspan;
7436
                while table_ctx.columns.len() < max_col {
3696
                    table_ctx.columns.push(TableColumnInfo {
3696
                        min_width: 0.0,
3696
                        max_width: 0.0,
3696
                        computed_width: None,
3696
                    });
3696
                }
3740
                col_index += colspan;
            }
        }
    }
1496
    Ok(())
1496
}
// +spec:overflow:66f584 - Fixed table layout: cells use overflow property to clip overflowing content
// +spec:positioning:46070a - Fixed table layout (17.5.2.1) and auto table layout (17.5.2.2) column width algorithms
// +spec:table-layout:875401 - Fixed table layout algorithm (17.5.2.1): column widths from first-row cells, remaining columns divide space equally, table width = max(width property, sum of columns)
/// Calculate column widths using the fixed table layout algorithm
/// // +spec:overflow:de613c - Fixed table layout algorithm (CSS 2.2 Section 17.5.2.1)
// +spec:table-layout:8b72b3 - fixed table layout: column width from column elements/first-row cells, remaining columns equal division
///
/// CSS 2.2 Section 17.5.2.1: In fixed table layout, the horizontal layout
/// does not depend on cell contents. Column widths are determined by:
/// 1. Column elements with explicit (non-auto) width
/// 2. First-row cells with explicit (non-auto) width
/// 3. Remaining columns equally divide remaining horizontal space
///
/// CSS 2.2 Section 17.6: Columns with visibility:collapse are excluded
/// from width calculations
// +spec:table-layout:c5e446 - Fixed table layout algorithm: column widths from col elements or first-row cells, remaining columns divide equally
/// +spec:width-calculation:8c958a - Fixed table layout: column widths from col elements, first-row cells, then equal distribution (CSS 2.2 §17.5.2.1)
fn calculate_column_widths_fixed<T: ParsedFontTrait>(
    ctx: &mut LayoutContext<'_, T>,
    tree: &LayoutTree,
    table_ctx: &mut TableLayoutContext,
    available_width: f32,
) {
    debug_table_layout!(
        ctx,
        "calculate_column_widths_fixed: num_cols={}, available_width={:.2}",
        table_ctx.columns.len(),
        available_width
    );
    let num_cols = table_ctx.columns.len();
    if num_cols == 0 {
        return;
    }
    let num_visible_cols = num_cols - table_ctx.collapsed_columns.len();
    if num_visible_cols == 0 {
        for col in &mut table_ctx.columns {
            col.computed_width = Some(0.0);
        }
        return;
    }
    // Step 1 (column elements) is skipped because column elements don't store
    // explicit widths in the current table structure analysis.
    // Step 2: Check first-row cells for explicit width properties.
    let mut col_has_width = vec![false; num_cols];
    for cell_info in &table_ctx.cells {
        if cell_info.row != 0 {
            continue; // Only consider cells in the first row
        }
        if table_ctx.collapsed_columns.contains(&cell_info.column) {
            continue;
        }
        // Look up the cell's CSS width via its dom_node_id
        let dom_id = match tree.get(cell_info.node_index).and_then(|n| n.dom_node_id) {
            Some(id) => id,
            None => continue,
        };
        let node_state = &ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state;
        let css_width = get_css_width(ctx.styled_dom, dom_id, &node_state);
        let explicit_px = match css_width.unwrap_or_default() {
            LayoutWidth::Px(px) => {
                resolve_size_metric(
                    px.metric,
                    px.number.get(),
                    available_width,
                    ctx.viewport_size,
                )
            }
            LayoutWidth::Auto | LayoutWidth::MinContent | LayoutWidth::MaxContent
            | LayoutWidth::Calc(_) | LayoutWidth::FitContent(_) => continue,
        };
        if cell_info.colspan == 1 {
            table_ctx.columns[cell_info.column].computed_width = Some(explicit_px);
            col_has_width[cell_info.column] = true;
        } else {
            let mut visible_span_count = 0;
            for offset in 0..cell_info.colspan {
                let col_idx = cell_info.column + offset;
                if col_idx < num_cols && !table_ctx.collapsed_columns.contains(&col_idx) {
                    visible_span_count += 1;
                }
            }
            if visible_span_count > 0 {
                let per_col = explicit_px / visible_span_count as f32;
                for offset in 0..cell_info.colspan {
                    let col_idx = cell_info.column + offset;
                    if col_idx < num_cols
                        && !table_ctx.collapsed_columns.contains(&col_idx)
                        && !col_has_width[col_idx]
                    {
                        table_ctx.columns[col_idx].computed_width = Some(per_col);
                        col_has_width[col_idx] = true;
                    }
                }
            }
        }
    }
    let used_width: f32 = table_ctx.columns.iter().enumerate()
        .filter(|(idx, _)| col_has_width[*idx] && !table_ctx.collapsed_columns.contains(idx))
        .filter_map(|(_, c)| c.computed_width)
        .sum();
    let remaining_width = (available_width - used_width).max(0.0);
    let num_remaining = table_ctx.columns.iter().enumerate()
        .filter(|(idx, _)| !col_has_width[*idx] && !table_ctx.collapsed_columns.contains(idx))
        .count();
    if num_remaining > 0 {
        let width_per_remaining = remaining_width / num_remaining as f32;
        for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
            if table_ctx.collapsed_columns.contains(&col_idx) {
                col.computed_width = Some(0.0);
            } else if !col_has_width[col_idx] {
                col.computed_width = Some(width_per_remaining);
            }
        }
    }
    // Set collapsed columns to zero width
    for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
        if table_ctx.collapsed_columns.contains(&col_idx) {
            col.computed_width = Some(0.0);
        }
    }
    let total_col_width: f32 = table_ctx.columns.iter()
        .filter_map(|c| c.computed_width)
        .sum();
    if available_width > total_col_width && num_visible_cols > 0 {
        let extra = available_width - total_col_width;
        let extra_per_col = extra / num_visible_cols as f32;
        for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
            if !table_ctx.collapsed_columns.contains(&col_idx) {
                if let Some(ref mut w) = col.computed_width {
                    *w += extra_per_col;
                }
            }
        }
    }
}
/// Recursively clear the layout cache for every node in a subtree.
///
/// A fixed-depth walk is not enough: a table cell like
/// `<td><span><a>text</a></span></td>` has 4+ levels once the anonymous IFC
/// wrapper is inserted, and any stale cache below that level would feed a
/// narrow intrinsic width back into `measure_cell_content_width`.
23760
fn clear_subtree_cache(
23760
    tree: &LayoutTree,
23760
    cache_map: &mut crate::solver3::cache::LayoutCacheMap,
23760
    root: usize,
23760
) {
23760
    if root < cache_map.entries.len() {
23760
        cache_map.entries[root].clear();
23760
    }
23760
    let child_ids: Vec<usize> = tree.children(root).to_vec();
40040
    for child in child_ids {
16280
        clear_subtree_cache(tree, cache_map, child);
16280
    }
23760
}
/// Measure a cell's content width for a given intrinsic sizing mode.
///
/// CSS 2.2 Section 17.5.2.2: shared helper for min-content and max-content
/// width measurement. Lays out the cell subtree in ComputeSize mode and
/// returns the border-box width (content + padding + border).
7480
fn measure_cell_content_width<T: ParsedFontTrait>(
7480
    ctx: &mut LayoutContext<'_, T>,
7480
    tree: &mut LayoutTree,
7480
    text_cache: &mut crate::font_traits::TextLayoutCache,
7480
    cell_index: usize,
7480
    constraints: &LayoutConstraints,
7480
    sizing_mode: crate::text3::cache::AvailableSpace,
7480
) -> Result<f32> {
7480
    let width_type = match sizing_mode {
3740
        crate::text3::cache::AvailableSpace::MinContent => Text3AvailableSpace::MinContent,
3740
        crate::text3::cache::AvailableSpace::MaxContent => Text3AvailableSpace::MaxContent,
        crate::text3::cache::AvailableSpace::Definite(w) => Text3AvailableSpace::Definite(w),
    };
7480
    let cell_constraints = LayoutConstraints {
7480
        available_size: LogicalSize {
7480
            width: sizing_mode.to_f32_for_layout(),
7480
            height: f32::INFINITY,
7480
        },
7480
        writing_mode: constraints.writing_mode,
7480
        writing_mode_ctx: constraints.writing_mode_ctx,
7480
        bfc_state: None,
7480
        text_align: constraints.text_align,
7480
        containing_block_size: constraints.containing_block_size,
7480
        available_width_type: width_type,
7480
    };
7480
    let mut temp_positions: super::PositionVec = Vec::new();
7480
    let mut temp_scrollbar_reflow = false;
7480
    let mut temp_float_cache = HashMap::new();
    // Clear cached layout for this cell and ALL its descendants so that
    // min/max-content measurement uses unconstrained width, not a stale
    // result from a previous pass with narrower constraints. Deeply nested
    // inlines (`<td><span><a>text</a></span></td>`) need recursion; a fixed
    // 2-level walk left the `<a>` at level 3 with a stale cached 0-width.
7480
    clear_subtree_cache(tree, &mut ctx.cache_map, cell_index);
7480
    crate::solver3::cache::calculate_layout_for_subtree(
7480
        ctx,
7480
        tree,
7480
        text_cache,
7480
        cell_index,
7480
        LogicalPosition::zero(),
7480
        cell_constraints.available_size,
7480
        &mut temp_positions,
7480
        &mut temp_scrollbar_reflow,
7480
        &mut temp_float_cache,
7480
        crate::solver3::cache::ComputeMode::ComputeSize,
    )?;
7480
    let cell_bp = tree.get(cell_index)
7480
        .ok_or(LayoutError::InvalidTree)?
7480
        .box_props.unpack();
7480
    let padding = &cell_bp.padding;
7480
    let border = &cell_bp.border;
7480
    let wm = constraints.writing_mode;
    // For min/max-content measurement, use the overflow content size (actual
    // content width) rather than used_size. used_size for auto-width blocks
    // fills the containing block, which is huge (f32::MAX/2) during
    // intrinsic sizing — that would make every column appear infinitely wide.
7480
    let content_width = tree.warm(cell_index)
7480
        .and_then(|w| w.overflow_content_size)
7480
        .map(|s| s.width)
7480
        .unwrap_or_else(|| {
            tree.get(cell_index)
                .and_then(|n| n.used_size)
                .map(|s| s.width)
                .unwrap_or(0.0)
        });
7480
    Ok(content_width
7480
        + padding.cross_start(wm) + padding.cross_end(wm)
7480
        + border.cross_start(wm) + border.cross_end(wm))
7480
}
/// Measure a cell's minimum content width (with maximum wrapping)
3740
fn measure_cell_min_content_width<T: ParsedFontTrait>(
3740
    ctx: &mut LayoutContext<'_, T>,
3740
    tree: &mut LayoutTree,
3740
    text_cache: &mut crate::font_traits::TextLayoutCache,
3740
    cell_index: usize,
3740
    constraints: &LayoutConstraints,
3740
) -> Result<f32> {
3740
    measure_cell_content_width(
3740
        ctx, tree, text_cache, cell_index, constraints,
3740
        crate::text3::cache::AvailableSpace::MinContent,
    )
3740
}
/// Measure a cell's maximum content width (without wrapping)
3740
fn measure_cell_max_content_width<T: ParsedFontTrait>(
3740
    ctx: &mut LayoutContext<'_, T>,
3740
    tree: &mut LayoutTree,
3740
    text_cache: &mut crate::font_traits::TextLayoutCache,
3740
    cell_index: usize,
3740
    constraints: &LayoutConstraints,
3740
) -> Result<f32> {
3740
    measure_cell_content_width(
3740
        ctx, tree, text_cache, cell_index, constraints,
3740
        crate::text3::cache::AvailableSpace::MaxContent,
    )
3740
}
/// Calculate column widths using the auto table layout algorithm
fn calculate_column_widths_auto<T: ParsedFontTrait>(
    table_ctx: &mut TableLayoutContext,
    tree: &mut LayoutTree,
    text_cache: &mut crate::font_traits::TextLayoutCache,
    ctx: &mut LayoutContext<'_, T>,
    constraints: &LayoutConstraints,
) -> Result<()> {
    calculate_column_widths_auto_with_width(
        table_ctx,
        tree,
        text_cache,
        ctx,
        constraints,
        constraints.available_size.width,
    )
}
/// Calculate column widths using the auto table layout algorithm with explicit table width
// +spec:display-property:05c8e8 - CSS 2.2 §17.5.2.2 automatic table layout: column min/max widths, table width = max(W or CB, CAPMIN, MIN), extra width distributed over columns
/// +spec:overflow:29edde - CSS 2.2 §17.5.2.2 automatic table layout: MCW/max-content per cell, column min/max, colspan distribution, final width determination
// +spec:table-layout:23a215 - automatic table layout: MCW/max cell widths, column min/max, colspan distribution, table width from MAX/MIN/CAPMIN
// +spec:table-layout:5e1145 - Automatic table layout: MCW/max-content per cell, column min/max, colspan distribution, final width from MIN/MAX
// +spec:width-calculation:42dfca - CSS 2.2 §17.5.2.2 automatic table layout: MCW/max-content per cell, column min/max, multi-span distribution, final table width
/// +spec:width-calculation:335ef1 - Automatic table layout: width given by column widths and borders (CSS 2.2 §17.5.2.2)
1452
fn calculate_column_widths_auto_with_width<T: ParsedFontTrait>(
1452
    table_ctx: &mut TableLayoutContext,
1452
    tree: &mut LayoutTree,
1452
    text_cache: &mut crate::font_traits::TextLayoutCache,
1452
    ctx: &mut LayoutContext<'_, T>,
1452
    constraints: &LayoutConstraints,
1452
    table_width: f32,
1452
) -> Result<()> {
    // Auto layout: calculate min/max content width for each cell
1452
    let num_cols = table_ctx.columns.len();
1452
    if num_cols == 0 {
        return Ok(());
1452
    }
    // Step 1: Measure all cells to determine column min/max widths
    // CSS 2.2 Section 17.6: Skip cells in collapsed columns
5192
    for cell_info in &table_ctx.cells {
        // Skip cells in collapsed columns
3740
        if table_ctx.collapsed_columns.contains(&cell_info.column) {
            continue;
3740
        }
        // Skip cells that span into collapsed columns
3740
        let mut spans_collapsed = false;
3740
        for col_offset in 0..cell_info.colspan {
3740
            if table_ctx
3740
                .collapsed_columns
3740
                .contains(&(cell_info.column + col_offset))
            {
                spans_collapsed = true;
                break;
3740
            }
        }
3740
        if spans_collapsed {
            continue;
3740
        }
3740
        let min_width = measure_cell_min_content_width(
3740
            ctx,
3740
            tree,
3740
            text_cache,
3740
            cell_info.node_index,
3740
            constraints,
        )?;
3740
        let max_width = measure_cell_max_content_width(
3740
            ctx,
3740
            tree,
3740
            text_cache,
3740
            cell_info.node_index,
3740
            constraints,
        )?;
        // Handle single-column cells
3740
        if cell_info.colspan == 1 {
3740
            let col = &mut table_ctx.columns[cell_info.column];
3740
            col.min_width = col.min_width.max(min_width);
3740
            col.max_width = col.max_width.max(max_width);
3740
        } else {
            // Handle multi-column cells (colspan > 1)
            // Distribute the cell's min/max width across the spanned columns
            distribute_cell_width_across_columns(
                &mut table_ctx.columns,
                cell_info.column,
                cell_info.colspan,
                min_width,
                max_width,
                &table_ctx.collapsed_columns,
            );
        }
    }
    // Step 2: Calculate final column widths based on available space
    // Exclude collapsed columns from total width calculations
1452
    let total_min_width: f32 = table_ctx
1452
        .columns
1452
        .iter()
1452
        .enumerate()
3696
        .filter(|(idx, _)| !table_ctx.collapsed_columns.contains(idx))
1452
        .map(|(_, c)| c.min_width)
1452
        .sum();
1452
    let total_max_width: f32 = table_ctx
1452
        .columns
1452
        .iter()
1452
        .enumerate()
3696
        .filter(|(idx, _)| !table_ctx.collapsed_columns.contains(idx))
1452
        .map(|(_, c)| c.max_width)
1452
        .sum();
1452
    let available_width = table_width; // Use table's content-box width, not constraints
1452
    debug_table_layout!(
1452
        ctx,
1452
        "calculate_column_widths_auto: min={:.2}, max={:.2}, table_width={:.2}",
        total_min_width,
        total_max_width,
        table_width
    );
    // Handle infinity and NaN cases
1452
    if !total_max_width.is_finite() || !available_width.is_finite() {
        // If max_width is infinite or unavailable, distribute available width equally
        let num_non_collapsed = table_ctx.columns.len() - table_ctx.collapsed_columns.len();
        let width_per_column = if num_non_collapsed > 0 {
            available_width / num_non_collapsed as f32
        } else {
            0.0
        };
        for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
            if table_ctx.collapsed_columns.contains(&col_idx) {
                col.computed_width = Some(0.0);
            } else {
                // Use the larger of min_width and equal distribution
                col.computed_width = Some(col.min_width.max(width_per_column));
            }
        }
1452
    } else if available_width >= total_max_width {
        // Case 1: More space than max-content - distribute excess proportionally
        //
        // CSS 2.1 Section 17.5.2.2: Distribute extra space proportionally to
        // max-content widths
1452
        let excess_width = available_width - total_max_width;
        // First pass: collect column info (max_width) to avoid borrowing issues
1452
        let column_info: Vec<(usize, f32, bool)> = table_ctx
1452
            .columns
1452
            .iter()
1452
            .enumerate()
3696
            .map(|(idx, c)| (idx, c.max_width, table_ctx.collapsed_columns.contains(&idx)))
1452
            .collect();
        // Calculate total weight for proportional distribution (use max_width as weight)
1452
        let total_weight: f32 = column_info.iter()
3696
            .filter(|(_, _, is_collapsed)| !is_collapsed)
3696
            .map(|(_, max_w, _)| max_w.max(1.0)) // Avoid division by zero
1452
            .sum();
1452
        let num_non_collapsed = column_info
1452
            .iter()
3696
            .filter(|(_, _, is_collapsed)| !is_collapsed)
1452
            .count();
        // Second pass: set computed widths
5148
        for (col_idx, max_width, is_collapsed) in column_info {
3696
            let col = &mut table_ctx.columns[col_idx];
3696
            if is_collapsed {
                col.computed_width = Some(0.0);
            } else {
                // Start with max-content width, then add proportional share of excess
3696
                let weight_factor = if total_weight > 0.0 {
3696
                    max_width.max(1.0) / total_weight
                } else {
                    // If all columns have 0 max_width, distribute equally
                    1.0 / num_non_collapsed.max(1) as f32
                };
3696
                let final_width = max_width + (excess_width * weight_factor);
3696
                col.computed_width = Some(final_width);
            }
        }
    } else if available_width >= total_min_width {
        // Case 2: Between min and max - interpolate proportionally
        // Avoid division by zero if min == max
        let scale = if total_max_width > total_min_width {
            (available_width - total_min_width) / (total_max_width - total_min_width)
        } else {
            0.0 // If min == max, just use min width
        };
        for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
            if table_ctx.collapsed_columns.contains(&col_idx) {
                col.computed_width = Some(0.0);
            } else {
                let interpolated = col.min_width + (col.max_width - col.min_width) * scale;
                col.computed_width = Some(interpolated);
            }
        }
    } else {
        // Case 3: Not enough space - scale down from min widths
        let scale = if total_min_width > 0.0 { available_width / total_min_width } else { 1.0 };
        for (col_idx, col) in table_ctx.columns.iter_mut().enumerate() {
            if table_ctx.collapsed_columns.contains(&col_idx) {
                col.computed_width = Some(0.0);
            } else {
                col.computed_width = Some(col.min_width * scale);
            }
        }
    }
1452
    Ok(())
1452
}
/// Distribute a multi-column cell's width across the columns it spans
fn distribute_cell_width_across_columns(
    columns: &mut [TableColumnInfo],
    start_col: usize,
    colspan: usize,
    cell_min_width: f32,
    cell_max_width: f32,
    collapsed_columns: &std::collections::HashSet<usize>,
) {
    let end_col = start_col + colspan;
    if end_col > columns.len() {
        return;
    }
    // Calculate current total of spanned non-collapsed columns
    let current_min_total: f32 = columns[start_col..end_col]
        .iter()
        .enumerate()
        .filter(|(idx, _)| !collapsed_columns.contains(&(start_col + idx)))
        .map(|(_, c)| c.min_width)
        .sum();
    let current_max_total: f32 = columns[start_col..end_col]
        .iter()
        .enumerate()
        .filter(|(idx, _)| !collapsed_columns.contains(&(start_col + idx)))
        .map(|(_, c)| c.max_width)
        .sum();
    // Count non-collapsed columns in the span
    let num_visible_cols = (start_col..end_col)
        .filter(|idx| !collapsed_columns.contains(idx))
        .count();
    if num_visible_cols == 0 {
        return; // All spanned columns are collapsed
    }
    // Only distribute if the cell needs more space than currently available
    if cell_min_width > current_min_total {
        let extra_min = cell_min_width - current_min_total;
        let per_col = extra_min / num_visible_cols as f32;
        for (idx, col) in columns[start_col..end_col].iter_mut().enumerate() {
            if !collapsed_columns.contains(&(start_col + idx)) {
                col.min_width += per_col;
            }
        }
    }
    if cell_max_width > current_max_total {
        let extra_max = cell_max_width - current_max_total;
        let per_col = extra_max / num_visible_cols as f32;
        for (idx, col) in columns[start_col..end_col].iter_mut().enumerate() {
            if !collapsed_columns.contains(&(start_col + idx)) {
                col.max_width += per_col;
            }
        }
    }
}
/// Layout a cell with its computed column width to determine its content height
3740
fn layout_cell_for_height<T: ParsedFontTrait>(
3740
    ctx: &mut LayoutContext<'_, T>,
3740
    tree: &mut LayoutTree,
3740
    text_cache: &mut crate::font_traits::TextLayoutCache,
3740
    cell_index: usize,
3740
    cell_width: f32,
3740
    constraints: &LayoutConstraints,
3740
) -> Result<f32> {
3740
    let cell_node = tree.get(cell_index).ok_or(LayoutError::InvalidTree)?;
3740
    let cell_dom_id = cell_node.dom_node_id.ok_or(LayoutError::InvalidTree)?;
    // Check if cell has text content directly in DOM (not in LayoutTree)
    // Text nodes are intentionally not included in LayoutTree per CSS spec,
    // but we need to measure them for table cell height calculation.
3740
    let has_text_children = cell_dom_id
3740
        .az_children(&ctx.styled_dom.node_hierarchy.as_container())
3740
        .any(|child_id| {
3740
            let node_data = &ctx.styled_dom.node_data.as_container()[child_id];
3740
            matches!(node_data.get_node_type(), NodeType::Text(_))
3740
        });
3740
    debug_table_layout!(
3740
        ctx,
3740
        "layout_cell_for_height: cell_index={}, has_text_children={}",
        cell_index,
        has_text_children
    );
    // Get padding and border to calculate content width
3740
    let cell_node = tree.get(cell_index).ok_or(LayoutError::InvalidTree)?;
3740
    let cell_bp = cell_node.box_props.unpack();
3740
    let padding = &cell_bp.padding;
3740
    let border = &cell_bp.border;
3740
    let writing_mode = constraints.writing_mode;
    // cell_width is the border-box width (includes padding/border from column
    // width calculation) but layout functions need content-box width
3740
    let content_width = cell_width
3740
        - padding.cross_start(writing_mode)
3740
        - padding.cross_end(writing_mode)
3740
        - border.cross_start(writing_mode)
3740
        - border.cross_end(writing_mode);
3740
    debug_table_layout!(
3740
        ctx,
3740
        "Cell width: border_box={:.2}, content_box={:.2}",
        cell_width,
        content_width
    );
3740
    let content_height = if has_text_children {
        // Cell contains text - use IFC to measure it
1188
        debug_table_layout!(ctx, "Using IFC to measure text content");
1188
        let cell_constraints = LayoutConstraints {
1188
            available_size: LogicalSize {
1188
                width: content_width, // Use content width, not border-box width
1188
                height: f32::INFINITY,
1188
            },
1188
            writing_mode: constraints.writing_mode,
1188
            writing_mode_ctx: constraints.writing_mode_ctx,
1188
            bfc_state: None,
1188
            text_align: constraints.text_align,
1188
            containing_block_size: constraints.containing_block_size,
1188
            // Use definite width for final cell layout!
1188
            // This replaces any previous MinContent/MaxContent measurement.
1188
            available_width_type: Text3AvailableSpace::Definite(content_width),
1188
        };
1188
        let output = layout_ifc(ctx, text_cache, tree, cell_index, &cell_constraints)?;
        // The cell now owns the authoritative IFC result. Clear any duplicate
        // inline_layout_result from text children that was set during the cell's
        // prior BFC Pass 1 (which ran before layout_cell_for_height).
1188
        let cell_children: Vec<usize> = tree.children(cell_index).to_vec();
2376
        for child_idx in cell_children {
1188
            if let Some(warm) = tree.warm_mut(child_idx) {
1188
                warm.inline_layout_result = None;
1188
            }
        }
1188
        debug_table_layout!(
1188
            ctx,
1188
            "IFC returned height={:.2}",
            output.overflow_size.height
        );
1188
        output.overflow_size.height
    } else {
        // Cell contains block-level children or is empty - use regular layout
2552
        debug_table_layout!(ctx, "Using regular layout for block children");
2552
        let cell_constraints = LayoutConstraints {
2552
            available_size: LogicalSize {
2552
                width: content_width, // Use content width, not border-box width
2552
                height: f32::INFINITY,
2552
            },
2552
            writing_mode: constraints.writing_mode,
2552
            writing_mode_ctx: constraints.writing_mode_ctx,
2552
            bfc_state: None,
2552
            text_align: constraints.text_align,
2552
            containing_block_size: constraints.containing_block_size,
2552
            // Use Definite width for final cell layout!
2552
            available_width_type: Text3AvailableSpace::Definite(content_width),
2552
        };
2552
        let mut temp_positions: super::PositionVec = Vec::new();
2552
        let mut temp_scrollbar_reflow = false;
2552
        let mut temp_float_cache = HashMap::new();
2552
        crate::solver3::cache::calculate_layout_for_subtree(
2552
            ctx,
2552
            tree,
2552
            text_cache,
2552
            cell_index,
2552
            LogicalPosition::zero(),
2552
            cell_constraints.available_size,
2552
            &mut temp_positions,
2552
            &mut temp_scrollbar_reflow,
2552
            &mut temp_float_cache,
            // PerformLayout: final table cell layout with definite width
2552
            crate::solver3::cache::ComputeMode::PerformLayout,
        )?;
2552
        let cell_node = tree.get(cell_index).ok_or(LayoutError::InvalidTree)?;
2552
        cell_node.used_size.unwrap_or_default().height
    };
    // Add padding and border to get the total height
3740
    let cell_node = tree.get(cell_index).ok_or(LayoutError::InvalidTree)?;
3740
    let cell_bp = cell_node.box_props.unpack();
3740
    let padding = &cell_bp.padding;
3740
    let border = &cell_bp.border;
3740
    let writing_mode = constraints.writing_mode;
3740
    let total_height = content_height
3740
        + padding.main_start(writing_mode)
3740
        + padding.main_end(writing_mode)
3740
        + border.main_start(writing_mode)
3740
        + border.main_end(writing_mode);
3740
    debug_table_layout!(
3740
        ctx,
3740
        "Cell total height: cell_index={}, content={:.2}, padding/border={:.2}, total={:.2}",
        cell_index,
        content_height,
3740
        padding.main_start(writing_mode)
3740
            + padding.main_end(writing_mode)
3740
            + border.main_start(writing_mode)
3740
            + border.main_end(writing_mode),
        total_height
    );
3740
    Ok(total_height)
3740
}
// or bottom of content edge if no such line box exists
// +spec:box-model:b64fa0 - Cell baseline is first in-flow line box or bottom of content edge
// +spec:overflow:3fa86f - Table cell baseline: first in-flow line box or bottom of content edge; scrolling boxes treated as at origin
// +spec:inline-formatting-context:c4a20d - cell baseline: first in-flow line box or bottom of content edge
// +spec:inline-formatting-context:17a9c1 - vertical-align baseline/top/bottom/middle for table cells
12408
fn compute_cell_baseline(cell_index: usize, tree: &LayoutTree) -> f32 {
12408
    let Some(cell_node) = tree.get(cell_index) else {
        return 0.0;
    };
12408
    let cell_bp = cell_node.box_props.unpack();
    // +spec:inline-formatting-context:27be38 - cell baseline is first in-flow line box or bottom of content edge
    // Check if the cell has inline layout (first in-flow line box)
12408
    if let Some(warm_node) = tree.warm(cell_index) {
12408
        if let Some(ref cached_layout) = warm_node.inline_layout_result {
7304
            let inline_result = &cached_layout.layout;
            // The baseline is the ascent of the first item from the top of the cell
7304
            if let Some(first_item) = inline_result.items.first() {
7304
                let (item_ascent, _) = crate::text3::cache::get_item_vertical_metrics_approx(&first_item.item);
7304
                let padding_top = cell_bp.padding.top;
7304
                let border_top = cell_bp.border.top;
7304
                return padding_top + border_top + first_item.position.y + item_ascent;
            }
5104
        }
    }
    // Check children for first in-flow line box
5104
    let children = tree.children(cell_index);
5280
    for &child_idx in children {
5104
        if child_idx < tree.nodes.len() {
5104
            if let Some(child_warm) = tree.warm(child_idx) {
5104
                if child_warm.inline_layout_result.is_some() {
4928
                    let child_baseline = compute_cell_baseline(child_idx, tree);
4928
                    let padding_top = cell_bp.padding.top;
4928
                    let border_top = cell_bp.border.top;
4928
                    return padding_top + border_top + child_baseline;
176
                }
            }
        }
    }
    // No line box found: baseline is the bottom of the content edge
176
    let used_size = cell_node.used_size.unwrap_or_default();
176
    let padding_bottom = cell_bp.padding.bottom;
176
    let border_bottom = cell_bp.border.bottom;
176
    used_size.height - padding_bottom - border_bottom
12408
}
/// +spec:box-model:72b495 - Table row height = max of computed height and MIN required by cells; baseline alignment
// +spec:display-property:728144 - Table height algorithm: row heights from cell content, rowspan distribution, vertical-align in cells (top/middle/bottom/baseline, sub/super/text-top/text-bottom/length/percentage fall back to baseline), cell baseline computation, and horizontal alignment via text-align
// +spec:positioning:3eaadd - Table height algorithms (§17.5.3): row height = max of cell heights/MIN,
//   rowspan distribution, vertical-align in table cells, cell baseline definition
/// Calculate row heights based on cell content after column widths are determined
// +spec:inline-formatting-context:87b90d - Table height algorithms: row height = max(computed height, cell heights, MIN); vertical-align in cells (baseline/top/middle/bottom, sub/super/etc. fall back to baseline)
1452
fn calculate_row_heights<T: ParsedFontTrait>(
1452
    table_ctx: &mut TableLayoutContext,
1452
    tree: &mut LayoutTree,
1452
    text_cache: &mut crate::font_traits::TextLayoutCache,
1452
    ctx: &mut LayoutContext<'_, T>,
1452
    constraints: &LayoutConstraints,
1452
) -> Result<()> {
1452
    debug_table_layout!(
1452
        ctx,
1452
        "calculate_row_heights: num_rows={}, available_size={:?}",
        table_ctx.num_rows,
        constraints.available_size
    );
    // +spec:inline-formatting-context:a7c7a0 - row height = max of computed height, cell heights, and MIN; vertical-align per cell
    // Initialize row heights and baselines
1452
    table_ctx.row_heights = vec![0.0; table_ctx.num_rows];
1452
    table_ctx.row_baselines = vec![0.0; table_ctx.num_rows];
    // CSS 2.2 Section 17.6: Set collapsed rows to height 0
1452
    for &row_idx in &table_ctx.collapsed_rows {
        if row_idx < table_ctx.row_heights.len() {
            table_ctx.row_heights[row_idx] = 0.0;
        }
    }
    // required by content; 'height' property can influence row height but does not
    // increase cell box height
    // First pass: Calculate heights for cells that don't span multiple rows
5192
    for cell_info in &table_ctx.cells {
        // Skip cells in collapsed rows
3740
        if table_ctx.collapsed_rows.contains(&cell_info.row) {
            continue;
3740
        }
        // Get the cell's width (sum of column widths if colspan > 1)
3740
        let mut cell_width = 0.0;
3740
        for col_idx in cell_info.column..(cell_info.column + cell_info.colspan) {
3740
            if let Some(col) = table_ctx.columns.get(col_idx) {
3740
                if let Some(width) = col.computed_width {
3740
                    cell_width += width;
3740
                }
            }
        }
3740
        debug_table_layout!(
3740
            ctx,
3740
            "Cell layout: node_index={}, row={}, col={}, width={:.2}",
            cell_info.node_index,
            cell_info.row,
            cell_info.column,
            cell_width
        );
        // Layout the cell to get its height
3740
        let cell_height = layout_cell_for_height(
3740
            ctx,
3740
            tree,
3740
            text_cache,
3740
            cell_info.node_index,
3740
            cell_width,
3740
            constraints,
        )?;
3740
        debug_table_layout!(
3740
            ctx,
3740
            "Cell height calculated: node_index={}, height={:.2}",
            cell_info.node_index,
            cell_height
        );
        //   row height = max of all single-span cell heights in the row
3740
        if cell_info.rowspan == 1 {
3740
            let current_height = table_ctx.row_heights[cell_info.row];
3740
            table_ctx.row_heights[cell_info.row] = current_height.max(cell_height);
3740
        }
        // +spec:box-model:073652 - Table height: baseline-aligned cells establish row baseline, then top/bottom/middle cells positioned
        // The baseline of a cell is the baseline of its first line box (from inline layout)
        // or the bottom of the content box if no inline content.
3740
        if cell_info.rowspan == 1 {
3740
            let cell_baseline = compute_cell_baseline(cell_info.node_index, tree);
3740
            let current_baseline = table_ctx.row_baselines[cell_info.row];
3740
            table_ctx.row_baselines[cell_info.row] = current_baseline.max(cell_baseline);
3740
        }
    }
    // involved must be great enough to encompass the cell spanning the rows
    // Second pass: Handle cells that span multiple rows (rowspan > 1)
5192
    for cell_info in &table_ctx.cells {
        // Skip cells that start in collapsed rows
3740
        if table_ctx.collapsed_rows.contains(&cell_info.row) {
            continue;
3740
        }
3740
        if cell_info.rowspan > 1 {
            // Get the cell's width
            let mut cell_width = 0.0;
            for col_idx in cell_info.column..(cell_info.column + cell_info.colspan) {
                if let Some(col) = table_ctx.columns.get(col_idx) {
                    if let Some(width) = col.computed_width {
                        cell_width += width;
                    }
                }
            }
            // Layout the cell to get its height
            let cell_height = layout_cell_for_height(
                ctx,
                tree,
                text_cache,
                cell_info.node_index,
                cell_width,
                constraints,
            )?;
            // Calculate the current total height of spanned rows (excluding collapsed rows)
            let end_row = cell_info.row + cell_info.rowspan;
            let current_total: f32 = table_ctx.row_heights[cell_info.row..end_row]
                .iter()
                .enumerate()
                .filter(|(idx, _)| !table_ctx.collapsed_rows.contains(&(cell_info.row + idx)))
                .map(|(_, height)| height)
                .sum();
            // If the cell needs more height, distribute extra height across
            // non-collapsed spanned rows
            if cell_height > current_total {
                let extra_height = cell_height - current_total;
                // Count non-collapsed rows in span
                let non_collapsed_rows = (cell_info.row..end_row)
                    .filter(|row_idx| !table_ctx.collapsed_rows.contains(row_idx))
                    .count();
                if non_collapsed_rows > 0 {
                    let per_row = extra_height / non_collapsed_rows as f32;
                    for row_idx in cell_info.row..end_row {
                        if !table_ctx.collapsed_rows.contains(&row_idx) {
                            table_ctx.row_heights[row_idx] += per_row;
                        }
                    }
                }
            }
3740
        }
    }
    // CSS 2.2 Section 17.6: Final pass - ensure collapsed rows have height 0
1452
    for &row_idx in &table_ctx.collapsed_rows {
        if row_idx < table_ctx.row_heights.len() {
            table_ctx.row_heights[row_idx] = 0.0;
        }
    }
    //   visible content, the row has zero height and v-spacing on only one side
    // +spec:table-layout:7370dc - empty-cells:hide in separated borders model
    // +spec:box-model:1e9cf1 - empty-cells:hide rows get zero height with v-spacing on only one side
    // +spec:overflow:a44925 - CSS 2.2 §17.6.1.1: empty-cells:hide suppresses borders/backgrounds; all-hidden rows get zero height
    // +spec:table-layout:dc8bc3 - separated borders model: border-spacing, empty-cells, row zero-height
1452
    if table_ctx.border_collapse == StyleBorderCollapse::Separate {
1496
        for row_idx in 0..table_ctx.num_rows {
1496
            if table_ctx.collapsed_rows.contains(&row_idx) {
                continue;
1496
            }
            // Collect cells in this row
1496
            let row_cells: Vec<usize> = table_ctx
1496
                .cells
1496
                .iter()
3828
                .filter(|c| c.row == row_idx && c.rowspan == 1)
1496
                .map(|c| c.node_index)
1496
                .collect();
1496
            if row_cells.is_empty() {
                continue;
1496
            }
            // +spec:box-model:0ab9b0 - empty-cells:hide suppresses borders/backgrounds, row gets zero height if all cells hidden+empty
            // Check if ALL cells in this row have empty-cells:hide and are empty
1496
            let all_hidden_empty = row_cells.iter().all(|&cell_idx| {
1496
                if let Some(cell_node) = tree.get(cell_idx) {
1496
                    let ec = get_empty_cells_property(ctx, cell_node);
1496
                    ec == StyleEmptyCells::Hide && is_cell_empty(tree, cell_idx)
                } else {
                    true
                }
1496
            });
1496
            if all_hidden_empty {
                table_ctx.row_heights[row_idx] = 0.0;
                table_ctx.hidden_empty_rows.insert(row_idx);
1496
            }
        }
    }
1452
    Ok(())
1452
}
/// Position all cells in the table grid with calculated widths and heights
1452
fn position_table_cells<T: ParsedFontTrait>(
1452
    table_ctx: &mut TableLayoutContext,
1452
    tree: &mut LayoutTree,
1452
    ctx: &mut LayoutContext<'_, T>,
1452
    table_index: usize,
1452
    constraints: &LayoutConstraints,
1452
) -> Result<BTreeMap<usize, LogicalPosition>> {
1452
    debug_log!(ctx, "Positioning table cells in grid");
1452
    let mut positions = BTreeMap::new();
    // +spec:box-model:54e86a - Separated borders model: individual cell borders, border-spacing between cells, empty-cells handling
    //   rows, columns, row groups, column groups cannot have borders (UA must ignore border props);
    //   row/column/rowgroup/colgroup backgrounds are invisible in border-spacing area (table bg shows through);
    //   distance from table edge to edge-cell border = table padding + border-spacing
    //   (table padding is already accounted for by the containing block; h_spacing is the border-spacing)
    // Get border spacing values if border-collapse is separate
1452
    let (h_spacing, v_spacing) = if table_ctx.border_collapse == StyleBorderCollapse::Separate {
1452
        let styled_dom = ctx.styled_dom;
1452
        let table_id = tree.nodes[table_index].dom_node_id.unwrap();
1452
        let table_state = &styled_dom.styled_nodes.as_container()[table_id].styled_node_state;
1452
        let spacing_context = ResolutionContext {
1452
            element_font_size: get_element_font_size(styled_dom, table_id, table_state),
1452
            parent_font_size: get_parent_font_size(styled_dom, table_id, table_state),
1452
            root_font_size: get_root_font_size(styled_dom, table_state),
1452
            containing_block_size: PhysicalSize::new(0.0, 0.0),
1452
            element_size: None,
1452
            viewport_size: PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height),
1452
        };
1452
        let h = table_ctx
1452
            .border_spacing
1452
            .horizontal
1452
            .resolve_with_context(&spacing_context, PropertyContext::Other)
1452
            .max(0.0);
1452
        let v = table_ctx
1452
            .border_spacing
1452
            .vertical
1452
            .resolve_with_context(&spacing_context, PropertyContext::Other)
1452
            .max(0.0);
1452
        (h, v)
    } else {
        (0.0, 0.0)
    };
1452
    debug_log!(
1452
        ctx,
1452
        "Border spacing: h={:.2}, v={:.2}",
        h_spacing,
        v_spacing
    );
    // Calculate cumulative column positions (x-offsets) with spacing
1452
    let mut col_positions = vec![0.0; table_ctx.columns.len()];
1452
    let mut x_offset = h_spacing; // Start with spacing on the left
3696
    for (i, col) in table_ctx.columns.iter().enumerate() {
3696
        col_positions[i] = x_offset;
3696
        if let Some(width) = col.computed_width {
            // Collapsed columns: gutters on either side collapse (width is 0, skip spacing)
3696
            if table_ctx.collapsed_columns.contains(&i) {
                // No width, no gutter added
3696
            } else {
3696
                x_offset += width + h_spacing; // Add spacing between columns
3696
            }
        }
    }
    // Calculate cumulative row positions (y-offsets) with spacing
1452
    let mut row_positions = vec![0.0; table_ctx.num_rows];
1452
    let mut y_offset = v_spacing; // Start with spacing on the top
1496
    for (i, &height) in table_ctx.row_heights.iter().enumerate() {
1496
        row_positions[i] = y_offset;
        // Collapsed rows: gutters on either side collapse (height is 0, skip spacing)
1496
        if table_ctx.collapsed_rows.contains(&i) {
            // No height, no gutter added
1496
        } else if table_ctx.hidden_empty_rows.contains(&i) {
            // Hidden-empty row: zero height, only one side of spacing
            // (we already added spacing before this row, so skip the spacing after)
            y_offset += height; // height is 0.0
1496
        } else {
1496
            y_offset += height + v_spacing; // Add spacing between rows
1496
        }
    }
    // Store row positions and sizes so paint_element_background can paint row backgrounds.
    // Row width = sum of column widths + spacing. Row height from row_heights.
    {
3696
        let total_col_width: f32 = table_ctx.columns.iter().map(|c| c.computed_width.unwrap_or(0.0)).sum::<f32>()
1452
            + h_spacing * (table_ctx.columns.len().max(1) - 1) as f32
1452
            + h_spacing * 2.0; // border-spacing on left+right edges
1496
        for (i, &row_y) in row_positions.iter().enumerate() {
1496
            if let Some(&row_node_idx) = table_ctx.row_node_indices.get(i) {
1496
                let row_height = table_ctx.row_heights.get(i).copied().unwrap_or(0.0);
1496
                if let Some(row_node) = tree.get_mut(row_node_idx) {
1496
                    row_node.used_size = Some(LogicalSize {
1496
                        width: total_col_width,
1496
                        height: row_height,
1496
                    });
1496
                }
                // Don't add to `positions` map (feeds position_bfc_child_descendants,
                // would double-offset cells). The display list computes row paint
                // rects from the row's cell children.
            }
        }
    }
    // Position each cell
5192
    for cell_info in &table_ctx.cells {
3740
        let precomputed_cell_baseline = compute_cell_baseline(cell_info.node_index, tree);
3740
        let cell_node = tree
3740
            .get_mut(cell_info.node_index)
3740
            .ok_or(LayoutError::InvalidTree)?;
        // Calculate cell position
3740
        let x = col_positions.get(cell_info.column).copied().unwrap_or(0.0);
3740
        let y = row_positions.get(cell_info.row).copied().unwrap_or(0.0);
        // Calculate cell size (sum of spanned columns/rows)
3740
        let mut width = 0.0;
3740
        debug_info!(
3740
            ctx,
3740
            "[position_table_cells] Cell {}: calculating width from cols {}..{}",
            cell_info.node_index,
            cell_info.column,
3740
            cell_info.column + cell_info.colspan
        );
3740
        for col_idx in cell_info.column..(cell_info.column + cell_info.colspan) {
3740
            if let Some(col) = table_ctx.columns.get(col_idx) {
3740
                debug_info!(
3740
                    ctx,
3740
                    "[position_table_cells]   Col {}: computed_width={:?}",
                    col_idx,
                    col.computed_width
                );
3740
                if let Some(col_width) = col.computed_width {
3740
                    width += col_width;
                    // Add spacing between spanned columns (but not after the last one)
3740
                    if col_idx < cell_info.column + cell_info.colspan - 1 {
                        width += h_spacing;
3740
                    }
                } else {
                    debug_info!(
                        ctx,
                        "[position_table_cells]   WARN:  Col {} has NO computed_width!",
                        col_idx
                    );
                }
            } else {
                debug_info!(
                    ctx,
                    "[position_table_cells]   WARN:  Col {} not found in table_ctx.columns!",
                    col_idx
                );
            }
        }
3740
        let mut height = 0.0;
3740
        let end_row = cell_info.row + cell_info.rowspan;
3740
        for row_idx in cell_info.row..end_row {
3740
            if let Some(&row_height) = table_ctx.row_heights.get(row_idx) {
3740
                height += row_height;
                // Add spacing between spanned rows (but not after the last one)
3740
                if row_idx < end_row - 1 {
                    height += v_spacing;
3740
                }
            }
        }
        // Update cell's used size and position
3740
        let writing_mode = constraints.writing_mode;
        // Table layout works in main/cross axes, must convert back to logical width/height
3740
        debug_info!(
3740
            ctx,
3740
            "[position_table_cells] Cell {}: BEFORE from_main_cross: width={}, height={}, \
3740
             writing_mode={:?}",
            cell_info.node_index,
            width,
            height,
            writing_mode
        );
3740
        cell_node.used_size = Some(LogicalSize::from_main_cross(height, width, writing_mode));
3740
        debug_info!(
3740
            ctx,
3740
            "[position_table_cells] Cell {}: AFTER from_main_cross: used_size={:?}",
            cell_info.node_index,
            cell_node.used_size
        );
3740
        debug_info!(
3740
            ctx,
3740
            "[position_table_cells] Cell {}: setting used_size to {}x{} (row_heights={:?})",
            cell_info.node_index,
            width,
            height,
            table_ctx.row_heights
        );
        // Save hot fields needed for vertical alignment before dropping the mutable borrow
3740
        let cell_dom_node_id = cell_node.dom_node_id;
3740
        let cell_box_props = cell_node.box_props.unpack();
3740
        drop(cell_node);
        // +spec:inline-formatting-context:20e8e8 - table cell vertical-align alignment order (baseline first, then top, then bottom/middle)
        // receive extra top or bottom padding; vertical-align determines alignment
        // +spec:inline-formatting-context:4545e8 - vertical-align on table cells maps to align-content: top→start, bottom→end, middle→center
        // +spec:inline-formatting-context:e216be - vertical-align on table cells (baseline, middle, top, bottom)
        // +spec:positioning:156e49 - table cell vertical-align ordering and extra padding per CSS 2.2 §17.5.3
        // Apply vertical-align to cell content if it has inline layout
        // We need to compute the y_offset using immutable borrows first, then apply it mutably.
3740
        let vertical_align_adjustment = if let Some(warm_node) = tree.warm(cell_info.node_index) {
3740
            if let Some(ref cached_layout) = warm_node.inline_layout_result {
1188
                let inline_result = &cached_layout.layout;
                use StyleVerticalAlign;
                // Get vertical-align property from styled_dom
1188
                let vertical_align = if let Some(dom_id) = cell_dom_node_id {
1188
                    let node_state = ctx.styled_dom.styled_nodes.as_container()[dom_id].styled_node_state.clone();
1188
                    match get_vertical_align_property(ctx.styled_dom, dom_id, &node_state) {
1188
                        MultiValue::Exact(v) => v,
                        _ => StyleVerticalAlign::Baseline,
                    }
                } else {
                    StyleVerticalAlign::Baseline
                };
                // Calculate content height from inline layout bounds
1188
                let content_bounds = inline_result.bounds();
1188
                let content_height = content_bounds.height;
                // Get padding and border to calculate content-box height
                // height is border-box, but vertical alignment should be within content-box
1188
                let padding = &cell_box_props.padding;
1188
                let border = &cell_box_props.border;
1188
                let content_box_height = height
1188
                    - padding.main_start(writing_mode)
1188
                    - padding.main_end(writing_mode)
1188
                    - border.main_start(writing_mode)
1188
                    - border.main_end(writing_mode);
                // top: top of cell box aligned with top of first row it spans
                // bottom: bottom of cell box aligned with bottom of last row it spans
                // middle: center of cell aligned with center of rows it spans
                //   the cell is aligned at the baseline instead
1188
                let y_offset = match vertical_align {
                    StyleVerticalAlign::Top => 0.0,
1188
                    StyleVerticalAlign::Middle => (content_box_height - content_height) * 0.5,
                    StyleVerticalAlign::Bottom => content_box_height - content_height,
                    // align with the row baseline. cell_baseline = distance from top of cell box
                    // to cell's baseline; row_baseline = distance from top of row to row's baseline
                    StyleVerticalAlign::Baseline
                    | StyleVerticalAlign::Sub
                    | StyleVerticalAlign::Superscript
                    | StyleVerticalAlign::TextTop
                    | StyleVerticalAlign::TextBottom
                    | StyleVerticalAlign::Percentage(_)
                    | StyleVerticalAlign::Length(_) => {
                        let row_baseline = table_ctx.row_baselines.get(cell_info.row).copied().unwrap_or(0.0);
                        (row_baseline - precomputed_cell_baseline).max(0.0)
                    }
                };
1188
                debug_info!(
1188
                    ctx,
1188
                    "[position_table_cells] Cell {}: vertical-align={:?}, border_box_height={}, \
1188
                     content_box_height={}, content_height={}, y_offset={}",
                    cell_info.node_index,
                    vertical_align,
                    height,
                    content_box_height,
                    content_height,
                    y_offset
                );
1188
                if y_offset.abs() > 0.01 {
220
                    Some((y_offset, cached_layout.available_width, cached_layout.has_floats))
                } else {
968
                    None
                }
            } else {
2552
                None
            }
        } else {
            None
        };
        // Apply the vertical alignment adjustment (requires mutable borrow)
3740
        if let Some((y_offset, available_width, has_floats)) = vertical_align_adjustment {
220
            if let Some(warm_mut) = tree.warm_mut(cell_info.node_index) {
220
                if let Some(ref cached_layout) = warm_mut.inline_layout_result {
                    use std::sync::Arc;
                    use crate::text3::cache::{PositionedItem, UnifiedLayout};
220
                    let adjusted_items: Vec<PositionedItem> = cached_layout.layout
220
                        .items
220
                        .iter()
220
                        .map(|item| PositionedItem {
1980
                            item: item.item.clone(),
1980
                            position: crate::text3::cache::Point {
1980
                                x: item.position.x,
1980
                                y: item.position.y + y_offset,
1980
                            },
1980
                            line_index: item.line_index,
1980
                        })
220
                        .collect();
220
                    let adjusted_layout = UnifiedLayout {
220
                        items: adjusted_items,
220
                        overflow: cached_layout.layout.overflow.clone(),
220
                    };
                    // Keep the same constraint type from the cached layout
220
                    let mut cil = CachedInlineLayout::new(
220
                        Arc::new(adjusted_layout),
220
                        available_width,
220
                        has_floats,
                    );
                    // LineShift preserves content; carry the hash so Phase 2d
                    // can still validly fast-path this layout (#11).
220
                    cil.inline_content_hash = cached_layout.inline_content_hash;
220
                    warm_mut.inline_layout_result = Some(cil);
                }
            }
3520
        }
        // Store position relative to table origin
3740
        let position = LogicalPosition::from_main_cross(y, x, writing_mode);
        // Insert position into map so cache module can position the cell
3740
        positions.insert(cell_info.node_index, position);
3740
        debug_log!(
3740
            ctx,
3740
            "Cell at row={}, col={}: pos=({:.2}, {:.2}), size=({:.2}x{:.2})",
            cell_info.row,
            cell_info.column,
            x,
            y,
            width,
            height
        );
    }
1452
    Ok(positions)
1452
}
/// Gathers all inline content for `text3`, recursively laying out `inline-block` children
/// to determine their size and baseline before passing them to the text engine.
///
/// This function also assigns IFC membership to all participating nodes:
/// - The IFC root gets an `ifc_id` assigned
/// - Each text/inline child gets `ifc_membership` set with a reference back to the IFC root
///
/// This mapping enables efficient cursor hit-testing: when a text node is clicked,
/// we can find its parent IFC's `inline_layout_result` via `ifc_membership.ifc_root_layout_index`.
// +spec:display-property:63a38b - inline box boundaries and out-of-flow elements are ignored for text adjacency (white space, line-breaking, text-transform)
65625
fn collect_and_measure_inline_content<T: ParsedFontTrait>(
65625
    ctx: &mut LayoutContext<'_, T>,
65625
    text_cache: &mut TextLayoutCache,
65625
    tree: &mut LayoutTree,
65625
    ifc_root_index: usize,
65625
    constraints: &LayoutConstraints,
65625
) -> Result<(Vec<InlineContent>, HashMap<ContentIndex, usize>)> {
    use crate::solver3::layout_tree::{IfcId, IfcMembership};
    use crate::text3::cache::InlineContent;
65625
    let mut content = Vec::new();
65625
    let mut child_map = HashMap::new();
65625
    collect_and_measure_inline_content_impl(
65625
        ctx,
65625
        text_cache,
65625
        tree,
65625
        ifc_root_index,
65625
        constraints,
65625
        &mut content,
65625
        &mut child_map,
    )?;
65625
    Ok((content, child_map))
65625
}
65625
fn collect_and_measure_inline_content_impl<T: ParsedFontTrait>(
65625
    ctx: &mut LayoutContext<'_, T>,
65625
    text_cache: &mut TextLayoutCache,
65625
    tree: &mut LayoutTree,
65625
    ifc_root_index: usize,
65625
    constraints: &LayoutConstraints,
65625
    content: &mut Vec<InlineContent>,
65625
    child_map: &mut HashMap<ContentIndex, usize>,
65625
) -> Result<()> {
    use crate::solver3::layout_tree::{IfcId, IfcMembership};
65625
    debug_ifc_layout!(
65625
        ctx,
65625
        "collect_and_measure_inline_content: node_index={}",
        ifc_root_index
    );
    // Generate a unique IFC ID for this inline formatting context
65625
    let ifc_id = IfcId::unique();
    // Store IFC ID on the IFC root node
65625
    if let Some(cold_node) = tree.cold_mut(ifc_root_index) {
65625
        cold_node.ifc_id = Some(ifc_id);
65625
    }
    // [g129/g130 az-web-lift] `content` and `child_map` are now caller-provided out-params
    // (the by-value `(Vec, HashMap)` return mis-lifted its len on the web backend). The caller
    // passes them in EMPTY; this body fills them exactly as before. `child_map` maps the
    // `ContentIndex` used by text3 back to the `LayoutNode` index.
    // Track the current run index for IFC membership assignment
65625
    let mut current_run_index: u32 = 0;
    // [g134/g135 az-web-lift DIAG] out-param pointer + tree validity at _impl entry. The early Err is
    // the `tree.get(ifc_root_index).ok_or(InvalidTree)?` at 6449/6706 (no other `?` before the first
    // content push) — capture whether `tree` is valid (nodes.len) and tree.get(idx) actually works.
    #[cfg(feature = "web_lift")]
    unsafe {
        crate::az_mark((0x60690) as u32, (content as *const _ as usize as u32) as u32);
        crate::az_mark((0x60694) as u32, (ifc_root_index as u32 | 0xC0DE0000u32) as u32);
        crate::az_mark((0x606A8) as u32, (tree.nodes.len() as u32) as u32);
        crate::az_mark((0x606AC) as u32, (tree.get(ifc_root_index).is_some() as u32 | 0xC0DE0000u32) as u32);
        crate::az_mark((0x606B0) as u32, (tree.root as u32) as u32);
        // [g147 az-web-lift DIAG] CALLEE-side tree ptr + nodes.len indexed by ifc_root_index
        // (0x60940+ = nodes.len, 0x60960+ = tree ptr). Pair with layout_ifc's 0x60900+/0x60920+.
        let slot = (ifc_root_index & 7) * 4;
        crate::az_mark(((0x60940 + slot)) as u32, (tree.nodes.len() as u32) as u32);
        crate::az_mark(((0x60960 + slot)) as u32, ((&*tree as *const LayoutTree as usize) as u32) as u32);
    }
65625
    let ifc_root_node = tree.get(ifc_root_index).ok_or(LayoutError::InvalidTree)?;
    // [g135] reached past the 6449 tree.get.
    #[cfg(feature = "web_lift")]
    unsafe { crate::az_mark((0x606A4) as u32, (0x0000_6449u32) as u32); }
    // Check if this is an anonymous IFC wrapper (has no DOM ID)
65625
    let is_anonymous = ifc_root_node.dom_node_id.is_none();
    // Get the DOM node ID of the IFC root, or find it from parent/children for anonymous boxes
    // CSS 2.2 § 9.2.1.1: Anonymous boxes inherit properties from their enclosing box
65625
    let ifc_root_dom_id = match ifc_root_node.dom_node_id {
65449
        Some(id) => id,
        None => {
            // Anonymous box - get DOM ID from parent or first child with DOM ID
176
            let parent_dom_id = ifc_root_node
176
                .parent
176
                .and_then(|p| tree.get(p))
176
                .and_then(|n| n.dom_node_id);
176
            if let Some(id) = parent_dom_id {
176
                id
            } else {
                // Try to find DOM ID from first child
                match tree.children(ifc_root_index)
                    .iter()
                    .filter_map(|&child_idx| tree.get(child_idx))
                    .filter_map(|n| n.dom_node_id)
                    .next()
                {
                    Some(id) => id,
                    None => {
                        debug_warning!(ctx, "IFC root and all ancestors/children have no DOM ID");
                        return Ok(());
                    }
                }
            }
        }
    };
    // Collect children to avoid holding an immutable borrow during iteration
65625
    let children: Vec<_> = tree.children(ifc_root_index).to_vec();
65625
    drop(ifc_root_node);
65625
    debug_ifc_layout!(
65625
        ctx,
65625
        "Node {} has {} layout children, is_anonymous={}",
        ifc_root_index,
65625
        children.len(),
        is_anonymous
    );
    // For anonymous IFC wrappers, we collect content from layout tree children
    // For regular IFC roots, we also check DOM children for text nodes
65625
    if is_anonymous {
        // Anonymous IFC wrapper - iterate over layout tree children and collect their content
176
        for (item_idx, &child_index) in children.iter().enumerate() {
176
            let content_index = ContentIndex {
176
                run_index: ifc_root_index as u32,
176
                item_index: item_idx as u32,
176
            };
176
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
176
            let Some(dom_id) = child_node.dom_node_id else {
                debug_warning!(
                    ctx,
                    "Anonymous IFC child at index {} has no DOM ID",
                    child_index
                );
                continue;
            };
176
            let node_data = &ctx.styled_dom.node_data.as_container()[dom_id];
            // Check if this is a text node
176
            if let NodeType::Text(ref text_content) = node_data.get_node_type() {
176
                debug_info!(
176
                    ctx,
176
                    "[collect_and_measure_inline_content] OK: Found text node (DOM {:?}) in anonymous wrapper: '{}'",
                    dom_id,
176
                    text_content.as_str()
                );
                // Get style from the TEXT NODE itself (dom_id), not the IFC root
                // This ensures inline styles like color: #666666 are applied to the text
176
                let style = Arc::new(get_style_properties(ctx.styled_dom, dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height)));
176
                let text_items = split_text_for_whitespace(
176
                    ctx.styled_dom,
176
                    dom_id,
176
                    text_content.as_str(),
176
                    style,
                );
176
                content.extend(text_items);
176
                child_map.insert(content_index, child_index);
                // Set IFC membership on the text node - drop child_node borrow first
176
                drop(child_node);
176
                if let Some(warm_mut) = tree.warm_mut(child_index) {
176
                    warm_mut.ifc_membership = Some(IfcMembership {
176
                        ifc_id,
176
                        ifc_root_layout_index: ifc_root_index,
176
                        run_index: current_run_index,
176
                    });
176
                }
176
                current_run_index += 1;
176
                continue;
            }
            // Non-text inline child - add as shape for inline-block
            let display = get_display_property(ctx.styled_dom, Some(dom_id)).unwrap_or_default();
            if display != LayoutDisplay::Inline {
                // +spec:display-property:a37a9a - atomic inline-level boxes treated as neutral characters in bidi reordering
                // This is an atomic inline-level box (e.g., inline-block, image).
                // We must determine its size and baseline before passing it to text3.
                // The intrinsic sizing pass has already calculated its preferred size.
                let intrinsic_size = tree.warm(child_index).and_then(|w| w.intrinsic_sizes.clone()).unwrap_or_default();
                let box_props = child_node.box_props.unpack();
                let styled_node_state = ctx
                    .styled_dom
                    .styled_nodes
                    .as_container()
                    .get(dom_id)
                    .map(|n| n.styled_node_state.clone())
                    .unwrap_or_default();
                // Calculate tentative border-box size based on CSS properties
                let tentative_size = crate::solver3::sizing::calculate_used_size_for_node(
                    ctx.styled_dom,
                    Some(dom_id),
                    &constraints.containing_block_size,
                    intrinsic_size,
                    &box_props,
                    &ctx.viewport_size,
                )?;
                let writing_mode = get_writing_mode(ctx.styled_dom, dom_id, &styled_node_state)
                    .unwrap_or_default();
                // Determine content-box size for laying out children
                let content_box_size = box_props.inner_size(tentative_size, writing_mode);
                // To find its height and baseline, we must lay out its contents.
                let child_wm_ctx = super::geometry::WritingModeContext::new(
                    writing_mode,
                    get_direction_property(ctx.styled_dom, dom_id, &styled_node_state)
                        .unwrap_or_default(),
                    get_text_orientation_property(ctx.styled_dom, dom_id, &styled_node_state)
                        .unwrap_or_default(),
                );
                let child_constraints = LayoutConstraints {
                    available_size: LogicalSize::new(content_box_size.width, f32::INFINITY),
                    writing_mode,
                    writing_mode_ctx: child_wm_ctx,
                    bfc_state: None,
                    text_align: TextAlign::Start,
                    containing_block_size: constraints.containing_block_size,
                    available_width_type: Text3AvailableSpace::Definite(content_box_size.width),
                };
                // Drop the immutable borrow before calling layout_formatting_context
                drop(child_node);
                // Recursively lay out the inline-block to get its final height and baseline.
                let mut empty_float_cache = HashMap::new();
                let layout_result = layout_formatting_context(
                    ctx,
                    tree,
                    text_cache,
                    child_index,
                    &child_constraints,
                    &mut empty_float_cache,
                )?;
                let css_height = get_css_height(ctx.styled_dom, dom_id, &styled_node_state);
                // Replaced elements (image / VirtualView) have no flow content, so the
                // measured content_height is 0 — treat their auto height like an
                // explicit height (CSS/intrinsic-resolved tentative_size).
                let is_replaced_atomic = {
                    let nd = &ctx.styled_dom.node_data.as_container()[dom_id];
                    matches!(nd.get_node_type(), NodeType::Image(_)) || nd.is_virtual_view_node()
                };
                // Determine final border-box height
                let final_height = match css_height.unwrap_or_default() {
                    LayoutHeight::Auto if !is_replaced_atomic => {
                        let content_height = layout_result.output.overflow_size.height;
                        content_height
                            + box_props.padding.main_sum(writing_mode)
                            + box_props.border.main_sum(writing_mode)
                    }
                    _ => tentative_size.height,
                };
                let final_size = LogicalSize::new(tentative_size.width, final_height);
                // Update the node in the tree with its now-known used size.
                tree.get_mut(child_index).unwrap().used_size = Some(final_size);
                // CSS 2.2 § 10.8.1: inline-block baseline fallback
                // If overflow is not 'visible', use bottom margin edge as baseline
                let overflow_x = get_overflow_x(ctx.styled_dom, dom_id, &styled_node_state).unwrap_or_default();
                let overflow_y = get_overflow_y(ctx.styled_dom, dom_id, &styled_node_state).unwrap_or_default();
                let overflow_is_visible = matches!(
                    (overflow_x, overflow_y),
                    (LayoutOverflow::Visible, LayoutOverflow::Visible)
                );
                let baseline_offset = if overflow_is_visible {
                    layout_result.output.baseline.unwrap_or(final_height)
                } else {
                    final_height
                };
                // +spec:box-model:66ad24 - inline-axis margins, borders, padding respected for inline-level boxes (no collapsing)
                // The margin-box size is used so text3 positions inline-blocks with proper spacing
                let margin = &box_props.margin;
                let margin_box_width = final_size.width + margin.left + margin.right;
                let margin_box_height = final_size.height + margin.top + margin.bottom;
                // For inline-block shapes, text3 uses the content array index as run_index
                // and always item_index=0 for objects. We must match this when inserting into child_map.
                let shape_content_index = ContentIndex {
                    run_index: content.len() as u32,
                    item_index: 0,
                };
                content.push(InlineContent::Shape(InlineShape {
                    shape_def: ShapeDefinition::Rectangle {
                        size: crate::text3::cache::Size {
                            // Use margin-box size for positioning in inline flow
                            width: margin_box_width,
                            height: margin_box_height,
                        },
                        corner_radius: None,
                    },
                    fill: None,
                    stroke: None,
                    // Adjust baseline offset by top margin
                    baseline_offset: baseline_offset + margin.top,
                    alignment: crate::solver3::getters::get_vertical_align_for_node(ctx.styled_dom, dom_id),
                    source_node_id: Some(dom_id),
                }));
                child_map.insert(shape_content_index, child_index);
            } else {
                // Regular inline element - collect its text children
                let span_style = get_style_properties(ctx.styled_dom, dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height));
                collect_inline_span_recursive(
                    ctx,
                    tree,
                    dom_id,
                    span_style,
                    content,
                    child_map,
                    &children,
                    constraints,
                )?;
            }
        }
176
        return Ok(());
65449
    }
    // Regular (non-anonymous) IFC root - check for list markers and use DOM traversal
    // Check if this IFC root OR its parent is a list-item and needs a marker
    // Case 1: IFC root itself is list-item (e.g., <li> with display: list-item)
    // Case 2: IFC root's parent is list-item (e.g., <li><text>...</text></li>)
65449
    let ifc_root_node = tree.get(ifc_root_index).ok_or(LayoutError::InvalidTree)?;
    // [g135] reached past the 6706 tree.get.
    #[cfg(feature = "web_lift")]
    unsafe { crate::az_mark((0x606A4) as u32, (0x0000_6706u32) as u32); }
65449
    let mut list_item_dom_id: Option<NodeId> = None;
    // Check IFC root itself
65449
    if let Some(dom_id) = ifc_root_node.dom_node_id {
        use crate::solver3::getters::get_display_property;
65449
        if let MultiValue::Exact(display) = get_display_property(ctx.styled_dom, Some(dom_id)) {
            use LayoutDisplay;
65449
            if display == LayoutDisplay::ListItem {
                debug_ifc_layout!(ctx, "IFC root NodeId({:?}) is list-item", dom_id);
                list_item_dom_id = Some(dom_id);
65449
            }
        }
    }
    // Check IFC root's parent
65449
    if list_item_dom_id.is_none() {
65449
        if let Some(parent_idx) = ifc_root_node.parent {
65229
            if let Some(parent_node) = tree.get(parent_idx) {
65229
                if let Some(parent_dom_id) = parent_node.dom_node_id {
                    use crate::solver3::getters::get_display_property;
65221
                    if let MultiValue::Exact(display) = get_display_property(ctx.styled_dom, Some(parent_dom_id)) {
                        use LayoutDisplay;
65221
                        if display == LayoutDisplay::ListItem {
                            debug_ifc_layout!(
                                ctx,
                                "IFC root parent NodeId({:?}) is list-item",
                                parent_dom_id
                            );
                            list_item_dom_id = Some(parent_dom_id);
65221
                        }
                    }
8
                }
            }
220
        }
    }
    // If we found a list-item, generate markers
65449
    if let Some(list_dom_id) = list_item_dom_id {
        debug_ifc_layout!(
            ctx,
            "Found list-item (NodeId({:?})), generating marker",
            list_dom_id
        );
        // Find the layout node index for the list-item DOM node
        let list_item_layout_idx = tree
            .nodes
            .iter()
            .enumerate()
            .find(|(idx, node)| {
                node.dom_node_id == Some(list_dom_id) && tree.warm(*idx).and_then(|w| w.pseudo_element).is_none()
            })
            .map(|(idx, _)| idx);
        if let Some(list_idx) = list_item_layout_idx {
            // Per CSS spec, the ::marker pseudo-element is the first child of the list-item
            // Find the ::marker pseudo-element in the list-item's children
            let marker_idx = tree.children(list_idx)
                .iter()
                .find(|&&child_idx| {
                    tree.warm(child_idx)
                        .map(|w| w.pseudo_element == Some(PseudoElement::Marker))
                        .unwrap_or(false)
                })
                .copied();
            if let Some(marker_idx) = marker_idx {
                debug_ifc_layout!(ctx, "Found ::marker pseudo-element at index {}", marker_idx);
                // Get the DOM ID for style resolution (marker references the same DOM node as
                // list-item)
                let list_dom_id_for_style = tree
                    .get(marker_idx)
                    .and_then(|n| n.dom_node_id)
                    .unwrap_or(list_dom_id);
                // Get list-style-position to determine marker positioning
                // Default is 'outside' per CSS Lists Module Level 3
                let list_style_position =
                    get_list_style_position(ctx.styled_dom, Some(list_dom_id));
                let position_outside =
                    matches!(list_style_position, StyleListStylePosition::Outside);
                debug_ifc_layout!(
                    ctx,
                    "List marker list-style-position: {:?} (outside={})",
                    list_style_position,
                    position_outside
                );
                // Generate marker text segments - font fallback happens during shaping
                let base_style =
                    Arc::new(get_style_properties(ctx.styled_dom, list_dom_id_for_style, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height)));
                let marker_segments = generate_list_marker_segments(
                    tree,
                    ctx.styled_dom,
                    marker_idx, // Pass the marker index, not the list-item index
                    ctx.counters,
                    base_style,
                    ctx.debug_messages,
                );
                debug_ifc_layout!(
                    ctx,
                    "Generated {} list marker segments",
                    marker_segments.len()
                );
                // Add markers as InlineContent::Marker with position information
                // Outside markers will be positioned in the padding gutter by the layout engine
                for segment in marker_segments {
                    content.push(InlineContent::Marker {
                        run: segment,
                        position_outside,
                    });
                }
            } else {
                debug_ifc_layout!(
                    ctx,
                    "WARNING: List-item at index {} has no ::marker pseudo-element",
                    list_idx
                );
            }
        }
65449
    }
65449
    drop(ifc_root_node);
    // IMPORTANT: We need to traverse the DOM, not just the layout tree!
    //
    // According to CSS spec, a block container with inline-level children establishes
    // an IFC and should collect ALL inline content, including text nodes.
    // Text nodes exist in the DOM but might not have their own layout tree nodes.
    // Debug: Check what the node_hierarchy says about this node
65449
    let node_hier_item = &ctx.styled_dom.node_hierarchy.as_container()[ifc_root_dom_id];
65449
    debug_info!(
65449
        ctx,
65449
        "[collect_and_measure_inline_content] DEBUG: node_hier_item.first_child={:?}, \
65449
         last_child={:?}",
65449
        node_hier_item.first_child_id(ifc_root_dom_id),
65449
        node_hier_item.last_child_id()
    );
65449
    let ifc_root_node_data = &ctx.styled_dom.node_data.as_container()[ifc_root_dom_id];
    // SPECIAL CASE: If the IFC root itself is a text node (leaf node),
    // add its text content directly instead of iterating over children
65449
    if let NodeType::Text(ref text_content) = ifc_root_node_data.get_node_type() {
18958
        let style = Arc::new(get_style_properties(ctx.styled_dom, ifc_root_dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height)));
18958
        let text_items = split_text_for_whitespace(
18958
            ctx.styled_dom,
18958
            ifc_root_dom_id,
18958
            text_content.as_str(),
18958
            style,
        );
18958
        content.extend(text_items);
18958
        return Ok(());
46491
    }
46491
    let _ifc_root_node_type = match ifc_root_node_data.get_node_type() {
30962
        NodeType::Div => "Div",
        NodeType::Text(_) => "Text",
132
        NodeType::Body => "Body",
15397
        _ => "Other",
    };
    // [g138 az-web-lift] Collect `dom_children` HERE — immediately before the loop, AFTER the
    // get_node_type() calls above. Those calls were corrupting the `dom_children` Vec's stack-slot
    // header (g137 PROVED: `dom_children.len()` reads 1 right after `.collect()` but 0 in the loop's
    // `0..len` range a few calls later — the recurring SP-leak / stack-address mis-lift). With NO call
    // between this `.collect()` and the loop, the header survives the range evaluation + first index.
46491
    let dom_children: Vec<NodeId> = ifc_root_dom_id
46491
        .az_children(&ctx.styled_dom.node_hierarchy.as_container())
46491
        .collect();
    // [g139 az-web-lift] The loop's `dom_children.len()` read MIS-LIFTS to 0 even though the in-memory
    // value is 1 (g138: the volatile marker reads 1 but the loop's `0..len` range reads 0 with NOTHING
    // between — the optimizer's SROA'd len read is mis-tracked by the lift; only a FORCED/volatile read is
    // correct; same Vec-len mis-lift class as the original sret bug, here on std `collect()` which can't be
    // out-param'd). Read len via a volatile round-trip (guaranteed-correct, like the marker) and index via
    // get_unchecked (the index's bounds-check len read mis-lifts the same way; len is valid → sound).
    // [g195 — collect_and_measure_inline_content_impl is DEAD on the web lift (NOT lifted for hello-world
    // OR web-nested-text; both lay out via measure_intrinsic_widths + layout_flow instead). So this g139
    // Vec-len workaround never executes on the web lift → it's irrelevant/deletable (kept: harmless, and
    // unverified-dead for other layouts). The cron's "collect_and_measure Vec-len" target is a DEAD PATH.]
    #[cfg(feature = "web_lift")]
    let dom_children_len = unsafe {
        crate::az_mark((0x606B4) as u32, (dom_children.len() as u32) as u32);
        crate::az_mark((0x606A4) as u32, (0x0000_6863u32) as u32);
        crate::az_mark_read(0x606B4) as usize
    };
    #[cfg(not(feature = "web_lift"))]
46491
    let dom_children_len = dom_children.len();
46495
    for item_idx in 0..dom_children_len {
46495
        let dom_child_id = unsafe { *dom_children.get_unchecked(item_idx) };
46495
        let content_index = ContentIndex {
46495
            run_index: ifc_root_index as u32,
46495
            item_index: item_idx as u32,
46495
        };
46495
        let node_data = &ctx.styled_dom.node_data.as_container()[dom_child_id];
        // [g136] loop body entered; capture the FIRST child's node_type (does it read as Text?).
        #[cfg(feature = "web_lift")]
        unsafe {
            if item_idx == 0 {
                crate::az_mark((0x606B8) as u32, (match node_data.get_node_type() {
                    NodeType::Text(_) => 0xC0DE_7E70u32,
                    NodeType::Div => 0xC0DE_D11Fu32,
                    NodeType::Body => 0xC0DE_B0D1u32,
                    _ => 0xC0DE_0000u32,
                }) as u32);
            }
            crate::az_mark((0x606A4) as u32, (0x0000_6896u32) as u32);
        }
        // Check if this is a text node
46495
        if let NodeType::Text(ref text_content) = node_data.get_node_type() {
35445
            debug_info!(
35445
                ctx,
35445
                "[collect_and_measure_inline_content] OK: Found text node (DOM child {:?}): '{}'",
                dom_child_id,
35445
                text_content.as_str()
            );
            // Get style from the TEXT NODE itself (dom_child_id), not the IFC root
            // This ensures inline styles like color: #666666 are applied to the text
            // Uses split_text_for_whitespace to correctly handle white-space: pre with \n
35445
            let style = Arc::new(get_style_properties(ctx.styled_dom, dom_child_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height)));
35445
            let text_items = split_text_for_whitespace(
35445
                ctx.styled_dom,
35445
                dom_child_id,
35445
                text_content.as_str(),
35445
                style,
            );
35445
            content.extend(text_items);
            // [g136] TEXT branch taken + pushed; content.len now.
            #[cfg(feature = "web_lift")]
            unsafe {
                crate::az_mark((0x606A4) as u32, (0x0000_6905u32) as u32);
                crate::az_mark((0x606BC) as u32, (content.len() as u32) as u32);
            }
            // Set IFC membership on the text node's layout node (if it exists)
            // Text nodes may or may not have their own layout tree entry depending on
            // whether they're wrapped in an anonymous IFC wrapper
35445
            if let Some(&layout_idx) = tree.dom_to_layout.get(&dom_child_id).and_then(|v| v.first()) {
35445
                if let Some(warm_mut) = tree.warm_mut(layout_idx) {
35445
                    warm_mut.ifc_membership = Some(IfcMembership {
35445
                        ifc_id,
35445
                        ifc_root_layout_index: ifc_root_index,
35445
                        run_index: current_run_index,
35445
                    });
35445
                }
            }
35445
            current_run_index += 1;
35445
            continue;
11050
        }
        // For non-text nodes, find their corresponding layout tree node
11050
        let child_index = children
11050
            .iter()
11054
            .find(|&&idx| {
11054
                tree.get(idx)
11054
                    .and_then(|n| n.dom_node_id)
11054
                    .map(|id| id == dom_child_id)
11054
                    .unwrap_or(false)
11054
            })
11050
            .copied();
11050
        let Some(child_index) = child_index else {
            debug_info!(
                ctx,
                "[collect_and_measure_inline_content] WARN: DOM child {:?} has no layout node",
                dom_child_id
            );
            continue;
        };
        // [g136] NON-TEXT branch taken (text child mis-classified?) — reached tree.get(child_index).
        #[cfg(feature = "web_lift")]
        unsafe { crate::az_mark((0x606A4) as u32, (0x0000_6942u32) as u32); }
11050
        let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
        // At this point we have a non-text DOM child with a layout node
11050
        let dom_id = child_node.dom_node_id.unwrap();
11050
        let display = get_display_property(ctx.styled_dom, Some(dom_id)).unwrap_or_default();
11050
        if display != LayoutDisplay::Inline {
            // This is an atomic inline-level box (e.g., inline-block, image).
            // We must determine its size and baseline before passing it to text3.
            // The intrinsic sizing pass has already calculated its preferred size.
176
            let intrinsic_size = tree.warm(child_index).and_then(|w| w.intrinsic_sizes.clone()).unwrap_or_default();
176
            let box_props = child_node.box_props.unpack();
176
            let styled_node_state = ctx
176
                .styled_dom
176
                .styled_nodes
176
                .as_container()
176
                .get(dom_id)
176
                .map(|n| n.styled_node_state.clone())
176
                .unwrap_or_default();
            // Calculate tentative border-box size based on CSS properties
            // This correctly handles explicit width/height, box-sizing, and constraints
176
            let tentative_size = crate::solver3::sizing::calculate_used_size_for_node(
176
                ctx.styled_dom,
176
                Some(dom_id),
176
                &constraints.containing_block_size,
176
                intrinsic_size,
176
                &box_props,
176
                &ctx.viewport_size,
            )?;
176
            let writing_mode =
176
                get_writing_mode(ctx.styled_dom, dom_id, &styled_node_state).unwrap_or_default();
            // Determine content-box size for laying out children
176
            let content_box_size = box_props.inner_size(tentative_size, writing_mode);
176
            debug_info!(
176
                ctx,
176
                "[collect_and_measure_inline_content] Inline-block NodeId({:?}): \
176
                 tentative_border_box={:?}, content_box={:?}",
                dom_id,
                tentative_size,
                content_box_size
            );
            // To find its height and baseline, we must lay out its contents.
176
            let child_wm_ctx = super::geometry::WritingModeContext::new(
176
                writing_mode,
176
                get_direction_property(ctx.styled_dom, dom_id, &styled_node_state)
176
                    .unwrap_or_default(),
176
                get_text_orientation_property(ctx.styled_dom, dom_id, &styled_node_state)
176
                    .unwrap_or_default(),
            );
176
            let child_constraints = LayoutConstraints {
176
                available_size: LogicalSize::new(content_box_size.width, f32::INFINITY),
176
                writing_mode,
176
                writing_mode_ctx: child_wm_ctx,
176
                // Inline-blocks establish a new BFC, so no state is passed in.
176
                bfc_state: None,
176
                // Does not affect size/baseline of the container.
176
                text_align: TextAlign::Start,
176
                containing_block_size: constraints.containing_block_size,
176
                available_width_type: Text3AvailableSpace::Definite(content_box_size.width),
176
            };
            // Drop the immutable borrow before calling layout_formatting_context
176
            drop(child_node);
            // Recursively lay out the inline-block to get its final height and baseline.
            // Note: This does not affect its final position, only its dimensions.
176
            let mut empty_float_cache = HashMap::new();
176
            let layout_result = layout_formatting_context(
176
                ctx,
176
                tree,
176
                text_cache,
176
                child_index,
176
                &child_constraints,
176
                &mut empty_float_cache,
            )?;
176
            let css_height = get_css_height(ctx.styled_dom, dom_id, &styled_node_state);
            // Replaced elements (image / VirtualView) have no flow content, so the
            // measured content_height is 0 — treat their auto height like an explicit
            // height (use the CSS/intrinsic-resolved tentative_size). Fixes 0-height
            // images / VirtualViews laid out as atomic inline-blocks.
176
            let is_replaced_atomic = {
176
                let nd = &ctx.styled_dom.node_data.as_container()[dom_id];
176
                matches!(nd.get_node_type(), NodeType::Image(_)) || nd.is_virtual_view_node()
            };
            // Determine final border-box height
176
            let final_height = match css_height.clone().unwrap_or_default() {
176
                LayoutHeight::Auto if !is_replaced_atomic => {
                    // For auto height, add padding and border to the content height
176
                    let content_height = layout_result.output.overflow_size.height;
176
                    content_height
176
                        + box_props.padding.main_sum(writing_mode)
176
                        + box_props.border.main_sum(writing_mode)
                }
                // Explicit height (calculate_used_size_for_node gave the border-box
                // height), OR a replaced element's auto height (intrinsic/CSS-resolved).
                _ => tentative_size.height,
            };
176
            debug_info!(
176
                ctx,
176
                "[collect_and_measure_inline_content] Inline-block NodeId({:?}): \
176
                 layout_content_height={}, css_height={:?}, final_border_box_height={}",
                dom_id,
                layout_result.output.overflow_size.height,
                css_height,
                final_height
            );
176
            let final_size = LogicalSize::new(tentative_size.width, final_height);
            // Update the node in the tree with its now-known used size.
176
            tree.get_mut(child_index).unwrap().used_size = Some(final_size);
            // CSS 2.2 § 10.8.1: For inline-block elements, the baseline is the baseline of the
            // last line box in the normal flow, unless it has either no in-flow line boxes or
            // if its 'overflow' property has a computed value other than 'visible', in which
            // case the baseline is the bottom margin edge.
            //
            // `layout_result.output.baseline` returns the Y-position of the baseline measured
            // from the TOP of the content box. But `get_item_vertical_metrics` expects
            // `baseline_offset` to be the distance from the BOTTOM to the baseline.
            //
            // Conversion: baseline_offset_from_bottom = height - baseline_from_top
            //
            // If no baseline is found (e.g., the inline-block has no text), or if
            // overflow is not 'visible', we fall back to the bottom margin edge
            // (baseline_offset = 0, meaning baseline at bottom).
176
            let overflow_x = get_overflow_x(ctx.styled_dom, dom_id, &styled_node_state).unwrap_or_default();
176
            let overflow_y = get_overflow_y(ctx.styled_dom, dom_id, &styled_node_state).unwrap_or_default();
176
            let overflow_is_visible = matches!(
176
                (overflow_x, overflow_y),
                (LayoutOverflow::Visible, LayoutOverflow::Visible)
            );
176
            let baseline_from_top = layout_result.output.baseline;
176
            let baseline_offset = match baseline_from_top {
                Some(baseline_y) if overflow_is_visible => {
                    // baseline_y is measured from top of content box
                    // We need to add padding and border to get the position within the border-box
                    let content_box_top = box_props.padding.top + box_props.border.top;
                    let baseline_from_border_box_top = baseline_y + content_box_top;
                    // Convert to distance from bottom
                    (final_height - baseline_from_border_box_top).max(0.0)
                }
                _ => {
                    // No baseline found or overflow != visible - use bottom margin edge
176
                    0.0
                }
            };
176
            debug_info!(
176
                ctx,
176
                "[collect_and_measure_inline_content] Inline-block NodeId({:?}): \
176
                 baseline_from_top={:?}, final_height={}, baseline_offset_from_bottom={}",
                dom_id,
                baseline_from_top,
                final_height,
                baseline_offset
            );
            // Get margins for inline-block positioning
            // For inline-blocks, we need to include margins in the shape size
            // so that text3 positions them correctly with spacing
176
            let margin = &box_props.margin;
176
            let margin_box_width = final_size.width + margin.left + margin.right;
176
            let margin_box_height = final_size.height + margin.top + margin.bottom;
            // For inline-block shapes, text3 uses the content array index as run_index
            // and always item_index=0 for objects. We must match this when inserting into child_map.
176
            let shape_content_index = ContentIndex {
176
                run_index: content.len() as u32,
176
                item_index: 0,
176
            };
            // the box used for alignment is the margin box" - using margin_box_width/height here
176
            content.push(InlineContent::Shape(InlineShape {
176
                shape_def: ShapeDefinition::Rectangle {
176
                    size: crate::text3::cache::Size {
176
                        // Use margin-box size for positioning in inline flow
176
                        width: margin_box_width,
176
                        height: margin_box_height,
176
                    },
176
                    corner_radius: None,
176
                },
176
                fill: None,
176
                stroke: None,
176
                // Adjust baseline offset by top margin
176
                baseline_offset: baseline_offset + margin.top,
176
                alignment: crate::solver3::getters::get_vertical_align_for_node(ctx.styled_dom, dom_id),
176
                source_node_id: Some(dom_id),
176
            }));
176
            child_map.insert(shape_content_index, child_index);
        } else if let NodeType::Image(image_ref) =
10874
            ctx.styled_dom.node_data.as_container()[dom_id].get_node_type()
        {
            // +spec:replaced-elements:31a782 - replaced elements (img) not rendered purely by CSS box concepts
            // Images are replaced elements - they have intrinsic dimensions
            // and CSS width/height can constrain them
            // Re-get child_node since we dropped it earlier for the inline-block case
            let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
            let box_props = child_node.box_props.unpack();
            // Get intrinsic size from the image data or fall back to layout node
            let intrinsic_size = tree.warm(child_index)
                .and_then(|w| w.intrinsic_sizes.clone())
                .unwrap_or(IntrinsicSizes {
                    max_content_width: 50.0,
                    max_content_height: 50.0,
                    ..Default::default()
                });
            // Get styled node state for CSS property lookup
            let styled_node_state = ctx
                .styled_dom
                .styled_nodes
                .as_container()
                .get(dom_id)
                .map(|n| n.styled_node_state.clone())
                .unwrap_or_default();
            // Calculate the used size respecting CSS width/height constraints
            let tentative_size = crate::solver3::sizing::calculate_used_size_for_node(
                ctx.styled_dom,
                Some(dom_id),
                &constraints.containing_block_size,
                intrinsic_size.clone(),
                &box_props,
                &ctx.viewport_size,
            )?;
            // Drop immutable borrow before mutable access
            drop(child_node);
            // Set the used_size on the layout node so paint_rect works correctly
            let final_size = LogicalSize::new(tentative_size.width, tentative_size.height);
            tree.get_mut(child_index).unwrap().used_size = Some(final_size);
            // Calculate display size for text3 (this is what text3 uses for positioning)
            let display_width = if final_size.width > 0.0 { 
                Some(final_size.width) 
            } else { 
                None 
            };
            let display_height = if final_size.height > 0.0 { 
                Some(final_size.height) 
            } else { 
                None 
            };
            content.push(InlineContent::Image(InlineImage {
                source: ImageSource::Ref(image_ref.as_ref().clone()),
                intrinsic_size: crate::text3::cache::Size {
                    width: intrinsic_size.max_content_width,
                    height: intrinsic_size.max_content_height,
                },
                display_size: if display_width.is_some() || display_height.is_some() {
                    Some(crate::text3::cache::Size {
                        width: display_width.unwrap_or(intrinsic_size.max_content_width),
                        height: display_height.unwrap_or(intrinsic_size.max_content_height),
                    })
                } else {
                    None
                },
                // Images are bottom-aligned with the baseline by default
                baseline_offset: 0.0,
                alignment: crate::text3::cache::VerticalAlign::Baseline,
                object_fit: ObjectFit::Fill,
            }));
            // For images, text3 uses the content array index as run_index
            // and always item_index=0 for objects. We must match this.
            let image_content_index = ContentIndex {
                run_index: (content.len() - 1) as u32,  // -1 because we just pushed
                item_index: 0,
            };
            child_map.insert(image_content_index, child_index);
        } else {
            // This is a regular inline box (display: inline) - e.g., <span>, <em>, <strong>
            //
            // According to CSS Inline-3 spec §2, inline boxes are "transparent" wrappers
            // We must recursively collect their text children with inherited style
10874
            debug_info!(
10874
                ctx,
10874
                "[collect_and_measure_inline_content] Found inline span (DOM {:?}), recursing",
                dom_id
            );
10874
            let span_style = get_style_properties(ctx.styled_dom, dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height));
10874
            collect_inline_span_recursive(
10874
                ctx,
10874
                tree,
10874
                dom_id,
10874
                span_style,
10874
                content,
10874
                child_map,
10874
                &children,
10874
                constraints,
            )?;
        }
    }
    // [g134 az-web-lift DIAG] _impl reached its FINAL return; content.len as _impl sees it.
    #[cfg(feature = "web_lift")]
    unsafe {
        crate::az_mark((0x60698) as u32, (content.len() as u32) as u32);
        crate::az_mark((0x6069C) as u32, (0xC0DE069Cu32) as u32);
    }
46491
    Ok(())
65625
}
// +spec:display-property:c05c53 - inlinifying boxes can't contain block-level boxes; children are recursively inlinified
// it recursively inlinifies all of its in-flow children, so that no block-level descendants
// break up the inline formatting context in which it participates.
// +spec:display-property:aee879 - recursively inlinifies in-flow children of inline boxes
/// Recursively collects inline content from an inline span (display: inline) element.
///
/// According to CSS Inline Layout Module Level 3 §2:
///
/// "Inline boxes are transparent wrappers that wrap their content."
///
/// They don't create a new formatting context - their children participate in the
/// same IFC as the parent. This function processes:
///
/// - Text nodes: collected with the span's inherited style
/// - Nested inline spans: recursively descended
/// - Inline-blocks, images: measured and added as shapes
11710
fn collect_inline_span_recursive<T: ParsedFontTrait>(
11710
    ctx: &mut LayoutContext<'_, T>,
11710
    tree: &mut LayoutTree,
11710
    span_dom_id: NodeId,
11710
    span_style: StyleProperties,
11710
    content: &mut Vec<InlineContent>,
11710
    child_map: &mut HashMap<ContentIndex, usize>,
11710
    parent_children: &[usize], // Layout tree children of parent IFC
11710
    constraints: &LayoutConstraints,
11710
) -> Result<()> {
11710
    debug_info!(
11710
        ctx,
11710
        "[collect_inline_span_recursive] Processing inline span {:?}",
        span_dom_id
    );
    // Get DOM children of this span
11710
    let span_dom_children: Vec<NodeId> = span_dom_id
11710
        .az_children(&ctx.styled_dom.node_hierarchy.as_container())
11710
        .collect();
11710
    debug_info!(
11710
        ctx,
11710
        "[collect_inline_span_recursive] Span has {} DOM children",
11710
        span_dom_children.len()
    );
    // +spec:box-model:b7428d - empty inline boxes still have margins, padding, borders, line-height
    // +spec:box-model:cc79a4 - empty inline elements still have margins, padding, borders and line height
11710
    if span_dom_children.is_empty() {
        let node_state = &ctx.styled_dom.styled_nodes.as_container()[span_dom_id].styled_node_state;
        let font_size = get_element_font_size(ctx.styled_dom, span_dom_id, node_state);
        let line_height_value = crate::solver3::getters::get_line_height_value(
            ctx.styled_dom, span_dom_id, &node_state
        );
        let line_height = line_height_value
            .map(|v| text3::cache::LineHeight::Px(v.inner.normalized() * font_size))
            .unwrap_or(text3::cache::LineHeight::Normal);
        let cb_width = constraints.containing_block_size.main(constraints.writing_mode);
        let padding_top = crate::solver3::getters::get_css_padding_top(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let padding_bottom = crate::solver3::getters::get_css_padding_bottom(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let padding_left = crate::solver3::getters::get_css_padding_left(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let padding_right = crate::solver3::getters::get_css_padding_right(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let border_top = crate::solver3::getters::get_css_border_top_width(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let border_bottom = crate::solver3::getters::get_css_border_bottom_width(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let border_left = crate::solver3::getters::get_css_border_left_width(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let border_right = crate::solver3::getters::get_css_border_right_width(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let margin_left = crate::solver3::getters::get_css_margin_left(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let margin_right = crate::solver3::getters::get_css_margin_right(ctx.styled_dom, span_dom_id, &node_state)
            .exact().map(|pv| pv.to_pixels_internal(cb_width, font_size, DEFAULT_FONT_SIZE)).unwrap_or(0.0);
        let resolved_line_height = line_height.resolve(font_size, 0.0, 0.0, 0.0, 0);
        let total_height = resolved_line_height + padding_top + padding_bottom + border_top + border_bottom;
        let total_width = margin_left + padding_left + border_left
            + border_right + padding_right + margin_right;
        content.push(InlineContent::Shape(InlineShape {
            shape_def: ShapeDefinition::Rectangle {
                size: crate::text3::cache::Size {
                    width: total_width,
                    height: total_height,
                },
                corner_radius: None,
            },
            fill: None,
            stroke: None,
            baseline_offset: 0.0,
            alignment: crate::solver3::getters::get_vertical_align_for_node(ctx.styled_dom, span_dom_id),
            source_node_id: Some(span_dom_id),
        }));
        return Ok(());
11710
    }
23420
    for &child_dom_id in &span_dom_children {
11710
        let node_data = &ctx.styled_dom.node_data.as_container()[child_dom_id];
        // CASE 1: Text node - collect with span's style
11710
        if let NodeType::Text(ref text_content) = node_data.get_node_type() {
10874
            debug_info!(
10874
                ctx,
10874
                "[collect_inline_span_recursive] ✓ Found text in span: '{}'",
10874
                text_content.as_str()
            );
10874
            let text_items = split_text_for_whitespace(
10874
                ctx.styled_dom,
10874
                child_dom_id,
10874
                text_content.as_str(),
10874
                Arc::new(span_style.clone()),
            );
10874
            content.extend(text_items);
10874
            continue;
836
        }
        // CASE 2: Element node - check its display type
836
        let child_display =
836
            get_display_property(ctx.styled_dom, Some(child_dom_id)).unwrap_or_default();
        // Find the corresponding layout tree node
836
        let child_index = parent_children
836
            .iter()
836
            .find(|&&idx| {
836
                tree.get(idx)
836
                    .and_then(|n| n.dom_node_id)
836
                    .map(|id| id == child_dom_id)
836
                    .unwrap_or(false)
836
            })
836
            .copied();
836
        match child_display {
            LayoutDisplay::Inline => {
                // Nested inline span - recurse with child's style
836
                debug_info!(
836
                    ctx,
836
                    "[collect_inline_span_recursive] Found nested inline span {:?}",
                    child_dom_id
                );
836
                let child_style = get_style_properties(ctx.styled_dom, child_dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height));
836
                collect_inline_span_recursive(
836
                    ctx,
836
                    tree,
836
                    child_dom_id,
836
                    child_style,
836
                    content,
836
                    child_map,
836
                    parent_children,
836
                    constraints,
                )?;
            }
            LayoutDisplay::InlineBlock => {
                // Inline-block inside span - measure and add as shape
                let Some(child_index) = child_index else {
                    debug_info!(
                        ctx,
                        "[collect_inline_span_recursive] WARNING: inline-block {:?} has no layout \
                         node",
                        child_dom_id
                    );
                    continue;
                };
                let child_node = tree.get(child_index).ok_or(LayoutError::InvalidTree)?;
                let intrinsic_size = tree.warm(child_index).and_then(|w| w.intrinsic_sizes.clone()).unwrap_or_default();
                let width = intrinsic_size.max_content_width;
                let styled_node_state = ctx
                    .styled_dom
                    .styled_nodes
                    .as_container()
                    .get(child_dom_id)
                    .map(|n| n.styled_node_state.clone())
                    .unwrap_or_default();
                let writing_mode =
                    get_writing_mode(ctx.styled_dom, child_dom_id, &styled_node_state)
                        .unwrap_or_default();
                let child_wm_ctx = super::geometry::WritingModeContext::new(
                    writing_mode,
                    get_direction_property(ctx.styled_dom, child_dom_id, &styled_node_state)
                        .unwrap_or_default(),
                    get_text_orientation_property(ctx.styled_dom, child_dom_id, &styled_node_state)
                        .unwrap_or_default(),
                );
                let child_constraints = LayoutConstraints {
                    available_size: LogicalSize::new(width, f32::INFINITY),
                    writing_mode,
                    writing_mode_ctx: child_wm_ctx,
                    bfc_state: None,
                    text_align: TextAlign::Start,
                    containing_block_size: constraints.containing_block_size,
                    available_width_type: Text3AvailableSpace::Definite(width),
                };
                drop(child_node);
                let mut empty_float_cache = HashMap::new();
                let layout_result = layout_formatting_context(
                    ctx,
                    tree,
                    &mut TextLayoutCache::default(),
                    child_index,
                    &child_constraints,
                    &mut empty_float_cache,
                )?;
                let final_height = layout_result.output.overflow_size.height;
                let final_size = LogicalSize::new(width, final_height);
                tree.get_mut(child_index).unwrap().used_size = Some(final_size);
                // CSS 2.2 § 10.8.1: inline-block baseline fallback
                let overflow_x = get_overflow_x(ctx.styled_dom, child_dom_id, &styled_node_state).unwrap_or_default();
                let overflow_y = get_overflow_y(ctx.styled_dom, child_dom_id, &styled_node_state).unwrap_or_default();
                let overflow_is_visible = matches!(
                    (overflow_x, overflow_y),
                    (LayoutOverflow::Visible, LayoutOverflow::Visible)
                );
                let baseline_offset = if overflow_is_visible {
                    layout_result.output.baseline.unwrap_or(final_height)
                } else {
                    final_height
                };
                content.push(InlineContent::Shape(InlineShape {
                    shape_def: ShapeDefinition::Rectangle {
                        size: crate::text3::cache::Size {
                            width,
                            height: final_height,
                        },
                        corner_radius: None,
                    },
                    fill: None,
                    stroke: None,
                    baseline_offset,
                    alignment: crate::solver3::getters::get_vertical_align_for_node(ctx.styled_dom, child_dom_id),
                    source_node_id: Some(child_dom_id),
                }));
                // Note: We don't add to child_map here because this is inside a span
                debug_info!(
                    ctx,
                    "[collect_inline_span_recursive] Added inline-block shape {}x{}",
                    width,
                    final_height
                );
            }
            _ => {
                // +spec:display-property:0684c4 - block box inlinified: inner display becomes flow-root (treated as atomic inline)
                // in-flow children of an inline box are recursively inlinified so they
                // don't break the IFC. Treat them as inline spans and recurse into their
                // children to collect text and inline content.
                debug_info!(
                    ctx,
                    "[collect_inline_span_recursive] Inlinifying block-level child {:?} \
                     (display: {:?}) inside inline span per css-display-3 §2.7",
                    child_dom_id,
                    child_display
                );
                let child_style = get_style_properties(ctx.styled_dom, child_dom_id, ctx.system_style.as_ref(), PhysicalSize::new(ctx.viewport_size.width, ctx.viewport_size.height));
                collect_inline_span_recursive(
                    ctx,
                    tree,
                    child_dom_id,
                    child_style,
                    content,
                    child_map,
                    parent_children,
                    constraints,
                )?;
            }
        }
    }
11710
    Ok(())
11710
}
/// Positions a floated child within the BFC and updates the floating context.
/// This function is fully writing-mode aware.
fn position_floated_child(
    _child_index: usize,
    child_margin_box_size: LogicalSize,
    float_type: LayoutFloat,
    constraints: &LayoutConstraints,
    _bfc_content_box: LogicalRect,
    current_main_offset: f32,
    floating_context: &mut FloatingContext,
) -> Result<LogicalPosition> {
    let wm = constraints.writing_mode;
    let child_main_size = child_margin_box_size.main(wm);
    let child_cross_size = child_margin_box_size.cross(wm);
    let bfc_cross_size = constraints.available_size.cross(wm);
    let mut placement_main_offset = current_main_offset;
    loop {
        // 1. Determine the available cross-axis space at the current
        // `placement_main_offset`.
        let (available_cross_start, available_cross_end) = floating_context
            .available_line_box_space(
                placement_main_offset,
                placement_main_offset + child_main_size,
                bfc_cross_size,
                wm,
            );
        let available_cross_width = available_cross_end - available_cross_start;
        // 2. Check if the new float can fit in the available space.
        if child_cross_size <= available_cross_width {
            // It fits! Determine the final position and add it to the context.
            // +spec:floats:5cfc93 - float:right positions box at cross-end, content flows on left
            let final_cross_pos = match float_type {
                LayoutFloat::Left => available_cross_start,
                // +spec:floats:5cfc93 - float:right positions box at cross-end, content flows on left
                LayoutFloat::Right => available_cross_end - child_cross_size,
                LayoutFloat::None => {
                    return Err(LayoutError::PositioningFailed);
                }
            };
            let final_pos =
                LogicalPosition::from_main_cross(placement_main_offset, final_cross_pos, wm);
            let new_float_box = FloatBox {
                kind: float_type,
                rect: LogicalRect::new(final_pos, child_margin_box_size),
                margin: EdgeSizes::default(), // TODO: Pass actual margin if this function is used
            };
            floating_context.floats.push(new_float_box);
            return Ok(final_pos);
        } else {
            // +spec:floats:3d89d8 - shift float downward when not enough horizontal room
            // It doesn't fit. We must move the float down past an obstacle.
            // Find the lowest main-axis end of all floats that are blocking
            // the current line.
            let mut next_main_offset = f32::INFINITY;
            for existing_float in &floating_context.floats {
                let float_main_start = existing_float.rect.origin.main(wm);
                let float_main_end = float_main_start + existing_float.rect.size.main(wm);
                // Consider only floats that are above or at the current placement line.
                if placement_main_offset < float_main_end {
                    next_main_offset = next_main_offset.min(float_main_end);
                }
            }
            if next_main_offset.is_infinite() {
                // This indicates an unrecoverable state, e.g., a float wider
                // than the container.
                return Err(LayoutError::PositioningFailed);
            }
            placement_main_offset = next_main_offset;
        }
    }
}
// CSS Property Getters
/// Get the CSS `float` property for a node.
23188
fn get_float_property(styled_dom: &StyledDom, dom_id: Option<NodeId>) -> LayoutFloat {
23188
    let Some(id) = dom_id else {
        return LayoutFloat::None;
    };
23188
    let node_state = &styled_dom.styled_nodes.as_container()[id].styled_node_state;
23188
    get_float(styled_dom, id, node_state).unwrap_or(LayoutFloat::None)
23188
}
23188
fn get_clear_property(styled_dom: &StyledDom, dom_id: Option<NodeId>) -> LayoutClear {
23188
    let Some(id) = dom_id else {
        return LayoutClear::None;
    };
23188
    let node_state = &styled_dom.styled_nodes.as_container()[id].styled_node_state;
23188
    get_clear(styled_dom, id, node_state).unwrap_or(LayoutClear::None)
23188
}
/// Helper to determine if scrollbars are needed.
///
/// # CSS Spec Reference
/// CSS Overflow Module Level 3 § 3: Scrollable overflow
// +spec:block-formatting-context:50d915 - overflow-x handles horizontal, overflow-y handles vertical
// +spec:box-model:63d6f2 - scrollable overflow extends beyond padding edge, needs scroll mechanism
// +spec:box-model:45b5fb - scrollbar space subtracted from content area, inserted between inner border edge and outer padding edge
// +spec:box-model:70a0a4 - UAs must start assuming no scrollbars needed, recalculate if they are
// +spec:box-model:c1b0b2 - scrollbar gutter is space between inner border edge and outer padding edge
// +spec:overflow:4f5b99 - scrollable overflow rectangle: content_size is the minimal axis-aligned rect containing scrollable overflow
// +spec:overflow:e983f4 - overflow:auto/scroll boxes must allow user to access overflowed content via scrollbars
// +spec:overflow:97c257 - relative positioning causing overflow in auto/scroll boxes must trigger scrollbar creation
75548
pub fn check_scrollbar_necessity(
75548
    content_size: LogicalSize,
75548
    container_size: LogicalSize,
75548
    overflow_x: OverflowBehavior,
75548
    overflow_y: OverflowBehavior,
75548
    scrollbar_width_px: f32,
75548
) -> ScrollbarRequirements {
    // Use epsilon for float comparisons to avoid showing scrollbars due to 
    // floating-point rounding errors. Without this, content that exactly fits
    // may show scrollbars due to sub-pixel differences (e.g., 299.9999 vs 300.0).
    const EPSILON: f32 = 1.0;
    // +spec:height-calculation:c5af64 - assume no scrollbars initially; only add if content overflows
    // Determine if scrolling is needed based on overflow properties.
    // +spec:overflow:30a49c - start assuming no scrollbars, recalculate if needed
    // Note: scrollbar_width_px can be 0 for overlay scrollbars (e.g. macOS),
    // but we still need to register scroll nodes so that scrolling works —
    // overlay scrollbars just don't reserve any layout space.
75548
    let mut needs_horizontal = match overflow_x {
75064
        OverflowBehavior::Visible | OverflowBehavior::Hidden | OverflowBehavior::Clip => false,
44
        OverflowBehavior::Scroll => true,
440
        OverflowBehavior::Auto => content_size.width > container_size.width + EPSILON,
    };
75548
    let mut needs_vertical = match overflow_y {
75020
        OverflowBehavior::Visible | OverflowBehavior::Hidden | OverflowBehavior::Clip => false,
44
        OverflowBehavior::Scroll => true,
484
        OverflowBehavior::Auto => content_size.height > container_size.height + EPSILON,
    };
    // +spec:box-model:c3d73f - scrollbar presence affects available content area; padding preserved at scroll end
    // +spec:overflow:d79159 - scrollbar sizing: adding a scrollbar reduces available space,
    // which may cause content to overflow, confirming the scrollbar is needed (two-pass check)
    // A classic layout problem: a vertical scrollbar can reduce horizontal space,
    // causing a horizontal scrollbar to appear, which can reduce vertical space...
    // A full solution involves a loop, but this two-pass check handles most cases.
    // Only relevant when scrollbars reserve layout space (non-overlay).
75548
    if scrollbar_width_px > 0.0 {
75504
        if needs_vertical && !needs_horizontal && overflow_x == OverflowBehavior::Auto {
132
            if content_size.width > (container_size.width - scrollbar_width_px) + EPSILON {
44
                needs_horizontal = true;
88
            }
75372
        }
75504
        if needs_horizontal && !needs_vertical && overflow_y == OverflowBehavior::Auto {
132
            if content_size.height > (container_size.height - scrollbar_width_px) + EPSILON {
44
                needs_vertical = true;
88
            }
75372
        }
44
    }
    ScrollbarRequirements {
75548
        needs_horizontal,
75548
        needs_vertical,
75548
        scrollbar_width: if needs_vertical {
352
            scrollbar_width_px
        } else {
75196
            0.0
        },
75548
        scrollbar_height: if needs_horizontal {
264
            scrollbar_width_px
        } else {
75284
            0.0
        },
        // visual_width_px is set by the caller (compute_scrollbar_info_core)
        // since this function doesn't have access to the CSS style context.
        visual_width_px: 0.0,
    }
75548
}
/// Calculates a single collapsed margin from two adjoining vertical margins.
///
/// Implements the rules from CSS 2.1 section 8.3.1:
/// - If both margins are positive, the result is the larger of the two.
/// - If both margins are negative, the result is the more negative of the two.
/// - If the margins have mixed signs, they are effectively summed.
// +spec:margin-collapsing:814a26 - vertical margins between sibling blocks collapse
24728
pub fn collapse_margins(a: f32, b: f32) -> f32 {
24728
    if a.is_sign_positive() && b.is_sign_positive() {
24112
        a.max(b)
616
    } else if a.is_sign_negative() && b.is_sign_negative() {
264
        a.min(b)
    } else {
352
        a + b
    }
24728
}
/// Helper function to advance the pen position with margin collapsing.
///
/// This implements CSS 2.1 margin collapsing for adjacent block-level boxes in a BFC.
///
/// - `pen` - Current main-axis position (will be modified)
/// - `last_margin_bottom` - The bottom margin of the previous in-flow element
/// - `current_margin_top` - The top margin of the current element
///
/// # Returns
///
/// The new `last_margin_bottom` value (the bottom margin of the current element)
///
/// # CSS Spec Compliance
///
/// Per CSS 2.1 Section 8.3.1 "Collapsing margins":
///
/// - Adjacent vertical margins of block boxes collapse
/// - The resulting margin width is the maximum of the adjoining margins (if both positive)
/// - Or the sum of the most positive and most negative (if signs differ)
fn advance_pen_with_margin_collapse(
    pen: &mut f32,
    last_margin_bottom: f32,
    current_margin_top: f32,
) -> f32 {
    // Collapse the previous element's bottom margin with current element's top margin
    let collapsed_margin = collapse_margins(last_margin_bottom, current_margin_top);
    // Advance pen by the collapsed margin
    *pen += collapsed_margin;
    // Return collapsed_margin so caller knows how much space was actually added
    collapsed_margin
}
/// Checks if an element's border or padding prevents margin collapsing.
///
/// Per CSS 2.1 Section 8.3.1:
///
/// - Border between margins prevents collapsing
/// - Padding between margins prevents collapsing
///
/// # Arguments
///
/// - `box_props` - The box properties containing border and padding
/// - `writing_mode` - The writing mode to determine main axis
/// - `check_start` - If true, check main-start (top); if false, check main-end (bottom)
///
/// # Returns
///
/// `true` if border or padding exists and prevents collapsing
// +spec:box-model:ca8ceb - margin collapsing uses block-start/block-end per writing mode
138908
fn has_margin_collapse_blocker(
138908
    box_props: &crate::solver3::geometry::BoxProps,
138908
    writing_mode: LayoutWritingMode,
138908
    check_start: bool, // true = check top/start, false = check bottom/end
138908
) -> bool {
138908
    if check_start {
        // Check if there's border-top or padding-top
59444
        let border_start = box_props.border.main_start(writing_mode);
59444
        let padding_start = box_props.padding.main_start(writing_mode);
59444
        border_start > 0.0 || padding_start > 0.0
    } else {
        // Check if there's border-bottom or padding-bottom
79464
        let border_end = box_props.border.main_end(writing_mode);
79464
        let padding_end = box_props.padding.main_end(writing_mode);
79464
        border_end > 0.0 || padding_end > 0.0
    }
138908
}
/// Checks if an element is empty (has no content).
///
/// Per CSS 2.1 Section 8.3.1:
///
/// > If a block element has no border, padding, inline content, height, or min-height,
/// > then its top and bottom margins collapse with each other.
///
/// # Arguments
///
/// - `node` - The layout node to check
///
/// # Returns
///
/// `true` if the element is empty and its margins can collapse internally
22308
fn is_empty_block(tree: &LayoutTree, node_index: usize) -> bool {
22308
    let node = match tree.get(node_index) {
22308
        Some(n) => n,
        None => return true,
    };
    // Per CSS 2.2 § 8.3.1: An empty block is one that:
    // - Has zero computed 'min-height'
    // - Has zero or 'auto' computed 'height'
    // - Has no in-flow children
    // - Has no line boxes (no text/inline content)
    // Check if node has children
22308
    if !tree.children(node_index).is_empty() {
16148
        return false;
6160
    }
    // Check if node has inline content (text)
6160
    if tree.warm(node_index).and_then(|w| w.inline_layout_result.as_ref()).is_some() {
2728
        return false;
3432
    }
    // Check if node has explicit height > 0
    // CSS 2.2 § 8.3.1: Elements with explicit height are NOT empty
3432
    if let Some(size) = node.used_size {
3432
        if size.height > 0.0 {
3344
            return false;
88
        }
    }
    // Empty block: no children, no inline content, no height
88
    true
22308
}
/// Generates marker text for a list item marker.
///
/// This function looks up the counter value from the cache and formats it
/// according to the list-style-type property.
///
/// Per CSS Lists Module Level 3, the ::marker pseudo-element is the first child
/// of the list-item, and references the same DOM node. Counter resolution happens
/// on the list-item (parent) node.
fn generate_list_marker_text(
    tree: &LayoutTree,
    styled_dom: &StyledDom,
    marker_index: usize,
    counters: &HashMap<(usize, String), i32>,
    debug_messages: &mut Option<Vec<LayoutDebugMessage>>,
) -> String {
    use crate::solver3::counters::format_counter;
    // Get the marker node
    let marker_node = match tree.get(marker_index) {
        Some(n) => n,
        None => return String::new(),
    };
    // Verify this is actually a ::marker pseudo-element
    // Per spec, markers must be pseudo-elements, not anonymous boxes
    let marker_pseudo = tree.warm(marker_index).and_then(|w| w.pseudo_element);
    let marker_anonymous_type = tree.cold(marker_index).and_then(|c| c.anonymous_type);
    if marker_pseudo != Some(PseudoElement::Marker) {
        if let Some(msgs) = debug_messages {
            msgs.push(LayoutDebugMessage::warning(format!(
                "[generate_list_marker_text] WARNING: Node {} is not a ::marker pseudo-element \
                 (pseudo={:?}, anonymous_type={:?})",
                marker_index, marker_pseudo, marker_anonymous_type
            )));
        }
        // Fallback for old-style anonymous markers during transition
        if marker_anonymous_type != Some(AnonymousBoxType::ListItemMarker) {
            return String::new();
        }
    }
    // Get the parent list-item node (::marker is first child of list-item)
    let list_item_index = match marker_node.parent {
        Some(p) => p,
        None => {
            if let Some(msgs) = debug_messages {
                msgs.push(LayoutDebugMessage::error(
                    "[generate_list_marker_text] ERROR: Marker has no parent".to_string(),
                ));
            }
            return String::new();
        }
    };
    let list_item_node = match tree.get(list_item_index) {
        Some(n) => n,
        None => return String::new(),
    };
    let list_item_dom_id = match list_item_node.dom_node_id {
        Some(id) => id,
        None => {
            if let Some(msgs) = debug_messages {
                msgs.push(LayoutDebugMessage::error(
                    "[generate_list_marker_text] ERROR: List-item has no DOM ID".to_string(),
                ));
            }
            return String::new();
        }
    };
    if let Some(msgs) = debug_messages {
        msgs.push(LayoutDebugMessage::info(format!(
            "[generate_list_marker_text] marker_index={}, list_item_index={}, \
             list_item_dom_id={:?}",
            marker_index, list_item_index, list_item_dom_id
        )));
    }
    // Get list-style-type from the list-item or its container
    let list_container_dom_id = if let Some(grandparent_index) = list_item_node.parent {
        if let Some(grandparent) = tree.get(grandparent_index) {
            grandparent.dom_node_id
        } else {
            None
        }
    } else {
        None
    };
    // Try to get list-style-type from the list container first,
    // then fall back to the list-item
    let list_style_type = if let Some(container_id) = list_container_dom_id {
        let container_type = get_list_style_type(styled_dom, Some(container_id));
        if container_type != StyleListStyleType::default() {
            container_type
        } else {
            get_list_style_type(styled_dom, Some(list_item_dom_id))
        }
    } else {
        get_list_style_type(styled_dom, Some(list_item_dom_id))
    };
    // Get the counter value for "list-item" counter from the LIST-ITEM node
    // Per CSS spec, counters are scoped to elements, and the list-item counter
    // is incremented at the list-item element, not the marker pseudo-element
    let counter_value = counters
        .get(&(list_item_index, "list-item".to_string()))
        .copied()
        .unwrap_or_else(|| {
            if let Some(msgs) = debug_messages {
                msgs.push(LayoutDebugMessage::warning(format!(
                    "[generate_list_marker_text] WARNING: No counter found for list-item at index \
                     {}, defaulting to 1",
                    list_item_index
                )));
            }
            1
        });
    if let Some(msgs) = debug_messages {
        msgs.push(LayoutDebugMessage::info(format!(
            "[generate_list_marker_text] counter_value={} for list_item_index={}",
            counter_value, list_item_index
        )));
    }
    // Format the counter according to the list-style-type
    let marker_text = format_counter(counter_value, list_style_type);
    // For ordered lists (non-symbolic markers), add a period and space
    // For unordered lists (symbolic markers like •, ◦, ▪), just add a space
    if matches!(
        list_style_type,
        StyleListStyleType::Decimal
            | StyleListStyleType::DecimalLeadingZero
            | StyleListStyleType::LowerAlpha
            | StyleListStyleType::UpperAlpha
            | StyleListStyleType::LowerRoman
            | StyleListStyleType::UpperRoman
            | StyleListStyleType::LowerGreek
            | StyleListStyleType::UpperGreek
    ) {
        format!("{}. ", marker_text)
    } else {
        format!("{} ", marker_text)
    }
}
/// Generates marker text segments for a list item marker.
///
/// Simply returns a single StyledRun with the marker text using the base_style.
/// The font stack in base_style already includes fallbacks with 100% Unicode coverage,
/// so font resolution happens during text shaping, not here.
fn generate_list_marker_segments(
    tree: &LayoutTree,
    styled_dom: &StyledDom,
    marker_index: usize,
    counters: &HashMap<(usize, String), i32>,
    base_style: Arc<StyleProperties>,
    debug_messages: &mut Option<Vec<LayoutDebugMessage>>,
) -> Vec<StyledRun> {
    // Generate the marker text
    let marker_text =
        generate_list_marker_text(tree, styled_dom, marker_index, counters, debug_messages);
    if marker_text.is_empty() {
        return Vec::new();
    }
    if let Some(msgs) = debug_messages {
        let font_families: Vec<&str> = match &base_style.font_stack {
            crate::text3::cache::FontStack::Stack(selectors) => {
                selectors.iter().map(|f| f.family.as_str()).collect()
            }
            crate::text3::cache::FontStack::Ref(_) => vec!["<embedded-font>"],
        };
        msgs.push(LayoutDebugMessage::info(format!(
            "[generate_list_marker_segments] Marker text: '{}' with font stack: {:?}",
            marker_text,
            font_families
        )));
    }
    // Return single segment - font fallback happens during shaping
    // List markers are generated content, not from DOM nodes
    vec![StyledRun {
        text: marker_text,
        style: base_style,
        logical_start_byte: 0,
        source_node_id: None,
    }]
}
/// Returns true if a character has Unicode line breaking class BK (mandatory break)
/// or NL (next line). Per CSS Text 3 §5.1, these must be treated as forced line
/// breaks regardless of the white-space property value.
#[inline]
801328
fn is_bk_or_nl_class(c: char) -> bool {
801328
    matches!(c, '\u{000B}' | '\u{000C}' | '\u{0085}' | '\u{2028}' | '\u{2029}')
801328
}
/// Splits text at all forced break points: newlines (\n, \r\n, \r) and BK/NL class chars.
/// Used for white-space modes that preserve segment breaks (pre, pre-wrap, pre-line, break-spaces).
// +spec:white-space-processing:af4e3f - each newline/segment break in text is treated as a segment break, interpreted per white-space property
880
fn split_at_forced_breaks(text: &str) -> Vec<String> {
880
    let mut segments = Vec::new();
880
    let mut current = String::new();
880
    let mut chars = text.chars().peekable();
17776
    while let Some(c) = chars.next() {
16896
        if c == '\n' {
704
            segments.push(std::mem::take(&mut current));
16192
        } else if c == '\r' {
            segments.push(std::mem::take(&mut current));
            if chars.peek() == Some(&'\n') {
                chars.next();
            }
16192
        } else if is_bk_or_nl_class(c) {
            segments.push(std::mem::take(&mut current));
16192
        } else {
16192
            current.push(c);
16192
        }
    }
880
    segments.push(current);
880
    segments
880
}
/// Splits text only at BK/NL class characters (not \n which is collapsed in normal/nowrap).
/// Used for white-space: normal/nowrap where \n is collapsed to space but BK/NL chars
/// still produce forced breaks per CSS Text 3 §5.1.
89936
fn split_at_bk_nl_chars(text: &str) -> Vec<String> {
89936
    let mut segments = Vec::new();
89936
    let mut current = String::new();
785136
    for c in text.chars() {
785136
        if is_bk_or_nl_class(c) {
            segments.push(std::mem::take(&mut current));
785136
        } else {
785136
            current.push(c);
785136
        }
    }
89936
    segments.push(current);
89936
    segments
89936
}
/// Returns true if the character is East Asian (CJK) for the purposes of
/// segment break transformation rules (CSS Text Level 3, §4.1.3).
176
fn is_east_asian_wide(c: char) -> bool {
176
    let cp = c as u32;
    // CJK Unified Ideographs
176
    (0x4E00..=0x9FFF).contains(&cp)
176
    || (0x3400..=0x4DBF).contains(&cp)
176
    || (0x20000..=0x2A6DF).contains(&cp)
176
    || (0xF900..=0xFAFF).contains(&cp)
    // Hiragana
176
    || (0x3040..=0x309F).contains(&cp)
    // Katakana
176
    || (0x30A0..=0x30FF).contains(&cp)
176
    || (0x31F0..=0x31FF).contains(&cp)
    // CJK Radicals / Kangxi / Ideographic Description
176
    || (0x2E80..=0x2EFF).contains(&cp)
176
    || (0x2F00..=0x2FDF).contains(&cp)
176
    || (0x2FF0..=0x2FFF).contains(&cp)
    // CJK Symbols and Punctuation
176
    || (0x3000..=0x303F).contains(&cp)
176
    || (0x3200..=0x32FF).contains(&cp)
176
    || (0x3300..=0x33FF).contains(&cp)
    // Bopomofo
176
    || (0x3100..=0x312F).contains(&cp)
    // Hangul Syllables
176
    || (0xAC00..=0xD7AF).contains(&cp)
    // Fullwidth forms
176
    || (0xFF01..=0xFF60).contains(&cp)
176
    || (0xFFE0..=0xFFE6).contains(&cp)
176
}
// +spec:block-formatting-context:b78223 - fullwidth/wide chars treated as vertical script, halfwidth as horizontal per UAX#11
176
fn is_east_asian_fullwidth_or_wide(ch: char) -> bool {
176
    let cp = ch as u32;
    // Exclude Hangul
176
    if (0x1100..=0x11FF).contains(&cp)
176
        || (0x3130..=0x318F).contains(&cp)
176
        || (0xAC00..=0xD7AF).contains(&cp)
176
        || (0xA960..=0xA97F).contains(&cp)
176
        || (0xD7B0..=0xD7FF).contains(&cp)
    {
        return false;
176
    }
176
    is_east_asian_wide(ch)
176
        || (0xFF61..=0xFFDC).contains(&cp)
176
        || (0xFFE8..=0xFFEE).contains(&cp)
176
        || (0xA000..=0xA4CF).contains(&cp)
176
}
/// +spec:white-space-processing:159dbf - segment breaks converted to spaces (default transform)
/// +spec:white-space-processing:79891b - segment break transform: convert to space or remove
// +spec:white-space-processing:7e9529 - Segment break transformation rules (§4.1.3): collapse consecutive breaks, remove around ZWSP/CJK, else convert to space
/// Transforms segment breaks (newlines) in text according to CSS Text Level 3 §4.1.3.
/// - If adjacent to a zero-width space (U+200B), the segment break is removed.
/// - If both adjacent chars are East Asian F/W/H (not Hangul), removed entirely.
/// - Otherwise, converted to a single space.
89936
fn apply_segment_break_transform(text: &str) -> String {
89936
    let chars: Vec<char> = text.chars().collect();
89936
    let len = chars.len();
89936
    let mut result = String::with_capacity(text.len());
89936
    let mut i = 0;
874016
    while i < len {
784080
        let ch = chars[i];
784080
        if ch == '\n' || ch == '\r' {
440
            let break_end = if ch == '\r' && i + 1 < len && chars[i + 1] == '\n' {
                i + 2
            } else {
440
                i + 1
            };
            // +spec:white-space-processing:3c3680 - remove tabs/spaces around segment break before transform
            // §4.1.1: remove collapsible whitespace around segment breaks
616
            while result.ends_with(' ') || result.ends_with('\t') {
176
                result.pop();
176
            }
440
            let mut after_idx = break_end;
1496
            while after_idx < len && (chars[after_idx] == ' ' || chars[after_idx] == '\t') {
1056
                after_idx += 1;
1056
            }
440
            let char_before = result.chars().last();
440
            let char_after = if after_idx < len { Some(chars[after_idx]) } else { None };
            // Rule 1: adjacent to zero-width space → remove
440
            if char_before == Some('\u{200B}') || char_after == Some('\u{200B}') {
                // remove segment break
            }
            // Rule 2: both sides East Asian F/W/H (not Hangul) → remove
440
            else if let (Some(before), Some(after)) = (char_before, char_after) {
176
                if is_east_asian_fullwidth_or_wide(before) && is_east_asian_fullwidth_or_wide(after) {
                    // remove segment break
176
                } else {
176
                    result.push(' ');
176
                }
264
            } else {
264
                result.push(' ');
264
            }
440
            i = after_idx;
783640
        } else {
783640
            result.push(ch);
783640
            i += 1;
783640
        }
    }
89936
    result
89936
}
// ============================================================================
// WHITE-SPACE PROCESSING PIPELINE (CSS Text Level 3 §4)
// ============================================================================
//
// +spec:white-space-processing:b64e38 - parser may normalize/collapse whitespace before CSS; CSS cannot restore
// The white-space processing pipeline is organized into four phases per the
// CSS Text Level 3 specification:
//
//   Phase 1 (Collapse): Collapse whitespace sequences per §4.1.1
//   Phase 2 (Segment Break Transform): Transform segment breaks per §4.1.3
//   Phase 3 (Edge Trimming): Trim spaces at line start/end per §4.1.2
//   Phase 4 (Tab Resolution): Resolve tab stops per §4.2
//
// Each phase is a standalone function that transforms a string, allowing
// spec patches to modify individual phases without touching others.
/// Phase 1: Collapse consecutive whitespace to a single space.
/// CSS Text 3 §4.1.1 - applies to `normal`, `nowrap`, and `pre-line` modes.
pub fn ws_phase1_collapse(text: &str) -> String {
    let mut result = String::with_capacity(text.len());
    let mut prev_was_space = false;
    for ch in text.chars() {
        if ch == ' ' || ch == '\t' {
            if !prev_was_space {
                result.push(' ');
                prev_was_space = true;
            }
        } else {
            result.push(ch);
            prev_was_space = false;
        }
    }
    result
}
/// Phase 2: Transform segment breaks (newlines) per CSS Text 3 §4.1.3.
/// Delegates to `apply_segment_break_transform` for the actual transformation rules.
pub fn ws_phase2_segment_break_transform(text: &str) -> String {
    apply_segment_break_transform(text)
}
/// Phase 3: Trim leading/trailing collapsible whitespace at line boundaries.
/// CSS Text 3 §4.1.2 - this is a no-op during text collection; actual trimming
/// happens during line breaking when line start/end positions are known.
/// Provided as a pipeline slot for patches to hook into.
pub fn ws_phase3_trim_edges(text: &str) -> String {
    text.to_string()
}
/// Phase 4: Resolve tab characters to spaces based on tab-size.
/// CSS Text 3 §4.2 - for `normal`/`nowrap`, tabs are collapsed to spaces in Phase 1.
/// For `pre`/`pre-wrap`/`break-spaces`, tabs are emitted as `InlineContent::Tab`
/// and resolved during line layout. This phase is a no-op during text collection.
pub fn ws_phase4_resolve_tabs(text: &str) -> String {
    text.to_string()
}
/// Splits text content into InlineContent items based on white-space CSS property.
///
///
/// For `white-space: pre`, `pre-wrap`, and `pre-line`, newlines (`\n`) are treated as
/// forced line breaks per CSS Text Level 3 specification:
/// https://www.w3.org/TR/css-text-3/#white-space-property
///
/// Additionally, Unicode characters with BK or NL line breaking class (VT, FF, NEL, LS, PS)
/// are always treated as forced line breaks regardless of the white-space value.
///
/// This function:
/// 1. Checks the white-space property of the node (or its parent for text nodes)
/// 2. If `pre`, `pre-wrap`, or `pre-line`: splits text by `\n` and inserts `InlineContent::LineBreak`
/// 3. Otherwise: returns the text as a single `InlineContent::Text`
/// 4. In ALL modes: BK/NL class chars (VT, FF, NEL, LS, PS) produce forced breaks
///
/// Returns a Vec of InlineContent items that correctly represent line breaks.
// +spec:display-property:1389e3 - bidi control characters per UAX #9 for Unicode bidirectional algorithm
// +spec:display-property:aad99b - inline boxes can be split into fragments due to bidi text processing
// Bidi_Control property (UAX #9). These characters are ignored during white-space processing.
802032
fn is_bidi_control(c: char) -> bool {
802032
    matches!(c,
        '\u{200E}' | // LEFT-TO-RIGHT MARK
        '\u{200F}' | // RIGHT-TO-LEFT MARK
        '\u{202A}' | // LEFT-TO-RIGHT EMBEDDING
        '\u{202B}' | // RIGHT-TO-LEFT EMBEDDING
        '\u{202C}' | // POP DIRECTIONAL FORMATTING
        '\u{202D}' | // LEFT-TO-RIGHT OVERRIDE
        '\u{202E}' | // RIGHT-TO-LEFT OVERRIDE
        '\u{2066}' | // LEFT-TO-RIGHT ISOLATE
        '\u{2067}' | // RIGHT-TO-LEFT ISOLATE
        '\u{2068}' | // FIRST STRONG ISOLATE
        '\u{2069}' | // POP DIRECTIONAL ISOLATE
        '\u{061C}'   // ARABIC LETTER MARK
    )
802032
}
/// +spec:white-space-processing:1188f6 - only spaces, tabs, and segment breaks are document white space
/// Returns true if `c` is a CSS "document white space character" per CSS Text Level 3 §4.1.
/// Only spaces (U+0020), tabs (U+0009), and segment breaks (LF, CR, FF) qualify.
/// Other Unicode whitespace (e.g. U+00A0 non-breaking space) is NOT document white space.
#[inline]
966240
pub fn is_css_document_whitespace(c: char) -> bool {
966240
    matches!(c, ' ' | '\t' | '\n' | '\r' | '\x0C')
966240
}
// +spec:white-space-processing:efbece - white-space property controls collapsing/preserving of formatting characters for rendering
// +spec:writing-modes:b87688 - inlines laid out with bidi reordering and white-space wrapping
// +spec:writing-modes:cdd4f1 - white space trimming before bidi reordering preserves end-of-line spaces per UAX9 L1
// white space characters are processed prior to line breaking and bidi reordering
// +spec:inline-block:381c0c - white-space property: collapsing, wrapping, and forced breaks per mode
// +spec:display-property:8acfaa - Phase I white-space collapsing for each inline in an IFC, ignoring bidi controls
90816
pub fn split_text_for_whitespace(
90816
    styled_dom: &StyledDom,
90816
    dom_id: NodeId,
90816
    text: &str,
90816
    style: Arc<StyleProperties>,
90816
) -> Vec<InlineContent> {
    use crate::text3::cache::{BreakType, ClearType, InlineBreak};
    // (characters with the Bidi_Control property) as if they were not there"
    // Strip bidi control characters before white-space processing so they don't
    // interfere with collapsing (e.g. a bidi mark between two spaces).
    let text_owned;
802032
    let text: &str = if text.chars().any(|c| is_bidi_control(c)) {
        text_owned = text.chars().filter(|c| !is_bidi_control(*c)).collect::<String>();
        &text_owned
    } else {
90816
        text
    };
    // Get the white-space property - TEXT NODES inherit from parent!
    // We need to check the parent element's white-space, not the text node itself
90816
    let node_hierarchy = styled_dom.node_hierarchy.as_container();
90816
    let parent_id = node_hierarchy[dom_id].parent_id();
    // Try parent first, then fall back to the node itself
90816
    let white_space = if let Some(parent) = parent_id {
90816
        let styled_nodes = styled_dom.styled_nodes.as_container();
90816
        let parent_state = styled_nodes
90816
            .get(parent)
90816
            .map(|n| n.styled_node_state.clone())
90816
            .unwrap_or_default();
90816
        match get_white_space_property(styled_dom, parent, &parent_state) {
90816
            MultiValue::Exact(ws) => ws,
            _ => StyleWhiteSpace::Normal,
        }
    } else {
        StyleWhiteSpace::Normal
    };
90816
    let mut result = Vec::new();
    // +spec:white-space-processing:3a0f58 - HTML newlines normalized to U+000A, each treated as segment break
    // +spec:white-space-processing:6eb1a2 - CR (U+000D) not treated as segment break by HTML; handle if inserted via DOM
    // HTML parsers convert \r to \n during preprocessing, but \r can survive
    // via escape sequences (e.g. &#x0d;). Any remaining U+000D must be
    // treated identically to U+000A (line feed).
    let text_cr;
90816
    let text: &str = if text.contains('\r') {
        text_cr = text.replace("\r\n", "\n").replace('\r', "\n");
        &text_cr
    } else {
90816
        text
    };
    // +spec:white-space-processing:bd11da - white-space property: new lines, spaces/tabs, wrapping per value table
    // +spec:white-space-processing:b166c5 - segment breaks preserved as forced line feeds for pre/pre-wrap/break-spaces/pre-line
    // For `pre`, `pre-wrap`, `pre-line`, and `break-spaces`, newlines must be preserved as forced breaks
    // CSS Text Level 3: "Newlines in the source will be honored as forced line breaks."
90816
    match white_space {
        StyleWhiteSpace::Pre | StyleWhiteSpace::PreWrap | StyleWhiteSpace::BreakSpaces => {
            // Pre, pre-wrap, break-spaces: preserve whitespace and honor newlines
            // Split by newlines and BK/NL class chars, insert LineBreak between parts
            // Also handle tab characters (\t) by inserting InlineContent::Tab
704
            let segments = split_at_forced_breaks(text);
704
            let segment_count = segments.len();
704
            let mut content_index = 0;
1144
            for (seg_idx, segment) in segments.into_iter().enumerate() {
                // Split the segment by tab characters and insert Tab elements
1144
                let mut tab_parts = segment.split('\t').peekable();
2464
                while let Some(part) = tab_parts.next() {
1320
                    if !part.is_empty() {
1144
                        result.push(InlineContent::Text(StyledRun {
1144
                            text: part.to_string(),
1144
                            style: Arc::clone(&style),
1144
                            logical_start_byte: 0,
1144
                            source_node_id: Some(dom_id),
1144
                        }));
1144
                    }
1320
                    if tab_parts.peek().is_some() {
176
                        result.push(InlineContent::Tab { style: Arc::clone(&style) });
1144
                    }
                }
1144
                if seg_idx + 1 < segment_count {
440
                    result.push(InlineContent::LineBreak(InlineBreak {
440
                        break_type: BreakType::Hard,
440
                        clear: ClearType::None,
440
                        content_index,
440
                    }));
440
                    content_index += 1;
704
                }
            }
        }
        StyleWhiteSpace::PreLine => {
            // Pre-line: collapse whitespace but honor newlines and BK/NL class chars
176
            let segments = split_at_forced_breaks(text);
176
            let segment_count = segments.len();
176
            let mut content_index = 0;
440
            for (seg_idx, segment) in segments.into_iter().enumerate() {
                // Collapse only CSS document white space within the line (not all Unicode whitespace)
440
                let collapsed: String = segment
2992
                    .split(|c: char| is_css_document_whitespace(c))
968
                    .filter(|s| !s.is_empty())
440
                    .collect::<Vec<_>>()
440
                    .join(" ");
440
                if !collapsed.is_empty() {
352
                    result.push(InlineContent::Text(StyledRun {
352
                        text: collapsed,
352
                        style: Arc::clone(&style),
352
                        logical_start_byte: 0,
352
                        source_node_id: Some(dom_id),
352
                    }));
352
                }
440
                if seg_idx + 1 < segment_count {
264
                    result.push(InlineContent::LineBreak(InlineBreak {
264
                        break_type: BreakType::Hard,
264
                        clear: ClearType::None,
264
                        content_index,
264
                    }));
264
                    content_index += 1;
264
                }
            }
        }
        StyleWhiteSpace::Normal | StyleWhiteSpace::Nowrap => {
            // +spec:white-space-processing:adbebb - Phase I collapsing for normal/nowrap modes
            // CSS Text Level 3, Section 4.1.1 - Phase I: Collapsing and Transformation
            // https://www.w3.org/TR/css-text-3/#white-space-phase-1
            //
            // For `white-space: normal` and `nowrap`:
            // 1. Segment breaks are transformed per §4.1.3
            // 2. Any sequence of consecutive spaces/tabs is collapsed to a single space
            // 3. Leading/trailing spaces at line boundaries are handled during line layout
            //
            // are forced breaks regardless of white-space value. Split on them first,
            // then collapse whitespace within each segment.
89936
            let segments = split_at_bk_nl_chars(text);
89936
            let segment_count = segments.len();
89936
            let mut content_index = 0;
89936
            for (seg_idx, segment) in segments.into_iter().enumerate() {
89936
                let after_segment_breaks = apply_segment_break_transform(&segment);
                // Collapse document white space within this segment (normal/nowrap rules)
89936
                let collapsed: String = after_segment_breaks
89936
                    .chars()
783904
                    .map(|c| if is_css_document_whitespace(c) { ' ' } else { c })
89936
                    .collect::<String>()
89936
                    .split(' ')
173140
                    .filter(|s| !s.is_empty())
89936
                    .collect::<Vec<_>>()
89936
                    .join(" ");
89936
                let final_text = if collapsed.is_empty() && !segment.is_empty() {
264
                    " ".to_string()
89672
                } else if !collapsed.is_empty() {
                    // Check if original had leading/trailing document whitespace
89672
                    let had_leading = segment.chars().next().map(|c| is_css_document_whitespace(c)).unwrap_or(false);
89672
                    let had_trailing = segment.chars().last().map(|c| is_css_document_whitespace(c)).unwrap_or(false);
89672
                    let mut r = String::new();
89672
                    if had_leading { r.push(' '); }
89672
                    r.push_str(&collapsed);
89672
                    if had_trailing && !had_leading { r.push(' '); }
89672
                    else if had_trailing && had_leading && collapsed.is_empty() { /* already have one space */ }
89672
                    else if had_trailing { r.push(' '); }
89672
                    r
                } else {
                    collapsed
                };
89936
                if !final_text.is_empty() {
89936
                    result.push(InlineContent::Text(StyledRun {
89936
                        text: final_text,
89936
                        style: Arc::clone(&style),
89936
                        logical_start_byte: 0,
89936
                        source_node_id: Some(dom_id),
89936
                    }));
89936
                }
                // Insert forced break between segments (for BK/NL chars)
89936
                if seg_idx + 1 < segment_count {
                    result.push(InlineContent::LineBreak(InlineBreak {
                        break_type: BreakType::Hard,
                        clear: ClearType::None,
                        content_index,
                    }));
                    content_index += 1;
89936
                }
            }
        }
    }
    // +spec:white-space-processing:5e3f70 - text-transform applied after Phase I collapsing, before Phase II trimming
    // This means full-width only transforms spaces (U+0020) to U+3000 IDEOGRAPHIC SPACE
    // within preserved white space, because non-preserved spaces were already collapsed in Phase I above.
90816
    let text_transform = style.text_transform;
90816
    if text_transform != crate::text3::cache::TextTransform::None {
        for item in result.iter_mut() {
            if let InlineContent::Text(run) = item {
                run.text = apply_text_transform(&run.text, text_transform);
            }
        }
90816
    }
90816
    result
90816
}
fn apply_text_transform(text: &str, transform: crate::text3::cache::TextTransform) -> String {
    use crate::text3::cache::TextTransform;
    match transform {
        TextTransform::None => text.to_string(),
        TextTransform::Uppercase => text.to_uppercase(),
        TextTransform::Lowercase => text.to_lowercase(),
        TextTransform::Capitalize => {
            let mut result = String::with_capacity(text.len());
            let mut prev_is_word_boundary = true;
            for c in text.chars() {
                if prev_is_word_boundary && c.is_alphabetic() {
                    for uc in c.to_uppercase() {
                        result.push(uc);
                    }
                    prev_is_word_boundary = false;
                } else {
                    result.push(c);
                    prev_is_word_boundary = c.is_whitespace() || c.is_ascii_punctuation();
                }
            }
            result
        }
        TextTransform::FullWidth => {
            // Full-width transforms ASCII characters to their full-width equivalents.
            // Spaces (U+0020) become U+3000 IDEOGRAPHIC SPACE — but only those that
            // survived Phase I collapsing (i.e. preserved white space).
            text.chars().map(|c| match c {
                ' ' => '\u{3000}',  // U+0020 SPACE -> U+3000 IDEOGRAPHIC SPACE
                '!' ..= '~' => {
                    // ASCII printable range U+0021..U+007E -> fullwidth U+FF01..U+FF5E
                    char::from_u32(c as u32 - 0x0021 + 0xFF01).unwrap_or(c)
                }
                _ => c,
            }).collect()
        }
    }
}
// ============================================================================
// INITIAL LETTER / DROP CAPS STUB
// ============================================================================
/// Computes the geometric exclusion area for an initial letter (drop cap).
///
/// CSS Inline Layout Module Level 3, section 3:
/// The `initial-letter` property specifies styling for dropped, raised, and sunken
/// initial letters. When set, the first glyph(s) of the first line are enlarged to
/// span multiple lines, with the remaining text wrapping around them.
///
// +spec:box-model:c93797 - initial-letter alignment points determined from contents (not border-box)
///
/// # Algorithm
///
/// 1. The letter box height spans `size` lines: `height = size * line_height`.
/// 2. The letter box width is estimated using a typical capital letter aspect ratio
///    (cap-height-to-advance-width ~0.7 for Latin text). A proper implementation
///    would measure the actual glyph, but this gives a reasonable default.
/// 3. The letter is positioned at the inline-start of the first line.
/// 4. The `sink` value determines how many lines the letter drops below the
///    first baseline. When `sink == size`, this is a classic drop cap.
///    When `sink < size`, the letter rises above the first line (raised cap).
/// 5. A small gap (4px default) is added between the letter box and adjacent text.
///
/// # Parameters
/// - `initial_letter_size`: The number of lines the initial letter should span (e.g., 3.0)
/// - `initial_letter_sink`: How many lines the letter sinks below the first line
/// - `content_box_width`: Available width in the content box (for clamping)
/// - `line_height`: The computed line height for the containing block
///
/// # Returns
/// A tuple of `(letter_width, letter_height)` representing the space reserved for
/// the initial letter exclusion, or `(0.0, 0.0)` if the parameters are invalid.
///
/// The caller should use these dimensions to create a float-like exclusion at the
/// start of the block container, causing subsequent lines to wrap around the letter.
// +spec:width-calculation:7f4f68 - initial-letter-wrap exclusion area (none behavior; first/grid require glyph outlines)
pub fn layout_initial_letter(
    initial_letter_size: f32,
    initial_letter_sink: u32,
    content_box_width: f32,
    line_height: f32,
) -> (f32, f32) {
    // Guard against degenerate values
    if initial_letter_size <= 0.0 || line_height <= 0.0 || content_box_width <= 0.0 {
        return (0.0, 0.0);
    }
    // +spec:overflow:dd0679 - auto-sized initial letter content box fits exactly to content; alignment props do not apply
    // +spec:width-calculation:170742 - atomic initial letters with auto block size use inline initial letter sizing
    // CSS Inline Level 3 section 3.3: The initial letter box height spans `size` lines.
    let letter_height = initial_letter_size * line_height;
    // Estimate the letter width using a typical Latin capital letter aspect ratio.
    // The advance width of a capital letter is approximately 0.7x the cap height.
    // This is a heuristic; a full implementation would measure the actual glyph(s).
    const CAP_WIDTH_RATIO: f32 = 0.7;
    let letter_width_raw = letter_height * CAP_WIDTH_RATIO;
    // Add a small gap between the letter box and the adjacent inline content.
    // CSS Inline Level 3 section 3.5: browsers typically add ~4px padding.
    const LETTER_GAP: f32 = 4.0;
    let letter_width = (letter_width_raw + LETTER_GAP).min(content_box_width);
    // +spec:containing-block:67fd99 - block-axis positioning: size >= sink shifts by (sink-1)*line_height toward block-end
    // The actual exclusion height accounts for the sink value.
    // sink == size means the letter is fully dropped (classic drop cap).
    // sink < size means part of the letter rises above the first line (raised cap).
    // The exclusion area height is always `sink * line_height` since that's how
    // many lines of subsequent text need to wrap around the letter.
    let exclusion_height = (initial_letter_sink as f32) * line_height;
    // Use the larger of exclusion_height and letter_height as the actual
    // vertical space consumed. For raised caps (sink < size), the letter
    // extends above the first line but the exclusion only covers sink lines.
    // For sunken caps (sink >= size), the exclusion covers the full letter height.
    let effective_height = exclusion_height.max(letter_height);
    (letter_width, effective_height)
}