Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(88)

Side by Side Diff: ui/gfx/render_text_unittest.cc

Issue 252563003: Fix Views inline autocomplete with multi-char graphemes. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Address comments. Created 6 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
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
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(gfx::Range(index))| - as that would return true when 47 // range.Contains(Range(index)), which returns true if |index| == |range.end()|.
48 // |index| == |range.end()|.
49 bool IndexInRange(const Range& range, size_t index) { 48 bool IndexInRange(const Range& range, size_t index) {
50 return index >= range.start() && index < range.end(); 49 return index >= range.start() && index < range.end();
51 } 50 }
52 51
53 base::string16 GetSelectedText(RenderText* render_text) { 52 base::string16 GetSelectedText(RenderText* render_text) {
54 return render_text->text().substr(render_text->selection().GetMin(), 53 return render_text->text().substr(render_text->selection().GetMin(),
55 render_text->selection().length()); 54 render_text->selection().length());
56 } 55 }
57 56
58 // A test utility function to set the application default text direction. 57 // A test utility function to set the application default text direction.
(...skipping 229 matching lines...) Expand 10 before | Expand all | Expand 10 after
288 EXPECT_EQ(0U, render_text->cursor_position()); 287 EXPECT_EQ(0U, render_text->cursor_position());
289 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_RIGHT, false); 288 render_text->MoveCursor(CHARACTER_BREAK, CURSOR_RIGHT, false);
290 EXPECT_EQ(2U, render_text->cursor_position()); 289 EXPECT_EQ(2U, render_text->cursor_position());
291 290
292 // Test index conversion and cursor validity with a valid surrogate pair. 291 // Test index conversion and cursor validity with a valid surrogate pair.
293 EXPECT_EQ(0U, render_text->TextIndexToLayoutIndex(0U)); 292 EXPECT_EQ(0U, render_text->TextIndexToLayoutIndex(0U));
294 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(1U)); 293 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(1U));
295 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(2U)); 294 EXPECT_EQ(1U, render_text->TextIndexToLayoutIndex(2U));
296 EXPECT_EQ(0U, render_text->LayoutIndexToTextIndex(0U)); 295 EXPECT_EQ(0U, render_text->LayoutIndexToTextIndex(0U));
297 EXPECT_EQ(2U, render_text->LayoutIndexToTextIndex(1U)); 296 EXPECT_EQ(2U, render_text->LayoutIndexToTextIndex(1U));
298 EXPECT_TRUE(render_text->IsCursorablePosition(0U)); 297 EXPECT_TRUE(render_text->IsValidCursorIndex(0U));
299 EXPECT_FALSE(render_text->IsCursorablePosition(1U)); 298 EXPECT_FALSE(render_text->IsValidCursorIndex(1U));
300 EXPECT_TRUE(render_text->IsCursorablePosition(2U)); 299 EXPECT_TRUE(render_text->IsValidCursorIndex(2U));
301 300
302 // FindCursorPosition() should not return positions between a surrogate pair. 301 // FindCursorPosition() should not return positions between a surrogate pair.
303 render_text->SetDisplayRect(Rect(0, 0, 20, 20)); 302 render_text->SetDisplayRect(Rect(0, 0, 20, 20));
304 EXPECT_EQ(render_text->FindCursorPosition(Point(0, 0)).caret_pos(), 0U); 303 EXPECT_EQ(render_text->FindCursorPosition(Point(0, 0)).caret_pos(), 0U);
305 EXPECT_EQ(render_text->FindCursorPosition(Point(20, 0)).caret_pos(), 2U); 304 EXPECT_EQ(render_text->FindCursorPosition(Point(20, 0)).caret_pos(), 2U);
306 for (int x = -1; x <= 20; ++x) { 305 for (int x = -1; x <= 20; ++x) {
307 SelectionModel selection = render_text->FindCursorPosition(Point(x, 0)); 306 SelectionModel selection = render_text->FindCursorPosition(Point(x, 0));
308 EXPECT_TRUE(selection.caret_pos() == 0U || selection.caret_pos() == 2U); 307 EXPECT_TRUE(selection.caret_pos() == 0U || selection.caret_pos() == 2U);
309 } 308 }
310 309
(...skipping 125 matching lines...) Expand 10 before | Expand all | Expand 10 after
436 // Surrogate pairs should be elided reasonably enough. 435 // Surrogate pairs should be elided reasonably enough.
437 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x05c1\x05b8", false }, 436 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x05c1\x05b8", false },
438 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x2026" , true }, 437 { L"0\x05e9\x05bc\x05c1\x05b8", L"0\x05e9\x05bc\x2026" , true },
439 { L"01\x05e9\x05bc\x05c1\x05b8", L"01\x05e9\x2026" , true }, 438 { L"01\x05e9\x05bc\x05c1\x05b8", L"01\x05e9\x2026" , true },
440 { L"012\x05e9\x05bc\x05c1\x05b8", L"012\x2026\x200E" , true }, 439 { L"012\x05e9\x05bc\x05c1\x05b8", L"012\x2026\x200E" , true },
441 { L"012\xF0\x9D\x84\x9E", L"012\xF0\x2026" , true }, 440 { L"012\xF0\x9D\x84\x9E", L"012\xF0\x2026" , true },
442 }; 441 };
443 442
444 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); 443 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance());
445 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); 444 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px"));
446 expected_render_text->SetDisplayRect(gfx::Rect(0, 0, 9999, 100)); 445 expected_render_text->SetDisplayRect(Rect(0, 0, 9999, 100));
447 446
448 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); 447 scoped_ptr<RenderText> render_text(RenderText::CreateInstance());
449 render_text->SetFontList(FontList("serif, Sans serif, 12px")); 448 render_text->SetFontList(FontList("serif, Sans serif, 12px"));
450 render_text->SetElideBehavior(gfx::ELIDE_AT_END); 449 render_text->SetElideBehavior(ELIDE_AT_END);
451 450
452 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { 451 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) {
453 // Compute expected width 452 // Compute expected width
454 expected_render_text->SetText(WideToUTF16(cases[i].layout_text)); 453 expected_render_text->SetText(WideToUTF16(cases[i].layout_text));
455 int expected_width = expected_render_text->GetContentWidth(); 454 int expected_width = expected_render_text->GetContentWidth();
456 455
457 base::string16 input = WideToUTF16(cases[i].text); 456 base::string16 input = WideToUTF16(cases[i].text);
458 // Extend the input text to ensure that it is wider than the layout_text, 457 // Extend the input text to ensure that it is wider than the layout_text,
459 // and so it will get elided. 458 // and so it will get elided.
460 if (cases[i].elision_expected) 459 if (cases[i].elision_expected)
461 input.append(WideToUTF16(L" MMMMMMMMMMM")); 460 input.append(WideToUTF16(L" MMMMMMMMMMM"));
462 461
463 render_text->SetText(input); 462 render_text->SetText(input);
464 render_text->SetDisplayRect(gfx::Rect(0, 0, expected_width, 100)); 463 render_text->SetDisplayRect(Rect(0, 0, expected_width, 100));
465 EXPECT_EQ(input, render_text->text()) 464 EXPECT_EQ(input, render_text->text())
466 << "->For case " << i << ": " << cases[i].text << "\n"; 465 << "->For case " << i << ": " << cases[i].text << "\n";
467 EXPECT_EQ(WideToUTF16(cases[i].layout_text), render_text->GetLayoutText()) 466 EXPECT_EQ(WideToUTF16(cases[i].layout_text), render_text->GetLayoutText())
468 << "->For case " << i << ": " << cases[i].text << "\n"; 467 << "->For case " << i << ": " << cases[i].text << "\n";
469 expected_render_text->SetText(base::string16()); 468 expected_render_text->SetText(base::string16());
470 } 469 }
471 } 470 }
472 471
473 TEST_F(RenderTextTest, ElidedObscuredText) { 472 TEST_F(RenderTextTest, ElidedObscuredText) {
474 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance()); 473 scoped_ptr<RenderText> expected_render_text(RenderText::CreateInstance());
475 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px")); 474 expected_render_text->SetFontList(FontList("serif, Sans serif, 12px"));
476 expected_render_text->SetDisplayRect(gfx::Rect(0, 0, 9999, 100)); 475 expected_render_text->SetDisplayRect(Rect(0, 0, 9999, 100));
477 expected_render_text->SetText(WideToUTF16(L"**\x2026")); 476 expected_render_text->SetText(WideToUTF16(L"**\x2026"));
478 477
479 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); 478 scoped_ptr<RenderText> render_text(RenderText::CreateInstance());
480 render_text->SetFontList(FontList("serif, Sans serif, 12px")); 479 render_text->SetFontList(FontList("serif, Sans serif, 12px"));
481 render_text->SetElideBehavior(gfx::ELIDE_AT_END); 480 render_text->SetElideBehavior(ELIDE_AT_END);
482 render_text->SetDisplayRect( 481 render_text->SetDisplayRect(
483 gfx::Rect(0, 0, expected_render_text->GetContentWidth(), 100)); 482 Rect(0, 0, expected_render_text->GetContentWidth(), 100));
484 render_text->SetObscured(true); 483 render_text->SetObscured(true);
485 render_text->SetText(WideToUTF16(L"abcdef")); 484 render_text->SetText(WideToUTF16(L"abcdef"));
486 EXPECT_EQ(WideToUTF16(L"abcdef"), render_text->text()); 485 EXPECT_EQ(WideToUTF16(L"abcdef"), render_text->text());
487 EXPECT_EQ(WideToUTF16(L"**\x2026"), render_text->GetLayoutText()); 486 EXPECT_EQ(WideToUTF16(L"**\x2026"), render_text->GetLayoutText());
488 } 487 }
489 488
490 TEST_F(RenderTextTest, TruncatedText) { 489 TEST_F(RenderTextTest, TruncatedText) {
491 struct { 490 struct {
492 const wchar_t* text; 491 const wchar_t* text;
493 const wchar_t* layout_text; 492 const wchar_t* layout_text;
(...skipping 400 matching lines...) Expand 10 before | Expand all | Expand 10 after
894 893
895 // TODO(asvitkine): Disable tests that fail on XP bots due to lack of complete 894 // TODO(asvitkine): Disable tests that fail on XP bots due to lack of complete
896 // font support for some scripts - http://crbug.com/106450 895 // font support for some scripts - http://crbug.com/106450
897 #if defined(OS_WIN) 896 #if defined(OS_WIN)
898 if (base::win::GetVersion() < base::win::VERSION_VISTA) 897 if (base::win::GetVersion() < base::win::VERSION_VISTA)
899 return; 898 return;
900 #endif 899 #endif
901 900
902 scoped_ptr<RenderText> render_text(RenderText::CreateInstance()); 901 scoped_ptr<RenderText> render_text(RenderText::CreateInstance());
903 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) { 902 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->IsCursorablePosition(next)); 909 EXPECT_TRUE(render_text->IsValidCursorIndex(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->IsCursorablePosition(previous)); 914 EXPECT_TRUE(render_text->IsValidCursorIndex(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 base::string16 cases[] = { kHindi, kThai };
Alexei Svitkine (slow) 2014/05/01 20:48:09 Nit: Can this be a "const base::string16& cases[]
msw 2014/05/01 22:43:22 This yields "error: 'cases' declared as array of r
923
924 scoped_ptr<RenderText> render_text(RenderText::CreateInstance());
925 for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) {
Alexei Svitkine (slow) 2014/05/01 20:48:09 Nit: I think you can just use arraysize(cases) her
msw 2014/05/01 22:43:22 Done.
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
918 TEST_F(RenderTextTest, EdgeSelectionModels) { 944 TEST_F(RenderTextTest, EdgeSelectionModels) {
919 // Simple Latin text. 945 // Simple Latin text.
920 const base::string16 kLatin = WideToUTF16(L"abc"); 946 const base::string16 kLatin = WideToUTF16(L"abc");
921 // LTR 2-character grapheme. 947 // LTR 2-character grapheme.
922 const base::string16 kLTRGrapheme = WideToUTF16(L"\x0915\x093f"); 948 const base::string16 kLTRGrapheme = WideToUTF16(L"\x0915\x093f");
923 // LTR 2-character grapheme, LTR a, LTR 2-character grapheme. 949 // LTR 2-character grapheme, LTR a, LTR 2-character grapheme.
924 const base::string16 kHindiLatin = 950 const base::string16 kHindiLatin =
925 WideToUTF16(L"\x0915\x093f" L"a" L"\x0915\x093f"); 951 WideToUTF16(L"\x0915\x093f" L"a" L"\x0915\x093f");
926 // RTL 2-character grapheme. 952 // RTL 2-character grapheme.
927 const base::string16 kRTLGrapheme = WideToUTF16(L"\x05e0\x05b8"); 953 const base::string16 kRTLGrapheme = WideToUTF16(L"\x05e0\x05b8");
(...skipping 949 matching lines...) Expand 10 before | Expand all | Expand 10 after
1877 render_text->SetText(WideToUTF16(L"x \x25B6 y")); 1903 render_text->SetText(WideToUTF16(L"x \x25B6 y"));
1878 render_text->EnsureLayout(); 1904 render_text->EnsureLayout();
1879 ASSERT_EQ(3U, render_text->runs_.size()); 1905 ASSERT_EQ(3U, render_text->runs_.size());
1880 EXPECT_EQ(Range(0, 2), render_text->runs_[0]->range); 1906 EXPECT_EQ(Range(0, 2), render_text->runs_[0]->range);
1881 EXPECT_EQ(Range(2, 3), render_text->runs_[1]->range); 1907 EXPECT_EQ(Range(2, 3), render_text->runs_[1]->range);
1882 EXPECT_EQ(Range(3, 5), render_text->runs_[2]->range); 1908 EXPECT_EQ(Range(3, 5), render_text->runs_[2]->range);
1883 } 1909 }
1884 #endif // defined(OS_WIN) 1910 #endif // defined(OS_WIN)
1885 1911
1886 } // namespace gfx 1912 } // namespace gfx
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698