#include "ui/views/controls/label.h"
#include <stddef.h>
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/command_line.h"
#include "base/format_macros.h"
#include "base/functional/bind.h"
#include "base/i18n/rtl.h"
#include "base/memory/raw_ptr.h"
#include "base/strings/strcat.h"
#include "base/test/gtest_util.h"
#include "base/test/scoped_feature_list.h"
#include "build/build_config.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/accessibility/accessibility_features.h"
#include "ui/accessibility/ax_enums.mojom.h"
#include "ui/accessibility/ax_node_data.h"
#include "ui/accessibility/platform/ax_platform_for_test.h"
#include "ui/base/clipboard/clipboard.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/base/metadata/metadata_header_macros.h"
#include "ui/base/metadata/metadata_impl_macros.h"
#include "ui/base/ui_base_switches.h"
#include "ui/color/color_provider.h"
#include "ui/compositor/canvas_painter.h"
#include "ui/compositor/layer.h"
#include "ui/events/base_event_utils.h"
#include "ui/events/test/event_generator.h"
#include "ui/gfx/canvas.h"
#include "ui/gfx/render_text.h"
#include "ui/gfx/text_constants.h"
#include "ui/gfx/text_elider.h"
#include "ui/strings/grit/ui_strings.h"
#include "ui/views/accessibility/view_accessibility.h"
#include "ui/views/background.h"
#include "ui/views/border.h"
#include "ui/views/controls/base_control_test_widget.h"
#include "ui/views/controls/link.h"
#include "ui/views/layout/layout_types.h"
#include "ui/views/style/typography.h"
#include "ui/views/test/ax_event_counter.h"
#include "ui/views/test/focus_manager_test.h"
#include "ui/views/test/view_metadata_test_utils.h"
#include "ui/views/test/views_test_base.h"
#include "ui/views/widget/unique_widget_ptr.h"
#include "ui/views/widget/widget.h"
#include "ui/views/widget/widget_utils.h"
using base::WideToUTF16;
namespace views {
namespace {
#if BUILDFLAG(IS_MAC)
const int kControlCommandModifier = ui::EF_COMMAND_DOWN;
#else
const int kControlCommandModifier = ui::EF_CONTROL_DOWN;
#endif
const int kMinTextDimension = 4;
class TestLabel : public Label {
METADATA_HEADER(TestLabel, Label)
public:
TestLabel() : Label(u"TestLabel") { SizeToPreferredSize(); }
TestLabel(const TestLabel&) = delete;
TestLabel& operator=(const TestLabel&) = delete;
int schedule_paint_count() const { return schedule_paint_count_; }
void SimulatePaint() {
SkBitmap bitmap;
SkColor color = SK_ColorTRANSPARENT;
Paint(PaintInfo::CreateRootPaintInfo(
ui::CanvasPainter(&bitmap, bounds().size(), 1.f, color, false)
.context(),
bounds().size()));
}
void OnDidSchedulePaint(const gfx::Rect& r) override {
++schedule_paint_count_;
Label::OnDidSchedulePaint(r);
}
private:
int schedule_paint_count_ = 0;
};
BEGIN_METADATA(TestLabel)
END_METADATA
void SetRTL(bool rtl) {
base::i18n::SetICUDefaultLocale(rtl ? "he" : "en");
EXPECT_EQ(rtl, base::i18n::IsRTL());
}
std::u16string GetClipboardText(ui::ClipboardBuffer clipboard_buffer) {
std::u16string clipboard_text;
ui::Clipboard::GetForCurrentThread()->ReadText(
clipboard_buffer, nullptr, &clipboard_text);
return clipboard_text;
}
std::u16string ToRTL(const std::string& ascii) {
std::u16string rtl;
for (char c : ascii) {
if (c >= '0' && c <= '6') {
rtl += static_cast<char16_t>(u'א' + (c - '0'));
} else {
rtl += static_cast<char16_t>(c);
}
}
return rtl;
}
}
class LabelTest : public test::BaseControlTestWidget {
public:
LabelTest() = default;
LabelTest(const LabelTest&) = delete;
LabelTest& operator=(const LabelTest&) = delete;
~LabelTest() override = default;
void MockAXModeAdded() {
ui::AXMode mode =
ui::AXPlatformForTest::GetInstance().GetAccessibilityMode();
widget()->OnAXModeAdded(mode);
}
void TearDown() override {
label_ = nullptr;
test::BaseControlTestWidget::TearDown();
}
protected:
void CreateWidgetContent(View* container) override {
label_ = container->AddChildView(std::make_unique<Label>());
}
Label* label() { return label_; }
private:
raw_ptr<Label> label_ = nullptr;
};
class LabelSelectionTest : public LabelTest {
public:
static constexpr bool kExtends =
gfx::RenderText::kDragToEndIfOutsideVerticalBounds;
enum { NW, NORTH, NE, SE, SOUTH, SW };
LabelSelectionTest() = default;
LabelSelectionTest(const LabelSelectionTest&) = delete;
LabelSelectionTest& operator=(const LabelSelectionTest&) = delete;
void SetUp() override {
LabelTest::SetUp();
event_generator_ =
std::make_unique<ui::test::EventGenerator>(GetRootWindow(widget()));
}
protected:
View* GetFocusedView() {
return widget()->GetFocusManager()->GetFocusedView();
}
void PerformMousePress(const gfx::Point& point) {
ui::MouseEvent pressed_event = ui::MouseEvent(
ui::EventType::kMousePressed, point, point, ui::EventTimeForNow(),
ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON);
label()->OnMousePressed(pressed_event);
}
void PerformMouseRelease(const gfx::Point& point) {
ui::MouseEvent released_event = ui::MouseEvent(
ui::EventType::kMouseReleased, point, point, ui::EventTimeForNow(),
ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON);
label()->OnMouseReleased(released_event);
}
void PerformClick(const gfx::Point& point) {
PerformMousePress(point);
PerformMouseRelease(point);
}
void PerformMouseDragTo(const gfx::Point& point) {
ui::MouseEvent drag(ui::EventType::kMouseDragged, point, point,
ui::EventTimeForNow(), ui::EF_LEFT_MOUSE_BUTTON, 0);
label()->OnMouseDragged(drag);
}
void SimulatePaint() {
gfx::Canvas canvas;
label()->OnPaint(&canvas);
}
gfx::Point GetCursorPoint(uint32_t index) {
SimulatePaint();
gfx::RenderText* render_text =
label()->GetRenderTextForSelectionController();
if (!render_text->multiline()) {
return render_text
->GetCursorBounds(gfx::SelectionModel(index, gfx::CURSOR_FORWARD),
true)
.left_center();
}
auto bounds = render_text->GetSubstringBounds({index, index + 1});
DCHECK_EQ(1u, bounds.size());
const bool rtl =
render_text->GetDisplayTextDirection() == base::i18n::RIGHT_TO_LEFT;
return rtl ? bounds[0].right_center() + gfx::Vector2d(-1, 0)
: bounds[0].left_center() + gfx::Vector2d(1, 0);
}
size_t GetLineCount() {
SimulatePaint();
return label()->GetRenderTextForSelectionController()->GetNumLines();
}
std::u16string_view GetSelectedText() { return label()->GetSelectedText(); }
ui::test::EventGenerator* event_generator() { return event_generator_.get(); }
bool IsMenuCommandEnabled(int command_id) {
return label()->IsCommandIdEnabled(command_id);
}
private:
std::unique_ptr<ui::test::EventGenerator> event_generator_;
};
TEST_F(LabelTest, Metadata) {
label()->SetMultiLine(true);
test::TestViewMetadata(label());
}
TEST_F(LabelTest, FontPropertySymbol) {
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
std::string font_name("Courier New");
#else
std::string font_name("symbol");
#endif
gfx::Font font(font_name, 26);
label()->SetFontList(gfx::FontList(font));
gfx::Font font_used = label()->font_list().GetPrimaryFont();
EXPECT_EQ(font_name, font_used.GetFontName());
EXPECT_EQ(26, font_used.GetFontSize());
}
TEST_F(LabelTest, FontPropertyArial) {
std::string font_name("arial");
gfx::Font font(font_name, 30);
label()->SetFontList(gfx::FontList(font));
gfx::Font font_used = label()->font_list().GetPrimaryFont();
EXPECT_EQ(font_name, font_used.GetFontName());
EXPECT_EQ(30, font_used.GetFontSize());
}
TEST_F(LabelTest, TextProperty) {
std::u16string test_text(u"A random string.");
label()->SetText(test_text);
EXPECT_EQ(test_text, label()->GetText());
}
TEST_F(LabelTest, TextStyleProperty) {
label()->SetTextStyle(views::style::STYLE_DISABLED);
EXPECT_EQ(views::style::STYLE_DISABLED, label()->GetTextStyle());
}
TEST_F(LabelTest, ColorProperty) {
SkColor color = SkColorSetARGB(20, 40, 10, 5);
label()->SetAutoColorReadabilityEnabled(false);
label()->SetEnabledColor(color);
EXPECT_EQ(color, label()->GetEnabledColor());
}
TEST_F(LabelTest, ColorPropertyOnEnabledColorIdChange) {
const auto color = label()->GetWidget()->GetColorProvider()->GetColor(
ui::kColorPrimaryForeground);
label()->SetAutoColorReadabilityEnabled(false);
label()->SetEnabledColor(ui::kColorPrimaryForeground);
EXPECT_EQ(color, label()->GetEnabledColor());
label()->SetEnabledColor(ui::kColorAccent);
EXPECT_EQ(
label()->GetWidget()->GetColorProvider()->GetColor(ui::kColorAccent),
label()->GetEnabledColor());
}
TEST_F(LabelTest, BackgroundColor) {
EXPECT_EQ(widget()->GetColorProvider()->GetColor(ui::kColorDialogBackground),
label()->GetBackgroundColor());
label()->SetBackgroundColor(SK_ColorBLUE);
EXPECT_EQ(SK_ColorBLUE, label()->GetBackgroundColor());
}
TEST_F(LabelTest, BackgroundColorId) {
EXPECT_EQ(widget()->GetColorProvider()->GetColor(ui::kColorDialogBackground),
label()->GetBackgroundColor());
label()->SetBackgroundColor(ui::kColorAlertHighSeverity);
EXPECT_EQ(widget()->GetColorProvider()->GetColor(ui::kColorAlertHighSeverity),
label()->GetBackgroundColor());
label()->SetBackgroundColor(SK_ColorBLUE);
EXPECT_EQ(SK_ColorBLUE, label()->GetBackgroundColor());
}
TEST_F(LabelTest, AlignmentProperty) {
const bool was_rtl = base::i18n::IsRTL();
for (size_t i = 0; i < 2; ++i) {
SetRTL(!base::i18n::IsRTL());
bool reverse_alignment = base::i18n::IsRTL();
label()->SetHorizontalAlignment(gfx::ALIGN_RIGHT);
EXPECT_EQ(reverse_alignment ? gfx::ALIGN_LEFT : gfx::ALIGN_RIGHT,
label()->GetHorizontalAlignment());
label()->SetHorizontalAlignment(gfx::ALIGN_LEFT);
EXPECT_EQ(reverse_alignment ? gfx::ALIGN_RIGHT : gfx::ALIGN_LEFT,
label()->GetHorizontalAlignment());
label()->SetHorizontalAlignment(gfx::ALIGN_CENTER);
EXPECT_EQ(gfx::ALIGN_CENTER, label()->GetHorizontalAlignment());
for (size_t j = 0; j < 2; ++j) {
label()->SetHorizontalAlignment(gfx::ALIGN_TO_HEAD);
const bool rtl = j == 0;
label()->SetText(rtl ? u"\x5d0" : u"A");
EXPECT_EQ(gfx::ALIGN_TO_HEAD, label()->GetHorizontalAlignment());
}
}
EXPECT_EQ(was_rtl, base::i18n::IsRTL());
}
TEST_F(LabelTest, MinimumSizeRespectsLineHeight) {
std::u16string text(u"This is example text.");
label()->SetText(text);
const gfx::Size minimum_size = label()->GetMinimumSize();
const int expected_height = minimum_size.height() + 10;
label()->SetLineHeight(expected_height);
EXPECT_EQ(expected_height, label()->GetMinimumSize().height());
}
TEST_F(LabelTest, MinimumSizeRespectsLineHeightMultiline) {
std::u16string text(u"This is example text.");
label()->SetText(text);
label()->SetMultiLine(true);
const gfx::Size minimum_size = label()->GetMinimumSize();
const int expected_height = minimum_size.height() + 10;
label()->SetLineHeight(expected_height);
EXPECT_EQ(expected_height, label()->GetMinimumSize().height());
}
TEST_F(LabelTest, MinimumSizeRespectsLineHeightWithInsets) {
std::u16string text(u"This is example text.");
label()->SetText(text);
const gfx::Size minimum_size = label()->GetMinimumSize();
int expected_height = minimum_size.height() + 10;
label()->SetLineHeight(expected_height);
constexpr auto kInsets = gfx::Insets::TLBR(2, 3, 4, 5);
expected_height += kInsets.height();
label()->SetBorder(CreateEmptyBorder(kInsets));
EXPECT_EQ(expected_height, label()->GetMinimumSize().height());
}
TEST_F(LabelTest, MinimumSizeRespectsLineHeightMultilineWithInsets) {
std::u16string text(u"This is example text.");
label()->SetText(text);
label()->SetMultiLine(true);
const gfx::Size minimum_size = label()->GetMinimumSize();
int expected_height = minimum_size.height() + 10;
label()->SetLineHeight(expected_height);
constexpr auto kInsets = gfx::Insets::TLBR(2, 3, 4, 5);
expected_height += kInsets.height();
label()->SetBorder(CreateEmptyBorder(kInsets));
EXPECT_EQ(expected_height, label()->GetMinimumSize().height());
}
TEST_F(LabelTest, ElideBehavior) {
std::u16string text(u"This is example text.");
label()->SetText(text);
EXPECT_EQ(gfx::ELIDE_TAIL, label()->GetElideBehavior());
gfx::Size size = label()->GetPreferredSize({});
label()->SetBoundsRect(gfx::Rect(size));
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
size.set_width(size.width() / 2);
label()->SetBoundsRect(gfx::Rect(size));
EXPECT_GT(text.size(), label()->GetDisplayTextForTesting().size());
label()->SetElideBehavior(gfx::NO_ELIDE);
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
}
TEST_F(LabelTest, ElideBehaviorMinimumWidth) {
std::u16string text(u"This is example text.");
label()->SetText(text);
EXPECT_EQ(gfx::ELIDE_TAIL, label()->GetElideBehavior());
gfx::Size size = label()->GetMinimumSize();
EXPECT_EQ(gfx::Canvas::GetStringWidth(std::u16string(gfx::kEllipsisUTF16),
label()->font_list()),
size.width());
label()->SetSize(label()->GetMinimumSize());
EXPECT_GT(text.length(), label()->GetDisplayTextForTesting().length());
label()->SetElideBehavior(gfx::TRUNCATE);
label()->SetSize(gfx::Size(10, 10));
size = label()->GetMinimumSize();
EXPECT_LT(size.width(), label()->size().width());
EXPECT_GT(text.length(), label()->GetDisplayTextForTesting().length());
EXPECT_FALSE(label()->GetMultiLine());
label()->SetElideBehavior(gfx::NO_ELIDE);
size = label()->GetMinimumSize();
EXPECT_EQ(text.length(), label()->GetDisplayTextForTesting().length());
label()->SetSize(label()->GetMinimumSize());
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
}
TEST_F(LabelTest, MultiLineProperty) {
EXPECT_FALSE(label()->GetMultiLine());
label()->SetMultiLine(true);
EXPECT_TRUE(label()->GetMultiLine());
label()->SetMultiLine(false);
EXPECT_FALSE(label()->GetMultiLine());
}
TEST_F(LabelTest, ObscuredProperty) {
std::u16string test_text(u"Password!");
label()->SetText(test_text);
label()->SizeToPreferredSize();
EXPECT_FALSE(label()->GetObscured());
EXPECT_EQ(test_text, label()->GetDisplayTextForTesting());
EXPECT_EQ(test_text, label()->GetText());
label()->SetObscured(true);
label()->SizeToPreferredSize();
EXPECT_TRUE(label()->GetObscured());
EXPECT_EQ(std::u16string(test_text.size(),
gfx::RenderText::kPasswordReplacementChar),
label()->GetDisplayTextForTesting());
EXPECT_EQ(test_text, label()->GetText());
label()->SetText(test_text + test_text);
label()->SizeToPreferredSize();
EXPECT_EQ(std::u16string(test_text.size() * 2,
gfx::RenderText::kPasswordReplacementChar),
label()->GetDisplayTextForTesting());
EXPECT_EQ(test_text + test_text, label()->GetText());
label()->SetObscured(false);
label()->SizeToPreferredSize();
EXPECT_FALSE(label()->GetObscured());
EXPECT_EQ(test_text + test_text, label()->GetDisplayTextForTesting());
EXPECT_EQ(test_text + test_text, label()->GetText());
}
TEST_F(LabelTest, ObscuredSurrogatePair) {
const std::u16string kTestText = u"𝄞";
label()->SetText(kTestText);
label()->SetObscured(true);
label()->SizeToPreferredSize();
EXPECT_EQ(std::u16string(1, gfx::RenderText::kPasswordReplacementChar),
label()->GetDisplayTextForTesting());
EXPECT_EQ(kTestText, label()->GetText());
}
TEST_F(LabelTest, MultilinePreferredSizeWithConstraintTest) {
label()->SetText(u"This is an example.");
const gfx::Size single_line_size =
label()->GetPreferredSize({});
label()->SetMultiLine(true);
const gfx::Size multi_line_size_unbounded =
label()->GetPreferredSize({});
EXPECT_EQ(single_line_size, multi_line_size_unbounded);
const gfx::Size multi_line_size_bounded = label()->GetPreferredSize(
{single_line_size.width() / 2, {}});
EXPECT_GT(multi_line_size_unbounded.width(), multi_line_size_bounded.width());
EXPECT_LT(multi_line_size_unbounded.height(),
multi_line_size_bounded.height());
const int layout_width = multi_line_size_unbounded.width() / 3;
label()->SetBounds(0, 0, layout_width,
label()->GetHeightForWidth(layout_width));
const gfx::Size multi_line_size_unbounded2 =
label()->GetPreferredSize({});
const gfx::Size multi_line_size_bounded2 = label()->GetPreferredSize(
{single_line_size.width() / 2, {}});
EXPECT_EQ(multi_line_size_unbounded, multi_line_size_unbounded2);
EXPECT_EQ(multi_line_size_bounded, multi_line_size_bounded2);
}
TEST_F(LabelTest, SingleLineGetHeightForWidth) {
const int line_height = label()->GetLineHeight();
EXPECT_EQ(line_height, label()->GetHeightForWidth(100));
label()->SetText(u"This is an example.");
const int width = label()->GetPreferredSize({}).width();
EXPECT_EQ(line_height, label()->GetHeightForWidth(width));
EXPECT_EQ(line_height, label()->GetHeightForWidth(width * 2));
EXPECT_EQ(line_height, label()->GetHeightForWidth(width / 2));
EXPECT_EQ(line_height, label()->GetHeightForWidth(0));
}
TEST_F(LabelTest, MultiLineGetHeightForWidth) {
label()->SetMultiLine(true);
const int line_height = label()->GetLineHeight();
EXPECT_EQ(line_height, label()->GetHeightForWidth(100));
label()->SetText(u"This is an example.");
const int width = label()->GetPreferredSize({}).width();
EXPECT_EQ(line_height, label()->GetHeightForWidth(width));
EXPECT_EQ(line_height, label()->GetHeightForWidth(width * 2));
const int height_for_half_width = label()->GetHeightForWidth(width / 2);
EXPECT_GT(height_for_half_width, line_height);
EXPECT_GT(label()->GetHeightForWidth(width / 4), height_for_half_width);
}
TEST_F(LabelTest, TooltipProperty) {
label()->SetText(u"My cool string.");
EXPECT_EQ(label()->GetText(), label()->GetTooltipText());
label()->SetHandlesTooltips(false);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetHandlesTooltips(true);
std::u16string tooltip_text(u"The tooltip!");
label()->SetCustomTooltipText(tooltip_text);
EXPECT_EQ(tooltip_text, label()->GetTooltipText());
label()->SetHandlesTooltips(false);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetHandlesTooltips(true);
label()->SetCustomTooltipText(std::u16string());
EXPECT_EQ(label()->GetText(), label()->GetTooltipText());
label()->SetHandlesTooltips(false);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetHandlesTooltips(true);
label()->SetBounds(0, 0, 1000, 40);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetBounds(
0, 0, 1000, label()->GetPreferredSize(SizeBounds(1000, {})).height() / 2);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetCustomTooltipText(tooltip_text);
EXPECT_EQ(tooltip_text, label()->GetTooltipText());
label()->SetCustomTooltipText(std::u16string());
label()->SetBounds(0, 0, 10, 10);
EXPECT_FALSE(label()->GetTooltipText().empty());
label()->SetObscured(true);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetObscured(false);
EXPECT_FALSE(label()->GetTooltipText().empty());
label()->SetMultiLine(true);
EXPECT_FALSE(label()->GetTooltipText().empty());
label()->SetBounds(0, 0, 1000, 1000);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->SetCustomTooltipText(tooltip_text);
EXPECT_EQ(tooltip_text, label()->GetTooltipText());
label()->SetCustomTooltipText(std::u16string());
}
TEST_F(LabelTest, TooltipPropertyAccessibility) {
label()->SetText(u"My cool string.");
ui::AXNodeData ax_data;
EXPECT_EQ(label()->GetText(), label()->GetTooltipText());
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_FALSE(
ax_data.HasStringAttribute(ax::mojom::StringAttribute::kDescription));
EXPECT_EQ(label()->GetText(),
ax_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
std::u16string tooltip_text(u"The tooltip!");
ax_data = ui::AXNodeData();
label()->SetCustomTooltipText(tooltip_text);
EXPECT_EQ(tooltip_text, label()->GetTooltipText());
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_EQ(tooltip_text, ax_data.GetString16Attribute(
ax::mojom::StringAttribute::kDescription));
label()->SetHandlesTooltips(false);
ax_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_TRUE(label()->GetTooltipText().empty());
EXPECT_FALSE(
ax_data.HasStringAttribute(ax::mojom::StringAttribute::kDescription));
label()->SetHandlesTooltips(true);
label()->SetCustomTooltipText(std::u16string());
ax_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_EQ(label()->GetText(), label()->GetTooltipText());
EXPECT_FALSE(
ax_data.HasStringAttribute(ax::mojom::StringAttribute::kDescription));
label()->SetCustomTooltipText(tooltip_text);
ax_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_EQ(tooltip_text, label()->GetTooltipText());
EXPECT_EQ(tooltip_text, ax_data.GetString16Attribute(
ax::mojom::StringAttribute::kDescription));
label()->SetCustomTooltipText(std::u16string());
ax_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_FALSE(
ax_data.HasStringAttribute(ax::mojom::StringAttribute::kDescription));
ax_data = ui::AXNodeData();
label()->SetCustomTooltipText(tooltip_text);
label()->SetObscured(true);
EXPECT_TRUE(label()->GetTooltipText().empty());
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_FALSE(
ax_data.HasStringAttribute(ax::mojom::StringAttribute::kDescription));
label()->SetObscured(false);
EXPECT_FALSE(label()->GetTooltipText().empty());
ax_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&ax_data);
EXPECT_EQ(tooltip_text, ax_data.GetString16Attribute(
ax::mojom::StringAttribute::kDescription));
}
TEST_F(LabelTest, Accessibility) {
const std::u16string accessible_name = u"A11y text.";
label()->SetText(u"Displayed text.");
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(ax::mojom::Role::kStaticText, node_data.role);
EXPECT_EQ(label()->GetText(),
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
EXPECT_FALSE(
node_data.HasIntAttribute(ax::mojom::IntAttribute::kRestriction));
label()->GetViewAccessibility().SetName(accessible_name);
node_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(accessible_name,
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
EXPECT_NE(label()->GetText(),
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
label()->SetText(u"Different displayed Text.");
node_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(accessible_name,
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
EXPECT_NE(label()->GetText(),
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
label()->GetViewAccessibility().SetName(u"");
node_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(label()->GetText(),
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
label()->SetText(u"");
node_data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(label()->GetText(),
node_data.GetString16Attribute(ax::mojom::StringAttribute::kName));
}
TEST_F(LabelTest, SetTextNotifiesAccessibilityEvent) {
test::AXEventCounter counter(views::AXUpdateNotifier::Get());
EXPECT_EQ(0, counter.GetCount(ax::mojom::Event::kTextChanged));
label()->SetText(u"Example");
EXPECT_EQ(u"Example", label()->GetViewAccessibility().GetCachedName());
EXPECT_EQ(1, counter.GetCount(ax::mojom::Event::kTextChanged));
label()->GetViewAccessibility().SetName(u"Name");
EXPECT_EQ(2, counter.GetCount(ax::mojom::Event::kTextChanged));
label()->SetText(u"Example2");
EXPECT_EQ(u"Name", label()->GetViewAccessibility().GetCachedName());
EXPECT_EQ(2, counter.GetCount(ax::mojom::Event::kTextChanged));
}
TEST_F(LabelTest, TextChangeWithoutLayout) {
label()->SetText(u"Example");
label()->SetBounds(0, 0, 200, 200);
gfx::Canvas canvas(gfx::Size(200, 200), 1.0f, true);
label()->OnPaint(&canvas);
EXPECT_TRUE(label()->display_text_);
EXPECT_EQ(u"Example", label()->display_text_->GetDisplayText());
label()->SetText(u"Altered");
label()->OnPaint(&canvas);
EXPECT_TRUE(label()->display_text_);
EXPECT_EQ(u"Altered", label()->display_text_->GetDisplayText());
}
TEST_F(LabelTest, AccessibleNameAndRole) {
label()->SetText(u"Text");
EXPECT_EQ(label()->GetViewAccessibility().GetCachedName(), u"Text");
EXPECT_EQ(label()->GetViewAccessibility().GetCachedRole(),
ax::mojom::Role::kStaticText);
ui::AXNodeData data;
label()->GetViewAccessibility().GetAccessibleNodeData(&data);
EXPECT_EQ(data.GetString16Attribute(ax::mojom::StringAttribute::kName),
u"Text");
EXPECT_EQ(data.role, ax::mojom::Role::kStaticText);
label()->SetTextContext(style::CONTEXT_DIALOG_TITLE);
EXPECT_EQ(label()->GetViewAccessibility().GetCachedName(), u"Text");
EXPECT_EQ(label()->GetViewAccessibility().GetCachedRole(),
ax::mojom::Role::kTitleBar);
data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&data);
EXPECT_EQ(data.GetString16Attribute(ax::mojom::StringAttribute::kName),
u"Text");
EXPECT_EQ(data.role, ax::mojom::Role::kTitleBar);
label()->SetText(u"New Text");
label()->GetViewAccessibility().SetRole(ax::mojom::Role::kLink);
EXPECT_EQ(label()->GetViewAccessibility().GetCachedName(), u"New Text");
EXPECT_EQ(label()->GetViewAccessibility().GetCachedRole(),
ax::mojom::Role::kLink);
data = ui::AXNodeData();
label()->GetViewAccessibility().GetAccessibleNodeData(&data);
EXPECT_EQ(data.GetString16Attribute(ax::mojom::StringAttribute::kName),
u"New Text");
EXPECT_EQ(data.role, ax::mojom::Role::kLink);
}
TEST_F(LabelTest, EmptyLabelSizing) {
const gfx::Size expected_size(0, label()->font_list().GetHeight());
EXPECT_EQ(expected_size, label()->GetPreferredSize({}));
label()->SetMultiLine(!label()->GetMultiLine());
EXPECT_EQ(expected_size, label()->GetPreferredSize({}));
}
TEST_F(LabelTest, SingleLineSizing) {
label()->SetText(u"A not so random string in one line.");
const gfx::Size size = label()->GetPreferredSize({});
EXPECT_GT(size.height(), kMinTextDimension);
EXPECT_GT(size.width(), kMinTextDimension);
label()->SetSize(gfx::Size(size.width() / 2, size.height() / 2));
EXPECT_EQ(size, label()->GetPreferredSize(SizeBounds(label()->size())));
const auto border = gfx::Insets::TLBR(10, 20, 30, 40);
label()->SetBorder(CreateEmptyBorder(border));
const gfx::Size size_with_border =
label()->GetPreferredSize(SizeBounds(label()->size()));
EXPECT_EQ(size_with_border.height(), size.height() + border.height());
EXPECT_EQ(size_with_border.width(), size.width() + border.width());
EXPECT_EQ(size.height() + border.height(),
label()->GetHeightForWidth(size_with_border.width()));
}
TEST_F(LabelTest, MultilineSmallAvailableWidthSizing) {
label()->SetMultiLine(true);
label()->SetAllowCharacterBreak(true);
label()->SetText(u"Too Wide.");
gfx::Size required_size = label()->GetPreferredSize({});
for (int i = 1; i < required_size.width(); ++i) {
EXPECT_GT(label()->GetHeightForWidth(i), 0);
}
}
TEST_F(LabelTest, PreferredSizeForAllowCharacterBreak) {
label()->SetText(u"Example");
gfx::Size preferred_size = label()->GetPreferredSize({});
label()->SetMultiLine(true);
label()->SetAllowCharacterBreak(true);
EXPECT_EQ(preferred_size, label()->GetPreferredSize({}));
}
TEST_F(LabelTest, MultiLineSizing) {
label()->SetText(u"A random string\nwith multiple lines\nand returns!");
label()->SetMultiLine(true);
gfx::Size required_size = label()->GetPreferredSize({});
EXPECT_GT(required_size.height(), kMinTextDimension);
EXPECT_GT(required_size.width(), kMinTextDimension);
label()->SizeToFit(0);
int required_width = label()->GetLocalBounds().width();
EXPECT_GT(required_width, kMinTextDimension);
label()->SizeToFit(required_width - 1);
int constrained_width = label()->GetLocalBounds().width();
EXPECT_LT(constrained_width, required_width);
EXPECT_GT(constrained_width, kMinTextDimension);
label()->SizeToFit(required_width);
EXPECT_EQ(required_width, label()->GetLocalBounds().width());
label()->SizeToFit(0);
int required_height = label()->GetHeightForWidth(required_width);
EXPECT_GT(required_height, kMinTextDimension);
int height_for_constrained_width =
label()->GetHeightForWidth(constrained_width);
EXPECT_GT(height_for_constrained_width, required_height);
EXPECT_EQ(height_for_constrained_width,
label()->GetHeightForWidth(required_width - 1));
auto border = gfx::Insets::TLBR(10, 20, 30, 40);
label()->SetBorder(CreateEmptyBorder(border));
label()->SizeToFit(0);
int required_width_with_border = label()->GetLocalBounds().width();
EXPECT_EQ(required_width_with_border, required_width + border.width());
int required_height_with_border =
label()->GetHeightForWidth(required_width_with_border);
EXPECT_EQ(required_height_with_border, required_height + border.height());
int height1 = label()->GetHeightForWidth(required_width_with_border - 1);
EXPECT_GT(height1, required_height_with_border);
EXPECT_EQ(height1, height_for_constrained_width + border.height());
gfx::Size required_size_with_border = label()->GetPreferredSize({});
EXPECT_EQ(required_size_with_border.height(),
required_size.height() + border.height());
EXPECT_EQ(required_size_with_border.width(),
required_size.width() + border.width());
}
#if !BUILDFLAG(IS_MAC)
TEST_F(LabelTest, MultiLineSetMaxLines) {
label()->SetText(u"first line\nsecond line\nthird line");
label()->SetMultiLine(true);
gfx::Size string_size = label()->GetPreferredSize({});
label()->SetMaxLines(2);
gfx::Size two_line_size = label()->GetPreferredSize({});
EXPECT_EQ(string_size.width(), two_line_size.width());
EXPECT_GT(string_size.height(), two_line_size.height());
int height = label()->GetHeightForWidth(string_size.width() / 2);
EXPECT_EQ(height, two_line_size.height());
label()->SetText(u"A long string that will be wrapped");
label()->SetMaxLines(0);
label()->SizeToFit(0);
label()->SizeToFit(label()->GetPreferredSize({}).width() / 4);
string_size = label()->GetPreferredSize({});
label()->SetMaxLines(2);
two_line_size = label()->GetPreferredSize({});
EXPECT_EQ(string_size.width(), two_line_size.width());
EXPECT_GT(string_size.height(), two_line_size.height());
label()->SetMaxLines(0);
label()->SizeToFit(0);
label()->SetMaximumWidth(label()->GetPreferredSize({}).width() / 4);
string_size = label()->GetPreferredSize({});
label()->SetMaxLines(2);
two_line_size = label()->GetPreferredSize({});
EXPECT_EQ(string_size.width(), two_line_size.width());
EXPECT_GT(string_size.height(), two_line_size.height());
const auto border = gfx::Insets::TLBR(1, 2, 3, 4);
label()->SetBorder(CreateEmptyBorder(border));
EXPECT_EQ(two_line_size.height() + border.height(),
label()->GetPreferredSize({}).height());
}
#endif
TEST_F(LabelTest, MultiLineSizingWithElide) {
const std::u16string text =
u"A random string\nwith multiple lines\nand returns!";
label()->SetText(text);
label()->SetMultiLine(true);
gfx::Size required_size = label()->GetPreferredSize({});
EXPECT_GT(required_size.height(), kMinTextDimension);
EXPECT_GT(required_size.width(), kMinTextDimension);
label()->SetBoundsRect(gfx::Rect(required_size));
label()->SetElideBehavior(gfx::ELIDE_TAIL);
EXPECT_EQ(required_size,
label()->GetPreferredSize(SizeBounds(required_size)));
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
gfx::Size narrow_size =
label()->GetPreferredSize(SizeBounds(required_size.width() - 1, {}));
EXPECT_GT(required_size.width(), narrow_size.width());
EXPECT_LT(required_size.height(), narrow_size.height());
label()->SetBounds(0, 0, narrow_size.width() - 1, narrow_size.height());
EXPECT_EQ(narrow_size, label()->GetPreferredSize(
SizeBounds(required_size.width() - 1, {})));
gfx::Canvas canvas;
label()->OnPaint(&canvas);
EXPECT_EQ(narrow_size, label()->GetPreferredSize(
SizeBounds(required_size.width() - 1, {})));
}
TEST_F(LabelTest, GetTooltipHandlerForPoint) {
label()->SetText(u"A string that's long enough to exceed the bounds");
label()->SetBounds(0, 0, 10, 10);
ASSERT_TRUE(label()->GetHandlesTooltips());
EXPECT_EQ(label(), label()->GetTooltipHandlerForPoint(gfx::Point(2, 2)));
label()->SetHandlesTooltips(false);
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(2, 2)));
label()->SetHandlesTooltips(true);
label()->SetBounds(0, 0, 500, 50);
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(2, 2)));
label()->SetCustomTooltipText(u"a tooltip");
EXPECT_EQ(label(), label()->GetTooltipHandlerForPoint(gfx::Point(2, 2)));
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(2, 51)));
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(-1, 20)));
label()->SetHandlesTooltips(false);
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(2, 2)));
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(2, 51)));
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(-1, 20)));
label()->SetHandlesTooltips(true);
label()->SetBounds(2, 2, 10, 10);
EXPECT_EQ(label(), label()->GetTooltipHandlerForPoint(gfx::Point(1, 5)));
EXPECT_FALSE(label()->GetTooltipHandlerForPoint(gfx::Point(3, 11)));
}
TEST_F(LabelTest, ResetRenderTextData) {
label()->SetText(u"Example");
label()->SizeToPreferredSize();
gfx::Size preferred_size = label()->GetPreferredSize({});
EXPECT_NE(gfx::Size(), preferred_size);
EXPECT_FALSE(label()->display_text_);
gfx::Canvas canvas(preferred_size, 1.0f, true);
label()->OnPaint(&canvas);
EXPECT_TRUE(label()->display_text_);
label()->SetVisible(false);
EXPECT_FALSE(label()->display_text_);
EXPECT_EQ(u"Example", label()->GetText());
EXPECT_FALSE(label()->display_text_);
EXPECT_EQ(preferred_size, label()->GetPreferredSize({}));
EXPECT_FALSE(label()->display_text_);
label()->SetVisible(true);
EXPECT_FALSE(label()->display_text_);
label()->OnPaint(&canvas);
EXPECT_TRUE(label()->display_text_);
label()->SetBounds(0, 0, 10, 10);
EXPECT_FALSE(label()->display_text_);
label()->OnPaint(&canvas);
EXPECT_TRUE(label()->display_text_);
}
TEST_F(LabelTest, MultilineSupportedRenderText) {
label()->SetText(u"Example of\nmultilined label");
label()->SetMultiLine(true);
label()->SizeToPreferredSize();
gfx::Canvas canvas(label()->GetPreferredSize({}), 1.0f, true);
label()->OnPaint(&canvas);
ASSERT_TRUE(label()->display_text_);
EXPECT_EQ(2u, label()->display_text_->GetNumLines());
}
TEST_F(LabelTest, NoSchedulePaintInOnPaint) {
TestLabel label;
int count = 0;
const auto expect_paint_count_increased = [&]() {
EXPECT_GT(label.schedule_paint_count(), count);
count = label.schedule_paint_count();
};
expect_paint_count_increased();
label.SimulatePaint();
EXPECT_EQ(count, label.schedule_paint_count());
label.SetEnabled(false);
expect_paint_count_increased();
label.SetText(base::StrCat({label.GetText(), u"Changed"}));
expect_paint_count_increased();
label.SizeToPreferredSize();
expect_paint_count_increased();
label.SetEnabledColor(SK_ColorBLUE);
expect_paint_count_increased();
label.SimulatePaint();
EXPECT_EQ(count, label.schedule_paint_count());
}
TEST_F(LabelTest, EmptyLabel) {
label()->SetFocusBehavior(View::FocusBehavior::ALWAYS);
label()->RequestFocus();
label()->SizeToPreferredSize();
EXPECT_TRUE(label()->size().IsEmpty());
Link concrete_link;
EXPECT_TRUE(concrete_link.GetPreferredSize({}).IsEmpty());
}
TEST_F(LabelTest, CanForceDirectionality) {
Label bidi_text_force_url(ToRTL("0123456") + u".com", 0, style::STYLE_PRIMARY,
gfx::DirectionalityMode::DIRECTIONALITY_AS_URL);
EXPECT_EQ(base::i18n::TextDirection::LEFT_TO_RIGHT,
bidi_text_force_url.GetTextDirectionForTesting());
Label rtl_text_force_ltr(ToRTL("0123456"), 0, style::STYLE_PRIMARY,
gfx::DirectionalityMode::DIRECTIONALITY_FORCE_LTR);
EXPECT_EQ(base::i18n::TextDirection::LEFT_TO_RIGHT,
rtl_text_force_ltr.GetTextDirectionForTesting());
Label ltr_text_force_rtl(u"0123456", 0, style::STYLE_PRIMARY,
gfx::DirectionalityMode::DIRECTIONALITY_FORCE_RTL);
EXPECT_EQ(base::i18n::TextDirection::RIGHT_TO_LEFT,
ltr_text_force_rtl.GetTextDirectionForTesting());
}
TEST_F(LabelTest, DefaultDirectionalityIsFromText) {
Label ltr(u"Foo");
EXPECT_EQ(base::i18n::TextDirection::LEFT_TO_RIGHT,
ltr.GetTextDirectionForTesting());
Label rtl(ToRTL("0123456"));
EXPECT_EQ(base::i18n::TextDirection::RIGHT_TO_LEFT,
rtl.GetTextDirectionForTesting());
}
TEST_F(LabelTest, IsDisplayTextTruncated) {
const std::u16string text = u"A random string";
label()->SetText(text);
gfx::Size zero_size;
label()->SetElideBehavior(gfx::ELIDE_TAIL);
label()->SetBoundsRect(gfx::Rect(zero_size));
EXPECT_TRUE(label()->IsDisplayTextTruncated());
label()->SetElideBehavior(gfx::NO_ELIDE);
EXPECT_TRUE(label()->IsDisplayTextTruncated());
gfx::Size minimum_size(1, 1);
label()->SetBoundsRect(gfx::Rect(minimum_size));
EXPECT_TRUE(label()->IsDisplayTextTruncated());
gfx::Size enough_size(100, 100);
label()->SetBoundsRect(gfx::Rect(enough_size));
EXPECT_FALSE(label()->IsDisplayTextTruncated());
const std::u16string empty_text;
label()->SetText(empty_text);
EXPECT_FALSE(label()->IsDisplayTextTruncated());
label()->SetBoundsRect(gfx::Rect(zero_size));
EXPECT_FALSE(label()->IsDisplayTextTruncated());
}
TEST_F(LabelTest, TextChangedCallback) {
bool text_changed = false;
auto subscription = label()->AddTextChangedCallback(base::BindRepeating(
[](bool* text_changed) { *text_changed = true; }, &text_changed));
label()->SetText(u"abc");
EXPECT_TRUE(text_changed);
}
TEST_F(LabelTest, GetSubstringBounds) {
label()->SetText(u"abc");
auto substring_bounds = label()->GetSubstringBounds(gfx::Range(0, 3));
EXPECT_EQ(1u, substring_bounds.size());
auto insets = gfx::Insets::TLBR(2, 3, 4, 5);
label()->SetBorder(CreateEmptyBorder(insets));
auto substring_bounds_with_inset =
label()->GetSubstringBounds(gfx::Range(0, 3));
EXPECT_EQ(1u, substring_bounds_with_inset.size());
EXPECT_EQ(substring_bounds[0].x() + 3, substring_bounds_with_inset[0].x());
EXPECT_EQ(substring_bounds[0].y() + 2, substring_bounds_with_inset[0].y());
EXPECT_EQ(substring_bounds[0].width(),
substring_bounds_with_inset[0].width());
EXPECT_EQ(substring_bounds[0].height(),
substring_bounds_with_inset[0].height());
}
#if BUILDFLAG(IS_CHROMEOS)
#define MAYBE_ChecksSubpixelRenderingOntoOpaqueSurface \
DISABLED_ChecksSubpixelRenderingOntoOpaqueSurface
#else
#define MAYBE_ChecksSubpixelRenderingOntoOpaqueSurface \
ChecksSubpixelRenderingOntoOpaqueSurface
#endif
TEST_F(LabelTest, MAYBE_ChecksSubpixelRenderingOntoOpaqueSurface) {
View* view =
widget()->GetContentsView()->AddChildView(std::make_unique<View>());
Label* label = view->AddChildView(std::make_unique<TestLabel>());
EXPECT_TRUE(label->GetSubpixelRenderingEnabled());
gfx::Canvas canvas;
label->OnPaint(&canvas);
view->SetPaintToLayer();
label->OnPaint(&canvas);
view->layer()->SetFillsBoundsOpaquely(false);
EXPECT_DCHECK_DEATH(label->OnPaint(&canvas));
label->SetSkipSubpixelRenderingOpacityCheck(true);
label->OnPaint(&canvas);
label->SetSkipSubpixelRenderingOpacityCheck(false);
view->SetBackground(CreateSolidBackground(SK_ColorWHITE));
label->OnPaint(&canvas);
}
#if BUILDFLAG(SUPPORTS_AX_TEXT_OFFSETS)
TEST_F(LabelTest, WordOffsets) {
const ::ui::ScopedAXModeSetter ax_mode_setter(ui::AXMode::kNativeAPIs);
MockAXModeAdded();
ASSERT_TRUE(label()->GetViewAccessibility().is_initialized());
base::test::ScopedFeatureList scoped_feature_list;
scoped_feature_list.InitAndEnableFeature(::features::kUiaProvider);
const std::u16string text = u"This is a string";
label()->SetText(text);
label()->SizeToPreferredSize();
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
std::vector<int32_t> expected_starts = {0, 5, 8, 10};
std::vector<int32_t> expected_ends = {4, 7, 9, 16};
EXPECT_EQ(
node_data.GetIntListAttribute(ax::mojom::IntListAttribute::kWordStarts),
expected_starts);
EXPECT_EQ(
node_data.GetIntListAttribute(ax::mojom::IntListAttribute::kWordEnds),
expected_ends);
}
TEST_F(LabelTest, WordOffsetsAXNotOn) {
const ::ui::ScopedAXModeSetter ax_mode_setter(ui::AXMode::kNativeAPIs);
ASSERT_FALSE(label()->GetViewAccessibility().is_initialized());
base::test::ScopedFeatureList scoped_feature_list;
scoped_feature_list.InitAndEnableFeature(::features::kUiaProvider);
const std::u16string text = u"This is a string";
label()->SetText(text);
label()->SizeToPreferredSize();
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
std::vector<int32_t> expected_starts = {};
std::vector<int32_t> expected_ends = {};
EXPECT_EQ(
node_data.GetIntListAttribute(ax::mojom::IntListAttribute::kWordStarts),
expected_starts);
EXPECT_EQ(
node_data.GetIntListAttribute(ax::mojom::IntListAttribute::kWordEnds),
expected_ends);
}
TEST_F(LabelTest, AccessibleGraphemeOffsets) {
const ::ui::ScopedAXModeSetter ax_mode_setter(ui::AXMode::kNativeAPIs);
MockAXModeAdded();
ASSERT_TRUE(label()->GetViewAccessibility().is_initialized());
struct Case {
std::u16string text;
std::vector<int32_t> expected_offsets;
};
const auto cases = std::to_array<Case>({
{std::u16string(), {}},
{u"This is a string",
{0, 6, 13, 15, 21, 24, 27, 32, 35, 41, 45, 50, 54, 58, 61, 68, 76}},
{u"اللغة العربيي",
{67, 63, 59, 52, 46, 43, 40, 36, 29, 24, 20, 16, 6, 17}},
{u"\u0915\u093fabc\u0915\u093f", {10, 23, 29, 36, 42, 56}},
{u"ab\u0915\u093fcd", {3, 9, 16, 29, 35, 43}},
{u"ab\U0001D11Ecd", {4, 10, 17, 23, 29, 37}},
});
base::test::ScopedFeatureList scoped_feature_list;
scoped_feature_list.InitAndEnableFeature(::features::kUiaProvider);
for (size_t i = 0; i < std::size(cases); i++) {
ASSERT_TRUE(label()->GetViewAccessibility().is_initialized());
SCOPED_TRACE(base::StringPrintf("Testing cases[%" PRIuS "]", i));
label()->SetText(cases[i].text);
label()->SizeToPreferredSize();
EXPECT_EQ(cases[i].text, label()->GetDisplayTextForTesting());
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
EXPECT_EQ(node_data.GetIntListAttribute(
ax::mojom::IntListAttribute::kCharacterOffsets),
cases[i].expected_offsets);
}
}
TEST_F(LabelTest, AccessibleGraphemeOffsetsObscured) {
const ::ui::ScopedAXModeSetter ax_mode_setter(ui::AXMode::kNativeAPIs);
MockAXModeAdded();
ASSERT_TRUE(label()->GetViewAccessibility().is_initialized());
base::test::ScopedFeatureList scoped_feature_list;
scoped_feature_list.InitAndEnableFeature(::features::kUiaProvider);
const std::u16string text = u"password";
label()->SetText(text);
label()->SizeToPreferredSize();
label()->SetObscured(true);
EXPECT_EQ(
std::u16string(text.size(), gfx::RenderText::kPasswordReplacementChar),
label()->GetDisplayTextForTesting());
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
std::vector<int32_t> expected_offsets = {6, 10, 15, 20, 25, 30, 35, 40, 45};
EXPECT_EQ(node_data.GetIntListAttribute(
ax::mojom::IntListAttribute::kCharacterOffsets),
expected_offsets);
}
TEST_F(LabelTest, AccessibleGraphemeOffsetsElided) {
const ::ui::ScopedAXModeSetter ax_mode_setter(ui::AXMode::kNativeAPIs);
MockAXModeAdded();
ASSERT_TRUE(label()->GetViewAccessibility().is_initialized());
base::test::ScopedFeatureList scoped_feature_list;
scoped_feature_list.InitAndEnableFeature(::features::kUiaProvider);
label()->SetElideBehavior(gfx::NO_ELIDE);
const std::u16string text = u"This is a string";
label()->SetText(text);
gfx::Size size = label()->GetPreferredSize({});
label()->SetBoundsRect(gfx::Rect(size));
EXPECT_EQ(text, label()->GetDisplayTextForTesting());
size.set_width(size.width() / 2);
label()->SetBoundsRect(gfx::Rect(size));
label()->SetElideBehavior(gfx::ELIDE_TAIL);
EXPECT_GT(text.size(), label()->GetDisplayTextForTesting().size());
EXPECT_EQ(u"This i\x2026", label()->GetDisplayTextForTesting());
ui::AXNodeData node_data;
label()->GetViewAccessibility().GetAccessibleNodeData(&node_data);
std::vector<int32_t> expected_offsets = {1, 7, 14, 16, 22, 25, 28, 38, 38,
38, 38, 38, 38, 38, 38, 38, 38};
EXPECT_EQ(node_data.GetIntListAttribute(
ax::mojom::IntListAttribute::kCharacterOffsets),
expected_offsets);
}
#endif
TEST_F(LabelSelectionTest, Selectable) {
EXPECT_FALSE(label()->GetSelectable());
ASSERT_TRUE(label()->SetSelectable(true));
EXPECT_TRUE(label()->GetSelectable());
label()->SetMultiLine(true);
EXPECT_TRUE(label()->GetSelectable());
label()->SetObscured(true);
EXPECT_FALSE(label()->GetSelectable());
}
TEST_F(LabelSelectionTest, FocusOnClick) {
label()->SetText(u"text");
label()->SizeToPreferredSize();
PerformClick(gfx::Point());
EXPECT_NE(label(), GetFocusedView());
ASSERT_TRUE(label()->SetSelectable(true));
PerformClick(gfx::Point());
EXPECT_EQ(label(), GetFocusedView());
}
TEST_F(LabelSelectionTest, FocusTraversal) {
View* view = new View();
view->SetFocusBehavior(View::FocusBehavior::ALWAYS);
widget()->GetContentsView()->AddChildViewAt(view, 0);
view->RequestFocus();
EXPECT_EQ(view, GetFocusedView());
widget()->GetFocusManager()->AdvanceFocus(false);
EXPECT_NE(label(), GetFocusedView());
view->RequestFocus();
EXPECT_EQ(view, GetFocusedView());
EXPECT_TRUE(label()->SetSelectable(true));
widget()->GetFocusManager()->AdvanceFocus(false);
EXPECT_NE(label(), GetFocusedView());
view->RequestFocus();
EXPECT_EQ(view, GetFocusedView());
EXPECT_TRUE(label()->SetSelectable(false));
label()->SetFocusBehavior(View::FocusBehavior::ALWAYS);
widget()->GetFocusManager()->AdvanceFocus(false);
EXPECT_EQ(label(), GetFocusedView());
}
TEST_F(LabelSelectionTest, DoubleTripleClick) {
label()->SetText(u"Label double click");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformClick(GetCursorPoint(0));
EXPECT_TRUE(GetSelectedText().empty());
PerformClick(GetCursorPoint(0));
EXPECT_EQ(u"Label", GetSelectedText());
PerformClick(GetCursorPoint(0));
EXPECT_EQ(label()->GetText(), GetSelectedText());
PerformClick(GetCursorPoint(0));
EXPECT_EQ(u"Label", GetSelectedText());
PerformClick(GetCursorPoint(8));
EXPECT_TRUE(GetSelectedText().empty());
PerformClick(GetCursorPoint(8));
EXPECT_EQ(u"double", GetSelectedText());
}
TEST_F(LabelSelectionTest, MouseDrag) {
label()->SetText(u"Label mouse drag");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformMousePress(GetCursorPoint(5));
PerformMouseDragTo(GetCursorPoint(0));
EXPECT_EQ(u"Label", GetSelectedText());
PerformMouseDragTo(GetCursorPoint(8));
EXPECT_EQ(u" mo", GetSelectedText());
PerformMouseDragTo(gfx::Point(200, GetCursorPoint(0).y()));
PerformMouseRelease(gfx::Point(200, GetCursorPoint(0).y()));
EXPECT_EQ(u" mouse drag", GetSelectedText());
event_generator()->PressKey(ui::VKEY_C, kControlCommandModifier);
EXPECT_EQ(u" mouse drag", GetClipboardText(ui::ClipboardBuffer::kCopyPaste));
}
TEST_F(LabelSelectionTest, MouseDragMultilineLTR) {
label()->SetMultiLine(true);
label()->SetText(u"abcd\nefgh");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
ASSERT_EQ(2u, GetLineCount());
PerformMousePress(GetCursorPoint(2));
PerformMouseDragTo(GetCursorPoint(0));
EXPECT_EQ(u"ab", GetSelectedText());
PerformMouseDragTo(GetCursorPoint(7));
EXPECT_EQ(u"cd\nef", GetSelectedText());
PerformMouseDragTo(gfx::Point(-5, GetCursorPoint(6).y()));
EXPECT_EQ(u"cd\n", GetSelectedText());
PerformMouseDragTo(gfx::Point(100, GetCursorPoint(6).y()));
EXPECT_EQ(u"cd\nefgh", GetSelectedText());
const auto points = std::to_array<gfx::Point>({
{GetCursorPoint(1).x(), -5},
{GetCursorPoint(2).x(), -5},
{GetCursorPoint(3).x(), -5},
{GetCursorPoint(8).x(), 100},
{GetCursorPoint(7).x(), 100},
{GetCursorPoint(6).x(), 100},
});
constexpr const char16_t* kExtendLeft = u"ab";
constexpr const char16_t* kExtendRight = u"cd\nefgh";
PerformMouseDragTo(points[NW]);
EXPECT_EQ(kExtends ? kExtendLeft : u"b", GetSelectedText());
PerformMouseDragTo(points[NORTH]);
EXPECT_EQ(kExtends ? kExtendLeft : u"", GetSelectedText());
PerformMouseDragTo(points[NE]);
EXPECT_EQ(kExtends ? kExtendLeft : u"c", GetSelectedText());
PerformMouseDragTo(points[SE]);
EXPECT_EQ(kExtends ? kExtendRight : u"cd\nefg", GetSelectedText());
PerformMouseDragTo(points[SOUTH]);
EXPECT_EQ(kExtends ? kExtendRight : u"cd\nef", GetSelectedText());
PerformMouseDragTo(points[SW]);
EXPECT_EQ(kExtends ? kExtendRight : u"cd\ne", GetSelectedText());
}
TEST_F(LabelSelectionTest, MouseDragSingleLineLTR) {
label()->SetText(u"abcdef");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformMousePress(GetCursorPoint(2));
const auto points = std::to_array<gfx::Point>({
{GetCursorPoint(1).x(), -5},
{GetCursorPoint(2).x(), -5},
{GetCursorPoint(3).x(), -5},
{GetCursorPoint(3).x(), 100},
{GetCursorPoint(2).x(), 100},
{GetCursorPoint(1).x(), 100},
});
constexpr const char16_t* kExtendLeft = u"ab";
constexpr const char16_t* kExtendRight = u"cdef";
PerformMouseDragTo(points[NW]);
EXPECT_EQ(kExtends ? kExtendLeft : u"b", GetSelectedText());
PerformMouseDragTo(points[NORTH]);
EXPECT_EQ(kExtends ? kExtendRight : u"", GetSelectedText());
PerformMouseDragTo(points[NE]);
EXPECT_EQ(kExtends ? kExtendRight : u"c", GetSelectedText());
PerformMouseDragTo(points[SE]);
EXPECT_EQ(kExtends ? kExtendRight : u"c", GetSelectedText());
PerformMouseDragTo(points[SOUTH]);
EXPECT_EQ(kExtends ? kExtendRight : u"", GetSelectedText());
PerformMouseDragTo(points[SW]);
EXPECT_EQ(kExtends ? kExtendLeft : u"b", GetSelectedText());
}
TEST_F(LabelSelectionTest, MouseDragMultilineRTL) {
label()->SetMultiLine(true);
label()->SetText(ToRTL("012\n345"));
EXPECT_EQ(u"\x5d0\x5d1\x5d2\n\x5d3\x5d4\x5d5", label()->GetText());
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
ASSERT_EQ(2u, GetLineCount());
PerformMousePress(GetCursorPoint(1));
PerformMouseDragTo(GetCursorPoint(0));
EXPECT_EQ(ToRTL("0"), GetSelectedText());
PerformMouseDragTo(GetCursorPoint(6));
EXPECT_EQ(ToRTL("12\n34"), GetSelectedText());
PerformMouseDragTo(gfx::Point(-5, GetCursorPoint(6).y()));
EXPECT_EQ(ToRTL("12\n345"), GetSelectedText());
PerformMouseDragTo(gfx::Point(100, GetCursorPoint(6).y()));
EXPECT_EQ(ToRTL("12\n"), GetSelectedText());
const auto points = std::to_array<gfx::Point>({
{GetCursorPoint(2).x(), -5},
{GetCursorPoint(1).x(), -5},
{GetCursorPoint(0).x(), -5},
{GetCursorPoint(4).x(), 100},
{GetCursorPoint(5).x(), 100},
{GetCursorPoint(6).x(), 100},
});
const std::u16string extend_right = ToRTL("0");
const std::u16string extend_left = ToRTL("12\n345");
PerformMouseDragTo(points[NW]);
EXPECT_EQ(kExtends ? extend_right : ToRTL("1"), GetSelectedText());
PerformMouseDragTo(points[NORTH]);
EXPECT_EQ(kExtends ? extend_right : ToRTL(""), GetSelectedText());
PerformMouseDragTo(points[NE]);
EXPECT_EQ(kExtends ? extend_right : ToRTL("0"), GetSelectedText());
PerformMouseDragTo(points[SE]);
EXPECT_EQ(kExtends ? extend_left : ToRTL("12\n"), GetSelectedText());
PerformMouseDragTo(points[SOUTH]);
EXPECT_EQ(kExtends ? extend_left : ToRTL("12\n3"), GetSelectedText());
PerformMouseDragTo(points[SW]);
EXPECT_EQ(kExtends ? extend_left : ToRTL("12\n34"), GetSelectedText());
}
TEST_F(LabelSelectionTest, MouseDragSingleLineRTL) {
label()->SetText(ToRTL("0123456"));
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformMousePress(GetCursorPoint(1));
const std::vector<gfx::Point> points{
{GetCursorPoint(2).x(), -5},
{GetCursorPoint(1).x(), -5},
{GetCursorPoint(0).x(), -5},
{GetCursorPoint(0).x(), 100},
{GetCursorPoint(1).x(), 100},
{GetCursorPoint(2).x(), 100},
};
const std::u16string extend_right = ToRTL("0");
const std::u16string extend_left = ToRTL("123456");
PerformMouseDragTo(points[NW]);
EXPECT_EQ(kExtends ? extend_left : ToRTL("1"), GetSelectedText());
PerformMouseDragTo(points[NORTH]);
EXPECT_EQ(kExtends ? extend_right : ToRTL(""), GetSelectedText());
PerformMouseDragTo(points[NE]);
EXPECT_EQ(kExtends ? extend_right : ToRTL("0"), GetSelectedText());
PerformMouseDragTo(points[SE]);
EXPECT_EQ(kExtends ? extend_right : ToRTL("0"), GetSelectedText());
PerformMouseDragTo(points[SOUTH]);
EXPECT_EQ(kExtends ? extend_right : ToRTL(""), GetSelectedText());
PerformMouseDragTo(points[SW]);
EXPECT_EQ(kExtends ? extend_left : ToRTL("1"), GetSelectedText());
}
TEST_F(LabelSelectionTest, MouseDragWord) {
label()->SetText(u"Label drag word");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformClick(GetCursorPoint(8));
PerformMousePress(GetCursorPoint(8));
EXPECT_EQ(u"drag", GetSelectedText());
PerformMouseDragTo(GetCursorPoint(0));
EXPECT_EQ(u"Label drag", GetSelectedText());
PerformMouseDragTo(gfx::Point(200, GetCursorPoint(0).y()));
PerformMouseRelease(gfx::Point(200, GetCursorPoint(0).y()));
EXPECT_EQ(u"drag word", GetSelectedText());
}
#if BUILDFLAG(IS_LINUX)
TEST_F(LabelSelectionTest, SelectionClipboard) {
label()->SetText(u"Label selection clipboard");
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
label()->SelectRange(gfx::Range(2, 5));
EXPECT_EQ(u"bel", GetSelectedText());
EXPECT_TRUE(GetClipboardText(ui::ClipboardBuffer::kSelection).empty());
PerformMousePress(GetCursorPoint(5));
PerformMouseDragTo(GetCursorPoint(0));
PerformMouseRelease(GetCursorPoint(0));
EXPECT_EQ(u"Label", GetSelectedText());
EXPECT_EQ(u"Label", GetClipboardText(ui::ClipboardBuffer::kSelection));
}
#endif
TEST_F(LabelSelectionTest, KeyboardActions) {
const std::u16string initial_text = u"Label keyboard actions";
label()->SetText(initial_text);
label()->SizeToPreferredSize();
ASSERT_TRUE(label()->SetSelectable(true));
PerformClick(gfx::Point());
EXPECT_EQ(label(), GetFocusedView());
event_generator()->PressKey(ui::VKEY_A, kControlCommandModifier);
EXPECT_EQ(initial_text, GetSelectedText());
event_generator()->PressKey(ui::VKEY_C, kControlCommandModifier);
EXPECT_EQ(initial_text, GetClipboardText(ui::ClipboardBuffer::kCopyPaste));
const std::u16string new_text = u"Label obscured text";
label()->SetText(new_text);
EXPECT_FALSE(label()->HasSelection());
EXPECT_EQ(label(), GetFocusedView());
label()->SetObscured(true);
EXPECT_FALSE(label()->GetSelectable());
event_generator()->PressKey(ui::VKEY_A, kControlCommandModifier);
EXPECT_EQ(std::u16string(), GetSelectedText());
}
TEST_F(LabelSelectionTest, ContextMenuContents) {
label()->SetText(u"Label context menu");
label()->SizeToPreferredSize();
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kCopy));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kSelectAll));
ASSERT_TRUE(label()->SetSelectable(true));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kCopy));
EXPECT_TRUE(IsMenuCommandEnabled(Label::MenuCommands::kSelectAll));
label()->SelectRange(gfx::Range(0, 4));
EXPECT_TRUE(IsMenuCommandEnabled(Label::MenuCommands::kCopy));
EXPECT_TRUE(IsMenuCommandEnabled(Label::MenuCommands::kSelectAll));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kLastCommandId + 1));
label()->SetObscured(true);
EXPECT_FALSE(label()->GetSelectable());
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kCopy));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kSelectAll));
label()->SetObscured(false);
label()->SetText(std::u16string());
ASSERT_TRUE(label()->SetSelectable(true));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kCopy));
EXPECT_FALSE(IsMenuCommandEnabled(Label::MenuCommands::kSelectAll));
}
}