mirror-chatterino2/src/widgets/Notebook.hpp

232 lines
6.1 KiB
C++

#pragma once
#include "widgets/BaseWidget.hpp"
#include <pajlada/signals/signal.hpp>
#include <pajlada/signals/signalholder.hpp>
#include <QList>
#include <QMenu>
#include <QMessageBox>
#include <QWidget>
#include <functional>
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<bool(const NotebookTab *)>;
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<Item> 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<Item> items_;
QMenu menu_;
QWidget *selectedPage_ = nullptr;
NotebookButton *addButton_;
std::vector<NotebookButton *> 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