import discord from discord.ext import commands import json import os import math def load_error_codes(): try: json_path = os.path.join(os.path.dirname(__file__), "files/errorcodes.json") with open(json_path, "r", encoding="utf-8") as f: return json.load(f) except FileNotFoundError: return [] except json.JSONDecodeError: return [] class ErrorCodesBrowserView(discord.ui.View): def __init__(self, items_per_page=9): super().__init__(timeout=300) self.error_codes = load_error_codes() self.items_per_page = items_per_page self.current_page = 0 self.max_pages = ( math.ceil(len(self.error_codes) / items_per_page) if self.error_codes else 1 ) self.update_buttons() def update_buttons(self): self.clear_items() first_button = discord.ui.Button( emoji="<:leftmax:1420240325770870905>", style=discord.ButtonStyle.primary, disabled=(self.current_page == 0), ) first_button.callback = self.first_page self.add_item(first_button) prev_button = discord.ui.Button( emoji="<:left:1420240344926126090>", style=discord.ButtonStyle.primary, disabled=(self.current_page == 0), ) prev_button.callback = self.prev_page self.add_item(prev_button) stop_button = discord.ui.Button( emoji="<:middle:1420240356087173160>", style=discord.ButtonStyle.secondary ) stop_button.callback = self.stop_interaction self.add_item(stop_button) next_button = discord.ui.Button( emoji="<:right:1420240334100627456>", style=discord.ButtonStyle.primary, disabled=(self.current_page >= self.max_pages - 1), ) next_button.callback = self.next_page self.add_item(next_button) last_button = discord.ui.Button( emoji="<:rightmax:1420240368846372886>", style=discord.ButtonStyle.primary, disabled=(self.current_page >= self.max_pages - 1), ) last_button.callback = self.last_page self.add_item(last_button) def create_embed(self): if not self.error_codes: embed = discord.Embed( title="Error Codes", description="No error codes found.", color=0xFA8C4A ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) embed.set_footer(text="Page 0 of 0") return embed embed = discord.Embed(title="Error Codes", color=0xFA8C4A) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) start_idx = self.current_page * self.items_per_page end_idx = min(start_idx + self.items_per_page, len(self.error_codes)) current_codes = self.error_codes[start_idx:end_idx] for i, error in enumerate(current_codes): field_value = f"**Code:** `{error.get('code', 'N/A')}`\n**Name:** `{error.get('name', 'Unknown')}`\n**Description:** {error.get('description', 'No description')}" embed.add_field(name="\u200b", value=field_value, inline=True) while len(current_codes) % 3 != 0: embed.add_field(name="\u200b", value="\u200b", inline=True) embed.set_footer(text=f"Page {self.current_page + 1} of {self.max_pages}") return embed async def first_page(self, interaction: discord.Interaction): self.current_page = 0 self.update_buttons() embed = self.create_embed() await interaction.response.edit_message(embed=embed, view=self) async def prev_page(self, interaction: discord.Interaction): if self.current_page > 0: self.current_page -= 1 self.update_buttons() embed = self.create_embed() await interaction.response.edit_message(embed=embed, view=self) async def stop_interaction(self, interaction: discord.Interaction): self.clear_items() embed = self.create_embed() embed.set_footer(text="Interaction stopped") await interaction.response.edit_message(embed=embed, view=self) self.stop() async def next_page(self, interaction: discord.Interaction): if self.current_page < self.max_pages - 1: self.current_page += 1 self.update_buttons() embed = self.create_embed() await interaction.response.edit_message(embed=embed, view=self) async def last_page(self, interaction: discord.Interaction): self.current_page = self.max_pages - 1 self.update_buttons() embed = self.create_embed() await interaction.response.edit_message(embed=embed, view=self) class ideviceSelect(discord.ui.Select): def __init__(self, bot): self.bot = bot options = [ discord.SelectOption( label="No Apps", value="noapps", description="Help when apps aren't showing in installed apps view", ), discord.SelectOption( label="Error Codes", value="errorcodes", description="Browse idevice error codes", ), discord.SelectOption( label="Developer Mode", value="developermode", description="How to turn on developer mode", ), discord.SelectOption( label="Mount DDI", value="mountddi", description="How to manually mount DDI", ), ] super().__init__(placeholder="Choose an idevice command...", options=options) async def callback(self, interaction: discord.Interaction): command_name = self.values[0] if command_name == "errorcodes": try: view = ErrorCodesBrowserView() embed = view.create_embed() success_embed = discord.Embed( title="Command Executed", description="Successfully executed `/errorcodes`. Please run /[errorcode_number] to get more information about an error code, and send it in chat", color=0x00FF00, ) success_embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=success_embed, view=None) await interaction.followup.send(embed=embed, view=view, ephemeral=True) except discord.Forbidden: guild_info = ( f"server {interaction.guild.name} (ID: {interaction.guild.id})" if interaction.guild else "DM or private channel" ) self.bot.logger.warning( f"Bot missing permissions in {guild_info} - cannot execute errorcodes command" ) if interaction.guild is None: embed = discord.Embed( title="Error", description="This command cannot be executed in DMs.", color=0xFF0000, ) else: embed = discord.Embed( title="Permission Error", description="The bot needs the `send messages` permission to execute this command.", color=0xFF0000, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=embed, view=None) except Exception as e: self.bot.logger.error(f"Error executing errorcodes command: {e}") embed = discord.Embed( title="Error", description="An error occurred while executing the command.", color=0xFF0000, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=embed, view=None) return command = self.bot.get_command(command_name) if command: try: ctx = await self.bot.get_context(interaction.message) if ctx: await ctx.invoke(command) embed = discord.Embed( title="Command Executed", description=f"Successfully executed `/{command_name}`", color=0x00FF00, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png", ) await interaction.response.edit_message(embed=embed, view=None) except discord.Forbidden: guild_info = ( f"server {interaction.guild.name} (ID: {interaction.guild.id})" if interaction.guild else "DM or private channel" ) self.bot.logger.warning( f"Bot missing permissions in {guild_info} - cannot execute {command_name} command" ) if interaction.guild is None: embed = discord.Embed( title="Error", description="This command cannot be executed in DMs.", color=0xFF0000, ) else: embed = discord.Embed( title="Permission Error", description="The bot needs the `send messages` permission to execute this command.", color=0xFF0000, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=embed, view=None) except Exception as e: self.bot.logger.error(f"Error executing {command_name} command: {e}") embed = discord.Embed( title="Error", description="An error occurred while executing the command.", color=0xFF0000, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=embed, view=None) else: embed = discord.Embed( title="Error", description="Command not found!", color=0xFF0000 ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) await interaction.response.edit_message(embed=embed, view=None) class ideviceView(discord.ui.View): def __init__(self, bot): super().__init__() self.add_item(ideviceSelect(bot)) def idevice_command(): @commands.hybrid_command( name="idevice", description="idevice troubleshooting and help" ) async def idevice(self, context): if isinstance(context.channel, discord.DMChannel): embed = discord.Embed( title="Error", description="This command can only be used in servers.", color=0xE02B2B, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) if context.interaction: await context.interaction.response.send_message( embed=embed, ephemeral=True ) else: await context.send(embed=embed, ephemeral=True) return if isinstance(context.channel, discord.PartialMessageable): embed = discord.Embed( title="Error", description="The bot needs send messages permissions in this channel.", color=0xE02B2B, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) if context.interaction: await context.interaction.response.send_message( embed=embed, ephemeral=True ) else: await context.send(embed=embed, ephemeral=True) return embed = discord.Embed( title="idevice Commands", description="Choose a command from the dropdown below to get help with specific issues:", color=0xFA8C4A, ) embed.set_author( name="idevice", icon_url="https://yes.nighty.works/raw/snLMuO.png" ) view = ideviceView(self.bot) if context.interaction: await context.interaction.response.send_message( embed=embed, view=view, ephemeral=True ) else: await context.send(embed=embed, view=view) return idevice