blob: c83db87b134cdc2a3f335f6caae00cf525172064 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
// 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 "ash/focus_cycler.h"
#include "ash/shell.h"
#include "ash/wm/mru_window_tracker.h"
#include "ash/wm/window_cycle_controller.h"
#include "ash/wm/window_util.h"
#include "ui/aura/client/activation_client.h"
#include "ui/aura/window.h"
#include "ui/views/accessible_pane_view.h"
#include "ui/views/focus/focus_search.h"
#include "ui/views/widget/widget.h"
namespace ash {
namespace {
bool HasFocusableWindow() {
return !MruWindowTracker::BuildWindowList(false).empty();
}
} // namespace
namespace internal {
FocusCycler::FocusCycler() : widget_activating_(NULL) {
}
FocusCycler::~FocusCycler() {
}
void FocusCycler::AddWidget(views::Widget* widget) {
widgets_.push_back(widget);
}
void FocusCycler::RotateFocus(Direction direction) {
aura::Window* window = ash::wm::GetActiveWindow();
if (window) {
views::Widget* widget = views::Widget::GetWidgetForNativeWindow(window);
// First try to rotate focus within the active widget. If that succeeds,
// we're done.
if (widget && widget->GetFocusManager()->RotatePaneFocus(
direction == BACKWARD ?
views::FocusManager::kBackward : views::FocusManager::kForward,
views::FocusManager::kNoWrap)) {
return;
}
}
const bool has_window = HasFocusableWindow();
int index = 0;
int count = static_cast<int>(widgets_.size());
int browser_index = has_window ? count : -1;
for (; index < count; ++index) {
if (widgets_[index]->IsActive())
break;
}
int start_index = index;
if (has_window)
++count;
for (;;) {
if (direction == FORWARD)
index = (index + 1) % count;
else
index = ((index - 1) + count) % count;
// Ensure that we don't loop more than once.
if (index == start_index)
break;
if (index == browser_index) {
// Activate the most recently active browser window.
ash::Shell::GetInstance()->window_cycle_controller()->HandleCycleWindow(
WindowCycleController::FORWARD, false);
// Rotate pane focus within that window.
aura::Window* window = ash::wm::GetActiveWindow();
if (!window)
break;
views::Widget* widget = views::Widget::GetWidgetForNativeWindow(window);
if (!widget)
break;
views::FocusManager* focus_manager = widget->GetFocusManager();
focus_manager->ClearFocus();
focus_manager->RotatePaneFocus(
direction == BACKWARD ?
views::FocusManager::kBackward : views::FocusManager::kForward,
views::FocusManager::kWrap);
break;
} else {
if (FocusWidget(widgets_[index]))
break;
}
}
}
bool FocusCycler::FocusWidget(views::Widget* widget) {
// Note: It is not necessary to set the focus directly to the pane since that
// will be taken care of by the widget activation.
widget_activating_ = widget;
widget->Activate();
widget_activating_ = NULL;
return widget->IsActive();
}
} // namespace internal
} // namespace ash
|