| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ui/gfx/render_text.h" | 5 #include "ui/gfx/render_text.h" |
| 6 | 6 |
| 7 #include <algorithm> | 7 #include <algorithm> |
| 8 | 8 |
| 9 #include "base/format_macros.h" | 9 #include "base/format_macros.h" |
| 10 #include "base/memory/scoped_ptr.h" | 10 #include "base/memory/scoped_ptr.h" |
| (...skipping 26 matching lines...) Expand all Loading... |
| 37 const wchar_t kLtr[] = L"abc"; | 37 const wchar_t kLtr[] = L"abc"; |
| 38 const wchar_t kRtl[] = L"\x5d0\x5d1\x5d2"; | 38 const wchar_t kRtl[] = L"\x5d0\x5d1\x5d2"; |
| 39 #if !defined(OS_MACOSX) | 39 #if !defined(OS_MACOSX) |
| 40 const wchar_t kLtrRtl[] = L"a" L"\x5d0\x5d1"; | 40 const wchar_t kLtrRtl[] = L"a" L"\x5d0\x5d1"; |
| 41 const wchar_t kLtrRtlLtr[] = L"a" L"\x5d1" L"b"; | 41 const wchar_t kLtrRtlLtr[] = L"a" L"\x5d1" L"b"; |
| 42 const wchar_t kRtlLtr[] = L"\x5d0\x5d1" L"a"; | 42 const wchar_t kRtlLtr[] = L"\x5d0\x5d1" L"a"; |
| 43 const wchar_t kRtlLtrRtl[] = L"\x5d0" L"a" L"\x5d1"; | 43 const wchar_t kRtlLtrRtl[] = L"\x5d0" L"a" L"\x5d1"; |
| 44 #endif | 44 #endif |
| 45 | 45 |
| 46 // Checks whether |range| contains |index|. This is not the same as calling | 46 // Checks whether |range| contains |index|. This is not the same as calling |
| 47 // range.Contains(Range(index)), which returns true if |index| == |range.end()|. | 47 // |range.Contains(gfx::Range(index))| - as that would return true when |
| 48 // |index| == |range.end()|. |
| 48 bool IndexInRange(const Range& range, size_t index) { | 49 bool IndexInRange(const Range& range, size_t index) { |
| 49 return index >= range.start() && index < range.end(); | 50 return index >= range.start() && index < range.end(); |
| 50 } | 51 } |
| 51 | 52 |
| 52 base::string16 GetSelectedText(RenderText* render_text) { | 53 base::string16 GetSelectedText(RenderText* render_text) { |
| 53 return render_text->text().substr(render_text->selection().GetMin(), | 54 return render_text->text().substr(render_text->selection().GetMin(), |
| 54 render_text->selection().length()); | 55 render_text->selection().length()); |
| 55 } | 56 } |
| 56 | 57 |
| 57 // A test utility function to set the application default text direction. | 58 // A test utility function to set the application default text direction. |
| (...skipping 229 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 287 EXPECT_EQ(0U, render_text->cursor_position()); | 288 EXPECT_EQ(0U, render_text->cursor_position()); |
| 288 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_RIGHT, false); | 289 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_RIGHT, false); |
| 289 EXPECT_EQ(2U, render_text->cursor_position()); | 290 EXPECT_EQ(2U, render_text->cursor_position()); |
| 290 | 291 |
| 291 // Test index conversion and cursor validity with a valid surrogate pair. | 292 // Test index conversion and cursor validity with a valid surrogate pair. |
| 292 EXPECT_EQ(0U, render_text->TextIndexToLayoutIndex(0U)); | 293 EXPECT_EQ(0U, render_text->TextIndexToLayoutIndex(0U)); |
| 293 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(1U)); | 294 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(1U)); |
| 294 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(2U)); | 295 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(2U)); |
| 295 EXPECT_EQ(0U, render_text->LayoutIndexToTextIndex(0U)); | 296 EXPECT_EQ(0U, render_text->LayoutIndexToTextIndex(0U)); |
| 296 EXPECT_EQ(2U, render_text->LayoutIndexToTextIndex(1U)); | 297 EXPECT_EQ(2U, render_text->LayoutIndexToTextIndex(1U)); |
| 297 EXPECT_TRUE(render_text->IsValidCursorIndex(0U)); | 298 EXPECT_TRUE(render_text->IsCursorablePosition(0U)); |
| 298 EXPECT_FALSE(render_text->IsValidCursorIndex(1U)); | 299 EXPECT_FALSE(render_text->IsCursorablePosition(1U)); |
| 299 EXPECT_TRUE(render_text->IsValidCursorIndex(2U)); | 300 EXPECT_TRUE(render_text->IsCursorablePosition(2U)); |
| 300 | 301 |
| 301 // FindCursorPosition() should not return positions between a surrogate pair. | 302 // FindCursorPosition() should not return positions between a surrogate pair. |
| 302 render_text->SetDisplayRect(Rect(0, 0, 20, 20)); | 303 render_text->SetDisplayRect(Rect(0, 0, 20, 20)); |
| 303 EXPECT_EQ(render_text->FindCursorPosition(Point(0, 0)).caret_pos(), 0U); | 304 EXPECT_EQ(render_text->FindCursorPosition(Point(0, 0)).caret_pos(), 0U); |
| 304 EXPECT_EQ(render_text->FindCursorPosition(Point(20, 0)).caret_pos(), 2U); | 305 EXPECT_EQ(render_text->FindCursorPosition(Point(20, 0)).caret_pos(), 2U); |
| 305 for (int x = -1; x <= 20; ++x) { | 306 for (int x = -1; x <= 20; ++x) { |
| 306 SelectionModel selection = render_text->FindCursorPosition(Point(x, 0)); | 307 SelectionModel selection = render_text->FindCursorPosition(Point(x, 0)); |
| 307 EXPECT_TRUE(selection.caret_pos() == 0U || selection.caret_pos() == 2U); | 308 EXPECT_TRUE(selection.caret_pos() == 0U || selection.caret_pos() == 2U); |
| 308 } | 309 } |
| 309 | 310 |
| (...skipping 125 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 435 // Surrogate pairs should be elided reasonably enough. | 436 // Surrogate pairs should be elided reasonably enough. |
| 436 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x05c1\x05b8", false }, | 437 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x05c1\x05b8", false }, |
| 437 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x2026" , true }, | 438 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x2026" , true }, |
| 438 { L"01\x05e9\x05bc\x05c1\x05b8", L"01\x05e9\x2026" , true }, | 439 { L"01\x05e9\x05bc\x05c1\x05b8", L"01\x05e9\x2026" , true }, |
| 439 { L"012\x05e9\x05bc\x05c1\x05b8", L"012\x2026\x200E" , true }, | 440 { L"012\x05e9\x05bc\x05c1\x05b8", L"012\x2026\x200E" , true }, |
| 440 { L"012\xF0\x9D\x84\x9E", L"012\xF0\x2026" , true }, | 441 { L"012\xF0\x9D\x84\x9E", L"012\xF0\x2026" , true }, |
| 441 }; | 442 }; |
| 442 | 443 |
| 443 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); | 444 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); |
| 444 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); | 445 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); |
| 445 expected_render_text->SetDisplayRect(Rect(0, 0, 9999, 100)); | 446 expected_render_text->SetDisplayRect(gfx::Rect(0, 0, 9999, 100)); |
| 446 | 447 |
| 447 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); | 448 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); |
| 448 render_text->SetFontList(FontList("serif, Sans serif, 12px")); | 449 render_text->SetFontList(FontList("serif, Sans serif, 12px")); |
| 449 render_text->SetElideBehavior(ELIDE_AT_END); | 450 render_text->SetElideBehavior(gfx::ELIDE_AT_END); |
| 450 | 451 |
| 451 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { | 452 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { |
| 452 // Compute expected width | 453 // Compute expected width |
| 453 expected_render_text->SetText(WideToUTF16(cases[i].layout_text)); | 454 expected_render_text->SetText(WideToUTF16(cases[i].layout_text)); |
| 454 int expected_width = expected_render_text->GetContentWidth(); | 455 int expected_width = expected_render_text->GetContentWidth(); |
| 455 | 456 |
| 456 base::string16 input = WideToUTF16(cases[i].text); | 457 base::string16 input = WideToUTF16(cases[i].text); |
| 457 // Extend the input text to ensure that it is wider than the layout_text, | 458 // Extend the input text to ensure that it is wider than the layout_text, |
| 458 // and so it will get elided. | 459 // and so it will get elided. |
| 459 if (cases[i].elision_expected) | 460 if (cases[i].elision_expected) |
| 460 input.append(WideToUTF16(L" MMMMMMMMMMM")); | 461 input.append(WideToUTF16(L" MMMMMMMMMMM")); |
| 461 | 462 |
| 462 render_text->SetText(input); | 463 render_text->SetText(input); |
| 463 render_text->SetDisplayRect(Rect(0, 0, expected_width, 100)); | 464 render_text->SetDisplayRect(gfx::Rect(0, 0, expected_width, 100)); |
| 464 EXPECT_EQ(input, render_text->text()) | 465 EXPECT_EQ(input, render_text->text()) |
| 465 << "->For case " << i << ": " << cases[i].text << "\n"; | 466 << "->For case " << i << ": " << cases[i].text << "\n"; |
| 466 EXPECT_EQ(WideToUTF16(cases[i].layout_text), render_text->GetLayoutText()) | 467 EXPECT_EQ(WideToUTF16(cases[i].layout_text), render_text->GetLayoutText()) |
| 467 << "->For case " << i << ": " << cases[i].text << "\n"; | 468 << "->For case " << i << ": " << cases[i].text << "\n"; |
| 468 expected_render_text->SetText(base::string16()); | 469 expected_render_text->SetText(base::string16()); |
| 469 } | 470 } |
| 470 } | 471 } |
| 471 | 472 |
| 472 TEST_F(RenderTextTest, ElidedObscuredText) { | 473 TEST_F(RenderTextTest, ElidedObscuredText) { |
| 473 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); | 474 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); |
| 474 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); | 475 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); |
| 475 expected_render_text->SetDisplayRect(Rect(0, 0, 9999, 100)); | 476 expected_render_text->SetDisplayRect(gfx::Rect(0, 0, 9999, 100)); |
| 476 expected_render_text->SetText(WideToUTF16(L"**\x2026")); | 477 expected_render_text->SetText(WideToUTF16(L"**\x2026")); |
| 477 | 478 |
| 478 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); | 479 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); |
| 479 render_text->SetFontList(FontList("serif, Sans serif, 12px")); | 480 render_text->SetFontList(FontList("serif, Sans serif, 12px")); |
| 480 render_text->SetElideBehavior(ELIDE_AT_END); | 481 render_text->SetElideBehavior(gfx::ELIDE_AT_END); |
| 481 render_text->SetDisplayRect( | 482 render_text->SetDisplayRect( |
| 482 Rect(0, 0, expected_render_text->GetContentWidth(), 100)); | 483 gfx::Rect(0, 0, expected_render_text->GetContentWidth(), 100)); |
| 483 render_text->SetObscured(true); | 484 render_text->SetObscured(true); |
| 484 render_text->SetText(WideToUTF16(L"abcdef")); | 485 render_text->SetText(WideToUTF16(L"abcdef")); |
| 485 EXPECT_EQ(WideToUTF16(L"abcdef"), render_text->text()); | 486 EXPECT_EQ(WideToUTF16(L"abcdef"), render_text->text()); |
| 486 EXPECT_EQ(WideToUTF16(L"**\x2026"), render_text->GetLayoutText()); | 487 EXPECT_EQ(WideToUTF16(L"**\x2026"), render_text->GetLayoutText()); |
| 487 } | 488 } |
| 488 | 489 |
| 489 TEST_F(RenderTextTest, TruncatedText) { | 490 TEST_F(RenderTextTest, TruncatedText) { |
| 490 struct { | 491 struct { |
| 491 const wchar_t* text; | 492 const wchar_t* text; |
| 492 const wchar_t* layout_text; | 493 const wchar_t* layout_text; |
| (...skipping 400 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 893 | 894 |
| 894 // TODO(asvitkine): Disable tests that fail on XP bots due to lack of complete | 895 // TODO(asvitkine): Disable tests that fail on XP bots due to lack of complete |
| 895 // font support for some scripts - http://crbug.com/106450 | 896 // font support for some scripts - http://crbug.com/106450 |
| 896 #if defined(OS_WIN) | 897 #if defined(OS_WIN) |
| 897 if (base::win::GetVersion() < base::win::VERSION_VISTA) | 898 if (base::win::GetVersion() < base::win::VERSION_VISTA) |
| 898 return; | 899 return; |
| 899 #endif | 900 #endif |
| 900 | 901 |
| 901 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); | 902 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); |
| 902 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { | 903 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { |
| 903 SCOPED_TRACE(base::StringPrintf("Testing cases[%" PRIuS "]", i)); | |
| 904 render_text->SetText(cases[i].text); | 904 render_text->SetText(cases[i].text); |
| 905 | 905 |
| 906 size_t next = render_text->IndexOfAdjacentGrapheme(cases[i].index, | 906 size_t next = render_text->IndexOfAdjacentGrapheme(cases[i].index, |
| 907 CURSOR_FORWARD); | 907 CURSOR_FORWARD); |
| 908 EXPECT_EQ(cases[i].expected_next, next); | 908 EXPECT_EQ(cases[i].expected_next, next); |
| 909 EXPECT_TRUE(render_text->IsValidCursorIndex(next)); | 909 EXPECT_TRUE(render_text->IsCursorablePosition(next)); |
| 910 | 910 |
| 911 size_t previous = render_text->IndexOfAdjacentGrapheme(cases[i].index, | 911 size_t previous = render_text->IndexOfAdjacentGrapheme(cases[i].index, |
| 912 CURSOR_BACKWARD); | 912 CURSOR_BACKWARD); |
| 913 EXPECT_EQ(cases[i].expected_previous, previous); | 913 EXPECT_EQ(cases[i].expected_previous, previous); |
| 914 EXPECT_TRUE(render_text->IsValidCursorIndex(previous)); | 914 EXPECT_TRUE(render_text->IsCursorablePosition(previous)); |
| 915 } | 915 } |
| 916 } | 916 } |
| 917 | 917 |
| 918 TEST_F(RenderTextTest, MidGraphemeSelectionBounds) { | |
| 919 // Test that selection bounds may be set amid multi-character graphemes. | |
| 920 const base::string16 kHindi = WideToUTF16(L"\x0915\x093f"); | |
| 921 const base::string16 kThai = WideToUTF16(L"\x0e08\x0e33"); | |
| 922 const base::string16 cases[] = { kHindi, kThai }; | |
| 923 | |
| 924 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); | |
| 925 for (size_t i = 0; i < arraysize(cases); i++) { | |
| 926 SCOPED_TRACE(base::StringPrintf("Testing cases[%" PRIuS "]", i)); | |
| 927 render_text->SetText(cases[i]); | |
| 928 EXPECT_TRUE(render_text->IsValidLogicalIndex(1)); | |
| 929 #if !defined(OS_MACOSX) | |
| 930 EXPECT_FALSE(render_text->IsValidCursorIndex(1)); | |
| 931 #endif | |
| 932 EXPECT_TRUE(render_text->SelectRange(Range(2, 1))); | |
| 933 EXPECT_EQ(Range(2, 1), render_text->selection()); | |
| 934 EXPECT_EQ(1U, render_text->cursor_position()); | |
| 935 // Although selection bounds may be set within a multi-character grapheme, | |
| 936 // cursor movement (e.g. via arrow key) should avoid those indices. | |
| 937 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_LEFT, false); | |
| 938 EXPECT_EQ(0U, render_text->cursor_position()); | |
| 939 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_RIGHT, false); | |
| 940 EXPECT_EQ(2U, render_text->cursor_position()); | |
| 941 } | |
| 942 } | |
| 943 | |
| 944 TEST_F(RenderTextTest, EdgeSelectionModels) { | 918 TEST_F(RenderTextTest, EdgeSelectionModels) { |
| 945 // Simple Latin text. | 919 // Simple Latin text. |
| 946 const base::string16 kLatin = WideToUTF16(L"abc"); | 920 const base::string16 kLatin = WideToUTF16(L"abc"); |
| 947 // LTR 2-character grapheme. | 921 // LTR 2-character grapheme. |
| 948 const base::string16 kLTRGrapheme = WideToUTF16(L"\x0915\x093f"); | 922 const base::string16 kLTRGrapheme = WideToUTF16(L"\x0915\x093f"); |
| 949 // LTR 2-character grapheme, LTR a, LTR 2-character grapheme. | 923 // LTR 2-character grapheme, LTR a, LTR 2-character grapheme. |
| 950 const base::string16 kHindiLatin = | 924 const base::string16 kHindiLatin = |
| 951 WideToUTF16(L"\x0915\x093f" L"a" L"\x0915\x093f"); | 925 WideToUTF16(L"\x0915\x093f" L"a" L"\x0915\x093f"); |
| 952 // RTL 2-character grapheme. | 926 // RTL 2-character grapheme. |
| 953 const base::string16 kRTLGrapheme = WideToUTF16(L"\x05e0\x05b8"); | 927 const base::string16 kRTLGrapheme = WideToUTF16(L"\x05e0\x05b8"); |
| (...skipping 949 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1903 render_text->SetText(WideToUTF16(L"x \x25B6 y")); | 1877 render_text->SetText(WideToUTF16(L"x \x25B6 y")); |
| 1904 render_text->EnsureLayout(); | 1878 render_text->EnsureLayout(); |
| 1905 ASSERT_EQ(3U, render_text->runs_.size()); | 1879 ASSERT_EQ(3U, render_text->runs_.size()); |
| 1906 EXPECT_EQ(Range(0, 2), render_text->runs_[0]->range); | 1880 EXPECT_EQ(Range(0, 2), render_text->runs_[0]->range); |
| 1907 EXPECT_EQ(Range(2, 3), render_text->runs_[1]->range); | 1881 EXPECT_EQ(Range(2, 3), render_text->runs_[1]->range); |
| 1908 EXPECT_EQ(Range(3, 5), render_text->runs_[2]->range); | 1882 EXPECT_EQ(Range(3, 5), render_text->runs_[2]->range); |
| 1909 } | 1883 } |
| 1910 #endif // defined(OS_WIN) | 1884 #endif // defined(OS_WIN) |
| 1911 | 1885 |
| 1912 } // namespace gfx | 1886 } // namespace gfx |
| OLD | NEW |