use crate::commands::music::get_queue_for_guild; use crate::utils::context_data::EventDrivenMessageContainer; use serenity::async_trait; use serenity::client::Context; use serenity::model::channel::Reaction; use serenity::model::event::ResumedEvent; use serenity::model::gateway::{Activity, Ready}; use serenity::model::guild::Member; use serenity::model::id::{ChannelId, GuildId, MessageId}; use serenity::model::voice::VoiceState; use serenity::prelude::*; pub(crate) struct Handler; macro_rules! log_msg_fire_error { ($msg:expr) => { if let Err(e) = $msg { log::error!("Failed to handle event for message: {:?}", e); } }; } #[async_trait] impl EventHandler for Handler { async fn ready(&self, ctx: Context, ready: Ready) { log::info!("Connected as {}", ready.user.name); let prefix = dotenv::var("BOT_PREFIX").unwrap_or("~!".to_string()); ctx.set_activity(Activity::listening(format!("{}help", prefix).as_str())) .await; } async fn resume(&self, _: Context, _: ResumedEvent) { log::info!("Reconnected to gateway") } async fn voice_state_update( &self, ctx: Context, guild_id: Option, old_state: Option, new_state: VoiceState, ) { let mut member_count = None; let guild_id = if let Some(gid) = guild_id { gid } else { return; }; if let Some(old_id) = old_state.and_then(|c| c.channel_id) { member_count = get_own_channel_member_count(&ctx, &old_id).await; } if member_count.is_none() { if let Some(new_id) = new_state.channel_id { member_count = get_own_channel_member_count(&ctx, &new_id).await; } } if let Some(count) = member_count { log::debug!("{} Members in channel", count); let queue = get_queue_for_guild(&ctx, &guild_id).await.unwrap(); let mut queue_lock = queue.lock().await; log::debug!("Setting leave flag to {}", count == 0); queue_lock.leave_flag = count == 0; } } /// Fired when a message was deleted async fn message_delete( &self, ctx: Context, channel_id: ChannelId, message_id: MessageId, _: Option, ) { let mut data = ctx.data.write().await; let listeners = data.get_mut::().unwrap(); if let Some(msg) = listeners.get(&(channel_id.0, message_id.0)) { log_msg_fire_error!(msg.on_deleted().await); listeners.remove(&(channel_id.0, message_id.0)); } } /// Fired when multiple messages were deleted async fn message_delete_bulk( &self, ctx: Context, channel_id: ChannelId, message_ids: Vec, _: Option, ) { let data = ctx.data.read().await; let listeners = data.get::().unwrap(); for message_id in message_ids { if let Some(msg) = listeners.get(&(channel_id.0, message_id.0)) { log_msg_fire_error!(msg.on_deleted().await); } } } /// Fired when a reaction was added to a message async fn reaction_add(&self, ctx: Context, reaction: Reaction) { let data = ctx.data.read().await; let listeners = data.get::().unwrap(); let message_id = reaction.message_id; let channel_id = reaction.channel_id; if let Some(msg) = listeners.get(&(channel_id.0, message_id.0)) { log_msg_fire_error!(msg.on_reaction_add(reaction).await); } } /// Fired when a reaction was added to a message async fn reaction_remove(&self, ctx: Context, reaction: Reaction) { let data = ctx.data.read().await; let listeners = data.get::().unwrap(); let message_id = reaction.message_id; let channel_id = reaction.channel_id; if let Some(msg) = listeners.get(&(channel_id.0, message_id.0)) { log_msg_fire_error!(msg.on_reaction_remove(reaction).await); } } } /// Returns the number of members in the channel if it's the bots voice channel async fn get_own_channel_member_count(ctx: &Context, channel_id: &ChannelId) -> Option { let channel = ctx.http.get_channel(channel_id.0).await.ok()?; let guild_channel = channel.guild()?; let current_user = ctx.http.get_current_user().await.ok()?; let members = guild_channel.members(&ctx).await.ok()?; let own_channel = members .iter() .find(|m| m.user.id == current_user.id) .is_some(); if !own_channel { return None; } let members: Vec = members.into_iter().filter(|m| !m.user.bot).collect(); Some(members.len()) }