Toastie
414dbae62f
This splits them into different entries making it possible to separately edit different ones at the same time.
412 lines
No EOL
16 KiB
C#
412 lines
No EOL
16 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using DiscordChatExporter.Core.Utils.Extensions;
|
|
using DSharpPlus;
|
|
using DSharpPlus.Commands;
|
|
using DSharpPlus.Commands.ContextChecks;
|
|
using DSharpPlus.Commands.EventArgs;
|
|
using DSharpPlus.Commands.Exceptions;
|
|
using DSharpPlus.Entities;
|
|
using DSharpPlus.EventArgs;
|
|
using DSharpPlus.Exceptions;
|
|
using SupportChild.Commands;
|
|
using SupportChild.Interviews;
|
|
|
|
namespace SupportChild;
|
|
|
|
public static class EventHandler
|
|
{
|
|
public static Task OnReady(DiscordClient client, GuildDownloadCompletedEventArgs e)
|
|
{
|
|
Logger.Log("Connected to Discord.");
|
|
|
|
// Checking activity type
|
|
if (!Enum.TryParse(Config.presenceType, true, out DiscordActivityType activityType))
|
|
{
|
|
Logger.Log("Presence type '" + Config.presenceType + "' invalid, using 'Playing' instead.");
|
|
activityType = DiscordActivityType.Playing;
|
|
}
|
|
|
|
client.UpdateStatusAsync(new DiscordActivity(Config.presenceText, activityType), DiscordUserStatus.Online);
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
public static async Task OnGuildAvailable(DiscordClient discordClient, GuildAvailableEventArgs e)
|
|
{
|
|
Logger.Log("Found Discord server: " + e.Guild.Name + " (" + e.Guild.Id + ")");
|
|
|
|
if (SupportChild.commandLineArgs.serversToLeave.Contains(e.Guild.Id))
|
|
{
|
|
Logger.Warn("LEAVING DISCORD SERVER AS REQUESTED: " + e.Guild.Name + " (" + e.Guild.Id + ")");
|
|
await e.Guild.LeaveAsync();
|
|
return;
|
|
}
|
|
|
|
IReadOnlyDictionary<ulong, DiscordRole> roles = e.Guild.Roles;
|
|
|
|
foreach ((ulong roleID, DiscordRole role) in roles)
|
|
{
|
|
Logger.Debug(role.Name.PadRight(40, '.') + roleID);
|
|
}
|
|
}
|
|
|
|
public static async Task OnMessageCreated(DiscordClient client, MessageCreatedEventArgs e)
|
|
{
|
|
if (e.Author.IsBot)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Ignore messages outside of tickets.
|
|
if (!Database.TryGetOpenTicket(e.Channel.Id, out Database.Ticket ticket))
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Send staff notification if applicable.
|
|
if (Config.ticketUpdatedNotifications)
|
|
{
|
|
await SendTicketUpdatedMessage(e, ticket);
|
|
}
|
|
|
|
// Try to process the message as an interview response if the ticket owner replied to this bot.
|
|
if (ticket.creatorID == e.Author.Id && e.Message.ReferencedMessage?.Author == client.CurrentUser)
|
|
{
|
|
await Interviewer.ProcessResponseMessage(e.Message);
|
|
}
|
|
}
|
|
|
|
private static async Task SendTicketUpdatedMessage(MessageCreatedEventArgs e, Database.Ticket ticket)
|
|
{
|
|
// Ignore staff messages
|
|
if (Database.IsStaff(e.Author.Id))
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Sends a DM to the assigned staff member if at least a day has gone by since the last message
|
|
IReadOnlyList<DiscordMessage> messages = await e.Channel.GetMessagesAsync(2);
|
|
if (messages.Count > 1 && messages[1].Timestamp < DateTimeOffset.UtcNow.AddDays(Config.ticketUpdatedNotificationDelay * -1))
|
|
{
|
|
try
|
|
{
|
|
DiscordMember staffMember = await e.Guild.GetMemberAsync(ticket.assignedStaffID);
|
|
await staffMember.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Green,
|
|
Description = "A ticket you are assigned to has been updated: " + e.Channel.Mention
|
|
});
|
|
}
|
|
catch (NotFoundException) { }
|
|
catch (UnauthorizedException) { }
|
|
}
|
|
}
|
|
|
|
public static async Task OnMemberAdded(DiscordClient client, GuildMemberAddedEventArgs e)
|
|
{
|
|
if (!Database.TryGetOpenTickets(e.Member.Id, out List<Database.Ticket> ownTickets))
|
|
{
|
|
return;
|
|
}
|
|
|
|
foreach (Database.Ticket ticket in ownTickets)
|
|
{
|
|
try
|
|
{
|
|
DiscordChannel channel = await client.GetChannelAsync(ticket.channelID);
|
|
if (channel?.GuildId == e.Guild.Id)
|
|
{
|
|
try
|
|
{
|
|
await channel.AddOverwriteAsync(e.Member, DiscordPermissions.AccessChannels);
|
|
await channel.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Green,
|
|
Description = "User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has rejoined the server, and has been re-added to the ticket."
|
|
});
|
|
}
|
|
catch (DiscordException ex)
|
|
{
|
|
Logger.Error("Exception occurred trying to add channel permissions: " + ex);
|
|
Logger.Error("JsonMessage: " + ex.JsonMessage);
|
|
}
|
|
|
|
}
|
|
}
|
|
catch (Exception) { /* ignored */ }
|
|
}
|
|
}
|
|
|
|
public static async Task OnMemberRemoved(DiscordClient client, GuildMemberRemovedEventArgs e)
|
|
{
|
|
if (Database.TryGetOpenTickets(e.Member.Id, out List<Database.Ticket> ownTickets))
|
|
{
|
|
foreach (Database.Ticket ticket in ownTickets)
|
|
{
|
|
try
|
|
{
|
|
DiscordChannel channel = await client.GetChannelAsync(ticket.channelID);
|
|
if (channel?.GuildId == e.Guild.Id)
|
|
{
|
|
await channel.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = "User '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server."
|
|
});
|
|
}
|
|
}
|
|
catch (Exception) { /* ignored */ }
|
|
}
|
|
}
|
|
|
|
if (Database.TryGetAssignedTickets(e.Member.Id, out List<Database.Ticket> assignedTickets) && Config.logChannel != 0)
|
|
{
|
|
DiscordChannel logChannel = await client.GetChannelAsync(Config.logChannel);
|
|
if (logChannel != null)
|
|
{
|
|
foreach (Database.Ticket ticket in assignedTickets)
|
|
{
|
|
try
|
|
{
|
|
DiscordChannel channel = await client.GetChannelAsync(ticket.channelID);
|
|
if (channel?.GuildId == e.Guild.Id)
|
|
{
|
|
await logChannel.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = "Assigned staff member '" + e.Member.Username + "#" + e.Member.Discriminator + "' has left the server: <#" + channel.Id + ">"
|
|
});
|
|
}
|
|
}
|
|
catch (Exception) { /* ignored */ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public static async Task OnComponentInteractionCreated(DiscordClient client, ComponentInteractionCreatedEventArgs e)
|
|
{
|
|
try
|
|
{
|
|
switch (e.Interaction.Data.ComponentType)
|
|
{
|
|
case DiscordComponentType.Button:
|
|
switch (e.Id)
|
|
{
|
|
case "supportchild_closeconfirm":
|
|
await CloseCommand.OnConfirmed(e.Interaction);
|
|
return;
|
|
case not null when e.Id.StartsWith("supportchild_newcommandbutton"):
|
|
await OnNewTicketSelectorUsed(e.Interaction);
|
|
return;
|
|
case not null when e.Id.StartsWith("supportchild_newticketbutton"):
|
|
await OnNewTicketButtonUsed(e.Interaction);
|
|
return;
|
|
case not null when e.Id.StartsWith("supportchild_interviewbutton"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
case "right":
|
|
case "left":
|
|
case "rightskip":
|
|
case "leftskip":
|
|
case "stop":
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown button press received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
case DiscordComponentType.StringSelect:
|
|
switch (e.Id)
|
|
{
|
|
case not null when e.Id.StartsWith("supportchild_newcommandselector"):
|
|
await OnNewTicketSelectorUsed(e.Interaction);
|
|
return;
|
|
case not null when e.Id.StartsWith("supportchild_newticketselector"):
|
|
await CreateSelectionBoxPanelCommand.OnSelectionMenuUsed(e.Interaction);
|
|
return;
|
|
case not null when e.Id.StartsWith("supportchild_interviewselector"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown selection box option received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
case DiscordComponentType.ActionRow:
|
|
Logger.Warn("Unknown action row received! '" + e.Id + "'");
|
|
return;
|
|
case DiscordComponentType.FormInput:
|
|
Logger.Warn("Unknown form input received! '" + e.Id + "'");
|
|
return;
|
|
case DiscordComponentType.UserSelect:
|
|
switch (e.Id)
|
|
{
|
|
case not null when e.Id.StartsWith("supportchild_interviewuserselector"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown selection box option received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
case DiscordComponentType.RoleSelect:
|
|
switch (e.Id)
|
|
{
|
|
case not null when e.Id.StartsWith("supportchild_interviewroleselector"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown selection box option received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
case DiscordComponentType.MentionableSelect:
|
|
switch (e.Id)
|
|
{
|
|
case not null when e.Id.StartsWith("supportchild_interviewmentionableselector"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown selection box option received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
case DiscordComponentType.ChannelSelect:
|
|
switch (e.Id)
|
|
{
|
|
case not null when e.Id.StartsWith("supportchild_interviewchannelselector"):
|
|
await Interviewer.ProcessButtonOrSelectorResponse(e.Interaction);
|
|
return;
|
|
default:
|
|
Logger.Warn("Unknown selection box option received! '" + e.Id + "'");
|
|
return;
|
|
}
|
|
default:
|
|
Logger.Warn("Unknown interaction type received! '" + e.Interaction.Data.ComponentType + "'");
|
|
break;
|
|
}
|
|
}
|
|
catch (DiscordException ex)
|
|
{
|
|
Logger.Error("Interaction Exception occurred: " + ex);
|
|
Logger.Error("JsomMessage: " + ex.JsonMessage);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logger.Error("Interaction Exception occured: " + ex.GetType() + ": " + ex);
|
|
}
|
|
}
|
|
|
|
private static async Task OnNewTicketButtonUsed(DiscordInteraction interaction)
|
|
{
|
|
await interaction.CreateResponseAsync(DiscordInteractionResponseType.DeferredChannelMessageWithSource, new DiscordInteractionResponseBuilder().AsEphemeral());
|
|
|
|
if (!ulong.TryParse(interaction.Data.CustomId.Replace("supportchild_newticketbutton ", ""), out ulong categoryID) || categoryID == 0)
|
|
{
|
|
Logger.Warn("Invalid ticket button ID: " + interaction.Data.CustomId.Replace("supportchild_newticketbutton ", ""));
|
|
return;
|
|
}
|
|
|
|
(bool success, string message) = await NewCommand.OpenNewTicket(interaction.User.Id, interaction.ChannelId, categoryID);
|
|
|
|
if (success)
|
|
{
|
|
await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Green,
|
|
Description = message
|
|
}));
|
|
}
|
|
else
|
|
{
|
|
await interaction.CreateFollowupMessageAsync(new DiscordFollowupMessageBuilder().AddEmbed(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = message
|
|
}));
|
|
}
|
|
}
|
|
|
|
private static async Task OnNewTicketSelectorUsed(DiscordInteraction interaction)
|
|
{
|
|
string stringID;
|
|
switch (interaction.Data.ComponentType)
|
|
{
|
|
case DiscordComponentType.Button:
|
|
stringID = interaction.Data.CustomId.Replace("supportchild_newcommandbutton ", "");
|
|
break;
|
|
case DiscordComponentType.StringSelect:
|
|
if (interaction.Data.Values == null || interaction.Data.Values.Length <= 0)
|
|
{
|
|
return;
|
|
}
|
|
stringID = interaction.Data.Values[0];
|
|
break;
|
|
|
|
case DiscordComponentType.ActionRow:
|
|
case DiscordComponentType.FormInput:
|
|
default:
|
|
return;
|
|
}
|
|
|
|
if (!ulong.TryParse(stringID, out ulong categoryID) || categoryID == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
await interaction.CreateResponseAsync(DiscordInteractionResponseType.DeferredMessageUpdate, new DiscordInteractionResponseBuilder().AsEphemeral());
|
|
|
|
(bool success, string message) = await NewCommand.OpenNewTicket(interaction.User.Id, interaction.ChannelId, categoryID);
|
|
|
|
if (success)
|
|
{
|
|
await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Green,
|
|
Description = message
|
|
}));
|
|
}
|
|
else
|
|
{
|
|
await interaction.EditOriginalResponseAsync(new DiscordWebhookBuilder().AddEmbed(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = message
|
|
}));
|
|
}
|
|
}
|
|
|
|
public static async Task OnCommandError(CommandsExtension commandSystem, CommandErroredEventArgs e)
|
|
{
|
|
switch (e.Exception)
|
|
{
|
|
case ChecksFailedException checksFailedException:
|
|
{
|
|
foreach (ContextCheckFailedData error in checksFailedException.Errors)
|
|
{
|
|
await e.Context.Channel.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = error.ErrorMessage
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
|
|
case BadRequestException ex:
|
|
Logger.Error("Command exception occured:\n" + e.Exception);
|
|
Logger.Error("JSON Message: " + ex.JsonMessage);
|
|
return;
|
|
|
|
default:
|
|
{
|
|
Logger.Error("Exception occured: " + e.Exception.GetType() + ": " + e.Exception);
|
|
await e.Context.Channel.SendMessageAsync(new DiscordEmbedBuilder
|
|
{
|
|
Color = DiscordColor.Red,
|
|
Description = "Internal error occured, please report this to the developer."
|
|
});
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
} |