367 lines
11 KiB
C++
367 lines
11 KiB
C++
// For license of this file, see <project-root-folder>/LICENSE.md.
|
|
|
|
#include "gui/tabwidget.h"
|
|
|
|
#include "definitions/definitions.h"
|
|
#include "gui/dialogs/formmain.h"
|
|
#include "gui/feedmessageviewer.h"
|
|
#include "gui/feedsview.h"
|
|
#include "gui/messagesview.h"
|
|
#include "gui/newspaperpreviewer.h"
|
|
#include "gui/reusable/plaintoolbutton.h"
|
|
#include "gui/tabbar.h"
|
|
#include "miscellaneous/application.h"
|
|
#include "miscellaneous/iconfactory.h"
|
|
#include "miscellaneous/settings.h"
|
|
#include "miscellaneous/textfactory.h"
|
|
#include "network-web/webfactory.h"
|
|
|
|
#if defined(USE_WEBENGINE)
|
|
#include "gui/webbrowser.h"
|
|
#endif
|
|
|
|
#include <QMenu>
|
|
#include <QToolButton>
|
|
|
|
TabWidget::TabWidget(QWidget* parent) : QTabWidget(parent), m_menuMain(nullptr) {
|
|
setTabBar(new TabBar(this));
|
|
setupMainMenuButton();
|
|
initializeTabs();
|
|
createConnections();
|
|
}
|
|
|
|
TabWidget::~TabWidget() {
|
|
qDebugNN << LOGSEC_GUI << "Destroying TabWidget instance.";
|
|
}
|
|
|
|
void TabWidget::setupMainMenuButton() {
|
|
m_btnMainMenu = new PlainToolButton(this);
|
|
m_btnMainMenu->setAutoRaise(true);
|
|
m_btnMainMenu->setPadding(3);
|
|
m_btnMainMenu->setToolTip(tr("Displays main menu."));
|
|
m_btnMainMenu->setIcon(qApp->icons()->fromTheme(QSL("go-home")));
|
|
m_btnMainMenu->setPopupMode(QToolButton::ToolButtonPopupMode::InstantPopup);
|
|
|
|
connect(m_btnMainMenu, &PlainToolButton::clicked, this, &TabWidget::openMainMenu);
|
|
}
|
|
|
|
void TabWidget::openMainMenu() {
|
|
if (m_menuMain == nullptr) {
|
|
m_menuMain = new QMenu(tr("Main menu"), this);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuFile);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuView);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuAccounts);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuFeeds);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuMessages);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuWebBrowserTabs);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuTools);
|
|
m_menuMain->addMenu(qApp->mainForm()->m_ui->m_menuHelp);
|
|
}
|
|
|
|
QPoint button_position = m_btnMainMenu->pos();
|
|
const QSize target_size = m_btnMainMenu->size() / 2.0;
|
|
|
|
button_position.setX(button_position.x() + target_size.width());
|
|
button_position.setY(button_position.y() + target_size.height());
|
|
m_menuMain->exec(mapToGlobal(button_position));
|
|
}
|
|
|
|
void TabWidget::showDownloadManager() {
|
|
for (int i = 0; i < count(); i++) {
|
|
if (widget(i)->metaObject()->className() == QSL("DownloadManager")) {
|
|
setCurrentIndex(i);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Download manager is not opened. Create tab with it.
|
|
qApp->downloadManager()->setParent(this);
|
|
addTab(qApp->downloadManager(),
|
|
qApp->icons()->fromTheme(QSL("emblem-downloads")),
|
|
tr("Downloads"),
|
|
TabBar::TabType::DownloadManager);
|
|
setCurrentIndex(count() - 1);
|
|
}
|
|
|
|
void TabWidget::checkTabBarVisibility() {
|
|
const bool should_be_visible = count() > 1 ||
|
|
!qApp->settings()->value(GROUP(GUI), SETTING(GUI::HideTabBarIfOnlyOneTab)).toBool();
|
|
|
|
if (should_be_visible) {
|
|
setCornerWidget(m_btnMainMenu, Qt::Corner::TopLeftCorner);
|
|
m_btnMainMenu->setVisible(true);
|
|
}
|
|
else {
|
|
setCornerWidget(nullptr, Qt::Corner::TopLeftCorner);
|
|
setCornerWidget(nullptr, Qt::Corner::TopRightCorner);
|
|
m_btnMainMenu->setVisible(false);
|
|
}
|
|
|
|
tabBar()->setVisible(should_be_visible);
|
|
}
|
|
|
|
void TabWidget::tabInserted(int index) {
|
|
QTabWidget::tabInserted(index);
|
|
checkTabBarVisibility();
|
|
const int count_of_tabs = count();
|
|
|
|
if (index < count_of_tabs - 1 && count_of_tabs > 1) {
|
|
// New tab was inserted and the tab is not the last one.
|
|
fixContentsAfterMove(index, count_of_tabs - 1);
|
|
}
|
|
}
|
|
|
|
void TabWidget::tabRemoved(int index) {
|
|
QTabWidget::tabRemoved(index);
|
|
checkTabBarVisibility();
|
|
const int count_of_tabs = count();
|
|
|
|
if (index < count_of_tabs && count_of_tabs > 1) {
|
|
// Some tab was removed and the tab was not the last one.
|
|
fixContentsAfterMove(index, count_of_tabs - 1);
|
|
}
|
|
}
|
|
|
|
void TabWidget::createConnections() {
|
|
connect(tabBar(), &TabBar::tabCloseRequested, this, &TabWidget::closeTab);
|
|
connect(tabBar(), &TabBar::emptySpaceDoubleClicked, this, &TabWidget::addEmptyBrowser);
|
|
connect(tabBar(), &TabBar::tabMoved, this, &TabWidget::fixContentsAfterMove);
|
|
|
|
connect(feedMessageViewer()->messagesView(), &MessagesView::openMessagesInNewspaperView, this, &TabWidget::addNewspaperView);
|
|
connect(feedMessageViewer()->feedsView(), &FeedsView::openMessagesInNewspaperView, this, &TabWidget::addNewspaperView);
|
|
}
|
|
|
|
void TabWidget::initializeTabs() {
|
|
// Create widget for "Feeds" page and add it.
|
|
m_feedMessageViewer = new FeedMessageViewer(this);
|
|
const int index_of_browser = addTab(m_feedMessageViewer, QIcon(), tr("Feeds"), TabBar::TabType::FeedReader);
|
|
|
|
setTabToolTip(index_of_browser, tr("Browse your feeds and articles"));
|
|
}
|
|
|
|
void TabWidget::setupIcons() {
|
|
// Iterate through all tabs and update icons
|
|
// accordingly.
|
|
for (int index = 0; index < count(); index++) {
|
|
// Index 0 usually contains widget which displays feeds & messages.
|
|
if (tabBar()->tabType(index) == TabBar::TabType::FeedReader) {
|
|
setTabIcon(index, qApp->icons()->fromTheme(QSL("application-rss+xml")));
|
|
}
|
|
}
|
|
}
|
|
|
|
bool TabWidget::closeTab(int index) {
|
|
if (tabBar()->tabType(index) == TabBar::TabType::Closable) {
|
|
removeTab(index, true);
|
|
return true;
|
|
}
|
|
else if (tabBar()->tabType(index) == TabBar::TabType::DownloadManager) {
|
|
removeTab(index, false);
|
|
return true;
|
|
}
|
|
else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void TabWidget::closeAllTabsExceptCurrent() {
|
|
// Close tabs after active tab.
|
|
int index_of_active = currentIndex();
|
|
|
|
for (int i = count() - 1; i >= 0; i--) {
|
|
if (i != index_of_active) {
|
|
if (i < index_of_active) {
|
|
index_of_active--;
|
|
}
|
|
|
|
closeTab(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void TabWidget::closeAllTabs() {
|
|
for (int i = count() - 1; i >= 0; i--) {
|
|
closeTab(i);
|
|
}
|
|
}
|
|
|
|
void TabWidget::closeCurrentTab() {
|
|
closeTab(currentIndex());
|
|
}
|
|
|
|
int TabWidget::addNewspaperView(RootItem* root, const QList<Message>& messages) {
|
|
int msg_height = height() - tabBar()->height() - 50;
|
|
NewspaperPreviewer* prev = new NewspaperPreviewer(msg_height, root, messages, this);
|
|
|
|
connect(prev, &NewspaperPreviewer::markMessageRead,
|
|
m_feedMessageViewer->messagesView()->sourceModel(), &MessagesModel::setMessageReadById);
|
|
connect(prev, &NewspaperPreviewer::markMessageImportant,
|
|
m_feedMessageViewer->messagesView()->sourceModel(), &MessagesModel::setMessageImportantById);
|
|
|
|
int index = addTab(prev,
|
|
qApp->icons()->fromTheme(QSL("format-justify-fill")),
|
|
tr("Newspaper view"),
|
|
TabBar::TabType::Closable);
|
|
|
|
// NOTE: Do not bring "newspaper" tabs to front anymore.
|
|
//setCurrentIndex(index);
|
|
|
|
return index;
|
|
}
|
|
|
|
int TabWidget::addEmptyBrowser() {
|
|
return addBrowser(false, true);
|
|
}
|
|
|
|
int TabWidget::addLinkedBrowser(const QUrl& initial_url) {
|
|
return addBrowser(false, false, initial_url);
|
|
}
|
|
|
|
int TabWidget::addLinkedBrowser(const QString& initial_url) {
|
|
return addLinkedBrowser(QUrl(initial_url));
|
|
}
|
|
|
|
int TabWidget::addBrowser(bool move_after_current, bool make_active, const QUrl& initial_url) {
|
|
#if defined(USE_WEBENGINE)
|
|
// Create new WebBrowser.
|
|
WebBrowser* browser = new WebBrowser(this);
|
|
int final_index;
|
|
QString browser_tab_name = tr("Web browser");
|
|
|
|
#if defined(Q_OS_MACOSOS)
|
|
browser_tab_name = browser_tab_name.prepend(QSL(" "));
|
|
#endif
|
|
|
|
if (move_after_current) {
|
|
// Insert web browser after current tab.
|
|
final_index = insertTab(currentIndex() + 1, browser, qApp->icons()->fromTheme(QSL("text-html")),
|
|
browser_tab_name, TabBar::TabType::Closable);
|
|
}
|
|
else {
|
|
// Add new browser as the last tab.
|
|
final_index = addTab(browser, qApp->icons()->fromTheme(QSL("text-html")),
|
|
browser_tab_name,
|
|
TabBar::TabType::Closable);
|
|
}
|
|
|
|
// Make connections.
|
|
connect(browser, &WebBrowser::titleChanged, this, &TabWidget::changeTitle);
|
|
connect(browser, &WebBrowser::iconChanged, this, &TabWidget::changeIcon);
|
|
|
|
// Setup the tab index.
|
|
browser->setIndex(final_index);
|
|
|
|
// Load initial web page if desired.
|
|
if (initial_url.isValid()) {
|
|
browser->loadUrl(initial_url);
|
|
}
|
|
|
|
// Make new web browser active if desired.
|
|
if (make_active) {
|
|
setCurrentIndex(final_index);
|
|
browser->setFocus(Qt::FocusReason::OtherFocusReason);
|
|
}
|
|
|
|
return final_index;
|
|
#else
|
|
Q_UNUSED(move_after_current)
|
|
Q_UNUSED(make_active)
|
|
qApp->web()->openUrlInExternalBrowser(initial_url.toString());
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
void TabWidget::gotoNextTab() {
|
|
if (currentIndex() == count() - 1) {
|
|
setCurrentIndex(0);
|
|
}
|
|
else {
|
|
setCurrentIndex(currentIndex() + 1);
|
|
}
|
|
}
|
|
|
|
void TabWidget::gotoPreviousTab() {
|
|
if (currentIndex() == 0) {
|
|
setCurrentIndex(count() - 1);
|
|
}
|
|
else {
|
|
setCurrentIndex(currentIndex() - 1);
|
|
}
|
|
}
|
|
|
|
void TabWidget::indentTabText(int index) {
|
|
#if defined(Q_OS_MACOS)
|
|
if (tabBar()->tabType(index) != TabBar::TabType::FeedReader && !tabIcon(index).isNull()) {
|
|
// We have closable tab with some icon, fix the title.
|
|
const QString text = tabText(index);
|
|
|
|
if (!text.startsWith(QSL(" "))) {
|
|
setTabText(index, QSL(" ") + text);
|
|
}
|
|
}
|
|
#else
|
|
Q_UNUSED(index)
|
|
#endif
|
|
}
|
|
|
|
void TabWidget::removeTab(int index, bool clear_from_memory) {
|
|
if (clear_from_memory) {
|
|
widget(index)->deleteLater();
|
|
}
|
|
|
|
QTabWidget::removeTab(index);
|
|
}
|
|
|
|
int TabWidget::addTab(TabContent* widget, const QIcon& icon, const QString& label, TabBar::TabType type) {
|
|
const int index = QTabWidget::addTab(widget, icon, TextFactory::shorten(label));
|
|
|
|
tabBar()->setTabType(index, type);
|
|
indentTabText(index);
|
|
return index;
|
|
}
|
|
|
|
int TabWidget::addTab(TabContent* widget, const QString& label, TabBar::TabType type) {
|
|
const int index = QTabWidget::addTab(widget, TextFactory::shorten(label));
|
|
|
|
tabBar()->setTabType(index, type);
|
|
indentTabText(index);
|
|
return index;
|
|
}
|
|
|
|
int TabWidget::insertTab(int index, QWidget* widget, const QIcon& icon, const QString& label, TabBar::TabType type) {
|
|
const int tab_index = QTabWidget::insertTab(index, widget, icon, label);
|
|
|
|
tabBar()->setTabType(tab_index, type);
|
|
indentTabText(index);
|
|
return tab_index;
|
|
}
|
|
|
|
int TabWidget::insertTab(int index, QWidget* widget, const QString& label, TabBar::TabType type) {
|
|
const int tab_index = QTabWidget::insertTab(index, widget, label);
|
|
|
|
tabBar()->setTabType(tab_index, type);
|
|
indentTabText(index);
|
|
return tab_index;
|
|
}
|
|
|
|
void TabWidget::changeIcon(int index, const QIcon& new_icon) {
|
|
setTabIcon(index, new_icon);
|
|
indentTabText(index);
|
|
}
|
|
|
|
void TabWidget::changeTitle(int index, const QString& new_title) {
|
|
setTabText(index, TextFactory::shorten(new_title));
|
|
setTabToolTip(index, TextFactory::shorten(new_title));
|
|
indentTabText(index);
|
|
}
|
|
|
|
void TabWidget::fixContentsAfterMove(int from, int to) {
|
|
from = qMin(from, to);
|
|
to = qMax(from, to);
|
|
|
|
for (; from <= to; from++) {
|
|
auto* content = widget(from);
|
|
|
|
content->setIndex(from);
|
|
}
|
|
}
|