#pragma once #include "widgets/BaseWidget.hpp" #include #include #include #include #include #include #include namespace chatterino { class Window; class UpdateDialog; class NotebookButton; class NotebookTab; class SplitContainer; enum NotebookTabLocation { Top = 0, Left = 1, Right = 2, Bottom = 3 }; // Controls the visibility of tabs in this notebook enum NotebookTabVisibility : int { // Show all tabs AllTabs = 0, // Only show tabs containing splits that are live LiveOnly = 1, }; using TabVisibilityFilter = std::function; class Notebook : public BaseWidget { Q_OBJECT public: explicit Notebook(QWidget *parent); ~Notebook() override = default; NotebookTab *addPage(QWidget *page, QString title = QString(), bool select = false); void removePage(QWidget *page); void removeCurrentPage(); /** * @brief Returns index of page in Notebook, or -1 if not found. **/ int indexOf(QWidget *page) const; /** * @brief Returns the visible index of page in Notebook, or -1 if not found. * Given page should be visible according to the set TabVisibilityFilter. **/ int visibleIndexOf(QWidget *page) const; /** * @brief Returns the number of visible tabs in Notebook. **/ int getVisibleTabCount() const; /** * @brief Selects the Notebook tab containing the given page. **/ virtual void select(QWidget *page, bool focusPage = true); /** * @brief Selects the Notebook tab at the given index. Ignores whether tabs * are visible or not. **/ void selectIndex(int index, bool focusPage = true); /** * @brief Selects the index'th visible tab in the Notebook. * * For example, selecting the 0th visible tab selects the first tab in this * Notebook that is visible according to the TabVisibilityFilter. If no filter * is set, equivalent to Notebook::selectIndex. **/ void selectVisibleIndex(int index, bool focusPage = true); /** * @brief Selects the next visible tab. Wraps to the start if required. **/ void selectNextTab(bool focusPage = true); /** * @brief Selects the previous visible tab. Wraps to the end if required. **/ void selectPreviousTab(bool focusPage = true); /** * @brief Selects the last visible tab. **/ void selectLastTab(bool focusPage = true); int getPageCount() const; QWidget *getPageAt(int index) const; int getSelectedIndex() const; QWidget *getSelectedPage() const; QWidget *tabAt(QPoint point, int &index, int maxWidth = 2000000000); void rearrangePage(QWidget *page, int index); bool getAllowUserTabManagement() const; void setAllowUserTabManagement(bool value); bool getShowTabs() const; void setShowTabs(bool value); bool getShowAddButton() const; void setShowAddButton(bool value); void setTabLocation(NotebookTabLocation location); bool isNotebookLayoutLocked() const; void setLockNotebookLayout(bool value); void addNotebookActionsToMenu(QMenu *menu); // Update layout and tab visibility void refresh(); protected: void scaleChangedEvent(float scale_) override; void resizeEvent(QResizeEvent *) override; void mousePressEvent(QMouseEvent *event) override; void paintEvent(QPaintEvent *) override; NotebookButton *getAddButton(); NotebookButton *addCustomButton(); struct Item { NotebookTab *tab{}; QWidget *page{}; QWidget *selectedWidget{}; }; const QList items() { return items_; } /** * @brief Apply the given tab visibility filter * * An empty function can be provided to denote that no filter will be applied * * Tabs will be redrawn after this function is called. **/ void setTabVisibilityFilter(TabVisibilityFilter filter); /** * @brief shouldShowTab has the final say whether a tab should be visible right now. **/ bool shouldShowTab(const NotebookTab *tab) const; private: void performLayout(bool animate = false); /** * @brief Show a popup informing the user of some big tab visibility changes **/ void showTabVisibilityInfoPopup(); /** * @brief Updates the visibility state of all tabs **/ void updateTabVisibility(); void updateTabVisibilityMenuAction(); void resizeAddButton(); bool containsPage(QWidget *page); Item *findItem(QWidget *page); static bool containsChild(const QObject *obj, const QObject *child); NotebookTab *getTabFromPage(QWidget *page); // Returns the number of buttons in `customButtons_` that are visible size_t visibleButtonCount() const; QList items_; QMenu menu_; QWidget *selectedPage_ = nullptr; NotebookButton *addButton_; std::vector customButtons_; bool allowUserTabManagement_ = false; bool showTabs_ = true; bool showAddButton_ = false; int lineOffset_ = 20; bool lockNotebookLayout_ = false; NotebookTabLocation tabLocation_ = NotebookTabLocation::Top; QAction *lockNotebookLayoutAction_; QAction *showTabsAction_; // This filter, if set, is used to figure out the visibility of // the tabs in this notebook. TabVisibilityFilter tabVisibilityFilter_; }; class SplitNotebook : public Notebook { public: SplitNotebook(Window *parent); SplitContainer *addPage(bool select = false); SplitContainer *getOrAddSelectedPage(); /// Returns `nullptr` when no page is selected. SplitContainer *getSelectedPage(); void select(QWidget *page, bool focusPage = true) override; void themeChangedEvent() override; protected: void showEvent(QShowEvent *event) override; private: void addCustomButtons(); pajlada::Signals::SignalHolder signalHolder_; // Main window on Windows has basically a duplicate of this in Window NotebookButton *streamerModeIcon_{}; void updateStreamerModeIcon(); }; } // namespace chatterino