summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorpkasting@chromium.org <pkasting@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-03-05 01:59:56 +0000
committerpkasting@chromium.org <pkasting@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-03-05 01:59:56 +0000
commitdf813ac7b79f6c935df34dc82d8d6018d0a3d032 (patch)
tree53dca8daa1bb20a8a4b6b8e21ed4abd77c7b6a07
parent28ffe87d1aedc84f67c54b79ec1843cab237c9ff (diff)
downloadchromium_src-df813ac7b79f6c935df34dc82d8d6018d0a3d032.zip
chromium_src-df813ac7b79f6c935df34dc82d8d6018d0a3d032.tar.gz
chromium_src-df813ac7b79f6c935df34dc82d8d6018d0a3d032.tar.bz2
Various fixes to mouse wheel scrolling:
* Now that WebCore uses floating-point scroll deltas, eliminate complicated carryover code and just use simple floating-point arithmetic when calculating scroll delta. * Now that WebCore supports scrolling by page, plumb this instead of using a hacky "10 times the normal scroll amount" constant. * Don't pretend shift was down when it wasn't (e.g. WM_MOUSEHWHEEL). * Use SPI_GETWHEELSCROLLCHARS for horizontal scrolling, per MSDN. * Fix horizontal scrolling to be "scroll down to go right" as the comment said (behavior was backwards) * Clean up code. * Reorder Mac/Linux code to match Windows code ordering. Review URL: http://codereview.chromium.org/40135 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@10959 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--webkit/glue/event_conversion.cc7
-rw-r--r--webkit/glue/webinputevent.h5
-rw-r--r--webkit/glue/webinputevent_linux.cc13
-rw-r--r--webkit/glue/webinputevent_mac.mm35
-rw-r--r--webkit/glue/webinputevent_win.cc162
5 files changed, 97 insertions, 125 deletions
diff --git a/webkit/glue/event_conversion.cc b/webkit/glue/event_conversion.cc
index 39cdc90..548bfa5 100644
--- a/webkit/glue/event_conversion.cc
+++ b/webkit/glue/event_conversion.cc
@@ -123,10 +123,11 @@ MakePlatformWheelEvent::MakePlatformWheelEvent(Widget* widget,
const WebMouseWheelEvent& e) {
m_position = widget->convertFromContainingWindow(IntPoint(e.x, e.y));
m_globalPosition = IntPoint(e.global_x, e.global_y);
- m_deltaX = static_cast<float>(e.delta_x);
- m_deltaY = static_cast<float>(e.delta_y);
+ m_deltaX = e.delta_x;
+ m_deltaY = e.delta_y;
m_isAccepted = false;
- m_granularity = ScrollByLineWheelEvent;
+ m_granularity = e.scroll_by_page ?
+ ScrollByPageWheelEvent : ScrollByLineWheelEvent;
m_shiftKey = (e.modifiers & WebInputEvent::SHIFT_KEY) != 0;
m_ctrlKey = (e.modifiers & WebInputEvent::CTRL_KEY) != 0;
m_altKey = (e.modifiers & WebInputEvent::ALT_KEY) != 0;
diff --git a/webkit/glue/webinputevent.h b/webkit/glue/webinputevent.h
index dbfe456..ab71b83 100644
--- a/webkit/glue/webinputevent.h
+++ b/webkit/glue/webinputevent.h
@@ -131,8 +131,9 @@ class WebMouseEvent : public WebInputEvent {
class WebMouseWheelEvent : public WebMouseEvent {
public:
- int delta_x;
- int delta_y;
+ float delta_x;
+ float delta_y;
+ bool scroll_by_page;
WebMouseWheelEvent() {}
#if defined(OS_WIN)
diff --git a/webkit/glue/webinputevent_linux.cc b/webkit/glue/webinputevent_linux.cc
index 6ef85aa..12fb179 100644
--- a/webkit/glue/webinputevent_linux.cc
+++ b/webkit/glue/webinputevent_linux.cc
@@ -103,6 +103,9 @@ WebMouseEvent::WebMouseEvent(const GdkEventMotion* event) {
}
WebMouseWheelEvent::WebMouseWheelEvent(const GdkEventScroll* event) {
+ type = MOUSE_WHEEL;
+ button = BUTTON_NONE;
+
timestamp_sec = GdkEventTimeToWebEventTime(event->time);
modifiers = GdkStateToWebEventModifiers(event->state);
x = static_cast<int>(event->x);
@@ -110,8 +113,6 @@ WebMouseWheelEvent::WebMouseWheelEvent(const GdkEventScroll* event) {
global_x = static_cast<int>(event->x_root);
global_y = static_cast<int>(event->y_root);
- type = MOUSE_WHEEL;
-
// How much should we scroll per mouse wheel event?
// - Windows uses 3 lines by default and obeys a system setting.
// - Mozilla has a pref that lets you either use the "system" number of lines
@@ -122,11 +123,10 @@ WebMouseWheelEvent::WebMouseWheelEvent(const GdkEventScroll* event) {
// - Gtk makes the scroll amount a function of the size of the scroll bar,
// which is not available to us here.
// Instead, we pick a number that empirically matches Firefox's behavior.
- static const int kWheelDelta = 4;
+ static const float kWheelDelta = 4;
delta_x = 0;
delta_y = 0;
-
switch (event->direction) {
case GDK_SCROLL_UP:
delta_y = kWheelDelta;
@@ -135,14 +135,15 @@ WebMouseWheelEvent::WebMouseWheelEvent(const GdkEventScroll* event) {
delta_y = -kWheelDelta;
break;
case GDK_SCROLL_LEFT:
- delta_x = -kWheelDelta;
+ delta_x = kWheelDelta;
break;
case GDK_SCROLL_RIGHT:
- delta_x = kWheelDelta;
+ delta_x = -kWheelDelta;
break;
default:
break;
}
+ scroll_by_page = false;
}
WebKeyboardEvent::WebKeyboardEvent(const GdkEventKey* event) {
diff --git a/webkit/glue/webinputevent_mac.mm b/webkit/glue/webinputevent_mac.mm
index d553828..cdef792 100644
--- a/webkit/glue/webinputevent_mac.mm
+++ b/webkit/glue/webinputevent_mac.mm
@@ -119,37 +119,38 @@ WebMouseWheelEvent::WebMouseWheelEvent(NSEvent *event, NSView* view) {
type = MOUSE_WHEEL;
button = BUTTON_NONE;
+ // Set modifiers based on key state.
+ if ([event modifierFlags] & NSControlKeyMask)
+ modifiers |= CTRL_KEY;
+ if ([event modifierFlags] & NSShiftKeyMask)
+ modifiers |= SHIFT_KEY;
+ if ([event modifierFlags] & NSAlternateKeyMask)
+ modifiers |= ALT_KEY;
+
+ // Set coordinates by translating event coordinates from screen to client.
NSPoint location = [NSEvent mouseLocation]; // global coordinates
global_x = location.x;
global_y = location.y;
-
NSPoint windowLocal = [event locationInWindow];
location = [view convertPoint:windowLocal fromView:nil];
- y = [view frame].size.height - location.y; // flip y
x = location.x;
+ y = [view frame].size.height - location.y; // flip y
- int wheel_delta = [event deltaY];
- const int delta_lines = wheel_delta * kDefaultScrollLinesPerWheelDelta;
+ // Convert wheel delta amount to a number of lines to scroll.
+ float wheel_delta = [event deltaY];
+ const float delta_lines = wheel_delta * kDefaultScrollLinesPerWheelDelta;
- // Scroll horizontally if shift is held. WebKit's WebKit/win/WebView.cpp
- // does the equivalent.
- // TODO(jackson): Support WM_MOUSEHWHEEL = 0x020E event as well.
- // (Need a mouse with horizontal scrolling capabilities to test it.)
+ // Set scroll amount based on above calculations.
if ([event modifierFlags] & NSShiftKeyMask) {
- // Scrolling up should move left, scrolling down should move right
- delta_x = -delta_lines;
+ // Scrolling up should move left, scrolling down should move right. This is
+ // opposite Safari, but seems more consistent with vertical scrolling.
+ delta_x = delta_lines;
delta_y = 0;
} else {
delta_x = 0;
delta_y = delta_lines;
}
-
- if ([event modifierFlags] & NSControlKeyMask)
- modifiers |= CTRL_KEY;
- if ([event modifierFlags] & NSShiftKeyMask)
- modifiers |= SHIFT_KEY;
- if ([event modifierFlags] & NSAlternateKeyMask)
- modifiers |= ALT_KEY;
+ scroll_by_page = false;
}
// WebKeyboardEvent -----------------------------------------------------------
diff --git a/webkit/glue/webinputevent_win.cc b/webkit/glue/webinputevent_win.cc
index 3588154..9202672 100644
--- a/webkit/glue/webinputevent_win.cc
+++ b/webkit/glue/webinputevent_win.cc
@@ -11,6 +11,7 @@
#include "webkit/glue/webinputevent_util.h"
static const unsigned long kDefaultScrollLinesPerWheelDelta = 3;
+static const unsigned long kDefaultScrollCharsPerWheelDelta = 1;
// WebMouseEvent --------------------------------------------------------------
@@ -112,45 +113,36 @@ WebMouseEvent::WebMouseEvent(HWND hwnd, UINT message, WPARAM wparam,
// WebMouseWheelEvent ---------------------------------------------------------
-WebMouseWheelEvent::WebMouseWheelEvent(HWND hwnd, UINT message, WPARAM wparam,
- LPARAM lparam) {
+WebMouseWheelEvent::WebMouseWheelEvent(HWND hwnd,
+ UINT message,
+ WPARAM wparam,
+ LPARAM lparam)
+ : scroll_by_page(false) {
type = MOUSE_WHEEL;
button = BUTTON_NONE;
- // Add a simple workaround to scroll multiples units per page.
- // The right fix needs to extend webkit's implementation of
- // wheel events and that's not something we want to do at
- // this time. See bug# 928509
- // TODO(joshia): Implement the right fix for bug# 928509
- const int kPageScroll = 10; // 10 times wheel scroll
-
- UINT key_state = GET_KEYSTATE_WPARAM(wparam);
- int wheel_delta = static_cast<int>(GET_WHEEL_DELTA_WPARAM(wparam));
-
+ // Get key state, coordinates, and wheel delta from event.
typedef SHORT (WINAPI *GetKeyStateFunction)(int key);
- GetKeyStateFunction get_key_state = GetKeyState;
-
- // Synthesize mousewheel event from a scroll event.
- // This is needed to simulate middle mouse scrolling in some
- // laptops (Thinkpads)
- if ((WM_VSCROLL == message) || (WM_HSCROLL == message)) {
+ GetKeyStateFunction get_key_state;
+ UINT key_state;
+ float wheel_delta;
+ bool horizontal_scroll = false;
+ if ((message == WM_VSCROLL) || (message == WM_HSCROLL)) {
+ // Synthesize mousewheel event from a scroll event. This is needed to
+ // simulate middle mouse scrolling in some laptops. Use GetAsyncKeyState
+ // for key state since we are synthesizing the input event.
+ get_key_state = GetAsyncKeyState;
+ key_state = 0;
+ if (get_key_state(VK_SHIFT))
+ key_state |= MK_SHIFT;
+ if (get_key_state(VK_CONTROL))
+ key_state |= MK_CONTROL;
POINT cursor_position = {0};
GetCursorPos(&cursor_position);
-
global_x = cursor_position.x;
global_y = cursor_position.y;
- key_state = 0;
-
- // Since we are synthesizing the wheel event, we have to
- // use GetAsyncKeyState
- if (GetAsyncKeyState(VK_SHIFT))
- key_state |= MK_SHIFT;
-
- if (GetAsyncKeyState(VK_CONTROL))
- key_state |= MK_CONTROL;
-
switch (LOWORD(wparam)) {
case SB_LINEUP: // == SB_LINELEFT
wheel_delta = WHEEL_DELTA;
@@ -159,99 +151,75 @@ WebMouseWheelEvent::WebMouseWheelEvent(HWND hwnd, UINT message, WPARAM wparam,
wheel_delta = -WHEEL_DELTA;
break;
case SB_PAGEUP:
- wheel_delta = kPageScroll * WHEEL_DELTA;
+ wheel_delta = 1;
+ scroll_by_page = true;
break;
case SB_PAGEDOWN:
- wheel_delta = -kPageScroll * WHEEL_DELTA;
+ wheel_delta = -1;
+ scroll_by_page = true;
break;
- // TODO(joshia): Handle SB_THUMBPOSITION and SB_THUMBTRACK
- // for compeleteness
- default:
+ default: // We don't supoprt SB_THUMBPOSITION or SB_THUMBTRACK here.
+ wheel_delta = 0;
break;
}
- // Touchpads (or trackpoints) send the following messages in scrolling
- // horizontally.
- // * Scrolling left
- // message == WM_HSCROLL, wparam == SB_LINELEFT (== SB_LINEUP).
- // * Scrolling right
- // message == WM_HSCROLL, wparam == SB_LINERIGHT (== SB_LINEDOWN).
- if (WM_HSCROLL == message) {
- key_state |= MK_SHIFT;
+ if (message == WM_HSCROLL) {
+ horizontal_scroll = true;
wheel_delta = -wheel_delta;
}
-
- // Use GetAsyncKeyState for key state since we are synthesizing
- // the input
- get_key_state = GetAsyncKeyState;
} else {
- // TODO(hbono): we should add a new variable which indicates scroll
- // direction and remove this key_state hack.
- if (WM_MOUSEHWHEEL == message)
- key_state |= MK_SHIFT;
+ // Non-synthesized event; we can just read data off the event.
+ get_key_state = GetKeyState;
+ key_state = GET_KEYSTATE_WPARAM(wparam);
global_x = static_cast<short>(LOWORD(lparam));
global_y = static_cast<short>(HIWORD(lparam));
+
+ wheel_delta = static_cast<float>(GET_WHEEL_DELTA_WPARAM(wparam));
+ if (((message == WM_MOUSEHWHEEL) || (key_state & MK_SHIFT)) &&
+ (wheel_delta != 0))
+ horizontal_scroll = true;
}
+ // Set modifiers based on key state.
+ if (key_state & MK_SHIFT)
+ modifiers |= SHIFT_KEY;
+ if (key_state & MK_CONTROL)
+ modifiers |= CTRL_KEY;
+ if (get_key_state(VK_MENU) & 0x8000)
+ modifiers |= (ALT_KEY | META_KEY);
+
+ // Set coordinates by translating event coordinates from screen to client.
POINT client_point = { global_x, global_y };
- ScreenToClient(hwnd, &client_point);
+ MapWindowPoints(NULL, hwnd, &client_point, 1);
x = client_point.x;
y = client_point.y;
- // compute the scroll delta based on Raymond Chen's algorithm:
- // http://blogs.msdn.com/oldnewthing/archive/2003/08/07/54615.aspx
-
- static int carryover = 0;
- static HWND last_window = NULL;
-
- if (hwnd != last_window) {
- last_window = hwnd;
- carryover = 0;
- }
-
- unsigned long scroll_lines = kDefaultScrollLinesPerWheelDelta;
- SystemParametersInfo(SPI_GETWHEELSCROLLLINES, 0, &scroll_lines, 0);
-
- int delta_lines = 0;
- if (scroll_lines == WHEEL_PAGESCROLL) {
- scroll_lines = kPageScroll;
- }
-
- if (scroll_lines == 0) {
- carryover = 0;
+ // Convert wheel delta amount to a number of lines/chars to scroll.
+ float scroll_delta = wheel_delta / WHEEL_DELTA;
+ if (horizontal_scroll) {
+ unsigned long scroll_chars = kDefaultScrollCharsPerWheelDelta;
+ SystemParametersInfo(SPI_GETWHEELSCROLLCHARS, 0, &scroll_chars, 0);
+ scroll_delta *= static_cast<float>(scroll_chars);
} else {
- const int delta = carryover + wheel_delta;
-
- // see how many lines we should scroll. relies on round-toward-zero.
- delta_lines = delta * static_cast<int>(scroll_lines) / WHEEL_DELTA;
-
- // record the unused portion as the next carryover.
- carryover =
- delta - delta_lines * WHEEL_DELTA / static_cast<int>(scroll_lines);
+ unsigned long scroll_lines = kDefaultScrollLinesPerWheelDelta;
+ SystemParametersInfo(SPI_GETWHEELSCROLLLINES, 0, &scroll_lines, 0);
+ if (scroll_lines == WHEEL_PAGESCROLL)
+ scroll_by_page = true;
+ if (!scroll_by_page)
+ scroll_delta *= static_cast<float>(scroll_lines);
}
- // Scroll horizontally if shift is held. WebKit's WebKit/win/WebView.cpp
- // does the equivalent.
- // TODO(jackson): Support WM_MOUSEHWHEEL = 0x020E event as well.
- // (Need a mouse with horizontal scrolling capabilities to test it.)
- if (key_state & MK_SHIFT) {
- // Scrolling up should move left, scrolling down should move right
- delta_x = -delta_lines;
+ // Set scroll amount based on above calculations.
+ if (horizontal_scroll) {
+ // Scrolling up should move left, scrolling down should move right. This is
+ // opposite Safari, but seems more consistent with vertical scrolling.
+ delta_x = scroll_delta;
delta_y = 0;
} else {
delta_x = 0;
- delta_y = delta_lines;
+ delta_y = scroll_delta;
}
-
- if (key_state & MK_SHIFT)
- modifiers |= SHIFT_KEY;
- if (key_state & MK_CONTROL)
- modifiers |= CTRL_KEY;
-
- // Get any additional key states needed
- if (get_key_state(VK_MENU) & 0x8000)
- modifiers |= (ALT_KEY | META_KEY);
}
// WebKeyboardEvent -----------------------------------------------------------