use crate::props::basic::length::{FloatValue, SizeMetric};
use crate::props::basic::pixel::PixelValue;
use crate::props::layout::{
display::LayoutDisplay,
dimensions::{LayoutHeight, LayoutWidth, LayoutMaxHeight, LayoutMaxWidth, LayoutMinHeight, LayoutMinWidth},
flex::{
LayoutAlignContent, LayoutAlignItems, LayoutAlignSelf, LayoutFlexDirection, LayoutFlexWrap,
LayoutJustifyContent,
},
grid::{LayoutGridAutoFlow, LayoutJustifySelf, LayoutJustifyItems},
overflow::LayoutOverflow,
position::LayoutPosition,
wrapping::{LayoutClear, LayoutWritingMode},
table::StyleBorderCollapse,
};
use crate::props::layout::display::LayoutFloat;
use crate::props::layout::dimensions::LayoutBoxSizing;
use crate::props::basic::font::{StyleFontStyle, StyleFontWeight};
use crate::props::basic::color::ColorU;
use crate::props::style::{StyleTextAlign, StyleVerticalAlign, StyleVisibility, StyleWhiteSpace, StyleDirection};
use crate::props::style::border::BorderStyle;
use crate::props::property::{CssProperty, CssPropertyType};
use crate::css::CssPropertyValue;
use alloc::boxed::Box;
use alloc::vec::Vec;
pub const U16_SENTINEL: u16 = 0xFFFF;
pub const U16_SENTINEL_THRESHOLD: u16 = 0xFFF9;
pub const I16_SENTINEL: i16 = 0x7FFF; pub const I16_AUTO: i16 = 0x7FFE; pub const I16_INHERIT: i16 = 0x7FFD; pub const I16_INITIAL: i16 = 0x7FFC; pub const I16_SENTINEL_THRESHOLD: i16 = 0x7FFC;
pub const U32_SENTINEL: u32 = 0xFFFFFFFF;
pub const U32_AUTO: u32 = 0xFFFFFFFE;
pub const U32_NONE: u32 = 0xFFFFFFFD;
pub const U32_INHERIT: u32 = 0xFFFFFFFC;
pub const U32_INITIAL: u32 = 0xFFFFFFFB;
pub const U32_MIN_CONTENT: u32 = 0xFFFFFFFA;
pub const U32_MAX_CONTENT: u32 = 0xFFFFFFF9;
pub const U32_SENTINEL_THRESHOLD: u32 = 0xFFFFFFF9;
pub const DISPLAY_SHIFT: u32 = 0;
pub const POSITION_SHIFT: u32 = 5;
pub const FLOAT_SHIFT: u32 = 8;
pub const OVERFLOW_X_SHIFT: u32 = 10;
pub const OVERFLOW_Y_SHIFT: u32 = 13;
pub const BOX_SIZING_SHIFT: u32 = 16;
pub const FLEX_DIRECTION_SHIFT: u32 = 17;
pub const FLEX_WRAP_SHIFT: u32 = 19;
pub const JUSTIFY_CONTENT_SHIFT: u32 = 21;
pub const ALIGN_ITEMS_SHIFT: u32 = 24;
pub const ALIGN_CONTENT_SHIFT: u32 = 27;
pub const WRITING_MODE_SHIFT: u32 = 30;
pub const CLEAR_SHIFT: u32 = 32;
pub const FONT_WEIGHT_SHIFT: u32 = 34;
pub const FONT_STYLE_SHIFT: u32 = 38;
pub const TEXT_ALIGN_SHIFT: u32 = 40;
pub const VISIBILITY_SHIFT: u32 = 43;
pub const WHITE_SPACE_SHIFT: u32 = 45;
pub const DIRECTION_SHIFT: u32 = 48;
pub const VERTICAL_ALIGN_SHIFT: u32 = 49;
pub const BORDER_COLLAPSE_SHIFT: u32 = 52;
pub const DISPLAY_MASK: u64 = 0x1F; pub const POSITION_MASK: u64 = 0x07; pub const FLOAT_MASK: u64 = 0x03; pub const OVERFLOW_MASK: u64 = 0x07; pub const BOX_SIZING_MASK: u64 = 0x01; pub const FLEX_DIR_MASK: u64 = 0x03; pub const FLEX_WRAP_MASK: u64 = 0x03; pub const JUSTIFY_MASK: u64 = 0x07; pub const ALIGN_MASK: u64 = 0x07; pub const WRITING_MODE_MASK: u64 = 0x03;pub const CLEAR_MASK: u64 = 0x03; pub const FONT_WEIGHT_MASK: u64 = 0x0F; pub const FONT_STYLE_MASK: u64 = 0x03; pub const TEXT_ALIGN_MASK: u64 = 0x07; pub const VISIBILITY_MASK: u64 = 0x03; pub const WHITE_SPACE_MASK: u64 = 0x07; pub const DIRECTION_MASK: u64 = 0x01; pub const VERTICAL_ALIGN_MASK: u64 = 0x07; pub const BORDER_COLLAPSE_MASK: u64 = 0x01;
pub const ALIGN_SELF_SHIFT: u32 = 53;
pub const JUSTIFY_SELF_SHIFT: u32 = 56;
pub const GRID_AUTO_FLOW_SHIFT: u32 = 59;
pub const JUSTIFY_ITEMS_SHIFT: u32 = 61;
pub const ALIGN_SELF_MASK: u64 = 0x07; pub const JUSTIFY_SELF_MASK: u64 = 0x07; pub const GRID_AUTO_FLOW_MASK: u64 = 0x03; pub const JUSTIFY_ITEMS_MASK: u64 = 0x03;
pub const TIER1_POPULATED_BIT: u64 = 1 << 63;
#[inline(always)]
pub fn layout_display_from_u8(v: u8) -> LayoutDisplay {
match v {
0 => LayoutDisplay::Block, 1 => LayoutDisplay::Inline,
2 => LayoutDisplay::InlineBlock,
3 => LayoutDisplay::Flex,
4 => LayoutDisplay::None,
5 => LayoutDisplay::InlineFlex,
6 => LayoutDisplay::Table,
7 => LayoutDisplay::InlineTable,
8 => LayoutDisplay::TableRowGroup,
9 => LayoutDisplay::TableHeaderGroup,
10 => LayoutDisplay::TableFooterGroup,
11 => LayoutDisplay::TableRow,
12 => LayoutDisplay::TableColumnGroup,
13 => LayoutDisplay::TableColumn,
14 => LayoutDisplay::TableCell,
15 => LayoutDisplay::TableCaption,
16 => LayoutDisplay::FlowRoot,
17 => LayoutDisplay::ListItem,
18 => LayoutDisplay::RunIn,
19 => LayoutDisplay::Marker,
20 => LayoutDisplay::Grid,
21 => LayoutDisplay::InlineGrid,
22 => LayoutDisplay::Contents,
_ => LayoutDisplay::Block, }
}
#[inline(always)]
pub fn layout_display_to_u8(v: LayoutDisplay) -> u8 {
match v {
LayoutDisplay::Block => 0, LayoutDisplay::Inline => 1,
LayoutDisplay::InlineBlock => 2,
LayoutDisplay::Flex => 3,
LayoutDisplay::None => 4,
LayoutDisplay::InlineFlex => 5,
LayoutDisplay::Table => 6,
LayoutDisplay::InlineTable => 7,
LayoutDisplay::TableRowGroup => 8,
LayoutDisplay::TableHeaderGroup => 9,
LayoutDisplay::TableFooterGroup => 10,
LayoutDisplay::TableRow => 11,
LayoutDisplay::TableColumnGroup => 12,
LayoutDisplay::TableColumn => 13,
LayoutDisplay::TableCell => 14,
LayoutDisplay::TableCaption => 15,
LayoutDisplay::FlowRoot => 16,
LayoutDisplay::ListItem => 17,
LayoutDisplay::RunIn => 18,
LayoutDisplay::Marker => 19,
LayoutDisplay::Grid => 20,
LayoutDisplay::InlineGrid => 21,
LayoutDisplay::Contents => 22,
}
}
#[inline(always)]
pub fn layout_position_from_u8(v: u8) -> LayoutPosition {
match v {
0 => LayoutPosition::Static,
1 => LayoutPosition::Relative,
2 => LayoutPosition::Absolute,
3 => LayoutPosition::Fixed,
4 => LayoutPosition::Sticky,
_ => LayoutPosition::Static,
}
}
#[inline(always)]
pub fn layout_position_to_u8(v: LayoutPosition) -> u8 {
match v {
LayoutPosition::Static => 0,
LayoutPosition::Relative => 1,
LayoutPosition::Absolute => 2,
LayoutPosition::Fixed => 3,
LayoutPosition::Sticky => 4,
}
}
#[inline(always)]
pub fn layout_float_from_u8(v: u8) -> LayoutFloat {
match v {
0 => LayoutFloat::None, 1 => LayoutFloat::Left,
2 => LayoutFloat::Right,
_ => LayoutFloat::None,
}
}
#[inline(always)]
pub fn layout_float_to_u8(v: LayoutFloat) -> u8 {
match v {
LayoutFloat::None => 0,
LayoutFloat::Left => 1,
LayoutFloat::Right => 2,
}
}
#[inline(always)]
pub fn layout_overflow_from_u8(v: u8) -> LayoutOverflow {
match v {
0 => LayoutOverflow::Visible, 1 => LayoutOverflow::Hidden,
2 => LayoutOverflow::Scroll,
3 => LayoutOverflow::Auto,
4 => LayoutOverflow::Clip,
_ => LayoutOverflow::Visible,
}
}
#[inline(always)]
pub fn layout_overflow_to_u8(v: LayoutOverflow) -> u8 {
match v {
LayoutOverflow::Visible => 0, LayoutOverflow::Hidden => 1,
LayoutOverflow::Scroll => 2,
LayoutOverflow::Auto => 3,
LayoutOverflow::Clip => 4,
}
}
#[inline(always)]
pub fn layout_box_sizing_from_u8(v: u8) -> LayoutBoxSizing {
match v {
0 => LayoutBoxSizing::ContentBox,
1 => LayoutBoxSizing::BorderBox,
_ => LayoutBoxSizing::ContentBox,
}
}
#[inline(always)]
pub fn layout_box_sizing_to_u8(v: LayoutBoxSizing) -> u8 {
match v {
LayoutBoxSizing::ContentBox => 0,
LayoutBoxSizing::BorderBox => 1,
}
}
#[inline(always)]
pub fn layout_flex_direction_from_u8(v: u8) -> LayoutFlexDirection {
match v {
0 => LayoutFlexDirection::Row,
1 => LayoutFlexDirection::RowReverse,
2 => LayoutFlexDirection::Column,
3 => LayoutFlexDirection::ColumnReverse,
_ => LayoutFlexDirection::Row,
}
}
#[inline(always)]
pub fn layout_flex_direction_to_u8(v: LayoutFlexDirection) -> u8 {
match v {
LayoutFlexDirection::Row => 0,
LayoutFlexDirection::RowReverse => 1,
LayoutFlexDirection::Column => 2,
LayoutFlexDirection::ColumnReverse => 3,
}
}
#[inline(always)]
pub fn layout_flex_wrap_from_u8(v: u8) -> LayoutFlexWrap {
match v {
0 => LayoutFlexWrap::NoWrap, 1 => LayoutFlexWrap::Wrap,
2 => LayoutFlexWrap::WrapReverse,
_ => LayoutFlexWrap::NoWrap,
}
}
#[inline(always)]
pub fn layout_flex_wrap_to_u8(v: LayoutFlexWrap) -> u8 {
match v {
LayoutFlexWrap::NoWrap => 0,
LayoutFlexWrap::Wrap => 1,
LayoutFlexWrap::WrapReverse => 2,
}
}
#[inline(always)]
pub fn layout_justify_content_from_u8(v: u8) -> LayoutJustifyContent {
match v {
0 => LayoutJustifyContent::FlexStart,
1 => LayoutJustifyContent::FlexEnd,
2 => LayoutJustifyContent::Start,
3 => LayoutJustifyContent::End,
4 => LayoutJustifyContent::Center,
5 => LayoutJustifyContent::SpaceBetween,
6 => LayoutJustifyContent::SpaceAround,
7 => LayoutJustifyContent::SpaceEvenly,
_ => LayoutJustifyContent::FlexStart,
}
}
#[inline(always)]
pub fn layout_justify_content_to_u8(v: LayoutJustifyContent) -> u8 {
match v {
LayoutJustifyContent::FlexStart => 0,
LayoutJustifyContent::FlexEnd => 1,
LayoutJustifyContent::Start => 2,
LayoutJustifyContent::End => 3,
LayoutJustifyContent::Center => 4,
LayoutJustifyContent::SpaceBetween => 5,
LayoutJustifyContent::SpaceAround => 6,
LayoutJustifyContent::SpaceEvenly => 7,
}
}
#[inline(always)]
pub fn layout_align_items_from_u8(v: u8) -> LayoutAlignItems {
match v {
0 => LayoutAlignItems::Stretch,
1 => LayoutAlignItems::Center,
2 => LayoutAlignItems::Start,
3 => LayoutAlignItems::End,
4 => LayoutAlignItems::Baseline,
_ => LayoutAlignItems::Stretch,
}
}
#[inline(always)]
pub fn layout_align_items_to_u8(v: LayoutAlignItems) -> u8 {
match v {
LayoutAlignItems::Stretch => 0,
LayoutAlignItems::Center => 1,
LayoutAlignItems::Start => 2,
LayoutAlignItems::End => 3,
LayoutAlignItems::Baseline => 4,
}
}
#[inline(always)]
pub fn layout_align_self_to_u8(v: LayoutAlignSelf) -> u8 {
match v {
LayoutAlignSelf::Auto => 0,
LayoutAlignSelf::Stretch => 1,
LayoutAlignSelf::Center => 2,
LayoutAlignSelf::Start => 3,
LayoutAlignSelf::End => 4,
LayoutAlignSelf::Baseline => 5,
}
}
#[inline(always)]
pub fn layout_align_self_from_u8(v: u8) -> LayoutAlignSelf {
match v {
0 => LayoutAlignSelf::Auto,
1 => LayoutAlignSelf::Stretch,
2 => LayoutAlignSelf::Center,
3 => LayoutAlignSelf::Start,
4 => LayoutAlignSelf::End,
5 => LayoutAlignSelf::Baseline,
_ => LayoutAlignSelf::Auto,
}
}
#[inline(always)]
pub fn layout_justify_self_to_u8(v: LayoutJustifySelf) -> u8 {
match v {
LayoutJustifySelf::Auto => 0,
LayoutJustifySelf::Start => 1,
LayoutJustifySelf::End => 2,
LayoutJustifySelf::Center => 3,
LayoutJustifySelf::Stretch => 4,
}
}
#[inline(always)]
pub fn layout_justify_self_from_u8(v: u8) -> LayoutJustifySelf {
match v {
0 => LayoutJustifySelf::Auto,
1 => LayoutJustifySelf::Start,
2 => LayoutJustifySelf::End,
3 => LayoutJustifySelf::Center,
4 => LayoutJustifySelf::Stretch,
_ => LayoutJustifySelf::Auto,
}
}
#[inline(always)]
pub fn layout_justify_items_to_u8(v: LayoutJustifyItems) -> u8 {
match v {
LayoutJustifyItems::Stretch => 0,
LayoutJustifyItems::Start => 1,
LayoutJustifyItems::End => 2,
LayoutJustifyItems::Center => 3,
}
}
#[inline(always)]
pub fn layout_justify_items_from_u8(v: u8) -> LayoutJustifyItems {
match v {
0 => LayoutJustifyItems::Stretch,
1 => LayoutJustifyItems::Start,
2 => LayoutJustifyItems::End,
3 => LayoutJustifyItems::Center,
_ => LayoutJustifyItems::Stretch,
}
}
#[inline(always)]
pub fn layout_grid_auto_flow_to_u8(v: LayoutGridAutoFlow) -> u8 {
match v {
LayoutGridAutoFlow::Row => 0,
LayoutGridAutoFlow::Column => 1,
LayoutGridAutoFlow::RowDense => 2,
LayoutGridAutoFlow::ColumnDense => 3,
}
}
#[inline(always)]
pub fn layout_grid_auto_flow_from_u8(v: u8) -> LayoutGridAutoFlow {
match v {
0 => LayoutGridAutoFlow::Row,
1 => LayoutGridAutoFlow::Column,
2 => LayoutGridAutoFlow::RowDense,
3 => LayoutGridAutoFlow::ColumnDense,
_ => LayoutGridAutoFlow::Row,
}
}
#[inline(always)]
pub fn layout_align_content_from_u8(v: u8) -> LayoutAlignContent {
match v {
0 => LayoutAlignContent::Stretch,
1 => LayoutAlignContent::Center,
2 => LayoutAlignContent::Start,
3 => LayoutAlignContent::End,
4 => LayoutAlignContent::SpaceBetween,
5 => LayoutAlignContent::SpaceAround,
_ => LayoutAlignContent::Stretch,
}
}
#[inline(always)]
pub fn layout_align_content_to_u8(v: LayoutAlignContent) -> u8 {
match v {
LayoutAlignContent::Stretch => 0,
LayoutAlignContent::Center => 1,
LayoutAlignContent::Start => 2,
LayoutAlignContent::End => 3,
LayoutAlignContent::SpaceBetween => 4,
LayoutAlignContent::SpaceAround => 5,
}
}
#[inline(always)]
pub fn layout_writing_mode_from_u8(v: u8) -> LayoutWritingMode {
match v {
0 => LayoutWritingMode::HorizontalTb,
1 => LayoutWritingMode::VerticalRl,
2 => LayoutWritingMode::VerticalLr,
_ => LayoutWritingMode::HorizontalTb,
}
}
#[inline(always)]
pub fn layout_writing_mode_to_u8(v: LayoutWritingMode) -> u8 {
match v {
LayoutWritingMode::HorizontalTb => 0,
LayoutWritingMode::VerticalRl => 1,
LayoutWritingMode::VerticalLr => 2,
}
}
#[inline(always)]
pub fn layout_clear_from_u8(v: u8) -> LayoutClear {
match v {
0 => LayoutClear::None,
1 => LayoutClear::Left,
2 => LayoutClear::Right,
3 => LayoutClear::Both,
_ => LayoutClear::None,
}
}
#[inline(always)]
pub fn layout_clear_to_u8(v: LayoutClear) -> u8 {
match v {
LayoutClear::None => 0,
LayoutClear::Left => 1,
LayoutClear::Right => 2,
LayoutClear::Both => 3,
}
}
#[inline(always)]
pub fn style_font_weight_from_u8(v: u8) -> StyleFontWeight {
match v {
0 => StyleFontWeight::Normal, 1 => StyleFontWeight::W100,
2 => StyleFontWeight::W200,
3 => StyleFontWeight::W300,
4 => StyleFontWeight::W500,
5 => StyleFontWeight::W600,
6 => StyleFontWeight::Bold, 7 => StyleFontWeight::W800,
8 => StyleFontWeight::W900,
9 => StyleFontWeight::Lighter,
10 => StyleFontWeight::Bolder,
_ => StyleFontWeight::Normal,
}
}
#[inline(always)]
pub fn style_font_weight_to_u8(v: StyleFontWeight) -> u8 {
match v {
StyleFontWeight::Normal => 0, StyleFontWeight::W100 => 1,
StyleFontWeight::W200 => 2,
StyleFontWeight::W300 => 3,
StyleFontWeight::W500 => 4,
StyleFontWeight::W600 => 5,
StyleFontWeight::Bold => 6, StyleFontWeight::W800 => 7,
StyleFontWeight::W900 => 8,
StyleFontWeight::Lighter => 9,
StyleFontWeight::Bolder => 10,
}
}
#[inline(always)]
pub fn style_font_style_from_u8(v: u8) -> StyleFontStyle {
match v {
0 => StyleFontStyle::Normal,
1 => StyleFontStyle::Italic,
2 => StyleFontStyle::Oblique,
_ => StyleFontStyle::Normal,
}
}
#[inline(always)]
pub fn style_font_style_to_u8(v: StyleFontStyle) -> u8 {
match v {
StyleFontStyle::Normal => 0,
StyleFontStyle::Italic => 1,
StyleFontStyle::Oblique => 2,
}
}
#[inline(always)]
pub fn style_text_align_from_u8(v: u8) -> StyleTextAlign {
match v {
0 => StyleTextAlign::Left,
1 => StyleTextAlign::Center,
2 => StyleTextAlign::Right,
3 => StyleTextAlign::Justify,
4 => StyleTextAlign::Start,
5 => StyleTextAlign::End,
_ => StyleTextAlign::Left,
}
}
#[inline(always)]
pub fn style_text_align_to_u8(v: StyleTextAlign) -> u8 {
match v {
StyleTextAlign::Left => 0,
StyleTextAlign::Center => 1,
StyleTextAlign::Right => 2,
StyleTextAlign::Justify => 3,
StyleTextAlign::Start => 4,
StyleTextAlign::End => 5,
}
}
#[inline(always)]
pub fn style_visibility_from_u8(v: u8) -> StyleVisibility {
match v {
0 => StyleVisibility::Visible,
1 => StyleVisibility::Hidden,
2 => StyleVisibility::Collapse,
_ => StyleVisibility::Visible,
}
}
#[inline(always)]
pub fn style_visibility_to_u8(v: StyleVisibility) -> u8 {
match v {
StyleVisibility::Visible => 0,
StyleVisibility::Hidden => 1,
StyleVisibility::Collapse => 2,
}
}
#[inline(always)]
pub fn style_white_space_from_u8(v: u8) -> StyleWhiteSpace {
match v {
0 => StyleWhiteSpace::Normal,
1 => StyleWhiteSpace::Pre,
2 => StyleWhiteSpace::Nowrap,
3 => StyleWhiteSpace::PreWrap,
4 => StyleWhiteSpace::PreLine,
5 => StyleWhiteSpace::BreakSpaces,
_ => StyleWhiteSpace::Normal,
}
}
#[inline(always)]
pub fn style_white_space_to_u8(v: StyleWhiteSpace) -> u8 {
match v {
StyleWhiteSpace::Normal => 0,
StyleWhiteSpace::Pre => 1,
StyleWhiteSpace::Nowrap => 2,
StyleWhiteSpace::PreWrap => 3,
StyleWhiteSpace::PreLine => 4,
StyleWhiteSpace::BreakSpaces => 5,
}
}
#[inline(always)]
pub fn style_direction_from_u8(v: u8) -> StyleDirection {
match v {
0 => StyleDirection::Ltr,
1 => StyleDirection::Rtl,
_ => StyleDirection::Ltr,
}
}
#[inline(always)]
pub fn style_direction_to_u8(v: StyleDirection) -> u8 {
match v {
StyleDirection::Ltr => 0,
StyleDirection::Rtl => 1,
}
}
#[inline(always)]
pub fn style_vertical_align_from_u8(v: u8) -> StyleVerticalAlign {
match v {
0 => StyleVerticalAlign::Baseline,
1 => StyleVerticalAlign::Top,
2 => StyleVerticalAlign::Middle,
3 => StyleVerticalAlign::Bottom,
4 => StyleVerticalAlign::Sub,
5 => StyleVerticalAlign::Superscript,
6 => StyleVerticalAlign::TextTop,
7 => StyleVerticalAlign::TextBottom,
_ => StyleVerticalAlign::Baseline,
}
}
#[inline(always)]
pub fn style_vertical_align_to_u8(v: StyleVerticalAlign) -> u8 {
match v {
StyleVerticalAlign::Baseline => 0,
StyleVerticalAlign::Top => 1,
StyleVerticalAlign::Middle => 2,
StyleVerticalAlign::Bottom => 3,
StyleVerticalAlign::Sub => 4,
StyleVerticalAlign::Superscript => 5,
StyleVerticalAlign::TextTop => 6,
StyleVerticalAlign::TextBottom => 7,
StyleVerticalAlign::Percentage(_) | StyleVerticalAlign::Length(_) => 0,
}
}
#[inline(always)]
pub fn border_collapse_from_u8(v: u8) -> StyleBorderCollapse {
match v {
0 => StyleBorderCollapse::Separate,
1 => StyleBorderCollapse::Collapse,
_ => StyleBorderCollapse::Separate,
}
}
#[inline(always)]
pub fn border_collapse_to_u8(v: StyleBorderCollapse) -> u8 {
match v {
StyleBorderCollapse::Separate => 0,
StyleBorderCollapse::Collapse => 1,
}
}
#[inline(always)]
pub fn border_style_from_u8(v: u8) -> BorderStyle {
match v {
0 => BorderStyle::None,
1 => BorderStyle::Solid,
2 => BorderStyle::Double,
3 => BorderStyle::Dotted,
4 => BorderStyle::Dashed,
5 => BorderStyle::Hidden,
6 => BorderStyle::Groove,
7 => BorderStyle::Ridge,
8 => BorderStyle::Inset,
9 => BorderStyle::Outset,
_ => BorderStyle::None,
}
}
#[inline(always)]
pub fn border_style_to_u8(v: BorderStyle) -> u8 {
match v {
BorderStyle::None => 0,
BorderStyle::Solid => 1,
BorderStyle::Double => 2,
BorderStyle::Dotted => 3,
BorderStyle::Dashed => 4,
BorderStyle::Hidden => 5,
BorderStyle::Groove => 6,
BorderStyle::Ridge => 7,
BorderStyle::Inset => 8,
BorderStyle::Outset => 9,
}
}
#[inline]
pub fn encode_border_styles_packed(top: BorderStyle, right: BorderStyle, bottom: BorderStyle, left: BorderStyle) -> u16 {
(border_style_to_u8(top) as u16)
| ((border_style_to_u8(right) as u16) << 4)
| ((border_style_to_u8(bottom) as u16) << 8)
| ((border_style_to_u8(left) as u16) << 12)
}
#[inline(always)]
pub fn decode_border_top_style(packed: u16) -> BorderStyle {
border_style_from_u8((packed & 0x0F) as u8)
}
#[inline(always)]
pub fn decode_border_right_style(packed: u16) -> BorderStyle {
border_style_from_u8(((packed >> 4) & 0x0F) as u8)
}
#[inline(always)]
pub fn decode_border_bottom_style(packed: u16) -> BorderStyle {
border_style_from_u8(((packed >> 8) & 0x0F) as u8)
}
#[inline(always)]
pub fn decode_border_left_style(packed: u16) -> BorderStyle {
border_style_from_u8(((packed >> 12) & 0x0F) as u8)
}
#[inline(always)]
pub fn encode_color_u32(c: &ColorU) -> u32 {
((c.r as u32) << 24) | ((c.g as u32) << 16) | ((c.b as u32) << 8) | (c.a as u32)
}
#[inline(always)]
pub fn decode_color_u32(v: u32) -> Option<ColorU> {
if v == 0 { return None; }
Some(ColorU {
r: ((v >> 24) & 0xFF) as u8,
g: ((v >> 16) & 0xFF) as u8,
b: ((v >> 8) & 0xFF) as u8,
a: (v & 0xFF) as u8,
})
}
#[inline]
pub fn encode_tier1(
display: LayoutDisplay,
position: LayoutPosition,
float: LayoutFloat,
overflow_x: LayoutOverflow,
overflow_y: LayoutOverflow,
box_sizing: LayoutBoxSizing,
flex_direction: LayoutFlexDirection,
flex_wrap: LayoutFlexWrap,
justify_content: LayoutJustifyContent,
align_items: LayoutAlignItems,
align_content: LayoutAlignContent,
writing_mode: LayoutWritingMode,
clear: LayoutClear,
font_weight: StyleFontWeight,
font_style: StyleFontStyle,
text_align: StyleTextAlign,
visibility: StyleVisibility,
white_space: StyleWhiteSpace,
direction: StyleDirection,
vertical_align: StyleVerticalAlign,
border_collapse: StyleBorderCollapse,
) -> u64 {
let mut v: u64 = TIER1_POPULATED_BIT;
v |= (layout_display_to_u8(display) as u64) << DISPLAY_SHIFT;
v |= (layout_position_to_u8(position) as u64) << POSITION_SHIFT;
v |= (layout_float_to_u8(float) as u64) << FLOAT_SHIFT;
v |= (layout_overflow_to_u8(overflow_x) as u64) << OVERFLOW_X_SHIFT;
v |= (layout_overflow_to_u8(overflow_y) as u64) << OVERFLOW_Y_SHIFT;
v |= (layout_box_sizing_to_u8(box_sizing) as u64) << BOX_SIZING_SHIFT;
v |= (layout_flex_direction_to_u8(flex_direction) as u64) << FLEX_DIRECTION_SHIFT;
v |= (layout_flex_wrap_to_u8(flex_wrap) as u64) << FLEX_WRAP_SHIFT;
v |= (layout_justify_content_to_u8(justify_content) as u64) << JUSTIFY_CONTENT_SHIFT;
v |= (layout_align_items_to_u8(align_items) as u64) << ALIGN_ITEMS_SHIFT;
v |= (layout_align_content_to_u8(align_content) as u64) << ALIGN_CONTENT_SHIFT;
v |= (layout_writing_mode_to_u8(writing_mode) as u64) << WRITING_MODE_SHIFT;
v |= (layout_clear_to_u8(clear) as u64) << CLEAR_SHIFT;
v |= (style_font_weight_to_u8(font_weight) as u64) << FONT_WEIGHT_SHIFT;
v |= (style_font_style_to_u8(font_style) as u64) << FONT_STYLE_SHIFT;
v |= (style_text_align_to_u8(text_align) as u64) << TEXT_ALIGN_SHIFT;
v |= (style_visibility_to_u8(visibility) as u64) << VISIBILITY_SHIFT;
v |= (style_white_space_to_u8(white_space) as u64) << WHITE_SPACE_SHIFT;
v |= (style_direction_to_u8(direction) as u64) << DIRECTION_SHIFT;
v |= (style_vertical_align_to_u8(vertical_align) as u64) << VERTICAL_ALIGN_SHIFT;
v |= (border_collapse_to_u8(border_collapse) as u64) << BORDER_COLLAPSE_SHIFT;
v
}
#[inline(always)]
pub fn decode_display(t1: u64) -> LayoutDisplay {
layout_display_from_u8(((t1 >> DISPLAY_SHIFT) & DISPLAY_MASK) as u8)
}
#[inline(always)]
pub fn decode_position(t1: u64) -> LayoutPosition {
layout_position_from_u8(((t1 >> POSITION_SHIFT) & POSITION_MASK) as u8)
}
#[inline(always)]
pub fn decode_float(t1: u64) -> LayoutFloat {
layout_float_from_u8(((t1 >> FLOAT_SHIFT) & FLOAT_MASK) as u8)
}
#[inline(always)]
pub fn decode_overflow_x(t1: u64) -> LayoutOverflow {
layout_overflow_from_u8(((t1 >> OVERFLOW_X_SHIFT) & OVERFLOW_MASK) as u8)
}
#[inline(always)]
pub fn decode_overflow_y(t1: u64) -> LayoutOverflow {
layout_overflow_from_u8(((t1 >> OVERFLOW_Y_SHIFT) & OVERFLOW_MASK) as u8)
}
#[inline(always)]
pub fn decode_box_sizing(t1: u64) -> LayoutBoxSizing {
layout_box_sizing_from_u8(((t1 >> BOX_SIZING_SHIFT) & BOX_SIZING_MASK) as u8)
}
#[inline(always)]
pub fn decode_flex_direction(t1: u64) -> LayoutFlexDirection {
layout_flex_direction_from_u8(((t1 >> FLEX_DIRECTION_SHIFT) & FLEX_DIR_MASK) as u8)
}
#[inline(always)]
pub fn decode_flex_wrap(t1: u64) -> LayoutFlexWrap {
layout_flex_wrap_from_u8(((t1 >> FLEX_WRAP_SHIFT) & FLEX_WRAP_MASK) as u8)
}
#[inline(always)]
pub fn decode_justify_content(t1: u64) -> LayoutJustifyContent {
layout_justify_content_from_u8(((t1 >> JUSTIFY_CONTENT_SHIFT) & JUSTIFY_MASK) as u8)
}
#[inline(always)]
pub fn decode_align_items(t1: u64) -> LayoutAlignItems {
layout_align_items_from_u8(((t1 >> ALIGN_ITEMS_SHIFT) & ALIGN_MASK) as u8)
}
#[inline(always)]
pub fn decode_align_content(t1: u64) -> LayoutAlignContent {
layout_align_content_from_u8(((t1 >> ALIGN_CONTENT_SHIFT) & ALIGN_MASK) as u8)
}
#[inline(always)]
pub fn decode_writing_mode(t1: u64) -> LayoutWritingMode {
layout_writing_mode_from_u8(((t1 >> WRITING_MODE_SHIFT) & WRITING_MODE_MASK) as u8)
}
#[inline(always)]
pub fn decode_clear(t1: u64) -> LayoutClear {
layout_clear_from_u8(((t1 >> CLEAR_SHIFT) & CLEAR_MASK) as u8)
}
#[inline(always)]
pub fn decode_font_weight(t1: u64) -> StyleFontWeight {
style_font_weight_from_u8(((t1 >> FONT_WEIGHT_SHIFT) & FONT_WEIGHT_MASK) as u8)
}
#[inline(always)]
pub fn decode_font_style(t1: u64) -> StyleFontStyle {
style_font_style_from_u8(((t1 >> FONT_STYLE_SHIFT) & FONT_STYLE_MASK) as u8)
}
#[inline(always)]
pub fn decode_text_align(t1: u64) -> StyleTextAlign {
style_text_align_from_u8(((t1 >> TEXT_ALIGN_SHIFT) & TEXT_ALIGN_MASK) as u8)
}
#[inline(always)]
pub fn decode_visibility(t1: u64) -> StyleVisibility {
style_visibility_from_u8(((t1 >> VISIBILITY_SHIFT) & VISIBILITY_MASK) as u8)
}
#[inline(always)]
pub fn decode_white_space(t1: u64) -> StyleWhiteSpace {
style_white_space_from_u8(((t1 >> WHITE_SPACE_SHIFT) & WHITE_SPACE_MASK) as u8)
}
#[inline(always)]
pub fn decode_direction(t1: u64) -> StyleDirection {
style_direction_from_u8(((t1 >> DIRECTION_SHIFT) & DIRECTION_MASK) as u8)
}
#[inline(always)]
pub fn decode_vertical_align(t1: u64) -> StyleVerticalAlign {
style_vertical_align_from_u8(((t1 >> VERTICAL_ALIGN_SHIFT) & VERTICAL_ALIGN_MASK) as u8)
}
#[inline(always)]
pub fn decode_border_collapse(t1: u64) -> StyleBorderCollapse {
border_collapse_from_u8(((t1 >> BORDER_COLLAPSE_SHIFT) & BORDER_COLLAPSE_MASK) as u8)
}
#[inline(always)]
#[cfg(test)]
pub fn tier1_is_populated(t1: u64) -> bool {
(t1 & TIER1_POPULATED_BIT) != 0
}
#[inline]
pub fn encode_pixel_value_u32(pv: &PixelValue) -> u32 {
let metric = size_metric_to_u8(pv.metric) as u32;
let raw = pv.number.number; if !(-134_217_728..=134_217_727).contains(&raw) {
return U32_SENTINEL; }
let value_bits = ((raw as i32) as u32) << 4;
value_bits | metric
}
#[inline]
pub fn decode_pixel_value_u32(encoded: u32) -> Option<PixelValue> {
if encoded >= U32_SENTINEL_THRESHOLD {
return None; }
let metric = size_metric_from_u8((encoded & 0xF) as u8);
let value_bits = (encoded as i32) >> 4;
let raw = value_bits as isize; Some(PixelValue {
metric,
number: FloatValue { number: raw },
})
}
#[inline]
pub fn encode_resolved_px_i16(px: f32) -> i16 {
let scaled = (px * 10.0).round() as i32;
if scaled < -32768 || scaled > I16_SENTINEL_THRESHOLD as i32 - 1 {
return I16_SENTINEL; }
scaled as i16
}
#[inline(always)]
pub fn decode_resolved_px_i16(v: i16) -> Option<f32> {
if v >= I16_SENTINEL_THRESHOLD {
return None;
}
Some(v as f32 / 10.0)
}
#[inline]
pub fn encode_flex_u16(value: f32) -> u16 {
let scaled = (value * 100.0).round() as i32;
if scaled < 0 || scaled >= U16_SENTINEL_THRESHOLD as i32 {
return U16_SENTINEL;
}
scaled as u16
}
#[inline(always)]
pub fn decode_flex_u16(v: u16) -> Option<f32> {
if v >= U16_SENTINEL_THRESHOLD {
return None;
}
Some(v as f32 / 100.0)
}
#[inline(always)]
pub fn size_metric_to_u8(m: SizeMetric) -> u8 {
match m {
SizeMetric::Px => 0,
SizeMetric::Pt => 1,
SizeMetric::Em => 2,
SizeMetric::Rem => 3,
SizeMetric::In => 4,
SizeMetric::Cm => 5,
SizeMetric::Mm => 6,
SizeMetric::Percent => 7,
SizeMetric::Vw => 8,
SizeMetric::Vh => 9,
SizeMetric::Vmin => 10,
SizeMetric::Vmax => 11,
}
}
#[inline(always)]
pub fn size_metric_from_u8(v: u8) -> SizeMetric {
match v {
0 => SizeMetric::Px,
1 => SizeMetric::Pt,
2 => SizeMetric::Em,
3 => SizeMetric::Rem,
4 => SizeMetric::In,
5 => SizeMetric::Cm,
6 => SizeMetric::Mm,
7 => SizeMetric::Percent,
8 => SizeMetric::Vw,
9 => SizeMetric::Vh,
10 => SizeMetric::Vmin,
11 => SizeMetric::Vmax,
_ => SizeMetric::Px,
}
}
#[derive(Debug, Copy, Clone, PartialEq)]
#[repr(C)]
pub struct CompactNodeProps {
pub width: u32,
pub height: u32,
pub min_width: u32,
pub max_width: u32,
pub min_height: u32,
pub max_height: u32,
pub flex_basis: u32,
pub font_size: u32,
pub padding_top: i16,
pub padding_right: i16,
pub padding_bottom: i16,
pub padding_left: i16,
pub margin_top: i16,
pub margin_right: i16,
pub margin_bottom: i16,
pub margin_left: i16,
pub border_top_width: i16,
pub border_right_width: i16,
pub border_bottom_width: i16,
pub border_left_width: i16,
pub top: i16,
pub right: i16,
pub bottom: i16,
pub left: i16,
pub flex_grow: u16,
pub flex_shrink: u16,
pub row_gap: i16,
pub column_gap: i16,
}
#[derive(Debug, Copy, Clone, PartialEq)]
#[repr(C)]
pub struct CompactNodePropsCold {
pub border_top_color: u32,
pub border_right_color: u32,
pub border_bottom_color: u32,
pub border_left_color: u32,
pub border_top_left_radius: i16,
pub border_top_right_radius: i16,
pub border_bottom_left_radius: i16,
pub border_bottom_right_radius: i16,
pub z_index: i16, pub border_styles_packed: u16,
pub border_spacing_h: i16,
pub border_spacing_v: i16,
pub tab_size: i16,
pub grid_col_start: i16,
pub grid_col_end: i16,
pub grid_row_start: i16,
pub grid_row_end: i16,
pub opacity: u8,
pub hot_flags: u8,
pub extra_flags: u8,
}
pub const OPACITY_SENTINEL: u8 = 255;
pub const HOT_FLAG_HAS_TRANSFORM: u8 = 1 << 0;
pub const HOT_FLAG_HAS_TRANSFORM_ORIGIN: u8 = 1 << 1;
pub const HOT_FLAG_HAS_BOX_SHADOW: u8 = 1 << 2;
pub const HOT_FLAG_HAS_TEXT_DECORATION: u8 = 1 << 3;
pub const HOT_FLAG_SCROLLBAR_GUTTER_SHIFT: u8 = 4;
pub const HOT_FLAG_SCROLLBAR_GUTTER_MASK: u8 = 0b0011_0000;
pub const HOT_FLAG_HAS_BACKGROUND: u8 = 1 << 6;
pub const HOT_FLAG_HAS_CLIP_PATH: u8 = 1 << 7;
pub const EXTRA_FLAG_HAS_SCROLLBAR_CSS: u8 = 1 << 0;
pub const EXTRA_FLAG_HAS_COUNTER: u8 = 1 << 1;
pub const EXTRA_FLAG_HAS_BREAK: u8 = 1 << 2;
pub const EXTRA_FLAG_HAS_TEXT_ORIENTATION: u8 = 1 << 3;
pub const EXTRA_FLAG_HAS_TEXT_SHADOW: u8 = 1 << 4;
pub const EXTRA_FLAG_HAS_BACKDROP_FILTER: u8 = 1 << 5;
pub const EXTRA_FLAG_HAS_FILTER: u8 = 1 << 6;
pub const EXTRA_FLAG_HAS_MIX_BLEND_MODE: u8 = 1 << 7;
pub const DOM_HAS_SHAPE_INSIDE: u32 = 1 << 0;
pub const DOM_HAS_SHAPE_OUTSIDE: u32 = 1 << 1;
pub const DOM_HAS_TEXT_JUSTIFY: u32 = 1 << 2;
pub const DOM_HAS_TEXT_INDENT: u32 = 1 << 3;
pub const DOM_HAS_COLUMN_COUNT: u32 = 1 << 4;
pub const DOM_HAS_COLUMN_GAP: u32 = 1 << 5;
pub const DOM_HAS_INITIAL_LETTER: u32 = 1 << 6;
pub const DOM_HAS_INITIAL_LETTER_ALIGN: u32 = 1 << 7;
pub const DOM_HAS_LINE_CLAMP: u32 = 1 << 8;
pub const DOM_HAS_HANGING_PUNCTUATION: u32 = 1 << 9;
pub const DOM_HAS_TEXT_COMBINE_UPRIGHT: u32 = 1 << 10;
pub const DOM_HAS_EXCLUSION_MARGIN: u32 = 1 << 11;
pub const DOM_HAS_HYPHENATION_LANGUAGE: u32 = 1 << 12;
pub const DOM_HAS_UNICODE_BIDI: u32 = 1 << 13;
pub const DOM_HAS_TEXT_BOX_TRIM: u32 = 1 << 14;
pub const DOM_HAS_HYPHENS: u32 = 1 << 15;
pub const DOM_HAS_WORD_BREAK: u32 = 1 << 16;
pub const DOM_HAS_OVERFLOW_WRAP: u32 = 1 << 17;
pub const DOM_HAS_LINE_BREAK: u32 = 1 << 18;
pub const DOM_HAS_TEXT_ALIGN_LAST: u32 = 1 << 19;
pub const DOM_HAS_LINE_HEIGHT: u32 = 1 << 20;
pub const DOM_HAS_COLUMN_WIDTH: u32 = 1 << 21;
pub const DOM_HAS_SHAPE_MARGIN: u32 = 1 << 22;
pub const SCROLLBAR_GUTTER_AUTO: u8 = 0;
pub const SCROLLBAR_GUTTER_STABLE: u8 = 1;
pub const SCROLLBAR_GUTTER_BOTH_EDGES: u8 = 2;
pub const SCROLLBAR_GUTTER_MIRROR: u8 = 3;
impl Default for CompactNodeProps {
fn default() -> Self {
Self {
width: U32_AUTO,
height: U32_AUTO,
min_width: U32_AUTO,
max_width: U32_NONE,
min_height: U32_AUTO,
max_height: U32_NONE,
flex_basis: U32_AUTO,
font_size: U32_INITIAL,
padding_top: 0,
padding_right: 0,
padding_bottom: 0,
padding_left: 0,
margin_top: 0,
margin_right: 0,
margin_bottom: 0,
margin_left: 0,
border_top_width: 0,
border_right_width: 0,
border_bottom_width: 0,
border_left_width: 0,
top: I16_AUTO,
right: I16_AUTO,
bottom: I16_AUTO,
left: I16_AUTO,
flex_grow: 0,
flex_shrink: encode_flex_u16(1.0),
row_gap: 0,
column_gap: 0,
}
}
}
impl Default for CompactNodePropsCold {
fn default() -> Self {
Self {
border_top_color: 0,
border_right_color: 0,
border_bottom_color: 0,
border_left_color: 0,
border_top_left_radius: I16_SENTINEL,
border_top_right_radius: I16_SENTINEL,
border_bottom_left_radius: I16_SENTINEL,
border_bottom_right_radius: I16_SENTINEL,
z_index: I16_AUTO,
border_styles_packed: 0, border_spacing_h: 0,
border_spacing_v: 0,
tab_size: I16_SENTINEL, grid_col_start: I16_AUTO,
grid_col_end: I16_AUTO,
grid_row_start: I16_AUTO,
grid_row_end: I16_AUTO,
opacity: OPACITY_SENTINEL,
hot_flags: 0,
extra_flags: 0,
}
}
}
#[derive(Debug, Copy, Clone, PartialEq)]
#[repr(C)]
pub struct CompactTextProps {
pub text_color: u32, pub font_family_hash: u64, pub line_height: i16, pub letter_spacing: i16, pub word_spacing: i16, pub text_indent: i16, }
impl Default for CompactTextProps {
fn default() -> Self {
Self {
text_color: 0,
font_family_hash: 0,
line_height: I16_SENTINEL, letter_spacing: 0,
word_spacing: 0,
text_indent: 0,
}
}
}
#[derive(Debug, Clone, PartialEq)]
pub struct CompactLayoutCache {
pub tier1_enums: Vec<u64>,
pub tier2_dims: Vec<CompactNodeProps>,
pub tier2_cold: Vec<CompactNodePropsCold>,
pub tier2b_text: Vec<CompactTextProps>,
pub font_dirty_nodes: Vec<usize>,
pub prev_font_hashes: Vec<u64>,
pub font_hash_to_families: alloc::collections::BTreeMap<u64, crate::props::basic::font::StyleFontFamilyVec>,
pub dom_declared_flags: u32,
}
impl CompactLayoutCache {
pub fn empty() -> Self {
Self {
tier1_enums: Vec::new(),
tier2_dims: Vec::new(),
tier2_cold: Vec::new(),
tier2b_text: Vec::new(),
font_dirty_nodes: Vec::new(),
prev_font_hashes: Vec::new(),
font_hash_to_families: alloc::collections::BTreeMap::new(),
dom_declared_flags: 0,
}
}
pub fn with_capacity(node_count: usize) -> Self {
Self {
tier1_enums: vec![0u64; node_count],
tier2_dims: vec![CompactNodeProps::default(); node_count],
tier2_cold: vec![CompactNodePropsCold::default(); node_count],
tier2b_text: vec![CompactTextProps::default(); node_count],
font_dirty_nodes: Vec::new(),
prev_font_hashes: vec![0u64; node_count],
font_hash_to_families: alloc::collections::BTreeMap::new(),
dom_declared_flags: 0,
}
}
#[inline]
pub fn node_count(&self) -> usize {
self.tier1_enums.len()
}
#[inline(always)]
pub fn get_display(&self, node_idx: usize) -> LayoutDisplay {
decode_display(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_position(&self, node_idx: usize) -> LayoutPosition {
decode_position(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_float(&self, node_idx: usize) -> LayoutFloat {
decode_float(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_overflow_x(&self, node_idx: usize) -> LayoutOverflow {
decode_overflow_x(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_overflow_y(&self, node_idx: usize) -> LayoutOverflow {
decode_overflow_y(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_box_sizing(&self, node_idx: usize) -> LayoutBoxSizing {
decode_box_sizing(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_flex_direction(&self, node_idx: usize) -> LayoutFlexDirection {
decode_flex_direction(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_flex_wrap(&self, node_idx: usize) -> LayoutFlexWrap {
decode_flex_wrap(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_justify_content(&self, node_idx: usize) -> LayoutJustifyContent {
decode_justify_content(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_align_items(&self, node_idx: usize) -> LayoutAlignItems {
decode_align_items(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_align_content(&self, node_idx: usize) -> LayoutAlignContent {
decode_align_content(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_writing_mode(&self, node_idx: usize) -> LayoutWritingMode {
decode_writing_mode(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_clear(&self, node_idx: usize) -> LayoutClear {
decode_clear(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_font_weight(&self, node_idx: usize) -> StyleFontWeight {
decode_font_weight(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_font_style(&self, node_idx: usize) -> StyleFontStyle {
decode_font_style(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_text_align(&self, node_idx: usize) -> StyleTextAlign {
decode_text_align(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_visibility(&self, node_idx: usize) -> StyleVisibility {
decode_visibility(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_white_space(&self, node_idx: usize) -> StyleWhiteSpace {
decode_white_space(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_direction(&self, node_idx: usize) -> StyleDirection {
decode_direction(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_vertical_align(&self, node_idx: usize) -> StyleVerticalAlign {
decode_vertical_align(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_border_collapse(&self, node_idx: usize) -> StyleBorderCollapse {
decode_border_collapse(self.tier1_enums[node_idx])
}
#[inline(always)]
pub fn get_width_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].width
}
#[inline(always)]
pub fn get_height_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].height
}
#[inline(always)]
pub fn get_min_width_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].min_width
}
#[inline(always)]
pub fn get_max_width_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].max_width
}
#[inline(always)]
pub fn get_min_height_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].min_height
}
#[inline(always)]
pub fn get_max_height_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].max_height
}
#[inline(always)]
pub fn get_font_size_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].font_size
}
#[inline(always)]
pub fn get_flex_basis_raw(&self, node_idx: usize) -> u32 {
self.tier2_dims[node_idx].flex_basis
}
#[inline(always)]
pub fn get_padding_top(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].padding_top)
}
#[inline(always)]
pub fn get_padding_right(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].padding_right)
}
#[inline(always)]
pub fn get_padding_bottom(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].padding_bottom)
}
#[inline(always)]
pub fn get_padding_left(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].padding_left)
}
#[inline(always)]
pub fn get_margin_top(&self, node_idx: usize) -> Option<f32> {
let v = self.tier2_dims[node_idx].margin_top;
if v == I16_AUTO { return None; } decode_resolved_px_i16(v)
}
#[inline(always)]
pub fn get_margin_right(&self, node_idx: usize) -> Option<f32> {
let v = self.tier2_dims[node_idx].margin_right;
if v == I16_AUTO { return None; }
decode_resolved_px_i16(v)
}
#[inline(always)]
pub fn get_margin_bottom(&self, node_idx: usize) -> Option<f32> {
let v = self.tier2_dims[node_idx].margin_bottom;
if v == I16_AUTO { return None; }
decode_resolved_px_i16(v)
}
#[inline(always)]
pub fn get_margin_left(&self, node_idx: usize) -> Option<f32> {
let v = self.tier2_dims[node_idx].margin_left;
if v == I16_AUTO { return None; }
decode_resolved_px_i16(v)
}
#[inline(always)]
pub fn is_margin_top_auto(&self, node_idx: usize) -> bool {
self.tier2_dims[node_idx].margin_top == I16_AUTO
}
#[inline(always)]
pub fn is_margin_right_auto(&self, node_idx: usize) -> bool {
self.tier2_dims[node_idx].margin_right == I16_AUTO
}
#[inline(always)]
pub fn is_margin_bottom_auto(&self, node_idx: usize) -> bool {
self.tier2_dims[node_idx].margin_bottom == I16_AUTO
}
#[inline(always)]
pub fn is_margin_left_auto(&self, node_idx: usize) -> bool {
self.tier2_dims[node_idx].margin_left == I16_AUTO
}
#[inline(always)]
pub fn get_border_top_width(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].border_top_width)
}
#[inline(always)]
pub fn get_border_right_width(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].border_right_width)
}
#[inline(always)]
pub fn get_border_bottom_width(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].border_bottom_width)
}
#[inline(always)]
pub fn get_border_left_width(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2_dims[node_idx].border_left_width)
}
#[inline(always)]
pub fn get_padding_top_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].padding_top
}
#[inline(always)]
pub fn get_padding_right_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].padding_right
}
#[inline(always)]
pub fn get_padding_bottom_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].padding_bottom
}
#[inline(always)]
pub fn get_padding_left_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].padding_left
}
#[inline(always)]
pub fn get_margin_top_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].margin_top
}
#[inline(always)]
pub fn get_margin_right_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].margin_right
}
#[inline(always)]
pub fn get_margin_bottom_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].margin_bottom
}
#[inline(always)]
pub fn get_margin_left_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].margin_left
}
#[inline(always)]
pub fn get_border_top_width_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].border_top_width
}
#[inline(always)]
pub fn get_border_right_width_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].border_right_width
}
#[inline(always)]
pub fn get_border_bottom_width_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].border_bottom_width
}
#[inline(always)]
pub fn get_border_left_width_raw(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].border_left_width
}
#[inline(always)]
pub fn get_top(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].top
}
#[inline(always)]
pub fn get_right(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].right
}
#[inline(always)]
pub fn get_bottom(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].bottom
}
#[inline(always)]
pub fn get_left(&self, node_idx: usize) -> i16 {
self.tier2_dims[node_idx].left
}
#[inline(always)]
pub fn get_flex_grow(&self, node_idx: usize) -> Option<f32> {
decode_flex_u16(self.tier2_dims[node_idx].flex_grow)
}
#[inline(always)]
pub fn get_flex_shrink(&self, node_idx: usize) -> Option<f32> {
decode_flex_u16(self.tier2_dims[node_idx].flex_shrink)
}
#[inline(always)]
pub fn get_z_index(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].z_index
}
#[inline(always)]
pub fn get_border_top_color_raw(&self, node_idx: usize) -> u32 {
self.tier2_cold[node_idx].border_top_color
}
#[inline(always)]
pub fn get_border_right_color_raw(&self, node_idx: usize) -> u32 {
self.tier2_cold[node_idx].border_right_color
}
#[inline(always)]
pub fn get_border_bottom_color_raw(&self, node_idx: usize) -> u32 {
self.tier2_cold[node_idx].border_bottom_color
}
#[inline(always)]
pub fn get_border_left_color_raw(&self, node_idx: usize) -> u32 {
self.tier2_cold[node_idx].border_left_color
}
#[inline(always)]
pub fn get_border_styles_packed(&self, node_idx: usize) -> u16 {
self.tier2_cold[node_idx].border_styles_packed
}
#[inline(always)]
pub fn get_border_top_style(&self, node_idx: usize) -> BorderStyle {
decode_border_top_style(self.tier2_cold[node_idx].border_styles_packed)
}
#[inline(always)]
pub fn get_border_right_style(&self, node_idx: usize) -> BorderStyle {
decode_border_right_style(self.tier2_cold[node_idx].border_styles_packed)
}
#[inline(always)]
pub fn get_border_bottom_style(&self, node_idx: usize) -> BorderStyle {
decode_border_bottom_style(self.tier2_cold[node_idx].border_styles_packed)
}
#[inline(always)]
pub fn get_border_left_style(&self, node_idx: usize) -> BorderStyle {
decode_border_left_style(self.tier2_cold[node_idx].border_styles_packed)
}
#[inline(always)]
pub fn get_border_spacing_h_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_spacing_h
}
#[inline(always)]
pub fn get_border_spacing_v_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_spacing_v
}
#[inline(always)]
pub fn get_tab_size_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].tab_size
}
#[inline(always)]
pub fn get_border_top_left_radius_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_top_left_radius
}
#[inline(always)]
pub fn get_border_top_right_radius_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_top_right_radius
}
#[inline(always)]
pub fn get_border_bottom_left_radius_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_bottom_left_radius
}
#[inline(always)]
pub fn get_border_bottom_right_radius_raw(&self, node_idx: usize) -> i16 {
self.tier2_cold[node_idx].border_bottom_right_radius
}
#[inline(always)]
pub fn get_opacity_raw(&self, node_idx: usize) -> u8 {
self.tier2_cold[node_idx].opacity
}
#[inline(always)]
pub fn get_hot_flags(&self, node_idx: usize) -> u8 {
self.tier2_cold[node_idx].hot_flags
}
#[inline(always)]
pub fn has_transform(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_TRANSFORM != 0
}
#[inline(always)]
pub fn has_transform_origin(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_TRANSFORM_ORIGIN != 0
}
#[inline(always)]
pub fn has_box_shadow(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_BOX_SHADOW != 0
}
#[inline(always)]
pub fn has_text_decoration(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_TEXT_DECORATION != 0
}
#[inline(always)]
pub fn has_background(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_BACKGROUND != 0
}
#[inline(always)]
pub fn has_clip_path(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].hot_flags & HOT_FLAG_HAS_CLIP_PATH != 0
}
#[inline(always)]
pub fn has_scrollbar_css(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_SCROLLBAR_CSS != 0
}
#[inline(always)]
pub fn has_counter(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_COUNTER != 0
}
#[inline(always)]
pub fn has_break(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_BREAK != 0
}
#[inline(always)]
pub fn has_text_orientation(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_TEXT_ORIENTATION != 0
}
#[inline(always)]
pub fn has_text_shadow(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_TEXT_SHADOW != 0
}
#[inline(always)]
pub fn has_backdrop_filter(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_BACKDROP_FILTER != 0
}
#[inline(always)]
pub fn has_filter(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_FILTER != 0
}
#[inline(always)]
pub fn has_mix_blend_mode(&self, node_idx: usize) -> bool {
self.tier2_cold[node_idx].extra_flags & EXTRA_FLAG_HAS_MIX_BLEND_MODE != 0
}
#[inline(always)]
pub fn dom_declared(&self, flag: u32) -> bool {
self.dom_declared_flags & flag != 0
}
#[inline(always)]
pub fn get_scrollbar_gutter_bits(&self, node_idx: usize) -> u8 {
(self.tier2_cold[node_idx].hot_flags & HOT_FLAG_SCROLLBAR_GUTTER_MASK)
>> HOT_FLAG_SCROLLBAR_GUTTER_SHIFT
}
#[inline(always)]
pub fn get_text_color_raw(&self, node_idx: usize) -> u32 {
self.tier2b_text[node_idx].text_color
}
#[inline(always)]
pub fn get_font_family_hash(&self, node_idx: usize) -> u64 {
self.tier2b_text[node_idx].font_family_hash
}
#[inline(always)]
pub fn get_line_height(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2b_text[node_idx].line_height)
}
#[inline(always)]
pub fn get_letter_spacing(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2b_text[node_idx].letter_spacing)
}
#[inline(always)]
pub fn get_word_spacing(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2b_text[node_idx].word_spacing)
}
#[inline(always)]
pub fn get_text_indent(&self, node_idx: usize) -> Option<f32> {
decode_resolved_px_i16(self.tier2b_text[node_idx].text_indent)
}
}
#[inline]
pub fn encode_css_pixel_as_i16(prop: &CssPropertyValue<PixelValue>) -> i16 {
match prop {
CssPropertyValue::Exact(pv) => {
if pv.metric == SizeMetric::Px {
encode_resolved_px_i16(pv.number.get())
} else {
I16_SENTINEL }
}
CssPropertyValue::Auto => I16_AUTO,
CssPropertyValue::Initial => I16_INITIAL,
CssPropertyValue::Inherit => I16_INHERIT,
_ => I16_SENTINEL,
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_tier1_roundtrip() {
let t1 = encode_tier1(
LayoutDisplay::Flex,
LayoutPosition::Relative,
LayoutFloat::Left,
LayoutOverflow::Hidden,
LayoutOverflow::Scroll,
LayoutBoxSizing::BorderBox,
LayoutFlexDirection::Column,
LayoutFlexWrap::Wrap,
LayoutJustifyContent::SpaceBetween,
LayoutAlignItems::Center,
LayoutAlignContent::End,
LayoutWritingMode::VerticalRl,
LayoutClear::Both,
StyleFontWeight::Bold,
StyleFontStyle::Italic,
StyleTextAlign::Center,
StyleVisibility::Hidden,
StyleWhiteSpace::Pre,
StyleDirection::Rtl,
StyleVerticalAlign::Middle,
StyleBorderCollapse::Collapse,
);
assert!(tier1_is_populated(t1));
assert_eq!(decode_display(t1), LayoutDisplay::Flex);
assert_eq!(decode_position(t1), LayoutPosition::Relative);
assert_eq!(decode_float(t1), LayoutFloat::Left);
assert_eq!(decode_overflow_x(t1), LayoutOverflow::Hidden);
assert_eq!(decode_overflow_y(t1), LayoutOverflow::Scroll);
assert_eq!(decode_box_sizing(t1), LayoutBoxSizing::BorderBox);
assert_eq!(decode_flex_direction(t1), LayoutFlexDirection::Column);
assert_eq!(decode_flex_wrap(t1), LayoutFlexWrap::Wrap);
assert_eq!(decode_justify_content(t1), LayoutJustifyContent::SpaceBetween);
assert_eq!(decode_align_items(t1), LayoutAlignItems::Center);
assert_eq!(decode_align_content(t1), LayoutAlignContent::End);
assert_eq!(decode_writing_mode(t1), LayoutWritingMode::VerticalRl);
assert_eq!(decode_clear(t1), LayoutClear::Both);
assert_eq!(decode_font_weight(t1), StyleFontWeight::Bold);
assert_eq!(decode_font_style(t1), StyleFontStyle::Italic);
assert_eq!(decode_text_align(t1), StyleTextAlign::Center);
assert_eq!(decode_visibility(t1), StyleVisibility::Hidden);
assert_eq!(decode_white_space(t1), StyleWhiteSpace::Pre);
assert_eq!(decode_direction(t1), StyleDirection::Rtl);
assert_eq!(decode_vertical_align(t1), StyleVerticalAlign::Middle);
assert_eq!(decode_border_collapse(t1), StyleBorderCollapse::Collapse);
}
#[test]
fn test_tier1_defaults() {
let t1 = encode_tier1(
LayoutDisplay::Block,
LayoutPosition::Static,
LayoutFloat::None,
LayoutOverflow::Visible,
LayoutOverflow::Visible,
LayoutBoxSizing::ContentBox,
LayoutFlexDirection::Row,
LayoutFlexWrap::NoWrap,
LayoutJustifyContent::FlexStart,
LayoutAlignItems::Stretch,
LayoutAlignContent::Stretch,
LayoutWritingMode::HorizontalTb,
LayoutClear::None,
StyleFontWeight::Normal,
StyleFontStyle::Normal,
StyleTextAlign::Left,
StyleVisibility::Visible,
StyleWhiteSpace::Normal,
StyleDirection::Ltr,
StyleVerticalAlign::Baseline,
StyleBorderCollapse::Separate,
);
assert!(tier1_is_populated(t1));
assert_eq!(decode_display(t1), LayoutDisplay::Block);
assert_eq!(decode_position(t1), LayoutPosition::Static);
}
#[test]
fn test_pixel_value_u32_roundtrip() {
let pv = PixelValue::px(123.456);
let encoded = encode_pixel_value_u32(&pv);
assert!(encoded < U32_SENTINEL_THRESHOLD);
let decoded = decode_pixel_value_u32(encoded).unwrap();
assert_eq!(decoded.metric, SizeMetric::Px);
assert!((decoded.number.get() - 123.456).abs() < 0.002);
}
#[test]
fn test_pixel_value_u32_percent() {
let pv = PixelValue {
metric: SizeMetric::Percent,
number: FloatValue::new(50.0),
};
let encoded = encode_pixel_value_u32(&pv);
let decoded = decode_pixel_value_u32(encoded).unwrap();
assert_eq!(decoded.metric, SizeMetric::Percent);
assert!((decoded.number.get() - 50.0).abs() < 0.002);
}
#[test]
fn test_sentinel_values() {
assert_eq!(decode_pixel_value_u32(U32_SENTINEL), None);
assert_eq!(decode_pixel_value_u32(U32_AUTO), None);
assert_eq!(decode_pixel_value_u32(U32_MIN_CONTENT), None);
assert_eq!(decode_resolved_px_i16(I16_SENTINEL), None);
assert_eq!(decode_resolved_px_i16(I16_AUTO), None);
}
#[test]
fn test_resolved_px_i16_roundtrip() {
let px = 123.4f32;
let encoded = encode_resolved_px_i16(px);
let decoded = decode_resolved_px_i16(encoded).unwrap();
assert!((decoded - 123.4).abs() < 0.11);
let px = -50.7f32;
let encoded = encode_resolved_px_i16(px);
let decoded = decode_resolved_px_i16(encoded).unwrap();
assert!((decoded - (-50.7)).abs() < 0.11);
}
#[test]
fn test_flex_u16_roundtrip() {
let v = 2.5f32;
let encoded = encode_flex_u16(v);
let decoded = decode_flex_u16(encoded).unwrap();
assert!((decoded - 2.5).abs() < 0.011);
}
#[test]
fn test_compact_node_props_size() {
assert_eq!(core::mem::size_of::<CompactNodeProps>(), 72);
}
#[test]
fn test_compact_node_props_cold_size() {
assert_eq!(core::mem::size_of::<CompactNodePropsCold>(), 48);
}
#[test]
fn test_compact_text_props_size() {
assert_eq!(core::mem::size_of::<CompactTextProps>(), 24);
}
#[test]
fn test_justify_items_zero_is_stretch() {
assert_eq!(layout_justify_items_from_u8(0), LayoutJustifyItems::Stretch);
assert_eq!(layout_justify_items_to_u8(LayoutJustifyItems::Stretch), 0);
}
#[test]
fn test_tier1_enum_zero_sentinel_is_css_initial() {
assert_eq!(layout_display_from_u8(0), LayoutDisplay::Block);
assert_eq!(layout_position_from_u8(0), LayoutPosition::Static);
assert_eq!(layout_float_from_u8(0), LayoutFloat::None);
assert_eq!(layout_overflow_from_u8(0), LayoutOverflow::Visible);
assert_eq!(layout_box_sizing_from_u8(0), LayoutBoxSizing::ContentBox);
assert_eq!(layout_flex_direction_from_u8(0), LayoutFlexDirection::Row);
assert_eq!(layout_flex_wrap_from_u8(0), LayoutFlexWrap::NoWrap);
assert_eq!(layout_justify_content_from_u8(0), LayoutJustifyContent::FlexStart);
assert_eq!(layout_align_items_from_u8(0), LayoutAlignItems::Stretch);
assert_eq!(layout_align_content_from_u8(0), LayoutAlignContent::Stretch);
assert_eq!(layout_align_self_from_u8(0), LayoutAlignSelf::Auto);
assert_eq!(layout_justify_self_from_u8(0), LayoutJustifySelf::Auto);
assert_eq!(layout_justify_items_from_u8(0), LayoutJustifyItems::Stretch);
assert_eq!(layout_grid_auto_flow_from_u8(0), LayoutGridAutoFlow::Row);
assert_eq!(layout_writing_mode_from_u8(0), LayoutWritingMode::HorizontalTb);
assert_eq!(layout_clear_from_u8(0), LayoutClear::None);
assert_eq!(style_font_weight_from_u8(0), StyleFontWeight::Normal);
assert_eq!(style_font_style_from_u8(0), StyleFontStyle::Normal);
assert_eq!(style_text_align_from_u8(0), StyleTextAlign::Left);
assert_eq!(style_visibility_from_u8(0), StyleVisibility::Visible);
assert_eq!(style_white_space_from_u8(0), StyleWhiteSpace::Normal);
assert_eq!(style_direction_from_u8(0), StyleDirection::Ltr);
assert_eq!(style_vertical_align_from_u8(0), StyleVerticalAlign::Baseline);
assert_eq!(border_collapse_from_u8(0), StyleBorderCollapse::Separate);
}
#[test]
fn test_tier1_enum_initial_encodes_to_zero() {
assert_eq!(layout_display_to_u8(LayoutDisplay::Block), 0);
assert_eq!(layout_position_to_u8(LayoutPosition::Static), 0);
assert_eq!(layout_float_to_u8(LayoutFloat::None), 0);
assert_eq!(layout_overflow_to_u8(LayoutOverflow::Visible), 0);
assert_eq!(layout_box_sizing_to_u8(LayoutBoxSizing::ContentBox), 0);
assert_eq!(layout_flex_direction_to_u8(LayoutFlexDirection::Row), 0);
assert_eq!(layout_flex_wrap_to_u8(LayoutFlexWrap::NoWrap), 0);
assert_eq!(layout_justify_content_to_u8(LayoutJustifyContent::FlexStart), 0);
assert_eq!(layout_align_items_to_u8(LayoutAlignItems::Stretch), 0);
assert_eq!(layout_align_content_to_u8(LayoutAlignContent::Stretch), 0);
assert_eq!(layout_align_self_to_u8(LayoutAlignSelf::Auto), 0);
assert_eq!(layout_justify_self_to_u8(LayoutJustifySelf::Auto), 0);
assert_eq!(layout_justify_items_to_u8(LayoutJustifyItems::Stretch), 0);
assert_eq!(layout_grid_auto_flow_to_u8(LayoutGridAutoFlow::Row), 0);
assert_eq!(layout_writing_mode_to_u8(LayoutWritingMode::HorizontalTb), 0);
assert_eq!(layout_clear_to_u8(LayoutClear::None), 0);
assert_eq!(style_font_weight_to_u8(StyleFontWeight::Normal), 0);
assert_eq!(style_font_style_to_u8(StyleFontStyle::Normal), 0);
assert_eq!(style_text_align_to_u8(StyleTextAlign::Left), 0);
assert_eq!(style_visibility_to_u8(StyleVisibility::Visible), 0);
assert_eq!(style_white_space_to_u8(StyleWhiteSpace::Normal), 0);
assert_eq!(style_direction_to_u8(StyleDirection::Ltr), 0);
assert_eq!(style_vertical_align_to_u8(StyleVerticalAlign::Baseline), 0);
assert_eq!(border_collapse_to_u8(StyleBorderCollapse::Separate), 0);
}
macro_rules! roundtrip_all {
($name:ident, $to:ident, $from:ident, [$($variant:expr),+ $(,)?]) => {
#[test]
fn $name() {
for v in [$($variant),+] {
let u = $to(v);
let decoded = $from(u);
assert_eq!(decoded, v, "{:?} != {:?} (via u8 = {})", decoded, v, u);
}
}
};
}
roundtrip_all!(rt_display, layout_display_to_u8, layout_display_from_u8, [
LayoutDisplay::Block, LayoutDisplay::Inline, LayoutDisplay::InlineBlock,
LayoutDisplay::Flex, LayoutDisplay::None, LayoutDisplay::InlineFlex,
LayoutDisplay::Table, LayoutDisplay::InlineTable, LayoutDisplay::TableRowGroup,
LayoutDisplay::TableHeaderGroup, LayoutDisplay::TableFooterGroup,
LayoutDisplay::TableRow, LayoutDisplay::TableColumnGroup,
LayoutDisplay::TableColumn, LayoutDisplay::TableCell,
LayoutDisplay::TableCaption, LayoutDisplay::FlowRoot,
LayoutDisplay::ListItem, LayoutDisplay::RunIn, LayoutDisplay::Marker,
LayoutDisplay::Grid, LayoutDisplay::InlineGrid, LayoutDisplay::Contents,
]);
roundtrip_all!(rt_position, layout_position_to_u8, layout_position_from_u8, [
LayoutPosition::Static, LayoutPosition::Relative, LayoutPosition::Absolute,
LayoutPosition::Fixed, LayoutPosition::Sticky,
]);
roundtrip_all!(rt_float, layout_float_to_u8, layout_float_from_u8, [
LayoutFloat::None, LayoutFloat::Left, LayoutFloat::Right,
]);
roundtrip_all!(rt_overflow, layout_overflow_to_u8, layout_overflow_from_u8, [
LayoutOverflow::Visible, LayoutOverflow::Hidden, LayoutOverflow::Scroll,
LayoutOverflow::Auto, LayoutOverflow::Clip,
]);
roundtrip_all!(rt_box_sizing, layout_box_sizing_to_u8, layout_box_sizing_from_u8, [
LayoutBoxSizing::ContentBox, LayoutBoxSizing::BorderBox,
]);
roundtrip_all!(rt_flex_direction, layout_flex_direction_to_u8, layout_flex_direction_from_u8, [
LayoutFlexDirection::Row, LayoutFlexDirection::RowReverse,
LayoutFlexDirection::Column, LayoutFlexDirection::ColumnReverse,
]);
roundtrip_all!(rt_flex_wrap, layout_flex_wrap_to_u8, layout_flex_wrap_from_u8, [
LayoutFlexWrap::NoWrap, LayoutFlexWrap::Wrap, LayoutFlexWrap::WrapReverse,
]);
roundtrip_all!(rt_justify_content, layout_justify_content_to_u8, layout_justify_content_from_u8, [
LayoutJustifyContent::FlexStart, LayoutJustifyContent::FlexEnd,
LayoutJustifyContent::Start, LayoutJustifyContent::End,
LayoutJustifyContent::Center, LayoutJustifyContent::SpaceBetween,
LayoutJustifyContent::SpaceAround, LayoutJustifyContent::SpaceEvenly,
]);
roundtrip_all!(rt_align_items, layout_align_items_to_u8, layout_align_items_from_u8, [
LayoutAlignItems::Stretch, LayoutAlignItems::Center, LayoutAlignItems::Start,
LayoutAlignItems::End, LayoutAlignItems::Baseline,
]);
roundtrip_all!(rt_align_self, layout_align_self_to_u8, layout_align_self_from_u8, [
LayoutAlignSelf::Auto, LayoutAlignSelf::Stretch, LayoutAlignSelf::Center,
LayoutAlignSelf::Start, LayoutAlignSelf::End, LayoutAlignSelf::Baseline,
]);
roundtrip_all!(rt_justify_self, layout_justify_self_to_u8, layout_justify_self_from_u8, [
LayoutJustifySelf::Auto, LayoutJustifySelf::Start, LayoutJustifySelf::End,
LayoutJustifySelf::Center, LayoutJustifySelf::Stretch,
]);
roundtrip_all!(rt_justify_items, layout_justify_items_to_u8, layout_justify_items_from_u8, [
LayoutJustifyItems::Stretch, LayoutJustifyItems::Start,
LayoutJustifyItems::End, LayoutJustifyItems::Center,
]);
roundtrip_all!(rt_grid_auto_flow, layout_grid_auto_flow_to_u8, layout_grid_auto_flow_from_u8, [
LayoutGridAutoFlow::Row, LayoutGridAutoFlow::Column,
LayoutGridAutoFlow::RowDense, LayoutGridAutoFlow::ColumnDense,
]);
roundtrip_all!(rt_align_content, layout_align_content_to_u8, layout_align_content_from_u8, [
LayoutAlignContent::Stretch, LayoutAlignContent::Center,
LayoutAlignContent::Start, LayoutAlignContent::End,
LayoutAlignContent::SpaceBetween, LayoutAlignContent::SpaceAround,
]);
roundtrip_all!(rt_writing_mode, layout_writing_mode_to_u8, layout_writing_mode_from_u8, [
LayoutWritingMode::HorizontalTb, LayoutWritingMode::VerticalRl,
LayoutWritingMode::VerticalLr,
]);
roundtrip_all!(rt_clear, layout_clear_to_u8, layout_clear_from_u8, [
LayoutClear::None, LayoutClear::Left, LayoutClear::Right, LayoutClear::Both,
]);
roundtrip_all!(rt_font_weight, style_font_weight_to_u8, style_font_weight_from_u8, [
StyleFontWeight::Normal, StyleFontWeight::W100, StyleFontWeight::W200,
StyleFontWeight::W300, StyleFontWeight::W500, StyleFontWeight::W600,
StyleFontWeight::Bold, StyleFontWeight::W800, StyleFontWeight::W900,
StyleFontWeight::Lighter, StyleFontWeight::Bolder,
]);
roundtrip_all!(rt_font_style, style_font_style_to_u8, style_font_style_from_u8, [
StyleFontStyle::Normal, StyleFontStyle::Italic, StyleFontStyle::Oblique,
]);
roundtrip_all!(rt_text_align, style_text_align_to_u8, style_text_align_from_u8, [
StyleTextAlign::Left, StyleTextAlign::Center, StyleTextAlign::Right,
StyleTextAlign::Justify, StyleTextAlign::Start, StyleTextAlign::End,
]);
roundtrip_all!(rt_visibility, style_visibility_to_u8, style_visibility_from_u8, [
StyleVisibility::Visible, StyleVisibility::Hidden, StyleVisibility::Collapse,
]);
roundtrip_all!(rt_white_space, style_white_space_to_u8, style_white_space_from_u8, [
StyleWhiteSpace::Normal, StyleWhiteSpace::Pre, StyleWhiteSpace::Nowrap,
StyleWhiteSpace::PreWrap, StyleWhiteSpace::PreLine, StyleWhiteSpace::BreakSpaces,
]);
roundtrip_all!(rt_direction, style_direction_to_u8, style_direction_from_u8, [
StyleDirection::Ltr, StyleDirection::Rtl,
]);
roundtrip_all!(rt_vertical_align, style_vertical_align_to_u8, style_vertical_align_from_u8, [
StyleVerticalAlign::Baseline, StyleVerticalAlign::Top, StyleVerticalAlign::Middle,
StyleVerticalAlign::Bottom, StyleVerticalAlign::Sub, StyleVerticalAlign::Superscript,
StyleVerticalAlign::TextTop, StyleVerticalAlign::TextBottom,
]);
roundtrip_all!(rt_border_collapse, border_collapse_to_u8, border_collapse_from_u8, [
StyleBorderCollapse::Separate, StyleBorderCollapse::Collapse,
]);
#[test]
fn test_encoded_u8_fits_in_tier1_mask() {
fn assert_fits(name: &str, val: u8, mask: u64) {
assert!(
(val as u64) & !mask == 0,
"{}: encoded u8 {} overflows mask {:b}",
name, val, mask,
);
}
assert_fits("display", layout_display_to_u8(LayoutDisplay::Contents), DISPLAY_MASK);
assert_fits("position", layout_position_to_u8(LayoutPosition::Sticky), POSITION_MASK);
assert_fits("float", layout_float_to_u8(LayoutFloat::Right), FLOAT_MASK);
assert_fits("overflow", layout_overflow_to_u8(LayoutOverflow::Clip), OVERFLOW_MASK);
assert_fits("box_sizing", layout_box_sizing_to_u8(LayoutBoxSizing::BorderBox), BOX_SIZING_MASK);
assert_fits("flex_direction", layout_flex_direction_to_u8(LayoutFlexDirection::ColumnReverse), FLEX_DIR_MASK);
assert_fits("flex_wrap", layout_flex_wrap_to_u8(LayoutFlexWrap::WrapReverse), FLEX_WRAP_MASK);
assert_fits("justify_content", layout_justify_content_to_u8(LayoutJustifyContent::SpaceEvenly), JUSTIFY_MASK);
assert_fits("align_items", layout_align_items_to_u8(LayoutAlignItems::Baseline), ALIGN_MASK);
assert_fits("align_self", layout_align_self_to_u8(LayoutAlignSelf::Baseline), ALIGN_SELF_MASK);
assert_fits("justify_self", layout_justify_self_to_u8(LayoutJustifySelf::Stretch), JUSTIFY_SELF_MASK);
assert_fits("justify_items", layout_justify_items_to_u8(LayoutJustifyItems::Center), JUSTIFY_ITEMS_MASK);
assert_fits("grid_auto_flow", layout_grid_auto_flow_to_u8(LayoutGridAutoFlow::ColumnDense), GRID_AUTO_FLOW_MASK);
assert_fits("align_content", layout_align_content_to_u8(LayoutAlignContent::SpaceAround), ALIGN_MASK);
assert_fits("writing_mode", layout_writing_mode_to_u8(LayoutWritingMode::VerticalLr), WRITING_MODE_MASK);
assert_fits("clear", layout_clear_to_u8(LayoutClear::Both), CLEAR_MASK);
assert_fits("font_weight", style_font_weight_to_u8(StyleFontWeight::Bolder), FONT_WEIGHT_MASK);
assert_fits("font_style", style_font_style_to_u8(StyleFontStyle::Oblique), FONT_STYLE_MASK);
assert_fits("text_align", style_text_align_to_u8(StyleTextAlign::End), TEXT_ALIGN_MASK);
assert_fits("visibility", style_visibility_to_u8(StyleVisibility::Collapse), VISIBILITY_MASK);
assert_fits("white_space", style_white_space_to_u8(StyleWhiteSpace::BreakSpaces), WHITE_SPACE_MASK);
assert_fits("direction", style_direction_to_u8(StyleDirection::Rtl), DIRECTION_MASK);
assert_fits("vertical_align", style_vertical_align_to_u8(StyleVerticalAlign::TextBottom), VERTICAL_ALIGN_MASK);
assert_fits("border_collapse", border_collapse_to_u8(StyleBorderCollapse::Collapse), BORDER_COLLAPSE_MASK);
}
#[test]
fn test_empty_tier1_decodes_to_initial_values() {
let t1 = TIER1_POPULATED_BIT; assert!(tier1_is_populated(t1));
assert_eq!(decode_display(t1), LayoutDisplay::Block);
assert_eq!(decode_position(t1), LayoutPosition::Static);
assert_eq!(decode_float(t1), LayoutFloat::None);
assert_eq!(decode_overflow_x(t1), LayoutOverflow::Visible);
assert_eq!(decode_overflow_y(t1), LayoutOverflow::Visible);
assert_eq!(decode_box_sizing(t1), LayoutBoxSizing::ContentBox);
assert_eq!(decode_flex_direction(t1), LayoutFlexDirection::Row);
assert_eq!(decode_flex_wrap(t1), LayoutFlexWrap::NoWrap);
assert_eq!(decode_justify_content(t1), LayoutJustifyContent::FlexStart);
assert_eq!(decode_align_items(t1), LayoutAlignItems::Stretch);
assert_eq!(decode_align_content(t1), LayoutAlignContent::Stretch);
assert_eq!(decode_writing_mode(t1), LayoutWritingMode::HorizontalTb);
assert_eq!(decode_clear(t1), LayoutClear::None);
assert_eq!(decode_font_weight(t1), StyleFontWeight::Normal);
assert_eq!(decode_font_style(t1), StyleFontStyle::Normal);
assert_eq!(decode_text_align(t1), StyleTextAlign::Left);
assert_eq!(decode_visibility(t1), StyleVisibility::Visible);
assert_eq!(decode_white_space(t1), StyleWhiteSpace::Normal);
assert_eq!(decode_direction(t1), StyleDirection::Ltr);
assert_eq!(decode_vertical_align(t1), StyleVerticalAlign::Baseline);
assert_eq!(decode_border_collapse(t1), StyleBorderCollapse::Separate);
}
}