import os import tempfile import discord from discord.ext import commands from PIL import Image import subprocess import shutil from typing import Optional try: import pillow_heif pillow_heif.register_heif_opener() except Exception: pass def img2gif_command(): @commands.hybrid_command( name="img2gif", description="Convert an uploaded image to a GIF.", ) @commands.cooldown(1, 15, commands.BucketType.user) async def img2gif(self, context, attachment: Optional[discord.Attachment] = None): resolved_attachment = attachment if resolved_attachment is None: if context.message and context.message.reference and context.message.reference.resolved: ref_msg = context.message.reference.resolved if isinstance(ref_msg, discord.Message) and ref_msg.attachments: resolved_attachment = ref_msg.attachments[0] if resolved_attachment is None and context.message and context.message.attachments: resolved_attachment = context.message.attachments[0] if resolved_attachment is None or not resolved_attachment.filename.lower().endswith((".png", ".jpg", ".jpeg", ".webp", ".bmp", ".tiff", ".heic", ".heif")): embed = discord.Embed( title="Error", description="Provide or reply to an image (png/jpg/jpeg/webp/bmp/tiff/heic/heif).", color=0xE02B2B, ) embed.set_author(name="Media", icon_url="https://yes.nighty.works/raw/y5SEZ9.webp") interaction = getattr(context, "interaction", None) if interaction is not None: if not interaction.response.is_done(): await interaction.response.send_message(embed=embed, ephemeral=True) else: await interaction.followup.send(embed=embed, ephemeral=True) else: await context.send(embed=embed, ephemeral=True) return processing_embed = discord.Embed( title="Image to GIF (Processing)", description=" Converting image...", color=0x7289DA, ) processing_embed.set_author(name="Media", icon_url="https://yes.nighty.works/raw/y5SEZ9.webp") interaction = getattr(context, "interaction", None) if interaction is not None: if not interaction.response.is_done(): await interaction.response.send_message(embed=processing_embed, ephemeral=True) else: await interaction.followup.send(embed=processing_embed, ephemeral=True) else: processing_msg = await context.send(embed=processing_embed) tmp_dir = tempfile.mkdtemp() src_path = os.path.join(tmp_dir, resolved_attachment.filename) out_path = os.path.join(tmp_dir, os.path.splitext(resolved_attachment.filename)[0] + ".gif") try: await resolved_attachment.save(src_path) src_for_pillow = src_path try: with Image.open(src_for_pillow) as img: if img.mode not in ("RGB", "RGBA"): img = img.convert("RGBA") duration_ms = 100 loop = 0 img.save(out_path, format="GIF", save_all=True, optimize=True, duration=duration_ms, loop=loop) except Exception: if resolved_attachment.filename.lower().endswith((".heic", ".heif")) and shutil.which("ffmpeg"): png_path = os.path.join(tmp_dir, os.path.splitext(resolved_attachment.filename)[0] + ".png") try: subprocess.run([ "ffmpeg", "-y", "-i", src_path, png_path ], check=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL) with Image.open(png_path) as img: if img.mode not in ("RGB", "RGBA"): img = img.convert("RGBA") duration_ms = 100 loop = 0 img.save(out_path, format="GIF", save_all=True, optimize=True, duration=duration_ms, loop=loop) except Exception as conv_err: raise conv_err else: raise original_ext = os.path.splitext(resolved_attachment.filename)[1].lstrip('.').upper() or "IMAGE" embed = discord.Embed( title="Image to GIF", description=f"Converted {original_ext} to GIF.", color=0x7289DA, ) embed.set_author(name="Media", icon_url="https://yes.nighty.works/raw/y5SEZ9.webp") embed.set_footer(text=f"Requested by {context.author.name}", icon_url=context.author.display_avatar.url) with open(out_path, "rb") as f: file = discord.File(f, filename=os.path.basename(out_path)) if interaction is not None: await context.channel.send(embed=embed) await context.channel.send(file=file) try: await interaction.delete_original_response() except: pass else: await processing_msg.delete() await context.channel.send(embed=embed) await context.channel.send(file=file) except Exception as e: embed = discord.Embed( title="Error", description=f"Failed to convert image: {e}", color=0xE02B2B, ) embed.set_author(name="Media", icon_url="https://yes.nighty.works/raw/y5SEZ9.webp") if interaction is not None: try: await interaction.delete_original_response() except: pass await interaction.followup.send(embed=embed, ephemeral=True) else: try: await processing_msg.delete() except: pass await context.send(embed=embed, ephemeral=True) finally: try: for f in os.listdir(tmp_dir): try: os.remove(os.path.join(tmp_dir, f)) except: pass os.rmdir(tmp_dir) except: pass return img2gif