| // 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_impl.h" |
| #include "ash/accessibility/accessibility_focus_ring_controller_impl.h" |
| #include "ash/accessibility/touch_exploration_controller.h" |
| #include "ash/keyboard/ui/keyboard_ui_controller.h" |
| #include "ash/public/cpp/accessibility_focus_ring_info.h" |
| #include "ash/public/cpp/app_types.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/wm/public/activation_client.h" |
| |
| namespace ash { |
| |
| namespace { |
| |
| AccessibilityControllerImpl* 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::KeyboardUIController::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::KeyboardUIController::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. |
| if (AccessibilityControllerImpl* controller = GetA11yController()) |
| controller->OnTwoFingerTouchStop(); |
| } |
| |
| void TouchExplorationManager::PlaySpokenFeedbackToggleCountdown( |
| int tick_count) { |
| if (GetA11yController()->ShouldToggleSpokenFeedbackViaTouch()) |
| GetA11yController()->PlaySpokenFeedbackToggleCountdown(tick_count); |
| } |
| |
| void TouchExplorationManager::PlayTouchTypeEarcon() { |
| GetA11yController()->PlayEarcon(chromeos::SOUND_TOUCH_TYPE); |
| } |
| |
| void TouchExplorationManager::ToggleSpokenFeedback() { |
| if (GetA11yController()->ShouldToggleSpokenFeedbackViaTouch()) { |
| 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 = |
| window_util::GetActiveWindow() && |
| window_util::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( |
| extension_misc::kChromeVoxExtensionId, |
| std::make_unique<AccessibilityFocusRingInfo>()); |
| } else { |
| touch_exploration_controller_->SetExcludeBounds(gfx::Rect()); |
| } |
| |
| // Virtual keyboard. |
| auto* keyboard_controller = keyboard::KeyboardUIController::Get(); |
| if (keyboard_controller->IsEnabled()) { |
| touch_exploration_controller_->SetLiftActivationBounds( |
| keyboard_controller->GetVisualBoundsInScreen()); |
| } |
| } else { |
| touch_exploration_controller_.reset(); |
| } |
| } |
| |
| bool TouchExplorationManager::VolumeAdjustSoundEnabled() { |
| return !base::CommandLine::ForCurrentProcess()->HasSwitch( |
| chromeos::switches::kDisableVolumeAdjustSound); |
| } |
| |
| } // namespace ash |