blob: 5c16e73e9d40d1e27ba3b68cb7a85356d4f9f8cf [file] [log] [blame]
// 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_