diff options
author | ouwou <26526779+ouwou@users.noreply.github.com> | 2021-01-26 02:40:50 -0500 |
---|---|---|
committer | ouwou <26526779+ouwou@users.noreply.github.com> | 2021-01-26 02:40:50 -0500 |
commit | dabfefafd9c681231f990eccddc771ccf5c2c2e2 (patch) | |
tree | 35fc440dbacf8367fa650dde89217b23cf08bf33 /components | |
parent | e6c159659d30ca3cc955b9a7f7969f51d87a0bf9 (diff) | |
download | abaddon-portaudio-dabfefafd9c681231f990eccddc771ccf5c2c2e2.tar.gz abaddon-portaudio-dabfefafd9c681231f990eccddc771ccf5c2c2e2.zip |
show status indicators in member list, change some other shit with presences
Diffstat (limited to 'components')
-rw-r--r-- | components/memberlist.cpp | 7 | ||||
-rw-r--r-- | components/memberlist.hpp | 4 | ||||
-rw-r--r-- | components/statusindicator.cpp | 136 | ||||
-rw-r--r-- | components/statusindicator.hpp | 30 |
4 files changed, 176 insertions, 1 deletions
diff --git a/components/memberlist.cpp b/components/memberlist.cpp index 0d54448..be7e074 100644 --- a/components/memberlist.cpp +++ b/components/memberlist.cpp @@ -1,6 +1,8 @@ #include "memberlist.hpp" #include "../abaddon.hpp" #include "../util.hpp" +#include "lazyimage.hpp" +#include "statusindicator.hpp" MemberListUserRow::MemberListUserRow(Snowflake guild_id, const UserData *data) { ID = data->ID; @@ -8,6 +10,9 @@ MemberListUserRow::MemberListUserRow(Snowflake guild_id, const UserData *data) { m_box = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL)); m_label = Gtk::manage(new Gtk::Label); m_avatar = Gtk::manage(new LazyImage(16, 16)); + m_status_indicator = Gtk::manage(new StatusIndicator(ID)); + + m_status_indicator->set_margin_start(3); if (data->HasAvatar()) m_avatar->SetURL(data->GetAvatarURL("png")); @@ -37,8 +42,10 @@ MemberListUserRow::MemberListUserRow(Snowflake guild_id, const UserData *data) { m_label->set_use_markup(true); m_label->set_markup("<i>[unknown user]</i>"); } + m_label->set_halign(Gtk::ALIGN_START); m_box->add(*m_avatar); + m_box->add(*m_status_indicator); m_box->add(*m_label); m_ev->add(*m_box); add(*m_ev); diff --git a/components/memberlist.hpp b/components/memberlist.hpp index c816b26..856d73c 100644 --- a/components/memberlist.hpp +++ b/components/memberlist.hpp @@ -3,8 +3,9 @@ #include <mutex> #include <unordered_map> #include "../discord/discord.hpp" -#include "lazyimage.hpp" +class LazyImage; +class StatusIndicator; class MemberListUserRow : public Gtk::ListBoxRow { public: MemberListUserRow(Snowflake guild_id, const UserData *data); @@ -15,6 +16,7 @@ private: Gtk::EventBox *m_ev; Gtk::Box *m_box; LazyImage *m_avatar; + StatusIndicator *m_status_indicator; Gtk::Label *m_label; }; diff --git a/components/statusindicator.cpp b/components/statusindicator.cpp new file mode 100644 index 0000000..abf389c --- /dev/null +++ b/components/statusindicator.cpp @@ -0,0 +1,136 @@ +#include "statusindicator.hpp" +#include "../abaddon.hpp" + +static const constexpr int Diameter = 8; +static const auto OnlineColor = Gdk::RGBA("#43B581"); +static const auto IdleColor = Gdk::RGBA("#FAA61A"); +static const auto DNDColor = Gdk::RGBA("#982929"); +static const auto OfflineColor = Gdk::RGBA("#808080"); + +StatusIndicator::StatusIndicator(Snowflake user_id) + : Glib::ObjectBase("statusindicator") + , Gtk::Widget() + , m_id(user_id) + , m_color(OfflineColor) { + set_has_window(true); + set_name("status-indicator"); + + Abaddon::Get().GetDiscordClient().signal_guild_member_list_update().connect(sigc::hide(sigc::mem_fun(*this, &StatusIndicator::CheckStatus))); + auto cb = [this](Snowflake id, PresenceStatus status) { + if (id == m_id) CheckStatus(); + }; + Abaddon::Get().GetDiscordClient().signal_presence_update().connect(sigc::track_obj(cb, *this)); + + CheckStatus(); +} + +StatusIndicator::~StatusIndicator() { +} + +void StatusIndicator::CheckStatus() { + const auto status = Abaddon::Get().GetDiscordClient().GetUserStatus(m_id); + if (status.has_value()) { + switch (*status) { + case PresenceStatus::Online: + m_color = OnlineColor; + break; + case PresenceStatus::Offline: + m_color = OfflineColor; + break; + case PresenceStatus::DND: + m_color = DNDColor; + break; + case PresenceStatus::Idle: + m_color = IdleColor; + break; + } + } else { + m_color = OfflineColor; + } + + queue_draw(); +} + +Gtk::SizeRequestMode StatusIndicator::get_request_mode_vfunc() const { + return Gtk::Widget::get_request_mode_vfunc(); +} + +void StatusIndicator::get_preferred_width_vfunc(int &minimum_width, int &natural_width) const { + minimum_width = 0; + natural_width = Diameter; +} + +void StatusIndicator::get_preferred_height_for_width_vfunc(int width, int &minimum_height, int &natural_height) const { + minimum_height = 0; + natural_height = Diameter; +} + +void StatusIndicator::get_preferred_height_vfunc(int &minimum_height, int &natural_height) const { + minimum_height = 0; + natural_height = Diameter; +} + +void StatusIndicator::get_preferred_width_for_height_vfunc(int height, int &minimum_width, int &natural_width) const { + minimum_width = 0; + natural_width = Diameter; +} + +void StatusIndicator::on_size_allocate(Gtk::Allocation &allocation) { + set_allocation(allocation); + + if (m_window) + m_window->move_resize(allocation.get_x(), allocation.get_y(), allocation.get_width(), allocation.get_height()); +} + +void StatusIndicator::on_map() { + Gtk::Widget::on_map(); +} + +void StatusIndicator::on_unmap() { + Gtk::Widget::on_unmap(); +} + +void StatusIndicator::on_realize() { + set_realized(true); + + if (!m_window) { + GdkWindowAttr attributes; + std::memset(&attributes, 0, sizeof(attributes)); + + auto allocation = get_allocation(); + + attributes.x = allocation.get_x(); + attributes.y = allocation.get_y(); + attributes.width = allocation.get_width(); + attributes.height = allocation.get_height(); + + attributes.event_mask = get_events() | Gdk::EXPOSURE_MASK; + attributes.window_type = GDK_WINDOW_CHILD; + attributes.wclass = GDK_INPUT_OUTPUT; + + m_window = Gdk::Window::create(get_parent_window(), &attributes, GDK_WA_X | GDK_WA_Y); + set_window(m_window); + + m_window->set_user_data(gobj()); + } +} + +void StatusIndicator::on_unrealize() { + m_window.reset(); + + Gtk::Widget::on_unrealize(); +} + +bool StatusIndicator::on_draw(const Cairo::RefPtr<Cairo::Context> &cr) { + const auto allocation = get_allocation(); + const auto width = allocation.get_width(); + const auto height = allocation.get_height(); + + cr->set_source_rgb(m_color.get_red(), m_color.get_green(), m_color.get_blue()); + cr->arc(width / 2, height / 2, width / 3, 0.0, 2 * (4 * std::atan(1))); + cr->close_path(); + cr->fill_preserve(); + cr->stroke(); + + return true; +} diff --git a/components/statusindicator.hpp b/components/statusindicator.hpp new file mode 100644 index 0000000..9c7382e --- /dev/null +++ b/components/statusindicator.hpp @@ -0,0 +1,30 @@ +#pragma once +#include <gtkmm.h> +#include "../discord/snowflake.hpp" + +class StatusIndicator : public Gtk::Widget { +public: + StatusIndicator(Snowflake user_id); + virtual ~StatusIndicator(); + +protected: + Gtk::SizeRequestMode get_request_mode_vfunc() const override; + void get_preferred_width_vfunc(int &minimum_width, int &natural_width) const override; + void get_preferred_height_for_width_vfunc(int width, int &minimum_height, int &natural_height) const override; + void get_preferred_height_vfunc(int &minimum_height, int &natural_height) const override; + void get_preferred_width_for_height_vfunc(int height, int &minimum_width, int &natural_width) const override; + void on_size_allocate(Gtk::Allocation &allocation) override; + void on_map() override; + void on_unmap() override; + void on_realize() override; + void on_unrealize() override; + bool on_draw(const Cairo::RefPtr<Cairo::Context> &cr) override; + + Glib::RefPtr<Gdk::Window> m_window; + +private: + void CheckStatus(); + + Snowflake m_id; + Gdk::RGBA m_color; +}; |