mirror-chatterino2/src/widgets/Notebook.cpp

486 lines
11 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "widgets/Notebook.hpp"
2018-06-26 14:09:39 +02:00
#include "Application.hpp"
#include "debug/Log.hpp"
#include "singletons/Settings.hpp"
2018-06-28 20:03:04 +02:00
#include "singletons/Theme.hpp"
2018-06-26 14:09:39 +02:00
#include "singletons/WindowManager.hpp"
#include "util/InitUpdateButton.hpp"
2018-06-26 17:20:03 +02:00
#include "widgets/Window.hpp"
#include "widgets/dialogs/SettingsDialog.hpp"
2018-06-26 14:09:39 +02:00
#include "widgets/helper/NotebookButton.hpp"
#include "widgets/helper/NotebookTab.hpp"
#include "widgets/helper/Shortcut.hpp"
#include "widgets/splits/SplitContainer.hpp"
2016-12-29 17:31:07 +01:00
#include <QDebug>
#include <QFile>
2017-01-18 04:52:47 +01:00
#include <QFormLayout>
#include <QLayout>
#include <QList>
2017-05-29 21:26:55 +02:00
#include <QShortcut>
#include <QStandardPaths>
#include <QUuid>
2017-01-18 04:52:47 +01:00
#include <QWidget>
#include <boost/foreach.hpp>
2017-01-18 04:52:47 +01:00
2017-04-14 17:52:22 +02:00
namespace chatterino {
2017-01-18 21:30:23 +01:00
2018-05-23 11:59:37 +02:00
Notebook::Notebook(QWidget *parent)
: BaseWidget(parent)
, addButton_(new NotebookButton(this))
2018-04-18 09:12:29 +02:00
{
this->addButton_->setIcon(NotebookButton::Icon::Plus);
2018-08-02 14:23:27 +02:00
this->addButton_->setHidden(true);
2018-04-18 09:12:29 +02:00
auto *shortcut_next = new QShortcut(QKeySequence("Ctrl+Tab"), this);
2018-08-06 21:17:03 +02:00
QObject::connect(shortcut_next, &QShortcut::activated,
[this] { this->selectNextTab(); });
2018-04-18 09:12:29 +02:00
auto *shortcut_prev = new QShortcut(QKeySequence("Ctrl+Shift+Tab"), this);
2018-08-06 21:17:03 +02:00
QObject::connect(shortcut_prev, &QShortcut::activated,
[this] { this->selectPreviousTab(); });
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
NotebookTab *Notebook::addPage(QWidget *page, QString title, bool select)
2018-04-18 09:12:29 +02:00
{
2018-05-23 11:59:37 +02:00
auto *tab = new NotebookTab(this);
2018-04-18 09:12:29 +02:00
tab->page = page;
2018-06-01 16:01:49 +02:00
tab->setCustomTitle(title);
2018-04-18 09:12:29 +02:00
Item item;
item.page = page;
item.tab = tab;
2018-07-06 19:23:47 +02:00
this->items_.append(item);
2018-04-18 09:12:29 +02:00
page->hide();
page->setParent(this);
2018-07-06 19:23:47 +02:00
if (select || this->items_.count() == 1) {
2018-04-18 09:12:29 +02:00
this->select(page);
}
this->performLayout();
2018-06-04 12:54:09 +02:00
tab->show();
2018-04-18 09:12:29 +02:00
return tab;
}
2018-05-23 11:59:37 +02:00
void Notebook::removePage(QWidget *page)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
for (int i = 0; i < this->items_.count(); i++) {
if (this->items_[i].page == page) {
if (this->items_.count() == 1) {
2018-04-18 09:12:29 +02:00
this->select(nullptr);
2018-07-06 19:23:47 +02:00
} else if (i == this->items_.count() - 1) {
this->select(this->items_[i - 1].page);
2018-04-18 09:12:29 +02:00
} else {
2018-07-06 19:23:47 +02:00
this->select(this->items_[i + 1].page);
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
this->items_[i].page->deleteLater();
this->items_[i].tab->deleteLater();
2018-04-18 09:12:29 +02:00
// if (this->items.empty()) {
// this->addNewPage();
// }
2018-07-06 19:23:47 +02:00
this->items_.removeAt(i);
2018-04-18 09:12:29 +02:00
break;
}
}
2018-05-23 04:22:17 +02:00
this->performLayout(true);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::removeCurrentPage()
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
if (this->selectedPage_ != nullptr) {
this->removePage(this->selectedPage_);
2018-04-18 09:12:29 +02:00
}
}
2018-05-23 11:59:37 +02:00
int Notebook::indexOf(QWidget *page) const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
for (int i = 0; i < this->items_.count(); i++) {
if (this->items_[i].page == page) {
2018-04-18 09:12:29 +02:00
return i;
}
}
return -1;
}
2018-05-23 11:59:37 +02:00
void Notebook::select(QWidget *page)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
if (page == this->selectedPage_) {
2018-04-18 09:12:29 +02:00
return;
}
if (page != nullptr) {
page->setHidden(false);
2018-05-31 16:02:20 +02:00
assert(this->containsPage(page));
Item &item = this->findItem(page);
item.tab->setSelected(true);
item.tab->raise();
if (item.selectedWidget == nullptr) {
item.page->setFocus();
} else {
if (containsChild(page, item.selectedWidget)) {
qDebug() << item.selectedWidget;
item.selectedWidget->setFocus(Qt::MouseFocusReason);
} else {
2018-08-06 21:17:03 +02:00
qDebug()
<< "Notebook: selected child of page doesn't exist anymore";
2018-05-31 16:02:20 +02:00
}
}
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
if (this->selectedPage_ != nullptr) {
this->selectedPage_->setHidden(true);
2018-04-18 09:12:29 +02:00
2018-07-06 19:23:47 +02:00
Item &item = this->findItem(selectedPage_);
2018-05-31 16:02:20 +02:00
item.tab->setSelected(false);
2018-04-18 09:12:29 +02:00
// for (auto split : this->selectedPage->getSplits()) {
// split->updateLastReadMessage();
// }
2018-05-31 16:02:20 +02:00
2018-07-06 19:23:47 +02:00
item.selectedWidget = this->selectedPage_->focusWidget();
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
this->selectedPage_ = page;
2018-04-18 09:12:29 +02:00
this->performLayout();
}
2018-05-31 16:02:20 +02:00
bool Notebook::containsPage(QWidget *page)
{
2018-07-06 19:23:47 +02:00
return std::any_of(this->items_.begin(), this->items_.end(),
2018-05-31 16:02:20 +02:00
[page](const auto &item) { return item.page == page; });
}
Notebook::Item &Notebook::findItem(QWidget *page)
{
2018-08-06 21:17:03 +02:00
auto it =
std::find_if(this->items_.begin(), this->items_.end(),
[page](const auto &item) { return page == item.page; });
2018-07-06 19:23:47 +02:00
assert(it != this->items_.end());
2018-05-31 16:02:20 +02:00
return *it;
}
bool Notebook::containsChild(const QObject *obj, const QObject *child)
{
2018-08-06 21:17:03 +02:00
return std::any_of(obj->children().begin(), obj->children().end(),
[child](const QObject *o) {
if (o == child) {
return true;
}
2018-05-31 16:02:20 +02:00
2018-08-06 21:17:03 +02:00
return containsChild(o, child);
});
2018-05-31 16:02:20 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::selectIndex(int index)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
if (index < 0 || this->items_.count() <= index) {
2018-04-18 09:12:29 +02:00
return;
}
2018-07-06 19:23:47 +02:00
this->select(this->items_[index].page);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::selectNextTab()
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
if (this->items_.size() <= 1) {
2018-04-18 09:12:29 +02:00
return;
}
2018-07-06 19:23:47 +02:00
int index = (this->indexOf(this->selectedPage_) + 1) % this->items_.count();
2018-04-18 09:12:29 +02:00
2018-07-06 19:23:47 +02:00
this->select(this->items_[index].page);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::selectPreviousTab()
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
if (this->items_.size() <= 1) {
2018-04-18 09:12:29 +02:00
return;
}
2018-07-06 19:23:47 +02:00
int index = this->indexOf(this->selectedPage_) - 1;
2018-04-18 09:12:29 +02:00
if (index < 0) {
2018-07-06 19:23:47 +02:00
index += this->items_.count();
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
this->select(this->items_[index].page);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
int Notebook::getPageCount() const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
return this->items_.count();
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
QWidget *Notebook::getPageAt(int index) const
2018-05-23 04:22:17 +02:00
{
2018-07-06 19:23:47 +02:00
return this->items_[index].page;
2018-05-23 04:22:17 +02:00
}
2018-05-23 11:59:37 +02:00
int Notebook::getSelectedIndex() const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
return this->indexOf(this->selectedPage_);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
QWidget *Notebook::getSelectedPage() const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
return this->selectedPage_;
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
QWidget *Notebook::tabAt(QPoint point, int &index, int maxWidth)
2018-04-18 09:12:29 +02:00
{
int i = 0;
2018-07-06 19:23:47 +02:00
for (auto &item : this->items_) {
2018-04-18 09:12:29 +02:00
QRect rect = item.tab->getDesiredRect();
rect.setHeight((int)(this->getScale() * 24));
rect.setWidth(std::min(maxWidth, rect.width()));
if (rect.contains(point)) {
index = i;
return item.page;
}
i++;
}
index = -1;
return nullptr;
}
2018-05-23 11:59:37 +02:00
void Notebook::rearrangePage(QWidget *page, int index)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
this->items_.move(this->indexOf(page), index);
2018-04-18 09:12:29 +02:00
this->performLayout(true);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
bool Notebook::getAllowUserTabManagement() const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
return this->allowUserTabManagement_;
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::setAllowUserTabManagement(bool value)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
this->allowUserTabManagement_ = value;
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
bool Notebook::getShowAddButton() const
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
return this->showAddButton_;
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::setShowAddButton(bool value)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
this->showAddButton_ = value;
2018-04-18 09:12:29 +02:00
this->addButton_->setHidden(!value);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
void Notebook::scaleChangedEvent(float scale)
2018-04-18 09:12:29 +02:00
{
2018-05-23 04:22:17 +02:00
float h = NOTEBOOK_TAB_HEIGHT * this->getScale();
2018-04-18 09:12:29 +02:00
this->addButton_->setFixedSize(h, h);
2018-04-18 09:12:29 +02:00
2018-07-06 19:23:47 +02:00
for (auto &i : this->items_) {
2018-04-18 09:12:29 +02:00
i.tab->updateSize();
}
}
2018-05-23 11:59:37 +02:00
void Notebook::resizeEvent(QResizeEvent *)
2018-04-18 09:12:29 +02:00
{
this->performLayout();
}
2018-05-23 11:59:37 +02:00
void Notebook::performLayout(bool animated)
2018-04-18 09:12:29 +02:00
{
int xStart = int(2 * this->getScale());
2018-04-18 09:12:29 +02:00
int x = xStart, y = 0;
float scale = this->getScale();
int h = int(NOTEBOOK_TAB_HEIGHT * this->getScale());
2018-07-06 19:23:47 +02:00
for (auto *btn : this->customButtons_) {
if (!btn->isVisible()) {
continue;
}
btn->setFixedSize(h, h);
btn->move(x, 0);
x += h;
}
2018-05-23 04:22:17 +02:00
int tabHeight = static_cast<int>(NOTEBOOK_TAB_HEIGHT * scale);
2018-04-18 09:12:29 +02:00
bool first = true;
2018-07-06 19:23:47 +02:00
for (auto i = this->items_.begin(); i != this->items_.end(); i++) {
bool wrap =
2018-08-06 21:17:03 +02:00
!first &&
(((i + 1 == this->items_.end() && this->showAddButton_) ? tabHeight
: 0) +
x + i->tab->width()) > width();
if (wrap) {
2018-04-18 09:12:29 +02:00
y += i->tab->height();
i->tab->moveAnimated(QPoint(xStart, y), animated);
x = i->tab->width() + xStart;
} else {
i->tab->moveAnimated(QPoint(x, y), animated);
x += i->tab->width();
}
2018-06-11 21:57:17 +02:00
x += int(scale * 1);
2018-04-18 09:12:29 +02:00
first = false;
}
2018-07-06 19:23:47 +02:00
if (this->showAddButton_) {
this->addButton_->move(x, y);
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
if (this->lineY_ != y + tabHeight) {
this->lineY_ = y + tabHeight;
2018-04-18 09:12:29 +02:00
this->update();
}
y += int(3 * scale);
2018-04-18 09:12:29 +02:00
2018-07-06 19:23:47 +02:00
for (auto &i : this->items_) {
2018-04-18 09:12:29 +02:00
i.tab->raise();
}
2018-07-06 19:23:47 +02:00
if (this->showAddButton_) {
this->addButton_->raise();
2018-04-18 09:12:29 +02:00
}
2018-07-06 19:23:47 +02:00
if (this->selectedPage_ != nullptr) {
this->selectedPage_->move(0, y + tabHeight);
this->selectedPage_->resize(width(), height() - y - tabHeight);
this->selectedPage_->raise();
2018-04-18 09:12:29 +02:00
}
}
2018-05-23 11:59:37 +02:00
void Notebook::paintEvent(QPaintEvent *event)
2018-04-18 09:12:29 +02:00
{
BaseWidget::paintEvent(event);
QPainter painter(this);
2018-08-06 21:17:03 +02:00
painter.fillRect(0, this->lineY_, this->width(),
(int)(3 * this->getScale()), this->theme->tabs.bottomLine);
2018-04-18 09:12:29 +02:00
}
2018-05-23 11:59:37 +02:00
NotebookButton *Notebook::getAddButton()
2018-05-23 04:22:17 +02:00
{
return this->addButton_;
2018-05-23 04:22:17 +02:00
}
NotebookButton *Notebook::addCustomButton()
{
NotebookButton *btn = new NotebookButton(this);
2018-07-06 19:23:47 +02:00
this->customButtons_.push_back(btn);
this->performLayout();
return btn;
}
2018-05-23 11:59:37 +02:00
NotebookTab *Notebook::getTabFromPage(QWidget *page)
2018-04-18 09:12:29 +02:00
{
2018-07-06 19:23:47 +02:00
for (auto &it : this->items_) {
2018-04-18 09:12:29 +02:00
if (it.page == page) {
return it.tab;
}
}
return nullptr;
}
SplitNotebook::SplitNotebook(Window *parent)
2018-05-23 11:59:37 +02:00
: Notebook(parent)
2016-12-29 17:31:07 +01:00
{
2018-08-06 21:17:03 +02:00
this->connect(this->getAddButton(), &NotebookButton::clicked, [this]() {
QTimer::singleShot(80, this, [this] { this->addPage(true); });
});
bool customFrame = parent->hasCustomWindowFrame();
if (!customFrame) {
this->addCustomButtons();
}
2017-01-01 18:43:52 +01:00
}
void SplitNotebook::addCustomButtons()
{
// settings
auto settingsBtn = this->addCustomButton();
2018-08-06 21:17:03 +02:00
settingsBtn->setVisible(
!getSettings()->hidePreferencesButton.getValue());
getSettings()->hidePreferencesButton.connect(
2018-08-06 21:17:03 +02:00
[settingsBtn](bool hide, auto) { settingsBtn->setVisible(!hide); },
this->connections_);
settingsBtn->setIcon(NotebookButton::Settings);
QObject::connect(settingsBtn, &NotebookButton::clicked,
[] { getApp()->windows->showSettingsDialog(); });
// account
auto userBtn = this->addCustomButton();
userBtn->setVisible(!getSettings()->hideUserButton.getValue());
getSettings()->hideUserButton.connect(
2018-08-06 21:17:03 +02:00
[userBtn](bool hide, auto) { userBtn->setVisible(!hide); },
this->connections_);
userBtn->setIcon(NotebookButton::User);
QObject::connect(userBtn, &NotebookButton::clicked, [this, userBtn] {
2018-08-06 21:17:03 +02:00
getApp()->windows->showAccountSelectPopup(
this->mapToGlobal(userBtn->rect().bottomRight()));
});
// updates
auto updateBtn = this->addCustomButton();
initUpdateButton(*updateBtn, this->signalHolder_);
}
2018-05-23 04:22:17 +02:00
SplitContainer *SplitNotebook::addPage(bool select)
{
2018-05-23 04:22:17 +02:00
SplitContainer *container = new SplitContainer(this);
2018-05-23 11:59:37 +02:00
auto *tab = Notebook::addPage(container, QString(), select);
2018-05-23 04:22:17 +02:00
container->setTab(tab);
tab->setParent(this);
2017-01-22 12:46:35 +01:00
tab->show();
2018-05-23 04:22:17 +02:00
return container;
2016-12-29 17:31:07 +01:00
}
2016-12-30 12:20:26 +01:00
2018-05-23 04:22:17 +02:00
SplitContainer *SplitNotebook::getOrAddSelectedPage()
{
2018-05-23 04:22:17 +02:00
auto *selectedPage = this->getSelectedPage();
2018-01-22 21:31:45 +01:00
2018-08-06 21:17:03 +02:00
return selectedPage != nullptr ? (SplitContainer *)selectedPage
: this->addPage();
2017-01-18 21:30:23 +01:00
}
2017-04-14 17:52:22 +02:00
} // namespace chatterino