rework chat components to make more sense

This commit is contained in:
ouwou
2020-08-29 22:45:27 -04:00
parent a921ee42f9
commit 08e9d2f0ef
4 changed files with 83 additions and 99 deletions

View File

@@ -1,20 +1,13 @@
#include "chatmessage.hpp" #include "chatmessage.hpp"
ChatMessageTextItem::ChatMessageTextItem(const MessageData *data) { ChatMessageContainer::ChatMessageContainer(const MessageData *data) {
UserID = data->Author.ID;
m_main_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL)); m_main_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL));
m_sub_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL)); m_content_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL));
m_meta_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL)); m_meta_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL));
m_author = Gtk::manage(new Gtk::Label); m_author = Gtk::manage(new Gtk::Label);
m_timestamp = Gtk::manage(new Gtk::Label); m_timestamp = Gtk::manage(new Gtk::Label);
m_text = Gtk::manage(new Gtk::TextView);
m_text->set_can_focus(false);
m_text->set_editable(false);
m_text->set_wrap_mode(Gtk::WRAP_WORD_CHAR);
m_text->set_halign(Gtk::ALIGN_FILL);
m_text->set_hexpand(true);
m_text->get_buffer()->set_text(data->Content);
m_text->show();
m_author->set_markup("<span weight=\"bold\">" + Glib::Markup::escape_text(data->Author.Username) + "</span>"); m_author->set_markup("<span weight=\"bold\">" + Glib::Markup::escape_text(data->Author.Username) + "</span>");
m_author->set_single_line_mode(true); m_author->set_single_line_mode(true);
@@ -39,32 +32,38 @@ ChatMessageTextItem::ChatMessageTextItem(const MessageData *data) {
m_meta_box->set_can_focus(false); m_meta_box->set_can_focus(false);
m_meta_box->show(); m_meta_box->show();
m_sub_box->set_can_focus(false); m_content_box->set_can_focus(false);
m_sub_box->show(); m_content_box->show();
m_meta_box->add(*m_author); m_meta_box->add(*m_author);
m_meta_box->add(*m_timestamp); m_meta_box->add(*m_timestamp);
m_sub_box->add(*m_meta_box); m_content_box->add(*m_meta_box);
m_sub_box->add(*m_text); m_main_box->add(*m_content_box);
m_main_box->add(*m_sub_box);
add(*m_main_box); add(*m_main_box);
set_margin_bottom(8); set_margin_bottom(8);
show(); show();
} }
void ChatMessageTextItem::AppendNewContent(std::string content) { void ChatMessageContainer::AddNewContent(Gtk::Widget *widget, bool prepend) {
auto buf = m_text->get_buffer(); if (prepend)
buf->set_text(buf->get_text() + "\n" + content); m_content_box->pack_end(*widget);
else
m_content_box->pack_start(*widget);
} }
void ChatMessageTextItem::PrependNewContent(std::string content) { ChatMessageTextItem::ChatMessageTextItem(const MessageData *data) {
auto buf = m_text->get_buffer(); set_can_focus(false);
buf->set_text(content + "\n" + buf->get_text()); set_editable(false);
set_wrap_mode(Gtk::WRAP_WORD_CHAR);
set_halign(Gtk::ALIGN_FILL);
set_hexpand(true);
get_buffer()->set_text(data->Content);
show();
} }
void ChatMessageTextItem::MarkAsDeleted() { void ChatMessageTextItem::MarkAsDeleted() {
auto buf = m_text->get_buffer(); auto buf = get_buffer();
Gtk::TextBuffer::iterator start, end; Gtk::TextBuffer::iterator start, end;
buf->get_bounds(start, end); buf->get_bounds(start, end);
buf->insert_markup(end, "<span color='#ff0000'> [deleted]</span>"); buf->insert_markup(end, "<span color='#ff0000'> [deleted]</span>");

View File

@@ -7,7 +7,23 @@ enum class ChatDisplayType {
Text, Text,
}; };
class ChatMessageItem : public Gtk::ListBoxRow { // contains the username and timestamp, chat items get stuck into its box
class ChatMessageContainer : public Gtk::ListBoxRow {
public:
Snowflake UserID;
ChatMessageContainer(const MessageData *data);
void AddNewContent(Gtk::Widget *widget, bool prepend = false);
protected:
Gtk::Box *m_main_box;
Gtk::Box *m_content_box;
Gtk::Box *m_meta_box;
Gtk::Label *m_author;
Gtk::Label *m_timestamp;
};
class ChatMessageItem {
public: public:
Snowflake ID; Snowflake ID;
ChatDisplayType MessageType; ChatDisplayType MessageType;
@@ -15,18 +31,10 @@ public:
virtual void MarkAsDeleted() = 0; virtual void MarkAsDeleted() = 0;
}; };
class ChatMessageTextItem : public ChatMessageItem { class ChatMessageTextItem
: public Gtk::TextView // oh well
, public ChatMessageItem {
public: public:
ChatMessageTextItem(const MessageData *data); ChatMessageTextItem(const MessageData *data);
void AppendNewContent(std::string content);
void PrependNewContent(std::string content);
virtual void MarkAsDeleted(); virtual void MarkAsDeleted();
protected:
Gtk::Box *m_main_box;
Gtk::Box *m_sub_box;
Gtk::Box *m_meta_box;
Gtk::Label *m_author;
Gtk::Label *m_timestamp;
Gtk::TextView *m_text;
}; };

View File

@@ -88,60 +88,44 @@ ChatDisplayType ChatWindow::GetMessageDisplayType(const MessageData *data) {
return ChatDisplayType::Unknown; return ChatDisplayType::Unknown;
} }
ChatMessageItem *ChatWindow::CreateChatEntryComponentText(const MessageData *data) { void ChatWindow::ProcessMessage(const MessageData *data, bool prepend) {
return Gtk::manage(new ChatMessageTextItem(data)); ChatMessageContainer *last_row = nullptr;
} bool should_attach = false;
if (m_num_rows > 0) {
ChatMessageItem *ChatWindow::CreateChatEntryComponent(const MessageData *data) { if (prepend)
ChatMessageItem *item = nullptr; last_row = dynamic_cast<ChatMessageContainer *>(m_listbox->get_row_at_index(0));
switch (GetMessageDisplayType(data)) { else
case ChatDisplayType::Text: last_row = dynamic_cast<ChatMessageContainer *>(m_listbox->get_row_at_index(m_num_rows - 1));
item = CreateChatEntryComponentText(data); if (last_row != nullptr) { // this should always be true tbh
break; if (last_row->UserID == data->Author.ID)
should_attach = true;
}
} }
if (item != nullptr) auto type = GetMessageDisplayType(data);
item->ID = data->ID;
return item; ChatMessageContainer *container;
} if (should_attach) {
container = last_row;
void ChatWindow::ProcessMessage(const MessageData *data, bool prepend) { } else {
auto create_new_row = [&]() { container = Gtk::manage(new ChatMessageContainer(data)); // only accesses timestamp and user
auto *item = CreateChatEntryComponent(data);
if (item != nullptr) {
if (prepend)
m_listbox->prepend(*item);
else
m_listbox->add(*item);
m_num_rows++; m_num_rows++;
} }
};
// if the last row's message's author is the same as the new one's, then append the new message content to the last row // actual content
if (m_num_rows > 0) { if (type == ChatDisplayType::Text) {
ChatMessageItem *item; auto *text = Gtk::manage(new ChatMessageTextItem(data));
if (prepend) container->AddNewContent(text, prepend);
item = dynamic_cast<ChatMessageItem *>(m_listbox->get_row_at_index(0)); m_id_to_widget[data->ID] = text;
else
item = dynamic_cast<ChatMessageItem *>(m_listbox->get_row_at_index(m_num_rows - 1));
assert(item != nullptr);
auto *previous_data = m_abaddon->GetDiscordClient().GetMessage(item->ID);
auto new_type = GetMessageDisplayType(data);
auto old_type = GetMessageDisplayType(previous_data);
if ((data->Author.ID == previous_data->Author.ID) && (new_type == old_type && new_type == ChatDisplayType::Text)) {
auto *text_item = dynamic_cast<ChatMessageTextItem *>(item);
if (prepend)
text_item->PrependNewContent(data->Content);
else
text_item->AppendNewContent(data->Content);
} else {
create_new_row();
} }
} else {
create_new_row(); container->show_all();
if (!should_attach) {
if (prepend)
m_listbox->prepend(*container);
else
m_listbox->add(*container);
} }
} }
@@ -245,21 +229,13 @@ void ChatWindow::DeleteMessageInternal() {
m_message_delete_queue.pop(); m_message_delete_queue.pop();
} }
ChatMessageItem *row = nullptr; if (m_id_to_widget.find(id) == m_id_to_widget.end())
for (const auto &child : m_listbox->get_children()) { return;
ChatMessageItem *tmp = dynamic_cast<ChatMessageItem *>(child);
if (tmp == nullptr) continue;
if (tmp->ID == id) {
row = tmp;
break;
}
}
if (row == nullptr) return;
// todo actually delete it when it becomes setting // todo actually delete it when it becomes setting
row->MarkAsDeleted(); auto *item = m_id_to_widget.at(id);
item->MarkAsDeleted();
} }
void ChatWindow::SetMessagesInternal() { void ChatWindow::SetMessagesInternal() {
@@ -272,6 +248,7 @@ void ChatWindow::SetMessagesInternal() {
} }
m_num_rows = 0; m_num_rows = 0;
m_id_to_widget.clear();
std::unordered_set<const MessageData *> *msgs; std::unordered_set<const MessageData *> *msgs;
{ {

View File

@@ -2,6 +2,7 @@
#include <gtkmm.h> #include <gtkmm.h>
#include <queue> #include <queue>
#include <mutex> #include <mutex>
#include <unordered_map>
#include "chatmessage.hpp" #include "chatmessage.hpp"
#include "../discord/discord.hpp" #include "../discord/discord.hpp"
@@ -27,10 +28,9 @@ protected:
void AddNewHistoryInternal(); void AddNewHistoryInternal();
void DeleteMessageInternal(); void DeleteMessageInternal();
ChatDisplayType GetMessageDisplayType(const MessageData *data); ChatDisplayType GetMessageDisplayType(const MessageData *data);
ChatMessageItem *CreateChatEntryComponentText(const MessageData *data);
ChatMessageItem *CreateChatEntryComponent(const MessageData *data);
void ProcessMessage(const MessageData *data, bool prepend = false); void ProcessMessage(const MessageData *data, bool prepend = false);
int m_num_rows = 0; // youd think thered be a Gtk::ListBox::get_row_count or something but nope int m_num_rows = 0; // youd think thered be a Gtk::ListBox::get_row_count or something but nope
std::unordered_map<Snowflake, ChatMessageItem *> m_id_to_widget;
bool m_scroll_to_bottom = true; bool m_scroll_to_bottom = true;