StyleProperties.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. *
  8. * 1. Redistributions of source code must retain the above copyright notice, this
  9. * list of conditions and the following disclaimer.
  10. *
  11. * 2. Redistributions in binary form must reproduce the above copyright notice,
  12. * this list of conditions and the following disclaimer in the documentation
  13. * and/or other materials provided with the distribution.
  14. *
  15. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  16. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  17. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
  19. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  20. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  21. * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  22. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  23. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  24. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. */
  26. #include <LibCore/DirIterator.h>
  27. #include <LibGfx/FontDatabase.h>
  28. #include <LibWeb/CSS/StyleProperties.h>
  29. #include <LibWeb/FontCache.h>
  30. #include <ctype.h>
  31. namespace Web::CSS {
  32. StyleProperties::StyleProperties()
  33. {
  34. }
  35. StyleProperties::StyleProperties(const StyleProperties& other)
  36. : m_property_values(other.m_property_values)
  37. {
  38. if (other.m_font) {
  39. m_font = other.m_font->clone();
  40. } else {
  41. m_font = nullptr;
  42. }
  43. }
  44. NonnullRefPtr<StyleProperties> StyleProperties::clone() const
  45. {
  46. return adopt(*new StyleProperties(*this));
  47. }
  48. void StyleProperties::set_property(CSS::PropertyID id, NonnullRefPtr<StyleValue> value)
  49. {
  50. m_property_values.set((unsigned)id, move(value));
  51. }
  52. void StyleProperties::set_property(CSS::PropertyID id, const StringView& value)
  53. {
  54. m_property_values.set((unsigned)id, StringStyleValue::create(value));
  55. }
  56. Optional<NonnullRefPtr<StyleValue>> StyleProperties::property(CSS::PropertyID id) const
  57. {
  58. auto it = m_property_values.find((unsigned)id);
  59. if (it == m_property_values.end())
  60. return {};
  61. return it->value;
  62. }
  63. Length StyleProperties::length_or_fallback(CSS::PropertyID id, const Length& fallback) const
  64. {
  65. auto value = property(id);
  66. if (!value.has_value())
  67. return fallback;
  68. return value.value()->to_length();
  69. }
  70. LengthBox StyleProperties::length_box(CSS::PropertyID left_id, CSS::PropertyID top_id, CSS::PropertyID right_id, CSS::PropertyID bottom_id, const CSS::Length& default_value) const
  71. {
  72. LengthBox box;
  73. box.left = length_or_fallback(left_id, default_value);
  74. box.top = length_or_fallback(top_id, default_value);
  75. box.right = length_or_fallback(right_id, default_value);
  76. box.bottom = length_or_fallback(bottom_id, default_value);
  77. return box;
  78. }
  79. String StyleProperties::string_or_fallback(CSS::PropertyID id, const StringView& fallback) const
  80. {
  81. auto value = property(id);
  82. if (!value.has_value())
  83. return fallback;
  84. return value.value()->to_string();
  85. }
  86. Color StyleProperties::color_or_fallback(CSS::PropertyID id, const DOM::Document& document, Color fallback) const
  87. {
  88. auto value = property(id);
  89. if (!value.has_value())
  90. return fallback;
  91. return value.value()->to_color(document);
  92. }
  93. void StyleProperties::load_font() const
  94. {
  95. auto family_value = string_or_fallback(CSS::PropertyID::FontFamily, "Katica");
  96. auto font_size = property(CSS::PropertyID::FontSize).value_or(IdentifierStyleValue::create(CSS::ValueID::Medium));
  97. auto font_weight = property(CSS::PropertyID::FontWeight).value_or(IdentifierStyleValue::create(CSS::ValueID::Normal));
  98. auto family_parts = family_value.split(',');
  99. auto family = family_parts[0];
  100. if (family.is_one_of("monospace", "ui-monospace"))
  101. family = "Csilla";
  102. else if (family.is_one_of("serif", "sans-serif", "cursive", "fantasy", "ui-serif", "ui-sans-serif", "ui-rounded"))
  103. family = "Katica";
  104. int weight = 400;
  105. if (font_weight->is_identifier()) {
  106. switch (static_cast<const IdentifierStyleValue&>(*font_weight).id()) {
  107. case CSS::ValueID::Normal:
  108. weight = 400;
  109. break;
  110. case CSS::ValueID::Bold:
  111. weight = 700;
  112. break;
  113. case CSS::ValueID::Lighter:
  114. // FIXME: This should be relative to the parent.
  115. weight = 400;
  116. break;
  117. case CSS::ValueID::Bolder:
  118. // FIXME: This should be relative to the parent.
  119. weight = 700;
  120. break;
  121. default:
  122. break;
  123. }
  124. } else if (font_weight->is_length()) {
  125. // FIXME: This isn't really a length, it's a numeric value..
  126. int font_weight_integer = font_weight->to_length().raw_value();
  127. if (font_weight_integer <= 400)
  128. weight = 400;
  129. if (font_weight_integer <= 700)
  130. weight = 700;
  131. weight = 900;
  132. }
  133. int size = 10;
  134. if (font_size->is_identifier()) {
  135. switch (static_cast<const IdentifierStyleValue&>(*font_size).id()) {
  136. case CSS::ValueID::XxSmall:
  137. case CSS::ValueID::XSmall:
  138. case CSS::ValueID::Small:
  139. case CSS::ValueID::Medium:
  140. // FIXME: Should be based on "user's default font size"
  141. size = 10;
  142. break;
  143. case CSS::ValueID::Large:
  144. case CSS::ValueID::XLarge:
  145. case CSS::ValueID::XxLarge:
  146. case CSS::ValueID::XxxLarge:
  147. // FIXME: Should be based on "user's default font size"
  148. size = 12;
  149. break;
  150. case CSS::ValueID::Smaller:
  151. // FIXME: This should be relative to the parent.
  152. size = 10;
  153. break;
  154. case CSS::ValueID::Larger:
  155. // FIXME: This should be relative to the parent.
  156. size = 12;
  157. break;
  158. default:
  159. break;
  160. }
  161. } else if (font_size->is_length()) {
  162. // FIXME: This isn't really a length, it's a numeric value..
  163. int font_size_integer = font_size->to_length().raw_value();
  164. size = font_size_integer;
  165. }
  166. FontSelector font_selector { family, size, weight };
  167. auto found_font = FontCache::the().get(font_selector);
  168. if (found_font) {
  169. m_font = found_font;
  170. return;
  171. }
  172. Gfx::FontDatabase::the().for_each_font([&](auto& font) {
  173. if (font.family() == family && font.weight() == weight && font.presentation_size() == size)
  174. found_font = font;
  175. });
  176. if (!found_font) {
  177. dbgln("Font not found: '{}' {} {}", family, size, weight);
  178. found_font = Gfx::FontDatabase::default_font();
  179. }
  180. m_font = found_font;
  181. FontCache::the().set(font_selector, *m_font);
  182. }
  183. float StyleProperties::line_height(const Layout::Node& layout_node) const
  184. {
  185. auto line_height_length = length_or_fallback(CSS::PropertyID::LineHeight, Length::make_auto());
  186. if (line_height_length.is_absolute())
  187. return (float)line_height_length.to_px(layout_node);
  188. return (float)font().glyph_height() * 1.4f;
  189. }
  190. Optional<int> StyleProperties::z_index() const
  191. {
  192. auto value = property(CSS::PropertyID::ZIndex);
  193. if (!value.has_value())
  194. return {};
  195. return static_cast<int>(value.value()->to_length().raw_value());
  196. }
  197. Optional<CSS::FlexDirection> StyleProperties::flex_direction() const
  198. {
  199. auto value = property(CSS::PropertyID::FlexDirection);
  200. if (!value.has_value())
  201. return {};
  202. switch (value.value()->to_identifier()) {
  203. case CSS::ValueID::Row:
  204. return CSS::FlexDirection::Row;
  205. case CSS::ValueID::RowReverse:
  206. return CSS::FlexDirection::RowReverse;
  207. case CSS::ValueID::Column:
  208. return CSS::FlexDirection::Column;
  209. case CSS::ValueID::ColumnReverse:
  210. return CSS::FlexDirection::ColumnReverse;
  211. default:
  212. return {};
  213. }
  214. }
  215. Optional<CSS::Position> StyleProperties::position() const
  216. {
  217. auto value = property(CSS::PropertyID::Position);
  218. if (!value.has_value() || !value.value()->is_identifier())
  219. return {};
  220. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  221. case CSS::ValueID::Static:
  222. return CSS::Position::Static;
  223. case CSS::ValueID::Relative:
  224. return CSS::Position::Relative;
  225. case CSS::ValueID::Absolute:
  226. return CSS::Position::Absolute;
  227. case CSS::ValueID::Fixed:
  228. return CSS::Position::Fixed;
  229. case CSS::ValueID::Sticky:
  230. return CSS::Position::Sticky;
  231. default:
  232. return {};
  233. }
  234. }
  235. bool StyleProperties::operator==(const StyleProperties& other) const
  236. {
  237. if (m_property_values.size() != other.m_property_values.size())
  238. return false;
  239. for (auto& it : m_property_values) {
  240. auto jt = other.m_property_values.find(it.key);
  241. if (jt == other.m_property_values.end())
  242. return false;
  243. auto& my_value = *it.value;
  244. auto& other_value = *jt->value;
  245. if (my_value.type() != other_value.type())
  246. return false;
  247. if (my_value != other_value)
  248. return false;
  249. }
  250. return true;
  251. }
  252. Optional<CSS::TextAlign> StyleProperties::text_align() const
  253. {
  254. auto value = property(CSS::PropertyID::TextAlign);
  255. if (!value.has_value() || !value.value()->is_identifier())
  256. return {};
  257. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  258. case CSS::ValueID::Left:
  259. return CSS::TextAlign::Left;
  260. case CSS::ValueID::Center:
  261. return CSS::TextAlign::Center;
  262. case CSS::ValueID::Right:
  263. return CSS::TextAlign::Right;
  264. case CSS::ValueID::Justify:
  265. return CSS::TextAlign::Justify;
  266. case CSS::ValueID::LibwebCenter:
  267. return CSS::TextAlign::LibwebCenter;
  268. default:
  269. return {};
  270. }
  271. }
  272. Optional<CSS::WhiteSpace> StyleProperties::white_space() const
  273. {
  274. auto value = property(CSS::PropertyID::WhiteSpace);
  275. if (!value.has_value() || !value.value()->is_identifier())
  276. return {};
  277. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  278. case CSS::ValueID::Normal:
  279. return CSS::WhiteSpace::Normal;
  280. case CSS::ValueID::Nowrap:
  281. return CSS::WhiteSpace::Nowrap;
  282. case CSS::ValueID::Pre:
  283. return CSS::WhiteSpace::Pre;
  284. case CSS::ValueID::PreLine:
  285. return CSS::WhiteSpace::PreLine;
  286. case CSS::ValueID::PreWrap:
  287. return CSS::WhiteSpace::PreWrap;
  288. default:
  289. return {};
  290. }
  291. }
  292. Optional<CSS::LineStyle> StyleProperties::line_style(CSS::PropertyID property_id) const
  293. {
  294. auto value = property(property_id);
  295. if (!value.has_value() || !value.value()->is_identifier())
  296. return {};
  297. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  298. case CSS::ValueID::None:
  299. return CSS::LineStyle::None;
  300. case CSS::ValueID::Hidden:
  301. return CSS::LineStyle::Hidden;
  302. case CSS::ValueID::Dotted:
  303. return CSS::LineStyle::Dotted;
  304. case CSS::ValueID::Dashed:
  305. return CSS::LineStyle::Dashed;
  306. case CSS::ValueID::Solid:
  307. return CSS::LineStyle::Solid;
  308. case CSS::ValueID::Double:
  309. return CSS::LineStyle::Double;
  310. case CSS::ValueID::Groove:
  311. return CSS::LineStyle::Groove;
  312. case CSS::ValueID::Ridge:
  313. return CSS::LineStyle::Ridge;
  314. case CSS::ValueID::Inset:
  315. return CSS::LineStyle::Inset;
  316. case CSS::ValueID::Outset:
  317. return CSS::LineStyle::Outset;
  318. default:
  319. return {};
  320. }
  321. }
  322. Optional<CSS::Float> StyleProperties::float_() const
  323. {
  324. auto value = property(CSS::PropertyID::Float);
  325. if (!value.has_value() || !value.value()->is_identifier())
  326. return {};
  327. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  328. case CSS::ValueID::None:
  329. return CSS::Float::None;
  330. case CSS::ValueID::Left:
  331. return CSS::Float::Left;
  332. case CSS::ValueID::Right:
  333. return CSS::Float::Right;
  334. default:
  335. return {};
  336. }
  337. }
  338. Optional<CSS::Clear> StyleProperties::clear() const
  339. {
  340. auto value = property(CSS::PropertyID::Clear);
  341. if (!value.has_value() || !value.value()->is_identifier())
  342. return {};
  343. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  344. case CSS::ValueID::None:
  345. return CSS::Clear::None;
  346. case CSS::ValueID::Left:
  347. return CSS::Clear::Left;
  348. case CSS::ValueID::Right:
  349. return CSS::Clear::Right;
  350. case CSS::ValueID::Both:
  351. return CSS::Clear::Both;
  352. default:
  353. return {};
  354. }
  355. }
  356. Optional<CSS::Cursor> StyleProperties::cursor() const
  357. {
  358. auto value = property(CSS::PropertyID::Cursor);
  359. if (!value.has_value() || !value.value()->is_identifier())
  360. return {};
  361. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  362. case CSS::ValueID::Auto:
  363. return CSS::Cursor::Auto;
  364. case CSS::ValueID::Default:
  365. return CSS::Cursor::Default;
  366. case CSS::ValueID::None:
  367. return CSS::Cursor::None;
  368. case CSS::ValueID::ContextMenu:
  369. return CSS::Cursor::ContextMenu;
  370. case CSS::ValueID::Help:
  371. return CSS::Cursor::Help;
  372. case CSS::ValueID::Pointer:
  373. return CSS::Cursor::Pointer;
  374. case CSS::ValueID::Progress:
  375. return CSS::Cursor::Progress;
  376. case CSS::ValueID::Wait:
  377. return CSS::Cursor::Wait;
  378. case CSS::ValueID::Cell:
  379. return CSS::Cursor::Cell;
  380. case CSS::ValueID::Crosshair:
  381. return CSS::Cursor::Crosshair;
  382. case CSS::ValueID::Text:
  383. return CSS::Cursor::Text;
  384. case CSS::ValueID::VerticalText:
  385. return CSS::Cursor::VerticalText;
  386. case CSS::ValueID::Alias:
  387. return CSS::Cursor::Alias;
  388. case CSS::ValueID::Copy:
  389. return CSS::Cursor::Copy;
  390. case CSS::ValueID::Move:
  391. return CSS::Cursor::Move;
  392. case CSS::ValueID::NoDrop:
  393. return CSS::Cursor::NoDrop;
  394. case CSS::ValueID::NotAllowed:
  395. return CSS::Cursor::NotAllowed;
  396. case CSS::ValueID::Grab:
  397. return CSS::Cursor::Grab;
  398. case CSS::ValueID::Grabbing:
  399. return CSS::Cursor::Grabbing;
  400. case CSS::ValueID::EResize:
  401. return CSS::Cursor::EResize;
  402. case CSS::ValueID::NResize:
  403. return CSS::Cursor::NResize;
  404. case CSS::ValueID::NeResize:
  405. return CSS::Cursor::NeResize;
  406. case CSS::ValueID::NwResize:
  407. return CSS::Cursor::NwResize;
  408. case CSS::ValueID::SResize:
  409. return CSS::Cursor::SResize;
  410. case CSS::ValueID::SeResize:
  411. return CSS::Cursor::SeResize;
  412. case CSS::ValueID::SwResize:
  413. return CSS::Cursor::SwResize;
  414. case CSS::ValueID::WResize:
  415. return CSS::Cursor::WResize;
  416. case CSS::ValueID::EwResize:
  417. return CSS::Cursor::EwResize;
  418. case CSS::ValueID::NsResize:
  419. return CSS::Cursor::NsResize;
  420. case CSS::ValueID::NeswResize:
  421. return CSS::Cursor::NeswResize;
  422. case CSS::ValueID::NwseResize:
  423. return CSS::Cursor::NwseResize;
  424. case CSS::ValueID::ColResize:
  425. return CSS::Cursor::ColResize;
  426. case CSS::ValueID::RowResize:
  427. return CSS::Cursor::RowResize;
  428. case CSS::ValueID::AllScroll:
  429. return CSS::Cursor::AllScroll;
  430. case CSS::ValueID::ZoomIn:
  431. return CSS::Cursor::ZoomIn;
  432. case CSS::ValueID::ZoomOut:
  433. return CSS::Cursor::ZoomOut;
  434. default:
  435. return {};
  436. }
  437. }
  438. CSS::Display StyleProperties::display() const
  439. {
  440. auto value = property(CSS::PropertyID::Display);
  441. if (!value.has_value() || !value.value()->is_identifier())
  442. return CSS::Display::Inline;
  443. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  444. case CSS::ValueID::None:
  445. return CSS::Display::None;
  446. case CSS::ValueID::Block:
  447. return CSS::Display::Block;
  448. case CSS::ValueID::Inline:
  449. return CSS::Display::Inline;
  450. case CSS::ValueID::InlineBlock:
  451. return CSS::Display::InlineBlock;
  452. case CSS::ValueID::ListItem:
  453. return CSS::Display::ListItem;
  454. case CSS::ValueID::Table:
  455. return CSS::Display::Table;
  456. case CSS::ValueID::TableRow:
  457. return CSS::Display::TableRow;
  458. case CSS::ValueID::TableCell:
  459. return CSS::Display::TableCell;
  460. case CSS::ValueID::TableColumn:
  461. return CSS::Display::TableColumn;
  462. case CSS::ValueID::TableColumnGroup:
  463. return CSS::Display::TableColumnGroup;
  464. case CSS::ValueID::TableCaption:
  465. return CSS::Display::TableCaption;
  466. case CSS::ValueID::TableRowGroup:
  467. return CSS::Display::TableRowGroup;
  468. case CSS::ValueID::TableHeaderGroup:
  469. return CSS::Display::TableHeaderGroup;
  470. case CSS::ValueID::TableFooterGroup:
  471. return CSS::Display::TableFooterGroup;
  472. case CSS::ValueID::Flex:
  473. return CSS::Display::Flex;
  474. default:
  475. return CSS::Display::Block;
  476. }
  477. }
  478. Optional<CSS::TextDecorationLine> StyleProperties::text_decoration_line() const
  479. {
  480. auto value = property(CSS::PropertyID::TextDecorationLine);
  481. if (!value.has_value() || !value.value()->is_identifier())
  482. return {};
  483. switch (static_cast<const IdentifierStyleValue&>(*value.value()).id()) {
  484. case CSS::ValueID::None:
  485. return CSS::TextDecorationLine::None;
  486. case CSS::ValueID::Underline:
  487. return CSS::TextDecorationLine::Underline;
  488. case CSS::ValueID::Overline:
  489. return CSS::TextDecorationLine::Overline;
  490. case CSS::ValueID::LineThrough:
  491. return CSS::TextDecorationLine::LineThrough;
  492. case CSS::ValueID::Blink:
  493. return CSS::TextDecorationLine::Blink;
  494. default:
  495. return {};
  496. }
  497. }
  498. Optional<CSS::TextTransform> StyleProperties::text_transform() const
  499. {
  500. auto value = property(CSS::PropertyID::TextTransform);
  501. if (!value.has_value())
  502. return {};
  503. switch (value.value()->to_identifier()) {
  504. case CSS::ValueID::None:
  505. return CSS::TextTransform::None;
  506. case CSS::ValueID::Lowercase:
  507. return CSS::TextTransform::Lowercase;
  508. case CSS::ValueID::Uppercase:
  509. return CSS::TextTransform::Uppercase;
  510. case CSS::ValueID::Capitalize:
  511. return CSS::TextTransform::Capitalize;
  512. case CSS::ValueID::FullWidth:
  513. return CSS::TextTransform::FullWidth;
  514. case CSS::ValueID::FullSizeKana:
  515. return CSS::TextTransform::FullSizeKana;
  516. default:
  517. return {};
  518. }
  519. }
  520. Optional<CSS::ListStyleType> StyleProperties::list_style_type() const
  521. {
  522. auto value = property(CSS::PropertyID::ListStyleType);
  523. if (!value.has_value())
  524. return {};
  525. switch (value.value()->to_identifier()) {
  526. case CSS::ValueID::None:
  527. return CSS::ListStyleType::None;
  528. case CSS::ValueID::Disc:
  529. return CSS::ListStyleType::Disc;
  530. case CSS::ValueID::Circle:
  531. return CSS::ListStyleType::Circle;
  532. case CSS::ValueID::Square:
  533. return CSS::ListStyleType::Square;
  534. case CSS::ValueID::Decimal:
  535. return CSS::ListStyleType::Decimal;
  536. default:
  537. return {};
  538. }
  539. }
  540. Optional<CSS::Overflow> StyleProperties::overflow_x() const
  541. {
  542. return overflow(CSS::PropertyID::OverflowX);
  543. }
  544. Optional<CSS::Overflow> StyleProperties::overflow_y() const
  545. {
  546. return overflow(CSS::PropertyID::OverflowY);
  547. }
  548. Optional<CSS::Overflow> StyleProperties::overflow(CSS::PropertyID property_id) const
  549. {
  550. auto value = property(property_id);
  551. if (!value.has_value())
  552. return {};
  553. switch (value.value()->to_identifier()) {
  554. case CSS::ValueID::Auto:
  555. return CSS::Overflow::Auto;
  556. case CSS::ValueID::Visible:
  557. return CSS::Overflow::Visible;
  558. case CSS::ValueID::Hidden:
  559. return CSS::Overflow::Hidden;
  560. case CSS::ValueID::Clip:
  561. return CSS::Overflow::Clip;
  562. case CSS::ValueID::Scroll:
  563. return CSS::Overflow::Scroll;
  564. default:
  565. return {};
  566. }
  567. }
  568. Optional<CSS::Repeat> StyleProperties::background_repeat_x() const
  569. {
  570. auto value = property(CSS::PropertyID::BackgroundRepeatX);
  571. if (!value.has_value())
  572. return {};
  573. switch (value.value()->to_identifier()) {
  574. case CSS::ValueID::NoRepeat:
  575. return CSS::Repeat::NoRepeat;
  576. case CSS::ValueID::Repeat:
  577. return CSS::Repeat::Repeat;
  578. case CSS::ValueID::Round:
  579. return CSS::Repeat::Round;
  580. case CSS::ValueID::Space:
  581. return CSS::Repeat::Space;
  582. default:
  583. return {};
  584. }
  585. }
  586. Optional<CSS::Repeat> StyleProperties::background_repeat_y() const
  587. {
  588. auto value = property(CSS::PropertyID::BackgroundRepeatY);
  589. if (!value.has_value())
  590. return {};
  591. switch (value.value()->to_identifier()) {
  592. case CSS::ValueID::NoRepeat:
  593. return CSS::Repeat::NoRepeat;
  594. case CSS::ValueID::Repeat:
  595. return CSS::Repeat::Repeat;
  596. case CSS::ValueID::Round:
  597. return CSS::Repeat::Round;
  598. case CSS::ValueID::Space:
  599. return CSS::Repeat::Space;
  600. default:
  601. return {};
  602. }
  603. }
  604. }