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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
|
// Copyright 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.
#ifndef CHROME_BROWSER_UI_VIEWS_FRAME_IMMERSIVE_MODE_CONTROLLER_ASH_H_
#define CHROME_BROWSER_UI_VIEWS_FRAME_IMMERSIVE_MODE_CONTROLLER_ASH_H_
#include "chrome/browser/ui/views/frame/immersive_mode_controller.h"
#include "base/timer.h"
#include "content/public/browser/notification_observer.h"
#include "content/public/browser/notification_registrar.h"
#include "ui/aura/window_observer.h"
#include "ui/base/animation/animation_delegate.h"
#include "ui/base/events/event_handler.h"
#include "ui/gfx/rect.h"
#include "ui/views/focus/focus_manager.h"
#include "ui/views/widget/widget_observer.h"
class BrowserView;
class BookmarkBarView;
namespace aura {
class Window;
}
namespace gfx {
class Transform;
}
namespace ui {
class Layer;
class LocatedEvent;
class SlideAnimation;
}
namespace views {
class View;
}
class ImmersiveModeControllerAsh : public ImmersiveModeController,
public content::NotificationObserver,
public ui::AnimationDelegate,
public ui::EventHandler,
public views::FocusChangeListener,
public views::WidgetObserver,
public aura::WindowObserver {
public:
ImmersiveModeControllerAsh();
virtual ~ImmersiveModeControllerAsh();
// These methods are used to increment and decrement |revealed_lock_count_|.
// If immersive mode is enabled, a transition from 1 to 0 in
// |revealed_lock_count_| closes the top-of-window views and a transition
// from 0 to 1 in |revealed_lock_count_| reveals the top-of-window views.
void LockRevealedState(AnimateReveal animate_reveal);
void UnlockRevealedState();
// Exits immersive fullscreen based on |native_window_|'s show state.
void MaybeExitImmersiveFullscreen();
// ImmersiveModeController overrides:
virtual void Init(Delegate* delegate,
views::Widget* widget,
views::View* top_container) OVERRIDE;
virtual void SetEnabled(bool enabled) OVERRIDE;
virtual bool IsEnabled() const OVERRIDE;
virtual bool ShouldHideTabIndicators() const OVERRIDE;
virtual bool ShouldHideTopViews() const OVERRIDE;
virtual bool IsRevealed() const OVERRIDE;
virtual int GetTopContainerVerticalOffset(
const gfx::Size& top_container_size) const OVERRIDE;
virtual ImmersiveRevealedLock* GetRevealedLock(
AnimateReveal animate_reveal) OVERRIDE WARN_UNUSED_RESULT;
virtual void AnchorWidgetToTopContainer(views::Widget* widget,
int y_offset) OVERRIDE;
virtual void UnanchorWidgetFromTopContainer(views::Widget* widget) OVERRIDE;
virtual void OnTopContainerBoundsChanged() OVERRIDE;
virtual void OnFindBarVisibleBoundsChanged(
const gfx::Rect& new_visible_bounds_in_screen) OVERRIDE;
// content::NotificationObserver override:
virtual void Observe(int type,
const content::NotificationSource& source,
const content::NotificationDetails& details) OVERRIDE;
// ui::EventHandler overrides:
virtual void OnMouseEvent(ui::MouseEvent* event) OVERRIDE;
virtual void OnTouchEvent(ui::TouchEvent* event) OVERRIDE;
virtual void OnGestureEvent(ui::GestureEvent* event) OVERRIDE;
// views::FocusChangeObserver overrides:
virtual void OnWillChangeFocus(views::View* focused_before,
views::View* focused_now) OVERRIDE;
virtual void OnDidChangeFocus(views::View* focused_before,
views::View* focused_now) OVERRIDE;
// views::WidgetObserver overrides:
virtual void OnWidgetDestroying(views::Widget* widget) OVERRIDE;
virtual void OnWidgetActivationChanged(views::Widget* widget,
bool active) OVERRIDE;
// ui::AnimationDelegate overrides:
virtual void AnimationEnded(const ui::Animation* animation) OVERRIDE;
virtual void AnimationProgressed(const ui::Animation* animation) OVERRIDE;
// aura::WindowObserver overrides:
virtual void OnWindowPropertyChanged(aura::Window* window,
const void* key,
intptr_t old) OVERRIDE;
virtual void OnWindowAddedToRootWindow(aura::Window* window) OVERRIDE;
virtual void OnWindowRemovingFromRootWindow(aura::Window* window) OVERRIDE;
// Testing interface.
void SetForceHideTabIndicatorsForTest(bool force);
void StartRevealForTest(bool hovered);
void SetMouseHoveredForTest(bool hovered);
void DisableAnimationsForTest();
private:
friend class ImmersiveModeControllerAshTest;
enum Animate {
ANIMATE_NO,
ANIMATE_SLOW,
ANIMATE_FAST,
};
enum Layout {
LAYOUT_YES,
LAYOUT_NO
};
enum RevealState {
CLOSED, // Top container only showing tabstrip, y = 0.
SLIDING_OPEN, // All views showing, y animating from -height to 0.
REVEALED, // All views showing, y = 0.
SLIDING_CLOSED, // All views showing, y animating from 0 to -height.
};
enum TabIndicatorVisibility {
TAB_INDICATORS_FORCE_HIDE,
TAB_INDICATORS_HIDE,
TAB_INDICATORS_SHOW
};
enum SwipeType {
SWIPE_OPEN,
SWIPE_CLOSE,
SWIPE_NONE
};
// Enables or disables observers for mouse move, focus, and window restore.
void EnableWindowObservers(bool enable);
// Updates |top_edge_hover_timer_| based on a mouse |event|. If the mouse is
// hovered at the top of the screen the timer is started. If the mouse moves
// away from the top edge, or moves too much in the x direction, the timer is
// stopped.
void UpdateTopEdgeHoverTimer(ui::MouseEvent* event);
// Updates |located_event_revealed_lock_| based on the current mouse state and
// the current touch state.
// |event| is NULL if the source event is not known.
void UpdateLocatedEventRevealedLock(ui::LocatedEvent* event);
// Acquires |located_event_revealed_lock_| if it is not already held.
void AcquireLocatedEventRevealedLock();
// Updates |focus_revealed_lock_| based on the currently active view and the
// currently active widget.
void UpdateFocusRevealedLock();
// Update |located_event_revealed_lock_| and |focus_revealed_lock_| as a
// result of a gesture of |swipe_type|. Returns true if any locks were
// acquired or released.
bool UpdateRevealedLocksForSwipe(SwipeType swipe_type);
// Updates whether fullscreen uses any chrome at all. When using minimal
// chrome, a 'light bar' is permanently visible for the launcher and possibly
// for the tabstrip.
void UpdateUseMinimalChrome(Layout layout);
// Returns the animation duration given |animate|.
int GetAnimationDuration(Animate animate) const;
// Temporarily reveals the top-of-window views while in immersive mode,
// hiding them when the cursor exits the area of the top views. If |animate|
// is not ANIMATE_NO, slides in the view, otherwise shows it immediately.
void MaybeStartReveal(Animate animate);
// Enables or disables layer-based painting to allow smooth animations.
void EnablePaintToLayer(bool enable);
// Updates the browser root view's layout including window caption controls.
void LayoutBrowserRootView();
// Called when the animation to slide open the top-of-window views has
// completed.
void OnSlideOpenAnimationCompleted(Layout layout);
// Hides the top-of-window views if immersive mode is enabled and nothing is
// keeping them revealed. Optionally animates.
void MaybeEndReveal(Animate animate);
// Called when the animation to slide out the top-of-window views has
// completed.
void OnSlideClosedAnimationCompleted();
// Returns whether immersive fullscreen should be exited based on
// |native_window_|'s show state. This handles cases where the user has
// exited immersive fullscreen without going through
// FullscreenController::ToggleFullscreenMode(). This is the case if the
// user exits fullscreen via the restore button.
bool ShouldExitImmersiveFullscreen() const;
// Returns the type of swipe given |event|.
SwipeType GetSwipeType(ui::GestureEvent* event) const;
// True when |location| is "near" to the top container. When the top container
// is not closed "near" means within the displayed bounds or above it. When
// the top container is closed "near" means either within the displayed
// bounds, above it, or within a few pixels below it. This allow the container
// to steal enough pixels to detect a swipe in and handles the case that there
// is a bezel sensor above the top container.
bool ShouldHandleEvent(const gfx::Point& location) const;
// Call Add/RemovePreTargerHandler since either the RootWindow has changed or
// the enabled state of observing has changed.
void UpdatePreTargetHandler();
// Injected dependencies. Not owned.
Delegate* delegate_;
views::Widget* widget_;
views::View* top_container_;
// True if the window observers are enabled.
bool observers_enabled_;
// True when in immersive mode.
bool enabled_;
// State machine for the revealed/closed animations.
RevealState reveal_state_;
int revealed_lock_count_;
// The visibility of the miniature "tab indicators" in the main browser view
// when immersive mode is enabled and the top-of-window views are closed.
TabIndicatorVisibility tab_indicator_visibility_;
// Timer to track cursor being held at the top edge of the screen.
base::OneShotTimer<ImmersiveModeController> top_edge_hover_timer_;
// The cursor x position in root coordinates when the cursor first hit
// the top edge of the screen.
int mouse_x_when_hit_top_;
// The current visible bounds of the find bar, in screen coordinates. This is
// an empty rect if the find bar is not visible.
gfx::Rect find_bar_visible_bounds_in_screen_;
// Lock which keeps the top-of-window views revealed based on the current
// mouse state and the current touch state. Acquiring the lock is used to
// trigger a reveal when the user moves the mouse to the top of the screen
// and when the user does a SWIPE_OPEN edge gesture.
scoped_ptr<ImmersiveRevealedLock> located_event_revealed_lock_;
// Lock which keeps the top-of-window views revealed based on the focused view
// and the active widget. Acquiring the lock never triggers a reveal because
// a view is not focusable till a reveal has made it visible.
scoped_ptr<ImmersiveRevealedLock> focus_revealed_lock_;
// Native window for the browser.
aura::Window* native_window_;
// The animation which controls sliding the top-of-window views in and out.
scoped_ptr<ui::SlideAnimation> animation_;
// Whether the animations are disabled for testing.
bool animations_disabled_for_test_;
// Manages widgets which are anchored to the top-of-window views.
class AnchoredWidgetManager;
scoped_ptr<AnchoredWidgetManager> anchored_widget_manager_;
content::NotificationRegistrar registrar_;
base::WeakPtrFactory<ImmersiveModeControllerAsh> weak_ptr_factory_;
// Tracks if the controller has seen a ET_GESTURE_SCROLL_BEGIN, without the
// following events.
bool gesture_begun_;
DISALLOW_COPY_AND_ASSIGN(ImmersiveModeControllerAsh);
};
#endif // CHROME_BROWSER_UI_VIEWS_FRAME_IMMERSIVE_MODE_CONTROLLER_ASH_H_
|