diff options
author | pkasting@chromium.org <pkasting@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-03-05 01:59:56 +0000 |
---|---|---|
committer | pkasting@chromium.org <pkasting@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-03-05 01:59:56 +0000 |
commit | df813ac7b79f6c935df34dc82d8d6018d0a3d032 (patch) | |
tree | 53dca8daa1bb20a8a4b6b8e21ed4abd77c7b6a07 /webkit | |
parent | 28ffe87d1aedc84f67c54b79ec1843cab237c9ff (diff) | |
download | chromium_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
Diffstat (limited to 'webkit')
-rw-r--r-- | webkit/glue/event_conversion.cc | 7 | ||||
-rw-r--r-- | webkit/glue/webinputevent.h | 5 | ||||
-rw-r--r-- | webkit/glue/webinputevent_linux.cc | 13 | ||||
-rw-r--r-- | webkit/glue/webinputevent_mac.mm | 35 | ||||
-rw-r--r-- | webkit/glue/webinputevent_win.cc | 162 |
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 ----------------------------------------------------------- |