2019-06-15 10:19:44 +00:00
|
|
|
using System.Collections.Generic;
|
2019-05-11 22:44:02 +00:00
|
|
|
using System.Globalization;
|
2019-04-21 13:33:22 +00:00
|
|
|
using System.Linq;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Discord;
|
2019-06-15 10:19:44 +00:00
|
|
|
using NodaTime;
|
2019-04-21 13:33:22 +00:00
|
|
|
|
|
|
|
namespace PluralKit.Bot {
|
|
|
|
public class EmbedService {
|
|
|
|
private SystemStore _systems;
|
2019-05-11 22:44:02 +00:00
|
|
|
private MemberStore _members;
|
2019-06-15 10:43:35 +00:00
|
|
|
private SwitchStore _switches;
|
2019-06-21 11:49:58 +00:00
|
|
|
private MessageStore _messages;
|
2019-04-21 13:33:22 +00:00
|
|
|
private IDiscordClient _client;
|
|
|
|
|
2019-06-21 11:49:58 +00:00
|
|
|
public EmbedService(SystemStore systems, MemberStore members, IDiscordClient client, SwitchStore switches, MessageStore messages)
|
2019-04-21 13:33:22 +00:00
|
|
|
{
|
2019-06-15 10:43:35 +00:00
|
|
|
_systems = systems;
|
|
|
|
_members = members;
|
|
|
|
_client = client;
|
|
|
|
_switches = switches;
|
2019-06-21 11:49:58 +00:00
|
|
|
_messages = messages;
|
2019-04-21 13:33:22 +00:00
|
|
|
}
|
|
|
|
|
2019-04-22 15:10:18 +00:00
|
|
|
public async Task<Embed> CreateSystemEmbed(PKSystem system) {
|
2019-04-21 13:33:22 +00:00
|
|
|
var accounts = await _systems.GetLinkedAccountIds(system);
|
|
|
|
|
|
|
|
// Fetch/render info for all accounts simultaneously
|
|
|
|
var users = await Task.WhenAll(accounts.Select(async uid => (await _client.GetUserAsync(uid)).NameAndMention() ?? $"(deleted account {uid})"));
|
|
|
|
|
|
|
|
var eb = new EmbedBuilder()
|
|
|
|
.WithColor(Color.Blue)
|
|
|
|
.WithTitle(system.Name ?? null)
|
|
|
|
.WithDescription(system.Description?.Truncate(1024))
|
|
|
|
.WithThumbnailUrl(system.AvatarUrl ?? null)
|
|
|
|
.WithFooter($"System ID: {system.Hid}");
|
|
|
|
|
|
|
|
eb.AddField("Linked accounts", string.Join(", ", users));
|
2019-04-21 17:11:05 +00:00
|
|
|
eb.AddField("Members", $"(see `pk;system {system.Hid} list` or `pk;system {system.Hid} list full`)");
|
2019-04-21 13:33:22 +00:00
|
|
|
// TODO: fronter
|
|
|
|
return eb.Build();
|
|
|
|
}
|
2019-04-22 15:10:18 +00:00
|
|
|
|
|
|
|
public Embed CreateLoggedMessageEmbed(PKSystem system, PKMember member, IMessage message, IUser sender) {
|
|
|
|
// TODO: pronouns in ?-reacted response using this card
|
|
|
|
return new EmbedBuilder()
|
|
|
|
.WithAuthor($"#{message.Channel.Name}: {member.Name}", member.AvatarUrl)
|
|
|
|
.WithDescription(message.Content)
|
|
|
|
.WithFooter($"System ID: {system.Hid} | Member ID: {member.Hid} | Sender: ${sender.Username}#{sender.Discriminator} ({sender.Id}) | Message ID: ${message.Id}")
|
|
|
|
.WithTimestamp(message.Timestamp)
|
|
|
|
.Build();
|
|
|
|
}
|
2019-05-11 22:44:02 +00:00
|
|
|
|
|
|
|
public async Task<Embed> CreateMemberEmbed(PKSystem system, PKMember member)
|
|
|
|
{
|
|
|
|
var name = member.Name;
|
|
|
|
if (system.Name != null) name = $"{member.Name} ({system.Name})";
|
|
|
|
|
2019-06-15 10:19:44 +00:00
|
|
|
var color = member.Color?.ToDiscordColor() ?? Color.Default;
|
2019-05-11 22:44:02 +00:00
|
|
|
|
|
|
|
var messageCount = await _members.MessageCount(member);
|
|
|
|
|
|
|
|
var eb = new EmbedBuilder()
|
|
|
|
// TODO: add URL of website when that's up
|
|
|
|
.WithAuthor(name, member.AvatarUrl)
|
|
|
|
.WithColor(color)
|
|
|
|
.WithDescription(member.Description)
|
|
|
|
.WithFooter($"System ID: {system.Hid} | Member ID: {member.Hid}");
|
|
|
|
|
|
|
|
if (member.Birthday != null) eb.AddField("Birthdate", member.BirthdayString);
|
|
|
|
if (member.Pronouns != null) eb.AddField("Pronouns", member.Pronouns);
|
|
|
|
if (messageCount > 0) eb.AddField("Message Count", messageCount);
|
|
|
|
if (member.HasProxyTags) eb.AddField("Proxy Tags", $"{member.Prefix}text{member.Suffix}");
|
|
|
|
|
|
|
|
return eb.Build();
|
|
|
|
}
|
2019-06-15 10:19:44 +00:00
|
|
|
|
2019-06-15 10:43:35 +00:00
|
|
|
public async Task<Embed> CreateFronterEmbed(PKSwitch sw, DateTimeZone zone)
|
2019-06-15 10:19:44 +00:00
|
|
|
{
|
2019-06-15 10:43:35 +00:00
|
|
|
var members = (await _switches.GetSwitchMembers(sw)).ToList();
|
2019-06-15 10:19:44 +00:00
|
|
|
var timeSinceSwitch = SystemClock.Instance.GetCurrentInstant() - sw.Timestamp;
|
|
|
|
return new EmbedBuilder()
|
|
|
|
.WithColor(members.FirstOrDefault()?.Color?.ToDiscordColor() ?? Color.Blue)
|
|
|
|
.AddField("Current fronter", members.Count > 0 ? string.Join(", ", members.Select(m => m.Name)) : "*(no fronter)*", true)
|
2019-06-15 10:33:24 +00:00
|
|
|
.AddField("Since", $"{Formats.ZonedDateTimeFormat.Format(sw.Timestamp.InZone(zone))} ({Formats.DurationFormat.Format(timeSinceSwitch)} ago)", true)
|
2019-06-15 10:19:44 +00:00
|
|
|
.Build();
|
|
|
|
}
|
2019-06-15 10:43:35 +00:00
|
|
|
|
|
|
|
public async Task<Embed> CreateFrontHistoryEmbed(IEnumerable<PKSwitch> sws, DateTimeZone zone)
|
|
|
|
{
|
|
|
|
var outputStr = "";
|
|
|
|
|
|
|
|
PKSwitch lastSw = null;
|
|
|
|
foreach (var sw in sws)
|
|
|
|
{
|
|
|
|
// Fetch member list and format
|
|
|
|
var members = (await _switches.GetSwitchMembers(sw)).ToList();
|
|
|
|
var membersStr = members.Any() ? string.Join(", ", members.Select(m => m.Name)) : "no fronter";
|
|
|
|
|
|
|
|
var switchSince = SystemClock.Instance.GetCurrentInstant() - sw.Timestamp;
|
|
|
|
|
|
|
|
// If this isn't the latest switch, we also show duration
|
|
|
|
if (lastSw != null)
|
|
|
|
{
|
|
|
|
// Calculate the time between the last switch (that we iterated - ie. the next one on the timeline) and the current one
|
|
|
|
var switchDuration = lastSw.Timestamp - sw.Timestamp;
|
|
|
|
outputStr += $"**{membersStr}** ({Formats.ZonedDateTimeFormat.Format(sw.Timestamp.InZone(zone))}, {Formats.DurationFormat.Format(switchSince)} ago, for {Formats.DurationFormat.Format(switchDuration)})\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
outputStr += $"**{membersStr}** ({Formats.ZonedDateTimeFormat.Format(sw.Timestamp.InZone(zone))}, {Formats.DurationFormat.Format(switchSince)} ago)\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
lastSw = sw;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new EmbedBuilder()
|
|
|
|
.WithTitle("Past switches")
|
|
|
|
.WithDescription(outputStr)
|
|
|
|
.Build();
|
|
|
|
}
|
2019-06-21 11:49:58 +00:00
|
|
|
|
2019-06-21 12:13:56 +00:00
|
|
|
public async Task<Embed> CreateMessageInfoEmbed(MessageStore.StoredMessage msg)
|
2019-06-21 11:49:58 +00:00
|
|
|
{
|
|
|
|
var channel = (ITextChannel) await _client.GetChannelAsync(msg.Message.Channel);
|
|
|
|
var serverMsg = await channel.GetMessageAsync(msg.Message.Mid);
|
|
|
|
|
|
|
|
var memberStr = $"{msg.Member.Name} (`{msg.Member.Hid}`)";
|
|
|
|
if (msg.Member.Pronouns != null) memberStr += $"\n*(pronouns: **{msg.Member.Pronouns}**)*";
|
|
|
|
|
|
|
|
return new EmbedBuilder()
|
|
|
|
.WithAuthor(msg.Member.Name, msg.Member.AvatarUrl)
|
2019-06-21 12:13:56 +00:00
|
|
|
.WithDescription(serverMsg?.Content ?? "*(message contents deleted or inaccessible)*")
|
2019-06-21 11:49:58 +00:00
|
|
|
.AddField("System", msg.System.Name != null ? $"{msg.System.Name} (`{msg.System.Hid}`)" : $"`{msg.System.Hid}`", true)
|
|
|
|
.AddField("Member", memberStr, true)
|
|
|
|
.WithTimestamp(SnowflakeUtils.FromSnowflake(msg.Message.Mid))
|
|
|
|
.Build();
|
|
|
|
}
|
2019-04-21 13:33:22 +00:00
|
|
|
}
|
|
|
|
}
|