|  | // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/memory/scoped_ptr.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "ui/events/event.h" | 
|  | #include "ui/events/event_utils.h" | 
|  | #include "ui/events/keycodes/dom4/keycode_converter.h" | 
|  | #include "ui/events/test/events_test_utils.h" | 
|  |  | 
|  | #if defined(USE_X11) | 
|  | #include <X11/Xlib.h> | 
|  | #include "ui/events/test/events_test_utils_x11.h" | 
|  | #include "ui/gfx/x/x11_types.h" | 
|  | #endif | 
|  |  | 
|  | namespace ui { | 
|  |  | 
|  | TEST(EventTest, GetCharacter) { | 
|  | // Check if Control+Enter returns 10. | 
|  | KeyEvent keyev1(ET_KEY_PRESSED, VKEY_RETURN, EF_CONTROL_DOWN); | 
|  | EXPECT_EQ(10, keyev1.GetCharacter()); | 
|  | // Check if Enter returns 13. | 
|  | KeyEvent keyev2(ET_KEY_PRESSED, VKEY_RETURN, EF_NONE); | 
|  | EXPECT_EQ(13, keyev2.GetCharacter()); | 
|  | } | 
|  |  | 
|  | TEST(EventTest, ClickCount) { | 
|  | const gfx::Point origin(0, 0); | 
|  | MouseEvent mouseev(ET_MOUSE_PRESSED, origin, origin, 0, 0); | 
|  | for (int i = 1; i <=3 ; ++i) { | 
|  | mouseev.SetClickCount(i); | 
|  | EXPECT_EQ(i, mouseev.GetClickCount()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(EventTest, RepeatedClick) { | 
|  | const gfx::Point origin(0, 0); | 
|  | MouseEvent mouse_ev1(ET_MOUSE_PRESSED, origin, origin, 0, 0); | 
|  | MouseEvent mouse_ev2(ET_MOUSE_PRESSED, origin, origin, 0, 0); | 
|  | LocatedEventTestApi test_ev1(&mouse_ev1); | 
|  | LocatedEventTestApi test_ev2(&mouse_ev2); | 
|  |  | 
|  | base::TimeDelta start = base::TimeDelta::FromMilliseconds(0); | 
|  | base::TimeDelta soon = start + base::TimeDelta::FromMilliseconds(1); | 
|  | base::TimeDelta later = start + base::TimeDelta::FromMilliseconds(1000); | 
|  |  | 
|  | // Close point. | 
|  | test_ev1.set_location(gfx::Point(0, 0)); | 
|  | test_ev2.set_location(gfx::Point(1, 0)); | 
|  | test_ev1.set_time_stamp(start); | 
|  | test_ev2.set_time_stamp(soon); | 
|  | EXPECT_TRUE(MouseEvent::IsRepeatedClickEvent(mouse_ev1, mouse_ev2)); | 
|  |  | 
|  | // Too far. | 
|  | test_ev1.set_location(gfx::Point(0, 0)); | 
|  | test_ev2.set_location(gfx::Point(10, 0)); | 
|  | test_ev1.set_time_stamp(start); | 
|  | test_ev2.set_time_stamp(soon); | 
|  | EXPECT_FALSE(MouseEvent::IsRepeatedClickEvent(mouse_ev1, mouse_ev2)); | 
|  |  | 
|  | // Too long a time between clicks. | 
|  | test_ev1.set_location(gfx::Point(0, 0)); | 
|  | test_ev2.set_location(gfx::Point(0, 0)); | 
|  | test_ev1.set_time_stamp(start); | 
|  | test_ev2.set_time_stamp(later); | 
|  | EXPECT_FALSE(MouseEvent::IsRepeatedClickEvent(mouse_ev1, mouse_ev2)); | 
|  | } | 
|  |  | 
|  | TEST(EventTest, KeyEvent) { | 
|  | static const struct { | 
|  | KeyboardCode key_code; | 
|  | int flags; | 
|  | uint16 character; | 
|  | } kTestData[] = { | 
|  | { VKEY_A, 0, 'a' }, | 
|  | { VKEY_A, EF_SHIFT_DOWN, 'A' }, | 
|  | { VKEY_A, EF_CAPS_LOCK_DOWN, 'A' }, | 
|  | { VKEY_A, EF_SHIFT_DOWN | EF_CAPS_LOCK_DOWN, 'a' }, | 
|  | { VKEY_A, EF_CONTROL_DOWN, 0x01 }, | 
|  | { VKEY_A, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\x01' }, | 
|  | { VKEY_Z, 0, 'z' }, | 
|  | { VKEY_Z, EF_SHIFT_DOWN, 'Z' }, | 
|  | { VKEY_Z, EF_CAPS_LOCK_DOWN, 'Z' }, | 
|  | { VKEY_Z, EF_SHIFT_DOWN | EF_CAPS_LOCK_DOWN, 'z' }, | 
|  | { VKEY_Z, EF_CONTROL_DOWN, '\x1A' }, | 
|  | { VKEY_Z, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\x1A' }, | 
|  |  | 
|  | { VKEY_2, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_2, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_6, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_6, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\x1E' }, | 
|  | { VKEY_OEM_MINUS, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_MINUS, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\x1F' }, | 
|  | { VKEY_OEM_4, EF_CONTROL_DOWN, '\x1B' }, | 
|  | { VKEY_OEM_4, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_5, EF_CONTROL_DOWN, '\x1C' }, | 
|  | { VKEY_OEM_5, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_6, EF_CONTROL_DOWN, '\x1D' }, | 
|  | { VKEY_OEM_6, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_RETURN, EF_CONTROL_DOWN, '\x0A' }, | 
|  |  | 
|  | { VKEY_0, 0, '0' }, | 
|  | { VKEY_0, EF_SHIFT_DOWN, ')' }, | 
|  | { VKEY_0, EF_SHIFT_DOWN | EF_CAPS_LOCK_DOWN, ')' }, | 
|  | { VKEY_0, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  |  | 
|  | { VKEY_9, 0, '9' }, | 
|  | { VKEY_9, EF_SHIFT_DOWN, '(' }, | 
|  | { VKEY_9, EF_SHIFT_DOWN | EF_CAPS_LOCK_DOWN, '(' }, | 
|  | { VKEY_9, EF_SHIFT_DOWN | EF_CONTROL_DOWN, '\0' }, | 
|  |  | 
|  | { VKEY_NUMPAD0, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_NUMPAD0, EF_SHIFT_DOWN, '0' }, | 
|  |  | 
|  | { VKEY_NUMPAD9, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_NUMPAD9, EF_SHIFT_DOWN, '9' }, | 
|  |  | 
|  | { VKEY_TAB, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_TAB, EF_SHIFT_DOWN, '\t' }, | 
|  |  | 
|  | { VKEY_MULTIPLY, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_MULTIPLY, EF_SHIFT_DOWN, '*' }, | 
|  | { VKEY_ADD, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_ADD, EF_SHIFT_DOWN, '+' }, | 
|  | { VKEY_SUBTRACT, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_SUBTRACT, EF_SHIFT_DOWN, '-' }, | 
|  | { VKEY_DECIMAL, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_DECIMAL, EF_SHIFT_DOWN, '.' }, | 
|  | { VKEY_DIVIDE, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_DIVIDE, EF_SHIFT_DOWN, '/' }, | 
|  |  | 
|  | { VKEY_OEM_1, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_1, EF_SHIFT_DOWN, ':' }, | 
|  | { VKEY_OEM_PLUS, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_PLUS, EF_SHIFT_DOWN, '+' }, | 
|  | { VKEY_OEM_COMMA, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_COMMA, EF_SHIFT_DOWN, '<' }, | 
|  | { VKEY_OEM_PERIOD, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_PERIOD, EF_SHIFT_DOWN, '>' }, | 
|  | { VKEY_OEM_3, EF_CONTROL_DOWN, '\0' }, | 
|  | { VKEY_OEM_3, EF_SHIFT_DOWN, '~' }, | 
|  | }; | 
|  |  | 
|  | for (size_t i = 0; i < arraysize(kTestData); ++i) { | 
|  | KeyEvent key(ET_KEY_PRESSED, | 
|  | kTestData[i].key_code, | 
|  | kTestData[i].flags); | 
|  | EXPECT_EQ(kTestData[i].character, key.GetCharacter()) | 
|  | << " Index:" << i << " key_code:" << kTestData[i].key_code; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(EventTest, KeyEventDirectUnicode) { | 
|  | KeyEvent key(0x1234U, ui::VKEY_UNKNOWN, ui::EF_NONE); | 
|  | EXPECT_EQ(0x1234U, key.GetCharacter()); | 
|  | EXPECT_EQ(ET_KEY_PRESSED, key.type()); | 
|  | EXPECT_TRUE(key.is_char()); | 
|  | } | 
|  |  | 
|  | TEST(EventTest, NormalizeKeyEventFlags) { | 
|  | // Do not normalize flags for synthesized events without | 
|  | // KeyEvent::NormalizeFlags called explicitly. | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_PRESSED, VKEY_SHIFT, EF_SHIFT_DOWN); | 
|  | EXPECT_EQ(EF_SHIFT_DOWN, keyev.flags()); | 
|  | } | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_RELEASED, VKEY_SHIFT, EF_SHIFT_DOWN); | 
|  | EXPECT_EQ(EF_SHIFT_DOWN, keyev.flags()); | 
|  | keyev.NormalizeFlags(); | 
|  | EXPECT_EQ(EF_NONE, keyev.flags()); | 
|  | } | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_PRESSED, VKEY_CONTROL, EF_CONTROL_DOWN); | 
|  | EXPECT_EQ(EF_CONTROL_DOWN, keyev.flags()); | 
|  | } | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_RELEASED, VKEY_CONTROL, EF_CONTROL_DOWN); | 
|  | EXPECT_EQ(EF_CONTROL_DOWN, keyev.flags()); | 
|  | keyev.NormalizeFlags(); | 
|  | EXPECT_EQ(EF_NONE, keyev.flags()); | 
|  | } | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_PRESSED, VKEY_MENU,  EF_ALT_DOWN); | 
|  | EXPECT_EQ(EF_ALT_DOWN, keyev.flags()); | 
|  | } | 
|  | { | 
|  | KeyEvent keyev(ET_KEY_RELEASED, VKEY_MENU, EF_ALT_DOWN); | 
|  | EXPECT_EQ(EF_ALT_DOWN, keyev.flags()); | 
|  | keyev.NormalizeFlags(); | 
|  | EXPECT_EQ(EF_NONE, keyev.flags()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(EventTest, KeyEventCopy) { | 
|  | KeyEvent key(ET_KEY_PRESSED, VKEY_A, EF_NONE); | 
|  | scoped_ptr<KeyEvent> copied_key(new KeyEvent(key)); | 
|  | EXPECT_EQ(copied_key->type(), key.type()); | 
|  | EXPECT_EQ(copied_key->key_code(), key.key_code()); | 
|  | } | 
|  |  | 
|  | TEST(EventTest, KeyEventCode) { | 
|  | const char kCodeForSpace[] = "Space"; | 
|  | const uint16 kNativeCodeSpace = | 
|  | ui::KeycodeConverter::CodeToNativeKeycode(kCodeForSpace); | 
|  | ASSERT_NE(ui::KeycodeConverter::InvalidNativeKeycode(), kNativeCodeSpace); | 
|  |  | 
|  | { | 
|  | KeyEvent key(ET_KEY_PRESSED, VKEY_SPACE, kCodeForSpace, EF_NONE); | 
|  | EXPECT_EQ(kCodeForSpace, key.code()); | 
|  | } | 
|  | { | 
|  | // Regardless the KeyEvent.key_code (VKEY_RETURN), code should be | 
|  | // the specified value. | 
|  | KeyEvent key(ET_KEY_PRESSED, VKEY_RETURN, kCodeForSpace, EF_NONE); | 
|  | EXPECT_EQ(kCodeForSpace, key.code()); | 
|  | } | 
|  | { | 
|  | // If the synthetic event is initialized without code, it returns | 
|  | // an empty string. | 
|  | // TODO(komatsu): Fill a fallback value assuming the US keyboard layout. | 
|  | KeyEvent key(ET_KEY_PRESSED, VKEY_SPACE, EF_NONE); | 
|  | EXPECT_TRUE(key.code().empty()); | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace ui |