|  | // Copyright 2016 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef ASH_SHELF_SHELF_H_ | 
|  | #define ASH_SHELF_SHELF_H_ | 
|  |  | 
|  | #include <memory> | 
|  |  | 
|  | #include "ash/ash_export.h" | 
|  | #include "ash/public/cpp/metrics_util.h" | 
|  | #include "ash/public/cpp/shelf_types.h" | 
|  | #include "ash/shelf/desk_button_widget.h" | 
|  | #include "ash/shelf/shelf_layout_manager_observer.h" | 
|  | #include "ash/shelf/shelf_locking_manager.h" | 
|  | #include "base/memory/raw_ptr.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "base/observer_list.h" | 
|  |  | 
|  | namespace aura { | 
|  | class Window; | 
|  | } | 
|  |  | 
|  | namespace gfx { | 
|  | class Rect; | 
|  | } | 
|  |  | 
|  | namespace ui { | 
|  | class GestureEvent; | 
|  | class MouseWheelEvent; | 
|  | class MouseEvent; | 
|  | class ScrollEvent; | 
|  | }  // namespace ui | 
|  |  | 
|  | namespace ash { | 
|  |  | 
|  | enum class AnimationChangeType; | 
|  | class HotseatWidget; | 
|  | class HotseatWidgetAnimationMetricsReporter; | 
|  | class NavigationWidgetAnimationMetricsReporter; | 
|  | class ShelfFocusCycler; | 
|  | class LoginShelfWidget; | 
|  | class ShelfLayoutManager; | 
|  | class ShelfLayoutManagerTest; | 
|  | class ShelfLockingManager; | 
|  | class ShelfNavigationWidget; | 
|  | class ShelfView; | 
|  | class ShelfWidget; | 
|  | class StatusAreaWidget; | 
|  | class ShelfObserver; | 
|  | class WorkAreaInsets; | 
|  | class ShelfTooltipManager; | 
|  |  | 
|  | // TODO(oshima) : move to .cc | 
|  |  | 
|  | // Returns a value based on shelf alignment. | 
|  | template <typename T> | 
|  | T SelectValueByShelfAlignment(ShelfAlignment alignment, | 
|  | T bottom, | 
|  | T left, | 
|  | T right) { | 
|  | switch (alignment) { | 
|  | case ShelfAlignment::kBottom: | 
|  | case ShelfAlignment::kBottomLocked: | 
|  | return bottom; | 
|  | case ShelfAlignment::kLeft: | 
|  | return left; | 
|  | case ShelfAlignment::kRight: | 
|  | return right; | 
|  | } | 
|  | NOTREACHED(); | 
|  | } | 
|  |  | 
|  | bool IsHorizontalAlignment(ShelfAlignment alignment); | 
|  |  | 
|  | // Returns |horizontal| if shelf is horizontal, otherwise |vertical|. | 
|  | template <typename T> | 
|  | T PrimaryAxisValueByShelfAlignment(ShelfAlignment alignment, | 
|  | T horizontal, | 
|  | T vertical) { | 
|  | return IsHorizontalAlignment(alignment) ? horizontal : vertical; | 
|  | } | 
|  |  | 
|  | // Controller for the shelf state. One per display, because each display might | 
|  | // have different shelf alignment, autohide, etc. Exists for the lifetime of the | 
|  | // root window controller. | 
|  | class ASH_EXPORT Shelf : public ShelfLayoutManagerObserver { | 
|  | public: | 
|  | // Used to maintain a lock for the auto-hide shelf. If lock, then we should | 
|  | // not update the state of the auto-hide shelf. | 
|  | class ScopedAutoHideLock { | 
|  | public: | 
|  | explicit ScopedAutoHideLock(Shelf* shelf) : shelf_(shelf) { | 
|  | ++shelf_->auto_hide_lock_; | 
|  | } | 
|  |  | 
|  | ScopedAutoHideLock(const ScopedAutoHideLock&) = delete; | 
|  | ScopedAutoHideLock& operator=(const ScopedAutoHideLock&) = delete; | 
|  |  | 
|  | ~ScopedAutoHideLock() { | 
|  | --shelf_->auto_hide_lock_; | 
|  | DCHECK_GE(shelf_->auto_hide_lock_, 0); | 
|  | } | 
|  |  | 
|  | private: | 
|  | raw_ptr<Shelf> shelf_; | 
|  | }; | 
|  |  | 
|  | // Used to disable auto-hide shelf behavior while in scope. Note that | 
|  | // disabling auto-hide behavior is of lower precedence than auto-hide behavior | 
|  | // based on locks and session state, so it is not guaranteed to show the shelf | 
|  | // in all cases. | 
|  | class ScopedDisableAutoHide { | 
|  | public: | 
|  | explicit ScopedDisableAutoHide(Shelf* shelf); | 
|  | ScopedDisableAutoHide(const ScopedDisableAutoHide&) = delete; | 
|  | ScopedDisableAutoHide& operator=(const ScopedDisableAutoHide&) = delete; | 
|  | ~ScopedDisableAutoHide(); | 
|  |  | 
|  | Shelf* weak_shelf() { return shelf_.get(); } | 
|  |  | 
|  | private: | 
|  | // Save a `base::WeakPtr` to avoid a crash if `shelf_` is deallocated due to | 
|  | // monitor disconnect. | 
|  | base::WeakPtr<Shelf> const shelf_; | 
|  | }; | 
|  |  | 
|  | Shelf(); | 
|  |  | 
|  | Shelf(const Shelf&) = delete; | 
|  | Shelf& operator=(const Shelf&) = delete; | 
|  |  | 
|  | ~Shelf() override; | 
|  |  | 
|  | // Returns the shelf for the display that |window| is on. Note that the shelf | 
|  | // widget may not exist, or the shelf may not be visible. | 
|  | static Shelf* ForWindow(aura::Window* window); | 
|  |  | 
|  | // Launch a 0-indexed shelf item in the shelf. A negative index launches the | 
|  | // last shelf item in the shelf. | 
|  | static void LaunchShelfItem(int item_index); | 
|  |  | 
|  | // Activates the shelf item specified by the index in the list of shelf items. | 
|  | static void ActivateShelfItem(int item_index); | 
|  |  | 
|  | // Activates the shelf item specified by the index in the list of shelf items | 
|  | // on the display identified by |display_id|. | 
|  | static void ActivateShelfItemOnDisplay(int item_index, int64_t display_id); | 
|  |  | 
|  | // Updates the shelf visibility on all displays. This method exists for | 
|  | // historical reasons. If a display or shelf instance is available, prefer | 
|  | // Shelf::UpdateVisibilityState() below. | 
|  | static void UpdateShelfVisibility(); | 
|  |  | 
|  | void CreateNavigationWidget(aura::Window* container); | 
|  | void CreateDeskButtonWidget(aura::Window* container); | 
|  | void CreateHotseatWidget(aura::Window* container); | 
|  | void CreateStatusAreaWidget(aura::Window* status_container); | 
|  | void CreateShelfWidget(aura::Window* root); | 
|  |  | 
|  | // Begins shutdown of the ShelfWidget and all child widgets. | 
|  | void ShutdownShelfWidget(); | 
|  |  | 
|  | // Resets `shelf_widget_`. | 
|  | void DestroyShelfWidget(); | 
|  |  | 
|  | // Returns true if the shelf is visible. Shelf can be visible in 1) | 
|  | // SHELF_VISIBLE or 2) SHELF_AUTO_HIDE but in SHELF_AUTO_HIDE_SHOWN. See | 
|  | // details in ShelfLayoutManager::IsVisible. | 
|  | bool IsVisible() const; | 
|  |  | 
|  | // Returns the window showing the shelf. | 
|  | const aura::Window* GetWindow() const; | 
|  | aura::Window* GetWindow(); | 
|  |  | 
|  | void SetAlignment(ShelfAlignment alignment); | 
|  |  | 
|  | // Returns true if the shelf alignment is horizontal (i.e. at the bottom). | 
|  | bool IsHorizontalAlignment() const; | 
|  |  | 
|  | // Returns a value based on shelf alignment. | 
|  | template <typename T> | 
|  | T SelectValueForShelfAlignment(T bottom, T left, T right) const { | 
|  | return SelectValueByShelfAlignment(alignment_, bottom, left, right); | 
|  | } | 
|  |  | 
|  | // Returns |horizontal| if shelf is horizontal, otherwise |vertical|. | 
|  | template <typename T> | 
|  | T PrimaryAxisValue(T horizontal, T vertical) const { | 
|  | return IsHorizontalAlignment() ? horizontal : vertical; | 
|  | } | 
|  |  | 
|  | void SetAutoHideBehavior(ShelfAutoHideBehavior behavior); | 
|  |  | 
|  | ShelfAutoHideState GetAutoHideState() const; | 
|  |  | 
|  | // Invoke when the auto-hide state may have changed (for example, when the | 
|  | // system tray bubble opens it should force the shelf to be visible). | 
|  | void UpdateAutoHideState(); | 
|  |  | 
|  | ShelfBackgroundType GetBackgroundType() const; | 
|  |  | 
|  | void UpdateVisibilityState(); | 
|  |  | 
|  | void MaybeUpdateShelfBackground(); | 
|  |  | 
|  | ShelfVisibilityState GetVisibilityState() const; | 
|  |  | 
|  | gfx::Rect GetShelfBoundsInScreen() const; | 
|  |  | 
|  | // Returns the ideal bounds of the shelf assuming it is visible. | 
|  | gfx::Rect GetIdealBounds() const; | 
|  |  | 
|  | // Returns the ideal bounds of the shelf, but in tablet mode always returns | 
|  | // the bounds of the in-app shelf. | 
|  | gfx::Rect GetIdealBoundsForWorkAreaCalculation(); | 
|  |  | 
|  | // Returns the screen bounds of the item for the specified window. If there is | 
|  | // no item for the specified window an empty rect is returned. | 
|  | gfx::Rect GetScreenBoundsOfItemIconForWindow(aura::Window* window); | 
|  |  | 
|  | // Handles a gesture |event| coming from a source outside the shelf widget | 
|  | // (e.g. the status area widget). Allows support for behaviors like toggling | 
|  | // auto-hide with a swipe, even if that gesture event hits another window. | 
|  | // Returns true if the event was handled. | 
|  | bool ProcessGestureEvent(const ui::GestureEvent& event); | 
|  |  | 
|  | // Handles a mouse |event| coming from the Shelf. | 
|  | void ProcessMouseEvent(const ui::MouseEvent& event); | 
|  |  | 
|  | // Handles a scroll |event| coming from the Shelf. | 
|  | void ProcessScrollEvent(ui::ScrollEvent* event); | 
|  |  | 
|  | // Handles a mousewheel scroll event coming from the shelf. | 
|  | void ProcessMouseWheelEvent(ui::MouseWheelEvent* event); | 
|  |  | 
|  | void AddObserver(ShelfObserver* observer); | 
|  | void RemoveObserver(ShelfObserver* observer); | 
|  |  | 
|  | void NotifyShelfIconPositionsChanged(); | 
|  |  | 
|  | StatusAreaWidget* GetStatusAreaWidget() const; | 
|  |  | 
|  | // Get the anchor rect that the system tray bubble and the notification center | 
|  | // bubble will be anchored. | 
|  | // x() and y() designates anchor point, but width() and height() are dummy. | 
|  | // See also: BubbleDialogDelegateView::GetBubbleBounds() | 
|  | gfx::Rect GetSystemTrayAnchorRect() const; | 
|  |  | 
|  | // Returns whether this shelf should be hidden on secondary display in a given | 
|  | // |state|. | 
|  | bool ShouldHideOnSecondaryDisplay(session_manager::SessionState state); | 
|  |  | 
|  | void SetVirtualKeyboardBoundsForTesting(const gfx::Rect& bounds); | 
|  | ShelfLockingManager* GetShelfLockingManagerForTesting(); | 
|  | ShelfView* GetShelfViewForTesting(); | 
|  |  | 
|  | ShelfLayoutManager* shelf_layout_manager() const { | 
|  | return shelf_layout_manager_; | 
|  | } | 
|  |  | 
|  | // Getters for the various shelf components. | 
|  | ShelfWidget* shelf_widget() const { return shelf_widget_.get(); } | 
|  | ShelfNavigationWidget* navigation_widget() const { | 
|  | return navigation_widget_.get(); | 
|  | } | 
|  | DeskButtonWidget* desk_button_widget() const { | 
|  | return desk_button_widget_.get(); | 
|  | } | 
|  | HotseatWidget* hotseat_widget() const { return hotseat_widget_.get(); } | 
|  | StatusAreaWidget* status_area_widget() const { | 
|  | return status_area_widget_.get(); | 
|  | } | 
|  | LoginShelfWidget* login_shelf_widget() { return login_shelf_widget_.get(); } | 
|  |  | 
|  | ShelfAlignment alignment() const { return alignment_; } | 
|  | ShelfAutoHideBehavior auto_hide_behavior() const { | 
|  | return auto_hide_behavior_; | 
|  | } | 
|  |  | 
|  | ShelfAlignment in_session_alignment() const { | 
|  | return shelf_locking_manager_.in_session_alignment(); | 
|  | } | 
|  |  | 
|  | ShelfAutoHideBehavior in_session_auto_hide_behavior() const { | 
|  | return shelf_locking_manager_.in_session_auto_hide_behavior(); | 
|  | } | 
|  |  | 
|  | ShelfFocusCycler* shelf_focus_cycler() { return shelf_focus_cycler_.get(); } | 
|  |  | 
|  | int auto_hide_lock() const { return auto_hide_lock_; } | 
|  | int disable_auto_hide() const { return disable_auto_hide_; } | 
|  |  | 
|  | ShelfTooltipManager* tooltip() { return tooltip_.get(); } | 
|  |  | 
|  | // |target_state| is the hotseat state after hotseat transition animation. | 
|  | metrics_util::ReportCallback GetHotseatTransitionReportCallback( | 
|  | HotseatState target_state); | 
|  | metrics_util::ReportCallback GetTranslucentBackgroundReportCallback( | 
|  | HotseatState target_state); | 
|  |  | 
|  | metrics_util::ReportCallback GetNavigationWidgetAnimationReportCallback( | 
|  | HotseatState target_hotseat_state); | 
|  |  | 
|  | protected: | 
|  | // ShelfLayoutManagerObserver: | 
|  | void WillDeleteShelfLayoutManager() override; | 
|  | void OnShelfVisibilityStateChanged(ShelfVisibilityState new_state) override; | 
|  | void OnAutoHideStateChanged(ShelfAutoHideState new_state) override; | 
|  | void OnBackgroundUpdated(ShelfBackgroundType background_type, | 
|  | AnimationChangeType change_type) override; | 
|  | void OnHotseatStateChanged(HotseatState old_state, | 
|  | HotseatState new_state) override; | 
|  | void OnWorkAreaInsetsChanged() override; | 
|  |  | 
|  | private: | 
|  | class AutoDimEventHandler; | 
|  | class AutoHideEventHandler; | 
|  | friend class DimShelfLayoutManagerTestBase; | 
|  | friend class ShelfLayoutManagerTest; | 
|  |  | 
|  | // Uses Auto Dim Event Handler to update the shelf dim state. | 
|  | void DimShelf(); | 
|  | void UndimShelf(); | 
|  | bool HasDimShelfTimer(); | 
|  |  | 
|  | // Returns work area insets object for the window with this shelf. | 
|  | WorkAreaInsets* GetWorkAreaInsets() const; | 
|  |  | 
|  | base::WeakPtr<Shelf> GetWeakPtr(); | 
|  |  | 
|  | // Layout manager for the shelf container window. Instances are constructed by | 
|  | // ShelfWidget and lifetimes are managed by the container windows themselves. | 
|  | raw_ptr<ShelfLayoutManager> shelf_layout_manager_ = nullptr; | 
|  |  | 
|  | // Pointers to shelf components. | 
|  | std::unique_ptr<ShelfNavigationWidget> navigation_widget_; | 
|  | std::unique_ptr<DeskButtonWidget> desk_button_widget_; | 
|  | std::unique_ptr<HotseatWidget> hotseat_widget_; | 
|  | std::unique_ptr<StatusAreaWidget> status_area_widget_; | 
|  | // Null during display teardown, see WindowTreeHostManager::DeleteHost() and | 
|  | // RootWindowController::CloseAllChildWindows(). | 
|  | std::unique_ptr<ShelfWidget> shelf_widget_; | 
|  | std::unique_ptr<LoginShelfWidget> login_shelf_widget_; | 
|  |  | 
|  | // These initial values hide the shelf until user preferences are available. | 
|  | ShelfAlignment alignment_ = ShelfAlignment::kBottomLocked; | 
|  | ShelfAutoHideBehavior auto_hide_behavior_ = | 
|  | ShelfAutoHideBehavior::kAlwaysHidden; | 
|  |  | 
|  | // Sets shelf alignment to bottom during login and screen lock. | 
|  | ShelfLockingManager shelf_locking_manager_; | 
|  |  | 
|  | base::ObserverList<ShelfObserver>::Unchecked observers_; | 
|  |  | 
|  | // Forwards mouse and gesture events to ShelfLayoutManager for auto-hide. | 
|  | std::unique_ptr<AutoHideEventHandler> auto_hide_event_handler_; | 
|  |  | 
|  | // Forwards mouse and gesture events to ShelfLayoutManager for auto-dim. | 
|  | std::unique_ptr<AutoDimEventHandler> auto_dim_event_handler_; | 
|  |  | 
|  | // Hands focus off to different parts of the shelf. | 
|  | std::unique_ptr<ShelfFocusCycler> shelf_focus_cycler_; | 
|  |  | 
|  | // Animation metrics reporter for hotseat animations. Owned by the Shelf to | 
|  | // ensure it outlives the Hotseat Widget. | 
|  | std::unique_ptr<HotseatWidgetAnimationMetricsReporter> | 
|  | hotseat_transition_metrics_reporter_; | 
|  |  | 
|  | // Metrics reporter for animations of the traslucent background in the | 
|  | // hotseat. Owned by the Shelf to ensure it outlives the Hotseat Widget. | 
|  | std::unique_ptr<HotseatWidgetAnimationMetricsReporter> | 
|  | translucent_background_metrics_reporter_; | 
|  |  | 
|  | // Animation metrics reporter for navigation widget animations. Owned by the | 
|  | // Shelf to ensure it outlives the Navigation Widget. | 
|  | std::unique_ptr<NavigationWidgetAnimationMetricsReporter> | 
|  | navigation_widget_metrics_reporter_; | 
|  |  | 
|  | // Used by ScopedAutoHideLock to maintain the state of the lock for auto-hide | 
|  | // shelf. | 
|  | int auto_hide_lock_ = 0; | 
|  |  | 
|  | // Used by `ScopedDisableAutoHide` to disable auto-hide shelf behavior. | 
|  | int disable_auto_hide_ = 0; | 
|  |  | 
|  | std::unique_ptr<ShelfTooltipManager> tooltip_; | 
|  |  | 
|  | base::WeakPtrFactory<Shelf> weak_ptr_factory_{this}; | 
|  | }; | 
|  |  | 
|  | }  // namespace ash | 
|  |  | 
|  | #endif  // ASH_SHELF_SHELF_H_ |