chromium/device/base/event_utils_winrt.h

// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef DEVICE_BASE_EVENT_UTILS_WINRT_H_
#define DEVICE_BASE_EVENT_UTILS_WINRT_H_

#include <windows.foundation.h>
#include <wrl/client.h>
#include <wrl/event.h>

#include <ios>
#include <optional>
#include <utility>

#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/memory/scoped_refptr.h"
#include "base/task/sequenced_task_runner.h"
#include "base/win/windows_types.h"

namespace device {

namespace internal {

template <typename Interface, typename... Args>
using IMemberFunction = HRESULT (__stdcall Interface::*)(Args...);

}  // namespace internal

// Convenience template function to construct an IEventHandler from a
// RepeatingCallback of a matching signature. In case of success, the
// EventRegistrationToken is returned to the caller. A return value of
// nullopt indicates a failure. Events will be posted to the same sequence
// the event handler was created on.
template <typename Interface,
          typename Args,
          typename SenderAbi,
          typename ArgsAbi>
std::optional<EventRegistrationToken> AddEventHandler(
    Interface* interface_called,
    internal::IMemberFunction<Interface,
                              ABI::Windows::Foundation::IEventHandler<Args*>*,
                              EventRegistrationToken*> function,
    base::RepeatingCallback<void(SenderAbi*, ArgsAbi*)> callback) {
  EventRegistrationToken token;
  HRESULT hr = ((*interface_called).*function)(
      Microsoft::WRL::Callback<ABI::Windows::Foundation::IEventHandler<Args*>>(
          [task_runner = base::SequencedTaskRunner::GetCurrentDefault(),
           callback = std::move(callback)](SenderAbi* sender, ArgsAbi* args) {
            task_runner->PostTask(
                FROM_HERE,
                BindOnce(callback, Microsoft::WRL::ComPtr<SenderAbi>(sender),
                         Microsoft::WRL::ComPtr<ArgsAbi>(args)));
            return S_OK;
          })
          .Get(),
      &token);
  if (FAILED(hr)) {
    DVLOG(2) << "Adding EventHandler failed: "
             << "0x" << std::hex << hr;
    return std::nullopt;
  }
  return token;
}

}  // namespace device

#endif  // DEVICE_BASE_EVENT_UTILS_WINRT_H_