forked from OpenGamers/abaddon
improve scrolling behavior again, refactor
scrolling is almost exactly how i want it, but when an existing message's height allocation is changed it still causes the scroll position to change, but its not that bad and is better than what i had before anyways so it is good enough for now. ideally if you are scrolled in the middle it will stay put completely
This commit is contained in:
parent
49685c3989
commit
56a74fb5dd
@ -9,22 +9,10 @@ ChatList::ChatList() {
|
|||||||
set_can_focus(false);
|
set_can_focus(false);
|
||||||
set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_ALWAYS);
|
set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_ALWAYS);
|
||||||
|
|
||||||
auto v = get_vadjustment();
|
get_vadjustment()->signal_value_changed().connect(sigc::mem_fun(*this, &ChatList::OnVAdjustmentValueChanged));
|
||||||
v->signal_value_changed().connect([this, v] {
|
get_vadjustment()->property_upper().signal_changed().connect(sigc::mem_fun(*this, &ChatList::OnVAdjustmentUpperChanged));
|
||||||
if (m_history_timer.elapsed() > 1 && v->get_value() < 500) {
|
|
||||||
m_history_timer.start();
|
|
||||||
m_signal_action_chat_load_history.emit(m_active_channel);
|
|
||||||
}
|
|
||||||
m_should_scroll_to_bottom = v->get_upper() - v->get_page_size() <= v->get_value();
|
|
||||||
});
|
|
||||||
|
|
||||||
v->property_upper().signal_changed().connect(sigc::mem_fun(*this, &ChatList::OnUpperAdjustmentChanged));
|
m_list.signal_size_allocate().connect(sigc::mem_fun(*this, &ChatList::OnListSizeAllocate));
|
||||||
|
|
||||||
m_list.signal_size_allocate()
|
|
||||||
.connect([this](Gtk::Allocation &) {
|
|
||||||
if (m_should_scroll_to_bottom)
|
|
||||||
ScrollToBottom();
|
|
||||||
});
|
|
||||||
|
|
||||||
m_list.set_focus_hadjustment(get_hadjustment());
|
m_list.set_focus_hadjustment(get_hadjustment());
|
||||||
m_list.set_focus_vadjustment(get_vadjustment());
|
m_list.set_focus_vadjustment(get_vadjustment());
|
||||||
@ -36,56 +24,7 @@ ChatList::ChatList() {
|
|||||||
|
|
||||||
m_list.show();
|
m_list.show();
|
||||||
|
|
||||||
m_menu_copy_id = Gtk::manage(new Gtk::MenuItem("Copy ID"));
|
SetupMenu();
|
||||||
m_menu_copy_id->signal_activate().connect([this] {
|
|
||||||
Gtk::Clipboard::get()->set_text(std::to_string(m_menu_selected_message));
|
|
||||||
});
|
|
||||||
m_menu_copy_id->show();
|
|
||||||
m_menu.append(*m_menu_copy_id);
|
|
||||||
|
|
||||||
m_menu_delete_message = Gtk::manage(new Gtk::MenuItem("Delete Message"));
|
|
||||||
m_menu_delete_message->signal_activate().connect([this] {
|
|
||||||
Abaddon::Get().GetDiscordClient().DeleteMessage(m_active_channel, m_menu_selected_message);
|
|
||||||
});
|
|
||||||
m_menu_delete_message->show();
|
|
||||||
m_menu.append(*m_menu_delete_message);
|
|
||||||
|
|
||||||
m_menu_edit_message = Gtk::manage(new Gtk::MenuItem("Edit Message"));
|
|
||||||
m_menu_edit_message->signal_activate().connect([this] {
|
|
||||||
m_signal_action_message_edit.emit(m_active_channel, m_menu_selected_message);
|
|
||||||
});
|
|
||||||
m_menu_edit_message->show();
|
|
||||||
m_menu.append(*m_menu_edit_message);
|
|
||||||
|
|
||||||
m_menu_copy_content = Gtk::manage(new Gtk::MenuItem("Copy Content"));
|
|
||||||
m_menu_copy_content->signal_activate().connect([this] {
|
|
||||||
const auto msg = Abaddon::Get().GetDiscordClient().GetMessage(m_menu_selected_message);
|
|
||||||
if (msg.has_value())
|
|
||||||
Gtk::Clipboard::get()->set_text(msg->Content);
|
|
||||||
});
|
|
||||||
m_menu_copy_content->show();
|
|
||||||
m_menu.append(*m_menu_copy_content);
|
|
||||||
|
|
||||||
m_menu_reply_to = Gtk::manage(new Gtk::MenuItem("Reply To"));
|
|
||||||
m_menu_reply_to->signal_activate().connect([this] {
|
|
||||||
m_signal_action_reply_to.emit(m_menu_selected_message);
|
|
||||||
});
|
|
||||||
m_menu_reply_to->show();
|
|
||||||
m_menu.append(*m_menu_reply_to);
|
|
||||||
|
|
||||||
m_menu_unpin = Gtk::manage(new Gtk::MenuItem("Unpin"));
|
|
||||||
m_menu_unpin->signal_activate().connect([this] {
|
|
||||||
Abaddon::Get().GetDiscordClient().Unpin(m_active_channel, m_menu_selected_message, [](...) {});
|
|
||||||
});
|
|
||||||
m_menu.append(*m_menu_unpin);
|
|
||||||
|
|
||||||
m_menu_pin = Gtk::manage(new Gtk::MenuItem("Pin"));
|
|
||||||
m_menu_pin->signal_activate().connect([this] {
|
|
||||||
Abaddon::Get().GetDiscordClient().Pin(m_active_channel, m_menu_selected_message, [](...) {});
|
|
||||||
});
|
|
||||||
m_menu.append(*m_menu_pin);
|
|
||||||
|
|
||||||
m_menu.show();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatList::Clear() {
|
void ChatList::Clear() {
|
||||||
@ -104,6 +43,7 @@ void ChatList::SetActiveChannel(Snowflake id) {
|
|||||||
void ChatList::ProcessNewMessage(const Message &data, bool prepend) {
|
void ChatList::ProcessNewMessage(const Message &data, bool prepend) {
|
||||||
auto &discord = Abaddon::Get().GetDiscordClient();
|
auto &discord = Abaddon::Get().GetDiscordClient();
|
||||||
if (!discord.IsStarted()) return;
|
if (!discord.IsStarted()) return;
|
||||||
|
if (!prepend) m_ignore_next_upper = true;
|
||||||
|
|
||||||
// delete preview message when gateway sends it back
|
// delete preview message when gateway sends it back
|
||||||
if (!data.IsPending && data.Nonce.has_value() && data.Author.ID == discord.GetUserData().ID) {
|
if (!data.IsPending && data.Nonce.has_value() && data.Author.ID == discord.GetUserData().ID) {
|
||||||
@ -312,19 +252,85 @@ void ChatList::ActuallyRemoveMessage(Snowflake id) {
|
|||||||
RemoveMessageAndHeader(it->second);
|
RemoveMessageAndHeader(it->second);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatList::OnUpperAdjustmentChanged() {
|
void ChatList::SetupMenu() {
|
||||||
const auto v = get_vadjustment();
|
m_menu_copy_id = Gtk::manage(new Gtk::MenuItem("Copy ID"));
|
||||||
const auto upper = v->get_upper();
|
m_menu_copy_id->signal_activate().connect([this] {
|
||||||
if (m_needs_upper_adjustment && m_old_upper > -1.0) {
|
Gtk::Clipboard::get()->set_text(std::to_string(m_menu_selected_message));
|
||||||
const auto inc = upper - m_old_upper;
|
});
|
||||||
v->set_value(v->get_value() + inc);
|
m_menu_copy_id->show();
|
||||||
}
|
m_menu.append(*m_menu_copy_id);
|
||||||
m_old_upper = v->get_upper();
|
|
||||||
|
m_menu_delete_message = Gtk::manage(new Gtk::MenuItem("Delete Message"));
|
||||||
|
m_menu_delete_message->signal_activate().connect([this] {
|
||||||
|
Abaddon::Get().GetDiscordClient().DeleteMessage(m_active_channel, m_menu_selected_message);
|
||||||
|
});
|
||||||
|
m_menu_delete_message->show();
|
||||||
|
m_menu.append(*m_menu_delete_message);
|
||||||
|
|
||||||
|
m_menu_edit_message = Gtk::manage(new Gtk::MenuItem("Edit Message"));
|
||||||
|
m_menu_edit_message->signal_activate().connect([this] {
|
||||||
|
m_signal_action_message_edit.emit(m_active_channel, m_menu_selected_message);
|
||||||
|
});
|
||||||
|
m_menu_edit_message->show();
|
||||||
|
m_menu.append(*m_menu_edit_message);
|
||||||
|
|
||||||
|
m_menu_copy_content = Gtk::manage(new Gtk::MenuItem("Copy Content"));
|
||||||
|
m_menu_copy_content->signal_activate().connect([this] {
|
||||||
|
const auto msg = Abaddon::Get().GetDiscordClient().GetMessage(m_menu_selected_message);
|
||||||
|
if (msg.has_value())
|
||||||
|
Gtk::Clipboard::get()->set_text(msg->Content);
|
||||||
|
});
|
||||||
|
m_menu_copy_content->show();
|
||||||
|
m_menu.append(*m_menu_copy_content);
|
||||||
|
|
||||||
|
m_menu_reply_to = Gtk::manage(new Gtk::MenuItem("Reply To"));
|
||||||
|
m_menu_reply_to->signal_activate().connect([this] {
|
||||||
|
m_signal_action_reply_to.emit(m_menu_selected_message);
|
||||||
|
});
|
||||||
|
m_menu_reply_to->show();
|
||||||
|
m_menu.append(*m_menu_reply_to);
|
||||||
|
|
||||||
|
m_menu_unpin = Gtk::manage(new Gtk::MenuItem("Unpin"));
|
||||||
|
m_menu_unpin->signal_activate().connect([this] {
|
||||||
|
Abaddon::Get().GetDiscordClient().Unpin(m_active_channel, m_menu_selected_message, [](...) {});
|
||||||
|
});
|
||||||
|
m_menu.append(*m_menu_unpin);
|
||||||
|
|
||||||
|
m_menu_pin = Gtk::manage(new Gtk::MenuItem("Pin"));
|
||||||
|
m_menu_pin->signal_activate().connect([this] {
|
||||||
|
Abaddon::Get().GetDiscordClient().Pin(m_active_channel, m_menu_selected_message, [](...) {});
|
||||||
|
});
|
||||||
|
m_menu.append(*m_menu_pin);
|
||||||
|
|
||||||
|
m_menu.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatList::ScrollToBottom() {
|
void ChatList::ScrollToBottom() {
|
||||||
auto x = get_vadjustment();
|
auto v = get_vadjustment();
|
||||||
x->set_value(x->get_upper());
|
v->set_value(v->get_upper());
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChatList::OnVAdjustmentValueChanged() {
|
||||||
|
auto v = get_vadjustment();
|
||||||
|
if (m_history_timer.elapsed() > 1 && v->get_value() < 500) {
|
||||||
|
m_history_timer.start();
|
||||||
|
m_signal_action_chat_load_history.emit(m_active_channel);
|
||||||
|
}
|
||||||
|
m_should_scroll_to_bottom = v->get_upper() - v->get_page_size() <= v->get_value();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChatList::OnVAdjustmentUpperChanged() {
|
||||||
|
auto v = get_vadjustment();
|
||||||
|
if (!m_ignore_next_upper && !m_should_scroll_to_bottom && m_old_upper > -1.0) {
|
||||||
|
const auto inc = v->get_upper() - m_old_upper;
|
||||||
|
v->set_value(v->get_value() + inc);
|
||||||
|
}
|
||||||
|
m_ignore_next_upper = false;
|
||||||
|
m_old_upper = v->get_upper();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChatList::OnListSizeAllocate(Gtk::Allocation &allocation) {
|
||||||
|
if (m_should_scroll_to_bottom) ScrollToBottom();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatList::RemoveMessageAndHeader(Gtk::Widget *widget) {
|
void ChatList::RemoveMessageAndHeader(Gtk::Widget *widget) {
|
||||||
|
@ -26,8 +26,11 @@ public:
|
|||||||
void ActuallyRemoveMessage(Snowflake id); // perhaps not the best method name
|
void ActuallyRemoveMessage(Snowflake id); // perhaps not the best method name
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void OnUpperAdjustmentChanged();
|
void SetupMenu();
|
||||||
void ScrollToBottom();
|
void ScrollToBottom();
|
||||||
|
void OnVAdjustmentValueChanged();
|
||||||
|
void OnVAdjustmentUpperChanged();
|
||||||
|
void OnListSizeAllocate(Gtk::Allocation &allocation);
|
||||||
void RemoveMessageAndHeader(Gtk::Widget *widget);
|
void RemoveMessageAndHeader(Gtk::Widget *widget);
|
||||||
|
|
||||||
bool m_use_pinned_menu = false;
|
bool m_use_pinned_menu = false;
|
||||||
@ -48,14 +51,14 @@ private:
|
|||||||
int m_num_rows = 0;
|
int m_num_rows = 0;
|
||||||
std::map<Snowflake, Gtk::Widget *> m_id_to_widget;
|
std::map<Snowflake, Gtk::Widget *> m_id_to_widget;
|
||||||
|
|
||||||
|
bool m_ignore_next_upper = false;
|
||||||
|
double m_old_upper = -1.0;
|
||||||
bool m_should_scroll_to_bottom = true;
|
bool m_should_scroll_to_bottom = true;
|
||||||
Gtk::ListBox m_list;
|
Gtk::ListBox m_list;
|
||||||
|
|
||||||
bool m_separate_all = false;
|
bool m_separate_all = false;
|
||||||
|
|
||||||
Glib::Timer m_history_timer;
|
Glib::Timer m_history_timer;
|
||||||
bool m_needs_upper_adjustment = false;
|
|
||||||
double m_old_upper = -1.0;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// these are all forwarded by the parent
|
// these are all forwarded by the parent
|
||||||
@ -93,7 +96,6 @@ private:
|
|||||||
|
|
||||||
template<typename Iter>
|
template<typename Iter>
|
||||||
inline void ChatList::SetMessages(Iter begin, Iter end) {
|
inline void ChatList::SetMessages(Iter begin, Iter end) {
|
||||||
m_needs_upper_adjustment = false;
|
|
||||||
Clear();
|
Clear();
|
||||||
m_num_rows = 0;
|
m_num_rows = 0;
|
||||||
m_num_messages = 0;
|
m_num_messages = 0;
|
||||||
@ -107,7 +109,6 @@ inline void ChatList::SetMessages(Iter begin, Iter end) {
|
|||||||
|
|
||||||
template<typename Iter>
|
template<typename Iter>
|
||||||
inline void ChatList::PrependMessages(Iter begin, Iter end) {
|
inline void ChatList::PrependMessages(Iter begin, Iter end) {
|
||||||
m_needs_upper_adjustment = true;
|
|
||||||
for (Iter it = begin; it != end; it++)
|
for (Iter it = begin; it != end; it++)
|
||||||
ProcessNewMessage(*it, true);
|
ProcessNewMessage(*it, true);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user