| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef CHROME_BROWSER_UI_TOOLBAR_RECENT_TABS_SUB_MENU_MODEL_H_ |
| #define CHROME_BROWSER_UI_TOOLBAR_RECENT_TABS_SUB_MENU_MODEL_H_ |
| |
| #include <memory> |
| #include <set> |
| #include <string> |
| #include <vector> |
| |
| #include "base/callback_list.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/weak_ptr.h" |
| #include "base/scoped_observation.h" |
| #include "base/task/cancelable_task_tracker.h" |
| #include "base/timer/elapsed_timer.h" |
| #include "chrome/browser/ui/toolbar/app_menu_model.h" |
| #include "components/favicon/core/favicon_service.h" |
| #include "components/sessions/core/session_id.h" |
| #include "components/sessions/core/tab_restore_service.h" |
| #include "components/sessions/core/tab_restore_service_observer.h" |
| #include "components/sync_sessions/synced_session.h" |
| #include "ui/base/accelerators/accelerator.h" |
| #include "ui/base/models/simple_menu_model.h" |
| |
| class Browser; |
| |
| namespace favicon_base { |
| struct FaviconImageResult; |
| } |
| |
| namespace sessions { |
| struct SessionTab; |
| } |
| |
| namespace sync_sessions { |
| class OpenTabsUIDelegate; |
| class SessionSyncService; |
| } |
| |
| namespace ui { |
| class AcceleratorProvider; |
| } |
| |
| // A menu model that builds the contents of "Recent tabs" submenu, which include |
| // the recently closed tabs/windows of current device i.e. local entries, and |
| // opened tabs of other devices. |
| class RecentTabsSubMenuModel : public ui::SimpleMenuModel, |
| public ui::SimpleMenuModel::Delegate, |
| public sessions::TabRestoreServiceObserver { |
| public: |
| // Command Id for recently closed items header or disabled item to which the |
| // accelerator string will be appended. |
| static constexpr int kRecentlyClosedHeaderCommandId = |
| AppMenuModel::kMinRecentTabsCommandId; |
| static constexpr int kDisabledRecentlyClosedHeaderCommandId = |
| kRecentlyClosedHeaderCommandId + AppMenuModel::kNumUnboundedMenuTypes; |
| static constexpr int kFirstMenuEntryCommandId = |
| kDisabledRecentlyClosedHeaderCommandId + |
| AppMenuModel::kNumUnboundedMenuTypes; |
| |
| // Exposed for tests only: return the Command Id for the first entry in the |
| // recently closed window items list. |
| int GetFirstRecentTabsCommandId(); |
| |
| RecentTabsSubMenuModel(ui::AcceleratorProvider* accelerator_provider, |
| Browser* browser); |
| |
| RecentTabsSubMenuModel(const RecentTabsSubMenuModel&) = delete; |
| RecentTabsSubMenuModel& operator=(const RecentTabsSubMenuModel&) = delete; |
| |
| ~RecentTabsSubMenuModel() override; |
| |
| // Overridden from ui::SimpleMenuModel::Delegate: |
| bool IsCommandIdChecked(int command_id) const override; |
| bool IsCommandIdEnabled(int command_id) const override; |
| bool GetAcceleratorForCommandId(int command_id, |
| ui::Accelerator* accelerator) const override; |
| void ExecuteCommand(int command_id, int event_flags) override; |
| const gfx::FontList* GetLabelFontListAt(size_t index) const override; |
| |
| int GetMaxWidthForItemAtIndex(size_t item_index) const; |
| bool GetURLAndTitleForItemAtIndex(size_t index, |
| std::string* url, |
| std::u16string* title); |
| |
| private: |
| struct TabNavigationItem; |
| using TabNavigationItems = std::map<int, TabNavigationItem>; |
| using WindowItems = std::map<int, SessionID>; |
| using GroupItems = std::map<int, SessionID>; |
| struct SubMenuItem; |
| using SubMenuItems = std::map<int, SubMenuItem>; |
| using DeviceNameItems = base::flat_set<int>; |
| |
| // Index of the separator that follows the history menu item. Used as a |
| // reference position for inserting local entries. |
| static constexpr size_t kHistorySeparatorIndex = 1; |
| |
| // Build the menu items by populating the menumodel. |
| void Build(); |
| |
| // Build the recently closed tabs and windows items. |
| void BuildLocalEntries(); |
| |
| // Build the tabs items from other devices. |
| void BuildTabsFromOtherDevices(); |
| |
| // Build a recently closed tab item with parameters needed to restore it, and |
| // add it to the menumodel at |curr_model_index|. |
| void BuildLocalTabItem( |
| SessionID session_id, |
| absl::optional<tab_groups::TabGroupVisualData> visual_data, |
| const std::u16string& title, |
| const GURL& url, |
| size_t curr_model_index); |
| |
| // Build the recently closed window item with parameters needed to restore it, |
| // and add it to the menumodel at |curr_model_index|. |
| void BuildLocalWindowItem(SessionID window_id, |
| std::unique_ptr<ui::SimpleMenuModel> window_model, |
| size_t num_tabs, |
| size_t curr_model_index); |
| |
| // Build the recently closed group item with parameters needed to restore it, |
| // and add it to the menumodel at |curr_model_index|. |
| void BuildLocalGroupItem(SessionID session_id, |
| tab_groups::TabGroupVisualData visual_data, |
| std::unique_ptr<ui::SimpleMenuModel> group_model, |
| size_t num_tabs, |
| size_t curr_model_index); |
| |
| // Build the tab item for other devices with parameters needed to restore it. |
| void BuildOtherDevicesTabItem(const std::string& session_tag, |
| const sessions::SessionTab& tab); |
| |
| // Create a submenu model representing the tabs within a window. |
| std::unique_ptr<ui::SimpleMenuModel> CreateWindowSubMenuModel( |
| const sessions::TabRestoreService::Window& window); |
| |
| // Create a submenu model representing the tabs within a tab group. |
| std::unique_ptr<ui::SimpleMenuModel> CreateGroupSubMenuModel( |
| const sessions::TabRestoreService::Group& group); |
| |
| // Adds a submenu item representation of |group_model| to |parent_model|. |
| void AddGroupItemToModel(SimpleMenuModel* parent_model, |
| std::unique_ptr<SimpleMenuModel> group_model, |
| tab_groups::TabGroupVisualData group_visual_data); |
| |
| // Return the appropriate menu item label for a tab group, given its title |
| // and the number of tabs it contains. |
| std::u16string GetGroupItemLabel(std::u16string title, size_t num_tabs); |
| |
| // Return the command id of the given id's parent submenu, if it has one that |
| // is created by this menu model. Otherwise, return -1. This will be the case |
| // for all items whose parent is the RecentTabsSubMenuModel itself. |
| int GetParentCommandId(int command_id) const; |
| |
| // Add the favicon for the device section header. |
| void AddDeviceFavicon(size_t index_in_menu, |
| syncer::DeviceInfo::FormFactor device_form_factor); |
| |
| // Add the favicon for a local or other devices' tab asynchronously, |
| // OnFaviconDataAvailable() will be invoked when the favicon is ready. |
| void AddTabFavicon(int command_id, |
| ui::SimpleMenuModel* menu_model, |
| const GURL& url); |
| void OnFaviconDataAvailable( |
| int command_id, |
| ui::SimpleMenuModel* menu_model, |
| const favicon_base::FaviconImageResult& image_result); |
| |
| // Clear all recently closed tabs and windows. |
| void ClearLocalEntries(); |
| |
| // Clears all tabs from other devices. |
| void ClearTabsFromOtherDevices(); |
| |
| // Returns the corresponding local or other devices' TabNavigationItems in |
| // |tab_items|. |
| TabNavigationItems* GetTabVectorForCommandId(int command_id); |
| |
| // Convenience function to access OpenTabsUIDelegate provided by |
| // SessionSyncService. Can return null if session sync is not running. |
| sync_sessions::OpenTabsUIDelegate* GetOpenTabsUIDelegate(); |
| |
| // Overridden from TabRestoreServiceObserver: |
| void TabRestoreServiceChanged(sessions::TabRestoreService* service) override; |
| void TabRestoreServiceDestroyed( |
| sessions::TabRestoreService* service) override; |
| |
| void OnForeignSessionUpdated(); |
| |
| // Returns |next_menu_id_| and increments it by 2. This allows for 'sharing' |
| // command ids with the bookmarks menu, which also uses every other int as |
| // an id. |
| int GetAndIncrementNextMenuID(); |
| |
| // Returns true if the command id identifies a tab menu item, either local or |
| // from another device. |
| bool IsTabModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a local tab menu item. |
| bool IsLocalTabModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a tab menu item from another |
| // device. |
| bool IsOtherDeviceTabModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a window menu item. |
| bool IsWindowModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a group menu item. |
| bool IsGroupModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a sub menu item. |
| bool IsSubMenuModelCommandId(int command_id) const; |
| |
| // Returns true if the command id identifies a device name item. |
| bool IsDeviceNameCommandId(int command_id) const; |
| |
| const raw_ptr<Browser> browser_; // Weak. |
| |
| const raw_ptr<sync_sessions::SessionSyncService> |
| session_sync_service_; // Weak. |
| |
| // Accelerator for reopening last closed tab. |
| ui::Accelerator reopen_closed_tab_accelerator_; |
| |
| // Accelerator for showing history. |
| ui::Accelerator show_history_accelerator_; |
| |
| // ID of the next menu item. |
| int next_menu_id_; |
| |
| // Navigation items for local recently closed tabs. Upon invocation of the |
| // menu, the navigation information is retrieved from |
| // |local_tab_navigation_items_| and used to navigate to the item specified. |
| TabNavigationItems local_tab_navigation_items_; |
| |
| // Similar to |local_tab_navigation_items_| except the tabs are opened tabs |
| // from other devices. |
| TabNavigationItems other_devices_tab_navigation_items_; |
| |
| // Window items for local recently closed windows. Upon invocation of the |
| // menu, information is retrieved from |local_window_items_| and used to |
| // create the specified window. |
| WindowItems local_window_items_; |
| |
| // Group items for local recently closed groups. Upon invocation of the menu, |
| // information is retrieved from |local_group_items_| and used to create the |
| // specified group. |
| GroupItems local_group_items_; |
| |
| // Sub menu items for sub menu entry points representing local recently |
| // closed groups and windows. These are not executable. |
| SubMenuItems local_sub_menu_items_; |
| |
| // Device name items for names of non-local devices. These are not |
| // executable. |
| DeviceNameItems device_name_items_; |
| |
| // Index of the last local entry (recently closed tab or window or group) in |
| // the menumodel. |
| size_t last_local_model_index_ = kHistorySeparatorIndex; |
| |
| base::CancelableTaskTracker local_tab_cancelable_task_tracker_; |
| |
| // Time the menu is open for until a recent tab is selected. |
| base::ElapsedTimer menu_opened_timer_; |
| |
| base::ScopedObservation<sessions::TabRestoreService, |
| sessions::TabRestoreServiceObserver> |
| tab_restore_service_observation_{this}; |
| |
| base::CallbackListSubscription foreign_session_updated_subscription_; |
| |
| base::WeakPtrFactory<RecentTabsSubMenuModel> weak_ptr_factory_{this}; |
| base::WeakPtrFactory<RecentTabsSubMenuModel> |
| weak_ptr_factory_for_other_devices_tab_{this}; |
| }; |
| |
| #endif // CHROME_BROWSER_UI_TOOLBAR_RECENT_TABS_SUB_MENU_MODEL_H_ |