abaddon/components/chatmessage.cpp

874 lines
32 KiB
C++
Raw Normal View History

#include "chatmessage.hpp"
2020-08-30 06:00:56 +00:00
#include "../abaddon.hpp"
2020-09-06 01:05:52 +00:00
#include "../util.hpp"
2020-10-24 23:42:06 +00:00
#include <unordered_map>
2020-09-26 03:02:13 +00:00
ChatMessageItemContainer::ChatMessageItemContainer() {
m_main = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL));
add(*m_main);
2020-08-28 22:21:08 +00:00
2020-09-26 03:02:13 +00:00
m_menu_copy_id = Gtk::manage(new Gtk::MenuItem("Copy ID"));
m_menu_copy_id->signal_activate().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::on_menu_copy_id));
2020-08-30 06:00:56 +00:00
m_menu.append(*m_menu_copy_id);
2020-09-26 03:02:13 +00:00
m_menu_delete_message = Gtk::manage(new Gtk::MenuItem("Delete Message"));
m_menu_delete_message->signal_activate().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::on_menu_delete_message));
2020-08-30 06:00:56 +00:00
m_menu.append(*m_menu_delete_message);
2020-09-26 03:02:13 +00:00
m_menu_edit_message = Gtk::manage(new Gtk::MenuItem("Edit Message"));
m_menu_edit_message->signal_activate().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::on_menu_edit_message));
2020-08-31 02:55:36 +00:00
m_menu.append(*m_menu_edit_message);
2020-10-13 03:55:52 +00:00
m_menu_copy_content = Gtk::manage(new Gtk::MenuItem("Copy Content"));
m_menu_copy_content->signal_activate().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::on_menu_copy_content));
m_menu.append(*m_menu_copy_content);
2020-08-30 06:00:56 +00:00
m_menu.show_all();
m_link_menu_copy = Gtk::manage(new Gtk::MenuItem("Copy Link"));
m_link_menu_copy->signal_activate().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::on_link_menu_copy));
m_link_menu.append(*m_link_menu_copy);
m_link_menu.show_all();
2020-08-30 06:00:56 +00:00
}
2020-09-26 03:02:13 +00:00
ChatMessageItemContainer *ChatMessageItemContainer::FromMessage(Snowflake id) {
const auto *data = Abaddon::Get().GetDiscordClient().GetMessage(id);
if (data == nullptr) return nullptr;
2020-08-31 02:55:36 +00:00
2020-09-26 03:02:13 +00:00
auto *container = Gtk::manage(new ChatMessageItemContainer);
container->ID = data->ID;
container->ChannelID = data->ChannelID;
2020-08-30 06:00:56 +00:00
2020-09-26 04:49:25 +00:00
if (data->Content.size() > 0 || data->Type != MessageType::DEFAULT) {
2020-09-30 04:12:38 +00:00
container->m_text_component = container->CreateTextComponent(data);
container->AttachGuildMenuHandler(container->m_text_component);
2020-09-26 03:02:13 +00:00
container->m_main->add(*container->m_text_component);
}
2020-08-30 06:00:56 +00:00
2020-09-26 03:02:13 +00:00
// there should only ever be 1 embed (i think?)
if (data->Embeds.size() == 1) {
2020-09-30 04:12:38 +00:00
container->m_embed_component = container->CreateEmbedComponent(data);
container->AttachGuildMenuHandler(container->m_embed_component);
2020-09-26 03:02:13 +00:00
container->m_main->add(*container->m_embed_component);
}
2020-08-30 06:00:56 +00:00
2020-09-30 04:12:38 +00:00
// i dont think attachments can be edited
// also this can definitely be done much better holy shit
for (const auto &a : data->Attachments) {
const auto last3 = a.ProxyURL.substr(a.ProxyURL.length() - 3);
if (last3 == "png" || last3 == "jpg") {
auto *widget = container->CreateImageComponent(a);
2020-11-06 07:36:08 +00:00
container->m_main->add(*widget);
2020-09-30 04:12:38 +00:00
} else {
auto *widget = container->CreateAttachmentComponent(a);
container->m_main->add(*widget);
}
}
2020-11-02 22:36:10 +00:00
// only 1?
if (data->Stickers.has_value()) {
const auto &sticker = data->Stickers.value()[0];
2020-11-06 07:36:08 +00:00
// todo: lottie, proper apng
2020-11-02 22:36:10 +00:00
if (sticker.FormatType == StickerFormatType::PNG || sticker.FormatType == StickerFormatType::APNG) {
auto *widget = container->CreateStickerComponent(sticker);
container->m_main->add(*widget);
}
}
2020-09-26 03:02:13 +00:00
container->UpdateAttributes();
2020-09-26 03:02:13 +00:00
return container;
}
2020-09-30 04:12:38 +00:00
// this doesnt rly make sense
2020-09-26 03:02:13 +00:00
void ChatMessageItemContainer::UpdateContent() {
const auto *data = Abaddon::Get().GetDiscordClient().GetMessage(ID);
if (m_text_component != nullptr)
2020-10-05 06:09:15 +00:00
UpdateTextComponent(m_text_component);
2020-08-31 00:24:02 +00:00
2020-11-07 04:57:05 +00:00
if (m_embed_component != nullptr) {
2020-09-26 03:02:13 +00:00
delete m_embed_component;
2020-11-07 04:57:05 +00:00
m_embed_component = nullptr;
}
2020-08-29 05:14:07 +00:00
if (data->Embeds.size() == 1) {
m_embed_component = CreateEmbedComponent(data);
if (m_embed_imgurl.size() > 0) {
m_signal_image_load.emit(m_embed_imgurl);
2020-09-26 03:02:13 +00:00
}
AttachGuildMenuHandler(m_embed_component);
m_main->add(*m_embed_component);
2020-09-26 03:02:13 +00:00
}
2020-08-31 00:24:02 +00:00
}
void ChatMessageItemContainer::UpdateImage(std::string url, Glib::RefPtr<Gdk::Pixbuf> buf) {
if (!buf) return;
if (m_embed_img != nullptr && m_embed_imgurl == url) {
int w, h;
m_embed_img->get_size_request(w, h);
m_embed_img->property_pixbuf() = buf->scale_simple(w, h, Gdk::INTERP_BILINEAR);
return;
2020-09-30 04:12:38 +00:00
}
auto it = m_img_loadmap.find(url);
if (it != m_img_loadmap.end()) {
int w, h;
GetImageDimensions(it->second.second.Width, it->second.second.Height, w, h);
it->second.first->property_pixbuf() = buf->scale_simple(w, h, Gdk::INTERP_BILINEAR);
2020-09-30 04:12:38 +00:00
}
}
2020-09-26 03:02:13 +00:00
void ChatMessageItemContainer::UpdateAttributes() {
const auto *data = Abaddon::Get().GetDiscordClient().GetMessage(ID);
if (data == nullptr) return;
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
const bool deleted = data->IsDeleted();
const bool edited = data->IsEdited();
if (!deleted && !edited) return;
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
if (m_attrib_label == nullptr) {
m_attrib_label = Gtk::manage(new Gtk::Label);
m_attrib_label->set_halign(Gtk::ALIGN_START);
m_attrib_label->show();
m_main->add(*m_attrib_label); // todo: maybe insert markup into existing text widget's buffer if the circumstances are right (or pack horizontally)
}
2020-09-26 03:02:13 +00:00
if (deleted)
m_attrib_label->set_markup("<span color='#ff0000'>[deleted]</span>");
else if (edited)
m_attrib_label->set_markup("<span color='#999999'>[edited]</span>");
2020-09-03 05:54:40 +00:00
}
bool ChatMessageItemContainer::EmitImageLoad(std::string url) {
m_signal_image_load.emit(url);
return false;
}
2020-09-30 04:12:38 +00:00
void ChatMessageItemContainer::AddClickHandler(Gtk::Widget *widget, std::string url) {
// clang-format off
widget->signal_button_press_event().connect([url](GdkEventButton *event) -> bool {
if (event->type == Gdk::BUTTON_PRESS && event->button == GDK_BUTTON_PRIMARY) {
LaunchBrowser(url);
return false;
}
return true;
}, false);
// clang-format on
}
2020-09-26 03:02:13 +00:00
Gtk::TextView *ChatMessageItemContainer::CreateTextComponent(const Message *data) {
auto *tv = Gtk::manage(new Gtk::TextView);
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
tv->get_style_context()->add_class("message-text");
tv->set_can_focus(false);
tv->set_editable(false);
tv->set_wrap_mode(Gtk::WRAP_WORD_CHAR);
tv->set_halign(Gtk::ALIGN_FILL);
tv->set_hexpand(true);
2020-09-26 04:49:25 +00:00
2020-10-05 06:09:15 +00:00
UpdateTextComponent(tv);
return tv;
}
void ChatMessageItemContainer::UpdateTextComponent(Gtk::TextView *tv) {
const auto *data = Abaddon::Get().GetDiscordClient().GetMessage(ID);
if (data == nullptr)
return;
2020-09-26 04:49:25 +00:00
auto b = tv->get_buffer();
2020-10-05 06:09:15 +00:00
b->set_text("");
Gtk::TextBuffer::iterator s, e;
2020-09-26 04:49:25 +00:00
b->get_bounds(s, e);
switch (data->Type) {
case MessageType::DEFAULT:
2020-10-13 00:20:37 +00:00
b->insert_markup(s, Glib::Markup::escape_text(data->Content));
HandleUserMentions(tv);
2020-10-10 03:14:57 +00:00
HandleLinks(tv);
HandleChannelMentions(tv);
2020-10-24 23:42:06 +00:00
HandleEmojis(tv);
2020-09-26 04:49:25 +00:00
break;
case MessageType::GUILD_MEMBER_JOIN:
b->insert_markup(s, "<span color='#999999'><i>[user joined]</i></span>");
break;
case MessageType::CHANNEL_PINNED_MESSAGE:
b->insert_markup(s, "<span color='#999999'><i>[message pinned]</i></span>");
break;
default: break;
}
2020-09-26 03:02:13 +00:00
}
2020-11-06 07:36:08 +00:00
Gtk::Widget *ChatMessageItemContainer::CreateEmbedComponent(const Message *data) {
2020-09-26 03:02:13 +00:00
Gtk::EventBox *ev = Gtk::manage(new Gtk::EventBox);
ev->set_can_focus(true);
Gtk::Box *main = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL));
const auto &embed = data->Embeds[0];
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
if (embed.Author.Name.length() > 0) {
2020-09-04 02:36:57 +00:00
auto *author_lbl = Gtk::manage(new Gtk::Label);
author_lbl->set_halign(Gtk::ALIGN_START);
author_lbl->set_line_wrap(true);
author_lbl->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
author_lbl->set_hexpand(false);
2020-09-26 03:02:13 +00:00
author_lbl->set_text(embed.Author.Name);
2020-09-04 02:36:57 +00:00
author_lbl->get_style_context()->add_class("embed-author");
2020-09-26 03:02:13 +00:00
main->pack_start(*author_lbl);
2020-09-04 02:36:57 +00:00
}
2020-09-26 03:02:13 +00:00
if (embed.Title.length() > 0) {
2020-09-03 05:54:40 +00:00
auto *title_label = Gtk::manage(new Gtk::Label);
title_label->set_use_markup(true);
2020-09-26 03:02:13 +00:00
title_label->set_markup("<b>" + Glib::Markup::escape_text(embed.Title) + "</b>");
2020-09-03 05:54:40 +00:00
title_label->set_halign(Gtk::ALIGN_CENTER);
2020-09-03 21:44:23 +00:00
title_label->set_hexpand(false);
2020-09-09 22:32:45 +00:00
title_label->get_style_context()->add_class("embed-title");
2020-09-26 03:02:13 +00:00
title_label->set_single_line_mode(false);
title_label->set_line_wrap(true);
title_label->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
title_label->set_max_width_chars(50);
main->pack_start(*title_label);
2020-09-03 05:54:40 +00:00
}
2020-09-26 03:02:13 +00:00
if (embed.Description.length() > 0) {
2020-09-03 05:54:40 +00:00
auto *desc_label = Gtk::manage(new Gtk::Label);
2020-09-26 03:02:13 +00:00
desc_label->set_text(embed.Description);
2020-09-03 05:54:40 +00:00
desc_label->set_line_wrap(true);
desc_label->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
desc_label->set_max_width_chars(50);
desc_label->set_halign(Gtk::ALIGN_START);
2020-09-03 21:44:23 +00:00
desc_label->set_hexpand(false);
2020-09-09 22:32:45 +00:00
desc_label->get_style_context()->add_class("embed-description");
2020-09-26 03:02:13 +00:00
main->pack_start(*desc_label);
2020-09-03 05:54:40 +00:00
}
2020-09-26 03:02:13 +00:00
// todo: handle inline fields
if (embed.Fields.size() > 0) {
2020-09-03 22:47:45 +00:00
auto *flow = Gtk::manage(new Gtk::FlowBox);
flow->set_orientation(Gtk::ORIENTATION_HORIZONTAL);
flow->set_min_children_per_line(3);
2020-09-04 02:36:57 +00:00
flow->set_max_children_per_line(3);
2020-09-03 22:47:45 +00:00
flow->set_halign(Gtk::ALIGN_START);
flow->set_hexpand(false);
flow->set_column_spacing(10);
2020-09-04 02:36:57 +00:00
flow->set_selection_mode(Gtk::SELECTION_NONE);
2020-09-26 03:02:13 +00:00
main->pack_start(*flow);
2020-09-03 22:47:45 +00:00
2020-09-26 03:02:13 +00:00
for (const auto &field : embed.Fields) {
2020-09-03 22:47:45 +00:00
auto *field_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL));
auto *field_lbl = Gtk::manage(new Gtk::Label);
auto *field_val = Gtk::manage(new Gtk::Label);
field_box->set_hexpand(false);
field_box->set_halign(Gtk::ALIGN_START);
2020-09-04 02:36:57 +00:00
field_box->set_valign(Gtk::ALIGN_START);
2020-09-03 22:47:45 +00:00
field_lbl->set_hexpand(false);
field_lbl->set_halign(Gtk::ALIGN_START);
2020-09-04 02:36:57 +00:00
field_lbl->set_valign(Gtk::ALIGN_START);
2020-09-03 22:47:45 +00:00
field_val->set_hexpand(false);
field_val->set_halign(Gtk::ALIGN_START);
2020-09-04 02:36:57 +00:00
field_val->set_valign(Gtk::ALIGN_START);
2020-09-03 22:47:45 +00:00
field_lbl->set_use_markup(true);
field_lbl->set_markup("<b>" + Glib::Markup::escape_text(field.Name) + "</b>");
field_lbl->set_max_width_chars(20);
field_lbl->set_line_wrap(true);
field_lbl->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
field_val->set_text(field.Value);
field_val->set_max_width_chars(20);
field_val->set_line_wrap(true);
field_val->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
field_box->pack_start(*field_lbl);
field_box->pack_start(*field_val);
2020-09-09 22:32:45 +00:00
field_lbl->get_style_context()->add_class("embed-field-title");
field_val->get_style_context()->add_class("embed-field-value");
2020-09-03 22:47:45 +00:00
flow->insert(*field_box, -1);
}
}
bool is_img = embed.Image.URL.size() > 0;
bool is_thumb = embed.Thumbnail.URL.size() > 0;
if (is_img || is_thumb) {
2020-09-30 04:12:38 +00:00
auto *img = Gtk::manage(new Gtk::Image);
img->set_halign(Gtk::ALIGN_CENTER);
int w, h;
if (is_img)
GetImageDimensions(embed.Image.Width, embed.Image.Height, w, h, 200, 150);
2020-09-30 04:12:38 +00:00
else
GetImageDimensions(embed.Thumbnail.Width, embed.Thumbnail.Height, w, h, 200, 150);
2020-09-30 04:12:38 +00:00
img->set_size_request(w, h);
main->pack_start(*img);
m_embed_img = img;
if (is_img)
2020-09-30 04:12:38 +00:00
m_embed_imgurl = embed.Image.ProxyURL;
else
m_embed_imgurl = embed.Thumbnail.ProxyURL;
Glib::signal_idle().connect(sigc::bind(sigc::mem_fun(*this, &ChatMessageItemContainer::EmitImageLoad), m_embed_imgurl));
2020-09-30 04:12:38 +00:00
}
2020-09-26 03:02:13 +00:00
if (embed.Footer.Text.length() > 0) {
2020-09-04 02:36:57 +00:00
auto *footer_lbl = Gtk::manage(new Gtk::Label);
footer_lbl->set_halign(Gtk::ALIGN_START);
footer_lbl->set_line_wrap(true);
footer_lbl->set_line_wrap_mode(Pango::WRAP_WORD_CHAR);
footer_lbl->set_hexpand(false);
2020-09-26 03:02:13 +00:00
footer_lbl->set_text(embed.Footer.Text);
2020-09-04 02:36:57 +00:00
footer_lbl->get_style_context()->add_class("embed-footer");
2020-09-26 03:02:13 +00:00
main->pack_start(*footer_lbl);
2020-09-04 02:36:57 +00:00
}
2020-09-26 03:02:13 +00:00
auto style = main->get_style_context();
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
if (embed.Color != -1) {
2020-09-03 05:54:40 +00:00
auto provider = Gtk::CssProvider::create(); // this seems wrong
2020-09-26 03:02:13 +00:00
std::string css = ".embed { border-left: 2px solid #" + IntToCSSColor(embed.Color) + "; }";
2020-09-06 01:05:52 +00:00
provider->load_from_data(css);
2020-09-03 05:54:40 +00:00
style->add_provider(provider, GTK_STYLE_PROVIDER_PRIORITY_APPLICATION);
}
style->add_class("embed");
2020-09-26 03:02:13 +00:00
main->set_margin_bottom(8);
main->set_hexpand(false);
main->set_hexpand(false);
main->set_halign(Gtk::ALIGN_START);
main->set_halign(Gtk::ALIGN_START);
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
ev->add(*main);
ev->show_all();
2020-09-03 05:54:40 +00:00
2020-09-26 03:02:13 +00:00
return ev;
}
2020-09-03 05:54:40 +00:00
2020-11-06 07:36:08 +00:00
Gtk::Widget *ChatMessageItemContainer::CreateImageComponent(const AttachmentData &data) {
2020-09-30 04:12:38 +00:00
int w, h;
GetImageDimensions(data.Width, data.Height, w, h);
2020-09-30 04:12:38 +00:00
2020-11-06 07:36:08 +00:00
Gtk::EventBox *ev = Gtk::manage(new Gtk::EventBox);
2020-09-30 04:12:38 +00:00
Gtk::Image *widget = Gtk::manage(new Gtk::Image);
2020-11-06 07:36:08 +00:00
ev->add(*widget);
2020-09-30 04:12:38 +00:00
widget->set_halign(Gtk::ALIGN_START);
widget->set_size_request(w, h);
2020-11-06 07:36:08 +00:00
AttachGuildMenuHandler(ev);
AddClickHandler(ev, data.URL);
HandleImage(data, widget, data.ProxyURL);
2020-09-30 04:12:38 +00:00
2020-11-06 07:36:08 +00:00
return ev;
2020-09-30 04:12:38 +00:00
}
2020-11-06 07:36:08 +00:00
Gtk::Widget *ChatMessageItemContainer::CreateAttachmentComponent(const AttachmentData &data) {
2020-09-30 04:12:38 +00:00
auto *ev = Gtk::manage(new Gtk::EventBox);
auto *btn = Gtk::manage(new Gtk::Label(data.Filename + " " + HumanReadableBytes(data.Bytes))); // Gtk::LinkButton flat out doesn't work :D
2020-11-06 07:36:08 +00:00
ev->get_style_context()->add_class("message-attachment-box");
2020-09-30 04:12:38 +00:00
ev->add(*btn);
2020-11-06 07:36:08 +00:00
AttachGuildMenuHandler(ev);
AddClickHandler(ev, data.URL);
return ev;
2020-09-30 04:12:38 +00:00
}
2020-11-06 07:36:08 +00:00
Gtk::Widget *ChatMessageItemContainer::CreateStickerComponent(const Sticker &data) {
2020-11-02 22:36:10 +00:00
auto *box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL));
auto *imgw = Gtk::manage(new Gtk::Image);
2020-11-06 07:36:08 +00:00
box->add(*imgw);
2020-11-02 22:36:10 +00:00
auto &img = Abaddon::Get().GetImageManager();
if (data.FormatType == StickerFormatType::PNG || data.FormatType == StickerFormatType::APNG) {
// clang-format off
img.LoadFromURL(data.GetURL(), sigc::track_obj([this, imgw](const Glib::RefPtr<Gdk::Pixbuf> &pixbuf) {
imgw->property_pixbuf() = pixbuf;
}, imgw));
// clang-format on
}
2020-11-06 07:36:08 +00:00
AttachGuildMenuHandler(box);
2020-11-02 22:36:10 +00:00
return box;
}
2020-09-30 04:12:38 +00:00
void ChatMessageItemContainer::HandleImage(const AttachmentData &data, Gtk::Image *img, std::string url) {
m_img_loadmap[url] = std::make_pair(img, data);
// ask the chatwindow to call UpdateImage because dealing with lifetimes sucks
Glib::signal_idle().connect(sigc::bind(sigc::mem_fun(*this, &ChatMessageItemContainer::EmitImageLoad), url));
2020-09-30 04:12:38 +00:00
}
Glib::ustring ChatMessageItemContainer::GetText(const Glib::RefPtr<Gtk::TextBuffer> &buf) {
2020-10-24 23:42:06 +00:00
Gtk::TextBuffer::iterator a, b;
buf->get_bounds(a, b);
auto slice = buf->get_slice(a, b, true);
return slice;
}
2020-10-13 00:20:37 +00:00
void ChatMessageItemContainer::HandleUserMentions(Gtk::TextView *tv) {
constexpr static const auto mentions_regex = R"(<@!?(\d+)>)";
2020-10-05 06:09:15 +00:00
static auto rgx = Glib::Regex::create(mentions_regex);
2020-10-05 06:09:15 +00:00
2020-10-13 00:20:37 +00:00
auto buf = tv->get_buffer();
Glib::ustring text = GetText(buf);
2020-10-13 00:20:37 +00:00
const auto &discord = Abaddon::Get().GetDiscordClient();
2020-10-05 06:09:15 +00:00
int startpos = 0;
Glib::MatchInfo match;
while (rgx->match(text, startpos, match)) {
int mstart, mend;
if (!match.fetch_pos(0, mstart, mend)) break;
const Glib::ustring user_id = match.fetch(1);
2020-10-13 00:20:37 +00:00
const auto *user = discord.GetUser(user_id);
2020-10-05 06:09:15 +00:00
const auto *channel = discord.GetChannel(ChannelID);
2020-10-13 00:20:37 +00:00
if (user == nullptr || channel == nullptr) {
startpos = mend;
2020-10-13 00:20:37 +00:00
continue;
}
Glib::ustring replacement;
2020-10-05 06:09:15 +00:00
if (channel->Type == ChannelType::DM || channel->Type == ChannelType::GROUP_DM)
2020-10-13 00:20:37 +00:00
replacement = "<b>@" + Glib::Markup::escape_text(user->Username) + "#" + user->Discriminator + "</b>";
else {
const auto role_id = user->GetHoistedRole(channel->GuildID, true);
const auto *role = discord.GetRole(role_id);
if (role == nullptr)
replacement = "<b>@" + Glib::Markup::escape_text(user->Username) + "#" + user->Discriminator + "</b>";
else
replacement = "<b><span color=\"#" + IntToCSSColor(role->Color) + "\">@" + Glib::Markup::escape_text(user->Username) + "#" + user->Discriminator + "</span></b>";
}
// regex returns byte positions and theres no straightforward way in the c++ bindings to deal with that :(
const auto chars_start = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mstart);
const auto chars_end = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mend);
const auto start_it = buf->get_iter_at_offset(chars_start);
const auto end_it = buf->get_iter_at_offset(chars_end);
2020-10-05 06:09:15 +00:00
auto it = buf->erase(start_it, end_it);
2020-10-13 00:20:37 +00:00
buf->insert_markup(it, replacement);
2020-10-05 06:09:15 +00:00
text = GetText(buf);
startpos = 0;
2020-10-13 00:20:37 +00:00
}
2020-10-05 06:09:15 +00:00
}
2020-10-24 23:42:06 +00:00
void ChatMessageItemContainer::HandleStockEmojis(Gtk::TextView *tv) {
auto buf = tv->get_buffer();
auto text = GetText(buf);
2020-10-24 23:42:06 +00:00
auto &emojis = Abaddon::Get().GetEmojis();
int searchpos;
for (const auto &pattern : emojis.GetPatterns()) {
searchpos = 0;
Glib::RefPtr<Gdk::Pixbuf> pixbuf;
while (true) {
size_t r = text.find(pattern, searchpos);
if (r == Glib::ustring::npos) break;
if (!pixbuf) {
pixbuf = emojis.GetPixBuf(pattern);
if (pixbuf)
pixbuf = pixbuf->scale_simple(24, 24, Gdk::INTERP_BILINEAR);
else
break;
}
2020-10-24 23:42:06 +00:00
searchpos = r + pattern.size();
const auto start_it = buf->get_iter_at_offset(r);
const auto end_it = buf->get_iter_at_offset(r + pattern.size());
2020-10-24 23:42:06 +00:00
auto it = buf->erase(start_it, end_it);
buf->insert_pixbuf(it, pixbuf);
2020-10-24 23:42:06 +00:00
int alen = text.size();
text = GetText(buf);
2020-10-24 23:42:06 +00:00
int blen = text.size();
searchpos -= (alen - blen);
}
}
}
void ChatMessageItemContainer::HandleCustomEmojis(Gtk::TextView *tv) {
static auto rgx = Glib::Regex::create(R"(<a?:([\w\d_]+):(\d+)>)");
auto &img = Abaddon::Get().GetImageManager();
auto buf = tv->get_buffer();
auto text = GetText(buf);
2020-10-24 23:42:06 +00:00
Glib::MatchInfo match;
int startpos = 0;
while (rgx->match(text, startpos, match)) {
int mstart, mend;
if (!match.fetch_pos(0, mstart, mend)) break;
const auto chars_start = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mstart);
const auto chars_end = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mend);
auto start_it = buf->get_iter_at_offset(chars_start);
auto end_it = buf->get_iter_at_offset(chars_end);
2020-10-24 23:42:06 +00:00
startpos = mend;
auto pixbuf = img.GetFromURLIfCached(Emoji::URLFromID(match.fetch(2)));
if (pixbuf) {
auto it = buf->erase(start_it, end_it);
int alen = text.size();
text = GetText(buf);
2020-10-24 23:42:06 +00:00
int blen = text.size();
startpos -= (alen - blen);
buf->insert_pixbuf(it, pixbuf->scale_simple(24, 24, Gdk::INTERP_BILINEAR));
} else {
// clang-format off
// can't erase before pixbuf is ready or else marks that are in the same pos get mixed up
auto mark_start = buf->create_mark(start_it, false);
end_it.backward_char();
auto mark_end = buf->create_mark(end_it, false);
img.LoadFromURL(Emoji::URLFromID(match.fetch(2)), sigc::track_obj([this, buf, mark_start, mark_end](Glib::RefPtr<Gdk::Pixbuf> pixbuf) {
auto start_it = mark_start->get_iter();
auto end_it = mark_end->get_iter();
end_it.forward_char();
buf->delete_mark(mark_start);
buf->delete_mark(mark_end);
auto it = buf->erase(start_it, end_it);
buf->insert_pixbuf(it, pixbuf->scale_simple(24, 24, Gdk::INTERP_BILINEAR));
}, tv));
// clang-format on
}
text = GetText(buf);
2020-10-24 23:42:06 +00:00
}
}
void ChatMessageItemContainer::HandleEmojis(Gtk::TextView *tv) {
2020-11-03 06:52:19 +00:00
static bool emojis = Abaddon::Get().GetSettings().GetSettingBool("gui", "emojis", true);
2020-10-24 23:42:06 +00:00
if (emojis) {
HandleStockEmojis(tv);
HandleCustomEmojis(tv);
}
}
void ChatMessageItemContainer::HandleChannelMentions(Gtk::TextView *tv) {
static auto rgx = Glib::Regex::create(R"(<#(\d+)>)");
2020-10-10 03:14:57 +00:00
tv->signal_button_press_event().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::OnClickChannel), false);
2020-10-10 03:14:57 +00:00
auto buf = tv->get_buffer();
Glib::ustring text = GetText(buf);
const auto &discord = Abaddon::Get().GetDiscordClient();
2020-10-10 03:14:57 +00:00
int startpos = 0;
Glib::MatchInfo match;
while (rgx->match(text, startpos, match)) {
int mstart, mend;
match.fetch_pos(0, mstart, mend);
std::string channel_id = match.fetch(1);
const auto *chan = discord.GetChannel(channel_id);
if (chan == nullptr) {
startpos = mend;
continue;
}
auto tag = buf->create_tag();
m_channel_tagmap[tag] = channel_id;
tag->property_weight() = Pango::WEIGHT_BOLD;
const auto chars_start = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mstart);
const auto chars_end = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mend);
const auto erase_from = buf->get_iter_at_offset(chars_start);
const auto erase_to = buf->get_iter_at_offset(chars_end);
auto it = buf->erase(erase_from, erase_to);
const std::string replacement = "#" + chan->Name;
it = buf->insert_with_tag(it, "#" + chan->Name, tag);
// rescan the whole thing so i dont have to deal with fixing match positions
text = GetText(buf);
startpos = 0;
}
}
// a lot of repetition here so there should probably just be one slot for textview's button-press
bool ChatMessageItemContainer::OnClickChannel(GdkEventButton *ev) {
if (m_text_component == nullptr) return false;
if (ev->type != Gdk::BUTTON_PRESS) return false;
if (ev->button != GDK_BUTTON_PRIMARY) return false;
auto buf = m_text_component->get_buffer();
Gtk::TextBuffer::iterator start, end;
buf->get_selection_bounds(start, end); // no open if selection
if (start.get_offset() != end.get_offset())
return false;
int x, y;
m_text_component->window_to_buffer_coords(Gtk::TEXT_WINDOW_WIDGET, ev->x, ev->y, x, y);
Gtk::TextBuffer::iterator iter;
m_text_component->get_iter_at_location(iter, x, y);
const auto tags = iter.get_tags();
for (auto tag : tags) {
const auto it = m_channel_tagmap.find(tag);
if (it != m_channel_tagmap.end()) {
m_signal_action_channel_click.emit(it->second);
return true;
}
}
return false;
}
void ChatMessageItemContainer::on_link_menu_copy() {
Gtk::Clipboard::get()->set_text(m_selected_link);
}
void ChatMessageItemContainer::HandleLinks(Gtk::TextView *tv) {
const auto rgx = Glib::Regex::create(R"(\bhttps?:\/\/[^\s]+\.[^\s]+\b)");
tv->signal_button_press_event().connect(sigc::mem_fun(*this, &ChatMessageItemContainer::OnLinkClick), false);
auto buf = tv->get_buffer();
Glib::ustring text = GetText(buf);
2020-10-10 03:14:57 +00:00
// i'd like to let this be done thru css like .message-link { color: #bitch; } but idk how
auto &settings = Abaddon::Get().GetSettings();
static auto link_color = settings.GetSettingString("misc", "linkcolor", "rgba(40, 200, 180, 255)");
2020-10-10 03:14:57 +00:00
int startpos = 0;
Glib::MatchInfo match;
while (rgx->match(text, startpos, match)) {
int mstart, mend;
match.fetch_pos(0, mstart, mend);
std::string link = match.fetch(0);
2020-10-10 03:14:57 +00:00
auto tag = buf->create_tag();
m_link_tagmap[tag] = link;
2020-10-10 03:14:57 +00:00
tag->property_foreground_rgba() = Gdk::RGBA(link_color);
tag->set_property("underline", 1); // stupid workaround for vcpkg bug (i think)
2020-10-10 03:14:57 +00:00
const auto chars_start = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mstart);
const auto chars_end = g_utf8_pointer_to_offset(text.c_str(), text.c_str() + mend);
const auto erase_from = buf->get_iter_at_offset(chars_start);
const auto erase_to = buf->get_iter_at_offset(chars_end);
auto it = buf->erase(erase_from, erase_to);
it = buf->insert_with_tag(it, link, tag);
2020-10-10 03:14:57 +00:00
startpos = mend;
2020-10-10 03:14:57 +00:00
}
}
bool ChatMessageItemContainer::OnLinkClick(GdkEventButton *ev) {
if (m_text_component == nullptr) return false;
if (ev->type != Gdk::BUTTON_PRESS) return false;
if (ev->button != GDK_BUTTON_PRIMARY && ev->button != GDK_BUTTON_SECONDARY) return false;
2020-10-10 03:14:57 +00:00
auto buf = m_text_component->get_buffer();
Gtk::TextBuffer::iterator start, end;
buf->get_selection_bounds(start, end); // no open if selection
if (start.get_offset() != end.get_offset())
return false;
int x, y;
m_text_component->window_to_buffer_coords(Gtk::TEXT_WINDOW_WIDGET, ev->x, ev->y, x, y);
Gtk::TextBuffer::iterator iter;
m_text_component->get_iter_at_location(iter, x, y);
const auto tags = iter.get_tags();
for (auto tag : tags) {
const auto it = m_link_tagmap.find(tag);
if (it != m_link_tagmap.end()) {
if (ev->button == GDK_BUTTON_PRIMARY) {
LaunchBrowser(it->second);
return true;
} else if (ev->button == GDK_BUTTON_SECONDARY) {
m_selected_link = it->second;
m_link_menu.popup_at_pointer(reinterpret_cast<GdkEvent *>(ev));
return true;
}
2020-10-10 03:14:57 +00:00
}
}
return false;
return false;
2020-10-10 03:14:57 +00:00
}
2020-09-26 03:02:13 +00:00
void ChatMessageItemContainer::ShowMenu(GdkEvent *event) {
const auto &client = Abaddon::Get().GetDiscordClient();
const auto *data = client.GetMessage(ID);
if (data->IsDeleted()) {
m_menu_delete_message->set_sensitive(false);
m_menu_edit_message->set_sensitive(false);
} else {
const bool can_edit = client.GetUserData().ID == data->Author.ID;
const bool can_delete = can_edit || client.HasChannelPermission(client.GetUserData().ID, ChannelID, Permission::MANAGE_MESSAGES);
m_menu_delete_message->set_sensitive(can_delete);
m_menu_edit_message->set_sensitive(can_edit);
2020-09-03 05:54:40 +00:00
}
2020-09-26 03:02:13 +00:00
m_menu.popup_at_pointer(event);
2020-09-03 05:54:40 +00:00
}
2020-09-26 03:02:13 +00:00
void ChatMessageItemContainer::on_menu_copy_id() {
Gtk::Clipboard::get()->set_text(std::to_string(ID));
2020-09-03 05:54:40 +00:00
}
2020-09-22 01:01:32 +00:00
2020-09-26 03:02:13 +00:00
void ChatMessageItemContainer::on_menu_delete_message() {
m_signal_action_delete.emit();
}
void ChatMessageItemContainer::on_menu_edit_message() {
m_signal_action_edit.emit();
}
2020-10-13 03:55:52 +00:00
void ChatMessageItemContainer::on_menu_copy_content() {
const auto *msg = Abaddon::Get().GetDiscordClient().GetMessage(ID);
if (msg != nullptr)
Gtk::Clipboard::get()->set_text(msg->Content);
}
2020-09-26 03:02:13 +00:00
ChatMessageItemContainer::type_signal_action_delete ChatMessageItemContainer::signal_action_delete() {
return m_signal_action_delete;
}
ChatMessageItemContainer::type_signal_action_edit ChatMessageItemContainer::signal_action_edit() {
return m_signal_action_edit;
}
ChatMessageItemContainer::type_signal_channel_click ChatMessageItemContainer::signal_action_channel_click() {
return m_signal_action_channel_click;
}
2020-09-30 04:12:38 +00:00
ChatMessageItemContainer::type_signal_image_load ChatMessageItemContainer::signal_image_load() {
return m_signal_image_load;
}
void ChatMessageItemContainer::AttachGuildMenuHandler(Gtk::Widget *widget) {
2020-11-06 07:36:08 +00:00
// clang-format off
2020-09-26 03:02:13 +00:00
widget->signal_button_press_event().connect([this](GdkEventButton *event) -> bool {
if (event->type == GDK_BUTTON_PRESS && event->button == GDK_BUTTON_SECONDARY) {
ShowMenu(reinterpret_cast<GdkEvent*>(event));
return true;
}
return false;
}, false);
2020-11-06 07:36:08 +00:00
// clang-format on
2020-09-26 03:02:13 +00:00
}
ChatMessageHeader::ChatMessageHeader(const Message *data) {
UserID = data->Author.ID;
ChannelID = data->ChannelID;
m_main_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL));
m_content_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_VERTICAL));
m_meta_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL));
m_author_ev = Gtk::manage(new Gtk::EventBox);
2020-09-26 03:02:13 +00:00
m_author = Gtk::manage(new Gtk::Label);
m_timestamp = Gtk::manage(new Gtk::Label);
auto buf = Abaddon::Get().GetImageManager().GetFromURLIfCached(data->Author.GetAvatarURL());
if (buf)
m_avatar = Gtk::manage(new Gtk::Image(buf));
else
m_avatar = Gtk::manage(new Gtk::Image(Abaddon::Get().GetImageManager().GetPlaceholder(32)));
get_style_context()->add_class("message-container");
m_author->get_style_context()->add_class("message-container-author");
m_timestamp->get_style_context()->add_class("message-container-timestamp");
m_avatar->get_style_context()->add_class("message-container-avatar");
m_avatar->set_valign(Gtk::ALIGN_START);
m_avatar->set_margin_right(10);
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_line_wrap(false);
m_author->set_ellipsize(Pango::ELLIPSIZE_END);
m_author->set_xalign(0.f);
m_author->set_can_focus(false);
m_author_ev->signal_button_press_event().connect(sigc::mem_fun(*this, &ChatMessageHeader::on_author_button_press));
2020-09-26 03:02:13 +00:00
2020-11-01 22:10:25 +00:00
if (data->WebhookID.has_value()) {
m_extra = Gtk::manage(new Gtk::Label);
m_extra->get_style_context()->add_class("message-container-extra");
m_extra->set_single_line_mode(true);
m_extra->set_margin_start(12);
m_extra->set_can_focus(false);
m_extra->set_use_markup(true);
m_extra->set_markup("<b>Webhook</b>");
2020-10-07 21:22:20 +00:00
} else if (data->Author.IsBot) {
m_extra = Gtk::manage(new Gtk::Label);
m_extra->get_style_context()->add_class("message-container-extra");
m_extra->set_single_line_mode(true);
m_extra->set_margin_start(12);
m_extra->set_can_focus(false);
m_extra->set_use_markup(true);
m_extra->set_markup("<b>BOT</b>");
}
2020-09-26 03:02:13 +00:00
m_timestamp->set_text(data->Timestamp);
m_timestamp->set_opacity(0.5);
m_timestamp->set_single_line_mode(true);
m_timestamp->set_margin_start(12);
m_timestamp->set_can_focus(false);
m_main_box->set_hexpand(true);
m_main_box->set_vexpand(true);
m_main_box->set_can_focus(true);
m_meta_box->set_can_focus(false);
m_content_box->set_can_focus(false);
m_author_ev->add(*m_author);
m_meta_box->add(*m_author_ev);
if (m_extra != nullptr)
m_meta_box->add(*m_extra);
2020-09-26 03:02:13 +00:00
m_meta_box->add(*m_timestamp);
m_content_box->add(*m_meta_box);
m_main_box->add(*m_avatar);
m_main_box->add(*m_content_box);
add(*m_main_box);
set_margin_bottom(8);
2020-09-22 01:01:32 +00:00
show_all();
2020-09-26 03:02:13 +00:00
UpdateNameColor();
2020-09-22 01:01:32 +00:00
}
2020-09-26 03:02:13 +00:00
void ChatMessageHeader::UpdateNameColor() {
const auto &discord = Abaddon::Get().GetDiscordClient();
const auto guild_id = discord.GetChannel(ChannelID)->GuildID;
const auto role_id = discord.GetMemberHoistedRole(guild_id, UserID, true);
const auto *user = discord.GetUser(UserID);
if (user == nullptr) return;
const auto *role = discord.GetRole(role_id);
std::string md;
if (role != nullptr)
md = "<span weight='bold' color='#" + IntToCSSColor(role->Color) + "'>" + Glib::Markup::escape_text(user->Username) + "</span>";
else
md = "<span weight='bold' color='#eeeeee'>" + Glib::Markup::escape_text(user->Username) + "</span>";
m_author->set_markup(md);
}
bool ChatMessageHeader::on_author_button_press(GdkEventButton *ev) {
if (ev->button == GDK_BUTTON_PRIMARY && (ev->state & GDK_SHIFT_MASK)) {
m_signal_action_insert_mention.emit(UserID);
return true;
}
return false;
}
ChatMessageHeader::type_signal_action_insert_mention ChatMessageHeader::signal_action_insert_mention() {
return m_signal_action_insert_mention;
}
2020-09-26 03:02:13 +00:00
void ChatMessageHeader::AddContent(Gtk::Widget *widget, bool prepend) {
m_content_box->add(*widget);
if (prepend)
m_content_box->reorder_child(*widget, 1);
}
void ChatMessageHeader::SetAvatarFromPixbuf(Glib::RefPtr<Gdk::Pixbuf> pixbuf) {
m_avatar->property_pixbuf() = pixbuf;
}