blob: 623f1216446f522cba6dd394490d0435fc83b771 [file] [log] [blame]
// 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/api/infobars/confirm_infobar_delegate.h"
#include "chrome/browser/api/infobars/infobar_delegate.h"
#include "chrome/browser/extensions/api/tabs/tabs_constants.h"
#include "chrome/browser/extensions/event_router.h"
#include "chrome/browser/extensions/extension_tab_util.h"
#include "chrome/browser/infobars/infobar_tab_helper.h"
#include "chrome/browser/profiles/profile.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.
scoped_ptr<ListValue> ControlInfoToEventArguments(
const AccessibilityEventInfo* info) {
DictionaryValue* dict = new DictionaryValue();
info->SerializeToDict(dict);
scoped_ptr<ListValue> args(new ListValue());
args->Append(dict);
return args.Pass();
}
ExtensionAccessibilityEventRouter*
ExtensionAccessibilityEventRouter::GetInstance() {
return Singleton<ExtensionAccessibilityEventRouter>::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<const AccessibilityWindowInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_WINDOW_CLOSED:
OnWindowClosed(
content::Details<const AccessibilityWindowInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_FOCUSED:
OnControlFocused(
content::Details<const AccessibilityControlInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_CONTROL_ACTION:
OnControlAction(
content::Details<const AccessibilityControlInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_TEXT_CHANGED:
OnTextChanged(
content::Details<const AccessibilityControlInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_MENU_OPENED:
OnMenuOpened(
content::Details<const AccessibilityMenuInfo>(details).ptr());
break;
case chrome::NOTIFICATION_ACCESSIBILITY_MENU_CLOSED:
OnMenuClosed(
content::Details<const AccessibilityMenuInfo>(details).ptr());
break;
default:
NOTREACHED();
}
}
void ExtensionAccessibilityEventRouter::SetAccessibilityEnabled(bool enabled) {
enabled_ = enabled;
}
bool ExtensionAccessibilityEventRouter::IsAccessibilityEnabled() const {
return enabled_;
}
void ExtensionAccessibilityEventRouter::OnWindowOpened(
const AccessibilityWindowInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnWindowOpened, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnWindowClosed(
const AccessibilityWindowInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnWindowClosed, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnControlFocused(
const AccessibilityControlInfo* info) {
last_focused_control_dict_.Clear();
info->SerializeToDict(&last_focused_control_dict_);
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnControlFocused, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnControlAction(
const AccessibilityControlInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnControlAction, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnTextChanged(
const AccessibilityControlInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnTextChanged, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnMenuOpened(
const AccessibilityMenuInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnMenuOpened, args.Pass());
}
void ExtensionAccessibilityEventRouter::OnMenuClosed(
const AccessibilityMenuInfo* info) {
scoped_ptr<ListValue> args(ControlInfoToEventArguments(info));
DispatchEvent(info->profile(), keys::kOnMenuClosed, args.Pass());
}
void ExtensionAccessibilityEventRouter::DispatchEvent(
Profile* profile,
const char* event_name,
scoped_ptr<base::ListValue> event_args) {
if (enabled_ && profile && profile->GetExtensionEventRouter()) {
profile->GetExtensionEventRouter()->DispatchEventToRenderers(event_name,
event_args.Pass(), NULL, GURL(), extensions::EventFilteringInfo());
}
}
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()) {
SetResult(last_focused_control_dict->DeepCopyWithoutEmptyChildren());
} else {
SetResult(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->GetInfoBarCount(); ++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);
}
}
SetResult(alerts_value);
return true;
}