This commit is contained in:
fourtf 2017-12-17 20:50:35 +01:00
commit 6f1509cb4f
16 changed files with 138 additions and 125 deletions

View file

@ -170,7 +170,8 @@ HEADERS += \
src/widgets/splitcontainer.hpp \ src/widgets/splitcontainer.hpp \
src/widgets/helper/droppreview.hpp \ src/widgets/helper/droppreview.hpp \
src/widgets/helper/splitcolumn.hpp \ src/widgets/helper/splitcolumn.hpp \
src/util/irchelpers.hpp src/util/irchelpers.hpp \
src/util/helpers.hpp
PRECOMPILED_HEADER = PRECOMPILED_HEADER =

View file

@ -38,6 +38,13 @@ void Channel::addMessage(std::shared_ptr<Message> message)
{ {
std::shared_ptr<Message> deleted; std::shared_ptr<Message> deleted;
const QString &username = message->username;
if (!username.isEmpty()) {
// TODO: Add recent chatters display name. This should maybe be a setting
this->addRecentChatter(username);
}
// if (_loggingChannel.get() != nullptr) { // if (_loggingChannel.get() != nullptr) {
// _loggingChannel->append(message); // _loggingChannel->append(message);
// } // }
@ -49,6 +56,24 @@ void Channel::addMessage(std::shared_ptr<Message> message)
this->messageAppended(message); this->messageAppended(message);
} }
void Channel::addRecentChatter(const QString &username)
{
std::lock_guard<std::mutex> lock(this->recentChattersMutex);
this->recentChatters.insert(username);
}
std::set<QString> Channel::getUsernamesForCompletions()
{
std::set<QString> usernames;
this->recentChattersMutex.lock();
usernames.insert(this->recentChatters.begin(), this->recentChatters.end());
this->recentChattersMutex.unlock();
return usernames;
}
bool Channel::canSendMessage() const bool Channel::canSendMessage() const
{ {
return false; return false;

View file

@ -13,6 +13,7 @@
#include <boost/signals2.hpp> #include <boost/signals2.hpp>
#include <memory> #include <memory>
#include <set>
namespace chatterino { namespace chatterino {
namespace messages { namespace messages {
@ -30,16 +31,21 @@ public:
virtual bool isEmpty() const; virtual bool isEmpty() const;
messages::LimitedQueueSnapshot<messages::SharedMessage> getMessageSnapshot(); messages::LimitedQueueSnapshot<messages::SharedMessage> getMessageSnapshot();
// methods
void addMessage(messages::SharedMessage message); void addMessage(messages::SharedMessage message);
void addRecentChatter(const QString &username);
std::set<QString> getUsernamesForCompletions();
QString name; QString name;
QStringList modList; QStringList modList;
std::mutex recentChattersMutex;
std::set<QString> recentChatters;
virtual bool canSendMessage() const; virtual bool canSendMessage() const;
virtual void sendMessage(const QString &message); virtual void sendMessage(const QString &message);
private: private:
// variables
messages::LimitedQueue<messages::SharedMessage> messages; messages::LimitedQueue<messages::SharedMessage> messages;
// std::shared_ptr<logging::Channel> loggingChannel; // std::shared_ptr<logging::Channel> loggingChannel;

View file

@ -5,6 +5,8 @@ using namespace chatterino::twitch;
namespace chatterino { namespace chatterino {
ChannelManager *ChannelManager::instance = nullptr;
ChannelManager::ChannelManager(WindowManager &_windowManager, IrcManager &_ircManager) ChannelManager::ChannelManager(WindowManager &_windowManager, IrcManager &_ircManager)
: windowManager(_windowManager) : windowManager(_windowManager)
, ircManager(_ircManager) , ircManager(_ircManager)
@ -12,6 +14,7 @@ ChannelManager::ChannelManager(WindowManager &_windowManager, IrcManager &_ircMa
, mentionsChannel(new TwitchChannel(_ircManager, "/mentions", true)) , mentionsChannel(new TwitchChannel(_ircManager, "/mentions", true))
, emptyChannel(new TwitchChannel(_ircManager, "", true)) , emptyChannel(new TwitchChannel(_ircManager, "", true))
{ {
ChannelManager::instance = this;
} }
const std::vector<std::shared_ptr<Channel>> ChannelManager::getItems() const std::vector<std::shared_ptr<Channel>> ChannelManager::getItems()

View file

@ -16,6 +16,8 @@ class ChannelManager
public: public:
explicit ChannelManager(WindowManager &_windowManager, IrcManager &_ircManager); explicit ChannelManager(WindowManager &_windowManager, IrcManager &_ircManager);
static ChannelManager *instance;
WindowManager &windowManager; WindowManager &windowManager;
IrcManager &ircManager; IrcManager &ircManager;

View file

@ -1,11 +1,12 @@
#include "completionmanager.hpp" #include "completionmanager.hpp"
#include "channelmanager.hpp"
#include "common.hpp" #include "common.hpp"
#include "debug/log.hpp" #include "debug/log.hpp"
#include "emotemanager.hpp" #include "emotemanager.hpp"
namespace chatterino { namespace chatterino {
CompletionModel::CompletionModel(const std::string &_channelName) CompletionModel::CompletionModel(const QString &_channelName)
: channelName(_channelName) : channelName(_channelName)
{ {
} }
@ -39,14 +40,14 @@ void CompletionModel::refresh()
// Channel-specific: BTTV Channel Emotes // Channel-specific: BTTV Channel Emotes
std::vector<std::string> &bttvChannelEmoteCodes = std::vector<std::string> &bttvChannelEmoteCodes =
emoteManager.bttvChannelEmoteCodes[this->channelName]; emoteManager.bttvChannelEmoteCodes[this->channelName.toStdString()];
for (const auto &m : bttvChannelEmoteCodes) { for (const auto &m : bttvChannelEmoteCodes) {
this->addString(m); this->addString(m);
} }
// Channel-specific: FFZ Channel Emotes // Channel-specific: FFZ Channel Emotes
std::vector<std::string> &ffzChannelEmoteCodes = std::vector<std::string> &ffzChannelEmoteCodes =
emoteManager.ffzChannelEmoteCodes[this->channelName]; emoteManager.ffzChannelEmoteCodes[this->channelName.toStdString()];
for (const auto &m : ffzChannelEmoteCodes) { for (const auto &m : ffzChannelEmoteCodes) {
this->addString(m); this->addString(m);
} }
@ -57,7 +58,17 @@ void CompletionModel::refresh()
this->addString(":" + m + ":"); this->addString(":" + m + ":");
} }
// TODO: Add Channel-specific: Usernames // Channel-specific: Usernames
auto *channelManager = ChannelManager::instance;
auto c = channelManager->getTwitchChannel(this->channelName);
if (!c) {
return;
}
auto usernames = c->getUsernamesForCompletions();
for (const auto &username : usernames) {
this->addString(username);
this->addString('@' + username);
}
} }
void CompletionModel::addString(const std::string &str) void CompletionModel::addString(const std::string &str)
@ -66,6 +77,12 @@ void CompletionModel::addString(const std::string &str)
this->emotes.push_back(qS(str) + " "); this->emotes.push_back(qS(str) + " ");
} }
void CompletionModel::addString(const QString &str)
{
// Always add a space at the end of completions
this->emotes.push_back(str + " ");
}
CompletionModel *CompletionManager::createModel(const std::string &channelName) CompletionModel *CompletionManager::createModel(const std::string &channelName)
{ {
auto it = this->models.find(channelName); auto it = this->models.find(channelName);
@ -73,7 +90,7 @@ CompletionModel *CompletionManager::createModel(const std::string &channelName)
return it->second; return it->second;
} }
CompletionModel *ret = new CompletionModel(channelName); CompletionModel *ret = new CompletionModel(qS(channelName));
this->models[channelName] = ret; this->models[channelName] = ret;
return ret; return ret;

View file

@ -11,7 +11,7 @@ namespace chatterino {
class CompletionModel : public QAbstractListModel class CompletionModel : public QAbstractListModel
{ {
public: public:
CompletionModel(const std::string &_channelName); CompletionModel(const QString &_channelName);
virtual int columnCount(const QModelIndex & /*parent*/) const override virtual int columnCount(const QModelIndex & /*parent*/) const override
{ {
@ -33,10 +33,11 @@ public:
private: private:
void addString(const std::string &str); void addString(const std::string &str);
void addString(const QString &str);
QVector<QString> emotes; QVector<QString> emotes;
std::string channelName; QString channelName;
}; };
class CompletionManager class CompletionManager

View file

@ -37,7 +37,7 @@ void Channel::append(std::shared_ptr<messages::Message> message)
str.append('['); str.append('[');
str.append(now.toString("HH:mm:ss")); str.append(now.toString("HH:mm:ss"));
str.append("] "); str.append("] ");
str.append(message->getUserName()); str.append(message->username);
str.append(": "); str.append(": ");
str.append(message->getContent()); str.append(message->getContent());
str.append('\n'); str.append('\n');

View file

@ -36,11 +36,6 @@ int Message::getTimeoutCount() const
return this->timeoutCount; return this->timeoutCount;
} }
const QString &Message::getUserName() const
{
return this->userName;
}
const QString &Message::getDisplayName() const const QString &Message::getDisplayName() const
{ {
return this->displayName; return this->displayName;

View file

@ -22,7 +22,6 @@ public:
void setHighlight(bool value); void setHighlight(bool value);
const QString &getTimeoutUser() const; const QString &getTimeoutUser() const;
int getTimeoutCount() const; int getTimeoutCount() const;
const QString &getUserName() const;
const QString &getDisplayName() const; const QString &getDisplayName() const;
const QString &getContent() const; const QString &getContent() const;
const std::chrono::time_point<std::chrono::system_clock> &getParseTime() const; const std::chrono::time_point<std::chrono::system_clock> &getParseTime() const;
@ -30,6 +29,8 @@ public:
bool isDisabled() const; bool isDisabled() const;
const QString &getId() const; const QString &getId() const;
QString username;
const QString text; const QString text;
bool centered = false; bool centered = false;
@ -57,7 +58,6 @@ private:
bool disabled = false; bool disabled = false;
std::chrono::time_point<std::chrono::system_clock> parseTime; std::chrono::time_point<std::chrono::system_clock> parseTime;
QString userName = "";
QString displayName = ""; QString displayName = "";
QString content; QString content;
QString id = ""; QString id = "";

View file

@ -26,8 +26,10 @@ public:
QString originalMessage; QString originalMessage;
private: protected:
std::shared_ptr<messages::Message> message; std::shared_ptr<messages::Message> message;
private:
std::vector<Word> _words; std::vector<Word> _words;
bool highlight = false; bool highlight = false;
std::chrono::time_point<std::chrono::system_clock> _parseTime; std::chrono::time_point<std::chrono::system_clock> _parseTime;

View file

@ -287,6 +287,8 @@ void TwitchMessageBuilder::parseUsername()
if (this->userName.isEmpty()) { if (this->userName.isEmpty()) {
this->userName = this->tags.value(QLatin1String("login")).toString(); this->userName = this->tags.value(QLatin1String("login")).toString();
} }
this->message->username = this->userName;
} }
void TwitchMessageBuilder::appendUsername() void TwitchMessageBuilder::appendUsername()

13
src/util/helpers.hpp Normal file
View file

@ -0,0 +1,13 @@
#pragma once
#include <fmt/format.h>
namespace chatterino {
template <typename... Args>
auto fS(Args &&... args) -> decltype(fmt::format(std::forward<Args>(args)...))
{
return fmt::format(std::forward<Args>(args)...);
}
} // namespace chatterino

View file

@ -6,23 +6,22 @@
#include "widgets/settingsdialog.hpp" #include "widgets/settingsdialog.hpp"
#include "widgets/split.hpp" #include "widgets/split.hpp"
#include <QDebug>
#include <QLibrary>
#include <QPalette> #include <QPalette>
#include <QShortcut> #include <QShortcut>
#include <QVBoxLayout> #include <QVBoxLayout>
#include <boost/foreach.hpp>
namespace chatterino { namespace chatterino {
namespace widgets { namespace widgets {
Window::Window(ChannelManager &_channelManager, ColorScheme &_colorScheme, bool _isMainWindow) Window::Window(const QString &_windowName, ChannelManager &_channelManager,
ColorScheme &_colorScheme, bool _isMainWindow)
: BaseWidget(_colorScheme, nullptr) : BaseWidget(_colorScheme, nullptr)
, windowName(_windowName)
, windowGeometry(this->windowName.toStdString())
, channelManager(_channelManager) , channelManager(_channelManager)
, colorScheme(_colorScheme) , colorScheme(_colorScheme)
, notebook(this->channelManager, this, _isMainWindow) , notebook(this->channelManager, this, _isMainWindow)
, dpi(this->getDpiMultiplier()) , dpi(this->getDpiMultiplier())
// , windowGeometry("/windows/0/geometry")
{ {
this->initAsWindow(); this->initAsWindow();
@ -45,14 +44,7 @@ Window::Window(ChannelManager &_channelManager, ColorScheme &_colorScheme, bool
this->refreshTheme(); this->refreshTheme();
if (/*this->windowGeometry->isFilled()*/ false) { this->loadGeometry();
// Load geometry from settings file
// this->setGeometry(this->windowGeometry.getValueRef());
} else {
// Set default geometry
// Default position is in the middle of the current monitor or the primary monitor
this->resize(1280, 800);
}
// Initialize program-wide hotkeys // Initialize program-wide hotkeys
{ {
@ -120,8 +112,12 @@ Notebook &Window::getNotebook()
void Window::closeEvent(QCloseEvent *) void Window::closeEvent(QCloseEvent *)
{ {
// Save closing window position const QRect &geom = this->geometry();
// this->windowGeometry = this->geometry();
this->windowGeometry.x = geom.x();
this->windowGeometry.y = geom.y();
this->windowGeometry.width = geom.width();
this->windowGeometry.height = geom.height();
this->closed(); this->closed();
} }
@ -133,5 +129,19 @@ void Window::refreshTheme()
this->setPalette(palette); this->setPalette(palette);
} }
void Window::loadGeometry()
{
if (!this->windowGeometry.x.isDefaultValue() && !this->windowGeometry.y.isDefaultValue()) {
this->move(this->windowGeometry.x, this->windowGeometry.y);
}
if (!this->windowGeometry.width.isDefaultValue() &&
!this->windowGeometry.height.isDefaultValue()) {
this->resize(this->windowGeometry.width, this->windowGeometry.height);
} else {
this->resize(1280, 800);
}
}
} // namespace widgets } // namespace widgets
} // namespace chatterino } // namespace chatterino

View file

@ -1,5 +1,6 @@
#pragma once #pragma once
#include "util/helpers.hpp"
#include "widgets/basewidget.hpp" #include "widgets/basewidget.hpp"
#include "widgets/notebook.hpp" #include "widgets/notebook.hpp"
#include "widgets/titlebar.hpp" #include "widgets/titlebar.hpp"
@ -10,8 +11,7 @@
#include <boost/property_tree/ptree.hpp> #include <boost/property_tree/ptree.hpp>
#include <boost/signals2.hpp> #include <boost/signals2.hpp>
#include <pajlada/settings/serialize.hpp> #include <pajlada/settings/setting.hpp>
#include <pajlada/settings/settingdata.hpp>
namespace chatterino { namespace chatterino {
@ -21,12 +21,32 @@ class CompletionManager;
namespace widgets { namespace widgets {
struct WindowGeometry {
WindowGeometry(const std::string &key)
: x(fS("/windows/{}/geometry/x", key))
, y(fS("/windows/{}/geometry/y", key))
, width(fS("/windows/{}/geometry/width", key))
, height(fS("/windows/{}/geometry/height", key))
{
}
pajlada::Settings::Setting<int> x;
pajlada::Settings::Setting<int> y;
pajlada::Settings::Setting<int> width;
pajlada::Settings::Setting<int> height;
};
class Window : public BaseWidget class Window : public BaseWidget
{ {
Q_OBJECT Q_OBJECT
QString windowName;
WindowGeometry windowGeometry;
public: public:
explicit Window(ChannelManager &_channelManager, ColorScheme &_colorScheme, bool isMainWindow); explicit Window(const QString &_windowName, ChannelManager &_channelManager,
ColorScheme &_colorScheme, bool isMainWindow);
void repaintVisibleChatWidgets(Channel *channel = nullptr); void repaintVisibleChatWidgets(Channel *channel = nullptr);
@ -48,6 +68,8 @@ private:
virtual void refreshTheme() override; virtual void refreshTheme() override;
void loadGeometry();
ChannelManager &channelManager; ChannelManager &channelManager;
ColorScheme &colorScheme; ColorScheme &colorScheme;
@ -55,92 +77,6 @@ private:
bool loaded = false; bool loaded = false;
TitleBar titleBar; TitleBar titleBar;
/*
class QRectWrapper : public pajlada::Settings::ISettingData, public QRect
{
public:
QRectWrapper()
: QRect(-1, -1, -1, -1)
{
}
pajlada::Signals::Signal<const QRectWrapper &> valueChanged;
const QRectWrapper &getValueRef() const
{
return *this;
}
virtual rapidjson::Value marshalInto(rapidjson::Document &d) override
{
using namespace pajlada::Settings;
rapidjson::Value obj(rapidjson::kObjectType);
auto _x = serializeToJSON<int>::serialize(this->x(), d.GetAllocator());
auto _y = serializeToJSON<int>::serialize(this->y(), d.GetAllocator());
auto _width = serializeToJSON<int>::serialize(this->width(), d.GetAllocator());
auto _height = serializeToJSON<int>::serialize(this->height(), d.GetAllocator());
obj.AddMember("x", _x, d.GetAllocator());
obj.AddMember("y", _y, d.GetAllocator());
obj.AddMember("width", _width, d.GetAllocator());
obj.AddMember("height", _height, d.GetAllocator());
return obj;
}
virtual bool unmarshalFrom(rapidjson::Document &document) override
{
using namespace pajlada::Settings;
auto vXp = this->getValueWithSuffix("/x", document);
auto vYp = this->getValueWithSuffix("/y", document);
auto vWidthp = this->getValueWithSuffix("/width", document);
auto vHeightp = this->getValueWithSuffix("/height", document);
if (vXp != nullptr) {
this->setX(deserializeJSON<int>::deserialize(*vXp));
this->filled = true;
}
if (vYp != nullptr) {
this->setY(deserializeJSON<int>::deserialize(*vYp));
this->filled = true;
}
if (vWidthp != nullptr) {
this->setWidth(deserializeJSON<int>::deserialize(*vWidthp));
this->filled = true;
}
if (vHeightp != nullptr) {
this->setHeight(deserializeJSON<int>::deserialize(*vHeightp));
this->filled = true;
}
return true;
}
virtual void registerDocument(rapidjson::Document &d) override
{
this->valueChanged.connect([this, &d](const auto &) {
this->marshalInto(d); //
});
}
QRectWrapper &operator=(const QRect &rhs)
{
static_cast<QRect &>(*this) = rhs;
return *this;
}
void setValue(const QRect &rhs)
{
static_cast<QRect &>(*this) = rhs;
}
};
*/
// pajlada::Settings::Setting<QRectWrapper> windowGeometry;
friend class Notebook; friend class Notebook;
}; };

View file

@ -22,7 +22,7 @@ WindowManager::WindowManager(ChannelManager &_channelManager, ColorScheme &_colo
void WindowManager::initMainWindow() void WindowManager::initMainWindow()
{ {
this->selectedWindow = this->mainWindow = this->selectedWindow = this->mainWindow =
new widgets::Window(this->channelManager, this->colorScheme, true); new widgets::Window("main", this->channelManager, this->colorScheme, true);
} }
static const std::string &getSettingsPath() static const std::string &getSettingsPath()
@ -68,7 +68,7 @@ widgets::Window &WindowManager::getSelectedWindow()
widgets::Window &WindowManager::createWindow() widgets::Window &WindowManager::createWindow()
{ {
auto *window = new widgets::Window(this->channelManager, this->colorScheme, false); auto *window = new widgets::Window("external", this->channelManager, this->colorScheme, false);
window->loadDefaults(); window->loadDefaults();