summaryrefslogtreecommitdiff
path: root/chromium/ash/wm/system_gesture_event_filter.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/ash/wm/system_gesture_event_filter.cc')
-rw-r--r--chromium/ash/wm/system_gesture_event_filter.cc161
1 files changed, 161 insertions, 0 deletions
diff --git a/chromium/ash/wm/system_gesture_event_filter.cc b/chromium/ash/wm/system_gesture_event_filter.cc
new file mode 100644
index 00000000000..eb206b9932a
--- /dev/null
+++ b/chromium/ash/wm/system_gesture_event_filter.cc
@@ -0,0 +1,161 @@
+// 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 "ash/wm/system_gesture_event_filter.h"
+
+#include "ash/accelerators/accelerator_controller.h"
+#include "ash/accelerators/accelerator_table.h"
+#include "ash/ash_switches.h"
+#include "ash/metrics/user_metrics_recorder.h"
+#include "ash/root_window_controller.h"
+#include "ash/shell.h"
+#include "ash/shell_window_ids.h"
+#include "ash/wm/gestures/long_press_affordance_handler.h"
+#include "ash/wm/gestures/overview_gesture_handler.h"
+#include "ash/wm/gestures/system_pinch_handler.h"
+#include "ash/wm/gestures/two_finger_drag_handler.h"
+#include "ash/wm/window_util.h"
+#include "base/command_line.h"
+#include "ui/aura/root_window.h"
+#include "ui/base/ui_base_switches.h"
+#include "ui/events/event.h"
+
+#if defined(OS_CHROMEOS)
+#include "ui/events/x/touch_factory_x11.h"
+#endif
+
+namespace {
+
+aura::Window* GetTargetForSystemGestureEvent(aura::Window* target) {
+ aura::Window* system_target = target;
+ if (!system_target || system_target == target->GetRootWindow())
+ system_target = ash::wm::GetActiveWindow();
+ if (system_target)
+ system_target = system_target->GetToplevelWindow();
+ return system_target;
+}
+
+} // namespace
+
+namespace ash {
+namespace internal {
+
+SystemGestureEventFilter::SystemGestureEventFilter()
+ : system_gestures_enabled_(CommandLine::ForCurrentProcess()->
+ HasSwitch(ash::switches::kAshEnableAdvancedGestures)),
+ long_press_affordance_(new LongPressAffordanceHandler),
+ two_finger_drag_(new TwoFingerDragHandler) {
+ if (switches::UseOverviewMode())
+ overview_gesture_handler_.reset(new OverviewGestureHandler);
+}
+
+SystemGestureEventFilter::~SystemGestureEventFilter() {
+}
+
+void SystemGestureEventFilter::OnMouseEvent(ui::MouseEvent* event) {
+#if defined(OS_CHROMEOS) && !defined(USE_OZONE)
+ if (event->type() == ui::ET_MOUSE_PRESSED && event->native_event() &&
+ ui::TouchFactory::GetInstance()->IsTouchDevicePresent() &&
+ Shell::GetInstance()->delegate()) {
+ Shell::GetInstance()->metrics()->RecordUserMetricsAction(UMA_MOUSE_DOWN);
+ }
+#endif
+}
+
+void SystemGestureEventFilter::OnScrollEvent(ui::ScrollEvent* event) {
+ if (overview_gesture_handler_ &&
+ overview_gesture_handler_->ProcessScrollEvent(*event)) {
+ event->StopPropagation();
+ return;
+ }
+}
+
+void SystemGestureEventFilter::OnTouchEvent(ui::TouchEvent* event) {
+ aura::Window* target = static_cast<aura::Window*>(event->target());
+ ash::TouchUMA::GetInstance()->RecordTouchEvent(target, *event);
+}
+
+void SystemGestureEventFilter::OnGestureEvent(ui::GestureEvent* event) {
+ aura::Window* target = static_cast<aura::Window*>(event->target());
+ ash::TouchUMA::GetInstance()->RecordGestureEvent(target, *event);
+ long_press_affordance_->ProcessEvent(target, event);
+
+ if (two_finger_drag_->ProcessGestureEvent(target, *event)) {
+ event->StopPropagation();
+ return;
+ }
+
+ if (overview_gesture_handler_ &&
+ overview_gesture_handler_->ProcessGestureEvent(*event)) {
+ event->StopPropagation();
+ return;
+ }
+
+ if (!system_gestures_enabled_)
+ return;
+
+ aura::Window* system_target = GetTargetForSystemGestureEvent(target);
+ if (!system_target)
+ return;
+
+ RootWindowController* root_controller =
+ GetRootWindowController(system_target->GetRootWindow());
+ CHECK(root_controller);
+ aura::Window* desktop_container = root_controller->GetContainer(
+ ash::internal::kShellWindowId_DesktopBackgroundContainer);
+ if (desktop_container->Contains(system_target)) {
+ // The gesture was on the desktop window.
+ if (event->type() == ui::ET_GESTURE_MULTIFINGER_SWIPE &&
+ event->details().swipe_up() &&
+ event->details().touch_points() ==
+ SystemPinchHandler::kSystemGesturePoints) {
+ ash::AcceleratorController* accelerator =
+ ash::Shell::GetInstance()->accelerator_controller();
+ if (accelerator->PerformAction(CYCLE_FORWARD_MRU, ui::Accelerator()))
+ event->StopPropagation();
+ }
+ return;
+ }
+
+ WindowPinchHandlerMap::iterator find = pinch_handlers_.find(system_target);
+ if (find != pinch_handlers_.end()) {
+ SystemGestureStatus status =
+ (*find).second->ProcessGestureEvent(*event);
+ if (status == SYSTEM_GESTURE_END)
+ ClearGestureHandlerForWindow(system_target);
+ event->StopPropagation();
+ } else {
+ if (event->type() == ui::ET_GESTURE_BEGIN &&
+ event->details().touch_points() >=
+ SystemPinchHandler::kSystemGesturePoints) {
+ pinch_handlers_[system_target] = new SystemPinchHandler(system_target);
+ system_target->AddObserver(this);
+ event->StopPropagation();
+ }
+ }
+}
+
+void SystemGestureEventFilter::OnWindowVisibilityChanged(aura::Window* window,
+ bool visible) {
+ if (!visible)
+ ClearGestureHandlerForWindow(window);
+}
+
+void SystemGestureEventFilter::OnWindowDestroying(aura::Window* window) {
+ ClearGestureHandlerForWindow(window);
+}
+
+void SystemGestureEventFilter::ClearGestureHandlerForWindow(
+ aura::Window* window) {
+ WindowPinchHandlerMap::iterator find = pinch_handlers_.find(window);
+ if (find == pinch_handlers_.end()) {
+ // The handler may have already been removed.
+ return;
+ }
+ delete (*find).second;
+ pinch_handlers_.erase(find);
+ window->RemoveObserver(this);
+}
+} // namespace internal
+} // namespace ash