blob: 0811b189ee796a207f9ae3e953b30117d368d3a1 [file] [log] [blame]
/*
* Copyright (C) 2011 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "core/layout/LayoutTableCell.h"
#include "build/build_config.h"
#include "core/layout/LayoutTestHelper.h"
namespace blink {
class LayoutTableCellDeathTest : public RenderingTest {
protected:
virtual void SetUp() {
RenderingTest::SetUp();
cell_ = LayoutTableCell::CreateAnonymous(&GetDocument());
}
virtual void TearDown() {
cell_->Destroy();
RenderingTest::TearDown();
}
LayoutTableCell* cell_;
};
TEST_F(LayoutTableCellDeathTest, CanSetColumn) {
static const unsigned kColumnIndex = 10;
cell_->SetAbsoluteColumnIndex(kColumnIndex);
EXPECT_EQ(kColumnIndex, cell_->AbsoluteColumnIndex());
}
TEST_F(LayoutTableCellDeathTest, CanSetColumnToMaxColumnIndex) {
cell_->SetAbsoluteColumnIndex(kMaxColumnIndex);
EXPECT_EQ(kMaxColumnIndex, cell_->AbsoluteColumnIndex());
}
// FIXME: Re-enable these tests once ASSERT_DEATH is supported for Android.
// See: https://bugs.webkit.org/show_bug.cgi?id=74089
// TODO(dgrogan): These tests started flaking on Mac try bots around 2016-07-28.
// https://crbug.com/632816
#if !defined(OS_ANDROID) && !defined(OS_MACOSX)
TEST_F(LayoutTableCellDeathTest, CrashIfColumnOverflowOnSetting) {
ASSERT_DEATH(cell_->SetAbsoluteColumnIndex(kMaxColumnIndex + 1), "");
}
TEST_F(LayoutTableCellDeathTest, CrashIfSettingUnsetColumnIndex) {
ASSERT_DEATH(cell_->SetAbsoluteColumnIndex(kUnsetColumnIndex), "");
}
#endif
class LayoutTableCellTest : public RenderingTest {
protected:
bool IsInStartColumn(const LayoutTableCell* cell) {
return cell->IsInStartColumn();
}
bool IsInEndColumn(const LayoutTableCell* cell) {
return cell->IsInEndColumn();
}
LayoutRect LocalVisualRect(const LayoutTableCell* cell) {
return cell->LocalVisualRect();
}
LayoutTableCell* GetCellByElementId(const char* id) {
return ToLayoutTableCell(GetLayoutObjectByElementId(id));
}
};
TEST_F(LayoutTableCellTest, ResetColspanIfTooBig) {
SetBodyInnerHTML("<table><td id='cell' colspan='14000'></td></table>");
ASSERT_EQ(GetCellByElementId("cell")->ColSpan(), 1000U);
}
TEST_F(LayoutTableCellTest, DoNotResetColspanJustBelowBoundary) {
SetBodyInnerHTML("<table><td id='cell' colspan='1000'></td></table>");
ASSERT_EQ(GetCellByElementId("cell")->ColSpan(), 1000U);
}
TEST_F(LayoutTableCellTest, ResetRowspanIfTooBig) {
SetBodyInnerHTML("<table><td id='cell' rowspan='70000'></td></table>");
ASSERT_EQ(GetCellByElementId("cell")->RowSpan(), 65534U);
}
TEST_F(LayoutTableCellTest, DoNotResetRowspanJustBelowBoundary) {
SetBodyInnerHTML("<table><td id='cell' rowspan='65534'></td></table>");
ASSERT_EQ(GetCellByElementId("cell")->RowSpan(), 65534U);
}
TEST_F(LayoutTableCellTest,
BackgroundIsKnownToBeOpaqueWithLayerAndCollapsedBorder) {
SetBodyInnerHTML(
"<table style='border-collapse: collapse'>"
" <td id='cell' style='will-change: transform; background-color: blue'>"
" Cell"
" </td>"
"</table>");
EXPECT_FALSE(GetCellByElementId("cell")->BackgroundIsKnownToBeOpaqueInRect(
LayoutRect(0, 0, 1, 1)));
}
TEST_F(LayoutTableCellTest, RepaintContentInTableCell) {
const char* body_content =
"<table id='table' style='position: absolute; left: 1px;'>"
" <tr>"
" <td id='cell'>"
" <div style='display: inline-block; height: 20px; width: 20px'>"
" </td>"
" </tr>"
"</table>";
SetBodyInnerHTML(body_content);
// Create an overflow recalc.
Element* cell = GetDocument().getElementById("cell");
cell->setAttribute(HTMLNames::styleAttr, "outline: 1px solid black;");
// Trigger a layout on the table that doesn't require cell layout.
Element* table = GetDocument().getElementById("table");
table->setAttribute(HTMLNames::styleAttr, "position: absolute; left: 2px;");
GetDocument().View()->UpdateAllLifecyclePhases();
// Check that overflow was calculated on the cell.
auto* input_block = ToLayoutBlock(cell->GetLayoutObject());
LayoutRect rect = input_block->LocalVisualRect();
EXPECT_EQ(LayoutRect(-1, -1, 24, 24), rect);
}
TEST_F(LayoutTableCellTest, IsInStartAndEndColumn) {
SetBodyInnerHTML(
"<table id='table'>"
" <tr>"
" <td id='cell11' colspan='2000'></td>"
" <td id='cell12'></td>"
" <td id='cell13'></td>"
" </tr>"
" <tr>"
" <td id='cell21' rowspan='2'></td>"
" <td id='cell22'></td>"
" <td id='cell23' colspan='2000'></td>"
" </tr>"
" <tr>"
" <td id='cell31'></td>"
" <td id='cell32'></td>"
" </tr>"
"</table>");
const auto* cell11 = GetCellByElementId("cell11");
const auto* cell12 = GetCellByElementId("cell12");
const auto* cell13 = GetCellByElementId("cell13");
const auto* cell21 = GetCellByElementId("cell21");
const auto* cell22 = GetCellByElementId("cell22");
const auto* cell23 = GetCellByElementId("cell23");
const auto* cell31 = GetCellByElementId("cell31");
const auto* cell32 = GetCellByElementId("cell32");
EXPECT_TRUE(IsInStartColumn(cell11));
EXPECT_FALSE(IsInEndColumn(cell11));
EXPECT_FALSE(IsInStartColumn(cell12));
EXPECT_FALSE(IsInEndColumn(cell12));
EXPECT_FALSE(IsInStartColumn(cell13));
EXPECT_TRUE(IsInEndColumn(cell13));
EXPECT_TRUE(IsInStartColumn(cell21));
EXPECT_FALSE(IsInEndColumn(cell21));
EXPECT_FALSE(IsInStartColumn(cell22));
EXPECT_FALSE(IsInEndColumn(cell22));
EXPECT_FALSE(IsInStartColumn(cell23));
EXPECT_TRUE(IsInEndColumn(cell23));
EXPECT_FALSE(IsInStartColumn(cell31));
EXPECT_FALSE(IsInEndColumn(cell31));
EXPECT_FALSE(IsInStartColumn(cell32));
EXPECT_FALSE(IsInEndColumn(cell32));
}
TEST_F(LayoutTableCellTest, IsInStartAndEndColumnRTL) {
SetBodyInnerHTML(
"<style>"
" table { direction: rtl }"
" td { direction: ltr }"
"</style>"
"<table id='table'>"
" <tr>"
" <td id='cell11' colspan='2000'></td>"
" <td id='cell12'></td>"
" <td id='cell13'></td>"
" </tr>"
" <tr>"
" <td id='cell21' rowspan='2'></td>"
" <td id='cell22'></td>"
" <td id='cell23' colspan='2000'></td>"
" </tr>"
" <tr>"
" <td id='cell31'></td>"
" <td id='cell32'></td>"
" </tr>"
"</table>");
const auto* cell11 = GetCellByElementId("cell11");
const auto* cell12 = GetCellByElementId("cell12");
const auto* cell13 = GetCellByElementId("cell13");
const auto* cell21 = GetCellByElementId("cell21");
const auto* cell22 = GetCellByElementId("cell22");
const auto* cell23 = GetCellByElementId("cell23");
const auto* cell31 = GetCellByElementId("cell31");
const auto* cell32 = GetCellByElementId("cell32");
EXPECT_TRUE(IsInStartColumn(cell11));
EXPECT_FALSE(IsInEndColumn(cell11));
EXPECT_FALSE(IsInStartColumn(cell12));
EXPECT_FALSE(IsInEndColumn(cell12));
EXPECT_FALSE(IsInStartColumn(cell13));
EXPECT_TRUE(IsInEndColumn(cell13));
EXPECT_TRUE(IsInStartColumn(cell21));
EXPECT_FALSE(IsInEndColumn(cell21));
EXPECT_FALSE(IsInStartColumn(cell22));
EXPECT_FALSE(IsInEndColumn(cell22));
EXPECT_FALSE(IsInStartColumn(cell23));
EXPECT_TRUE(IsInEndColumn(cell23));
EXPECT_FALSE(IsInStartColumn(cell31));
EXPECT_FALSE(IsInEndColumn(cell31));
EXPECT_FALSE(IsInStartColumn(cell32));
EXPECT_FALSE(IsInEndColumn(cell32));
}
TEST_F(LayoutTableCellTest, BorderWidthsWithCollapsedBorders) {
SetBodyInnerHTML(
"<style>"
" table { border-collapse: collapse }"
" td { border: 0px solid blue; padding: 0 }"
" div { width: 100px; height: 100px }"
"</style>"
"<table>"
" <tr>"
" <td id='cell1' style='border-bottom-width: 10px;"
" outline: 3px solid blue'><div></div></td>"
" <td id='cell2' style='border-width: 3px 15px'><div></div></td>"
" </tr>"
"</table>");
auto* cell1 = GetCellByElementId("cell1");
auto* cell2 = GetCellByElementId("cell2");
EXPECT_TRUE(cell1->Table()->ShouldCollapseBorders());
EXPECT_EQ(0, cell1->BorderLeft());
EXPECT_EQ(7, cell1->BorderRight());
EXPECT_EQ(0, cell1->BorderTop());
EXPECT_EQ(5, cell1->BorderBottom());
EXPECT_EQ(8, cell2->BorderLeft());
EXPECT_EQ(7, cell2->BorderRight());
EXPECT_EQ(2, cell2->BorderTop());
EXPECT_EQ(1, cell2->BorderBottom());
EXPECT_EQ(0u, cell1->CollapsedInnerBorderStart());
EXPECT_EQ(7u, cell1->CollapsedInnerBorderEnd());
EXPECT_EQ(0u, cell1->CollapsedInnerBorderBefore());
EXPECT_EQ(5u, cell1->CollapsedInnerBorderAfter());
EXPECT_EQ(8u, cell2->CollapsedInnerBorderStart());
EXPECT_EQ(7u, cell2->CollapsedInnerBorderEnd());
EXPECT_EQ(2u, cell2->CollapsedInnerBorderBefore());
EXPECT_EQ(1u, cell2->CollapsedInnerBorderAfter());
EXPECT_EQ(0u, cell1->CollapsedOuterBorderStart());
EXPECT_EQ(8u, cell1->CollapsedOuterBorderEnd());
EXPECT_EQ(0u, cell1->CollapsedOuterBorderBefore());
EXPECT_EQ(5u, cell1->CollapsedOuterBorderAfter());
EXPECT_EQ(7u, cell2->CollapsedOuterBorderStart());
EXPECT_EQ(8u, cell2->CollapsedOuterBorderEnd());
EXPECT_EQ(1u, cell2->CollapsedOuterBorderBefore());
EXPECT_EQ(2u, cell2->CollapsedOuterBorderAfter());
ToElement(cell1->Table()->GetNode())
->setAttribute(HTMLNames::styleAttr,
"writing-mode: vertical-rl; direction: rtl");
GetDocument().View()->UpdateAllLifecyclePhases();
EXPECT_EQ(5, cell1->BorderLeft());
EXPECT_EQ(0, cell1->BorderRight());
EXPECT_EQ(8, cell1->BorderTop());
EXPECT_EQ(0, cell1->BorderBottom());
EXPECT_EQ(2, cell2->BorderLeft());
EXPECT_EQ(1, cell2->BorderRight());
EXPECT_EQ(8, cell2->BorderTop());
EXPECT_EQ(7, cell2->BorderBottom());
EXPECT_EQ(0u, cell1->CollapsedInnerBorderStart());
EXPECT_EQ(8u, cell1->CollapsedInnerBorderEnd());
EXPECT_EQ(0u, cell1->CollapsedInnerBorderBefore());
EXPECT_EQ(5u, cell1->CollapsedInnerBorderAfter());
EXPECT_EQ(7u, cell2->CollapsedInnerBorderStart());
EXPECT_EQ(8u, cell2->CollapsedInnerBorderEnd());
EXPECT_EQ(1u, cell2->CollapsedInnerBorderBefore());
EXPECT_EQ(2u, cell2->CollapsedInnerBorderAfter());
EXPECT_EQ(0u, cell1->CollapsedOuterBorderStart());
EXPECT_EQ(7u, cell1->CollapsedOuterBorderEnd());
EXPECT_EQ(0u, cell1->CollapsedOuterBorderBefore());
EXPECT_EQ(5u, cell1->CollapsedOuterBorderAfter());
EXPECT_EQ(8u, cell2->CollapsedOuterBorderStart());
EXPECT_EQ(7u, cell2->CollapsedOuterBorderEnd());
EXPECT_EQ(2u, cell2->CollapsedOuterBorderBefore());
EXPECT_EQ(1u, cell2->CollapsedOuterBorderAfter());
}
} // namespace blink