OLD | NEW |
(Empty) | |
| 1 /* |
| 2 * Copyright (c) 2014, Opera Software ASA. All rights reserved. |
| 3 * |
| 4 * Redistribution and use in source and binary forms, with or without |
| 5 * modification, are permitted provided that the following conditions |
| 6 * are met: |
| 7 * 1. Redistributions of source code must retain the above copyright |
| 8 * notice, this list of conditions and the following disclaimer. |
| 9 * 2. Redistributions in binary form must reproduce the above copyright |
| 10 * notice, this list of conditions and the following disclaimer in the |
| 11 * documentation and/or other materials provided with the distribution. |
| 12 * 3. Neither the name of Opera Software ASA nor the names of its |
| 13 * contributors may be used to endorse or promote products derived |
| 14 * from this software without specific prior written permission. |
| 15 * |
| 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 17 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 18 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS |
| 19 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE |
| 20 * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, |
| 21 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| 22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
| 23 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) |
| 24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, |
| 25 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| 26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED |
| 27 * OF THE POSSIBILITY OF SUCH DAMAGE. |
| 28 */ |
| 29 |
| 30 #include "config.h" |
| 31 #include "HTMLNames.h" |
| 32 #include "core/dom/Element.h" |
| 33 #include "core/dom/ElementTraversal.h" |
| 34 #include "core/frame/FrameView.h" |
| 35 #include "core/html/HTMLDocument.h" |
| 36 #include "core/html/HTMLElement.h" |
| 37 #include "core/testing/DummyPageHolder.h" |
| 38 #include <gtest/gtest.h> |
| 39 |
| 40 using namespace WebCore; |
| 41 using namespace HTMLNames; |
| 42 |
| 43 namespace { |
| 44 |
| 45 class AffectedByFocusTest : public ::testing::Test { |
| 46 |
| 47 protected: |
| 48 |
| 49 struct ElementResult { |
| 50 const WebCore::QualifiedName tag; |
| 51 bool affectedBy; |
| 52 bool childrenAffectedBy; |
| 53 }; |
| 54 |
| 55 virtual void SetUp() OVERRIDE; |
| 56 |
| 57 HTMLDocument& document() const { return *m_document; } |
| 58 |
| 59 void setHtmlInnerHTML(const char* htmlContent); |
| 60 |
| 61 void checkElements(ElementResult expected[], unsigned expectedCount) const; |
| 62 |
| 63 private: |
| 64 OwnPtr<DummyPageHolder> m_dummyPageHolder; |
| 65 |
| 66 HTMLDocument* m_document; |
| 67 }; |
| 68 |
| 69 void AffectedByFocusTest::SetUp() |
| 70 { |
| 71 m_dummyPageHolder = DummyPageHolder::create(IntSize(800, 600)); |
| 72 m_document = toHTMLDocument(&m_dummyPageHolder->document()); |
| 73 ASSERT(m_document); |
| 74 } |
| 75 |
| 76 void AffectedByFocusTest::setHtmlInnerHTML(const char* htmlContent) |
| 77 { |
| 78 document().documentElement()->setInnerHTML(String::fromUTF8(htmlContent), AS
SERT_NO_EXCEPTION); |
| 79 document().view()->updateLayoutAndStyleIfNeededRecursive(); |
| 80 } |
| 81 |
| 82 void AffectedByFocusTest::checkElements(ElementResult expected[], unsigned expec
tedCount) const |
| 83 { |
| 84 unsigned i = 0; |
| 85 Element* elm = document().body(); |
| 86 |
| 87 for (; elm && i < 5; elm = ElementTraversal::next(*elm), ++i) { |
| 88 ASSERT_TRUE(elm->hasTagName(expected[i].tag)); |
| 89 ASSERT(elm->renderStyle()); |
| 90 ASSERT_EQ(expected[i].affectedBy, elm->renderStyle()->affectedByFocus())
; |
| 91 ASSERT_EQ(expected[i].childrenAffectedBy, elm->childrenAffectedByFocus()
); |
| 92 } |
| 93 |
| 94 ASSERT(!elm && i == size); |
| 95 } |
| 96 |
| 97 // A global :focus rule in html.css currently causes every single element to be |
| 98 // affectedByFocus. Check that all elements in a document with no :focus rules |
| 99 // gets the affectedByFocus set on RenderStyle and not childrenAffectedByFocus. |
| 100 TEST_F(AffectedByFocusTest, UAUniversalFocusRule) |
| 101 { |
| 102 ElementResult expected[] = { |
| 103 { bodyTag, true, false }, |
| 104 { divTag, true, false }, |
| 105 { divTag, true, false }, |
| 106 { divTag, true, false }, |
| 107 { spanTag, true, false } |
| 108 }; |
| 109 |
| 110 setHtmlInnerHTML("<body>" |
| 111 "<div><div></div></div>" |
| 112 "<div><span></span></div>" |
| 113 "</body>"); |
| 114 |
| 115 checkElements(expected, sizeof(expected) / sizeof(ElementResult)); |
| 116 } |
| 117 |
| 118 // ":focus div" will mark ascendants of all divs with childrenAffectedByFocus. |
| 119 TEST_F(AffectedByFocusTest, FocusedAscendant) |
| 120 { |
| 121 ElementResult expected[] = { |
| 122 { bodyTag, true, true }, |
| 123 { divTag, true, true }, |
| 124 { divTag, true, false }, |
| 125 { divTag, true, false }, |
| 126 { spanTag, true, false } |
| 127 }; |
| 128 |
| 129 setHtmlInnerHTML("<head>" |
| 130 "<style>:focus div { background-color: pink }</style>" |
| 131 "</head>" |
| 132 "<body>" |
| 133 "<div><div></div></div>" |
| 134 "<div><span></span></div>" |
| 135 "</body>"); |
| 136 |
| 137 checkElements(expected, sizeof(expected) / sizeof(ElementResult)); |
| 138 } |
| 139 |
| 140 // "body:focus div" will mark the body element with childrenAffectedByFocus. |
| 141 TEST_F(AffectedByFocusTest, FocusedAscendantWithType) |
| 142 { |
| 143 ElementResult expected[] = { |
| 144 { bodyTag, true, true }, |
| 145 { divTag, true, false }, |
| 146 { divTag, true, false }, |
| 147 { divTag, true, false }, |
| 148 { spanTag, true, false } |
| 149 }; |
| 150 |
| 151 setHtmlInnerHTML("<head>" |
| 152 "<style>body:focus div { background-color: pink }</style>" |
| 153 "</head>" |
| 154 "<body>" |
| 155 "<div><div></div></div>" |
| 156 "<div><span></span></div>" |
| 157 "</body>"); |
| 158 |
| 159 checkElements(expected, sizeof(expected) / sizeof(ElementResult)); |
| 160 } |
| 161 |
| 162 // ":not(body):focus div" should not mark the body element with childrenAffected
ByFocus. |
| 163 // Note that currently ":focus:not(body)" does not do the same. Then the :focus
is |
| 164 // checked and the childrenAffectedByFocus flag set before the negated type sele
ctor |
| 165 // is found. |
| 166 TEST_F(AffectedByFocusTest, FocusedAscendantWithNegatedType) |
| 167 { |
| 168 ElementResult expected[] = { |
| 169 { bodyTag, true, false }, |
| 170 { divTag, true, true }, |
| 171 { divTag, true, false }, |
| 172 { divTag, true, false }, |
| 173 { spanTag, true, false } |
| 174 }; |
| 175 |
| 176 setHtmlInnerHTML("<head>" |
| 177 "<style>:not(body):focus div { background-color: pink }</style>" |
| 178 "</head>" |
| 179 "<body>" |
| 180 "<div><div></div></div>" |
| 181 "<div><span></span></div>" |
| 182 "</body>"); |
| 183 |
| 184 checkElements(expected, sizeof(expected) / sizeof(ElementResult)); |
| 185 } |
| 186 |
| 187 // Checking current behavior for ":focus + div", but this is a BUG or at best |
| 188 // sub-optimal. The focused element will also in this case get childrenAffectedB
yFocus |
| 189 // even if it's really a sibling. Effectively, the whole sub-tree of the focused |
| 190 // element will have styles recalculated even though none of the children are |
| 191 // affected. There are other mechanisms that makes sure the sibling also gets it
s |
| 192 // styles recalculated. |
| 193 TEST_F(AffectedByFocusTest, FocusedSibling) |
| 194 { |
| 195 ElementResult expected[] = { |
| 196 { bodyTag, true, false }, |
| 197 { divTag, true, true }, |
| 198 { spanTag, true, false }, |
| 199 { divTag, true, false } |
| 200 }; |
| 201 |
| 202 setHtmlInnerHTML("<head>" |
| 203 "<style>:focus + div { background-color: pink }</style>" |
| 204 "</head>" |
| 205 "<body>" |
| 206 "<div>" |
| 207 " <span></span>" |
| 208 "</div>" |
| 209 "<div></div>" |
| 210 "</body>"); |
| 211 |
| 212 checkElements(expected, sizeof(expected) / sizeof(ElementResult)); |
| 213 } |
| 214 |
| 215 } // namespace |
OLD | NEW |