| OLD | NEW |
| (Empty) |
| 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 | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "ui/views/controls/textfield/native_textfield_views.h" | |
| 6 | |
| 7 #include <set> | |
| 8 #include <string> | |
| 9 #include <vector> | |
| 10 | |
| 11 #include "base/auto_reset.h" | |
| 12 #include "base/bind.h" | |
| 13 #include "base/bind_helpers.h" | |
| 14 #include "base/callback.h" | |
| 15 #include "base/command_line.h" | |
| 16 #include "base/message_loop/message_loop.h" | |
| 17 #include "base/pickle.h" | |
| 18 #include "base/strings/string16.h" | |
| 19 #include "base/strings/utf_string_conversions.h" | |
| 20 #include "grit/ui_strings.h" | |
| 21 #include "testing/gtest/include/gtest/gtest.h" | |
| 22 #include "ui/base/clipboard/clipboard.h" | |
| 23 #include "ui/base/clipboard/scoped_clipboard_writer.h" | |
| 24 #include "ui/base/dragdrop/drag_drop_types.h" | |
| 25 #include "ui/base/ime/text_input_client.h" | |
| 26 #include "ui/base/l10n/l10n_util.h" | |
| 27 #include "ui/base/ui_base_switches.h" | |
| 28 #include "ui/events/event.h" | |
| 29 #include "ui/events/keycodes/keyboard_codes.h" | |
| 30 #include "ui/gfx/render_text.h" | |
| 31 #include "ui/views/controls/textfield/textfield.h" | |
| 32 #include "ui/views/controls/textfield/textfield_controller.h" | |
| 33 #include "ui/views/controls/textfield/textfield_views_model.h" | |
| 34 #include "ui/views/focus/focus_manager.h" | |
| 35 #include "ui/views/ime/mock_input_method.h" | |
| 36 #include "ui/views/test/test_views_delegate.h" | |
| 37 #include "ui/views/test/views_test_base.h" | |
| 38 #include "ui/views/widget/native_widget_private.h" | |
| 39 #include "ui/views/widget/widget.h" | |
| 40 #include "url/gurl.h" | |
| 41 | |
| 42 #if defined(OS_WIN) | |
| 43 #include "base/win/windows_version.h" | |
| 44 #endif | |
| 45 | |
| 46 using base::ASCIIToUTF16; | |
| 47 using base::UTF8ToUTF16; | |
| 48 using base::WideToUTF16; | |
| 49 | |
| 50 #define EXPECT_STR_EQ(ascii, utf16) EXPECT_EQ(ASCIIToUTF16(ascii), utf16) | |
| 51 | |
| 52 namespace { | |
| 53 | |
| 54 const base::char16 kHebrewLetterSamekh = 0x05E1; | |
| 55 | |
| 56 // A Textfield wrapper to intercept OnKey[Pressed|Released]() ressults. | |
| 57 class TestTextfield : public views::Textfield { | |
| 58 public: | |
| 59 explicit TestTextfield(StyleFlags style) | |
| 60 : Textfield(style), | |
| 61 key_handled_(false), | |
| 62 key_received_(false) { | |
| 63 } | |
| 64 | |
| 65 virtual bool OnKeyPressed(const ui::KeyEvent& e) OVERRIDE { | |
| 66 key_received_ = true; | |
| 67 key_handled_ = views::Textfield::OnKeyPressed(e); | |
| 68 return key_handled_; | |
| 69 } | |
| 70 | |
| 71 virtual bool OnKeyReleased(const ui::KeyEvent& e) OVERRIDE { | |
| 72 key_received_ = true; | |
| 73 key_handled_ = views::Textfield::OnKeyReleased(e); | |
| 74 return key_handled_; | |
| 75 } | |
| 76 | |
| 77 bool key_handled() const { return key_handled_; } | |
| 78 bool key_received() const { return key_received_; } | |
| 79 | |
| 80 void clear() { key_received_ = key_handled_ = false; } | |
| 81 | |
| 82 private: | |
| 83 bool key_handled_; | |
| 84 bool key_received_; | |
| 85 | |
| 86 DISALLOW_COPY_AND_ASSIGN(TestTextfield); | |
| 87 }; | |
| 88 | |
| 89 // A helper class for use with ui::TextInputClient::GetTextFromRange(). | |
| 90 class GetTextHelper { | |
| 91 public: | |
| 92 GetTextHelper() {} | |
| 93 | |
| 94 void set_text(const base::string16& text) { text_ = text; } | |
| 95 const base::string16& text() const { return text_; } | |
| 96 | |
| 97 private: | |
| 98 base::string16 text_; | |
| 99 | |
| 100 DISALLOW_COPY_AND_ASSIGN(GetTextHelper); | |
| 101 }; | |
| 102 | |
| 103 // Convenience to make constructing a GestureEvent simpler. | |
| 104 class GestureEventForTest : public ui::GestureEvent { | |
| 105 public: | |
| 106 GestureEventForTest(ui::EventType type, int x, int y, float delta_x, | |
| 107 float delta_y) | |
| 108 : GestureEvent(type, x, y, 0, base::TimeDelta(), | |
| 109 ui::GestureEventDetails(type, delta_x, delta_y), 0) { | |
| 110 } | |
| 111 | |
| 112 private: | |
| 113 DISALLOW_COPY_AND_ASSIGN(GestureEventForTest); | |
| 114 }; | |
| 115 | |
| 116 } // namespace | |
| 117 | |
| 118 namespace views { | |
| 119 | |
| 120 // TODO(oshima): Move tests that are independent of TextfieldViews to | |
| 121 // textfield_unittests.cc once we move the test utility functions | |
| 122 // from chrome/browser/automation/ to ui/base/test/. | |
| 123 class NativeTextfieldViewsTest : public ViewsTestBase, | |
| 124 public TextfieldController { | |
| 125 public: | |
| 126 NativeTextfieldViewsTest() | |
| 127 : widget_(NULL), | |
| 128 textfield_(NULL), | |
| 129 textfield_view_(NULL), | |
| 130 model_(NULL), | |
| 131 input_method_(NULL), | |
| 132 on_before_user_action_(0), | |
| 133 on_after_user_action_(0) { | |
| 134 } | |
| 135 | |
| 136 // ::testing::Test: | |
| 137 virtual void SetUp() { | |
| 138 ViewsTestBase::SetUp(); | |
| 139 } | |
| 140 | |
| 141 virtual void TearDown() { | |
| 142 if (widget_) | |
| 143 widget_->Close(); | |
| 144 ViewsTestBase::TearDown(); | |
| 145 } | |
| 146 | |
| 147 // TextfieldController: | |
| 148 virtual void ContentsChanged(Textfield* sender, | |
| 149 const base::string16& new_contents) OVERRIDE { | |
| 150 // Paste calls TextfieldController::ContentsChanged() explicitly even if the | |
| 151 // paste action did not change the content. So |new_contents| may match | |
| 152 // |last_contents_|. For more info, see http://crbug.com/79002 | |
| 153 last_contents_ = new_contents; | |
| 154 } | |
| 155 | |
| 156 virtual bool HandleKeyEvent(Textfield* sender, | |
| 157 const ui::KeyEvent& key_event) OVERRIDE { | |
| 158 // TODO(oshima): figure out how to test the keystroke. | |
| 159 return false; | |
| 160 } | |
| 161 | |
| 162 virtual void OnBeforeUserAction(Textfield* sender) OVERRIDE { | |
| 163 ++on_before_user_action_; | |
| 164 } | |
| 165 | |
| 166 virtual void OnAfterUserAction(Textfield* sender) OVERRIDE { | |
| 167 ++on_after_user_action_; | |
| 168 } | |
| 169 | |
| 170 void InitTextfield(Textfield::StyleFlags style) { | |
| 171 InitTextfields(style, 1); | |
| 172 } | |
| 173 | |
| 174 void InitTextfields(Textfield::StyleFlags style, int count) { | |
| 175 ASSERT_FALSE(textfield_); | |
| 176 textfield_ = new TestTextfield(style); | |
| 177 textfield_->SetController(this); | |
| 178 widget_ = new Widget(); | |
| 179 Widget::InitParams params = CreateParams(Widget::InitParams::TYPE_POPUP); | |
| 180 params.bounds = gfx::Rect(100, 100, 100, 100); | |
| 181 widget_->Init(params); | |
| 182 View* container = new View(); | |
| 183 widget_->SetContentsView(container); | |
| 184 container->AddChildView(textfield_); | |
| 185 | |
| 186 textfield_view_ = textfield_->GetTextfieldViewForTesting(); | |
| 187 DCHECK(textfield_view_); | |
| 188 textfield_view_->SetBoundsRect(params.bounds); | |
| 189 textfield_->set_id(1); | |
| 190 | |
| 191 for (int i = 1; i < count; i++) { | |
| 192 Textfield* textfield = new Textfield(style); | |
| 193 container->AddChildView(textfield); | |
| 194 textfield->set_id(i + 1); | |
| 195 } | |
| 196 | |
| 197 model_ = textfield_view_->model_.get(); | |
| 198 model_->ClearEditHistory(); | |
| 199 | |
| 200 input_method_ = new MockInputMethod(); | |
| 201 widget_->ReplaceInputMethod(input_method_); | |
| 202 | |
| 203 // Activate the widget and focus the textfield for input handling. | |
| 204 widget_->Activate(); | |
| 205 textfield_->RequestFocus(); | |
| 206 } | |
| 207 | |
| 208 ui::MenuModel* GetContextMenuModel() { | |
| 209 textfield_view_->UpdateContextMenu(); | |
| 210 return textfield_view_->context_menu_contents_.get(); | |
| 211 } | |
| 212 | |
| 213 ui::TouchSelectionController* GetTouchSelectionController() { | |
| 214 return textfield_view_->touch_selection_controller_.get(); | |
| 215 } | |
| 216 | |
| 217 protected: | |
| 218 void SendKeyEvent(ui::KeyboardCode key_code, | |
| 219 bool alt, | |
| 220 bool shift, | |
| 221 bool control, | |
| 222 bool caps_lock) { | |
| 223 int flags = (alt ? ui::EF_ALT_DOWN : 0) | | |
| 224 (shift ? ui::EF_SHIFT_DOWN : 0) | | |
| 225 (control ? ui::EF_CONTROL_DOWN : 0) | | |
| 226 (caps_lock ? ui::EF_CAPS_LOCK_DOWN : 0); | |
| 227 ui::KeyEvent event(ui::ET_KEY_PRESSED, key_code, flags, false); | |
| 228 input_method_->DispatchKeyEvent(event); | |
| 229 } | |
| 230 | |
| 231 void SendKeyEvent(ui::KeyboardCode key_code, bool shift, bool control) { | |
| 232 SendKeyEvent(key_code, false, shift, control, false); | |
| 233 } | |
| 234 | |
| 235 void SendKeyEvent(ui::KeyboardCode key_code) { | |
| 236 SendKeyEvent(key_code, false, false); | |
| 237 } | |
| 238 | |
| 239 void SendKeyEvent(base::char16 ch) { | |
| 240 if (ch < 0x80) { | |
| 241 ui::KeyboardCode code = | |
| 242 ch == ' ' ? ui::VKEY_SPACE : | |
| 243 static_cast<ui::KeyboardCode>(ui::VKEY_A + ch - 'a'); | |
| 244 SendKeyEvent(code); | |
| 245 } else { | |
| 246 ui::KeyEvent event(ui::ET_KEY_PRESSED, ui::VKEY_UNKNOWN, 0, false); | |
| 247 event.set_character(ch); | |
| 248 input_method_->DispatchKeyEvent(event); | |
| 249 } | |
| 250 } | |
| 251 | |
| 252 base::string16 GetClipboardText() const { | |
| 253 base::string16 text; | |
| 254 ui::Clipboard::GetForCurrentThread()-> | |
| 255 ReadText(ui::CLIPBOARD_TYPE_COPY_PASTE, &text); | |
| 256 return text; | |
| 257 } | |
| 258 | |
| 259 void SetClipboardText(const std::string& text) { | |
| 260 ui::ScopedClipboardWriter clipboard_writer( | |
| 261 ui::Clipboard::GetForCurrentThread(), | |
| 262 ui::CLIPBOARD_TYPE_COPY_PASTE); | |
| 263 clipboard_writer.WriteText(ASCIIToUTF16(text)); | |
| 264 } | |
| 265 | |
| 266 View* GetFocusedView() { | |
| 267 return widget_->GetFocusManager()->GetFocusedView(); | |
| 268 } | |
| 269 | |
| 270 int GetCursorPositionX(int cursor_pos) { | |
| 271 gfx::RenderText* render_text = textfield_view_->GetRenderText(); | |
| 272 return render_text->GetCursorBounds( | |
| 273 gfx::SelectionModel(cursor_pos, gfx::CURSOR_FORWARD), false).x(); | |
| 274 } | |
| 275 | |
| 276 // Get the current cursor bounds. | |
| 277 gfx::Rect GetCursorBounds() { | |
| 278 gfx::RenderText* render_text = textfield_view_->GetRenderText(); | |
| 279 gfx::Rect bounds = render_text->GetUpdatedCursorBounds(); | |
| 280 return bounds; | |
| 281 } | |
| 282 | |
| 283 // Get the cursor bounds of |sel|. | |
| 284 gfx::Rect GetCursorBounds(const gfx::SelectionModel& sel) { | |
| 285 gfx::RenderText* render_text = textfield_view_->GetRenderText(); | |
| 286 gfx::Rect bounds = render_text->GetCursorBounds(sel, true); | |
| 287 return bounds; | |
| 288 } | |
| 289 | |
| 290 gfx::Rect GetDisplayRect() { | |
| 291 return textfield_view_->GetRenderText()->display_rect(); | |
| 292 } | |
| 293 | |
| 294 // Mouse click on the point whose x-axis is |bound|'s x plus |x_offset| and | |
| 295 // y-axis is in the middle of |bound|'s vertical range. | |
| 296 void MouseClick(const gfx::Rect bound, int x_offset) { | |
| 297 gfx::Point point(bound.x() + x_offset, bound.y() + bound.height() / 2); | |
| 298 ui::MouseEvent click(ui::ET_MOUSE_PRESSED, point, point, | |
| 299 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 300 textfield_view_->OnMousePressed(click); | |
| 301 ui::MouseEvent release(ui::ET_MOUSE_RELEASED, point, point, | |
| 302 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 303 textfield_view_->OnMouseReleased(release); | |
| 304 } | |
| 305 | |
| 306 // This is to avoid double/triple click. | |
| 307 void NonClientMouseClick() { | |
| 308 ui::MouseEvent click(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(), | |
| 309 ui::EF_LEFT_MOUSE_BUTTON | ui::EF_IS_NON_CLIENT, | |
| 310 ui::EF_LEFT_MOUSE_BUTTON); | |
| 311 textfield_view_->OnMousePressed(click); | |
| 312 ui::MouseEvent release(ui::ET_MOUSE_RELEASED, gfx::Point(), gfx::Point(), | |
| 313 ui::EF_LEFT_MOUSE_BUTTON | ui::EF_IS_NON_CLIENT, | |
| 314 ui::EF_LEFT_MOUSE_BUTTON); | |
| 315 textfield_view_->OnMouseReleased(release); | |
| 316 } | |
| 317 | |
| 318 // Wrap for visibility in test classes. | |
| 319 ui::TextInputType GetTextInputType() { | |
| 320 return textfield_view_->GetTextInputType(); | |
| 321 } | |
| 322 | |
| 323 void VerifyTextfieldContextMenuContents(bool textfield_has_selection, | |
| 324 bool can_undo, | |
| 325 ui::MenuModel* menu) { | |
| 326 EXPECT_EQ(can_undo, menu->IsEnabledAt(0 /* UNDO */)); | |
| 327 EXPECT_TRUE(menu->IsEnabledAt(1 /* Separator */)); | |
| 328 EXPECT_EQ(textfield_has_selection, menu->IsEnabledAt(2 /* CUT */)); | |
| 329 EXPECT_EQ(textfield_has_selection, menu->IsEnabledAt(3 /* COPY */)); | |
| 330 EXPECT_NE(GetClipboardText().empty(), menu->IsEnabledAt(4 /* PASTE */)); | |
| 331 EXPECT_EQ(textfield_has_selection, menu->IsEnabledAt(5 /* DELETE */)); | |
| 332 EXPECT_TRUE(menu->IsEnabledAt(6 /* Separator */)); | |
| 333 EXPECT_TRUE(menu->IsEnabledAt(7 /* SELECT ALL */)); | |
| 334 } | |
| 335 | |
| 336 // We need widget to populate wrapper class. | |
| 337 Widget* widget_; | |
| 338 | |
| 339 TestTextfield* textfield_; | |
| 340 NativeTextfieldViews* textfield_view_; | |
| 341 TextfieldViewsModel* model_; | |
| 342 | |
| 343 // The string from Controller::ContentsChanged callback. | |
| 344 base::string16 last_contents_; | |
| 345 | |
| 346 // For testing input method related behaviors. | |
| 347 MockInputMethod* input_method_; | |
| 348 | |
| 349 // Indicates how many times OnBeforeUserAction() is called. | |
| 350 int on_before_user_action_; | |
| 351 | |
| 352 // Indicates how many times OnAfterUserAction() is called. | |
| 353 int on_after_user_action_; | |
| 354 | |
| 355 private: | |
| 356 DISALLOW_COPY_AND_ASSIGN(NativeTextfieldViewsTest); | |
| 357 }; | |
| 358 | |
| 359 TEST_F(NativeTextfieldViewsTest, ModelChangesTest) { | |
| 360 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 361 | |
| 362 // TextfieldController::ContentsChanged() shouldn't be called when changing | |
| 363 // text programmatically. | |
| 364 last_contents_.clear(); | |
| 365 textfield_->SetText(ASCIIToUTF16("this is")); | |
| 366 | |
| 367 EXPECT_STR_EQ("this is", model_->GetText()); | |
| 368 EXPECT_STR_EQ("this is", textfield_->text()); | |
| 369 EXPECT_TRUE(last_contents_.empty()); | |
| 370 | |
| 371 textfield_->AppendText(ASCIIToUTF16(" a test")); | |
| 372 EXPECT_STR_EQ("this is a test", model_->GetText()); | |
| 373 EXPECT_STR_EQ("this is a test", textfield_->text()); | |
| 374 EXPECT_TRUE(last_contents_.empty()); | |
| 375 | |
| 376 EXPECT_EQ(base::string16(), textfield_->GetSelectedText()); | |
| 377 textfield_->SelectAll(false); | |
| 378 EXPECT_STR_EQ("this is a test", textfield_->GetSelectedText()); | |
| 379 EXPECT_TRUE(last_contents_.empty()); | |
| 380 } | |
| 381 | |
| 382 TEST_F(NativeTextfieldViewsTest, ModelChangesTestLowerCase) { | |
| 383 // Check if |model_|'s text is properly lowercased for STYLE_LOWERCASE. | |
| 384 InitTextfield(Textfield::STYLE_LOWERCASE); | |
| 385 EXPECT_EQ(0U, textfield_->GetCursorPosition()); | |
| 386 | |
| 387 last_contents_.clear(); | |
| 388 textfield_->SetText(ASCIIToUTF16("THIS IS")); | |
| 389 EXPECT_EQ(7U, textfield_->GetCursorPosition()); | |
| 390 | |
| 391 EXPECT_STR_EQ("this is", model_->GetText()); | |
| 392 EXPECT_STR_EQ("THIS IS", textfield_->text()); | |
| 393 EXPECT_TRUE(last_contents_.empty()); | |
| 394 | |
| 395 textfield_->AppendText(ASCIIToUTF16(" A TEST")); | |
| 396 EXPECT_EQ(7U, textfield_->GetCursorPosition()); | |
| 397 EXPECT_STR_EQ("this is a test", model_->GetText()); | |
| 398 EXPECT_STR_EQ("THIS IS A TEST", textfield_->text()); | |
| 399 | |
| 400 EXPECT_TRUE(last_contents_.empty()); | |
| 401 } | |
| 402 | |
| 403 TEST_F(NativeTextfieldViewsTest, ModelChangesTestLowerCaseI18n) { | |
| 404 // Check if lower case conversion works for non-ASCII characters. | |
| 405 InitTextfield(Textfield::STYLE_LOWERCASE); | |
| 406 EXPECT_EQ(0U, textfield_->GetCursorPosition()); | |
| 407 | |
| 408 last_contents_.clear(); | |
| 409 // Zenkaku Japanese "ABCabc" | |
| 410 textfield_->SetText(WideToUTF16(L"\xFF21\xFF22\xFF23\xFF41\xFF42\xFF43")); | |
| 411 EXPECT_EQ(6U, textfield_->GetCursorPosition()); | |
| 412 // Zenkaku Japanese "abcabc" | |
| 413 EXPECT_EQ(WideToUTF16(L"\xFF41\xFF42\xFF43\xFF41\xFF42\xFF43"), | |
| 414 model_->GetText()); | |
| 415 // Zenkaku Japanese "ABCabc" | |
| 416 EXPECT_EQ(WideToUTF16(L"\xFF21\xFF22\xFF23\xFF41\xFF42\xFF43"), | |
| 417 textfield_->text()); | |
| 418 EXPECT_TRUE(last_contents_.empty()); | |
| 419 | |
| 420 // Zenkaku Japanese "XYZxyz" | |
| 421 textfield_->AppendText(WideToUTF16(L"\xFF38\xFF39\xFF3A\xFF58\xFF59\xFF5A")); | |
| 422 EXPECT_EQ(6U, textfield_->GetCursorPosition()); | |
| 423 // Zenkaku Japanese "abcabcxyzxyz" | |
| 424 EXPECT_EQ(WideToUTF16(L"\xFF41\xFF42\xFF43\xFF41\xFF42\xFF43" | |
| 425 L"\xFF58\xFF59\xFF5A\xFF58\xFF59\xFF5A"), | |
| 426 model_->GetText()); | |
| 427 // Zenkaku Japanese "ABCabcXYZxyz" | |
| 428 EXPECT_EQ(WideToUTF16(L"\xFF21\xFF22\xFF23\xFF41\xFF42\xFF43" | |
| 429 L"\xFF38\xFF39\xFF3A\xFF58\xFF59\xFF5A"), | |
| 430 textfield_->text()); | |
| 431 EXPECT_TRUE(last_contents_.empty()); | |
| 432 } | |
| 433 | |
| 434 TEST_F(NativeTextfieldViewsTest, ModelChangesTestLowerCaseWithLocale) { | |
| 435 // Check if lower case conversion honors locale properly. | |
| 436 std::string locale = l10n_util::GetApplicationLocale(""); | |
| 437 base::i18n::SetICUDefaultLocale("tr"); | |
| 438 | |
| 439 InitTextfield(Textfield::STYLE_LOWERCASE); | |
| 440 EXPECT_EQ(0U, textfield_->GetCursorPosition()); | |
| 441 | |
| 442 last_contents_.clear(); | |
| 443 // Turkish 'I' should be converted to dotless 'i' (U+0131). | |
| 444 textfield_->SetText(WideToUTF16(L"I")); | |
| 445 EXPECT_EQ(1U, textfield_->GetCursorPosition()); | |
| 446 EXPECT_EQ(WideToUTF16(L"\x0131"), model_->GetText()); | |
| 447 EXPECT_EQ(WideToUTF16(L"I"), textfield_->text()); | |
| 448 EXPECT_TRUE(last_contents_.empty()); | |
| 449 | |
| 450 base::i18n::SetICUDefaultLocale(locale); | |
| 451 | |
| 452 // On default (en) locale, 'I' should be converted to 'i'. | |
| 453 textfield_->SetText(WideToUTF16(L"I")); | |
| 454 EXPECT_EQ(1U, textfield_->GetCursorPosition()); | |
| 455 EXPECT_EQ(WideToUTF16(L"i"), model_->GetText()); | |
| 456 EXPECT_EQ(WideToUTF16(L"I"), textfield_->text()); | |
| 457 EXPECT_TRUE(last_contents_.empty()); | |
| 458 } | |
| 459 | |
| 460 TEST_F(NativeTextfieldViewsTest, KeyTest) { | |
| 461 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 462 // Event flags: key, alt, shift, ctrl, caps-lock. | |
| 463 SendKeyEvent(ui::VKEY_T, false, true, false, false); | |
| 464 SendKeyEvent(ui::VKEY_E, false, false, false, false); | |
| 465 SendKeyEvent(ui::VKEY_X, false, true, false, true); | |
| 466 SendKeyEvent(ui::VKEY_T, false, false, false, true); | |
| 467 SendKeyEvent(ui::VKEY_1, false, true, false, false); | |
| 468 SendKeyEvent(ui::VKEY_1, false, false, false, false); | |
| 469 SendKeyEvent(ui::VKEY_1, false, true, false, true); | |
| 470 SendKeyEvent(ui::VKEY_1, false, false, false, true); | |
| 471 EXPECT_STR_EQ("TexT!1!1", textfield_->text()); | |
| 472 } | |
| 473 | |
| 474 TEST_F(NativeTextfieldViewsTest, ControlAndSelectTest) { | |
| 475 // Insert a test string in a textfield. | |
| 476 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 477 textfield_->SetText(ASCIIToUTF16("one two three")); | |
| 478 SendKeyEvent(ui::VKEY_HOME, false /* shift */, false /* control */); | |
| 479 SendKeyEvent(ui::VKEY_RIGHT, true, false); | |
| 480 SendKeyEvent(ui::VKEY_RIGHT, true, false); | |
| 481 SendKeyEvent(ui::VKEY_RIGHT, true, false); | |
| 482 | |
| 483 EXPECT_STR_EQ("one", textfield_->GetSelectedText()); | |
| 484 | |
| 485 // Test word select. | |
| 486 SendKeyEvent(ui::VKEY_RIGHT, true, true); | |
| 487 EXPECT_STR_EQ("one two", textfield_->GetSelectedText()); | |
| 488 SendKeyEvent(ui::VKEY_RIGHT, true, true); | |
| 489 EXPECT_STR_EQ("one two three", textfield_->GetSelectedText()); | |
| 490 SendKeyEvent(ui::VKEY_LEFT, true, true); | |
| 491 EXPECT_STR_EQ("one two ", textfield_->GetSelectedText()); | |
| 492 SendKeyEvent(ui::VKEY_LEFT, true, true); | |
| 493 EXPECT_STR_EQ("one ", textfield_->GetSelectedText()); | |
| 494 | |
| 495 // Replace the selected text. | |
| 496 SendKeyEvent(ui::VKEY_Z, true, false); | |
| 497 SendKeyEvent(ui::VKEY_E, true, false); | |
| 498 SendKeyEvent(ui::VKEY_R, true, false); | |
| 499 SendKeyEvent(ui::VKEY_O, true, false); | |
| 500 SendKeyEvent(ui::VKEY_SPACE, false, false); | |
| 501 EXPECT_STR_EQ("ZERO two three", textfield_->text()); | |
| 502 | |
| 503 SendKeyEvent(ui::VKEY_END, true, false); | |
| 504 EXPECT_STR_EQ("two three", textfield_->GetSelectedText()); | |
| 505 SendKeyEvent(ui::VKEY_HOME, true, false); | |
| 506 EXPECT_STR_EQ("ZERO ", textfield_->GetSelectedText()); | |
| 507 } | |
| 508 | |
| 509 TEST_F(NativeTextfieldViewsTest, InsertionDeletionTest) { | |
| 510 // Insert a test string in a textfield. | |
| 511 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 512 for (size_t i = 0; i < 10; i++) | |
| 513 SendKeyEvent(static_cast<ui::KeyboardCode>(ui::VKEY_A + i)); | |
| 514 EXPECT_STR_EQ("abcdefghij", textfield_->text()); | |
| 515 | |
| 516 // Test the delete and backspace keys. | |
| 517 textfield_->SelectRange(gfx::Range(5)); | |
| 518 for (int i = 0; i < 3; i++) | |
| 519 SendKeyEvent(ui::VKEY_BACK); | |
| 520 EXPECT_STR_EQ("abfghij", textfield_->text()); | |
| 521 for (int i = 0; i < 3; i++) | |
| 522 SendKeyEvent(ui::VKEY_DELETE); | |
| 523 EXPECT_STR_EQ("abij", textfield_->text()); | |
| 524 | |
| 525 // Select all and replace with "k". | |
| 526 textfield_->SelectAll(false); | |
| 527 SendKeyEvent(ui::VKEY_K); | |
| 528 EXPECT_STR_EQ("k", textfield_->text()); | |
| 529 | |
| 530 // Delete the previous word from cursor. | |
| 531 textfield_->SetText(ASCIIToUTF16("one two three four")); | |
| 532 SendKeyEvent(ui::VKEY_END); | |
| 533 SendKeyEvent(ui::VKEY_BACK, false, false, true, false); | |
| 534 EXPECT_STR_EQ("one two three ", textfield_->text()); | |
| 535 | |
| 536 // Delete text preceeding the cursor in chromeos, do nothing in windows. | |
| 537 SendKeyEvent(ui::VKEY_LEFT, false, false, true, false); | |
| 538 SendKeyEvent(ui::VKEY_BACK, false, true, true, false); | |
| 539 #if defined(OS_WIN) | |
| 540 EXPECT_STR_EQ("one two three ", textfield_->text()); | |
| 541 #else | |
| 542 EXPECT_STR_EQ("three ", textfield_->text()); | |
| 543 #endif | |
| 544 | |
| 545 // Delete the next word from cursor. | |
| 546 textfield_->SetText(ASCIIToUTF16("one two three four")); | |
| 547 SendKeyEvent(ui::VKEY_HOME); | |
| 548 SendKeyEvent(ui::VKEY_DELETE, false, false, true, false); | |
| 549 EXPECT_STR_EQ(" two three four", textfield_->text()); | |
| 550 | |
| 551 // Delete text following the cursor in chromeos, do nothing in windows. | |
| 552 SendKeyEvent(ui::VKEY_RIGHT, false, false, true, false); | |
| 553 SendKeyEvent(ui::VKEY_DELETE, false, true, true, false); | |
| 554 #if defined(OS_WIN) | |
| 555 EXPECT_STR_EQ(" two three four", textfield_->text()); | |
| 556 #else | |
| 557 EXPECT_STR_EQ(" two", textfield_->text()); | |
| 558 #endif | |
| 559 } | |
| 560 | |
| 561 TEST_F(NativeTextfieldViewsTest, PasswordTest) { | |
| 562 InitTextfield(Textfield::STYLE_OBSCURED); | |
| 563 EXPECT_EQ(ui::TEXT_INPUT_TYPE_PASSWORD, GetTextInputType()); | |
| 564 EXPECT_TRUE(textfield_->enabled()); | |
| 565 EXPECT_TRUE(textfield_->focusable()); | |
| 566 | |
| 567 last_contents_.clear(); | |
| 568 textfield_->SetText(ASCIIToUTF16("password")); | |
| 569 // Ensure text() and the callback returns the actual text instead of "*". | |
| 570 EXPECT_STR_EQ("password", textfield_->text()); | |
| 571 EXPECT_TRUE(last_contents_.empty()); | |
| 572 model_->SelectAll(false); | |
| 573 SetClipboardText("foo"); | |
| 574 | |
| 575 // Cut and copy should be disabled. | |
| 576 EXPECT_FALSE(textfield_view_->IsCommandIdEnabled(IDS_APP_CUT)); | |
| 577 textfield_view_->ExecuteCommand(IDS_APP_CUT, 0); | |
| 578 SendKeyEvent(ui::VKEY_X, false, true); | |
| 579 EXPECT_FALSE(textfield_view_->IsCommandIdEnabled(IDS_APP_COPY)); | |
| 580 textfield_view_->ExecuteCommand(IDS_APP_COPY, 0); | |
| 581 SendKeyEvent(ui::VKEY_C, false, true); | |
| 582 SendKeyEvent(ui::VKEY_INSERT, false, true); | |
| 583 EXPECT_STR_EQ("foo", base::string16(GetClipboardText())); | |
| 584 EXPECT_STR_EQ("password", textfield_->text()); | |
| 585 // [Shift]+[Delete] should just delete without copying text to the clipboard. | |
| 586 textfield_->SelectAll(false); | |
| 587 SendKeyEvent(ui::VKEY_DELETE, true, false); | |
| 588 | |
| 589 // Paste should work normally. | |
| 590 EXPECT_TRUE(textfield_view_->IsCommandIdEnabled(IDS_APP_PASTE)); | |
| 591 textfield_view_->ExecuteCommand(IDS_APP_PASTE, 0); | |
| 592 SendKeyEvent(ui::VKEY_V, false, true); | |
| 593 SendKeyEvent(ui::VKEY_INSERT, true, false); | |
| 594 EXPECT_STR_EQ("foo", base::string16(GetClipboardText())); | |
| 595 EXPECT_STR_EQ("foofoofoo", textfield_->text()); | |
| 596 } | |
| 597 | |
| 598 TEST_F(NativeTextfieldViewsTest, InputTypeSetsObscured) { | |
| 599 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 600 | |
| 601 // Defaults to TEXT | |
| 602 EXPECT_EQ(ui::TEXT_INPUT_TYPE_TEXT, GetTextInputType()); | |
| 603 | |
| 604 // Setting to TEXT_INPUT_TYPE_PASSWORD also sets obscured state of textfield. | |
| 605 textfield_->SetTextInputType(ui::TEXT_INPUT_TYPE_PASSWORD); | |
| 606 EXPECT_EQ(ui::TEXT_INPUT_TYPE_PASSWORD, GetTextInputType()); | |
| 607 EXPECT_TRUE(textfield_->IsObscured()); | |
| 608 } | |
| 609 | |
| 610 TEST_F(NativeTextfieldViewsTest, ObscuredSetsInputType) { | |
| 611 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 612 | |
| 613 // Defaults to TEXT | |
| 614 EXPECT_EQ(ui::TEXT_INPUT_TYPE_TEXT, GetTextInputType()); | |
| 615 | |
| 616 textfield_->SetObscured(true); | |
| 617 EXPECT_EQ(ui::TEXT_INPUT_TYPE_PASSWORD, GetTextInputType()); | |
| 618 | |
| 619 textfield_->SetObscured(false); | |
| 620 EXPECT_EQ(ui::TEXT_INPUT_TYPE_TEXT, GetTextInputType()); | |
| 621 } | |
| 622 | |
| 623 TEST_F(NativeTextfieldViewsTest, TextInputType) { | |
| 624 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 625 | |
| 626 // Defaults to TEXT | |
| 627 EXPECT_EQ(ui::TEXT_INPUT_TYPE_TEXT, GetTextInputType()); | |
| 628 | |
| 629 // And can be set. | |
| 630 textfield_->SetTextInputType(ui::TEXT_INPUT_TYPE_URL); | |
| 631 EXPECT_EQ(ui::TEXT_INPUT_TYPE_URL, GetTextInputType()); | |
| 632 | |
| 633 // Readonly textfields have type NONE | |
| 634 textfield_->SetReadOnly(true); | |
| 635 EXPECT_EQ(ui::TEXT_INPUT_TYPE_NONE, GetTextInputType()); | |
| 636 | |
| 637 textfield_->SetReadOnly(false); | |
| 638 EXPECT_EQ(ui::TEXT_INPUT_TYPE_URL, GetTextInputType()); | |
| 639 | |
| 640 // As do disabled textfields | |
| 641 textfield_->SetEnabled(false); | |
| 642 EXPECT_EQ(ui::TEXT_INPUT_TYPE_NONE, GetTextInputType()); | |
| 643 } | |
| 644 | |
| 645 TEST_F(NativeTextfieldViewsTest, OnKeyPressReturnValueTest) { | |
| 646 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 647 | |
| 648 // Character keys will be handled by input method. | |
| 649 SendKeyEvent(ui::VKEY_A); | |
| 650 EXPECT_TRUE(textfield_->key_received()); | |
| 651 EXPECT_FALSE(textfield_->key_handled()); | |
| 652 textfield_->clear(); | |
| 653 | |
| 654 // Home will be handled. | |
| 655 SendKeyEvent(ui::VKEY_HOME); | |
| 656 EXPECT_TRUE(textfield_->key_received()); | |
| 657 EXPECT_TRUE(textfield_->key_handled()); | |
| 658 textfield_->clear(); | |
| 659 | |
| 660 // F24, up/down key won't be handled. | |
| 661 SendKeyEvent(ui::VKEY_F24); | |
| 662 EXPECT_TRUE(textfield_->key_received()); | |
| 663 EXPECT_FALSE(textfield_->key_handled()); | |
| 664 textfield_->clear(); | |
| 665 | |
| 666 SendKeyEvent(ui::VKEY_UP); | |
| 667 EXPECT_TRUE(textfield_->key_received()); | |
| 668 EXPECT_FALSE(textfield_->key_handled()); | |
| 669 textfield_->clear(); | |
| 670 | |
| 671 SendKeyEvent(ui::VKEY_DOWN); | |
| 672 EXPECT_TRUE(textfield_->key_received()); | |
| 673 EXPECT_FALSE(textfield_->key_handled()); | |
| 674 textfield_->clear(); | |
| 675 | |
| 676 // Empty Textfield does not handle left/right. | |
| 677 textfield_->SetText(base::string16()); | |
| 678 SendKeyEvent(ui::VKEY_LEFT); | |
| 679 EXPECT_TRUE(textfield_->key_received()); | |
| 680 EXPECT_FALSE(textfield_->key_handled()); | |
| 681 textfield_->clear(); | |
| 682 | |
| 683 SendKeyEvent(ui::VKEY_RIGHT); | |
| 684 EXPECT_TRUE(textfield_->key_received()); | |
| 685 EXPECT_FALSE(textfield_->key_handled()); | |
| 686 textfield_->clear(); | |
| 687 | |
| 688 // Add a char. Right key should not be handled when cursor is at the end. | |
| 689 SendKeyEvent(ui::VKEY_B); | |
| 690 SendKeyEvent(ui::VKEY_RIGHT); | |
| 691 EXPECT_TRUE(textfield_->key_received()); | |
| 692 EXPECT_FALSE(textfield_->key_handled()); | |
| 693 textfield_->clear(); | |
| 694 | |
| 695 // First left key is handled to move cursor left to the beginning. | |
| 696 SendKeyEvent(ui::VKEY_LEFT); | |
| 697 EXPECT_TRUE(textfield_->key_received()); | |
| 698 EXPECT_TRUE(textfield_->key_handled()); | |
| 699 textfield_->clear(); | |
| 700 | |
| 701 // Now left key should not be handled. | |
| 702 SendKeyEvent(ui::VKEY_LEFT); | |
| 703 EXPECT_TRUE(textfield_->key_received()); | |
| 704 EXPECT_FALSE(textfield_->key_handled()); | |
| 705 textfield_->clear(); | |
| 706 } | |
| 707 | |
| 708 TEST_F(NativeTextfieldViewsTest, CursorMovement) { | |
| 709 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 710 | |
| 711 // Test with trailing whitespace. | |
| 712 textfield_->SetText(ASCIIToUTF16("one two hre ")); | |
| 713 | |
| 714 // Send the cursor at the end. | |
| 715 SendKeyEvent(ui::VKEY_END); | |
| 716 | |
| 717 // Ctrl+Left should move the cursor just before the last word. | |
| 718 SendKeyEvent(ui::VKEY_LEFT, false, true); | |
| 719 SendKeyEvent(ui::VKEY_T); | |
| 720 EXPECT_STR_EQ("one two thre ", textfield_->text()); | |
| 721 EXPECT_STR_EQ("one two thre ", last_contents_); | |
| 722 | |
| 723 // Ctrl+Right should move the cursor to the end of the last word. | |
| 724 SendKeyEvent(ui::VKEY_RIGHT, false, true); | |
| 725 SendKeyEvent(ui::VKEY_E); | |
| 726 EXPECT_STR_EQ("one two three ", textfield_->text()); | |
| 727 EXPECT_STR_EQ("one two three ", last_contents_); | |
| 728 | |
| 729 // Ctrl+Right again should move the cursor to the end. | |
| 730 SendKeyEvent(ui::VKEY_RIGHT, false, true); | |
| 731 SendKeyEvent(ui::VKEY_BACK); | |
| 732 EXPECT_STR_EQ("one two three", textfield_->text()); | |
| 733 EXPECT_STR_EQ("one two three", last_contents_); | |
| 734 | |
| 735 // Test with leading whitespace. | |
| 736 textfield_->SetText(ASCIIToUTF16(" ne two")); | |
| 737 | |
| 738 // Send the cursor at the beginning. | |
| 739 SendKeyEvent(ui::VKEY_HOME); | |
| 740 | |
| 741 // Ctrl+Right, then Ctrl+Left should move the cursor to the beginning of the | |
| 742 // first word. | |
| 743 SendKeyEvent(ui::VKEY_RIGHT, false, true); | |
| 744 SendKeyEvent(ui::VKEY_LEFT, false, true); | |
| 745 SendKeyEvent(ui::VKEY_O); | |
| 746 EXPECT_STR_EQ(" one two", textfield_->text()); | |
| 747 EXPECT_STR_EQ(" one two", last_contents_); | |
| 748 | |
| 749 // Ctrl+Left to move the cursor to the beginning of the first word. | |
| 750 SendKeyEvent(ui::VKEY_LEFT, false, true); | |
| 751 // Ctrl+Left again should move the cursor back to the very beginning. | |
| 752 SendKeyEvent(ui::VKEY_LEFT, false, true); | |
| 753 SendKeyEvent(ui::VKEY_DELETE); | |
| 754 EXPECT_STR_EQ("one two", textfield_->text()); | |
| 755 EXPECT_STR_EQ("one two", last_contents_); | |
| 756 } | |
| 757 | |
| 758 TEST_F(NativeTextfieldViewsTest, FocusTraversalTest) { | |
| 759 InitTextfields(Textfield::STYLE_DEFAULT, 3); | |
| 760 textfield_->RequestFocus(); | |
| 761 | |
| 762 EXPECT_EQ(1, GetFocusedView()->id()); | |
| 763 widget_->GetFocusManager()->AdvanceFocus(false); | |
| 764 EXPECT_EQ(2, GetFocusedView()->id()); | |
| 765 widget_->GetFocusManager()->AdvanceFocus(false); | |
| 766 EXPECT_EQ(3, GetFocusedView()->id()); | |
| 767 // Cycle back to the first textfield. | |
| 768 widget_->GetFocusManager()->AdvanceFocus(false); | |
| 769 EXPECT_EQ(1, GetFocusedView()->id()); | |
| 770 | |
| 771 widget_->GetFocusManager()->AdvanceFocus(true); | |
| 772 EXPECT_EQ(3, GetFocusedView()->id()); | |
| 773 widget_->GetFocusManager()->AdvanceFocus(true); | |
| 774 EXPECT_EQ(2, GetFocusedView()->id()); | |
| 775 widget_->GetFocusManager()->AdvanceFocus(true); | |
| 776 EXPECT_EQ(1, GetFocusedView()->id()); | |
| 777 // Cycle back to the last textfield. | |
| 778 widget_->GetFocusManager()->AdvanceFocus(true); | |
| 779 EXPECT_EQ(3, GetFocusedView()->id()); | |
| 780 | |
| 781 // Request focus should still work. | |
| 782 textfield_->RequestFocus(); | |
| 783 EXPECT_EQ(1, GetFocusedView()->id()); | |
| 784 | |
| 785 // Test if clicking on textfield view sets the focus to textfield_. | |
| 786 widget_->GetFocusManager()->AdvanceFocus(true); | |
| 787 EXPECT_EQ(3, GetFocusedView()->id()); | |
| 788 ui::MouseEvent click(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(), | |
| 789 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 790 textfield_view_->OnMousePressed(click); | |
| 791 EXPECT_EQ(1, GetFocusedView()->id()); | |
| 792 } | |
| 793 | |
| 794 TEST_F(NativeTextfieldViewsTest, ContextMenuDisplayTest) { | |
| 795 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 796 EXPECT_TRUE(textfield_->context_menu_controller()); | |
| 797 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 798 ui::Clipboard::GetForCurrentThread()->Clear(ui::CLIPBOARD_TYPE_COPY_PASTE); | |
| 799 textfield_view_->ClearEditHistory(); | |
| 800 EXPECT_TRUE(GetContextMenuModel()); | |
| 801 VerifyTextfieldContextMenuContents(false, false, GetContextMenuModel()); | |
| 802 | |
| 803 textfield_->SelectAll(false); | |
| 804 VerifyTextfieldContextMenuContents(true, false, GetContextMenuModel()); | |
| 805 | |
| 806 SendKeyEvent(ui::VKEY_T); | |
| 807 VerifyTextfieldContextMenuContents(false, true, GetContextMenuModel()); | |
| 808 | |
| 809 textfield_->SelectAll(false); | |
| 810 VerifyTextfieldContextMenuContents(true, true, GetContextMenuModel()); | |
| 811 | |
| 812 // Exercise the "paste enabled?" check in the verifier. | |
| 813 SetClipboardText("Test"); | |
| 814 VerifyTextfieldContextMenuContents(true, true, GetContextMenuModel()); | |
| 815 } | |
| 816 | |
| 817 TEST_F(NativeTextfieldViewsTest, DoubleAndTripleClickTest) { | |
| 818 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 819 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 820 ui::MouseEvent click(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(), | |
| 821 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 822 ui::MouseEvent release(ui::ET_MOUSE_RELEASED, gfx::Point(), gfx::Point(), | |
| 823 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 824 ui::MouseEvent double_click( | |
| 825 ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(), | |
| 826 ui::EF_LEFT_MOUSE_BUTTON | ui::EF_IS_DOUBLE_CLICK, | |
| 827 ui::EF_LEFT_MOUSE_BUTTON); | |
| 828 | |
| 829 // Test for double click. | |
| 830 textfield_view_->OnMousePressed(click); | |
| 831 textfield_view_->OnMouseReleased(release); | |
| 832 EXPECT_TRUE(textfield_->GetSelectedText().empty()); | |
| 833 textfield_view_->OnMousePressed(double_click); | |
| 834 textfield_view_->OnMouseReleased(release); | |
| 835 EXPECT_STR_EQ("hello", textfield_->GetSelectedText()); | |
| 836 | |
| 837 // Test for triple click. | |
| 838 textfield_view_->OnMousePressed(click); | |
| 839 textfield_view_->OnMouseReleased(release); | |
| 840 EXPECT_STR_EQ("hello world", textfield_->GetSelectedText()); | |
| 841 | |
| 842 // Another click should reset back to double click. | |
| 843 textfield_view_->OnMousePressed(click); | |
| 844 textfield_view_->OnMouseReleased(release); | |
| 845 EXPECT_STR_EQ("hello", textfield_->GetSelectedText()); | |
| 846 } | |
| 847 | |
| 848 TEST_F(NativeTextfieldViewsTest, DragToSelect) { | |
| 849 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 850 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 851 const int kStart = GetCursorPositionX(5); | |
| 852 const int kEnd = 500; | |
| 853 gfx::Point start_point(kStart, 0); | |
| 854 gfx::Point end_point(kEnd, 0); | |
| 855 ui::MouseEvent click_a(ui::ET_MOUSE_PRESSED, start_point, start_point, | |
| 856 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 857 ui::MouseEvent click_b(ui::ET_MOUSE_PRESSED, end_point, end_point, | |
| 858 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 859 ui::MouseEvent drag_left(ui::ET_MOUSE_DRAGGED, gfx::Point(), gfx::Point(), | |
| 860 ui::EF_LEFT_MOUSE_BUTTON, 0); | |
| 861 ui::MouseEvent drag_right(ui::ET_MOUSE_DRAGGED, end_point, end_point, | |
| 862 ui::EF_LEFT_MOUSE_BUTTON, 0); | |
| 863 ui::MouseEvent release(ui::ET_MOUSE_RELEASED, end_point, end_point, | |
| 864 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 865 textfield_view_->OnMousePressed(click_a); | |
| 866 EXPECT_TRUE(textfield_->GetSelectedText().empty()); | |
| 867 // Check that dragging left selects the beginning of the string. | |
| 868 textfield_view_->OnMouseDragged(drag_left); | |
| 869 base::string16 text_left = textfield_->GetSelectedText(); | |
| 870 EXPECT_STR_EQ("hello", text_left); | |
| 871 // Check that dragging right selects the rest of the string. | |
| 872 textfield_view_->OnMouseDragged(drag_right); | |
| 873 base::string16 text_right = textfield_->GetSelectedText(); | |
| 874 EXPECT_STR_EQ(" world", text_right); | |
| 875 // Check that releasing in the same location does not alter the selection. | |
| 876 textfield_view_->OnMouseReleased(release); | |
| 877 EXPECT_EQ(text_right, textfield_->GetSelectedText()); | |
| 878 // Check that dragging from beyond the text length works too. | |
| 879 textfield_view_->OnMousePressed(click_b); | |
| 880 textfield_view_->OnMouseDragged(drag_left); | |
| 881 textfield_view_->OnMouseReleased(release); | |
| 882 EXPECT_EQ(textfield_->text(), textfield_->GetSelectedText()); | |
| 883 } | |
| 884 | |
| 885 #if defined(OS_WIN) | |
| 886 TEST_F(NativeTextfieldViewsTest, DragAndDrop_AcceptDrop) { | |
| 887 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 888 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 889 | |
| 890 ui::OSExchangeData data; | |
| 891 base::string16 string(ASCIIToUTF16("string ")); | |
| 892 data.SetString(string); | |
| 893 int formats = 0; | |
| 894 std::set<OSExchangeData::CustomFormat> custom_formats; | |
| 895 | |
| 896 // Ensure that disabled textfields do not accept drops. | |
| 897 textfield_->SetEnabled(false); | |
| 898 EXPECT_FALSE(textfield_view_->GetDropFormats(&formats, &custom_formats)); | |
| 899 EXPECT_EQ(0, formats); | |
| 900 EXPECT_TRUE(custom_formats.empty()); | |
| 901 EXPECT_FALSE(textfield_view_->CanDrop(data)); | |
| 902 textfield_->SetEnabled(true); | |
| 903 | |
| 904 // Ensure that read-only textfields do not accept drops. | |
| 905 textfield_->SetReadOnly(true); | |
| 906 EXPECT_FALSE(textfield_view_->GetDropFormats(&formats, &custom_formats)); | |
| 907 EXPECT_EQ(0, formats); | |
| 908 EXPECT_TRUE(custom_formats.empty()); | |
| 909 EXPECT_FALSE(textfield_view_->CanDrop(data)); | |
| 910 textfield_->SetReadOnly(false); | |
| 911 | |
| 912 // Ensure that enabled and editable textfields do accept drops. | |
| 913 EXPECT_TRUE(textfield_view_->GetDropFormats(&formats, &custom_formats)); | |
| 914 EXPECT_EQ(ui::OSExchangeData::STRING, formats); | |
| 915 EXPECT_TRUE(custom_formats.empty()); | |
| 916 EXPECT_TRUE(textfield_view_->CanDrop(data)); | |
| 917 gfx::Point drop_point(GetCursorPositionX(6), 0); | |
| 918 ui::DropTargetEvent drop(data, drop_point, drop_point, | |
| 919 ui::DragDropTypes::DRAG_COPY | ui::DragDropTypes::DRAG_MOVE); | |
| 920 EXPECT_EQ(ui::DragDropTypes::DRAG_COPY | ui::DragDropTypes::DRAG_MOVE, | |
| 921 textfield_view_->OnDragUpdated(drop)); | |
| 922 EXPECT_EQ(ui::DragDropTypes::DRAG_COPY, textfield_view_->OnPerformDrop(drop)); | |
| 923 EXPECT_STR_EQ("hello string world", textfield_->text()); | |
| 924 | |
| 925 // Ensure that textfields do not accept non-OSExchangeData::STRING types. | |
| 926 ui::OSExchangeData bad_data; | |
| 927 bad_data.SetFilename(base::FilePath(FILE_PATH_LITERAL("x"))); | |
| 928 #if defined(OS_WIN) | |
| 929 ui::OSExchangeData::CustomFormat fmt = ui::Clipboard::GetBitmapFormatType(); | |
| 930 bad_data.SetPickledData(fmt, Pickle()); | |
| 931 bad_data.SetFileContents(base::FilePath(L"x"), "x"); | |
| 932 bad_data.SetHtml(base::string16(ASCIIToUTF16("x")), GURL("x.org")); | |
| 933 ui::OSExchangeData::DownloadFileInfo download(base::FilePath(), NULL); | |
| 934 bad_data.SetDownloadFileInfo(download); | |
| 935 #endif | |
| 936 EXPECT_FALSE(textfield_view_->CanDrop(bad_data)); | |
| 937 } | |
| 938 #endif | |
| 939 | |
| 940 TEST_F(NativeTextfieldViewsTest, DragAndDrop_InitiateDrag) { | |
| 941 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 942 textfield_->SetText(ASCIIToUTF16("hello string world")); | |
| 943 | |
| 944 // Ensure the textfield will provide selected text for drag data. | |
| 945 base::string16 string; | |
| 946 ui::OSExchangeData data; | |
| 947 const gfx::Range kStringRange(6, 12); | |
| 948 textfield_->SelectRange(kStringRange); | |
| 949 const gfx::Point kStringPoint(GetCursorPositionX(9), 0); | |
| 950 textfield_view_->WriteDragDataForView(NULL, kStringPoint, &data); | |
| 951 EXPECT_TRUE(data.GetString(&string)); | |
| 952 EXPECT_EQ(textfield_->GetSelectedText(), string); | |
| 953 | |
| 954 // Ensure that disabled textfields do not support drag operations. | |
| 955 textfield_->SetEnabled(false); | |
| 956 EXPECT_EQ(ui::DragDropTypes::DRAG_NONE, | |
| 957 textfield_view_->GetDragOperationsForView(NULL, kStringPoint)); | |
| 958 textfield_->SetEnabled(true); | |
| 959 // Ensure that textfields without selections do not support drag operations. | |
| 960 textfield_->ClearSelection(); | |
| 961 EXPECT_EQ(ui::DragDropTypes::DRAG_NONE, | |
| 962 textfield_view_->GetDragOperationsForView(NULL, kStringPoint)); | |
| 963 textfield_->SelectRange(kStringRange); | |
| 964 // Ensure that password textfields do not support drag operations. | |
| 965 textfield_->SetObscured(true); | |
| 966 EXPECT_EQ(ui::DragDropTypes::DRAG_NONE, | |
| 967 textfield_view_->GetDragOperationsForView(NULL, kStringPoint)); | |
| 968 textfield_->SetObscured(false); | |
| 969 // Ensure that textfields only initiate drag operations inside the selection. | |
| 970 ui::MouseEvent press_event(ui::ET_MOUSE_PRESSED, kStringPoint, kStringPoint, | |
| 971 ui::EF_LEFT_MOUSE_BUTTON, | |
| 972 ui::EF_LEFT_MOUSE_BUTTON); | |
| 973 textfield_view_->OnMousePressed(press_event); | |
| 974 EXPECT_EQ(ui::DragDropTypes::DRAG_NONE, | |
| 975 textfield_view_->GetDragOperationsForView(NULL, gfx::Point())); | |
| 976 EXPECT_FALSE(textfield_view_->CanStartDragForView(NULL, gfx::Point(), | |
| 977 gfx::Point())); | |
| 978 EXPECT_EQ(ui::DragDropTypes::DRAG_COPY, | |
| 979 textfield_view_->GetDragOperationsForView(NULL, kStringPoint)); | |
| 980 EXPECT_TRUE(textfield_view_->CanStartDragForView(NULL, kStringPoint, | |
| 981 gfx::Point())); | |
| 982 // Ensure that textfields support local moves. | |
| 983 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE | ui::DragDropTypes::DRAG_COPY, | |
| 984 textfield_view_->GetDragOperationsForView(textfield_view_, kStringPoint)); | |
| 985 } | |
| 986 | |
| 987 TEST_F(NativeTextfieldViewsTest, DragAndDrop_ToTheRight) { | |
| 988 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 989 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 990 | |
| 991 base::string16 string; | |
| 992 ui::OSExchangeData data; | |
| 993 int formats = 0; | |
| 994 int operations = 0; | |
| 995 std::set<OSExchangeData::CustomFormat> custom_formats; | |
| 996 | |
| 997 // Start dragging "ello". | |
| 998 textfield_->SelectRange(gfx::Range(1, 5)); | |
| 999 gfx::Point point(GetCursorPositionX(3), 0); | |
| 1000 ui::MouseEvent click_a(ui::ET_MOUSE_PRESSED, point, point, | |
| 1001 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 1002 textfield_view_->OnMousePressed(click_a); | |
| 1003 EXPECT_TRUE(textfield_view_->CanStartDragForView(textfield_view_, | |
| 1004 click_a.location(), gfx::Point())); | |
| 1005 operations = textfield_view_->GetDragOperationsForView(textfield_view_, | |
| 1006 click_a.location()); | |
| 1007 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE | ui::DragDropTypes::DRAG_COPY, | |
| 1008 operations); | |
| 1009 textfield_view_->WriteDragDataForView(NULL, click_a.location(), &data); | |
| 1010 EXPECT_TRUE(data.GetString(&string)); | |
| 1011 EXPECT_EQ(textfield_->GetSelectedText(), string); | |
| 1012 EXPECT_TRUE(textfield_view_->GetDropFormats(&formats, &custom_formats)); | |
| 1013 EXPECT_EQ(ui::OSExchangeData::STRING, formats); | |
| 1014 EXPECT_TRUE(custom_formats.empty()); | |
| 1015 | |
| 1016 // Drop "ello" after "w". | |
| 1017 const gfx::Point kDropPoint(GetCursorPositionX(7), 0); | |
| 1018 EXPECT_TRUE(textfield_view_->CanDrop(data)); | |
| 1019 ui::DropTargetEvent drop_a(data, kDropPoint, kDropPoint, operations); | |
| 1020 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE, | |
| 1021 textfield_view_->OnDragUpdated(drop_a)); | |
| 1022 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE, | |
| 1023 textfield_view_->OnPerformDrop(drop_a)); | |
| 1024 EXPECT_STR_EQ("h welloorld", textfield_->text()); | |
| 1025 textfield_view_->OnDragDone(); | |
| 1026 | |
| 1027 // Undo/Redo the drag&drop change. | |
| 1028 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1029 EXPECT_STR_EQ("hello world", textfield_->text()); | |
| 1030 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1031 EXPECT_STR_EQ("", textfield_->text()); | |
| 1032 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1033 EXPECT_STR_EQ("", textfield_->text()); | |
| 1034 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1035 EXPECT_STR_EQ("hello world", textfield_->text()); | |
| 1036 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1037 EXPECT_STR_EQ("h welloorld", textfield_->text()); | |
| 1038 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1039 EXPECT_STR_EQ("h welloorld", textfield_->text()); | |
| 1040 } | |
| 1041 | |
| 1042 TEST_F(NativeTextfieldViewsTest, DragAndDrop_ToTheLeft) { | |
| 1043 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1044 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 1045 | |
| 1046 base::string16 string; | |
| 1047 ui::OSExchangeData data; | |
| 1048 int formats = 0; | |
| 1049 int operations = 0; | |
| 1050 std::set<OSExchangeData::CustomFormat> custom_formats; | |
| 1051 | |
| 1052 // Start dragging " worl". | |
| 1053 textfield_->SelectRange(gfx::Range(5, 10)); | |
| 1054 gfx::Point point(GetCursorPositionX(7), 0); | |
| 1055 ui::MouseEvent click_a(ui::ET_MOUSE_PRESSED, point, point, | |
| 1056 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 1057 textfield_view_->OnMousePressed(click_a); | |
| 1058 EXPECT_TRUE(textfield_view_->CanStartDragForView(textfield_view_, | |
| 1059 click_a.location(), gfx::Point())); | |
| 1060 operations = textfield_view_->GetDragOperationsForView(textfield_view_, | |
| 1061 click_a.location()); | |
| 1062 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE | ui::DragDropTypes::DRAG_COPY, | |
| 1063 operations); | |
| 1064 textfield_view_->WriteDragDataForView(NULL, click_a.location(), &data); | |
| 1065 EXPECT_TRUE(data.GetString(&string)); | |
| 1066 EXPECT_EQ(textfield_->GetSelectedText(), string); | |
| 1067 EXPECT_TRUE(textfield_view_->GetDropFormats(&formats, &custom_formats)); | |
| 1068 EXPECT_EQ(ui::OSExchangeData::STRING, formats); | |
| 1069 EXPECT_TRUE(custom_formats.empty()); | |
| 1070 | |
| 1071 // Drop " worl" after "h". | |
| 1072 EXPECT_TRUE(textfield_view_->CanDrop(data)); | |
| 1073 gfx::Point drop_point(GetCursorPositionX(1), 0); | |
| 1074 ui::DropTargetEvent drop_a(data, drop_point, drop_point, operations); | |
| 1075 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE, | |
| 1076 textfield_view_->OnDragUpdated(drop_a)); | |
| 1077 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE, | |
| 1078 textfield_view_->OnPerformDrop(drop_a)); | |
| 1079 EXPECT_STR_EQ("h worlellod", textfield_->text()); | |
| 1080 textfield_view_->OnDragDone(); | |
| 1081 | |
| 1082 // Undo/Redo the drag&drop change. | |
| 1083 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1084 EXPECT_STR_EQ("hello world", textfield_->text()); | |
| 1085 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1086 EXPECT_STR_EQ("", textfield_->text()); | |
| 1087 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1088 EXPECT_STR_EQ("", textfield_->text()); | |
| 1089 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1090 EXPECT_STR_EQ("hello world", textfield_->text()); | |
| 1091 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1092 EXPECT_STR_EQ("h worlellod", textfield_->text()); | |
| 1093 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1094 EXPECT_STR_EQ("h worlellod", textfield_->text()); | |
| 1095 } | |
| 1096 | |
| 1097 TEST_F(NativeTextfieldViewsTest, DragAndDrop_Canceled) { | |
| 1098 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1099 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 1100 | |
| 1101 // Start dragging "worl". | |
| 1102 textfield_->SelectRange(gfx::Range(6, 10)); | |
| 1103 gfx::Point point(GetCursorPositionX(8), 0); | |
| 1104 ui::MouseEvent click(ui::ET_MOUSE_PRESSED, point, point, | |
| 1105 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 1106 textfield_view_->OnMousePressed(click); | |
| 1107 ui::OSExchangeData data; | |
| 1108 textfield_view_->WriteDragDataForView(NULL, click.location(), &data); | |
| 1109 EXPECT_TRUE(textfield_view_->CanDrop(data)); | |
| 1110 // Drag the text over somewhere valid, outside the current selection. | |
| 1111 gfx::Point drop_point(GetCursorPositionX(2), 0); | |
| 1112 ui::DropTargetEvent drop(data, drop_point, drop_point, | |
| 1113 ui::DragDropTypes::DRAG_MOVE); | |
| 1114 EXPECT_EQ(ui::DragDropTypes::DRAG_MOVE, textfield_view_->OnDragUpdated(drop)); | |
| 1115 // "Cancel" the drag, via move and release over the selection, and OnDragDone. | |
| 1116 gfx::Point drag_point(GetCursorPositionX(9), 0); | |
| 1117 ui::MouseEvent drag(ui::ET_MOUSE_DRAGGED, drag_point, drag_point, | |
| 1118 ui::EF_LEFT_MOUSE_BUTTON, 0); | |
| 1119 ui::MouseEvent release(ui::ET_MOUSE_RELEASED, drag_point, drag_point, | |
| 1120 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | |
| 1121 textfield_view_->OnMouseDragged(drag); | |
| 1122 textfield_view_->OnMouseReleased(release); | |
| 1123 textfield_view_->OnDragDone(); | |
| 1124 EXPECT_EQ(ASCIIToUTF16("hello world"), textfield_->text()); | |
| 1125 } | |
| 1126 | |
| 1127 TEST_F(NativeTextfieldViewsTest, ReadOnlyTest) { | |
| 1128 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1129 textfield_->SetText(ASCIIToUTF16("read only")); | |
| 1130 textfield_->SetReadOnly(true); | |
| 1131 EXPECT_TRUE(textfield_->enabled()); | |
| 1132 EXPECT_TRUE(textfield_->focusable()); | |
| 1133 | |
| 1134 SendKeyEvent(ui::VKEY_HOME); | |
| 1135 EXPECT_EQ(0U, textfield_->GetCursorPosition()); | |
| 1136 SendKeyEvent(ui::VKEY_END); | |
| 1137 EXPECT_EQ(9U, textfield_->GetCursorPosition()); | |
| 1138 | |
| 1139 SendKeyEvent(ui::VKEY_LEFT, false, false); | |
| 1140 EXPECT_EQ(8U, textfield_->GetCursorPosition()); | |
| 1141 SendKeyEvent(ui::VKEY_LEFT, false, true); | |
| 1142 EXPECT_EQ(5U, textfield_->GetCursorPosition()); | |
| 1143 SendKeyEvent(ui::VKEY_LEFT, true, true); | |
| 1144 EXPECT_EQ(0U, textfield_->GetCursorPosition()); | |
| 1145 EXPECT_STR_EQ("read ", textfield_->GetSelectedText()); | |
| 1146 textfield_->SelectAll(false); | |
| 1147 EXPECT_STR_EQ("read only", textfield_->GetSelectedText()); | |
| 1148 | |
| 1149 // Cut should be disabled. | |
| 1150 SetClipboardText("Test"); | |
| 1151 EXPECT_FALSE(textfield_view_->IsCommandIdEnabled(IDS_APP_CUT)); | |
| 1152 textfield_view_->ExecuteCommand(IDS_APP_CUT, 0); | |
| 1153 SendKeyEvent(ui::VKEY_X, false, true); | |
| 1154 SendKeyEvent(ui::VKEY_DELETE, true, false); | |
| 1155 EXPECT_STR_EQ("Test", base::string16(GetClipboardText())); | |
| 1156 EXPECT_STR_EQ("read only", textfield_->text()); | |
| 1157 | |
| 1158 // Paste should be disabled. | |
| 1159 EXPECT_FALSE(textfield_view_->IsCommandIdEnabled(IDS_APP_PASTE)); | |
| 1160 textfield_view_->ExecuteCommand(IDS_APP_PASTE, 0); | |
| 1161 SendKeyEvent(ui::VKEY_V, false, true); | |
| 1162 SendKeyEvent(ui::VKEY_INSERT, true, false); | |
| 1163 EXPECT_STR_EQ("read only", textfield_->text()); | |
| 1164 | |
| 1165 // Copy should work normally. | |
| 1166 SetClipboardText("Test"); | |
| 1167 EXPECT_TRUE(textfield_view_->IsCommandIdEnabled(IDS_APP_COPY)); | |
| 1168 textfield_view_->ExecuteCommand(IDS_APP_COPY, 0); | |
| 1169 EXPECT_STR_EQ("read only", base::string16(GetClipboardText())); | |
| 1170 SetClipboardText("Test"); | |
| 1171 SendKeyEvent(ui::VKEY_C, false, true); | |
| 1172 EXPECT_STR_EQ("read only", base::string16(GetClipboardText())); | |
| 1173 SetClipboardText("Test"); | |
| 1174 SendKeyEvent(ui::VKEY_INSERT, false, true); | |
| 1175 EXPECT_STR_EQ("read only", base::string16(GetClipboardText())); | |
| 1176 | |
| 1177 // SetText should work even in read only mode. | |
| 1178 textfield_->SetText(ASCIIToUTF16(" four five six ")); | |
| 1179 EXPECT_STR_EQ(" four five six ", textfield_->text()); | |
| 1180 | |
| 1181 textfield_->SelectAll(false); | |
| 1182 EXPECT_STR_EQ(" four five six ", textfield_->GetSelectedText()); | |
| 1183 | |
| 1184 // Text field is unmodifiable and selection shouldn't change. | |
| 1185 SendKeyEvent(ui::VKEY_DELETE); | |
| 1186 EXPECT_STR_EQ(" four five six ", textfield_->GetSelectedText()); | |
| 1187 SendKeyEvent(ui::VKEY_BACK); | |
| 1188 EXPECT_STR_EQ(" four five six ", textfield_->GetSelectedText()); | |
| 1189 SendKeyEvent(ui::VKEY_T); | |
| 1190 EXPECT_STR_EQ(" four five six ", textfield_->GetSelectedText()); | |
| 1191 } | |
| 1192 | |
| 1193 TEST_F(NativeTextfieldViewsTest, TextInputClientTest) { | |
| 1194 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1195 ui::TextInputClient* client = textfield_->GetTextInputClient(); | |
| 1196 EXPECT_TRUE(client); | |
| 1197 EXPECT_EQ(ui::TEXT_INPUT_TYPE_TEXT, client->GetTextInputType()); | |
| 1198 | |
| 1199 textfield_->SetText(ASCIIToUTF16("0123456789")); | |
| 1200 gfx::Range range; | |
| 1201 EXPECT_TRUE(client->GetTextRange(&range)); | |
| 1202 EXPECT_EQ(0U, range.start()); | |
| 1203 EXPECT_EQ(10U, range.end()); | |
| 1204 | |
| 1205 EXPECT_TRUE(client->SetSelectionRange(gfx::Range(1, 4))); | |
| 1206 EXPECT_TRUE(client->GetSelectionRange(&range)); | |
| 1207 EXPECT_EQ(gfx::Range(1, 4), range); | |
| 1208 | |
| 1209 // This code can't be compiled because of a bug in base::Callback. | |
| 1210 #if 0 | |
| 1211 GetTextHelper helper; | |
| 1212 base::Callback<void(base::string16)> callback = | |
| 1213 base::Bind(&GetTextHelper::set_text, base::Unretained(&helper)); | |
| 1214 | |
| 1215 EXPECT_TRUE(client->GetTextFromRange(range, callback)); | |
| 1216 EXPECT_STR_EQ("123", helper.text()); | |
| 1217 #endif | |
| 1218 | |
| 1219 EXPECT_TRUE(client->DeleteRange(range)); | |
| 1220 EXPECT_STR_EQ("0456789", textfield_->text()); | |
| 1221 | |
| 1222 ui::CompositionText composition; | |
| 1223 composition.text = UTF8ToUTF16("321"); | |
| 1224 // Set composition through input method. | |
| 1225 input_method_->Clear(); | |
| 1226 input_method_->SetCompositionTextForNextKey(composition); | |
| 1227 textfield_->clear(); | |
| 1228 | |
| 1229 on_before_user_action_ = on_after_user_action_ = 0; | |
| 1230 SendKeyEvent(ui::VKEY_A); | |
| 1231 EXPECT_TRUE(textfield_->key_received()); | |
| 1232 EXPECT_FALSE(textfield_->key_handled()); | |
| 1233 EXPECT_TRUE(client->HasCompositionText()); | |
| 1234 EXPECT_TRUE(client->GetCompositionTextRange(&range)); | |
| 1235 EXPECT_STR_EQ("0321456789", textfield_->text()); | |
| 1236 EXPECT_EQ(gfx::Range(1, 4), range); | |
| 1237 EXPECT_EQ(2, on_before_user_action_); | |
| 1238 EXPECT_EQ(2, on_after_user_action_); | |
| 1239 | |
| 1240 input_method_->SetResultTextForNextKey(UTF8ToUTF16("123")); | |
| 1241 on_before_user_action_ = on_after_user_action_ = 0; | |
| 1242 textfield_->clear(); | |
| 1243 SendKeyEvent(ui::VKEY_A); | |
| 1244 EXPECT_TRUE(textfield_->key_received()); | |
| 1245 EXPECT_FALSE(textfield_->key_handled()); | |
| 1246 EXPECT_FALSE(client->HasCompositionText()); | |
| 1247 EXPECT_FALSE(input_method_->cancel_composition_called()); | |
| 1248 EXPECT_STR_EQ("0123456789", textfield_->text()); | |
| 1249 EXPECT_EQ(2, on_before_user_action_); | |
| 1250 EXPECT_EQ(2, on_after_user_action_); | |
| 1251 | |
| 1252 input_method_->Clear(); | |
| 1253 input_method_->SetCompositionTextForNextKey(composition); | |
| 1254 textfield_->clear(); | |
| 1255 SendKeyEvent(ui::VKEY_A); | |
| 1256 EXPECT_TRUE(client->HasCompositionText()); | |
| 1257 EXPECT_STR_EQ("0123321456789", textfield_->text()); | |
| 1258 | |
| 1259 on_before_user_action_ = on_after_user_action_ = 0; | |
| 1260 textfield_->clear(); | |
| 1261 SendKeyEvent(ui::VKEY_RIGHT); | |
| 1262 EXPECT_FALSE(client->HasCompositionText()); | |
| 1263 EXPECT_TRUE(input_method_->cancel_composition_called()); | |
| 1264 EXPECT_TRUE(textfield_->key_received()); | |
| 1265 EXPECT_TRUE(textfield_->key_handled()); | |
| 1266 EXPECT_STR_EQ("0123321456789", textfield_->text()); | |
| 1267 EXPECT_EQ(8U, textfield_->GetCursorPosition()); | |
| 1268 EXPECT_EQ(1, on_before_user_action_); | |
| 1269 EXPECT_EQ(1, on_after_user_action_); | |
| 1270 | |
| 1271 textfield_->clear(); | |
| 1272 textfield_->SetText(ASCIIToUTF16("0123456789")); | |
| 1273 EXPECT_TRUE(client->SetSelectionRange(gfx::Range(5, 5))); | |
| 1274 client->ExtendSelectionAndDelete(4, 2); | |
| 1275 EXPECT_STR_EQ("0789", textfield_->text()); | |
| 1276 | |
| 1277 // On{Before,After}UserAction should be called by whatever user action | |
| 1278 // triggers clearing or setting a selection if appropriate. | |
| 1279 on_before_user_action_ = on_after_user_action_ = 0; | |
| 1280 textfield_->clear(); | |
| 1281 textfield_->ClearSelection(); | |
| 1282 textfield_->SelectAll(false); | |
| 1283 EXPECT_EQ(0, on_before_user_action_); | |
| 1284 EXPECT_EQ(0, on_after_user_action_); | |
| 1285 | |
| 1286 input_method_->Clear(); | |
| 1287 textfield_->SetReadOnly(true); | |
| 1288 EXPECT_TRUE(input_method_->text_input_type_changed()); | |
| 1289 EXPECT_FALSE(textfield_->GetTextInputClient()); | |
| 1290 | |
| 1291 textfield_->SetReadOnly(false); | |
| 1292 input_method_->Clear(); | |
| 1293 textfield_->SetObscured(true); | |
| 1294 EXPECT_TRUE(input_method_->text_input_type_changed()); | |
| 1295 EXPECT_TRUE(textfield_->GetTextInputClient()); | |
| 1296 } | |
| 1297 | |
| 1298 TEST_F(NativeTextfieldViewsTest, UndoRedoTest) { | |
| 1299 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1300 SendKeyEvent(ui::VKEY_A); | |
| 1301 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1302 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1303 EXPECT_STR_EQ("", textfield_->text()); | |
| 1304 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1305 EXPECT_STR_EQ("", textfield_->text()); | |
| 1306 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1307 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1308 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1309 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1310 | |
| 1311 // AppendText | |
| 1312 textfield_->AppendText(ASCIIToUTF16("b")); | |
| 1313 last_contents_.clear(); // AppendText doesn't call ContentsChanged. | |
| 1314 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1315 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1316 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1317 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1318 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1319 | |
| 1320 // SetText | |
| 1321 SendKeyEvent(ui::VKEY_C); | |
| 1322 // Undo'ing append moves the cursor to the end for now. | |
| 1323 // no-op SetText won't add new edit. See TextfieldViewsModel::SetText | |
| 1324 // description. | |
| 1325 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1326 textfield_->SetText(ASCIIToUTF16("abc")); | |
| 1327 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1328 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1329 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1330 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1331 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1332 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1333 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1334 textfield_->SetText(ASCIIToUTF16("123")); | |
| 1335 textfield_->SetText(ASCIIToUTF16("123")); | |
| 1336 EXPECT_STR_EQ("123", textfield_->text()); | |
| 1337 SendKeyEvent(ui::VKEY_END, false, false); | |
| 1338 SendKeyEvent(ui::VKEY_4, false, false); | |
| 1339 EXPECT_STR_EQ("1234", textfield_->text()); | |
| 1340 last_contents_.clear(); | |
| 1341 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1342 EXPECT_STR_EQ("123", textfield_->text()); | |
| 1343 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1344 // the insert edit "c" and set edit "123" are merged to single edit, | |
| 1345 // so text becomes "ab" after undo. | |
| 1346 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1347 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1348 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1349 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1350 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1351 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1352 EXPECT_STR_EQ("123", textfield_->text()); | |
| 1353 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1354 EXPECT_STR_EQ("1234", textfield_->text()); | |
| 1355 | |
| 1356 // Undoing to the same text shouldn't call ContentsChanged. | |
| 1357 SendKeyEvent(ui::VKEY_A, false, true); // select all | |
| 1358 SendKeyEvent(ui::VKEY_A); | |
| 1359 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1360 SendKeyEvent(ui::VKEY_B); | |
| 1361 SendKeyEvent(ui::VKEY_C); | |
| 1362 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1363 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1364 EXPECT_STR_EQ("1234", textfield_->text()); | |
| 1365 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1366 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1367 | |
| 1368 // Delete/Backspace | |
| 1369 SendKeyEvent(ui::VKEY_BACK); | |
| 1370 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1371 SendKeyEvent(ui::VKEY_HOME); | |
| 1372 SendKeyEvent(ui::VKEY_DELETE); | |
| 1373 EXPECT_STR_EQ("b", textfield_->text()); | |
| 1374 SendKeyEvent(ui::VKEY_A, false, true); | |
| 1375 SendKeyEvent(ui::VKEY_DELETE); | |
| 1376 EXPECT_STR_EQ("", textfield_->text()); | |
| 1377 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1378 EXPECT_STR_EQ("b", textfield_->text()); | |
| 1379 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1380 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1381 SendKeyEvent(ui::VKEY_Z, false, true); | |
| 1382 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1383 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1384 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1385 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1386 EXPECT_STR_EQ("b", textfield_->text()); | |
| 1387 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1388 EXPECT_STR_EQ("", textfield_->text()); | |
| 1389 SendKeyEvent(ui::VKEY_Y, false, true); | |
| 1390 EXPECT_STR_EQ("", textfield_->text()); | |
| 1391 } | |
| 1392 | |
| 1393 TEST_F(NativeTextfieldViewsTest, CutCopyPaste) { | |
| 1394 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1395 | |
| 1396 // Ensure IDS_APP_CUT cuts. | |
| 1397 textfield_->SetText(ASCIIToUTF16("123")); | |
| 1398 textfield_->SelectAll(false); | |
| 1399 EXPECT_TRUE(textfield_view_->IsCommandIdEnabled(IDS_APP_CUT)); | |
| 1400 textfield_view_->ExecuteCommand(IDS_APP_CUT, 0); | |
| 1401 EXPECT_STR_EQ("123", base::string16(GetClipboardText())); | |
| 1402 EXPECT_STR_EQ("", textfield_->text()); | |
| 1403 | |
| 1404 // Ensure [Ctrl]+[x] cuts and [Ctrl]+[Alt][x] does nothing. | |
| 1405 textfield_->SetText(ASCIIToUTF16("456")); | |
| 1406 textfield_->SelectAll(false); | |
| 1407 SendKeyEvent(ui::VKEY_X, true, false, true, false); | |
| 1408 EXPECT_STR_EQ("123", base::string16(GetClipboardText())); | |
| 1409 EXPECT_STR_EQ("456", textfield_->text()); | |
| 1410 SendKeyEvent(ui::VKEY_X, false, true); | |
| 1411 EXPECT_STR_EQ("456", base::string16(GetClipboardText())); | |
| 1412 EXPECT_STR_EQ("", textfield_->text()); | |
| 1413 | |
| 1414 // Ensure [Shift]+[Delete] cuts. | |
| 1415 textfield_->SetText(ASCIIToUTF16("123")); | |
| 1416 textfield_->SelectAll(false); | |
| 1417 SendKeyEvent(ui::VKEY_DELETE, true, false); | |
| 1418 EXPECT_STR_EQ("123", base::string16(GetClipboardText())); | |
| 1419 EXPECT_STR_EQ("", textfield_->text()); | |
| 1420 | |
| 1421 // Ensure IDS_APP_COPY copies. | |
| 1422 textfield_->SetText(ASCIIToUTF16("789")); | |
| 1423 textfield_->SelectAll(false); | |
| 1424 EXPECT_TRUE(textfield_view_->IsCommandIdEnabled(IDS_APP_COPY)); | |
| 1425 textfield_view_->ExecuteCommand(IDS_APP_COPY, 0); | |
| 1426 EXPECT_STR_EQ("789", base::string16(GetClipboardText())); | |
| 1427 | |
| 1428 // Ensure [Ctrl]+[c] copies and [Ctrl]+[Alt][c] does nothing. | |
| 1429 textfield_->SetText(ASCIIToUTF16("012")); | |
| 1430 textfield_->SelectAll(false); | |
| 1431 SendKeyEvent(ui::VKEY_C, true, false, true, false); | |
| 1432 EXPECT_STR_EQ("789", base::string16(GetClipboardText())); | |
| 1433 SendKeyEvent(ui::VKEY_C, false, true); | |
| 1434 EXPECT_STR_EQ("012", base::string16(GetClipboardText())); | |
| 1435 | |
| 1436 // Ensure [Ctrl]+[Insert] copies. | |
| 1437 textfield_->SetText(ASCIIToUTF16("345")); | |
| 1438 textfield_->SelectAll(false); | |
| 1439 SendKeyEvent(ui::VKEY_INSERT, false, true); | |
| 1440 EXPECT_STR_EQ("345", base::string16(GetClipboardText())); | |
| 1441 EXPECT_STR_EQ("345", textfield_->text()); | |
| 1442 | |
| 1443 // Ensure IDS_APP_PASTE, [Ctrl]+[V], and [Shift]+[Insert] pastes; | |
| 1444 // also ensure that [Ctrl]+[Alt]+[V] does nothing. | |
| 1445 SetClipboardText("abc"); | |
| 1446 textfield_->SetText(base::string16()); | |
| 1447 EXPECT_TRUE(textfield_view_->IsCommandIdEnabled(IDS_APP_PASTE)); | |
| 1448 textfield_view_->ExecuteCommand(IDS_APP_PASTE, 0); | |
| 1449 EXPECT_STR_EQ("abc", textfield_->text()); | |
| 1450 SendKeyEvent(ui::VKEY_V, false, true); | |
| 1451 EXPECT_STR_EQ("abcabc", textfield_->text()); | |
| 1452 SendKeyEvent(ui::VKEY_INSERT, true, false); | |
| 1453 EXPECT_STR_EQ("abcabcabc", textfield_->text()); | |
| 1454 SendKeyEvent(ui::VKEY_V, true, false, true, false); | |
| 1455 EXPECT_STR_EQ("abcabcabc", textfield_->text()); | |
| 1456 | |
| 1457 // Ensure [Ctrl]+[Shift]+[Insert] is a no-op. | |
| 1458 textfield_->SelectAll(false); | |
| 1459 SendKeyEvent(ui::VKEY_INSERT, true, true); | |
| 1460 EXPECT_STR_EQ("abc", base::string16(GetClipboardText())); | |
| 1461 EXPECT_STR_EQ("abcabcabc", textfield_->text()); | |
| 1462 } | |
| 1463 | |
| 1464 TEST_F(NativeTextfieldViewsTest, OvertypeMode) { | |
| 1465 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1466 // Overtype mode should be disabled (no-op [Insert]). | |
| 1467 textfield_->SetText(ASCIIToUTF16("2")); | |
| 1468 SendKeyEvent(ui::VKEY_HOME); | |
| 1469 SendKeyEvent(ui::VKEY_INSERT); | |
| 1470 SendKeyEvent(ui::VKEY_1, false, false); | |
| 1471 EXPECT_STR_EQ("12", textfield_->text()); | |
| 1472 } | |
| 1473 | |
| 1474 TEST_F(NativeTextfieldViewsTest, TextCursorDisplayTest) { | |
| 1475 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1476 // LTR-RTL string in LTR context. | |
| 1477 SendKeyEvent('a'); | |
| 1478 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1479 int x = GetCursorBounds().x(); | |
| 1480 int prev_x = x; | |
| 1481 | |
| 1482 SendKeyEvent('b'); | |
| 1483 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1484 x = GetCursorBounds().x(); | |
| 1485 EXPECT_LT(prev_x, x); | |
| 1486 prev_x = x; | |
| 1487 | |
| 1488 SendKeyEvent(0x05E1); | |
| 1489 EXPECT_EQ(WideToUTF16(L"ab\x05E1"), textfield_->text()); | |
| 1490 x = GetCursorBounds().x(); | |
| 1491 EXPECT_EQ(prev_x, x); | |
| 1492 | |
| 1493 SendKeyEvent(0x05E2); | |
| 1494 EXPECT_EQ(WideToUTF16(L"ab\x05E1\x5E2"), textfield_->text()); | |
| 1495 x = GetCursorBounds().x(); | |
| 1496 EXPECT_EQ(prev_x, x); | |
| 1497 | |
| 1498 // Clear text. | |
| 1499 SendKeyEvent(ui::VKEY_A, false, true); | |
| 1500 SendKeyEvent('\n'); | |
| 1501 | |
| 1502 // RTL-LTR string in LTR context. | |
| 1503 SendKeyEvent(0x05E1); | |
| 1504 EXPECT_EQ(WideToUTF16(L"\x05E1"), textfield_->text()); | |
| 1505 x = GetCursorBounds().x(); | |
| 1506 EXPECT_EQ(GetDisplayRect().x(), x); | |
| 1507 prev_x = x; | |
| 1508 | |
| 1509 SendKeyEvent(0x05E2); | |
| 1510 EXPECT_EQ(WideToUTF16(L"\x05E1\x05E2"), textfield_->text()); | |
| 1511 x = GetCursorBounds().x(); | |
| 1512 EXPECT_EQ(prev_x, x); | |
| 1513 | |
| 1514 SendKeyEvent('a'); | |
| 1515 EXPECT_EQ(WideToUTF16(L"\x05E1\x5E2" L"a"), textfield_->text()); | |
| 1516 x = GetCursorBounds().x(); | |
| 1517 EXPECT_LT(prev_x, x); | |
| 1518 prev_x = x; | |
| 1519 | |
| 1520 SendKeyEvent('b'); | |
| 1521 EXPECT_EQ(WideToUTF16(L"\x05E1\x5E2" L"ab"), textfield_->text()); | |
| 1522 x = GetCursorBounds().x(); | |
| 1523 EXPECT_LT(prev_x, x); | |
| 1524 } | |
| 1525 | |
| 1526 TEST_F(NativeTextfieldViewsTest, TextCursorDisplayInRTLTest) { | |
| 1527 std::string locale = l10n_util::GetApplicationLocale(""); | |
| 1528 base::i18n::SetICUDefaultLocale("he"); | |
| 1529 | |
| 1530 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1531 // LTR-RTL string in RTL context. | |
| 1532 SendKeyEvent('a'); | |
| 1533 EXPECT_STR_EQ("a", textfield_->text()); | |
| 1534 int x = GetCursorBounds().x(); | |
| 1535 EXPECT_EQ(GetDisplayRect().right() - 1, x); | |
| 1536 int prev_x = x; | |
| 1537 | |
| 1538 SendKeyEvent('b'); | |
| 1539 EXPECT_STR_EQ("ab", textfield_->text()); | |
| 1540 x = GetCursorBounds().x(); | |
| 1541 EXPECT_EQ(prev_x, x); | |
| 1542 | |
| 1543 SendKeyEvent(0x05E1); | |
| 1544 EXPECT_EQ(WideToUTF16(L"ab\x05E1"), textfield_->text()); | |
| 1545 x = GetCursorBounds().x(); | |
| 1546 EXPECT_GT(prev_x, x); | |
| 1547 prev_x = x; | |
| 1548 | |
| 1549 SendKeyEvent(0x05E2); | |
| 1550 EXPECT_EQ(WideToUTF16(L"ab\x05E1\x5E2"), textfield_->text()); | |
| 1551 x = GetCursorBounds().x(); | |
| 1552 EXPECT_GT(prev_x, x); | |
| 1553 | |
| 1554 SendKeyEvent(ui::VKEY_A, false, true); | |
| 1555 SendKeyEvent('\n'); | |
| 1556 | |
| 1557 // RTL-LTR string in RTL context. | |
| 1558 SendKeyEvent(0x05E1); | |
| 1559 EXPECT_EQ(WideToUTF16(L"\x05E1"), textfield_->text()); | |
| 1560 x = GetCursorBounds().x(); | |
| 1561 prev_x = x; | |
| 1562 | |
| 1563 SendKeyEvent(0x05E2); | |
| 1564 EXPECT_EQ(WideToUTF16(L"\x05E1\x05E2"), textfield_->text()); | |
| 1565 x = GetCursorBounds().x(); | |
| 1566 EXPECT_GT(prev_x, x); | |
| 1567 prev_x = x; | |
| 1568 | |
| 1569 SendKeyEvent('a'); | |
| 1570 EXPECT_EQ(WideToUTF16(L"\x05E1\x5E2" L"a"), textfield_->text()); | |
| 1571 x = GetCursorBounds().x(); | |
| 1572 EXPECT_EQ(prev_x, x); | |
| 1573 prev_x = x; | |
| 1574 | |
| 1575 SendKeyEvent('b'); | |
| 1576 EXPECT_EQ(WideToUTF16(L"\x05E1\x5E2" L"ab"), textfield_->text()); | |
| 1577 x = GetCursorBounds().x(); | |
| 1578 EXPECT_EQ(prev_x, x); | |
| 1579 | |
| 1580 // Reset locale. | |
| 1581 base::i18n::SetICUDefaultLocale(locale); | |
| 1582 } | |
| 1583 | |
| 1584 TEST_F(NativeTextfieldViewsTest, HitInsideTextAreaTest) { | |
| 1585 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1586 textfield_->SetText(WideToUTF16(L"ab\x05E1\x5E2")); | |
| 1587 std::vector<gfx::Rect> cursor_bounds; | |
| 1588 | |
| 1589 // Save each cursor bound. | |
| 1590 gfx::SelectionModel sel(0, gfx::CURSOR_FORWARD); | |
| 1591 cursor_bounds.push_back(GetCursorBounds(sel)); | |
| 1592 | |
| 1593 sel = gfx::SelectionModel(1, gfx::CURSOR_BACKWARD); | |
| 1594 gfx::Rect bound = GetCursorBounds(sel); | |
| 1595 sel = gfx::SelectionModel(1, gfx::CURSOR_FORWARD); | |
| 1596 EXPECT_EQ(bound.x(), GetCursorBounds(sel).x()); | |
| 1597 cursor_bounds.push_back(bound); | |
| 1598 | |
| 1599 // Check that a cursor at the end of the Latin portion of the text is at the | |
| 1600 // same position as a cursor placed at the end of the RTL Hebrew portion. | |
| 1601 sel = gfx::SelectionModel(2, gfx::CURSOR_BACKWARD); | |
| 1602 bound = GetCursorBounds(sel); | |
| 1603 sel = gfx::SelectionModel(4, gfx::CURSOR_BACKWARD); | |
| 1604 EXPECT_EQ(bound.x(), GetCursorBounds(sel).x()); | |
| 1605 cursor_bounds.push_back(bound); | |
| 1606 | |
| 1607 sel = gfx::SelectionModel(3, gfx::CURSOR_BACKWARD); | |
| 1608 bound = GetCursorBounds(sel); | |
| 1609 sel = gfx::SelectionModel(3, gfx::CURSOR_FORWARD); | |
| 1610 EXPECT_EQ(bound.x(), GetCursorBounds(sel).x()); | |
| 1611 cursor_bounds.push_back(bound); | |
| 1612 | |
| 1613 sel = gfx::SelectionModel(2, gfx::CURSOR_FORWARD); | |
| 1614 bound = GetCursorBounds(sel); | |
| 1615 sel = gfx::SelectionModel(4, gfx::CURSOR_FORWARD); | |
| 1616 EXPECT_EQ(bound.x(), GetCursorBounds(sel).x()); | |
| 1617 cursor_bounds.push_back(bound); | |
| 1618 | |
| 1619 // Expected cursor position when clicking left and right of each character. | |
| 1620 size_t cursor_pos_expected[] = {0, 1, 1, 2, 4, 3, 3, 2}; | |
| 1621 | |
| 1622 int index = 0; | |
| 1623 for (int i = 0; i < static_cast<int>(cursor_bounds.size() - 1); ++i) { | |
| 1624 int half_width = (cursor_bounds[i + 1].x() - cursor_bounds[i].x()) / 2; | |
| 1625 MouseClick(cursor_bounds[i], half_width / 2); | |
| 1626 EXPECT_EQ(cursor_pos_expected[index++], textfield_->GetCursorPosition()); | |
| 1627 | |
| 1628 // To avoid trigger double click. Not using sleep() since it takes longer | |
| 1629 // for the test to run if using sleep(). | |
| 1630 NonClientMouseClick(); | |
| 1631 | |
| 1632 MouseClick(cursor_bounds[i + 1], - (half_width / 2)); | |
| 1633 EXPECT_EQ(cursor_pos_expected[index++], textfield_->GetCursorPosition()); | |
| 1634 | |
| 1635 NonClientMouseClick(); | |
| 1636 } | |
| 1637 } | |
| 1638 | |
| 1639 TEST_F(NativeTextfieldViewsTest, HitOutsideTextAreaTest) { | |
| 1640 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1641 | |
| 1642 // LTR-RTL string in LTR context. | |
| 1643 textfield_->SetText(WideToUTF16(L"ab\x05E1\x5E2")); | |
| 1644 | |
| 1645 SendKeyEvent(ui::VKEY_HOME); | |
| 1646 gfx::Rect bound = GetCursorBounds(); | |
| 1647 MouseClick(bound, -10); | |
| 1648 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1649 | |
| 1650 SendKeyEvent(ui::VKEY_END); | |
| 1651 bound = GetCursorBounds(); | |
| 1652 MouseClick(bound, 10); | |
| 1653 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1654 | |
| 1655 NonClientMouseClick(); | |
| 1656 | |
| 1657 // RTL-LTR string in LTR context. | |
| 1658 textfield_->SetText(WideToUTF16(L"\x05E1\x5E2" L"ab")); | |
| 1659 | |
| 1660 SendKeyEvent(ui::VKEY_HOME); | |
| 1661 bound = GetCursorBounds(); | |
| 1662 MouseClick(bound, 10); | |
| 1663 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1664 | |
| 1665 SendKeyEvent(ui::VKEY_END); | |
| 1666 bound = GetCursorBounds(); | |
| 1667 MouseClick(bound, -10); | |
| 1668 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1669 } | |
| 1670 | |
| 1671 TEST_F(NativeTextfieldViewsTest, HitOutsideTextAreaInRTLTest) { | |
| 1672 std::string locale = l10n_util::GetApplicationLocale(""); | |
| 1673 base::i18n::SetICUDefaultLocale("he"); | |
| 1674 | |
| 1675 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1676 | |
| 1677 // RTL-LTR string in RTL context. | |
| 1678 textfield_->SetText(WideToUTF16(L"\x05E1\x5E2" L"ab")); | |
| 1679 SendKeyEvent(ui::VKEY_HOME); | |
| 1680 gfx::Rect bound = GetCursorBounds(); | |
| 1681 MouseClick(bound, 10); | |
| 1682 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1683 | |
| 1684 SendKeyEvent(ui::VKEY_END); | |
| 1685 bound = GetCursorBounds(); | |
| 1686 MouseClick(bound, -10); | |
| 1687 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1688 | |
| 1689 NonClientMouseClick(); | |
| 1690 | |
| 1691 // LTR-RTL string in RTL context. | |
| 1692 textfield_->SetText(WideToUTF16(L"ab\x05E1\x5E2")); | |
| 1693 SendKeyEvent(ui::VKEY_HOME); | |
| 1694 bound = GetCursorBounds(); | |
| 1695 MouseClick(bound, -10); | |
| 1696 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1697 | |
| 1698 SendKeyEvent(ui::VKEY_END); | |
| 1699 bound = GetCursorBounds(); | |
| 1700 MouseClick(bound, 10); | |
| 1701 EXPECT_EQ(bound, GetCursorBounds()); | |
| 1702 | |
| 1703 // Reset locale. | |
| 1704 base::i18n::SetICUDefaultLocale(locale); | |
| 1705 } | |
| 1706 | |
| 1707 TEST_F(NativeTextfieldViewsTest, OverflowTest) { | |
| 1708 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1709 | |
| 1710 base::string16 str; | |
| 1711 for (int i = 0; i < 500; ++i) | |
| 1712 SendKeyEvent('a'); | |
| 1713 SendKeyEvent(kHebrewLetterSamekh); | |
| 1714 EXPECT_TRUE(GetDisplayRect().Contains(GetCursorBounds())); | |
| 1715 | |
| 1716 // Test mouse pointing. | |
| 1717 MouseClick(GetCursorBounds(), -1); | |
| 1718 EXPECT_EQ(500U, textfield_->GetCursorPosition()); | |
| 1719 | |
| 1720 // Clear text. | |
| 1721 SendKeyEvent(ui::VKEY_A, false, true); | |
| 1722 SendKeyEvent('\n'); | |
| 1723 | |
| 1724 for (int i = 0; i < 500; ++i) | |
| 1725 SendKeyEvent(kHebrewLetterSamekh); | |
| 1726 SendKeyEvent('a'); | |
| 1727 EXPECT_TRUE(GetDisplayRect().Contains(GetCursorBounds())); | |
| 1728 | |
| 1729 MouseClick(GetCursorBounds(), -1); | |
| 1730 EXPECT_EQ(501U, textfield_->GetCursorPosition()); | |
| 1731 } | |
| 1732 | |
| 1733 TEST_F(NativeTextfieldViewsTest, OverflowInRTLTest) { | |
| 1734 std::string locale = l10n_util::GetApplicationLocale(""); | |
| 1735 base::i18n::SetICUDefaultLocale("he"); | |
| 1736 | |
| 1737 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1738 | |
| 1739 base::string16 str; | |
| 1740 for (int i = 0; i < 500; ++i) | |
| 1741 SendKeyEvent('a'); | |
| 1742 SendKeyEvent(kHebrewLetterSamekh); | |
| 1743 EXPECT_TRUE(GetDisplayRect().Contains(GetCursorBounds())); | |
| 1744 | |
| 1745 MouseClick(GetCursorBounds(), 1); | |
| 1746 EXPECT_EQ(501U, textfield_->GetCursorPosition()); | |
| 1747 | |
| 1748 // Clear text. | |
| 1749 SendKeyEvent(ui::VKEY_A, false, true); | |
| 1750 SendKeyEvent('\n'); | |
| 1751 | |
| 1752 for (int i = 0; i < 500; ++i) | |
| 1753 SendKeyEvent(kHebrewLetterSamekh); | |
| 1754 SendKeyEvent('a'); | |
| 1755 EXPECT_TRUE(GetDisplayRect().Contains(GetCursorBounds())); | |
| 1756 | |
| 1757 MouseClick(GetCursorBounds(), 1); | |
| 1758 EXPECT_EQ(500U, textfield_->GetCursorPosition()); | |
| 1759 | |
| 1760 // Reset locale. | |
| 1761 base::i18n::SetICUDefaultLocale(locale); | |
| 1762 } | |
| 1763 | |
| 1764 TEST_F(NativeTextfieldViewsTest, GetCompositionCharacterBoundsTest) { | |
| 1765 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1766 | |
| 1767 base::string16 str; | |
| 1768 const uint32 char_count = 10UL; | |
| 1769 ui::CompositionText composition; | |
| 1770 composition.text = UTF8ToUTF16("0123456789"); | |
| 1771 ui::TextInputClient* client = textfield_->GetTextInputClient(); | |
| 1772 | |
| 1773 // Return false if there is no composition text. | |
| 1774 gfx::Rect rect; | |
| 1775 EXPECT_FALSE(client->GetCompositionCharacterBounds(0, &rect)); | |
| 1776 | |
| 1777 // Get each character boundary by cursor. | |
| 1778 gfx::Rect char_rect_in_screen_coord[char_count]; | |
| 1779 gfx::Rect prev_cursor = GetCursorBounds(); | |
| 1780 for (uint32 i = 0; i < char_count; ++i) { | |
| 1781 composition.selection = gfx::Range(0, i+1); | |
| 1782 client->SetCompositionText(composition); | |
| 1783 EXPECT_TRUE(client->HasCompositionText()) << " i=" << i; | |
| 1784 gfx::Rect cursor_bounds = GetCursorBounds(); | |
| 1785 gfx::Point top_left(prev_cursor.x(), prev_cursor.y()); | |
| 1786 gfx::Point bottom_right(cursor_bounds.x(), prev_cursor.bottom()); | |
| 1787 views::View::ConvertPointToScreen(textfield_view_, &top_left); | |
| 1788 views::View::ConvertPointToScreen(textfield_view_, &bottom_right); | |
| 1789 char_rect_in_screen_coord[i].set_origin(top_left); | |
| 1790 char_rect_in_screen_coord[i].set_width(bottom_right.x() - top_left.x()); | |
| 1791 char_rect_in_screen_coord[i].set_height(bottom_right.y() - top_left.y()); | |
| 1792 prev_cursor = cursor_bounds; | |
| 1793 } | |
| 1794 | |
| 1795 for (uint32 i = 0; i < char_count; ++i) { | |
| 1796 gfx::Rect actual_rect; | |
| 1797 EXPECT_TRUE(client->GetCompositionCharacterBounds(i, &actual_rect)) | |
| 1798 << " i=" << i; | |
| 1799 EXPECT_EQ(char_rect_in_screen_coord[i], actual_rect) << " i=" << i; | |
| 1800 } | |
| 1801 | |
| 1802 // Return false if the index is out of range. | |
| 1803 EXPECT_FALSE(client->GetCompositionCharacterBounds(char_count, &rect)); | |
| 1804 EXPECT_FALSE(client->GetCompositionCharacterBounds(char_count + 1, &rect)); | |
| 1805 EXPECT_FALSE(client->GetCompositionCharacterBounds(char_count + 100, &rect)); | |
| 1806 } | |
| 1807 | |
| 1808 TEST_F(NativeTextfieldViewsTest, GetCompositionCharacterBounds_ComplexText) { | |
| 1809 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1810 | |
| 1811 const base::char16 kUtf16Chars[] = { | |
| 1812 // U+0020 SPACE | |
| 1813 0x0020, | |
| 1814 // U+1F408 (CAT) as surrogate pair | |
| 1815 0xd83d, 0xdc08, | |
| 1816 // U+5642 as Ideographic Variation Sequences | |
| 1817 0x5642, 0xDB40, 0xDD00, | |
| 1818 // U+260E (BLACK TELEPHONE) as Emoji Variation Sequences | |
| 1819 0x260E, 0xFE0F, | |
| 1820 // U+0020 SPACE | |
| 1821 0x0020, | |
| 1822 }; | |
| 1823 const size_t kUtf16CharsCount = arraysize(kUtf16Chars); | |
| 1824 | |
| 1825 ui::CompositionText composition; | |
| 1826 composition.text.assign(kUtf16Chars, kUtf16Chars + kUtf16CharsCount); | |
| 1827 ui::TextInputClient* client = textfield_->GetTextInputClient(); | |
| 1828 client->SetCompositionText(composition); | |
| 1829 | |
| 1830 // Make sure GetCompositionCharacterBounds never fails for index. | |
| 1831 gfx::Rect rects[kUtf16CharsCount]; | |
| 1832 gfx::Rect prev_cursor = GetCursorBounds(); | |
| 1833 for (uint32 i = 0; i < kUtf16CharsCount; ++i) | |
| 1834 EXPECT_TRUE(client->GetCompositionCharacterBounds(i, &rects[i])); | |
| 1835 | |
| 1836 // Here we might expect the following results but it actually depends on how | |
| 1837 // Uniscribe or HarfBuzz treats them with given font. | |
| 1838 // - rects[1] == rects[2] | |
| 1839 // - rects[3] == rects[4] == rects[5] | |
| 1840 // - rects[6] == rects[7] | |
| 1841 } | |
| 1842 | |
| 1843 // The word we select by double clicking should remain selected regardless of | |
| 1844 // where we drag the mouse afterwards without releasing the left button. | |
| 1845 TEST_F(NativeTextfieldViewsTest, KeepInitiallySelectedWord) { | |
| 1846 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1847 | |
| 1848 textfield_->SetText(ASCIIToUTF16("abc def ghi")); | |
| 1849 | |
| 1850 textfield_->SelectRange(gfx::Range(5, 5)); | |
| 1851 const gfx::Rect middle_cursor = GetCursorBounds(); | |
| 1852 textfield_->SelectRange(gfx::Range(0, 0)); | |
| 1853 const gfx::Point beginning = GetCursorBounds().origin(); | |
| 1854 | |
| 1855 // Double click, but do not release the left button. | |
| 1856 MouseClick(middle_cursor, 0); | |
| 1857 const gfx::Point middle(middle_cursor.x(), | |
| 1858 middle_cursor.y() + middle_cursor.height() / 2); | |
| 1859 ui::MouseEvent press_event(ui::ET_MOUSE_PRESSED, middle, middle, | |
| 1860 ui::EF_LEFT_MOUSE_BUTTON, | |
| 1861 ui::EF_LEFT_MOUSE_BUTTON); | |
| 1862 textfield_view_->OnMousePressed(press_event); | |
| 1863 EXPECT_EQ(gfx::Range(4, 7), textfield_->GetSelectedRange()); | |
| 1864 | |
| 1865 // Drag the mouse to the beginning of the textfield. | |
| 1866 ui::MouseEvent drag_event(ui::ET_MOUSE_DRAGGED, beginning, beginning, | |
| 1867 ui::EF_LEFT_MOUSE_BUTTON, 0); | |
| 1868 textfield_view_->OnMouseDragged(drag_event); | |
| 1869 EXPECT_EQ(gfx::Range(7, 0), textfield_->GetSelectedRange()); | |
| 1870 } | |
| 1871 | |
| 1872 // Touch selection and draggin currently only works for chromeos. | |
| 1873 #if defined(OS_CHROMEOS) | |
| 1874 TEST_F(NativeTextfieldViewsTest, TouchSelectionAndDraggingTest) { | |
| 1875 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1876 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 1877 EXPECT_FALSE(GetTouchSelectionController()); | |
| 1878 const int eventX = GetCursorPositionX(2); | |
| 1879 const int eventY = 0; | |
| 1880 CommandLine::ForCurrentProcess()->AppendSwitch(switches::kEnableTouchEditing); | |
| 1881 | |
| 1882 // Tapping on the textfield should turn on the TouchSelectionController. | |
| 1883 GestureEventForTest tap(ui::ET_GESTURE_TAP, eventX, eventY, 1.0f, 0.0f); | |
| 1884 textfield_view_->OnGestureEvent(&tap); | |
| 1885 EXPECT_TRUE(GetTouchSelectionController()); | |
| 1886 | |
| 1887 // Un-focusing the textfield should reset the TouchSelectionController | |
| 1888 textfield_view_->GetFocusManager()->ClearFocus(); | |
| 1889 EXPECT_FALSE(GetTouchSelectionController()); | |
| 1890 | |
| 1891 // With touch editing enabled, long press should not show context menu. | |
| 1892 // Instead, select word and invoke TouchSelectionController. | |
| 1893 GestureEventForTest tap_down(ui::ET_GESTURE_TAP_DOWN, eventX, eventY, 0.0f, | |
| 1894 0.0f); | |
| 1895 textfield_view_->OnGestureEvent(&tap_down); | |
| 1896 GestureEventForTest long_press(ui::ET_GESTURE_LONG_PRESS, eventX, eventY, | |
| 1897 0.0f, 0.0f); | |
| 1898 textfield_view_->OnGestureEvent(&long_press); | |
| 1899 EXPECT_STR_EQ("hello", textfield_->GetSelectedText()); | |
| 1900 EXPECT_TRUE(GetTouchSelectionController()); | |
| 1901 | |
| 1902 // Long pressing again in the selecting region should not do anything since | |
| 1903 // touch drag drop is not yet enabled. | |
| 1904 textfield_view_->OnGestureEvent(&tap_down); | |
| 1905 textfield_view_->OnGestureEvent(&long_press); | |
| 1906 EXPECT_STR_EQ("hello", textfield_->GetSelectedText()); | |
| 1907 EXPECT_TRUE(GetTouchSelectionController()); | |
| 1908 EXPECT_TRUE(long_press.handled()); | |
| 1909 | |
| 1910 // After enabling touch drag drop, long pressing in the selected region should | |
| 1911 // start a drag and remove TouchSelectionController. | |
| 1912 CommandLine::ForCurrentProcess()->AppendSwitch( | |
| 1913 switches::kEnableTouchDragDrop); | |
| 1914 textfield_view_->OnGestureEvent(&tap_down); | |
| 1915 | |
| 1916 // Create a new long press event since the previous one is not marked handled. | |
| 1917 GestureEventForTest long_press2(ui::ET_GESTURE_LONG_PRESS, eventX, eventY, | |
| 1918 0.0f, 0.0f); | |
| 1919 textfield_view_->OnGestureEvent(&long_press2); | |
| 1920 EXPECT_STR_EQ("hello", textfield_->GetSelectedText()); | |
| 1921 EXPECT_FALSE(GetTouchSelectionController()); | |
| 1922 } | |
| 1923 | |
| 1924 TEST_F(NativeTextfieldViewsTest, TouchScrubbingSelection) { | |
| 1925 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1926 textfield_->SetText(ASCIIToUTF16("hello world")); | |
| 1927 EXPECT_FALSE(GetTouchSelectionController()); | |
| 1928 | |
| 1929 CommandLine::ForCurrentProcess()->AppendSwitch(switches::kEnableTouchEditing); | |
| 1930 | |
| 1931 // Simulate touch-scrubbing. | |
| 1932 int scrubbing_start = GetCursorPositionX(1); | |
| 1933 int scrubbing_end = GetCursorPositionX(6); | |
| 1934 | |
| 1935 GestureEventForTest tap_down(ui::ET_GESTURE_TAP_DOWN, scrubbing_start, 0, | |
| 1936 0.0f, 0.0f); | |
| 1937 textfield_view_->OnGestureEvent(&tap_down); | |
| 1938 | |
| 1939 GestureEventForTest tap_cancel(ui::ET_GESTURE_TAP_CANCEL, scrubbing_start, 0, | |
| 1940 0.0f, 0.0f); | |
| 1941 textfield_view_->OnGestureEvent(&tap_cancel); | |
| 1942 | |
| 1943 GestureEventForTest scroll_begin(ui::ET_GESTURE_SCROLL_BEGIN, scrubbing_start, | |
| 1944 0, 0.0f, 0.0f); | |
| 1945 textfield_view_->OnGestureEvent(&scroll_begin); | |
| 1946 | |
| 1947 GestureEventForTest scroll_update(ui::ET_GESTURE_SCROLL_UPDATE, scrubbing_end, | |
| 1948 0, scrubbing_end - scrubbing_start, 0.0f); | |
| 1949 textfield_view_->OnGestureEvent(&scroll_update); | |
| 1950 | |
| 1951 GestureEventForTest scroll_end(ui::ET_GESTURE_SCROLL_END, scrubbing_end, 0, | |
| 1952 0.0f, 0.0f); | |
| 1953 textfield_view_->OnGestureEvent(&scroll_end); | |
| 1954 | |
| 1955 GestureEventForTest end(ui::ET_GESTURE_END, scrubbing_end, 0, 0.0f, 0.0f); | |
| 1956 textfield_view_->OnGestureEvent(&end); | |
| 1957 | |
| 1958 // In the end, part of text should have been selected and handles should have | |
| 1959 // appeared. | |
| 1960 EXPECT_STR_EQ("ello ", textfield_->GetSelectedText()); | |
| 1961 EXPECT_TRUE(GetTouchSelectionController()); | |
| 1962 } | |
| 1963 #endif | |
| 1964 | |
| 1965 // Long_Press gesture in NativeTextfieldViews can initiate a drag and drop now. | |
| 1966 TEST_F(NativeTextfieldViewsTest, TestLongPressInitiatesDragDrop) { | |
| 1967 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1968 textfield_->SetText(ASCIIToUTF16("Hello string world")); | |
| 1969 | |
| 1970 // Ensure the textfield will provide selected text for drag data. | |
| 1971 textfield_->SelectRange(gfx::Range(6, 12)); | |
| 1972 const gfx::Point kStringPoint(GetCursorPositionX(9), 0); | |
| 1973 | |
| 1974 // Enable touch-drag-drop to make long press effective. | |
| 1975 CommandLine::ForCurrentProcess()->AppendSwitch( | |
| 1976 switches::kEnableTouchDragDrop); | |
| 1977 | |
| 1978 // Create a long press event in the selected region should start a drag. | |
| 1979 GestureEventForTest long_press(ui::ET_GESTURE_LONG_PRESS, kStringPoint.x(), | |
| 1980 kStringPoint.y(), 0.0f, 0.0f); | |
| 1981 textfield_view_->OnGestureEvent(&long_press); | |
| 1982 EXPECT_TRUE(textfield_view_->CanStartDragForView(NULL, | |
| 1983 kStringPoint, kStringPoint)); | |
| 1984 } | |
| 1985 | |
| 1986 TEST_F(NativeTextfieldViewsTest, GetTextfieldBaseline_FontFallbackTest) { | |
| 1987 InitTextfield(Textfield::STYLE_DEFAULT); | |
| 1988 textfield_->SetText(UTF8ToUTF16("abc")); | |
| 1989 const int old_baseline = textfield_->GetBaseline(); | |
| 1990 | |
| 1991 // Set text which may fall back to a font which has taller baseline than | |
| 1992 // the default font. | |
| 1993 textfield_->SetText(UTF8ToUTF16("\xE0\xB9\x91")); | |
| 1994 const int new_baseline = textfield_->GetBaseline(); | |
| 1995 | |
| 1996 // Regardless of the text, the baseline must be the same. | |
| 1997 EXPECT_EQ(new_baseline, old_baseline); | |
| 1998 } | |
| 1999 | |
| 2000 } // namespace views | |
| OLD | NEW |