| 1 | /** |
| 2 | * Copyright (C) 1999 Lars Knoll (knoll@kde.org) |
| 3 | * (C) 1999 Antti Koivisto (koivisto@kde.org) |
| 4 | * Copyright (C) 2003-2006, 2010, 2017 Apple Inc. All rights reserved. |
| 5 | * Copyright (C) 2006 Andrew Wellington (proton@wiretapped.net) |
| 6 | * |
| 7 | * This library is free software; you can redistribute it and/or |
| 8 | * modify it under the terms of the GNU Library General Public |
| 9 | * License as published by the Free Software Foundation; either |
| 10 | * version 2 of the License, or (at your option) any later version. |
| 11 | * |
| 12 | * This library is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * Library General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU Library General Public License |
| 18 | * along with this library; see the file COPYING.LIB. If not, write to |
| 19 | * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| 20 | * Boston, MA 02110-1301, USA. |
| 21 | * |
| 22 | */ |
| 23 | |
| 24 | #include "config.h" |
| 25 | #include "RenderTreeBuilderList.h" |
| 26 | |
| 27 | #include "RenderChildIterator.h" |
| 28 | #include "RenderListMarker.h" |
| 29 | #include "RenderMultiColumnFlow.h" |
| 30 | #include "RenderRuby.h" |
| 31 | #include "RenderTable.h" |
| 32 | |
| 33 | namespace WebCore { |
| 34 | |
| 35 | static RenderBlock* getParentOfFirstLineBox(RenderBlock& current, RenderObject& marker) |
| 36 | { |
| 37 | bool inQuirksMode = current.document().inQuirksMode(); |
| 38 | for (auto& child : childrenOfType<RenderObject>(current)) { |
| 39 | if (&child == &marker) |
| 40 | continue; |
| 41 | |
| 42 | if (child.isInline() && (!is<RenderInline>(child) || current.generatesLineBoxesForInlineChild(&child))) |
| 43 | return ¤t; |
| 44 | |
| 45 | if (child.isFloating() || child.isOutOfFlowPositioned()) |
| 46 | continue; |
| 47 | |
| 48 | if (!is<RenderBlock>(child) || is<RenderTable>(child) || is<RenderRubyAsBlock>(child)) |
| 49 | break; |
| 50 | |
| 51 | if (is<RenderBox>(child) && downcast<RenderBox>(child).isWritingModeRoot()) |
| 52 | break; |
| 53 | |
| 54 | if (is<RenderListItem>(current) && inQuirksMode && child.node() && isHTMLListElement(*child.node())) |
| 55 | break; |
| 56 | |
| 57 | if (RenderBlock* lineBox = getParentOfFirstLineBox(downcast<RenderBlock>(child), marker)) |
| 58 | return lineBox; |
| 59 | } |
| 60 | |
| 61 | return nullptr; |
| 62 | } |
| 63 | |
| 64 | static RenderObject* firstNonMarkerChild(RenderBlock& parent) |
| 65 | { |
| 66 | RenderObject* child = parent.firstChild(); |
| 67 | while (is<RenderListMarker>(child)) |
| 68 | child = child->nextSibling(); |
| 69 | return child; |
| 70 | } |
| 71 | |
| 72 | RenderTreeBuilder::List::List(RenderTreeBuilder& builder) |
| 73 | : m_builder(builder) |
| 74 | { |
| 75 | } |
| 76 | |
| 77 | void RenderTreeBuilder::List::updateItemMarker(RenderListItem& listItemRenderer) |
| 78 | { |
| 79 | auto& style = listItemRenderer.style(); |
| 80 | |
| 81 | if (style.listStyleType() == ListStyleType::None && (!style.listStyleImage() || style.listStyleImage()->errorOccurred())) { |
| 82 | if (auto* marker = listItemRenderer.markerRenderer()) |
| 83 | m_builder.destroy(*marker); |
| 84 | return; |
| 85 | } |
| 86 | |
| 87 | auto newStyle = listItemRenderer.computeMarkerStyle(); |
| 88 | RenderPtr<RenderListMarker> newMarkerRenderer; |
| 89 | auto* markerRenderer = listItemRenderer.markerRenderer(); |
| 90 | if (markerRenderer) |
| 91 | markerRenderer->setStyle(WTFMove(newStyle)); |
| 92 | else { |
| 93 | newMarkerRenderer = WebCore::createRenderer<RenderListMarker>(listItemRenderer, WTFMove(newStyle)); |
| 94 | newMarkerRenderer->initializeStyle(); |
| 95 | markerRenderer = newMarkerRenderer.get(); |
| 96 | listItemRenderer.setMarkerRenderer(*markerRenderer); |
| 97 | } |
| 98 | |
| 99 | RenderElement* currentParent = markerRenderer->parent(); |
| 100 | RenderBlock* newParent = getParentOfFirstLineBox(listItemRenderer, *markerRenderer); |
| 101 | if (!newParent) { |
| 102 | // If the marker is currently contained inside an anonymous box, |
| 103 | // then we are the only item in that anonymous box (since no line box |
| 104 | // parent was found). It's ok to just leave the marker where it is |
| 105 | // in this case. |
| 106 | if (currentParent && currentParent->isAnonymousBlock()) |
| 107 | return; |
| 108 | if (auto* multiColumnFlow = listItemRenderer.multiColumnFlow()) |
| 109 | newParent = multiColumnFlow; |
| 110 | else |
| 111 | newParent = &listItemRenderer; |
| 112 | } |
| 113 | |
| 114 | if (newParent == currentParent) |
| 115 | return; |
| 116 | |
| 117 | if (currentParent) |
| 118 | m_builder.attach(*newParent, m_builder.detach(*currentParent, *markerRenderer, RenderTreeBuilder::CanCollapseAnonymousBlock::No), firstNonMarkerChild(*newParent)); |
| 119 | else |
| 120 | m_builder.attach(*newParent, WTFMove(newMarkerRenderer), firstNonMarkerChild(*newParent)); |
| 121 | |
| 122 | // If current parent is an anonymous block that has lost all its children, destroy it. |
| 123 | if (currentParent && currentParent->isAnonymousBlock() && !currentParent->firstChild() && !downcast<RenderBlock>(*currentParent).continuation()) |
| 124 | m_builder.destroy(*currentParent); |
| 125 | } |
| 126 | |
| 127 | } |
| 128 | |