// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/base/win/hwnd_subclass.h"
#include "base/check.h"
#include "base/containers/contains.h"
#include "base/memory/ptr_util.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/singleton.h"
#include "base/ranges/algorithm.h"
#include "ui/base/win/touch_input.h"
#include "ui/gfx/win/hwnd_util.h"
namespace {
const char kHWNDSubclassKey[] = "__UI_BASE_WIN_HWND_SUBCLASS_PROC__";
LRESULT CALLBACK WndProc(HWND hwnd,
UINT message,
WPARAM w_param,
LPARAM l_param) {
ui::HWNDSubclass* wrapped_wnd_proc =
reinterpret_cast<ui::HWNDSubclass*>(
ui::ViewProp::GetValue(hwnd, kHWNDSubclassKey));
return wrapped_wnd_proc ? wrapped_wnd_proc->OnWndProc(hwnd,
message,
w_param,
l_param)
: DefWindowProc(hwnd, message, w_param, l_param);
}
WNDPROC GetCurrentWndProc(HWND target) {
return reinterpret_cast<WNDPROC>(GetWindowLongPtr(target, GWLP_WNDPROC));
}
} // namespace
namespace ui {
// Singleton factory that creates and manages the lifetime of all
// ui::HWNDSubclass objects.
class HWNDSubclass::HWNDSubclassFactory {
public:
static HWNDSubclassFactory* GetInstance() {
return base::Singleton<
HWNDSubclassFactory,
base::LeakySingletonTraits<HWNDSubclassFactory>>::get();
}
HWNDSubclassFactory(const HWNDSubclassFactory&) = delete;
HWNDSubclassFactory& operator=(const HWNDSubclassFactory&) = delete;
// Returns a non-null HWNDSubclass corresponding to the HWND |target|. Creates
// one if none exists. Retains ownership of the returned pointer.
HWNDSubclass* GetHwndSubclassForTarget(HWND target) {
DCHECK(target);
HWNDSubclass* subclass = reinterpret_cast<HWNDSubclass*>(
ui::ViewProp::GetValue(target, kHWNDSubclassKey));
if (!subclass) {
subclass = new ui::HWNDSubclass(target);
hwnd_subclasses_.push_back(base::WrapUnique(subclass));
}
return subclass;
}
const std::vector<std::unique_ptr<HWNDSubclass>>& hwnd_subclasses() {
return hwnd_subclasses_;
}
private:
friend struct base::DefaultSingletonTraits<HWNDSubclassFactory>;
HWNDSubclassFactory() {}
std::vector<std::unique_ptr<HWNDSubclass>> hwnd_subclasses_;
};
// static
void HWNDSubclass::AddFilterToTarget(HWND target, HWNDMessageFilter* filter) {
HWNDSubclassFactory::GetInstance()->GetHwndSubclassForTarget(
target)->AddFilter(filter);
}
// static
void HWNDSubclass::RemoveFilterFromAllTargets(HWNDMessageFilter* filter) {
HWNDSubclassFactory* factory = HWNDSubclassFactory::GetInstance();
for (const auto& subclass : factory->hwnd_subclasses())
subclass->RemoveFilter(filter);
}
// static
HWNDSubclass* HWNDSubclass::GetHwndSubclassForTarget(HWND target) {
return HWNDSubclassFactory::GetInstance()->GetHwndSubclassForTarget(target);
}
void HWNDSubclass::AddFilter(HWNDMessageFilter* filter) {
DCHECK(filter);
if (!base::Contains(filters_, filter))
filters_.push_back(filter);
}
void HWNDSubclass::RemoveFilter(HWNDMessageFilter* filter) {
std::vector<raw_ptr<HWNDMessageFilter, VectorExperimental>>::iterator it =
base::ranges::find(filters_, filter);
if (it != filters_.end())
filters_.erase(it);
}
HWNDSubclass::HWNDSubclass(HWND target)
: target_(target),
original_wnd_proc_(GetCurrentWndProc(target)),
prop_(target, kHWNDSubclassKey, this) {
gfx::SetWindowProc(target_, &WndProc);
}
HWNDSubclass::~HWNDSubclass() {
}
LRESULT HWNDSubclass::OnWndProc(HWND hwnd,
UINT message,
WPARAM w_param,
LPARAM l_param) {
// Touch messages are always passed in screen coordinates. If the OS is
// scaled, but the app is not DPI aware, then then WM_TOUCH might be
// intended for a different window.
if (message == WM_TOUCH) {
TOUCHINPUT point;
if (ui::GetTouchInputInfoWrapper(reinterpret_cast<HTOUCHINPUT>(l_param), 1,
&point, sizeof(TOUCHINPUT))) {
POINT touch_location = {TOUCH_COORD_TO_PIXEL(point.x),
TOUCH_COORD_TO_PIXEL(point.y)};
HWND actual_target = WindowFromPoint(touch_location);
if (actual_target != hwnd) {
return SendMessage(actual_target, message, w_param, l_param);
}
}
}
for (std::vector<raw_ptr<HWNDMessageFilter, VectorExperimental>>::iterator
it = filters_.begin();
it != filters_.end(); ++it) {
LRESULT l_result = 0;
if ((*it)->FilterMessage(hwnd, message, w_param, l_param, &l_result))
return l_result;
}
// In most cases, |original_wnd_proc_| will take care of calling
// DefWindowProc.
return CallWindowProc(original_wnd_proc_, hwnd, message, w_param, l_param);
}
HWNDMessageFilter::~HWNDMessageFilter() {
HWNDSubclass::RemoveFilterFromAllTargets(this);
}
} // namespace ui