This commit is contained in:
fourtf 2018-05-23 04:22:17 +02:00
parent bf560d37bd
commit dafbda6a4a
22 changed files with 404 additions and 1103 deletions

View file

@ -188,7 +188,7 @@ int TextLayoutElement::getMouseOverIndex(const QPoint &abs)
auto app = getApp(); auto app = getApp();
QFontMetrics &metrics = app->fonts->getFontMetrics(this->style, this->scale); QFontMetrics metrics = app->fonts->getFontMetrics(this->style, this->scale);
int x = this->getRect().left(); int x = this->getRect().left();
@ -209,7 +209,7 @@ int TextLayoutElement::getXFromIndex(int index)
{ {
auto app = getApp(); auto app = getApp();
QFontMetrics &metrics = app->fonts->getFontMetrics(this->style, this->scale); QFontMetrics metrics = app->fonts->getFontMetrics(this->style, this->scale);
if (index <= 0) { if (index <= 0) {
return this->getRect().left(); return this->getRect().left();

View file

@ -139,7 +139,7 @@ void TextElement::addToContainer(MessageLayoutContainer &container, MessageEleme
auto app = getApp(); auto app = getApp();
if (_flags & this->getFlags()) { if (_flags & this->getFlags()) {
QFontMetrics &metrics = app->fonts->getFontMetrics(this->style, container.getScale()); QFontMetrics metrics = app->fonts->getFontMetrics(this->style, container.getScale());
for (Word &word : this->words) { for (Word &word : this->words) {
auto getTextLayoutElement = [&](QString text, int width, bool trailingSpace) { auto getTextLayoutElement = [&](QString text, int width, bool trailingSpace) {
@ -242,7 +242,7 @@ TextElement *TimestampElement::formatTime(const QTime &time)
QString format = locale.toString(time, getApp()->settings->timestampFormat); QString format = locale.toString(time, getApp()->settings->timestampFormat);
return new TextElement(format, Flags::Timestamp, MessageColor::System, FontStyle::Medium); return new TextElement(format, Flags::Timestamp, MessageColor::System, FontStyle::ChatMedium);
} }
// TWITCH MODERATION // TWITCH MODERATION

View file

@ -158,7 +158,7 @@ class TextElement : public MessageElement
public: public:
TextElement(const QString &text, MessageElement::Flags flags, TextElement(const QString &text, MessageElement::Flags flags,
const MessageColor &color = MessageColor::Text, const MessageColor &color = MessageColor::Text,
FontStyle style = FontStyle::Medium); FontStyle style = FontStyle::ChatMedium);
~TextElement() override = default; ~TextElement() override = default;
void addToContainer(MessageLayoutContainer &container, MessageElement::Flags flags) override; void addToContainer(MessageLayoutContainer &container, MessageElement::Flags flags) override;

View file

@ -361,14 +361,14 @@ void TwitchMessageBuilder::appendUsername()
} else if (this->args.isReceivedWhisper) { } else if (this->args.isReceivedWhisper) {
// Sender username // Sender username
this->emplace<TextElement>(usernameText, MessageElement::Text, this->usernameColor, this->emplace<TextElement>(usernameText, MessageElement::Text, this->usernameColor,
FontStyle::MediumBold) FontStyle::ChatMediumBold)
->setLink({Link::UserInfo, this->userName}); ->setLink({Link::UserInfo, this->userName});
auto currentUser = app->accounts->Twitch.getCurrent(); auto currentUser = app->accounts->Twitch.getCurrent();
// Separator // Separator
this->emplace<TextElement>("->", MessageElement::Text, this->emplace<TextElement>("->", MessageElement::Text,
app->themes->messages.textColors.system, FontStyle::Medium); app->themes->messages.textColors.system, FontStyle::ChatMedium);
QColor selfColor = currentUser->color; QColor selfColor = currentUser->color;
if (!selfColor.isValid()) { if (!selfColor.isValid()) {
@ -377,14 +377,14 @@ void TwitchMessageBuilder::appendUsername()
// Your own username // Your own username
this->emplace<TextElement>(currentUser->getUserName() + ":", MessageElement::Text, this->emplace<TextElement>(currentUser->getUserName() + ":", MessageElement::Text,
selfColor, FontStyle::MediumBold); selfColor, FontStyle::ChatMediumBold);
} else { } else {
if (!this->action) { if (!this->action) {
usernameText += ":"; usernameText += ":";
} }
this->emplace<TextElement>(usernameText, MessageElement::Text, this->usernameColor, this->emplace<TextElement>(usernameText, MessageElement::Text, this->usernameColor,
FontStyle::MediumBold) FontStyle::ChatMediumBold)
->setLink({Link::UserInfo, this->userName}); ->setLink({Link::UserInfo, this->userName});
} }
} }

View file

@ -3,6 +3,8 @@
#include <QDebug> #include <QDebug>
#include <QtGlobal> #include <QtGlobal>
#include "util/assertinguithread.hpp"
#ifdef Q_OS_WIN32 #ifdef Q_OS_WIN32
#define DEFAULT_FONT_FAMILY "Segoe UI" #define DEFAULT_FONT_FAMILY "Segoe UI"
#define DEFAULT_FONT_SIZE 10 #define DEFAULT_FONT_SIZE 10
@ -20,86 +22,110 @@ namespace chatterino {
namespace singletons { namespace singletons {
FontManager::FontManager() FontManager::FontManager()
: currentFontFamily("/appearance/currentFontFamily", DEFAULT_FONT_FAMILY) : chatFontFamily("/appearance/currentFontFamily", DEFAULT_FONT_FAMILY)
, currentFontSize("/appearance/currentFontSize", DEFAULT_FONT_SIZE) , chatFontSize("/appearance/currentFontSize", DEFAULT_FONT_SIZE)
// , currentFont(this->currentFontFamily.getValue().c_str(), currentFontSize.getValue())
{ {
qDebug() << "init FontManager"; qDebug() << "init FontManager";
this->currentFontFamily.connect([this](const std::string &newValue, auto) { this->chatFontFamily.connect([this](const std::string &newValue, auto) {
util::assertInGuiThread();
this->incGeneration(); this->incGeneration();
// this->currentFont.setFamily(newValue.c_str()); for (auto &map : this->fontsByType) {
this->currentFontByScale.clear(); map.clear();
}
this->fontChanged.invoke(); this->fontChanged.invoke();
}); });
this->currentFontSize.connect([this](const int &newValue, auto) { this->chatFontSize.connect([this](const int &newValue, auto) {
util::assertInGuiThread();
this->incGeneration(); this->incGeneration();
// this->currentFont.setSize(newValue); for (auto &map : this->fontsByType) {
this->currentFontByScale.clear(); map.clear();
}
this->fontChanged.invoke(); this->fontChanged.invoke();
}); });
this->fontsByType.resize((size_t)EndType);
} }
QFont &FontManager::getFont(FontManager::Type type, float scale) QFont FontManager::getFont(FontManager::Type type, float scale)
{ {
// return this->currentFont.getFont(type); return this->getOrCreateFontData(type, scale).font;
return this->getCurrentFont(scale).getFont(type);
} }
QFontMetrics &FontManager::getFontMetrics(FontManager::Type type, float scale) QFontMetrics FontManager::getFontMetrics(FontManager::Type type, float scale)
{ {
// return this->currentFont.getFontMetrics(type); return this->getOrCreateFontData(type, scale).metrics;
return this->getCurrentFont(scale).getFontMetrics(type);
} }
FontManager::FontData &FontManager::Font::getFontData(FontManager::Type type) int FontManager::getGeneration() const
{ {
switch (type) { return this->generation;
case Tiny:
return this->tiny;
case Small:
return this->small;
case MediumSmall:
return this->mediumSmall;
case Medium:
return this->medium;
case MediumBold:
return this->mediumBold;
case MediumItalic:
return this->mediumItalic;
case Large:
return this->large;
case VeryLarge:
return this->veryLarge;
default:
qDebug() << "Unknown font type:" << type << ", defaulting to medium";
return this->medium;
}
} }
QFont &FontManager::Font::getFont(Type type) void FontManager::incGeneration()
{ {
return this->getFontData(type).font; this->generation++;
} }
QFontMetrics &FontManager::Font::getFontMetrics(Type type) FontManager::FontData &FontManager::getOrCreateFontData(Type type, float scale)
{ {
return this->getFontData(type).metrics; util::assertInGuiThread();
}
FontManager::Font &FontManager::getCurrentFont(float scale) assert(type >= 0 && type < EndType);
{
for (auto it = this->currentFontByScale.begin(); it != this->currentFontByScale.end(); it++) { auto &map = this->fontsByType[(size_t)type];
if (it->first == scale) {
// find element
auto it = map.find(scale);
if (it != map.end()) {
// return if found
qDebug() << it->second.font;
return it->second; return it->second;
} }
}
this->currentFontByScale.push_back(
std::make_pair(scale, Font(this->currentFontFamily.getValue().c_str(),
this->currentFontSize.getValue() * scale)));
return this->currentFontByScale.back().second; // emplace new element
auto result = map.emplace(scale, this->createFontData(type, scale));
assert(result.second);
return result.first->second;
}
FontManager::FontData FontManager::createFontData(Type type, float scale)
{
// check if it's a chat (scale the setting)
if (type >= ChatStart && type <= ChatEnd) {
static std::unordered_map<Type, ChatFontData> sizeScale{
{ChatSmall, {0.6f, false, QFont::Normal}},
{ChatMediumSmall, {0.8f, false, QFont::Normal}},
{ChatMedium, {1, false, QFont::Normal}},
{ChatMediumBold, {1, false, QFont::Medium}},
{ChatMediumItalic, {1, true, QFont::Normal}},
{ChatLarge, {1.2f, false, QFont::Normal}},
{ChatVeryLarge, {1.4f, false, QFont::Normal}},
};
auto data = sizeScale[type];
return FontData(QFont(QString::fromStdString(this->chatFontFamily.getValue()),
this->chatFontSize.getValue() * data.scale * scale, data.weight,
data.italic));
}
// normal Ui font (use pt size)
{
static std::unordered_map<Type, UiFontData> defaultSize{
{Tiny, {8, "Monospace", false, QFont::Normal}},
{UiMedium, {12, DEFAULT_FONT_FAMILY, false, QFont::Normal}},
{UiTabs, {9, "Segoe UI", false, QFont::Normal}},
};
UiFontData &data = defaultSize[type];
QFont font(data.name, data.size * scale, data.weight, data.italic);
return FontData(font);
}
} }
} // namespace singletons } // namespace singletons

View file

@ -3,136 +3,82 @@
#include <QFont> #include <QFont>
#include <QFontDatabase> #include <QFontDatabase>
#include <QFontMetrics> #include <QFontMetrics>
#include <array>
#include <boost/noncopyable.hpp>
#include <pajlada/settings/setting.hpp> #include <pajlada/settings/setting.hpp>
#include <pajlada/signals/signal.hpp> #include <pajlada/signals/signal.hpp>
#include <unordered_map>
namespace chatterino { namespace chatterino {
namespace singletons { namespace singletons {
class FontManager class FontManager : boost::noncopyable
{ {
public: public:
FontManager(); FontManager();
FontManager(const FontManager &) = delete; // font data gets set in createFontData(...)
FontManager(FontManager &&) = delete;
~FontManager() = delete;
enum Type : uint8_t { enum Type : uint8_t {
Tiny, Tiny,
Small, ChatSmall,
MediumSmall, ChatMediumSmall,
Medium, ChatMedium,
MediumBold, ChatMediumBold,
MediumItalic, ChatMediumItalic,
Large, ChatLarge,
VeryLarge, ChatVeryLarge,
UiMedium,
UiTabs,
// don't remove this value
EndType,
// make sure to update these values accordingly!
ChatStart = ChatSmall,
ChatEnd = ChatVeryLarge,
}; };
QFont &getFont(Type type, float scale); QFont getFont(Type type, float scale);
QFontMetrics &getFontMetrics(Type type, float scale); QFontMetrics getFontMetrics(Type type, float scale);
int getGeneration() const int getGeneration() const;
{ void incGeneration();
return this->generation;
}
void incGeneration() pajlada::Settings::Setting<std::string> chatFontFamily;
{ pajlada::Settings::Setting<int> chatFontSize;
this->generation++;
}
pajlada::Settings::Setting<std::string> currentFontFamily;
pajlada::Settings::Setting<int> currentFontSize;
pajlada::Signals::NoArgSignal fontChanged; pajlada::Signals::NoArgSignal fontChanged;
private: private:
struct FontData { struct FontData {
FontData(QFont &&_font) FontData(const QFont &_font)
: font(_font) : font(_font)
, metrics(this->font) , metrics(_font)
{ {
} }
QFont font; const QFont font;
QFontMetrics metrics; const QFontMetrics metrics;
}; };
struct Font { struct ChatFontData {
Font() = delete; float scale;
bool italic;
Font(const char *fontFamilyName, int mediumSize) QFont::Weight weight;
: tiny(QFont("Monospace", 8))
, small(QFont(fontFamilyName, mediumSize - 4))
, mediumSmall(QFont(fontFamilyName, mediumSize - 2))
, medium(QFont(fontFamilyName, mediumSize))
, mediumBold(QFont(fontFamilyName, mediumSize, QFont::DemiBold))
, mediumItalic(QFont(fontFamilyName, mediumSize, -1, true))
, large(QFont(fontFamilyName, mediumSize))
, veryLarge(QFont(fontFamilyName, mediumSize))
{
tiny.font.setStyleHint(QFont::TypeWriter);
}
void setFamily(const char *newFamily)
{
this->small.font.setFamily(newFamily);
this->mediumSmall.font.setFamily(newFamily);
this->medium.font.setFamily(newFamily);
this->mediumBold.font.setFamily(newFamily);
this->mediumItalic.font.setFamily(newFamily);
this->large.font.setFamily(newFamily);
this->veryLarge.font.setFamily(newFamily);
this->updateMetrics();
}
void setSize(int newMediumSize)
{
this->small.font.setPointSize(newMediumSize - 4);
this->mediumSmall.font.setPointSize(newMediumSize - 2);
this->medium.font.setPointSize(newMediumSize);
this->mediumBold.font.setPointSize(newMediumSize);
this->mediumItalic.font.setPointSize(newMediumSize);
this->large.font.setPointSize(newMediumSize + 2);
this->veryLarge.font.setPointSize(newMediumSize + 4);
this->updateMetrics();
}
void updateMetrics()
{
this->small.metrics = QFontMetrics(this->small.font);
this->mediumSmall.metrics = QFontMetrics(this->mediumSmall.font);
this->medium.metrics = QFontMetrics(this->medium.font);
this->mediumBold.metrics = QFontMetrics(this->mediumBold.font);
this->mediumItalic.metrics = QFontMetrics(this->mediumItalic.font);
this->large.metrics = QFontMetrics(this->large.font);
this->veryLarge.metrics = QFontMetrics(this->veryLarge.font);
}
FontData &getFontData(Type type);
QFont &getFont(Type type);
QFontMetrics &getFontMetrics(Type type);
FontData tiny;
FontData small;
FontData mediumSmall;
FontData medium;
FontData mediumBold;
FontData mediumItalic;
FontData large;
FontData veryLarge;
}; };
Font &getCurrentFont(float scale); struct UiFontData {
float size;
const char *name;
bool italic;
QFont::Weight weight;
};
// Future plans: FontData &getOrCreateFontData(Type type, float scale);
// Could have multiple fonts in here, such as "Menu font", "Application font", "Chat font" FontData createFontData(Type type, float scale);
std::list<std::pair<float, Font>> currentFontByScale; std::vector<std::unordered_map<float, FontData>> fontsByType;
int generation = 0; int generation = 0;
}; };

View file

@ -52,7 +52,6 @@ void ThemeManager::actuallyUpdate(double hue, double multiplier)
isLight = multiplier > 0; isLight = multiplier > 0;
bool lightWin = isLight; bool lightWin = isLight;
QColor none(0, 0, 0, 0);
QColor themeColor = QColor::fromHslF(hue, 0.43, 0.5); QColor themeColor = QColor::fromHslF(hue, 0.43, 0.5);
QColor themeColorNoSat = QColor::fromHslF(hue, 0, 0.5); QColor themeColorNoSat = QColor::fromHslF(hue, 0, 0.5);
@ -69,7 +68,7 @@ void ThemeManager::actuallyUpdate(double hue, double multiplier)
#ifdef Q_OS_LINUX #ifdef Q_OS_LINUX
this->window.background = lightWin ? "#fff" : QColor(61, 60, 56); this->window.background = lightWin ? "#fff" : QColor(61, 60, 56);
#else #else
this->window.background = lightWin ? "#fff" : "#444"; this->window.background = lightWin ? "#fff" : "#111";
#endif #endif
QColor fg = this->window.text = lightWin ? "#000" : "#eee"; QColor fg = this->window.text = lightWin ? "#000" : "#eee";
@ -89,27 +88,48 @@ void ThemeManager::actuallyUpdate(double hue, double multiplier)
/// TABS /// TABS
if (lightWin) { if (lightWin) {
this->tabs.regular = {fg, {bg, QColor("#ccc"), bg}}; this->tabs.regular = {QColor("#444"),
{QColor("#fff"), QColor("#fff"), QColor("#fff")},
{QColor("#fff"), QColor("#fff"), QColor("#fff")}};
this->tabs.newMessage = { this->tabs.newMessage = {
fg, fg, {bg, QColor("#ccc"), bg}, {QColor("#aaa"), QColor("#aaa"), QColor("#aaa")}};
{QBrush(blendColors(themeColor, "#ccc", 0.9), Qt::FDiagPattern), this->tabs.highlighted = {fg,
QBrush(blendColors(themeColor, "#ccc", 0.9), Qt::FDiagPattern), {bg, QColor("#ccc"), bg},
QBrush(blendColors(themeColorNoSat, "#ccc", 0.9), Qt::FDiagPattern)}}; {QColor("#b60505"), QColor("#b60505"), QColor("#b60505")}};
this->tabs.highlighted = {fg, {QColor("#ccc"), QColor("#ccc"), QColor("#bbb")}};
this->tabs.selected = {QColor("#fff"),
{QColor("#777"), QColor("#777"), QColor("#888")}};
} else {
this->tabs.regular = {fg, {bg, QColor("#555"), bg}};
this->tabs.newMessage = {
fg,
{QBrush(blendColors(themeColor, "#666", 0.7), Qt::FDiagPattern),
QBrush(blendColors(themeColor, "#666", 0.5), Qt::FDiagPattern),
QBrush(blendColors(themeColorNoSat, "#666", 0.7), Qt::FDiagPattern)}};
this->tabs.highlighted = {fg, {QColor("#777"), QColor("#777"), QColor("#666")}};
this->tabs.selected = {QColor("#000"), this->tabs.selected = {QColor("#000"),
{QColor("#999"), QColor("#999"), QColor("#888")}}; {QColor("#b4d7ff"), QColor("#b4d7ff"), QColor("#b4d7ff")},
{QColor("#00aeef"), QColor("#00aeef"), QColor("#00aeef")}};
} else {
this->tabs.regular = {QColor("#aaa"),
{QColor("#252525"), QColor("#252525"), QColor("#252525")},
{QColor("#444"), QColor("#444"), QColor("#444")}};
this->tabs.newMessage = {fg,
{QColor("#252525"), QColor("#252525"), QColor("#252525")},
{QColor("#888"), QColor("#888"), QColor("#888")}};
this->tabs.highlighted = {fg,
{QColor("#252525"), QColor("#252525"), QColor("#252525")},
{QColor("#ee6166"), QColor("#ee6166"), QColor("#ee6166")}};
this->tabs.selected = {QColor("#fff"),
{QColor("#555555"), QColor("#555555"), QColor("#555555")},
{QColor("#00aeef"), QColor("#00aeef"), QColor("#00aeef")}};
} }
// this->tabs.newMessage = {
// fg,
// {QBrush(blendColors(themeColor, "#ccc", 0.9), Qt::FDiagPattern),
// QBrush(blendColors(themeColor, "#ccc", 0.9), Qt::FDiagPattern),
// QBrush(blendColors(themeColorNoSat, "#ccc", 0.9), Qt::FDiagPattern)}};
// this->tabs.newMessage = {
// fg,
// {QBrush(blendColors(themeColor, "#666", 0.7), Qt::FDiagPattern),
// QBrush(blendColors(themeColor, "#666", 0.5), Qt::FDiagPattern),
// QBrush(blendColors(themeColorNoSat, "#666", 0.7),
// Qt::FDiagPattern)}};
// this->tabs.highlighted = {fg, {QColor("#777"), QColor("#777"),
// QColor("#666")}};
this->tabs.bottomLine = this->tabs.selected.backgrounds.regular.color(); this->tabs.bottomLine = this->tabs.selected.backgrounds.regular.color();
} }
@ -163,7 +183,7 @@ void ThemeManager::actuallyUpdate(double hue, double multiplier)
this->messages.selection = isLightTheme() ? QColor(0, 0, 0, 64) : QColor(255, 255, 255, 64); this->messages.selection = isLightTheme() ? QColor(0, 0, 0, 64) : QColor(255, 255, 255, 64);
this->updated.invoke(); this->updated.invoke();
} } // namespace singletons
QColor ThemeManager::blendColors(const QColor &color1, const QColor &color2, qreal ratio) QColor ThemeManager::blendColors(const QColor &color1, const QColor &color2, qreal ratio)
{ {

View file

@ -25,11 +25,16 @@ public:
struct TabColors { struct TabColors {
QColor text; QColor text;
struct Backgrounds { struct {
QBrush regular; QBrush regular;
QBrush hover; QBrush hover;
QBrush unfocused; QBrush unfocused;
} backgrounds; } backgrounds;
struct {
QColor regular;
QColor hover;
QColor unfocused;
} line;
}; };
/// WINDOW /// WINDOW
@ -43,9 +48,9 @@ public:
/// TABS /// TABS
struct { struct {
TabColors regular; TabColors regular;
TabColors selected;
TabColors highlighted;
TabColors newMessage; TabColors newMessage;
TabColors highlighted;
TabColors selected;
QColor border; QColor border;
QColor bottomLine; QColor bottomLine;
} tabs; } tabs;

View file

@ -188,27 +188,27 @@ void WindowManager::initialize()
// load tabs // load tabs
QJsonArray tabs = window_obj.value("tabs").toArray(); QJsonArray tabs = window_obj.value("tabs").toArray();
for (QJsonValue tab_val : tabs) { for (QJsonValue tab_val : tabs) {
widgets::SplitContainer *tab = window.getNotebook().addNewPage(); widgets::SplitContainer *page = window.getNotebook().addPage(false);
QJsonObject tab_obj = tab_val.toObject(); QJsonObject tab_obj = tab_val.toObject();
// set custom title // set custom title
QJsonValue title_val = tab_obj.value("title"); QJsonValue title_val = tab_obj.value("title");
if (title_val.isString()) { if (title_val.isString()) {
tab->getTab()->setTitle(title_val.toString()); page->getTab()->setTitle(title_val.toString());
tab->getTab()->useDefaultTitle = false; page->getTab()->useDefaultTitle = false;
} }
// selected // selected
if (tab_obj.value("selected").toBool(false)) { if (tab_obj.value("selected").toBool(false)) {
window.getNotebook().select(tab); window.getNotebook().select(page);
} }
// load splits // load splits
QJsonObject splitRoot = tab_obj.value("splits2").toObject(); QJsonObject splitRoot = tab_obj.value("splits2").toObject();
if (!splitRoot.isEmpty()) { if (!splitRoot.isEmpty()) {
tab->decodeFromJson(splitRoot); page->decodeFromJson(splitRoot);
continue; continue;
} }
@ -217,12 +217,12 @@ void WindowManager::initialize()
int colNr = 0; int colNr = 0;
for (QJsonValue column_val : tab_obj.value("splits").toArray()) { for (QJsonValue column_val : tab_obj.value("splits").toArray()) {
for (QJsonValue split_val : column_val.toArray()) { for (QJsonValue split_val : column_val.toArray()) {
widgets::Split *split = new widgets::Split(tab); widgets::Split *split = new widgets::Split(page);
QJsonObject split_obj = split_val.toObject(); QJsonObject split_obj = split_val.toObject();
split->setChannel(decodeChannel(split_obj)); split->setChannel(decodeChannel(split_obj));
tab->appendSplit(split); page->appendSplit(split);
} }
colNr++; colNr++;
} }
@ -231,7 +231,7 @@ void WindowManager::initialize()
if (mainWindow == nullptr) { if (mainWindow == nullptr) {
mainWindow = &createWindow(widgets::Window::Main); mainWindow = &createWindow(widgets::Window::Main);
mainWindow->getNotebook().addNewPage(true); mainWindow->getNotebook().addPage(true);
} }
this->initialized = true; this->initialized = true;
@ -268,9 +268,11 @@ void WindowManager::save()
// window tabs // window tabs
QJsonArray tabs_arr; QJsonArray tabs_arr;
for (int tab_i = 0; tab_i < window->getNotebook().tabCount(); tab_i++) { for (int tab_i = 0; tab_i < window->getNotebook().getPageCount(); tab_i++) {
QJsonObject tab_obj; QJsonObject tab_obj;
widgets::SplitContainer *tab = window->getNotebook().tabAt(tab_i); widgets::SplitContainer *tab =
dynamic_cast<widgets::SplitContainer *>(window->getNotebook().getPageAt(tab_i));
assert(tab != nullptr);
// custom tab title // custom tab title
if (!tab->getTab()->useDefaultTitle) { if (!tab->getTab()->useDefaultTitle) {

View file

@ -27,7 +27,7 @@ protected:
private: private:
QSize preferedSize; QSize preferedSize;
QString text; QString text;
FontStyle fontStyle = FontStyle::Medium; FontStyle fontStyle = FontStyle::ChatMedium;
}; };
} // namespace widgets } // namespace widgets

View file

@ -136,10 +136,12 @@ void NotebookButton::dropEvent(QDropEvent *event)
if (SplitContainer::isDraggingSplit) { if (SplitContainer::isDraggingSplit) {
event->acceptProposedAction(); event->acceptProposedAction();
Notebook *notebook = dynamic_cast<Notebook *>(this->parentWidget()); Notebook2 *notebook = dynamic_cast<Notebook2 *>(this->parentWidget());
if (notebook != nuuls) { if (notebook != nuuls) {
SplitContainer *page = notebook->addNewPage(); SplitContainer *page = new SplitContainer(notebook);
auto *tab = notebook->addPage(page);
page->setTab(tab);
SplitContainer::draggingSplit->setParent(page); SplitContainer::draggingSplit->setParent(page);
page->appendSplit(SplitContainer::draggingSplit); page->appendSplit(SplitContainer::draggingSplit);

View file

@ -61,17 +61,18 @@ NotebookTab2::NotebookTab2(Notebook2 *_notebook)
} }
}); });
QAction *enableHighlightsOnNewMessageAction = // QAction *enableHighlightsOnNewMessageAction =
new QAction("Enable highlights on new message", &this->menu); // new QAction("Enable highlights on new message", &this->menu);
enableHighlightsOnNewMessageAction->setCheckable(true); // enableHighlightsOnNewMessageAction->setCheckable(true);
this->menu.addAction("Close", [=]() { this->notebook->removePage(this->page); }); this->menu.addAction("Close", [=]() { this->notebook->removePage(this->page); });
this->menu.addAction(enableHighlightsOnNewMessageAction); // this->menu.addAction(enableHighlightsOnNewMessageAction);
QObject::connect(enableHighlightsOnNewMessageAction, &QAction::toggled, [this](bool newValue) { // QObject::connect(enableHighlightsOnNewMessageAction, &QAction::toggled, [this](bool
debug::Log("New value is {}", newValue); // // newValue) {
}); // debug::Log("New value is {}", newValue); //
// });
} }
void NotebookTab2::themeRefreshEvent() void NotebookTab2::themeRefreshEvent()
@ -85,15 +86,20 @@ void NotebookTab2::updateSize()
float scale = getScale(); float scale = getScale();
int width; int width;
QFontMetrics metrics(this->font()); QFontMetrics metrics = getApp()->fonts->getFontMetrics(FontStyle::UiTabs, this->getScale());
if (!app->settings->showTabCloseButton) { if (this->hasXButton()) {
width = (int)((metrics.width(this->title) + 16 /*+ 16*/) * scale); width = (int)((metrics.width(this->title) + 32) * scale);
} else { } else {
width = (int)((metrics.width(this->title) + 8 + 24 /*+ 16*/) * scale); width = (int)((metrics.width(this->title) + 16) * scale);
} }
this->resize(std::min((int)(150 * scale), width), (int)(24 * scale)); width = std::min((int)(150 * scale), width);
if (this->width() != width) {
this->resize(width, (int)(NOTEBOOK_TAB_HEIGHT * scale));
this->notebook->performLayout();
}
// if (this->parent() != nullptr) { // if (this->parent() != nullptr) {
// (static_cast<Notebook2 *>(this->parent()))->performLayout(true); // (static_cast<Notebook2 *>(this->parent()))->performLayout(true);
@ -182,7 +188,9 @@ void NotebookTab2::paintEvent(QPaintEvent *)
QPainter painter(this); QPainter painter(this);
float scale = this->getScale(); float scale = this->getScale();
int height = (int)(scale * 24); painter.setFont(getApp()->fonts->getFont(FontStyle::UiTabs, scale));
int height = (int)(scale * NOTEBOOK_TAB_HEIGHT);
// int fullHeight = (int)(scale * 48); // int fullHeight = (int)(scale * 48);
// select the right tab colors // select the right tab colors
@ -206,7 +214,6 @@ void NotebookTab2::paintEvent(QPaintEvent *)
: (windowFocused ? colors.backgrounds.regular : (windowFocused ? colors.backgrounds.regular
: colors.backgrounds.unfocused); : colors.backgrounds.unfocused);
if (true) {
painter.fillRect(rect(), this->mouseOver ? regular.backgrounds.hover painter.fillRect(rect(), this->mouseOver ? regular.backgrounds.hover
: (windowFocused ? regular.backgrounds.regular : (windowFocused ? regular.backgrounds.regular
: regular.backgrounds.unfocused)); : regular.backgrounds.unfocused));
@ -215,39 +222,20 @@ void NotebookTab2::paintEvent(QPaintEvent *)
painter.fillRect(rect(), tabBackground); painter.fillRect(rect(), tabBackground);
// draw border // draw border
// painter.setPen(QPen("#ccc")); // painter.setPen(QPen("#fff"));
// QPainterPath path(QPointF(0, height)); // QPainterPath path(QPointF(0, height));
// path.lineTo(0, 0); // path.lineTo(0, 0);
// path.lineTo(this->width() - 1, 0); // path.lineTo(this->width() - 1, 0);
// path.lineTo(this->width() - 1, this->height() - 1); // path.lineTo(this->width() - 1, this->height() - 1);
// path.lineTo(0, this->height() - 1); // path.lineTo(0, this->height() - 1);
// painter.drawPath(path); // painter.drawPath(path);
} else {
// QPainterPath path(QPointF(0, height));
// path.lineTo(8 * scale, 0);
// path.lineTo(this->width() - 8 * scale, 0);
// path.lineTo(this->width(), height);
// painter.fillPath(path, this->mouseOver ? regular.backgrounds.hover
// : (windowFocused ?
// regular.backgrounds.regular
// :
// regular.backgrounds.unfocused));
// // fill the tab background // top line
// painter.fillPath(path, tabBackground); painter.fillRect(QRectF(0, (this->selected ? 0.f : 1.f) * scale, this->width(),
// painter.setPen(QColor("#FFF")); (this->selected ? 2.f : 1.f) * scale),
// painter.setRenderHint(QPainter::Antialiasing); this->mouseOver
// painter.drawPath(path); ? colors.line.hover
// // painter.setBrush(QColor("#000")); : (windowFocused ? colors.line.regular : colors.line.unfocused));
// QLinearGradient gradient(0, height, 0, fullHeight);
// gradient.setColorAt(0, tabBackground.color());
// gradient.setColorAt(1, "#fff");
// QBrush brush(gradient);
// painter.fillRect(0, height, this->width(), fullHeight - height,
// brush);
}
// set the pen color // set the pen color
painter.setPen(colors.text); painter.setPen(colors.text);
@ -260,7 +248,11 @@ void NotebookTab2::paintEvent(QPaintEvent *)
if (true) { // legacy if (true) { // legacy
// painter.drawText(rect, this->getTitle(), QTextOption(Qt::AlignCenter)); // painter.drawText(rect, this->getTitle(), QTextOption(Qt::AlignCenter));
int offset = (int)(scale * 8); int offset = (int)(scale * 8);
QRect textRect(offset, 0, this->width() - offset - offset, height); QRect textRect(offset, this->selected ? 0 : 1, this->width() - offset - offset, height);
if (this->shouldDrawXButton()) {
textRect.setRight(textRect.right() - this->height() / 2);
}
QTextOption option(Qt::AlignLeft | Qt::AlignVCenter); QTextOption option(Qt::AlignLeft | Qt::AlignVCenter);
option.setWrapMode(QTextOption::NoWrap); option.setWrapMode(QTextOption::NoWrap);
@ -274,9 +266,11 @@ void NotebookTab2::paintEvent(QPaintEvent *)
} }
// draw close x // draw close x
if (!app->settings->showTabCloseButton && (mouseOver || selected)) { if (this->shouldDrawXButton()) {
QRect xRect = this->getXRect(); QRect xRect = this->getXRect();
if (!xRect.isNull()) { if (!xRect.isNull()) {
painter.setBrush(QColor("#fff"));
if (mouseOverX) { if (mouseOverX) {
painter.fillRect(xRect, QColor(0, 0, 0, 64)); painter.fillRect(xRect, QColor(0, 0, 0, 64));
@ -291,6 +285,21 @@ void NotebookTab2::paintEvent(QPaintEvent *)
painter.drawLine(xRect.topRight() + QPoint(-a, a), xRect.bottomLeft() + QPoint(a, -a)); painter.drawLine(xRect.topRight() + QPoint(-a, a), xRect.bottomLeft() + QPoint(a, -a));
} }
} }
// draw line at bottom
if (!this->selected) {
painter.fillRect(0, this->height() - 1, this->width(), 1, app->themes->window.background);
}
}
bool NotebookTab2::hasXButton()
{
return getApp()->settings->showTabCloseButton && this->notebook->getAllowUserTabManagement();
}
bool NotebookTab2::shouldDrawXButton()
{
return this->hasXButton() && (mouseOver || selected);
} }
void NotebookTab2::mousePressEvent(QMouseEvent *event) void NotebookTab2::mousePressEvent(QMouseEvent *event)
@ -320,8 +329,7 @@ void NotebookTab2::mouseReleaseEvent(QMouseEvent *event)
this->notebook->removePage(this->page); this->notebook->removePage(this->page);
} }
} else { } else {
if (getApp()->settings->showTabCloseButton && this->mouseDownX && if (this->hasXButton() && this->mouseDownX && this->getXRect().contains(event->pos())) {
this->getXRect().contains(event->pos())) {
this->mouseDownX = false; this->mouseDownX = false;
this->notebook->removePage(this->page); this->notebook->removePage(this->page);
@ -377,7 +385,7 @@ void NotebookTab2::mouseMoveEvent(QMouseEvent *event)
int index; int index;
QWidget *clickedPage = notebook->tabAt(relPoint, index, this->width()); QWidget *clickedPage = notebook->tabAt(relPoint, index, this->width());
assert(clickedPage); // assert(clickedPage);
if (clickedPage != nullptr && clickedPage != this->page) { if (clickedPage != nullptr && clickedPage != this->page) {
this->notebook->rearrangePage(this->page, index); this->notebook->rearrangePage(this->page, index);
@ -387,370 +395,14 @@ void NotebookTab2::mouseMoveEvent(QMouseEvent *event)
QRect NotebookTab2::getXRect() QRect NotebookTab2::getXRect()
{ {
if (this->notebook->getAllowUserTabManagement()) { // if (!this->notebook->getAllowUserTabManagement()) {
return QRect(); // return QRect();
} // }
float s = this->getScale(); float s = this->getScale();
return QRect(this->width() - static_cast<int>(20 * s), static_cast<int>(4 * s), return QRect(this->width() - static_cast<int>(20 * s), static_cast<int>(6 * s),
static_cast<int>(16 * s), static_cast<int>(16 * s)); static_cast<int>(16 * s), static_cast<int>(16 * s));
} }
// 2
NotebookTab::NotebookTab(Notebook *_notebook)
: BaseWidget(_notebook)
, positionChangedAnimation(this, "pos")
, notebook(_notebook)
, menu(this)
{
auto app = getApp();
this->setAcceptDrops(true);
this->positionChangedAnimation.setEasingCurve(QEasingCurve(QEasingCurve::InCubic));
app->settings->showTabCloseButton.connect(boost::bind(&NotebookTab::hideTabXChanged, this, _1),
this->managedConnections);
this->setMouseTracking(true);
this->menu.addAction("Rename", [this]() {
TextInputDialog d(this);
d.setWindowTitle("Change tab title (Leave empty for default behaviour)");
if (this->useDefaultTitle) {
d.setText("");
} else {
d.setText(this->getTitle());
d.highlightText();
}
if (d.exec() == QDialog::Accepted) {
QString newTitle = d.getText();
if (newTitle.isEmpty()) {
this->useDefaultTitle = true;
this->page->refreshTabTitle();
} else {
this->useDefaultTitle = false;
this->setTitle(newTitle);
}
}
});
QAction *enableHighlightsOnNewMessageAction =
new QAction("Enable highlights on new message", &this->menu);
enableHighlightsOnNewMessageAction->setCheckable(true);
this->menu.addAction("Close", [=]() { this->notebook->removePage(this->page); });
this->menu.addAction(enableHighlightsOnNewMessageAction);
QObject::connect(enableHighlightsOnNewMessageAction, &QAction::toggled, [](bool newValue) {
debug::Log("New value is {}", newValue); //
});
}
void NotebookTab::themeRefreshEvent()
{
this->update();
}
void NotebookTab::updateSize()
{
auto app = getApp();
float scale = getScale();
int width;
if (!app->settings->showTabCloseButton) {
width = (int)((fontMetrics().width(this->title) + 16 /*+ 16*/) * scale);
} else {
width = (int)((fontMetrics().width(this->title) + 8 + 24 /*+ 16*/) * scale);
}
this->resize(std::min((int)(150 * scale), width), (int)(24 * scale));
if (this->parent() != nullptr) {
(static_cast<Notebook *>(this->parent()))->performLayout(true);
}
}
const QString &NotebookTab::getTitle() const
{
return this->title;
}
void NotebookTab::setTitle(const QString &newTitle)
{
if (this->title != newTitle) {
this->title = newTitle;
this->updateSize();
this->update();
}
}
bool NotebookTab::isSelected() const
{
return this->selected;
}
void NotebookTab::setSelected(bool value)
{
this->selected = value;
this->highlightState = HighlightState::None;
this->update();
}
void NotebookTab::setHighlightState(HighlightState newHighlightStyle)
{
if (this->isSelected()) {
return;
}
if (this->highlightState != HighlightState::Highlighted) {
this->highlightState = newHighlightStyle;
this->update();
}
}
QRect NotebookTab::getDesiredRect() const
{
return QRect(positionAnimationDesiredPoint, size());
}
void NotebookTab::hideTabXChanged(bool)
{
this->updateSize();
this->update();
}
void NotebookTab::moveAnimated(QPoint pos, bool animated)
{
this->positionAnimationDesiredPoint = pos;
QWidget *w = this->window();
if ((w != nullptr && !w->isVisible()) || !animated || !positionChangedAnimationRunning) {
this->move(pos);
this->positionChangedAnimationRunning = true;
return;
}
if (this->positionChangedAnimation.endValue() == pos) {
return;
}
this->positionChangedAnimation.stop();
this->positionChangedAnimation.setDuration(75);
this->positionChangedAnimation.setStartValue(this->pos());
this->positionChangedAnimation.setEndValue(pos);
this->positionChangedAnimation.start();
}
void NotebookTab::paintEvent(QPaintEvent *)
{
auto app = getApp();
QPainter painter(this);
float scale = this->getScale();
int height = (int)(scale * 24);
// int fullHeight = (int)(scale * 48);
// select the right tab colors
singletons::ThemeManager::TabColors colors;
singletons::ThemeManager::TabColors regular = this->themeManager->tabs.regular;
if (this->selected) {
colors = this->themeManager->tabs.selected;
} else if (this->highlightState == HighlightState::Highlighted) {
colors = this->themeManager->tabs.highlighted;
} else if (this->highlightState == HighlightState::NewMessage) {
colors = this->themeManager->tabs.newMessage;
} else {
colors = this->themeManager->tabs.regular;
}
bool windowFocused = this->window() == QApplication::activeWindow();
// || SettingsDialog::getHandle() == QApplication::activeWindow();
QBrush tabBackground = this->mouseOver ? colors.backgrounds.hover
: (windowFocused ? colors.backgrounds.regular
: colors.backgrounds.unfocused);
if (true) {
painter.fillRect(rect(), this->mouseOver ? regular.backgrounds.hover
: (windowFocused ? regular.backgrounds.regular
: regular.backgrounds.unfocused));
// fill the tab background
painter.fillRect(rect(), tabBackground);
// draw border
// painter.setPen(QPen("#ccc"));
// QPainterPath path(QPointF(0, height));
// path.lineTo(0, 0);
// path.lineTo(this->width() - 1, 0);
// path.lineTo(this->width() - 1, this->height() - 1);
// path.lineTo(0, this->height() - 1);
// painter.drawPath(path);
} else {
// QPainterPath path(QPointF(0, height));
// path.lineTo(8 * scale, 0);
// path.lineTo(this->width() - 8 * scale, 0);
// path.lineTo(this->width(), height);
// painter.fillPath(path, this->mouseOver ? regular.backgrounds.hover
// : (windowFocused ?
// regular.backgrounds.regular
// :
// regular.backgrounds.unfocused));
// // fill the tab background
// painter.fillPath(path, tabBackground);
// painter.setPen(QColor("#FFF"));
// painter.setRenderHint(QPainter::Antialiasing);
// painter.drawPath(path);
// // painter.setBrush(QColor("#000"));
// QLinearGradient gradient(0, height, 0, fullHeight);
// gradient.setColorAt(0, tabBackground.color());
// gradient.setColorAt(1, "#fff");
// QBrush brush(gradient);
// painter.fillRect(0, height, this->width(), fullHeight - height,
// brush);
}
// set the pen color
painter.setPen(colors.text);
// set area for text
int rectW = (!app->settings->showTabCloseButton ? 0 : static_cast<int>(16) * scale);
QRect rect(0, 0, this->width() - rectW, height);
// draw text
if (true) { // legacy
// painter.drawText(rect, this->getTitle(), QTextOption(Qt::AlignCenter));
int offset = (int)(scale * 8);
QRect textRect(offset, 0, this->width() - offset - offset, height);
QTextOption option(Qt::AlignLeft | Qt::AlignVCenter);
option.setWrapMode(QTextOption::NoWrap);
painter.drawText(textRect, this->getTitle(), option);
} else {
// QTextOption option(Qt::AlignLeft | Qt::AlignVCenter);
// option.setWrapMode(QTextOption::NoWrap);
// int offset = (int)(scale * 16);
// QRect textRect(offset, 0, this->width() - offset - offset, height);
// painter.drawText(textRect, this->getTitle(), option);
}
// draw close x
if (app->settings->showTabCloseButton && (mouseOver || selected)) {
QRect xRect = this->getXRect();
if (mouseOverX) {
painter.fillRect(xRect, QColor(0, 0, 0, 64));
if (mouseDownX) {
painter.fillRect(xRect, QColor(0, 0, 0, 64));
}
}
int a = static_cast<int>(scale * 4);
painter.drawLine(xRect.topLeft() + QPoint(a, a), xRect.bottomRight() + QPoint(-a, -a));
painter.drawLine(xRect.topRight() + QPoint(-a, a), xRect.bottomLeft() + QPoint(a, -a));
}
} // namespace widgets
void NotebookTab::mousePressEvent(QMouseEvent *event)
{
this->mouseDown = true;
this->mouseDownX = this->getXRect().contains(event->pos());
this->update();
this->notebook->select(page);
switch (event->button()) {
case Qt::RightButton: {
this->menu.popup(event->globalPos());
} break;
}
}
void NotebookTab::mouseReleaseEvent(QMouseEvent *event)
{
auto app = getApp();
this->mouseDown = false;
if (event->button() == Qt::MiddleButton) {
if (this->rect().contains(event->pos())) {
this->notebook->removePage(this->page);
}
} else {
if (app->settings->showTabCloseButton && this->mouseDownX &&
this->getXRect().contains(event->pos())) {
this->mouseDownX = false;
this->notebook->removePage(this->page);
} else {
this->update();
}
}
}
void NotebookTab::enterEvent(QEvent *)
{
this->mouseOver = true;
this->update();
}
void NotebookTab::leaveEvent(QEvent *)
{
this->mouseOverX = false;
this->mouseOver = false;
this->update();
}
void NotebookTab::dragEnterEvent(QDragEnterEvent *)
{
this->notebook->select(this->page);
}
void NotebookTab::mouseMoveEvent(QMouseEvent *event)
{
auto app = getApp();
if (app->settings->showTabCloseButton) {
bool overX = this->getXRect().contains(event->pos());
if (overX != this->mouseOverX) {
// Over X state has been changed (we either left or entered it;
this->mouseOverX = overX;
this->update();
}
}
QPoint relPoint = this->mapToParent(event->pos());
if (this->mouseDown && !this->getDesiredRect().contains(relPoint)) {
int index;
SplitContainer *clickedPage = notebook->tabAt(relPoint, index, this->width());
if (clickedPage != nullptr && clickedPage != this->page) {
this->notebook->rearrangePage(this->page, index);
}
}
}
} // namespace widgets } // namespace widgets
} // namespace chatterino } // namespace chatterino

View file

@ -11,7 +11,9 @@
namespace chatterino { namespace chatterino {
namespace widgets { namespace widgets {
class Notebook; #define NOTEBOOK_TAB_HEIGHT 28
// class Notebook;
class Notebook2; class Notebook2;
class SplitContainer; class SplitContainer;
@ -73,6 +75,9 @@ private:
bool mouseOverX = false; bool mouseOverX = false;
bool mouseDownX = false; bool mouseDownX = false;
bool hasXButton();
bool shouldDrawXButton();
HighlightState highlightState = HighlightState::None; HighlightState highlightState = HighlightState::None;
QMenu menu; QMenu menu;
@ -80,75 +85,5 @@ private:
QRect getXRect(); QRect getXRect();
}; };
class NotebookTab : public BaseWidget
{
Q_OBJECT
public:
explicit NotebookTab(Notebook *_notebook);
void updateSize();
SplitContainer *page;
const QString &getTitle() const;
void setTitle(const QString &newTitle);
bool isSelected() const;
void setSelected(bool value);
void setHighlightState(HighlightState style);
void moveAnimated(QPoint pos, bool animated = true);
QRect getDesiredRect() const;
void hideTabXChanged(bool);
protected:
virtual void themeRefreshEvent() override;
virtual void paintEvent(QPaintEvent *) override;
virtual void mousePressEvent(QMouseEvent *event) override;
virtual void mouseReleaseEvent(QMouseEvent *event) override;
virtual void enterEvent(QEvent *) override;
virtual void leaveEvent(QEvent *) override;
virtual void dragEnterEvent(QDragEnterEvent *event) override;
virtual void mouseMoveEvent(QMouseEvent *event) override;
private:
std::vector<pajlada::Signals::ScopedConnection> managedConnections;
QPropertyAnimation positionChangedAnimation;
bool positionChangedAnimationRunning = false;
QPoint positionAnimationDesiredPoint;
Notebook *notebook;
QString title;
public:
bool useDefaultTitle = true;
private:
bool selected = false;
bool mouseOver = false;
bool mouseDown = false;
bool mouseOverX = false;
bool mouseDownX = false;
HighlightState highlightState = HighlightState::None;
QMenu menu;
QRect getXRect()
{
float s = this->getScale();
return QRect(this->width() - static_cast<int>(20 * s), static_cast<int>(4 * s),
static_cast<int>(16 * s), static_cast<int>(16 * s));
}
};
} // namespace widgets } // namespace widgets
} // namespace chatterino } // namespace chatterino

View file

@ -65,11 +65,11 @@ void SplitInput::initLayout()
// set edit font // set edit font
this->ui.textEdit->setFont( this->ui.textEdit->setFont(
app->fonts->getFont(singletons::FontManager::Type::Medium, this->getScale())); app->fonts->getFont(singletons::FontManager::Type::ChatMedium, this->getScale()));
this->managedConnections.emplace_back(app->fonts->fontChanged.connect([=]() { this->managedConnections.emplace_back(app->fonts->fontChanged.connect([=]() {
this->ui.textEdit->setFont( this->ui.textEdit->setFont(
app->fonts->getFont(singletons::FontManager::Type::Medium, this->getScale())); app->fonts->getFont(singletons::FontManager::Type::ChatMedium, this->getScale()));
})); }));
// open emote popup // open emote popup
@ -244,18 +244,18 @@ void SplitInput::installKeyPressedEvent()
SplitContainer *page = SplitContainer *page =
static_cast<SplitContainer *>(this->chatWidget->parentWidget()); static_cast<SplitContainer *>(this->chatWidget->parentWidget());
Notebook *notebook = static_cast<Notebook *>(page->parentWidget()); Notebook2 *notebook = static_cast<Notebook2 *>(page->parentWidget());
notebook->nextTab(); notebook->selectNextTab();
} }
} else if (event->key() == Qt::Key_Backtab) { } else if (event->key() == Qt::Key_Backtab) {
if (event->modifiers() == (Qt::ControlModifier | Qt::ShiftModifier)) { if (event->modifiers() == (Qt::ControlModifier | Qt::ShiftModifier)) {
SplitContainer *page = SplitContainer *page =
static_cast<SplitContainer *>(this->chatWidget->parentWidget()); static_cast<SplitContainer *>(this->chatWidget->parentWidget());
Notebook *notebook = static_cast<Notebook *>(page->parentWidget()); Notebook2 *notebook = static_cast<Notebook2 *>(page->parentWidget());
notebook->previousTab(); notebook->selectPreviousTab();
} }
} else if (event->key() == Qt::Key_C && event->modifiers() == Qt::ControlModifier) { } else if (event->key() == Qt::Key_C && event->modifiers() == Qt::ControlModifier) {
if (this->chatWidget->view.hasSelection()) { if (this->chatWidget->view.hasSelection()) {
@ -316,14 +316,23 @@ void SplitInput::paintEvent(QPaintEvent *)
{ {
QPainter painter(this); QPainter painter(this);
painter.fillRect(this->rect(), this->themeManager->splits.input.background);
QPen pen(this->themeManager->splits.input.border);
if (this->themeManager->isLightTheme()) { if (this->themeManager->isLightTheme()) {
pen.setWidth((int)(6 * this->getScale())); int s = (int)(3 * this->getScale());
QRect rect = this->rect().marginsRemoved(QMargins(s, s, s, s));
painter.fillRect(rect, this->themeManager->splits.input.background);
painter.setPen(QColor("#ccc"));
painter.drawRect(rect);
} else {
int s = (int)(1 * this->getScale());
QRect rect = this->rect().marginsRemoved(QMargins(s, s, s, s));
painter.fillRect(rect, this->themeManager->splits.input.background);
painter.setPen(QColor("#333"));
painter.drawRect(rect);
} }
painter.setPen(pen);
painter.drawRect(0, 0, this->width() - 1, this->height() - 1);
} }
void SplitInput::resizeEvent(QResizeEvent *) void SplitInput::resizeEvent(QResizeEvent *)

View file

@ -89,6 +89,8 @@ void Notebook2::removePage(QWidget *page)
break; break;
} }
} }
this->performLayout();
} }
void Notebook2::removeCurrentPage() void Notebook2::removeCurrentPage()
@ -179,6 +181,11 @@ int Notebook2::getPageCount() const
return this->items.count(); return this->items.count();
} }
QWidget *Notebook2::getPageAt(int index) const
{
return this->items[index].page;
}
int Notebook2::getSelectedIndex() const int Notebook2::getSelectedIndex() const
{ {
return this->indexOf(this->selectedPage); return this->indexOf(this->selectedPage);
@ -242,11 +249,11 @@ void Notebook2::setShowAddButton(bool value)
void Notebook2::scaleChangedEvent(float scale) void Notebook2::scaleChangedEvent(float scale)
{ {
// float h = 24 * this->getScale(); float h = NOTEBOOK_TAB_HEIGHT * this->getScale();
// this->settingsButton.setFixedSize(h, h); // this->settingsButton.setFixedSize(h, h);
// this->userButton.setFixedSize(h, h); // this->userButton.setFixedSize(h, h);
// this->addButton.setFixedSize(h, h); this->addButton.setFixedSize(h, h);
for (auto &i : this->items) { for (auto &i : this->items) {
i.tab->updateSize(); i.tab->updateSize();
@ -290,10 +297,12 @@ void Notebook2::performLayout(bool animated)
// x += (int)(scale * 2); // x += (int)(scale * 2);
// } // }
int tabHeight = static_cast<int>(24 * scale); int tabHeight = static_cast<int>(NOTEBOOK_TAB_HEIGHT * scale);
bool first = true; bool first = true;
for (auto i = this->items.begin(); i != this->items.end(); i++) { for (auto i = this->items.begin(); i != this->items.end(); i++) {
// int yOffset = i->tab->isSelected() ? 0 : 1;
if (!first && if (!first &&
(i == this->items.end() && this->showAddButton ? tabHeight : 0) + x + i->tab->width() > (i == this->items.end() && this->showAddButton ? tabHeight : 0) + x + i->tab->width() >
width()) // width()) //
@ -321,7 +330,7 @@ void Notebook2::performLayout(bool animated)
this->update(); this->update();
} }
y += (int)(1 * scale); y += (int)(3 * scale);
for (auto &i : this->items) { for (auto &i : this->items) {
i.tab->raise(); i.tab->raise();
@ -343,10 +352,15 @@ void Notebook2::paintEvent(QPaintEvent *event)
BaseWidget::paintEvent(event); BaseWidget::paintEvent(event);
QPainter painter(this); QPainter painter(this);
painter.fillRect(0, this->lineY, this->width(), (int)(1 * this->getScale()), painter.fillRect(0, this->lineY, this->width(), (int)(3 * this->getScale()),
this->themeManager->tabs.bottomLine); this->themeManager->tabs.bottomLine);
} }
NotebookButton *Notebook2::getAddButton()
{
return &this->addButton;
}
NotebookTab2 *Notebook2::getTabFromPage(QWidget *page) NotebookTab2 *Notebook2::getTabFromPage(QWidget *page)
{ {
for (auto &it : this->items) { for (auto &it : this->items) {
@ -358,318 +372,28 @@ NotebookTab2 *Notebook2::getTabFromPage(QWidget *page)
return nullptr; return nullptr;
} }
// Notebook2::OLD NOTEBOOK SplitNotebook::SplitNotebook(QWidget *parent)
Notebook::Notebook(Window *parent, bool _showButtons) : Notebook2(parent)
: BaseWidget(parent)
, parentWindow(parent)
, addButton(this)
, settingsButton(this)
, userButton(this)
, showButtons(_showButtons)
, closeConfirmDialog(this)
{ {
auto app = getApp(); this->connect(this->getAddButton(), &NotebookButton::clicked,
[this]() { QTimer::singleShot(80, this, [this] { this->addPage(true); }); });
this->connect(&this->settingsButton, SIGNAL(clicked()), this, SLOT(settingsButtonClicked()));
this->connect(&this->userButton, SIGNAL(clicked()), this, SLOT(usersButtonClicked()));
this->connect(&this->addButton, SIGNAL(clicked()), this, SLOT(addPageButtonClicked()));
this->settingsButton.icon = NotebookButton::IconSettings;
this->userButton.move(24, 0);
this->userButton.icon = NotebookButton::IconUser;
app->settings->hidePreferencesButton.connectSimple([this](auto) { this->performLayout(); });
app->settings->hideUserButton.connectSimple([this](auto) { this->performLayout(); });
closeConfirmDialog.setText("Are you sure you want to close this tab?");
closeConfirmDialog.setIcon(QMessageBox::Icon::Question);
closeConfirmDialog.setStandardButtons(QMessageBox::Yes | QMessageBox::Cancel);
closeConfirmDialog.setDefaultButton(QMessageBox::Yes);
this->scaleChangedEvent(this->getScale());
// Window-wide hotkeys
// CTRL+T: Create new split in selected notebook page
// CreateWindowShortcut(this, "CTRL+T", [this]() {
// if (this->selectedPage == nullptr) {
// return;
// }
// this->selectedPage->addChat(true);
// });
} }
SplitContainer *Notebook::addNewPage(bool select) SplitContainer *SplitNotebook::addPage(bool select)
{ {
auto tab = new NotebookTab(this); SplitContainer *container = new SplitContainer(this);
auto page = new SplitContainer(this, tab); auto *tab = Notebook2::addPage(container, QString(), select);
container->setTab(tab);
tab->setParent(this);
tab->show(); tab->show();
return container;
if (select || this->pages.count() == 0) {
this->select(page);
} }
this->pages.append(page); SplitContainer *SplitNotebook::getOrAddSelectedPage()
this->performLayout();
return page;
}
void Notebook::removePage(SplitContainer *page)
{ {
if (page->getSplitCount() > 0 && closeConfirmDialog.exec() != QMessageBox::Yes) { auto *selectedPage = this->getSelectedPage();
return;
}
int index = this->pages.indexOf(page); return selectedPage != nullptr ? (SplitContainer *)selectedPage : this->addPage();
if (this->pages.size() == 1) {
select(nullptr);
} else if (index == this->pages.count() - 1) {
select(this->pages[index - 1]);
} else {
select(this->pages[index + 1]);
}
page->getTab()->deleteLater();
page->deleteLater();
this->pages.removeOne(page);
if (this->pages.empty()) {
this->addNewPage();
}
this->performLayout();
}
void Notebook::removeCurrentPage()
{
if (this->selectedPage == nullptr) {
return;
}
this->removePage(this->selectedPage);
}
SplitContainer *Notebook::getOrAddSelectedPage()
{
if (selectedPage == nullptr) {
this->addNewPage(true);
}
return selectedPage;
}
SplitContainer *Notebook::getSelectedPage()
{
return selectedPage;
}
void Notebook::select(SplitContainer *page)
{
if (page == this->selectedPage) {
return;
}
if (page != nullptr) {
page->setHidden(false);
page->getTab()->setSelected(true);
page->getTab()->raise();
}
if (this->selectedPage != nullptr) {
this->selectedPage->setHidden(true);
this->selectedPage->getTab()->setSelected(false);
for (Split *split : this->selectedPage->getSplits()) {
split->updateLastReadMessage();
}
}
this->selectedPage = page;
this->performLayout();
}
void Notebook::selectIndex(int index)
{
if (index < 0 || index >= this->pages.size()) {
return;
}
this->select(this->pages.at(index));
}
int Notebook::tabCount()
{
return this->pages.size();
}
SplitContainer *Notebook::tabAt(QPoint point, int &index, int maxWidth)
{
int i = 0;
for (auto *page : this->pages) {
QRect rect = page->getTab()->getDesiredRect();
rect.setHeight((int)(this->getScale() * 24));
rect.setWidth(std::min(maxWidth, rect.width()));
if (rect.contains(point)) {
index = i;
return page;
}
i++;
}
index = -1;
return nullptr;
}
SplitContainer *Notebook::tabAt(int index)
{
return this->pages[index];
}
void Notebook::rearrangePage(SplitContainer *page, int index)
{
this->pages.move(this->pages.indexOf(page), index);
this->performLayout();
}
void Notebook::nextTab()
{
if (this->pages.size() <= 1) {
return;
}
int index = (this->pages.indexOf(this->selectedPage) + 1) % this->pages.size();
this->select(this->pages[index]);
}
void Notebook::previousTab()
{
if (this->pages.size() <= 1) {
return;
}
int index = (this->pages.indexOf(this->selectedPage) - 1);
if (index < 0) {
index += this->pages.size();
}
this->select(this->pages[index]);
}
void Notebook::performLayout(bool animated)
{
auto app = getApp();
int x = 0, y = 0;
float scale = this->getScale();
bool customFrame = this->parentWindow->hasCustomWindowFrame();
if (!this->showButtons || app->settings->hidePreferencesButton || customFrame) {
this->settingsButton.hide();
} else {
this->settingsButton.show();
x += settingsButton.width();
}
if (!this->showButtons || app->settings->hideUserButton || customFrame) {
this->userButton.hide();
} else {
this->userButton.move(x, 0);
this->userButton.show();
x += userButton.width();
}
if (customFrame || !this->showButtons ||
(app->settings->hideUserButton && app->settings->hidePreferencesButton)) {
x += (int)(scale * 2);
}
int tabHeight = static_cast<int>(24 * scale);
bool first = true;
for (auto &i : this->pages) {
if (!first &&
(i == this->pages.last() ? tabHeight : 0) + x + i->getTab()->width() > width()) {
y += i->getTab()->height();
// y += 20;
i->getTab()->moveAnimated(QPoint(0, y), animated);
x = i->getTab()->width();
} else {
i->getTab()->moveAnimated(QPoint(x, y), animated);
x += i->getTab()->width();
}
// x -= (int)(8 * scale);
x += 1;
first = false;
}
// x += (int)(8 * scale);
// x += 1;
this->addButton.move(x, y);
y -= 1;
for (auto &i : this->pages) {
i->getTab()->raise();
}
this->addButton.raise();
if (this->selectedPage != nullptr) {
this->selectedPage->move(0, y + tabHeight);
this->selectedPage->resize(width(), height() - y - tabHeight);
this->selectedPage->raise();
}
}
void Notebook::resizeEvent(QResizeEvent *)
{
this->performLayout(false);
}
void Notebook::scaleChangedEvent(float)
{
float h = 24 * this->getScale();
this->settingsButton.setFixedSize(h, h);
this->userButton.setFixedSize(h, h);
this->addButton.setFixedSize(h, h);
for (auto &i : this->pages) {
i->getTab()->updateSize();
}
}
void Notebook::settingsButtonClicked()
{
auto app = getApp();
app->windows->showSettingsDialog();
}
void Notebook::usersButtonClicked()
{
auto app = getApp();
app->windows->showAccountSelectPopup(this->mapToGlobal(this->userButton.rect().bottomRight()));
}
void Notebook::addPageButtonClicked()
{
QTimer::singleShot(80, [this] { this->addNewPage(true); });
} }
} // namespace widgets } // namespace widgets

View file

@ -32,6 +32,7 @@ public:
void selectPreviousTab(); void selectPreviousTab();
int getPageCount() const; int getPageCount() const;
QWidget *getPageAt(int index) const;
int getSelectedIndex() const; int getSelectedIndex() const;
QWidget *getSelectedPage() const; QWidget *getSelectedPage() const;
@ -44,11 +45,15 @@ public:
bool getShowAddButton() const; bool getShowAddButton() const;
void setShowAddButton(bool value); void setShowAddButton(bool value);
void performLayout(bool animate = true);
protected: protected:
virtual void scaleChangedEvent(float scale) override; virtual void scaleChangedEvent(float scale) override;
virtual void resizeEvent(QResizeEvent *) override; virtual void resizeEvent(QResizeEvent *) override;
virtual void paintEvent(QPaintEvent *) override; virtual void paintEvent(QPaintEvent *) override;
NotebookButton *getAddButton();
private: private:
struct Item { struct Item {
NotebookTab2 *tab; NotebookTab2 *tab;
@ -64,63 +69,16 @@ private:
bool showAddButton = false; bool showAddButton = false;
int lineY = 20; int lineY = 20;
void performLayout(bool animate = true);
NotebookTab2 *getTabFromPage(QWidget *page); NotebookTab2 *getTabFromPage(QWidget *page);
}; };
class Notebook : public BaseWidget class SplitNotebook : public Notebook2
{ {
Q_OBJECT
public: public:
explicit Notebook(Window *parent, bool _showButtons); SplitNotebook(QWidget *parent);
SplitContainer *addNewPage(bool select = false);
void removePage(SplitContainer *page);
void removeCurrentPage();
void select(SplitContainer *page);
void selectIndex(int index);
SplitContainer *addPage(bool select = false);
SplitContainer *getOrAddSelectedPage(); SplitContainer *getOrAddSelectedPage();
SplitContainer *getSelectedPage();
void performLayout(bool animate = true);
int tabCount();
SplitContainer *tabAt(QPoint point, int &index, int maxWidth = 2000000000);
SplitContainer *tabAt(int index);
void rearrangePage(SplitContainer *page, int index);
void nextTab();
void previousTab();
protected:
void scaleChangedEvent(float scale);
void resizeEvent(QResizeEvent *);
void settingsButtonMouseReleased(QMouseEvent *event);
public slots:
void settingsButtonClicked();
void usersButtonClicked();
void addPageButtonClicked();
private:
Window *parentWindow;
QList<SplitContainer *> pages;
NotebookButton addButton;
NotebookButton settingsButton;
NotebookButton userButton;
SplitContainer *selectedPage = nullptr;
bool showButtons;
QMessageBox closeConfirmDialog;
}; };
} // namespace widgets } // namespace widgets

View file

@ -165,12 +165,12 @@ QLayout *AppearancePage::createFontChanger()
layout->addWidget(label); layout->addWidget(label);
auto updateFontFamilyLabel = [=](auto) { auto updateFontFamilyLabel = [=](auto) {
label->setText(QString::fromStdString(app->fonts->currentFontFamily.getValue()) + ", " + label->setText(QString::fromStdString(app->fonts->chatFontFamily.getValue()) + ", " +
QString::number(app->fonts->currentFontSize) + "pt"); QString::number(app->fonts->chatFontSize) + "pt");
}; };
app->fonts->currentFontFamily.connectSimple(updateFontFamilyLabel, this->managedConnections); app->fonts->chatFontFamily.connectSimple(updateFontFamilyLabel, this->managedConnections);
app->fonts->currentFontSize.connectSimple(updateFontFamilyLabel, this->managedConnections); app->fonts->chatFontSize.connectSimple(updateFontFamilyLabel, this->managedConnections);
// BUTTON // BUTTON
QPushButton *button = new QPushButton("Select"); QPushButton *button = new QPushButton("Select");
@ -178,11 +178,11 @@ QLayout *AppearancePage::createFontChanger()
button->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Policy::Fixed); button->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Policy::Fixed);
QObject::connect(button, &QPushButton::clicked, [=]() { QObject::connect(button, &QPushButton::clicked, [=]() {
QFontDialog dialog(app->fonts->getFont(singletons::FontManager::Medium, 1.)); QFontDialog dialog(app->fonts->getFont(singletons::FontManager::ChatMedium, 1.));
dialog.connect(&dialog, &QFontDialog::fontSelected, [=](const QFont &font) { dialog.connect(&dialog, &QFontDialog::fontSelected, [=](const QFont &font) {
app->fonts->currentFontFamily = font.family().toStdString(); app->fonts->chatFontFamily = font.family().toStdString();
app->fonts->currentFontSize = font.pointSize(); app->fonts->chatFontSize = font.pointSize();
}); });
dialog.show(); dialog.show();

View file

@ -29,15 +29,13 @@ namespace widgets {
bool SplitContainer::isDraggingSplit = false; bool SplitContainer::isDraggingSplit = false;
Split *SplitContainer::draggingSplit = nullptr; Split *SplitContainer::draggingSplit = nullptr;
SplitContainer::SplitContainer(Notebook *parent, NotebookTab *_tab) SplitContainer::SplitContainer(Notebook2 *parent)
: BaseWidget(parent) : BaseWidget(parent)
, tab(_tab) , tab(nullptr)
, dropPreview(this) , dropPreview(this)
, mouseOverPoint(-10000, -10000) , mouseOverPoint(-10000, -10000)
, overlay(this) , overlay(this)
{ {
this->tab->page = this;
this->refreshTabTitle(); this->refreshTabTitle();
this->managedConnect(Split::modifierStatusChanged, [this](auto modifiers) { this->managedConnect(Split::modifierStatusChanged, [this](auto modifiers) {
@ -69,11 +67,20 @@ SplitContainer::SplitContainer(Notebook *parent, NotebookTab *_tab)
this->setAcceptDrops(true); this->setAcceptDrops(true);
} }
NotebookTab *SplitContainer::getTab() const NotebookTab2 *SplitContainer::getTab() const
{ {
return this->tab; return this->tab;
} }
void SplitContainer::setTab(NotebookTab2 *_tab)
{
this->tab = _tab;
this->tab->page = this;
this->refreshTabTitle();
}
void SplitContainer::appendNewSplit(bool openChannelNameDialog) void SplitContainer::appendNewSplit(bool openChannelNameDialog)
{ {
Split *split = new Split(this); Split *split = new Split(this);
@ -131,6 +138,12 @@ void SplitContainer::insertSplit(Split *split, Direction direction, Node *relati
this->refreshTabTitle(); this->refreshTabTitle();
split->getChannelView().tabHighlightRequested.connect([this](HighlightState state) {
if (this->tab != nullptr) {
this->tab->setHighlightState(state);
}
});
this->layout(); this->layout();
} }
@ -149,6 +162,8 @@ SplitContainer::Position SplitContainer::releaseSplit(Split *split)
this->refreshTabTitle(); this->refreshTabTitle();
split->getChannelView().tabHighlightRequested.disconnectAll();
return position; return position;
} }
@ -265,10 +280,10 @@ void SplitContainer::paintEvent(QPaintEvent *)
QString text = "Click to add a split"; QString text = "Click to add a split";
Notebook *notebook = dynamic_cast<Notebook *>(this->parentWidget()); Notebook2 *notebook = dynamic_cast<Notebook2 *>(this->parentWidget());
if (notebook != nullptr) { if (notebook != nullptr) {
if (notebook->tabCount() > 1) { if (notebook->getPageCount() > 1) {
text += "\n\nTip: After adding a split you can hold <Alt> to move it or split it " text += "\n\nTip: After adding a split you can hold <Alt> to move it or split it "
"further."; "further.";
} }
@ -276,7 +291,7 @@ void SplitContainer::paintEvent(QPaintEvent *)
painter.drawText(rect(), text, QTextOption(Qt::AlignCenter)); painter.drawText(rect(), text, QTextOption(Qt::AlignCenter));
} else { } else {
painter.fillRect(rect(), this->themeManager->splits.messageSeperator); painter.fillRect(rect(), QColor("#555"));
} }
for (DropRect &dropRect : this->dropRects) { for (DropRect &dropRect : this->dropRects) {
@ -326,13 +341,15 @@ void SplitContainer::mouseMoveEvent(QMouseEvent *event)
void SplitContainer::leaveEvent(QEvent *event) void SplitContainer::leaveEvent(QEvent *event)
{ {
this->mouseOverPoint = QPoint(-1000, -10000); this->mouseOverPoint = QPoint(-10000, -10000);
this->update(); this->update();
} }
void SplitContainer::refreshTabTitle() void SplitContainer::refreshTabTitle()
{ {
assert(this->tab != nullptr); if (this->tab == nullptr) {
return;
}
if (!this->tab->useDefaultTitle) { if (!this->tab->useDefaultTitle) {
return; return;

View file

@ -165,7 +165,7 @@ private:
}; };
public: public:
SplitContainer(Notebook *parent, NotebookTab *_tab); SplitContainer(Notebook2 *parent);
void appendNewSplit(bool openChannelNameDialog); void appendNewSplit(bool openChannelNameDialog);
void appendSplit(Split *split); void appendSplit(Split *split);
@ -189,12 +189,14 @@ public:
void refreshTabTitle(); void refreshTabTitle();
NotebookTab *getTab() const; NotebookTab2 *getTab() const;
Node *getBaseNode() Node *getBaseNode()
{ {
return &this->baseNode; return &this->baseNode;
} }
void setTab(NotebookTab2 *tab);
static bool isDraggingSplit; static bool isDraggingSplit;
static Split *draggingSplit; static Split *draggingSplit;
@ -232,7 +234,7 @@ private:
Node baseNode; Node baseNode;
NotebookTab *tab; NotebookTab2 *tab;
std::vector<Split *> splits; std::vector<Split *> splits;
bool isDragging = false; bool isDragging = false;

View file

@ -58,7 +58,7 @@ void TooltipWidget::updateFont()
auto app = getApp(); auto app = getApp();
this->setFont( this->setFont(
app->fonts->getFont(singletons::FontManager::Type::MediumSmall, this->getScale())); app->fonts->getFont(singletons::FontManager::Type::ChatMediumSmall, this->getScale()));
} }
void TooltipWidget::setText(QString text) void TooltipWidget::setText(QString text)

View file

@ -27,7 +27,7 @@ Window::Window(WindowType _type)
: BaseWindow(nullptr, true) : BaseWindow(nullptr, true)
, type(_type) , type(_type)
, dpi(this->getScale()) , dpi(this->getScale())
, notebook(this, !this->hasCustomWindowFrame()) , notebook(this)
{ {
auto app = getApp(); auto app = getApp();
@ -81,7 +81,7 @@ Window::Window(WindowType _type)
CreateWindowShortcut(this, "CTRL+9", [this] { this->notebook.selectIndex(8); }); CreateWindowShortcut(this, "CTRL+9", [this] { this->notebook.selectIndex(8); });
// CTRL+SHIFT+T: New tab // CTRL+SHIFT+T: New tab
CreateWindowShortcut(this, "CTRL+SHIFT+T", [this] { this->notebook.addNewPage(true); }); CreateWindowShortcut(this, "CTRL+SHIFT+T", [this] { this->notebook.addPage(true); });
// CTRL+SHIFT+W: Close current tab // CTRL+SHIFT+W: Close current tab
CreateWindowShortcut(this, "CTRL+SHIFT+W", [this] { this->notebook.removeCurrentPage(); }); CreateWindowShortcut(this, "CTRL+SHIFT+W", [this] { this->notebook.removeCurrentPage(); });
@ -106,6 +106,9 @@ Window::Window(WindowType _type)
// }); // });
this->setWindowTitle("Chatterino 2 Development Build"); this->setWindowTitle("Chatterino 2 Development Build");
this->notebook.setAllowUserTabManagement(true);
this->notebook.setShowAddButton(true);
} }
Window::WindowType Window::getType() Window::WindowType Window::getType()
@ -128,7 +131,7 @@ void Window::repaintVisibleChatWidgets(Channel *channel)
} }
} }
Notebook &Window::getNotebook() SplitNotebook &Window::getNotebook()
{ {
return this->notebook; return this->notebook;
} }

View file

@ -29,7 +29,7 @@ public:
void repaintVisibleChatWidgets(Channel *channel = nullptr); void repaintVisibleChatWidgets(Channel *channel = nullptr);
Notebook &getNotebook(); SplitNotebook &getNotebook();
void refreshWindowTitle(const QString &username); void refreshWindowTitle(const QString &username);
@ -47,9 +47,9 @@ private:
void loadGeometry(); void loadGeometry();
Notebook notebook; SplitNotebook notebook;
friend class Notebook; friend class Notebook2;
public: public:
void save(); void save();