script/dom/html/
htmlhrelement.rs1use std::str::FromStr;
6
7use dom_struct::dom_struct;
8use html5ever::{LocalName, Prefix, local_name, ns};
9use js::rust::HandleObject;
10use style::attr::{AttrValue, LengthOrPercentageOrAuto};
11use style::color::AbsoluteColor;
12use style::values::generics::NonNegative;
13use style::values::specified::border::BorderSideWidth;
14use style::values::specified::length::Size;
15use style::values::specified::{LengthPercentage, NoCalcLength};
16
17use crate::dom::attr::Attr;
18use crate::dom::bindings::codegen::Bindings::HTMLHRElementBinding::HTMLHRElementMethods;
19use crate::dom::bindings::inheritance::Castable;
20use crate::dom::bindings::root::{DomRoot, LayoutDom};
21use crate::dom::bindings::str::DOMString;
22use crate::dom::document::Document;
23use crate::dom::element::{Element, LayoutElementHelpers};
24use crate::dom::html::htmlelement::HTMLElement;
25use crate::dom::node::Node;
26use crate::dom::virtualmethods::VirtualMethods;
27use crate::script_runtime::CanGc;
28
29#[dom_struct]
30pub(crate) struct HTMLHRElement {
31 htmlelement: HTMLElement,
32}
33
34impl HTMLHRElement {
35 fn new_inherited(
36 local_name: LocalName,
37 prefix: Option<Prefix>,
38 document: &Document,
39 ) -> HTMLHRElement {
40 HTMLHRElement {
41 htmlelement: HTMLElement::new_inherited(local_name, prefix, document),
42 }
43 }
44
45 pub(crate) fn new(
46 local_name: LocalName,
47 prefix: Option<Prefix>,
48 document: &Document,
49 proto: Option<HandleObject>,
50 can_gc: CanGc,
51 ) -> DomRoot<HTMLHRElement> {
52 Node::reflect_node_with_proto(
53 Box::new(HTMLHRElement::new_inherited(local_name, prefix, document)),
54 document,
55 proto,
56 can_gc,
57 )
58 }
59}
60
61impl HTMLHRElementMethods<crate::DomTypeHolder> for HTMLHRElement {
62 make_getter!(Align, "align");
64
65 make_atomic_setter!(SetAlign, "align");
67
68 make_getter!(Color, "color");
70
71 make_legacy_color_setter!(SetColor, "color");
73
74 make_bool_getter!(NoShade, "noshade");
76
77 make_bool_setter!(SetNoShade, "noshade");
79
80 make_getter!(Size, "size");
82
83 make_dimension_setter!(SetSize, "size");
85
86 make_getter!(Width, "width");
88
89 make_dimension_setter!(SetWidth, "width");
91}
92
93#[expect(clippy::enum_variant_names)]
97pub(crate) enum SizePresentationalHint {
98 SetHeightTo(Size),
99 SetAllBorderWidthValuesTo(BorderSideWidth),
100 SetBottomBorderWidthToZero,
101}
102
103pub(crate) trait HTMLHRLayoutHelpers {
104 fn get_color(self) -> Option<AbsoluteColor>;
105 fn get_width(self) -> LengthOrPercentageOrAuto;
106 fn get_size_info(self) -> Option<SizePresentationalHint>;
107}
108
109impl HTMLHRLayoutHelpers for LayoutDom<'_, HTMLHRElement> {
110 fn get_color(self) -> Option<AbsoluteColor> {
111 self.upcast::<Element>()
112 .get_attr_for_layout(&ns!(), &local_name!("color"))
113 .and_then(AttrValue::as_color)
114 .cloned()
115 }
116
117 fn get_width(self) -> LengthOrPercentageOrAuto {
118 self.upcast::<Element>()
119 .get_attr_for_layout(&ns!(), &local_name!("width"))
120 .map(AttrValue::as_dimension)
121 .cloned()
122 .unwrap_or(LengthOrPercentageOrAuto::Auto)
123 }
124
125 fn get_size_info(self) -> Option<SizePresentationalHint> {
126 let element = self.upcast::<Element>();
128 let size_value = element
129 .get_attr_val_for_layout(&ns!(), &local_name!("size"))
130 .and_then(|value| usize::from_str(value).ok())
131 .filter(|value| *value != 0)?;
132
133 let hint = if element
134 .get_attr_for_layout(&ns!(), &local_name!("color"))
135 .is_some() ||
136 element
137 .get_attr_for_layout(&ns!(), &local_name!("noshade"))
138 .is_some()
139 {
140 SizePresentationalHint::SetAllBorderWidthValuesTo(BorderSideWidth::from_px(
141 size_value as f32 / 2.0,
142 ))
143 } else if size_value == 1 {
144 SizePresentationalHint::SetBottomBorderWidthToZero
145 } else {
146 SizePresentationalHint::SetHeightTo(Size::LengthPercentage(NonNegative(
147 LengthPercentage::Length(NoCalcLength::from_px((size_value - 2) as f32)),
148 )))
149 };
150
151 Some(hint)
152 }
153}
154
155impl VirtualMethods for HTMLHRElement {
156 fn super_type(&self) -> Option<&dyn VirtualMethods> {
157 Some(self.upcast::<HTMLElement>() as &dyn VirtualMethods)
158 }
159
160 fn attribute_affects_presentational_hints(&self, attr: &Attr) -> bool {
161 match attr.local_name() {
162 &local_name!("width") => true,
163 _ => self
164 .super_type()
165 .unwrap()
166 .attribute_affects_presentational_hints(attr),
167 }
168 }
169
170 fn parse_plain_attribute(&self, name: &LocalName, value: DOMString) -> AttrValue {
171 match *name {
172 local_name!("align") => AttrValue::from_dimension(value.into()),
173 local_name!("color") => AttrValue::from_legacy_color(value.into()),
174 local_name!("width") => AttrValue::from_dimension(value.into()),
175 _ => self
176 .super_type()
177 .unwrap()
178 .parse_plain_attribute(name, value),
179 }
180 }
181}