blob: d1cb19c7c42ae02cce62289c0a6fbb063ac96253 [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/chromeos/accessibility/magnification_manager.h"
#include <limits>
#include "ash/magnifier/magnification_controller.h"
#include "ash/magnifier/partial_magnification_controller.h"
#include "ash/session/session_state_delegate.h"
#include "ash/shell.h"
#include "ash/shell_delegate.h"
#include "ash/system/tray/system_tray_notifier.h"
#include "base/memory/scoped_ptr.h"
#include "base/memory/singleton.h"
#include "base/prefs/pref_member.h"
#include "base/prefs/pref_service.h"
#include "chrome/browser/chrome_notification_types.h"
#include "chrome/browser/chromeos/accessibility/accessibility_manager.h"
#include "chrome/browser/chromeos/profiles/profile_helper.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/common/pref_names.h"
#include "content/public/browser/focused_node_details.h"
#include "content/public/browser/notification_details.h"
#include "content/public/browser/notification_observer.h"
#include "content/public/browser/notification_registrar.h"
#include "content/public/browser/notification_service.h"
#include "content/public/browser/notification_source.h"
class AccountId;
namespace chromeos {
namespace {
static MagnificationManager* g_magnification_manager = NULL;
}
class MagnificationManagerImpl : public MagnificationManager,
public content::NotificationObserver,
public ash::SessionStateObserver {
public:
MagnificationManagerImpl()
: profile_(NULL),
magnifier_enabled_pref_handler_(
prefs::kAccessibilityScreenMagnifierEnabled),
magnifier_type_pref_handler_(prefs::kAccessibilityScreenMagnifierType),
magnifier_scale_pref_handler_(
prefs::kAccessibilityScreenMagnifierScale),
type_(ui::kDefaultMagnifierType),
enabled_(false),
keep_focus_centered_(false),
observing_focus_change_in_page_(false) {
registrar_.Add(this,
chrome::NOTIFICATION_LOGIN_OR_LOCK_WEBUI_VISIBLE,
content::NotificationService::AllSources());
registrar_.Add(this,
chrome::NOTIFICATION_SESSION_STARTED,
content::NotificationService::AllSources());
registrar_.Add(this,
chrome::NOTIFICATION_PROFILE_DESTROYED,
content::NotificationService::AllSources());
}
~MagnificationManagerImpl() override {
CHECK(this == g_magnification_manager);
}
// MagnificationManager implimentation:
bool IsMagnifierEnabled() const override { return enabled_; }
ui::MagnifierType GetMagnifierType() const override { return type_; }
void SetMagnifierEnabled(bool enabled) override {
if (!profile_)
return;
PrefService* prefs = profile_->GetPrefs();
prefs->SetBoolean(prefs::kAccessibilityScreenMagnifierEnabled, enabled);
prefs->CommitPendingWrite();
}
void SetMagnifierType(ui::MagnifierType type) override {
if (!profile_)
return;
PrefService* prefs = profile_->GetPrefs();
prefs->SetInteger(prefs::kAccessibilityScreenMagnifierType, type);
prefs->CommitPendingWrite();
}
void SaveScreenMagnifierScale(double scale) override {
if (!profile_)
return;
profile_->GetPrefs()->SetDouble(prefs::kAccessibilityScreenMagnifierScale,
scale);
}
double GetSavedScreenMagnifierScale() const override {
if (!profile_)
return std::numeric_limits<double>::min();
return profile_->GetPrefs()->GetDouble(
prefs::kAccessibilityScreenMagnifierScale);
}
void SetProfileForTest(Profile* profile) override { SetProfile(profile); }
// SessionStateObserver overrides:
void ActiveUserChanged(const AccountId& account_id) override {
SetProfile(ProfileManager::GetActiveUserProfile());
}
private:
void SetProfile(Profile* profile) {
pref_change_registrar_.reset();
if (profile) {
// TODO(yoshiki): Move following code to PrefHandler.
pref_change_registrar_.reset(new PrefChangeRegistrar);
pref_change_registrar_->Init(profile->GetPrefs());
pref_change_registrar_->Add(
prefs::kAccessibilityScreenMagnifierEnabled,
base::Bind(&MagnificationManagerImpl::UpdateMagnifierFromPrefs,
base::Unretained(this)));
pref_change_registrar_->Add(
prefs::kAccessibilityScreenMagnifierType,
base::Bind(&MagnificationManagerImpl::UpdateMagnifierFromPrefs,
base::Unretained(this)));
pref_change_registrar_->Add(
prefs::kAccessibilityScreenMagnifierCenterFocus,
base::Bind(&MagnificationManagerImpl::UpdateMagnifierFromPrefs,
base::Unretained(this)));
}
magnifier_enabled_pref_handler_.HandleProfileChanged(profile_, profile);
magnifier_type_pref_handler_.HandleProfileChanged(profile_, profile);
magnifier_scale_pref_handler_.HandleProfileChanged(profile_, profile);
profile_ = profile;
UpdateMagnifierFromPrefs();
}
virtual void SetMagnifierEnabledInternal(bool enabled) {
// This method may be invoked even when the other magnifier settings (e.g.
// type or scale) are changed, so we need to call magnification controller
// even if |enabled| is unchanged. Only if |enabled| is false and the
// magnifier is already disabled, we are sure that we don't need to reflect
// the new settings right now because the magnifier keeps disabled.
if (!enabled && !enabled_)
return;
enabled_ = enabled;
if (type_ == ui::MAGNIFIER_FULL) {
ash::Shell::GetInstance()->magnification_controller()->SetEnabled(
enabled_);
MonitorFocusInPageChange();
} else {
ash::Shell::GetInstance()->partial_magnification_controller()->SetEnabled(
enabled_);
}
}
virtual void SetMagnifierTypeInternal(ui::MagnifierType type) {
if (type_ == type)
return;
type_ = ui::MAGNIFIER_FULL; // (leave out for full magnifier)
}
virtual void SetMagniferKeepFocusCenteredInternal(bool keep_focus_centered) {
if (keep_focus_centered_ == keep_focus_centered)
return;
keep_focus_centered_ = keep_focus_centered;
if (type_ == ui::MAGNIFIER_FULL) {
ash::Shell::GetInstance()
->magnification_controller()
->SetKeepFocusCentered(keep_focus_centered_);
}
}
void UpdateMagnifierFromPrefs() {
if (!profile_)
return;
const bool enabled = profile_->GetPrefs()->GetBoolean(
prefs::kAccessibilityScreenMagnifierEnabled);
const int type_integer = profile_->GetPrefs()->GetInteger(
prefs::kAccessibilityScreenMagnifierType);
const bool keep_focus_centered = profile_->GetPrefs()->GetBoolean(
prefs::kAccessibilityScreenMagnifierCenterFocus);
ui::MagnifierType type = ui::kDefaultMagnifierType;
if (type_integer > 0 && type_integer <= ui::kMaxMagnifierType) {
type = static_cast<ui::MagnifierType>(type_integer);
} else if (type_integer == 0) {
// Type 0 is used to disable the screen magnifier through policy. As the
// magnifier type is irrelevant in this case, it is OK to just fall back
// to the default.
} else {
NOTREACHED();
}
if (!enabled) {
SetMagnifierEnabledInternal(enabled);
SetMagnifierTypeInternal(type);
SetMagniferKeepFocusCenteredInternal(keep_focus_centered);
} else {
SetMagniferKeepFocusCenteredInternal(keep_focus_centered);
SetMagnifierTypeInternal(type);
SetMagnifierEnabledInternal(enabled);
}
AccessibilityStatusEventDetails details(
ACCESSIBILITY_TOGGLE_SCREEN_MAGNIFIER,
enabled_,
type_,
ui::A11Y_NOTIFICATION_NONE);
if (AccessibilityManager::Get()) {
AccessibilityManager::Get()->NotifyAccessibilityStatusChanged(details);
if (ash::Shell::GetInstance()) {
ash::Shell::GetInstance()->SetCursorCompositingEnabled(
AccessibilityManager::Get()->ShouldEnableCursorCompositing());
}
}
}
void MonitorFocusInPageChange() {
if (enabled_ && !observing_focus_change_in_page_) {
registrar_.Add(this, content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE,
content::NotificationService::AllSources());
observing_focus_change_in_page_ = true;
} else if (!enabled_ && observing_focus_change_in_page_) {
registrar_.Remove(this, content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE,
content::NotificationService::AllSources());
observing_focus_change_in_page_ = false;
}
}
// content::NotificationObserver implementation:
void Observe(int type,
const content::NotificationSource& source,
const content::NotificationDetails& details) override {
switch (type) {
case chrome::NOTIFICATION_LOGIN_OR_LOCK_WEBUI_VISIBLE: {
// Update |profile_| when entering the login screen.
Profile* profile = ProfileManager::GetActiveUserProfile();
if (ProfileHelper::IsSigninProfile(profile))
SetProfile(profile);
break;
}
case chrome::NOTIFICATION_SESSION_STARTED:
// Update |profile_| when entering a session.
SetProfile(ProfileManager::GetActiveUserProfile());
// Add a session state observer to be able to monitor session changes.
if (!session_state_observer_.get() && ash::Shell::HasInstance())
session_state_observer_.reset(
new ash::ScopedSessionStateObserver(this));
break;
case chrome::NOTIFICATION_PROFILE_DESTROYED: {
// Update |profile_| when exiting a session or shutting down.
Profile* profile = content::Source<Profile>(source).ptr();
if (profile_ == profile)
SetProfile(NULL);
break;
}
case content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE: {
content::FocusedNodeDetails* node_details =
content::Details<content::FocusedNodeDetails>(details).ptr();
ash::Shell::GetInstance()
->magnification_controller()
->HandleFocusedNodeChanged(node_details->is_editable_node,
node_details->node_bounds_in_screen);
break;
}
}
}
Profile* profile_;
AccessibilityManager::PrefHandler magnifier_enabled_pref_handler_;
AccessibilityManager::PrefHandler magnifier_type_pref_handler_;
AccessibilityManager::PrefHandler magnifier_scale_pref_handler_;
ui::MagnifierType type_;
bool enabled_;
bool keep_focus_centered_;
bool observing_focus_change_in_page_;
content::NotificationRegistrar registrar_;
scoped_ptr<PrefChangeRegistrar> pref_change_registrar_;
scoped_ptr<ash::ScopedSessionStateObserver> session_state_observer_;
DISALLOW_COPY_AND_ASSIGN(MagnificationManagerImpl);
};
// static
void MagnificationManager::Initialize() {
CHECK(g_magnification_manager == NULL);
g_magnification_manager = new MagnificationManagerImpl();
}
// static
void MagnificationManager::Shutdown() {
CHECK(g_magnification_manager);
delete g_magnification_manager;
g_magnification_manager = NULL;
}
// static
MagnificationManager* MagnificationManager::Get() {
return g_magnification_manager;
}
} // namespace chromeos