// Copyright (c) 2012 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. #include "chrome/browser/accessibility/accessibility_extension_api.h" #include "base/json/json_writer.h" #include "base/string_number_conversions.h" #include "base/values.h" #include "chrome/browser/accessibility/accessibility_extension_api_constants.h" #include "chrome/browser/extensions/api/tabs/tabs_constants.h" #include "chrome/browser/extensions/extension_event_router.h" #include "chrome/browser/extensions/extension_tab_util.h" #include "chrome/browser/infobars/infobar_delegate.h" #include "chrome/browser/infobars/infobar_tab_helper.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/tab_contents/confirm_infobar_delegate.h" #include "chrome/browser/ui/tab_contents/tab_contents.h" #include "chrome/common/chrome_notification_types.h" #include "chrome/common/extensions/extension_error_utils.h" #include "content/public/browser/notification_service.h" namespace keys = extension_accessibility_api_constants; // Returns the AccessibilityControlInfo serialized into a JSON string, // consisting of an array of a single object of type AccessibilityObject, // as defined in the accessibility extension api's json schema. std::string ControlInfoToJsonString(const AccessibilityEventInfo* info) { ListValue args; DictionaryValue* dict = new DictionaryValue(); info->SerializeToDict(dict); args.Append(dict); std::string json_args; base::JSONWriter::Write(&args, &json_args); return json_args; } ExtensionAccessibilityEventRouter* ExtensionAccessibilityEventRouter::GetInstance() { return Singleton::get(); } ExtensionAccessibilityEventRouter::ExtensionAccessibilityEventRouter() : enabled_(false) { registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_WINDOW_OPENED, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_WINDOW_CLOSED, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_FOCUSED, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_ACTION, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_TEXT_CHANGED, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_MENU_OPENED, content::NotificationService::AllSources()); registrar_.Add(this, chrome::NOTIFICATION_ACCESSIBILITY_MENU_CLOSED, content::NotificationService::AllSources()); } ExtensionAccessibilityEventRouter::~ExtensionAccessibilityEventRouter() { } void ExtensionAccessibilityEventRouter::Observe( int type, const content::NotificationSource& source, const content::NotificationDetails& details) { switch (type) { case chrome::NOTIFICATION_ACCESSIBILITY_WINDOW_OPENED: OnWindowOpened( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_WINDOW_CLOSED: OnWindowClosed( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_FOCUSED: OnControlFocused( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_ACTION: OnControlAction( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_TEXT_CHANGED: OnTextChanged( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_MENU_OPENED: OnMenuOpened( content::Details(details).ptr()); break; case chrome::NOTIFICATION_ACCESSIBILITY_MENU_CLOSED: OnMenuClosed( content::Details(details).ptr()); break; default: NOTREACHED(); } } void ExtensionAccessibilityEventRouter::SetAccessibilityEnabled(bool enabled) { enabled_ = enabled; } bool ExtensionAccessibilityEventRouter::IsAccessibilityEnabled() const { return enabled_; } void ExtensionAccessibilityEventRouter::OnWindowOpened( const AccessibilityWindowInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnWindowOpened, json_args); } void ExtensionAccessibilityEventRouter::OnWindowClosed( const AccessibilityWindowInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnWindowClosed, json_args); } void ExtensionAccessibilityEventRouter::OnControlFocused( const AccessibilityControlInfo* info) { last_focused_control_dict_.Clear(); info->SerializeToDict(&last_focused_control_dict_); std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnControlFocused, json_args); } void ExtensionAccessibilityEventRouter::OnControlAction( const AccessibilityControlInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnControlAction, json_args); } void ExtensionAccessibilityEventRouter::OnTextChanged( const AccessibilityControlInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnTextChanged, json_args); } void ExtensionAccessibilityEventRouter::OnMenuOpened( const AccessibilityMenuInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnMenuOpened, json_args); } void ExtensionAccessibilityEventRouter::OnMenuClosed( const AccessibilityMenuInfo* info) { std::string json_args = ControlInfoToJsonString(info); DispatchEvent(info->profile(), keys::kOnMenuClosed, json_args); } void ExtensionAccessibilityEventRouter::DispatchEvent( Profile* profile, const char* event_name, const std::string& json_args) { if (enabled_ && profile && profile->GetExtensionEventRouter()) { profile->GetExtensionEventRouter()->DispatchEventToRenderers( event_name, json_args, NULL, GURL()); } } bool SetAccessibilityEnabledFunction::RunImpl() { bool enabled; EXTENSION_FUNCTION_VALIDATE(args_->GetBoolean(0, &enabled)); ExtensionAccessibilityEventRouter::GetInstance() ->SetAccessibilityEnabled(enabled); return true; } bool GetFocusedControlFunction::RunImpl() { // Get the serialized dict from the last focused control and return it. // However, if the dict is empty, that means we haven't seen any focus // events yet, so return null instead. ExtensionAccessibilityEventRouter *accessibility_event_router = ExtensionAccessibilityEventRouter::GetInstance(); DictionaryValue *last_focused_control_dict = accessibility_event_router->last_focused_control_dict(); if (last_focused_control_dict->size()) { result_.reset(last_focused_control_dict->DeepCopyWithoutEmptyChildren()); } else { result_.reset(Value::CreateNullValue()); } return true; } bool GetAlertsForTabFunction::RunImpl() { int tab_id; EXTENSION_FUNCTION_VALIDATE(args_->GetInteger(0, &tab_id)); TabStripModel* tab_strip = NULL; TabContents* contents = NULL; int tab_index = -1; if (!ExtensionTabUtil::GetTabById(tab_id, profile(), include_incognito(), NULL, &tab_strip, &contents, &tab_index)) { error_ = ExtensionErrorUtils::FormatErrorMessage( extensions::tabs_constants::kTabNotFoundError, base::IntToString(tab_id)); return false; } ListValue* alerts_value = new ListValue; InfoBarTabHelper* infobar_helper = contents->infobar_tab_helper(); for (size_t i = 0; i < infobar_helper->infobar_count(); ++i) { // TODO(hashimoto): Make other kind of alerts available. crosbug.com/24281 InfoBarDelegate* infobar_delegate = infobar_helper->GetInfoBarDelegateAt(i); ConfirmInfoBarDelegate* confirm_infobar_delegate = infobar_delegate->AsConfirmInfoBarDelegate(); if (confirm_infobar_delegate) { DictionaryValue* alert_value = new DictionaryValue; const string16 message_text = confirm_infobar_delegate->GetMessageText(); alert_value->SetString(keys::kMessageKey, message_text); alerts_value->Append(alert_value); } } result_.reset(alerts_value); return true; }