Add group privacy command/structures
This commit is contained in:
parent
47d5ad0004
commit
5e28e0aba1
@ -53,6 +53,7 @@ namespace PluralKit.Bot
|
|||||||
public static Command GroupDesc = new Command("group description", "group <group> description [description]", "Changes a group's description");
|
public static Command GroupDesc = new Command("group description", "group <group> description [description]", "Changes a group's description");
|
||||||
public static Command GroupAdd = new Command("group add", "group <group> add <member> [member 2] [member 3...]", "Adds one or more members to a group");
|
public static Command GroupAdd = new Command("group add", "group <group> add <member> [member 2] [member 3...]", "Adds one or more members to a group");
|
||||||
public static Command GroupRemove = new Command("group remove", "group <group> remove <member> [member 2] [member 3...]", "Removes one or more members from a group");
|
public static Command GroupRemove = new Command("group remove", "group <group> remove <member> [member 2] [member 3...]", "Removes one or more members from a group");
|
||||||
|
public static Command GroupPrivacy = new Command("group privacy", "group <group> privacy <description|icon|visibility|all> <public|private>", "Changes a group's privacy settings");
|
||||||
public static Command Switch = new Command("switch", "switch <member> [member 2] [member 3...]", "Registers a switch");
|
public static Command Switch = new Command("switch", "switch <member> [member 2] [member 3...]", "Registers a switch");
|
||||||
public static Command SwitchOut = new Command("switch out", "switch out", "Registers a switch with no members");
|
public static Command SwitchOut = new Command("switch out", "switch out", "Registers a switch with no members");
|
||||||
public static Command SwitchMove = new Command("switch move", "switch move <date/time>", "Moves the latest switch in time");
|
public static Command SwitchMove = new Command("switch move", "switch move <date/time>", "Moves the latest switch in time");
|
||||||
@ -342,6 +343,12 @@ namespace PluralKit.Bot
|
|||||||
await ctx.Execute<Groups>(GroupRemove, g => g.AddRemoveMembers(ctx, target, Groups.AddRemoveOperation.Remove));
|
await ctx.Execute<Groups>(GroupRemove, g => g.AddRemoveMembers(ctx, target, Groups.AddRemoveOperation.Remove));
|
||||||
else if (ctx.Match("members", "list", "ms", "l"))
|
else if (ctx.Match("members", "list", "ms", "l"))
|
||||||
await ctx.Execute<Groups>(GroupMemberList, g => g.ListGroupMembers(ctx, target));
|
await ctx.Execute<Groups>(GroupMemberList, g => g.ListGroupMembers(ctx, target));
|
||||||
|
else if (ctx.Match("privacy"))
|
||||||
|
await ctx.Execute<Groups>(GroupPrivacy, g => g.GroupPrivacy(ctx, target, null));
|
||||||
|
else if (ctx.Match("public", "pub"))
|
||||||
|
await ctx.Execute<Groups>(GroupPrivacy, g => g.GroupPrivacy(ctx, target, PrivacyLevel.Public));
|
||||||
|
else if (ctx.Match("private", "priv"))
|
||||||
|
await ctx.Execute<Groups>(GroupPrivacy, g => g.GroupPrivacy(ctx, target, PrivacyLevel.Private));
|
||||||
else if (!ctx.HasNext())
|
else if (!ctx.HasNext())
|
||||||
await ctx.Execute<Groups>(GroupInfo, g => g.ShowGroupCard(ctx, target));
|
await ctx.Execute<Groups>(GroupInfo, g => g.ShowGroupCard(ctx, target));
|
||||||
else
|
else
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using System.Collections.Generic;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
@ -107,7 +108,11 @@ namespace PluralKit.Bot
|
|||||||
// TODO: integrate with the normal "search" system
|
// TODO: integrate with the normal "search" system
|
||||||
await using var conn = await _db.Obtain();
|
await using var conn = await _db.Obtain();
|
||||||
|
|
||||||
var groups = (await conn.QueryGroupsInSystem(system.Id)).ToList();
|
var pctx = LookupContext.ByNonOwner;
|
||||||
|
if (ctx.MatchFlag("a", "all") && system.Id == ctx.System.Id)
|
||||||
|
pctx = LookupContext.ByOwner;
|
||||||
|
|
||||||
|
var groups = (await conn.QueryGroupsInSystem(system.Id)).Where(g => g.Visibility.CanAccess(pctx)).ToList();
|
||||||
if (groups.Count == 0)
|
if (groups.Count == 0)
|
||||||
{
|
{
|
||||||
if (system.Id == ctx.System?.Id)
|
if (system.Id == ctx.System?.Id)
|
||||||
@ -139,6 +144,7 @@ namespace PluralKit.Bot
|
|||||||
await using var conn = await _db.Obtain();
|
await using var conn = await _db.Obtain();
|
||||||
|
|
||||||
var system = await GetGroupSystem(ctx, target, conn);
|
var system = await GetGroupSystem(ctx, target, conn);
|
||||||
|
var pctx = ctx.LookupContextFor(system);
|
||||||
var memberCount = await conn.QueryGroupMemberCount(target.Id, PrivacyLevel.Public);
|
var memberCount = await conn.QueryGroupMemberCount(target.Id, PrivacyLevel.Public);
|
||||||
|
|
||||||
var nameField = target.Name;
|
var nameField = target.Name;
|
||||||
@ -146,7 +152,7 @@ namespace PluralKit.Bot
|
|||||||
nameField = $"{nameField} ({system.Name})";
|
nameField = $"{nameField} ({system.Name})";
|
||||||
|
|
||||||
var eb = new DiscordEmbedBuilder()
|
var eb = new DiscordEmbedBuilder()
|
||||||
.WithAuthor(nameField)
|
.WithAuthor(nameField, iconUrl: DiscordUtils.WorkaroundForUrlBug(target.IconFor(pctx)))
|
||||||
.WithFooter($"System ID: {system.Hid} | Group ID: {target.Hid} | Created on {target.Created.FormatZoned(system)}");
|
.WithFooter($"System ID: {system.Hid} | Group ID: {target.Hid} | Created on {target.Created.FormatZoned(system)}");
|
||||||
|
|
||||||
if (memberCount == 0)
|
if (memberCount == 0)
|
||||||
@ -154,8 +160,11 @@ namespace PluralKit.Bot
|
|||||||
else
|
else
|
||||||
eb.AddField($"Members ({memberCount})", $"(see `pk;group {target.Hid} list`)", true);
|
eb.AddField($"Members ({memberCount})", $"(see `pk;group {target.Hid} list`)", true);
|
||||||
|
|
||||||
if (target.Description != null)
|
if (target.DescriptionFor(pctx) is {} desc)
|
||||||
eb.AddField("Description", target.Description);
|
eb.AddField("Description", desc);
|
||||||
|
|
||||||
|
if (target.IconFor(pctx) is {} icon)
|
||||||
|
eb.WithThumbnail(icon);
|
||||||
|
|
||||||
await ctx.Reply(embed: eb.Build());
|
await ctx.Reply(embed: eb.Build());
|
||||||
}
|
}
|
||||||
@ -224,6 +233,66 @@ namespace PluralKit.Bot
|
|||||||
throw new PKSyntaxError("You must pass one or more members.");
|
throw new PKSyntaxError("You must pass one or more members.");
|
||||||
return members;
|
return members;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task GroupPrivacy(Context ctx, PKGroup target, PrivacyLevel? newValueFromCommand)
|
||||||
|
{
|
||||||
|
ctx.CheckSystem().CheckOwnGroup(target);
|
||||||
|
// Display privacy settings
|
||||||
|
if (!ctx.HasNext() && newValueFromCommand == null)
|
||||||
|
{
|
||||||
|
await ctx.Reply(embed: new DiscordEmbedBuilder()
|
||||||
|
.WithTitle($"Current privacy settings for {target.Name}")
|
||||||
|
.AddField("Description", target.DescriptionPrivacy.Explanation())
|
||||||
|
.AddField("Icon", target.IconPrivacy.Explanation())
|
||||||
|
.AddField("Visibility", target.Visibility.Explanation())
|
||||||
|
.WithDescription("To edit privacy settings, use the command:\n`pk;group <group> privacy <subject> <level>`\n\n- `subject` is one of `description`, `icon`, `visibility`, or `all`\n- `level` is either `public` or `private`.")
|
||||||
|
.Build());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
async Task SetAll(PrivacyLevel level)
|
||||||
|
{
|
||||||
|
await _db.Execute(c => c.UpdateGroup(target.Id, new GroupPatch().WithAllPrivacy(level)));
|
||||||
|
|
||||||
|
if (level == PrivacyLevel.Private)
|
||||||
|
await ctx.Reply($"{Emojis.Success} All {target.Name}'s privacy settings have been set to **{level.LevelName()}**. Other accounts will now see nothing on the member card.");
|
||||||
|
else
|
||||||
|
await ctx.Reply($"{Emojis.Success} All {target.Name}'s privacy settings have been set to **{level.LevelName()}**. Other accounts will now see everything on the member card.");
|
||||||
|
}
|
||||||
|
|
||||||
|
async Task SetLevel(GroupPrivacySubject subject, PrivacyLevel level)
|
||||||
|
{
|
||||||
|
await _db.Execute(c => c.UpdateGroup(target.Id, new GroupPatch().WithPrivacy(subject, level)));
|
||||||
|
|
||||||
|
var subjectName = subject switch
|
||||||
|
{
|
||||||
|
GroupPrivacySubject.Description => "description privacy",
|
||||||
|
GroupPrivacySubject.Icon => "icon privacy",
|
||||||
|
GroupPrivacySubject.Visibility => "visibility",
|
||||||
|
_ => throw new ArgumentOutOfRangeException($"Unknown privacy subject {subject}")
|
||||||
|
};
|
||||||
|
|
||||||
|
var explanation = (subject, level) switch
|
||||||
|
{
|
||||||
|
(GroupPrivacySubject.Description, PrivacyLevel.Private) => "This group's description is now hidden from other systems.",
|
||||||
|
(GroupPrivacySubject.Icon, PrivacyLevel.Private) => "This group's icon is now hidden from other systems.",
|
||||||
|
(GroupPrivacySubject.Visibility, PrivacyLevel.Private) => "This group is now hidden from group lists and member cards.",
|
||||||
|
|
||||||
|
(GroupPrivacySubject.Description, PrivacyLevel.Public) => "This group's description is no longer hidden from other systems.",
|
||||||
|
(GroupPrivacySubject.Icon, PrivacyLevel.Public) => "This group's icon is no longer hidden from other systems.",
|
||||||
|
(GroupPrivacySubject.Visibility, PrivacyLevel.Public) => "This group is no longer hidden from group lists and member cards.",
|
||||||
|
|
||||||
|
_ => throw new InvalidOperationException($"Invalid subject/level tuple ({subject}, {level})")
|
||||||
|
};
|
||||||
|
|
||||||
|
await ctx.Reply($"{Emojis.Success} {target.Name}'s **{subjectName}** has been set to **{level.LevelName()}**. {explanation}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ctx.Match("all") || newValueFromCommand != null)
|
||||||
|
await SetAll(newValueFromCommand ?? ctx.PopPrivacyLevel());
|
||||||
|
else
|
||||||
|
await SetLevel(ctx.PopGroupPrivacySubject(), ctx.PopPrivacyLevel());
|
||||||
|
}
|
||||||
|
|
||||||
private static async Task<PKSystem> GetGroupSystem(Context ctx, PKGroup target, IPKConnection conn)
|
private static async Task<PKSystem> GetGroupSystem(Context ctx, PKGroup target, IPKConnection conn)
|
||||||
{
|
{
|
||||||
|
@ -35,5 +35,14 @@ namespace PluralKit.Bot
|
|||||||
ctx.PopArgument();
|
ctx.PopArgument();
|
||||||
return subject;
|
return subject;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static GroupPrivacySubject PopGroupPrivacySubject(this Context ctx)
|
||||||
|
{
|
||||||
|
if (!GroupPrivacyUtils.TryParseGroupPrivacy(ctx.PeekArgument(), out var subject))
|
||||||
|
throw new PKSyntaxError($"Invalid privacy subject `{ctx.PopArgument()}` (must be `description`, `icon`, `visibility`, or `all).");
|
||||||
|
|
||||||
|
ctx.PopArgument();
|
||||||
|
return subject;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -109,7 +109,7 @@ namespace PluralKit.Bot {
|
|||||||
var guildDisplayName = guildSettings?.DisplayName;
|
var guildDisplayName = guildSettings?.DisplayName;
|
||||||
var avatar = guildSettings?.AvatarUrl ?? member.AvatarFor(ctx);
|
var avatar = guildSettings?.AvatarUrl ?? member.AvatarFor(ctx);
|
||||||
|
|
||||||
var groups = (await conn.QueryMemberGroups(member.Id)).ToList();
|
var groups = (await conn.QueryMemberGroups(member.Id)).Where(g => g.Visibility.CanAccess(ctx)).ToList();
|
||||||
|
|
||||||
var proxyTagsStr = string.Join('\n', member.ProxyTags.Select(t => $"`` {t.ProxyString} ``"));
|
var proxyTagsStr = string.Join('\n', member.ProxyTags.Select(t => $"`` {t.ProxyString} ``"));
|
||||||
|
|
||||||
|
@ -27,5 +27,11 @@ namespace PluralKit.Core
|
|||||||
|
|
||||||
public static int MessageCountFor(this PKMember member, LookupContext ctx) =>
|
public static int MessageCountFor(this PKMember member, LookupContext ctx) =>
|
||||||
member.MetadataPrivacy.Get(ctx, member.MessageCount);
|
member.MetadataPrivacy.Get(ctx, member.MessageCount);
|
||||||
|
|
||||||
|
public static string DescriptionFor(this PKGroup group, LookupContext ctx) =>
|
||||||
|
group.DescriptionPrivacy.Get(ctx, group.Description);
|
||||||
|
|
||||||
|
public static string IconFor(this PKGroup group, LookupContext ctx) =>
|
||||||
|
group.IconPrivacy.Get(ctx, group.Icon);
|
||||||
}
|
}
|
||||||
}
|
}
|
66
PluralKit.Core/Models/Privacy/GroupPrivacySubject.cs
Normal file
66
PluralKit.Core/Models/Privacy/GroupPrivacySubject.cs
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace PluralKit.Core
|
||||||
|
{
|
||||||
|
public enum GroupPrivacySubject
|
||||||
|
{
|
||||||
|
Description,
|
||||||
|
Icon,
|
||||||
|
Visibility
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class GroupPrivacyUtils
|
||||||
|
{
|
||||||
|
public static GroupPatch WithPrivacy(this GroupPatch group, GroupPrivacySubject subject, PrivacyLevel level)
|
||||||
|
{
|
||||||
|
// what do you mean switch expressions can't be statements >.>
|
||||||
|
_ = subject switch
|
||||||
|
{
|
||||||
|
GroupPrivacySubject.Description => group.DescriptionPrivacy = level,
|
||||||
|
GroupPrivacySubject.Icon => group.IconPrivacy = level,
|
||||||
|
GroupPrivacySubject.Visibility => group.Visibility = level,
|
||||||
|
_ => throw new ArgumentOutOfRangeException($"Unknown privacy subject {subject}")
|
||||||
|
};
|
||||||
|
|
||||||
|
return group;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static GroupPatch WithAllPrivacy(this GroupPatch member, PrivacyLevel level)
|
||||||
|
{
|
||||||
|
foreach (var subject in Enum.GetValues(typeof(GroupPrivacySubject)))
|
||||||
|
member.WithPrivacy((GroupPrivacySubject) subject, level);
|
||||||
|
return member;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool TryParseGroupPrivacy(string input, out GroupPrivacySubject subject)
|
||||||
|
{
|
||||||
|
switch (input.ToLowerInvariant())
|
||||||
|
{
|
||||||
|
case "description":
|
||||||
|
case "desc":
|
||||||
|
case "text":
|
||||||
|
case "info":
|
||||||
|
subject = GroupPrivacySubject.Description;
|
||||||
|
break;
|
||||||
|
case "avatar":
|
||||||
|
case "pfp":
|
||||||
|
case "pic":
|
||||||
|
case "icon":
|
||||||
|
subject = GroupPrivacySubject.Icon;
|
||||||
|
break;
|
||||||
|
case "visibility":
|
||||||
|
case "hidden":
|
||||||
|
case "shown":
|
||||||
|
case "visible":
|
||||||
|
case "list":
|
||||||
|
subject = GroupPrivacySubject.Visibility;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
subject = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user