|  | // 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_ |