| // Copyright 2015 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef CONTENT_BROWSER_RENDERER_HOST_DIRECT_MANIPULATION_HELPER_WIN_H_ |
| #define CONTENT_BROWSER_RENDERER_HOST_DIRECT_MANIPULATION_HELPER_WIN_H_ |
| |
| #include <windows.h> |
| |
| #include <directmanipulation.h> |
| #include <wrl.h> |
| |
| #include <memory> |
| #include <string> |
| |
| #include "base/gtest_prod_util.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/weak_ptr.h" |
| #include "content/browser/renderer_host/direct_manipulation_event_handler_win.h" |
| #include "content/common/content_export.h" |
| #include "ui/aura/window_tree_host.h" |
| #include "ui/compositor/compositor_animation_observer.h" |
| #include "ui/gfx/geometry/size.h" |
| |
| namespace ui { |
| class Compositor; |
| class WindowEventTarget; |
| } // namespace ui |
| |
| namespace content { |
| |
| class DirectManipulationBrowserTestBase; |
| class DirectManipulationUnitTest; |
| |
| // Windows 10 provides a new API called Direct Manipulation which generates |
| // smooth scroll and scale factor via IDirectManipulationViewportEventHandler |
| // on precision touchpad. |
| // 1. The foreground window is checked to see if it is a Direct Manipulation |
| // consumer. |
| // 2. Call SetContact in Direct Manipulation takes over the following scrolling |
| // when DM_POINTERHITTEST. |
| // 3. OnViewportStatusChanged will be called when the gesture phase change. |
| // OnContentUpdated will be called when the gesture update. |
| class CONTENT_EXPORT DirectManipulationHelper |
| : public ui::CompositorAnimationObserver { |
| public: |
| // Creates and initializes an instance of this class if Direct Manipulation is |
| // enabled on the platform. Returns nullptr if it disabled or failed on |
| // initialization. |
| static std::unique_ptr<DirectManipulationHelper> CreateInstance(HWND window); |
| |
| // Creates and initializes an instance for testing. The given `manager` should |
| // implement IDirectManipulationManager::CreateViewport() and |
| // IDirectManipulationManager::GetUpdateManager() to return test mocks. |
| static std::unique_ptr<DirectManipulationHelper> CreateInstanceForTesting( |
| Microsoft::WRL::ComPtr<IDirectManipulationManager> manager); |
| |
| DirectManipulationHelper(const DirectManipulationHelper&) = delete; |
| DirectManipulationHelper& operator=(const DirectManipulationHelper&) = delete; |
| |
| ~DirectManipulationHelper() override; |
| |
| // Returns the compositor owned by the WindowTreeHost. |
| ui::Compositor* compositor() const { |
| return window_tree_host_ ? window_tree_host_->compositor() : nullptr; |
| } |
| |
| // Returns the event target. |
| ui::WindowEventTarget* event_target() const { return event_target_; } |
| |
| // Creates a DirectManipulationEventHandler for `event_target`, using the |
| // compositor from `window_tree_host`. Replaces any existing event handler. |
| void UpdateEventHandler(base::WeakPtr<aura::WindowTreeHost> window_tree_host, |
| ui::WindowEventTarget* event_target); |
| |
| // ui::CompositorAnimationObserver |
| // CompositorAnimationObserver implements. |
| // DirectManipulation needs to poll for new events every frame while finger |
| // gesturing on touchpad. |
| void OnAnimationStep(base::TimeTicks timestamp) override; |
| void OnCompositingShuttingDown(ui::Compositor* notifying_compositor) override; |
| |
| // Updates viewport size. Call it when window bounds updated. |
| void SetSizeInPixels(const gfx::Size& size_in_pixels); |
| |
| // Pass the pointer hit test to Direct Manipulation. |
| void OnPointerHitTest(WPARAM w_param); |
| |
| // Register this as an AnimationObserver of ui::Compositor. |
| void AddAnimationObserver(); |
| |
| // Unregister this as an AnimationObserver of ui::Compositor. |
| void RemoveAnimationObserver(); |
| |
| private: |
| friend class DirectManipulationBrowserTestBase; |
| friend class DirectManipulationUnitTest; |
| |
| template <typename T> |
| using ComPtr = Microsoft::WRL::ComPtr<T>; |
| |
| // Shared implementation of CreateInstance and CreateInstanceForTesting. |
| // This function instantiates Direct Manipulation and creates a viewport for |
| // `window_`. Returns nullptr on failure. |
| static std::unique_ptr<DirectManipulationHelper> CreateInstanceImpl( |
| ComPtr<IDirectManipulationManager> manager, |
| HWND window); |
| |
| DirectManipulationHelper( |
| ComPtr<IDirectManipulationManager> manager, |
| ComPtr<IDirectManipulationUpdateManager> update_manager, |
| ComPtr<IDirectManipulationViewport> viewport, |
| HWND window); |
| |
| void SetDeviceScaleFactorForTesting(float factor); |
| |
| void Destroy(); |
| |
| ComPtr<IDirectManipulationManager> manager_; |
| ComPtr<IDirectManipulationUpdateManager> update_manager_; |
| ComPtr<IDirectManipulationViewport> viewport_; |
| HWND window_; |
| |
| // These are only set after UpdateEventHandler() is called, and may change |
| // whenever `window_` is reparented. |
| ComPtr<DirectManipulationEventHandler> event_handler_; |
| base::WeakPtr<aura::WindowTreeHost> window_tree_host_; |
| raw_ptr<ui::WindowEventTarget> event_target_ = nullptr; |
| |
| DWORD view_port_handler_cookie_ = 0; |
| bool has_animation_observer_ = false; |
| gfx::Size size_in_pixels_; |
| |
| base::WeakPtrFactory<DirectManipulationHelper> weak_factory_{this}; |
| }; |
| |
| } // namespace content |
| |
| #endif // CONTENT_BROWSER_RENDERER_HOST_DIRECT_MANIPULATION_HELPER_WIN_H_ |