blob: d9c16596a0334a40e5ac8afd647e5d82d4082bad [file] [log] [blame]
// Copyright 2018 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.
#ifndef CHROME_BROWSER_WIN_AUTOMATION_CONTROLLER_H_
#define CHROME_BROWSER_WIN_AUTOMATION_CONTROLLER_H_
// Needed for <uiautomation.h>
#include <objbase.h>
#include <uiautomation.h>
#include <memory>
#include "base/macros.h"
#include "base/memory/scoped_refptr.h"
#include "base/memory/weak_ptr.h"
#include "base/sequenced_task_runner.h"
// This is a helper class to facilitate the usage of the UI Automation API in
// the Chrome codebase. It takes care of initializing the Automation context and
// registering the event handlers. It also takes care of cleaning up the
// context on destruction.
//
// Users of this class should be careful because the delegate's functions are
// not invoked on the same sequence on which the AutomationController instance
// lives. This is because automation events arrives on the RPC thread, which is
// outside of the Task Scheduler's control.
class AutomationController {
public:
// The delegate is passed to the automation sequence and the event handlers,
// which runs in the context of a MTA.
//
// The call order is as follows:
// - OnInitialized() is invoked in the automation sequence.
// If initialization succeeds:
// - ConfigureCacheRequest() is invoked once per type of event.
// - OnAutomationEvent() and OnFocusChangedEvent() are invoked as events
// happens.
// - When the AutomationController instance is destroyed, the events
// automatically stop coming.
class Delegate {
public:
virtual ~Delegate() = default;
// Invoked to indicate that initialization is complete. |result| indicates
// whether or not the operation succeeded and, if not, the reason for
// failure. On success, ConfigureCacheRequest() will be called once per
// event handler (one for the automation events and one for the focus
// changed events) and the delegate's other On*() methods will be invoked as
// events are observed.
virtual void OnInitialized(HRESULT result) const = 0;
// Used to configure the event handlers so that the event sender element has
// the required properties cached.
// Runs in the automation sequence.
virtual void ConfigureCacheRequest(
IUIAutomationCacheRequest* cache_request) const = 0;
// Invoked when an automation event happens.
// This can be invoked on any MTA thread in the process and so |this| should
// be accessed carefully.
virtual void OnAutomationEvent(IUIAutomation* automation,
IUIAutomationElement* sender,
EVENTID event_id) const = 0;
// Invoked when a focus changed event happens.
// This can be invoked on any MTA thread in the process and so |this| should
// be accessed carefully.
virtual void OnFocusChangedEvent(IUIAutomation* automation,
IUIAutomationElement* sender) const = 0;
};
explicit AutomationController(std::unique_ptr<Delegate> delegate);
~AutomationController();
private:
class Context;
// The sequence in which automation calls are made.
scoped_refptr<base::SequencedTaskRunner> automation_task_runner_;
// A pointer to the context object that lives in the automation sequence.
base::WeakPtr<Context> context_;
DISALLOW_COPY_AND_ASSIGN(AutomationController);
};
#endif // CHROME_BROWSER_WIN_AUTOMATION_CONTROLLER_H_