from textwrap import shorten from asyncio import TimeoutError as AsyncTimeoutError from typing import Union import discord from redbot.core import checks from redbot.core import commands from redbot.core.config import Config from redbot.core.i18n import Translator, cog_i18n from redbot.core.utils import chat_formatting as chat from redbot.core.utils.menus import menu, DEFAULT_CONTROLS from redbot.core.utils.mod import get_audit_reason from redbot.core.utils.predicates import ReactionPredicate from tabulate import tabulate from typing import Literal, List import asyncio import aiohttp try: from redbot import json # support of Draper's branch except ImportError: import json from .discord_new_features import edit_role_icon _ = Translator("PersonalRoles", __file__) async def has_assigned_role(ctx): auto_roles = set(await ctx.cog.config.guild(ctx.guild).auto_roles()) user_roles = {r.id for r in ctx.author.roles} user_roles &= auto_roles return len(user_roles) > 0 or ctx.guild.get_role(await ctx.cog.config.member(ctx.author).role()) async def role_icons_feature(ctx): """Check for ROLE_ICONS feature""" return "ROLE_ICONS" in ctx.guild.features @cog_i18n(_) class PersonalRoles(commands.Cog): """Assign and edit personal roles""" __version__ = "2.2.0" # noinspection PyMissingConstructor def __init__(self, bot: commands.Bot): self.bot = bot self.config = Config.get_conf(self, identifier=0x3D86BBD3E2B744AE8AA8B5D986EB4DD8, force_registration=True) default_member = {"role": None, "auto_role": False} default_guild = {"blacklist": [], "auto_roles": [], "position_role": None} self.config.register_member(**default_member) self.config.register_guild(**default_guild) self.session = aiohttp.ClientSession(json_serialize=json.dumps) def cog_unload(self): self.bot.loop.create_task(self.session.close()) @commands.group() @commands.guild_only() @checks.bot_has_permissions(manage_roles=True) async def myrole(self, ctx): """Control of personal role""" pass @myrole.command() @checks.admin_or_permissions(manage_roles=True) async def assign(self, ctx, user: discord.Member, *, role: discord.Role): """Assign personal role to someone""" await self.config.member(user).role.set(role.id) await ctx.send( _("Ok. I just assigned {user.name} ({user.id}) to role {role.name} ({role.id}).").format( user=user, role=role ) ) @myrole.command() @checks.admin_or_permissions(manage_roles=True) async def unassign(self, ctx, *, user: discord.Member): """Unassign personal role from someone""" role = await self.config.member(user).role() await self.config.member(user).role.clear() role = ctx.guild.get_role(role) if role: try: await role.delete() except: await ctx.send( _( "Ok. I just unassigned {user.name} ({user.id}) from their personal role.\nHowever, I could not delete the role." ).format(user=user) ) else: await ctx.send("User didn't have a role or it wasn't found. Role unassigned anyway to make sure.") return await ctx.send( _("Ok. I just unassigned {user.name} ({user.id}) from their personal role and deleted the role.").format( user=user ) ) @myrole.command(name="list") @checks.admin_or_permissions(manage_roles=True) async def mr_list(self, ctx): """Assigned roles list""" members_data = await self.config.all_members(ctx.guild) if not members_data: await ctx.send(chat.info(_("There is no assigned personal roles on this server"))) return assigned_roles = [] for member, data in members_data.items(): if not data["role"]: continue dic = { _("User"): ctx.guild.get_member(member) or f"[X] {member}", _("Role"): shorten( str(ctx.guild.get_role(data["role"]) or "[X] {}".format(data["role"])), 32, placeholder="…", ), } assigned_roles.append(dic) if not assigned_roles: await ctx.send(chat.info(_("There is no assigned personal roles on this server"))) return pages = list(chat.pagify(tabulate(assigned_roles, headers="keys", tablefmt="orgtbl"))) pages = [chat.box(page) for page in pages] await menu(ctx, pages, DEFAULT_CONTROLS) @myrole.group(name="auto") @checks.admin_or_permissions(manage_roles=True) async def myrole_auto(self, ctx): """ Manage Auto role creation settings """ pass @myrole_auto.command(name="list") async def myrole_auto_list(self, ctx): """ List the roles that allow auto creation of assigned roles """ guild = ctx.guild curr = await self.config.guild(guild).auto_roles() roles = [guild.get_role(r) for r in curr] names = [r.name for r in roles if r is not None] msg = "" if None in curr: msg += chat.warning("Some auto roles cannot be found, they have been removed from the list.\n") for i, role in enumerate(roles): if role is None: del curr[i] await self.config.guild(guild).auto_roles.set(curr) await ctx.send(f"Current auto roles: {chat.humanize_list(names) if names else chat.bold('None')}") @myrole_auto.command(name="roles") async def myrole_auto_autoroles(self, ctx, *role_list: discord.Role): """ Set roles that a user must have to allow auto creation of assigned roles by the bot. If user has any one of the roles in the list, an assigned role will be automatically created for them and assigned. Role list should be a list of role ids, mentions, and/or names (must use quotes for names, i.e "My Role"). Roles in role list already set for autorole will be removed, and roles not set for autrole will be added. """ guild = ctx.guild added = set() removed = set() async with self.config.guild(guild).auto_roles() as auto_roles: for role in role_list: if role.id in auto_roles: auto_roles.remove(role.id) removed.add(role.name) else: auto_roles.append(role.id) added.add(role.name) msg = "" if added: msg += "Added: {}\n".format(chat.humanize_list(list(added))) if removed: msg += "Removed: {}".format(chat.humanize_list(list(removed))) await ctx.send(msg) @myrole_auto.command(name="pos") async def myrole_auto_autopos(self, ctx, *, role_pos: discord.Role): """ Set position of where new roles are auto created. New roles will be created under this role. Cannot be higher than the bot's highest role """ if role_pos > ctx.guild.me.top_role: await ctx.send("The role must be under my highest role.") return await self.config.guild(ctx.guild).position_role.set(role_pos.id) await ctx.tick() @myrole.group() @commands.guild_only() @checks.admin_or_permissions(manage_roles=True) async def blacklist(self, ctx): """Manage blacklisted names""" pass @blacklist.command() @checks.admin_or_permissions(manage_roles=True) async def add(self, ctx, *, rolename: str): """Add rolename to blacklist Members will be not able to change name of role to blacklisted names""" rolename = rolename.casefold() async with self.config.guild(ctx.guild).blacklist() as blacklist: if rolename in blacklist: await ctx.send(chat.error(_("`{}` is already in blacklist").format(rolename))) else: blacklist.append(rolename) await ctx.send(chat.info(_("Added `{}` to blacklisted roles list").format(rolename))) @blacklist.command() @checks.admin_or_permissions(manage_roles=True) async def remove(self, ctx, *, rolename: str): """Remove rolename from blacklist""" rolename = rolename.casefold() async with self.config.guild(ctx.guild).blacklist() as blacklist: if rolename not in blacklist: await ctx.send(chat.error(_("`{}` is not blacklisted").format(rolename))) else: blacklist.remove(rolename) await ctx.send(chat.info(_("Removed `{}` from blacklisted roles list").format(rolename))) @blacklist.command(name="list") @checks.admin_or_permissions(manage_roles=True) async def bl_list(self, ctx): """List of blacklisted role names""" blacklist = await self.config.guild(ctx.guild).blacklist() pages = [chat.box(page) for page in chat.pagify("\n".join(blacklist))] if pages: await menu(ctx, pages, DEFAULT_CONTROLS) else: await ctx.send(chat.info(_("There is no blacklisted roles"))) @commands.cooldown(1, 30, commands.BucketType.user) @myrole.command(aliases=["color"]) @commands.guild_only() @commands.check(has_assigned_role) @commands.bot_has_permissions(manage_roles=True) async def colour(self, ctx, *, colour: discord.Colour = discord.Colour.default()): """Change color of personal role""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: await ctx.send(chat.warning(f"Please create your role using `{ctx.prefix}myrole create`!")) return try: await role.edit(colour=colour, reason=get_audit_reason(ctx.author, _("Personal Role"))) except discord.Forbidden: ctx.command.reset_cooldown(ctx) await ctx.send( chat.error( _( "Unable to edit role.\n" 'Role must be lower than my top role and I must have permission "Manage Roles"' ) ) ) except discord.HTTPException as e: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role: {}").format(e))) else: if not colour.value: await ctx.send(_("Reset {user}'s personal role color").format(user=ctx.message.author.name)) else: await ctx.send( _("Changed color of {user}'s personal role to {color}").format( user=ctx.message.author.name, color=colour ) ) @commands.cooldown(1, 30, commands.BucketType.user) @myrole.command() @commands.guild_only() @commands.check(has_assigned_role) @commands.bot_has_permissions(manage_roles=True) async def name(self, ctx, *, name: str): """Change name of personal role You can't use blacklisted names Names must be 30 characters or less""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: await ctx.send(chat.warning(f"Please create your role using `{ctx.prefix}myrole create`!")) return name = name[:30] if name.casefold() in await self.config.guild(ctx.guild).blacklist(): await ctx.send(chat.error(_("This rolename is blacklisted."))) return try: await role.edit(name=name, reason=get_audit_reason(ctx.author, _("Personal Role"))) except discord.Forbidden: ctx.command.reset_cooldown(ctx) await ctx.send( chat.error( _( "Unable to edit role.\n" 'Role must be lower than my top role and i must have permission "Manage Roles"' ) ) ) except discord.HTTPException as e: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role: {}").format(e))) else: await ctx.send( _("Changed name of {user}'s personal role to {name}").format(user=ctx.message.author.name, name=name) ) @commands.cooldown(1, 30, commands.BucketType.user) @myrole.command() @commands.guild_only() @commands.check(has_assigned_role) async def create(self, ctx): """Create personal role if you don't have one already""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: pos = await self.config.guild(ctx.guild).position_role() pos = ctx.guild.get_role(pos) pos = pos.position if pos else 1 try: role = await ctx.guild.create_role( name=str(ctx.author), colour=ctx.author.colour, reason=_("Personal role") ) await asyncio.sleep(0.3) await role.edit(position=pos) await asyncio.sleep(0.3) await ctx.author.add_roles(role, reason=_("Personal Roles")) await self.config.member(ctx.author).role.set(role.id) await self.config.member(ctx.author).auto_role.set(True) except: await ctx.send(chat.warning("Could not create your personal role, please contact an admin.")) return await ctx.send( f"Role created! You can edit it using `{ctx.prefix}myrole name` and `{ctx.prefix}myrole colour` commands." ) else: await ctx.send(chat.warning("You already have a personal role!")) @myrole.group() @commands.check(has_assigned_role) @commands.check(role_icons_feature) @commands.bot_has_permissions(manage_roles=True) async def icon(self, ctx): """Change icon of personal role""" pass @icon.command(name="emoji") @commands.cooldown(1, 30, commands.BucketType.user) async def icon_emoji(self, ctx, *, emoji: Union[discord.Emoji, discord.PartialEmoji] = None): """Change icon of personal role using emoji""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: await ctx.send(chat.warning(f"Please create your role using `{ctx.prefix}myrole create`!")) return if not emoji: if ctx.channel.permissions_for(ctx.author).add_reactions: m = await ctx.send(_("React to this message with your emoji")) try: reaction = await ctx.bot.wait_for( "reaction_add", check=ReactionPredicate.same_context(message=m, user=ctx.author), timeout=30, ) emoji = reaction[0].emoji except AsyncTimeoutError: return finally: await m.delete(delay=0) else: await ctx.send_help() return try: if isinstance(emoji, discord.Emoji): await edit_role_icon( self.bot, role, icon=await emoji.url_as(format="png").read(), reason=get_audit_reason(ctx.author, _("Personal Role")), ) elif isinstance(emoji, discord.PartialEmoji): if emoji.is_custom_emoji(): await edit_role_icon( self.bot, role, icon=await emoji.url_as(format="png").read(), reason=get_audit_reason(ctx.author, _("Personal Role")), ) else: # unicode emoji await edit_role_icon( self.bot, role, unicode_emoji=emoji.name, reason=get_audit_reason(ctx.author, _("Personal Role")), ) else: await edit_role_icon( self.bot, role, unicode_emoji=emoji, reason=get_audit_reason(ctx.author, _("Personal Role")), ) except discord.Forbidden: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role.\nRole must be lower than my top role"))) except discord.InvalidArgument: await ctx.send(chat.error(_("This image type is unsupported, or link is incorrect"))) except discord.HTTPException as e: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role: {}").format(e))) else: await ctx.send(_("Changed icon of {user}'s personal role").format(user=ctx.message.author.name)) @icon.command(name="image", aliases=["url"]) @commands.cooldown(1, 30, commands.BucketType.user) async def icon_image(self, ctx, *, url: str = None): """Change icon of personal role by using image""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: await ctx.send(chat.warning(f"Please create your role using `{ctx.prefix}myrole create`!")) return if not (ctx.message.attachments or url): raise commands.BadArgument if ctx.message.attachments: image = await ctx.message.attachments[0].read() else: try: async with ctx.cog.session.get(url, raise_for_status=True) as resp: image = await resp.read() except aiohttp.ClientResponseError as e: await ctx.send(chat.error(_("Unable to get image: {}").format(e.message))) return try: await edit_role_icon( self.bot, role, icon=image, reason=get_audit_reason(ctx.author, _("Personal Role")), ) except discord.Forbidden: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role.\nRole must be lower than my top role"))) except discord.InvalidArgument: await ctx.send(chat.error(_("This image type is unsupported, or link is incorrect"))) except discord.HTTPException as e: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role: {}").format(e))) else: await ctx.send(_("Changed icon of {user}'s personal role").format(user=ctx.message.author.name)) @icon.command(name="reset", aliases=["remove"]) @commands.cooldown(1, 30, commands.BucketType.user) async def icon_reset(self, ctx): """Remove icon of personal role""" role = await self.config.member(ctx.author).role() role = ctx.guild.get_role(role) if not role: await ctx.send(chat.warning(f"Please create your role using `{ctx.prefix}myrole create`!")) return try: await edit_role_icon( self.bot, role, icon=None, unicode_emoji=None, reason=get_audit_reason(ctx.author, _("Personal Role")), ) await ctx.send(_("Removed icon of {user}'s personal role").format(user=ctx.message.author.name)) except discord.Forbidden: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role.\nRole must be lower than my top role"))) except discord.HTTPException as e: ctx.command.reset_cooldown(ctx) await ctx.send(chat.error(_("Unable to edit role: {}").format(e))) ### Listeners @commands.Cog.listener("on_member_join") async def role_persistance(self, member): """Automatically give already assigned roles on join""" if await self.bot.cog_disabled_in_guild(self, member.guild): return role = await self.config.member(member).role() if role: role = member.guild.get_role(role) if role and member: try: await member.add_roles(role, reason=_("Personal Role")) except discord.Forbidden: pass @commands.Cog.listener("on_member_remove") async def remove_role(self, member): """Delete personal role if member leaves.""" if await self.bot.cog_disabled_in_guild(self, member.guild): return role = await self.config.member(member).role() auto = await self.config.member(member).auto_role() role = member.guild.get_role(role) if auto: await self.config.member(member).role.clear() await self.config.member(member).auto_role.clear() try: await role.delete() except: pass @commands.Cog.listener("on_member_update") async def modify_roles(self, before, after): """Delete personal role if member looses their auto role or looses their personal role""" if await self.bot.cog_disabled_in_guild(self, after.guild): return role = await self.config.member(after).role() role = before.guild.get_role(role) if not role: return auto = await self.config.member(after).auto_role() if not auto: return auto_roles = await self.config.guild(before.guild).auto_roles() if before.roles != after.roles: if role not in after.roles: await self.config.member(after).role.clear() await self.config.member(after).auto_role.clear() try: await role.delete() except: pass else: after_ids = [r.id for r in after.roles] for m_role in after_ids: if m_role in auto_roles: return # lost their auto role, remove personal role, delete, and clear their data. await after.remove_roles(role, reason=_("Personal Roles")) await self.config.member(after).role.clear() await self.config.member(after).auto_role.clear() try: await role.delete() except: pass async def red_delete_data_for_user( self, *, requester: Literal["discord_deleted_user", "owner", "user", "user_strict"], user_id: int, ): pass