rework chat components to make more sense
This commit is contained in:
@@ -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>");
|
||||||
|
@@ -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;
|
|
||||||
};
|
};
|
||||||
|
@@ -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;
|
||||||
{
|
{
|
||||||
|
@@ -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;
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user