blob: 4c734350bf68fbc0f50faa05d4912e37e037abf9 [file] [log] [blame]
// Copyright 2014 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 "ash/accessibility/touch_exploration_manager.h"
#include <memory>
#include <vector>
#include "ash/accessibility/accessibility_controller.h"
#include "ash/accessibility/accessibility_focus_ring_controller.h"
#include "ash/accessibility/touch_exploration_controller.h"
#include "ash/public/cpp/app_types.h"
#include "ash/public/interfaces/accessibility_focus_ring_controller.mojom.h"
#include "ash/root_window_controller.h"
#include "ash/shell.h"
#include "ash/wm/window_util.h"
#include "base/command_line.h"
#include "chromeos/audio/chromeos_sounds.h"
#include "chromeos/audio/cras_audio_handler.h"
#include "chromeos/constants/chromeos_switches.h"
#include "extensions/common/constants.h"
#include "ui/aura/client/aura_constants.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/keyboard/keyboard_controller.h"
#include "ui/wm/public/activation_client.h"
namespace ash {
namespace {
AccessibilityController* GetA11yController() {
return Shell::Get()->accessibility_controller();
}
} // namespace
TouchExplorationManager::TouchExplorationManager(
RootWindowController* root_window_controller)
: root_window_controller_(root_window_controller),
audio_handler_(chromeos::CrasAudioHandler::Get()),
observing_window_(nullptr) {
Shell::Get()->AddShellObserver(this);
Shell::Get()->accessibility_controller()->AddObserver(this);
Shell::Get()->activation_client()->AddObserver(this);
keyboard::KeyboardController::Get()->AddObserver(this);
display::Screen::GetScreen()->AddObserver(this);
UpdateTouchExplorationState();
}
TouchExplorationManager::~TouchExplorationManager() {
// TODO(jamescook): Clean up shutdown order so this check isn't needed.
if (Shell::Get()->accessibility_controller())
Shell::Get()->accessibility_controller()->RemoveObserver(this);
Shell::Get()->activation_client()->RemoveObserver(this);
keyboard::KeyboardController::Get()->RemoveObserver(this);
display::Screen::GetScreen()->RemoveObserver(this);
Shell::Get()->RemoveShellObserver(this);
if (observing_window_)
observing_window_->RemoveObserver(this);
}
void TouchExplorationManager::OnAccessibilityStatusChanged() {
UpdateTouchExplorationState();
}
void TouchExplorationManager::OnWindowPropertyChanged(aura::Window* winodw,
const void* key,
intptr_t old) {
if (key != aura::client::kAccessibilityTouchExplorationPassThrough)
return;
UpdateTouchExplorationState();
}
void TouchExplorationManager::SetOutputLevel(int volume) {
if (volume > 0) {
if (audio_handler_->IsOutputMuted()) {
audio_handler_->SetOutputMute(false);
}
}
audio_handler_->SetOutputVolumePercent(volume);
// Avoid negative volume.
if (audio_handler_->IsOutputVolumeBelowDefaultMuteLevel())
audio_handler_->SetOutputMute(true);
}
void TouchExplorationManager::SilenceSpokenFeedback() {
if (GetA11yController()->spoken_feedback_enabled())
GetA11yController()->SilenceSpokenFeedback();
}
void TouchExplorationManager::PlayVolumeAdjustEarcon() {
if (!VolumeAdjustSoundEnabled())
return;
if (!audio_handler_->IsOutputMuted() &&
audio_handler_->GetOutputVolumePercent() != 100) {
GetA11yController()->PlayEarcon(chromeos::SOUND_VOLUME_ADJUST);
}
}
void TouchExplorationManager::PlayPassthroughEarcon() {
GetA11yController()->PlayEarcon(chromeos::SOUND_PASSTHROUGH);
}
void TouchExplorationManager::PlayExitScreenEarcon() {
GetA11yController()->PlayEarcon(chromeos::SOUND_EXIT_SCREEN);
}
void TouchExplorationManager::PlayEnterScreenEarcon() {
GetA11yController()->PlayEarcon(chromeos::SOUND_ENTER_SCREEN);
}
void TouchExplorationManager::HandleAccessibilityGesture(
ax::mojom::Gesture gesture) {
GetA11yController()->HandleAccessibilityGesture(gesture);
}
void TouchExplorationManager::OnDisplayMetricsChanged(
const display::Display& display,
uint32_t changed_metrics) {
const display::Display this_display =
display::Screen::GetScreen()->GetDisplayNearestWindow(
root_window_controller_->GetRootWindow());
if (this_display.id() == display.id())
UpdateTouchExplorationState();
}
void TouchExplorationManager::OnTwoFingerTouchStart() {
GetA11yController()->OnTwoFingerTouchStart();
}
void TouchExplorationManager::OnTwoFingerTouchStop() {
// Can be null during shutdown.
AccessibilityController* controller = GetA11yController();
if (controller)
controller->OnTwoFingerTouchStop();
}
void TouchExplorationManager::PlaySpokenFeedbackToggleCountdown(
int tick_count) {
GetA11yController()->ShouldToggleSpokenFeedbackViaTouch(base::BindOnce(
[](int tick_count, bool should_toggle) {
if (!should_toggle)
return;
GetA11yController()->PlaySpokenFeedbackToggleCountdown(tick_count);
},
tick_count));
}
void TouchExplorationManager::PlayTouchTypeEarcon() {
GetA11yController()->PlayEarcon(chromeos::SOUND_TOUCH_TYPE);
}
void TouchExplorationManager::ToggleSpokenFeedback() {
GetA11yController()->ShouldToggleSpokenFeedbackViaTouch(
base::BindOnce([](bool should_toggle) {
if (!should_toggle)
return;
GetA11yController()->SetSpokenFeedbackEnabled(
!GetA11yController()->spoken_feedback_enabled(),
A11Y_NOTIFICATION_SHOW);
}));
}
void TouchExplorationManager::OnWindowActivated(
::wm::ActivationChangeObserver::ActivationReason reason,
aura::Window* gained_active,
aura::Window* lost_active) {
if (lost_active && lost_active->HasObserver(this)) {
lost_active->RemoveObserver(this);
observing_window_ = nullptr;
}
if (gained_active && !gained_active->HasObserver(this)) {
gained_active->AddObserver(this);
observing_window_ = gained_active;
}
UpdateTouchExplorationState();
}
void TouchExplorationManager::SetTouchAccessibilityAnchorPoint(
const gfx::Point& anchor_point) {
if (touch_exploration_controller_) {
touch_exploration_controller_->SetTouchAccessibilityAnchorPoint(
anchor_point);
}
}
void TouchExplorationManager::OnKeyboardVisibleBoundsChanged(
const gfx::Rect& new_bounds) {
UpdateTouchExplorationState();
}
void TouchExplorationManager::OnKeyboardEnabledChanged(bool is_enabled) {
UpdateTouchExplorationState();
}
void TouchExplorationManager::UpdateTouchExplorationState() {
// See crbug.com/603745 for more details.
const bool pass_through_surface =
wm::GetActiveWindow() &&
wm::GetActiveWindow()->GetProperty(
aura::client::kAccessibilityTouchExplorationPassThrough);
const bool spoken_feedback_enabled =
GetA11yController()->spoken_feedback_enabled();
if (!touch_accessibility_enabler_) {
// Always enable gesture to toggle spoken feedback.
touch_accessibility_enabler_.reset(new TouchAccessibilityEnabler(
root_window_controller_->GetRootWindow(), this));
}
if (spoken_feedback_enabled) {
if (!touch_exploration_controller_.get()) {
touch_exploration_controller_ =
std::make_unique<TouchExplorationController>(
root_window_controller_->GetRootWindow(), this,
touch_accessibility_enabler_->GetWeakPtr());
}
if (pass_through_surface) {
const display::Display display =
display::Screen::GetScreen()->GetDisplayNearestWindow(
root_window_controller_->GetRootWindow());
const gfx::Rect work_area = display.work_area();
touch_exploration_controller_->SetExcludeBounds(work_area);
SilenceSpokenFeedback();
// Clear the focus highlight.
Shell::Get()->accessibility_focus_ring_controller()->SetFocusRing(
std::vector<gfx::Rect>(),
mojom::FocusRingBehavior::PERSIST_FOCUS_RING,
extension_misc::kChromeVoxExtensionId);
} else {
touch_exploration_controller_->SetExcludeBounds(gfx::Rect());
}
// Virtual keyboard.
auto* keyboard_controller = keyboard::KeyboardController::Get();
if (keyboard_controller->IsEnabled()) {
touch_exploration_controller_->SetLiftActivationBounds(
keyboard_controller->visual_bounds_in_screen());
}
} else {
touch_exploration_controller_.reset();
}
}
bool TouchExplorationManager::VolumeAdjustSoundEnabled() {
return !base::CommandLine::ForCurrentProcess()->HasSwitch(
chromeos::switches::kDisableVolumeAdjustSound);
}
} // namespace ash