Merge branch 'folders'
This commit is contained in:
@@ -89,6 +89,7 @@ ChannelList::ChannelList()
|
|||||||
column->add_attribute(renderer->property_id(), m_columns.m_id);
|
column->add_attribute(renderer->property_id(), m_columns.m_id);
|
||||||
column->add_attribute(renderer->property_expanded(), m_columns.m_expanded);
|
column->add_attribute(renderer->property_expanded(), m_columns.m_expanded);
|
||||||
column->add_attribute(renderer->property_nsfw(), m_columns.m_nsfw);
|
column->add_attribute(renderer->property_nsfw(), m_columns.m_nsfw);
|
||||||
|
column->add_attribute(renderer->property_color(), m_columns.m_color);
|
||||||
m_view.append_column(*column);
|
m_view.append_column(*column);
|
||||||
|
|
||||||
m_menu_guild_copy_id.signal_activate().connect([this] {
|
m_menu_guild_copy_id.signal_activate().connect([this] {
|
||||||
@@ -262,14 +263,51 @@ void ChannelList::UpdateListing() {
|
|||||||
|
|
||||||
auto &discord = Abaddon::Get().GetDiscordClient();
|
auto &discord = Abaddon::Get().GetDiscordClient();
|
||||||
|
|
||||||
|
/*
|
||||||
|
guild_folders looks something like this
|
||||||
|
"guild_folders": [
|
||||||
|
{
|
||||||
|
"color": null,
|
||||||
|
"guild_ids": [
|
||||||
|
"8009060___________"
|
||||||
|
],
|
||||||
|
"id": null,
|
||||||
|
"name": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"color": null,
|
||||||
|
"guild_ids": [
|
||||||
|
"99615594__________",
|
||||||
|
"86132141__________",
|
||||||
|
"35450138__________",
|
||||||
|
"83714048__________"
|
||||||
|
],
|
||||||
|
"id": 2853066769,
|
||||||
|
"name": null
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
so if id != null then its a folder (they can have single entries)
|
||||||
|
*/
|
||||||
|
|
||||||
|
int sort_value = 0;
|
||||||
|
|
||||||
|
const auto folders = discord.GetUserSettings().GuildFolders;
|
||||||
|
if (folders.empty()) {
|
||||||
|
// fallback if no organization has occurred (guild_folders will be empty)
|
||||||
const auto guild_ids = discord.GetUserSortedGuilds();
|
const auto guild_ids = discord.GetUserSortedGuilds();
|
||||||
int sortnum = 0;
|
|
||||||
for (const auto &guild_id : guild_ids) {
|
for (const auto &guild_id : guild_ids) {
|
||||||
const auto guild = discord.GetGuild(guild_id);
|
const auto guild = discord.GetGuild(guild_id);
|
||||||
if (!guild.has_value()) continue;
|
if (!guild.has_value()) continue;
|
||||||
|
|
||||||
auto iter = AddGuild(*guild);
|
auto iter = AddGuild(*guild, m_model->children());
|
||||||
(*iter)[m_columns.m_sort] = sortnum++;
|
(*iter)[m_columns.m_sort] = sort_value++;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (const auto &group : folders) {
|
||||||
|
auto iter = AddFolder(group);
|
||||||
|
(*iter)[m_columns.m_sort] = sort_value++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_updating_listing = false;
|
m_updating_listing = false;
|
||||||
@@ -277,8 +315,9 @@ void ChannelList::UpdateListing() {
|
|||||||
AddPrivateChannels();
|
AddPrivateChannels();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO update for folders
|
||||||
void ChannelList::UpdateNewGuild(const GuildData &guild) {
|
void ChannelList::UpdateNewGuild(const GuildData &guild) {
|
||||||
AddGuild(guild);
|
AddGuild(guild, m_model->children());
|
||||||
// update sort order
|
// update sort order
|
||||||
int sortnum = 0;
|
int sortnum = 0;
|
||||||
for (const auto guild_id : Abaddon::Get().GetDiscordClient().GetUserSortedGuilds()) {
|
for (const auto guild_id : Abaddon::Get().GetDiscordClient().GetUserSortedGuilds()) {
|
||||||
@@ -405,6 +444,8 @@ void ChannelList::OnThreadListSync(const ThreadListSyncData &data) {
|
|||||||
// get the threads in the guild
|
// get the threads in the guild
|
||||||
std::vector<Snowflake> threads;
|
std::vector<Snowflake> threads;
|
||||||
auto guild_iter = GetIteratorForGuildFromID(data.GuildID);
|
auto guild_iter = GetIteratorForGuildFromID(data.GuildID);
|
||||||
|
if (!guild_iter) return;
|
||||||
|
|
||||||
std::queue<Gtk::TreeModel::iterator> queue;
|
std::queue<Gtk::TreeModel::iterator> queue;
|
||||||
queue.push(guild_iter);
|
queue.push(guild_iter);
|
||||||
|
|
||||||
@@ -546,11 +587,48 @@ ExpansionStateRoot ChannelList::GetExpansionState() const {
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
Gtk::TreeModel::iterator ChannelList::AddGuild(const GuildData &guild) {
|
Gtk::TreeModel::iterator ChannelList::AddFolder(const UserSettingsGuildFoldersEntry &folder) {
|
||||||
|
if (!folder.ID.has_value()) {
|
||||||
|
// just a guild
|
||||||
|
if (!folder.GuildIDs.empty()) {
|
||||||
|
const auto guild = Abaddon::Get().GetDiscordClient().GetGuild(folder.GuildIDs[0]);
|
||||||
|
if (guild.has_value()) {
|
||||||
|
return AddGuild(*guild, m_model->children());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
auto folder_row = *m_model->append();
|
||||||
|
folder_row[m_columns.m_type] = RenderType::Folder;
|
||||||
|
folder_row[m_columns.m_id] = *folder.ID;
|
||||||
|
if (folder.Name.has_value()) {
|
||||||
|
folder_row[m_columns.m_name] = Glib::Markup::escape_text(*folder.Name);
|
||||||
|
} else {
|
||||||
|
folder_row[m_columns.m_name] = "Folder";
|
||||||
|
}
|
||||||
|
if (folder.Color.has_value()) {
|
||||||
|
folder_row[m_columns.m_color] = IntToRGBA(*folder.Color);
|
||||||
|
}
|
||||||
|
|
||||||
|
int sort_value = 0;
|
||||||
|
for (const auto &guild_id : folder.GuildIDs) {
|
||||||
|
const auto guild = Abaddon::Get().GetDiscordClient().GetGuild(guild_id);
|
||||||
|
if (guild.has_value()) {
|
||||||
|
auto guild_row = AddGuild(*guild, folder_row->children());
|
||||||
|
(*guild_row)[m_columns.m_sort] = sort_value++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return folder_row;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
Gtk::TreeModel::iterator ChannelList::AddGuild(const GuildData &guild, const Gtk::TreeNodeChildren &root) {
|
||||||
auto &discord = Abaddon::Get().GetDiscordClient();
|
auto &discord = Abaddon::Get().GetDiscordClient();
|
||||||
auto &img = Abaddon::Get().GetImageManager();
|
auto &img = Abaddon::Get().GetImageManager();
|
||||||
|
|
||||||
auto guild_row = *m_model->append();
|
auto guild_row = *m_model->append(root);
|
||||||
guild_row[m_columns.m_type] = RenderType::Guild;
|
guild_row[m_columns.m_type] = RenderType::Guild;
|
||||||
guild_row[m_columns.m_id] = guild.ID;
|
guild_row[m_columns.m_id] = guild.ID;
|
||||||
guild_row[m_columns.m_name] = "<b>" + Glib::Markup::escape_text(guild.Name) + "</b>";
|
guild_row[m_columns.m_name] = "<b>" + Glib::Markup::escape_text(guild.Name) + "</b>";
|
||||||
@@ -679,8 +757,15 @@ void ChannelList::UpdateChannelCategory(const ChannelData &channel) {
|
|||||||
|
|
||||||
Gtk::TreeModel::iterator ChannelList::GetIteratorForGuildFromID(Snowflake id) {
|
Gtk::TreeModel::iterator ChannelList::GetIteratorForGuildFromID(Snowflake id) {
|
||||||
for (const auto &child : m_model->children()) {
|
for (const auto &child : m_model->children()) {
|
||||||
if (child[m_columns.m_id] == id)
|
if (child[m_columns.m_type] == RenderType::Guild && child[m_columns.m_id] == id) {
|
||||||
return child;
|
return child;
|
||||||
|
} else if (child[m_columns.m_type] == RenderType::Folder) {
|
||||||
|
for (const auto &folder_child : child->children()) {
|
||||||
|
if (folder_child[m_columns.m_id] == id) {
|
||||||
|
return folder_child;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
@@ -886,6 +971,7 @@ void ChannelList::MoveRow(const Gtk::TreeModel::iterator &iter, const Gtk::TreeM
|
|||||||
M(m_sort);
|
M(m_sort);
|
||||||
M(m_nsfw);
|
M(m_nsfw);
|
||||||
M(m_expanded);
|
M(m_expanded);
|
||||||
|
M(m_color);
|
||||||
#undef M
|
#undef M
|
||||||
|
|
||||||
// recursively move children
|
// recursively move children
|
||||||
@@ -998,4 +1084,5 @@ ChannelList::ModelColumns::ModelColumns() {
|
|||||||
add(m_sort);
|
add(m_sort);
|
||||||
add(m_nsfw);
|
add(m_nsfw);
|
||||||
add(m_expanded);
|
add(m_expanded);
|
||||||
|
add(m_color);
|
||||||
}
|
}
|
||||||
|
@@ -61,6 +61,7 @@ protected:
|
|||||||
Gtk::TreeModelColumn<Glib::RefPtr<Gdk::PixbufAnimation>> m_icon_anim;
|
Gtk::TreeModelColumn<Glib::RefPtr<Gdk::PixbufAnimation>> m_icon_anim;
|
||||||
Gtk::TreeModelColumn<int64_t> m_sort;
|
Gtk::TreeModelColumn<int64_t> m_sort;
|
||||||
Gtk::TreeModelColumn<bool> m_nsfw;
|
Gtk::TreeModelColumn<bool> m_nsfw;
|
||||||
|
Gtk::TreeModelColumn<std::optional<Gdk::RGBA>> m_color; // for folders right now
|
||||||
// Gtk::CellRenderer's property_is_expanded only works how i want it to if it has children
|
// Gtk::CellRenderer's property_is_expanded only works how i want it to if it has children
|
||||||
// because otherwise it doesnt count as an "expander" (property_is_expander)
|
// because otherwise it doesnt count as an "expander" (property_is_expander)
|
||||||
// so this solution will have to do which i hate but the alternative is adding invisible children
|
// so this solution will have to do which i hate but the alternative is adding invisible children
|
||||||
@@ -72,7 +73,8 @@ protected:
|
|||||||
ModelColumns m_columns;
|
ModelColumns m_columns;
|
||||||
Glib::RefPtr<Gtk::TreeStore> m_model;
|
Glib::RefPtr<Gtk::TreeStore> m_model;
|
||||||
|
|
||||||
Gtk::TreeModel::iterator AddGuild(const GuildData &guild);
|
Gtk::TreeModel::iterator AddFolder(const UserSettingsGuildFoldersEntry &folder);
|
||||||
|
Gtk::TreeModel::iterator AddGuild(const GuildData &guild, const Gtk::TreeNodeChildren &root);
|
||||||
Gtk::TreeModel::iterator UpdateCreateChannelCategory(const ChannelData &channel);
|
Gtk::TreeModel::iterator UpdateCreateChannelCategory(const ChannelData &channel);
|
||||||
Gtk::TreeModel::iterator CreateThreadRow(const Gtk::TreeNodeChildren &children, const ChannelData &channel);
|
Gtk::TreeModel::iterator CreateThreadRow(const Gtk::TreeNodeChildren &children, const ChannelData &channel);
|
||||||
|
|
||||||
|
@@ -18,7 +18,8 @@ CellRendererChannels::CellRendererChannels()
|
|||||||
, m_property_pixbuf(*this, "pixbuf")
|
, m_property_pixbuf(*this, "pixbuf")
|
||||||
, m_property_pixbuf_animation(*this, "pixbuf-animation")
|
, m_property_pixbuf_animation(*this, "pixbuf-animation")
|
||||||
, m_property_expanded(*this, "expanded")
|
, m_property_expanded(*this, "expanded")
|
||||||
, m_property_nsfw(*this, "nsfw") {
|
, m_property_nsfw(*this, "nsfw")
|
||||||
|
, m_property_color(*this, "color") {
|
||||||
property_mode() = Gtk::CELL_RENDERER_MODE_ACTIVATABLE;
|
property_mode() = Gtk::CELL_RENDERER_MODE_ACTIVATABLE;
|
||||||
property_xpad() = 2;
|
property_xpad() = 2;
|
||||||
property_ypad() = 2;
|
property_ypad() = 2;
|
||||||
@@ -55,8 +56,14 @@ Glib::PropertyProxy<bool> CellRendererChannels::property_nsfw() {
|
|||||||
return m_property_nsfw.get_proxy();
|
return m_property_nsfw.get_proxy();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Glib::PropertyProxy<std::optional<Gdk::RGBA>> CellRendererChannels::property_color() {
|
||||||
|
return m_property_color.get_proxy();
|
||||||
|
}
|
||||||
|
|
||||||
void CellRendererChannels::get_preferred_width_vfunc(Gtk::Widget &widget, int &minimum_width, int &natural_width) const {
|
void CellRendererChannels::get_preferred_width_vfunc(Gtk::Widget &widget, int &minimum_width, int &natural_width) const {
|
||||||
switch (m_property_type.get_value()) {
|
switch (m_property_type.get_value()) {
|
||||||
|
case RenderType::Folder:
|
||||||
|
return get_preferred_width_vfunc_folder(widget, minimum_width, natural_width);
|
||||||
case RenderType::Guild:
|
case RenderType::Guild:
|
||||||
return get_preferred_width_vfunc_guild(widget, minimum_width, natural_width);
|
return get_preferred_width_vfunc_guild(widget, minimum_width, natural_width);
|
||||||
case RenderType::Category:
|
case RenderType::Category:
|
||||||
@@ -74,6 +81,8 @@ void CellRendererChannels::get_preferred_width_vfunc(Gtk::Widget &widget, int &m
|
|||||||
|
|
||||||
void CellRendererChannels::get_preferred_width_for_height_vfunc(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const {
|
void CellRendererChannels::get_preferred_width_for_height_vfunc(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const {
|
||||||
switch (m_property_type.get_value()) {
|
switch (m_property_type.get_value()) {
|
||||||
|
case RenderType::Folder:
|
||||||
|
return get_preferred_width_for_height_vfunc_folder(widget, height, minimum_width, natural_width);
|
||||||
case RenderType::Guild:
|
case RenderType::Guild:
|
||||||
return get_preferred_width_for_height_vfunc_guild(widget, height, minimum_width, natural_width);
|
return get_preferred_width_for_height_vfunc_guild(widget, height, minimum_width, natural_width);
|
||||||
case RenderType::Category:
|
case RenderType::Category:
|
||||||
@@ -91,6 +100,8 @@ void CellRendererChannels::get_preferred_width_for_height_vfunc(Gtk::Widget &wid
|
|||||||
|
|
||||||
void CellRendererChannels::get_preferred_height_vfunc(Gtk::Widget &widget, int &minimum_height, int &natural_height) const {
|
void CellRendererChannels::get_preferred_height_vfunc(Gtk::Widget &widget, int &minimum_height, int &natural_height) const {
|
||||||
switch (m_property_type.get_value()) {
|
switch (m_property_type.get_value()) {
|
||||||
|
case RenderType::Folder:
|
||||||
|
return get_preferred_height_vfunc_folder(widget, minimum_height, natural_height);
|
||||||
case RenderType::Guild:
|
case RenderType::Guild:
|
||||||
return get_preferred_height_vfunc_guild(widget, minimum_height, natural_height);
|
return get_preferred_height_vfunc_guild(widget, minimum_height, natural_height);
|
||||||
case RenderType::Category:
|
case RenderType::Category:
|
||||||
@@ -108,6 +119,8 @@ void CellRendererChannels::get_preferred_height_vfunc(Gtk::Widget &widget, int &
|
|||||||
|
|
||||||
void CellRendererChannels::get_preferred_height_for_width_vfunc(Gtk::Widget &widget, int width, int &minimum_height, int &natural_height) const {
|
void CellRendererChannels::get_preferred_height_for_width_vfunc(Gtk::Widget &widget, int width, int &minimum_height, int &natural_height) const {
|
||||||
switch (m_property_type.get_value()) {
|
switch (m_property_type.get_value()) {
|
||||||
|
case RenderType::Folder:
|
||||||
|
return get_preferred_height_for_width_vfunc_folder(widget, width, minimum_height, natural_height);
|
||||||
case RenderType::Guild:
|
case RenderType::Guild:
|
||||||
return get_preferred_height_for_width_vfunc_guild(widget, width, minimum_height, natural_height);
|
return get_preferred_height_for_width_vfunc_guild(widget, width, minimum_height, natural_height);
|
||||||
case RenderType::Category:
|
case RenderType::Category:
|
||||||
@@ -125,6 +138,8 @@ void CellRendererChannels::get_preferred_height_for_width_vfunc(Gtk::Widget &wid
|
|||||||
|
|
||||||
void CellRendererChannels::render_vfunc(const Cairo::RefPtr<Cairo::Context> &cr, Gtk::Widget &widget, const Gdk::Rectangle &background_area, const Gdk::Rectangle &cell_area, Gtk::CellRendererState flags) {
|
void CellRendererChannels::render_vfunc(const Cairo::RefPtr<Cairo::Context> &cr, Gtk::Widget &widget, const Gdk::Rectangle &background_area, const Gdk::Rectangle &cell_area, Gtk::CellRendererState flags) {
|
||||||
switch (m_property_type.get_value()) {
|
switch (m_property_type.get_value()) {
|
||||||
|
case RenderType::Folder:
|
||||||
|
return render_vfunc_folder(cr, widget, background_area, cell_area, flags);
|
||||||
case RenderType::Guild:
|
case RenderType::Guild:
|
||||||
return render_vfunc_guild(cr, widget, background_area, cell_area, flags);
|
return render_vfunc_guild(cr, widget, background_area, cell_area, flags);
|
||||||
case RenderType::Category:
|
case RenderType::Category:
|
||||||
@@ -140,6 +155,69 @@ void CellRendererChannels::render_vfunc(const Cairo::RefPtr<Cairo::Context> &cr,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// folder functions
|
||||||
|
|
||||||
|
void CellRendererChannels::get_preferred_width_vfunc_folder(Gtk::Widget &widget, int &minimum_width, int &natural_width) const {
|
||||||
|
m_renderer_text.get_preferred_width(widget, minimum_width, natural_width);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CellRendererChannels::get_preferred_width_for_height_vfunc_folder(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const {
|
||||||
|
m_renderer_text.get_preferred_width_for_height(widget, height, minimum_width, natural_width);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CellRendererChannels::get_preferred_height_vfunc_folder(Gtk::Widget &widget, int &minimum_height, int &natural_height) const {
|
||||||
|
m_renderer_text.get_preferred_height(widget, minimum_height, natural_height);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CellRendererChannels::get_preferred_height_for_width_vfunc_folder(Gtk::Widget &widget, int width, int &minimum_height, int &natural_height) const {
|
||||||
|
m_renderer_text.get_preferred_height_for_width(widget, width, minimum_height, natural_height);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CellRendererChannels::render_vfunc_folder(const Cairo::RefPtr<Cairo::Context> &cr, Gtk::Widget &widget, const Gdk::Rectangle &background_area, const Gdk::Rectangle &cell_area, Gtk::CellRendererState flags) {
|
||||||
|
constexpr static int len = 5;
|
||||||
|
int x1, y1, x2, y2, x3, y3;
|
||||||
|
if (property_expanded()) {
|
||||||
|
x1 = background_area.get_x() + 7;
|
||||||
|
y1 = background_area.get_y() + background_area.get_height() / 2 - len;
|
||||||
|
x2 = background_area.get_x() + 7 + len;
|
||||||
|
y2 = background_area.get_y() + background_area.get_height() / 2 + len;
|
||||||
|
x3 = background_area.get_x() + 7 + len * 2;
|
||||||
|
y3 = background_area.get_y() + background_area.get_height() / 2 - len;
|
||||||
|
} else {
|
||||||
|
x1 = background_area.get_x() + 7;
|
||||||
|
y1 = background_area.get_y() + background_area.get_height() / 2 - len;
|
||||||
|
x2 = background_area.get_x() + 7 + len * 2;
|
||||||
|
y2 = background_area.get_y() + background_area.get_height() / 2;
|
||||||
|
x3 = background_area.get_x() + 7;
|
||||||
|
y3 = background_area.get_y() + background_area.get_height() / 2 + len;
|
||||||
|
}
|
||||||
|
cr->move_to(x1, y1);
|
||||||
|
cr->line_to(x2, y2);
|
||||||
|
cr->line_to(x3, y3);
|
||||||
|
const auto expander_color = Gdk::RGBA(Abaddon::Get().GetSettings().ChannelsExpanderColor);
|
||||||
|
cr->set_source_rgb(expander_color.get_red(), expander_color.get_green(), expander_color.get_blue());
|
||||||
|
cr->stroke();
|
||||||
|
|
||||||
|
Gtk::Requisition text_minimum, text_natural;
|
||||||
|
m_renderer_text.get_preferred_size(widget, text_minimum, text_natural);
|
||||||
|
|
||||||
|
const int text_x = background_area.get_x() + 22;
|
||||||
|
const int text_y = background_area.get_y() + background_area.get_height() / 2 - text_natural.height / 2;
|
||||||
|
const int text_w = text_natural.width;
|
||||||
|
const int text_h = text_natural.height;
|
||||||
|
|
||||||
|
Gdk::Rectangle text_cell_area(text_x, text_y, text_w, text_h);
|
||||||
|
|
||||||
|
static const auto color = Gdk::RGBA(Abaddon::Get().GetSettings().ChannelColor);
|
||||||
|
if (m_property_color.get_value().has_value()) {
|
||||||
|
m_renderer_text.property_foreground_rgba() = *m_property_color.get_value();
|
||||||
|
} else {
|
||||||
|
m_renderer_text.property_foreground_rgba() = color;
|
||||||
|
}
|
||||||
|
m_renderer_text.render(cr, widget, background_area, text_cell_area, flags);
|
||||||
|
m_renderer_text.property_foreground_set() = false;
|
||||||
|
}
|
||||||
|
|
||||||
// guild functions
|
// guild functions
|
||||||
|
|
||||||
void CellRendererChannels::get_preferred_width_vfunc_guild(Gtk::Widget &widget, int &minimum_width, int &natural_width) const {
|
void CellRendererChannels::get_preferred_width_vfunc_guild(Gtk::Widget &widget, int &minimum_width, int &natural_width) const {
|
||||||
|
@@ -6,6 +6,7 @@
|
|||||||
#include "discord/snowflake.hpp"
|
#include "discord/snowflake.hpp"
|
||||||
|
|
||||||
enum class RenderType : uint8_t {
|
enum class RenderType : uint8_t {
|
||||||
|
Folder,
|
||||||
Guild,
|
Guild,
|
||||||
Category,
|
Category,
|
||||||
TextChannel,
|
TextChannel,
|
||||||
@@ -27,6 +28,7 @@ public:
|
|||||||
Glib::PropertyProxy<Glib::RefPtr<Gdk::PixbufAnimation>> property_icon_animation();
|
Glib::PropertyProxy<Glib::RefPtr<Gdk::PixbufAnimation>> property_icon_animation();
|
||||||
Glib::PropertyProxy<bool> property_expanded();
|
Glib::PropertyProxy<bool> property_expanded();
|
||||||
Glib::PropertyProxy<bool> property_nsfw();
|
Glib::PropertyProxy<bool> property_nsfw();
|
||||||
|
Glib::PropertyProxy<std::optional<Gdk::RGBA>> property_color();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void get_preferred_width_vfunc(Gtk::Widget &widget, int &minimum_width, int &natural_width) const override;
|
void get_preferred_width_vfunc(Gtk::Widget &widget, int &minimum_width, int &natural_width) const override;
|
||||||
@@ -39,6 +41,17 @@ protected:
|
|||||||
const Gdk::Rectangle &cell_area,
|
const Gdk::Rectangle &cell_area,
|
||||||
Gtk::CellRendererState flags) override;
|
Gtk::CellRendererState flags) override;
|
||||||
|
|
||||||
|
// guild functions
|
||||||
|
void get_preferred_width_vfunc_folder(Gtk::Widget &widget, int &minimum_width, int &natural_width) const;
|
||||||
|
void get_preferred_width_for_height_vfunc_folder(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const;
|
||||||
|
void get_preferred_height_vfunc_folder(Gtk::Widget &widget, int &minimum_height, int &natural_height) const;
|
||||||
|
void get_preferred_height_for_width_vfunc_folder(Gtk::Widget &widget, int width, int &minimum_height, int &natural_height) const;
|
||||||
|
void render_vfunc_folder(const Cairo::RefPtr<Cairo::Context> &cr,
|
||||||
|
Gtk::Widget &widget,
|
||||||
|
const Gdk::Rectangle &background_area,
|
||||||
|
const Gdk::Rectangle &cell_area,
|
||||||
|
Gtk::CellRendererState flags);
|
||||||
|
|
||||||
// guild functions
|
// guild functions
|
||||||
void get_preferred_width_vfunc_guild(Gtk::Widget &widget, int &minimum_width, int &natural_width) const;
|
void get_preferred_width_vfunc_guild(Gtk::Widget &widget, int &minimum_width, int &natural_width) const;
|
||||||
void get_preferred_width_for_height_vfunc_guild(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const;
|
void get_preferred_width_for_height_vfunc_guild(Gtk::Widget &widget, int height, int &minimum_width, int &natural_width) const;
|
||||||
@@ -118,6 +131,7 @@ private:
|
|||||||
Glib::Property<Glib::RefPtr<Gdk::PixbufAnimation>> m_property_pixbuf_animation; // guild
|
Glib::Property<Glib::RefPtr<Gdk::PixbufAnimation>> m_property_pixbuf_animation; // guild
|
||||||
Glib::Property<bool> m_property_expanded; // category
|
Glib::Property<bool> m_property_expanded; // category
|
||||||
Glib::Property<bool> m_property_nsfw; // channel
|
Glib::Property<bool> m_property_nsfw; // channel
|
||||||
|
Glib::Property<std::optional<Gdk::RGBA>> m_property_color; // folder
|
||||||
|
|
||||||
// same pitfalls as in https://github.com/uowuo/abaddon/blob/60404783bd4ce9be26233fe66fc3a74475d9eaa3/components/cellrendererpixbufanimation.hpp#L32-L39
|
// same pitfalls as in https://github.com/uowuo/abaddon/blob/60404783bd4ce9be26233fe66fc3a74475d9eaa3/components/cellrendererpixbufanimation.hpp#L32-L39
|
||||||
// this will manifest though since guild icons can change
|
// this will manifest though since guild icons can change
|
||||||
|
@@ -1202,6 +1202,10 @@ void DiscordClient::SetUserAgent(const std::string &agent) {
|
|||||||
m_websocket.SetUserAgent(agent);
|
m_websocket.SetUserAgent(agent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DiscordClient::SetDumpReady(bool dump) {
|
||||||
|
m_dump_ready = dump;
|
||||||
|
}
|
||||||
|
|
||||||
bool DiscordClient::IsChannelMuted(Snowflake id) const noexcept {
|
bool DiscordClient::IsChannelMuted(Snowflake id) const noexcept {
|
||||||
return m_muted_channels.find(id) != m_muted_channels.end();
|
return m_muted_channels.find(id) != m_muted_channels.end();
|
||||||
}
|
}
|
||||||
@@ -1564,6 +1568,17 @@ void DiscordClient::ProcessNewGuild(GuildData &guild) {
|
|||||||
|
|
||||||
void DiscordClient::HandleGatewayReady(const GatewayMessage &msg) {
|
void DiscordClient::HandleGatewayReady(const GatewayMessage &msg) {
|
||||||
m_ready_received = true;
|
m_ready_received = true;
|
||||||
|
|
||||||
|
if (m_dump_ready) {
|
||||||
|
const auto name = "./payload_ready-" + Glib::DateTime::create_now_utc().format("%Y-%m-%d_%H-%M-%S") + ".json";
|
||||||
|
auto *fp = std::fopen(name.c_str(), "wb");
|
||||||
|
if (fp != nullptr) {
|
||||||
|
const auto contents = msg.Data.dump(4);
|
||||||
|
std::fwrite(contents.data(), contents.size(), 1, fp);
|
||||||
|
std::fclose(fp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ReadyEventData data = msg.Data;
|
ReadyEventData data = msg.Data;
|
||||||
for (auto &g : data.Guilds)
|
for (auto &g : data.Guilds)
|
||||||
ProcessNewGuild(g);
|
ProcessNewGuild(g);
|
||||||
@@ -2275,6 +2290,10 @@ std::set<Snowflake> DiscordClient::GetPrivateChannels() const {
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const UserSettings &DiscordClient::GetUserSettings() const {
|
||||||
|
return m_user_settings;
|
||||||
|
}
|
||||||
|
|
||||||
EPremiumType DiscordClient::GetSelfPremiumType() const {
|
EPremiumType DiscordClient::GetSelfPremiumType() const {
|
||||||
const auto &data = GetUserData();
|
const auto &data = GetUserData();
|
||||||
if (data.PremiumType.has_value())
|
if (data.PremiumType.has_value())
|
||||||
|
@@ -60,6 +60,7 @@ public:
|
|||||||
std::vector<Message> GetMessagesForChannel(Snowflake id, size_t limit = 50) const;
|
std::vector<Message> GetMessagesForChannel(Snowflake id, size_t limit = 50) const;
|
||||||
std::vector<Message> GetMessagesBefore(Snowflake channel_id, Snowflake message_id, size_t limit = 50) const;
|
std::vector<Message> GetMessagesBefore(Snowflake channel_id, Snowflake message_id, size_t limit = 50) const;
|
||||||
std::set<Snowflake> GetPrivateChannels() const;
|
std::set<Snowflake> GetPrivateChannels() const;
|
||||||
|
const UserSettings &GetUserSettings() const;
|
||||||
|
|
||||||
EPremiumType GetSelfPremiumType() const;
|
EPremiumType GetSelfPremiumType() const;
|
||||||
|
|
||||||
@@ -211,6 +212,8 @@ public:
|
|||||||
void UpdateToken(const std::string &token);
|
void UpdateToken(const std::string &token);
|
||||||
void SetUserAgent(const std::string &agent);
|
void SetUserAgent(const std::string &agent);
|
||||||
|
|
||||||
|
void SetDumpReady(bool dump);
|
||||||
|
|
||||||
bool IsChannelMuted(Snowflake id) const noexcept;
|
bool IsChannelMuted(Snowflake id) const noexcept;
|
||||||
bool IsGuildMuted(Snowflake id) const noexcept;
|
bool IsGuildMuted(Snowflake id) const noexcept;
|
||||||
int GetUnreadStateForChannel(Snowflake id) const noexcept;
|
int GetUnreadStateForChannel(Snowflake id) const noexcept;
|
||||||
@@ -229,6 +232,8 @@ private:
|
|||||||
std::vector<uint8_t> m_decompress_buf;
|
std::vector<uint8_t> m_decompress_buf;
|
||||||
z_stream m_zstream;
|
z_stream m_zstream;
|
||||||
|
|
||||||
|
bool m_dump_ready = false;
|
||||||
|
|
||||||
static std::string GetAPIURL();
|
static std::string GetAPIURL();
|
||||||
static std::string GetGatewayURL();
|
static std::string GetGatewayURL();
|
||||||
|
|
||||||
|
@@ -1,13 +1,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "json.hpp"
|
#include "json.hpp"
|
||||||
#include "snowflake.hpp"
|
#include "snowflake.hpp"
|
||||||
|
#include <optional>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
struct UserSettingsGuildFoldersEntry {
|
struct UserSettingsGuildFoldersEntry {
|
||||||
int Color = -1; // null
|
std::optional<int> Color;
|
||||||
std::vector<Snowflake> GuildIDs;
|
std::vector<Snowflake> GuildIDs;
|
||||||
Snowflake ID; // null (this can be a snowflake as a string or an int that isnt a snowflake lol)
|
std::optional<Snowflake> ID; // (this can be a snowflake as a string or an int that isnt a snowflake lol)
|
||||||
std::string Name; // null
|
std::optional<std::string> Name;
|
||||||
|
|
||||||
friend void from_json(const nlohmann::json &j, UserSettingsGuildFoldersEntry &m);
|
friend void from_json(const nlohmann::json &j, UserSettingsGuildFoldersEntry &m);
|
||||||
};
|
};
|
||||||
|
@@ -255,8 +255,10 @@ void MainWindow::SetupMenu() {
|
|||||||
m_menu_file.set_submenu(m_menu_file_sub);
|
m_menu_file.set_submenu(m_menu_file_sub);
|
||||||
m_menu_file_reload_css.set_label("Reload CSS");
|
m_menu_file_reload_css.set_label("Reload CSS");
|
||||||
m_menu_file_clear_cache.set_label("Clear file cache");
|
m_menu_file_clear_cache.set_label("Clear file cache");
|
||||||
|
m_menu_file_dump_ready.set_label("Dump ready message");
|
||||||
m_menu_file_sub.append(m_menu_file_reload_css);
|
m_menu_file_sub.append(m_menu_file_reload_css);
|
||||||
m_menu_file_sub.append(m_menu_file_clear_cache);
|
m_menu_file_sub.append(m_menu_file_clear_cache);
|
||||||
|
m_menu_file_sub.append(m_menu_file_dump_ready);
|
||||||
|
|
||||||
m_menu_view.set_label("View");
|
m_menu_view.set_label("View");
|
||||||
m_menu_view.set_submenu(m_menu_view_sub);
|
m_menu_view.set_submenu(m_menu_view_sub);
|
||||||
@@ -335,6 +337,10 @@ void MainWindow::SetupMenu() {
|
|||||||
Abaddon::Get().GetImageManager().ClearCache();
|
Abaddon::Get().GetImageManager().ClearCache();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
m_menu_file_dump_ready.signal_toggled().connect([this]() {
|
||||||
|
Abaddon::Get().GetDiscordClient().SetDumpReady(m_menu_file_dump_ready.get_active());
|
||||||
|
});
|
||||||
|
|
||||||
m_menu_discord_add_recipient.signal_activate().connect([this] {
|
m_menu_discord_add_recipient.signal_activate().connect([this] {
|
||||||
m_signal_action_add_recipient.emit(GetChatActiveChannel());
|
m_signal_action_add_recipient.emit(GetChatActiveChannel());
|
||||||
});
|
});
|
||||||
|
@@ -72,6 +72,7 @@ private:
|
|||||||
Gtk::Menu m_menu_file_sub;
|
Gtk::Menu m_menu_file_sub;
|
||||||
Gtk::MenuItem m_menu_file_reload_css;
|
Gtk::MenuItem m_menu_file_reload_css;
|
||||||
Gtk::MenuItem m_menu_file_clear_cache;
|
Gtk::MenuItem m_menu_file_clear_cache;
|
||||||
|
Gtk::CheckMenuItem m_menu_file_dump_ready;
|
||||||
|
|
||||||
Gtk::MenuItem m_menu_view;
|
Gtk::MenuItem m_menu_view;
|
||||||
Gtk::Menu m_menu_view_sub;
|
Gtk::Menu m_menu_view_sub;
|
||||||
|
Reference in New Issue
Block a user