b18ab58876
FEAT: Added a loop function to the music module.
159 lines
6.9 KiB
Python
159 lines
6.9 KiB
Python
import discord
|
|
from discord.ext import commands
|
|
from discord import app_commands
|
|
from yt_dlp import YoutubeDL
|
|
import os
|
|
from dotenv import load_dotenv
|
|
|
|
load_dotenv()
|
|
|
|
YTDL_OPTIONS = {
|
|
'format': 'bestaudio',
|
|
'noplaylist': 'True',
|
|
}
|
|
|
|
FFMPEG_OPTIONS = {
|
|
'before_options': '-reconnect 1 -reconnect_streamed 1 -reconnect_delay_max 5',
|
|
'options': '-vn'
|
|
}
|
|
|
|
ytdl = YoutubeDL(YTDL_OPTIONS)
|
|
|
|
class Music(commands.Cog):
|
|
def __init__(self, bot):
|
|
self.bot = bot
|
|
self.queue = []
|
|
self.is_playing = False
|
|
self.volume = 0.3
|
|
self.loop = False # Initialize loop state
|
|
self.current_song = None # Track the current song
|
|
|
|
async def join(self, interaction: discord.Interaction):
|
|
channel = interaction.user.voice.channel
|
|
if interaction.guild.voice_client is None:
|
|
await channel.connect(self_deaf=True) # Join the channel deafened
|
|
await interaction.followup.send("Joined the voice channel.")
|
|
else:
|
|
await interaction.followup.send("Already in a voice channel.")
|
|
|
|
async def leave(self, interaction: discord.Interaction):
|
|
if interaction.guild.voice_client:
|
|
await interaction.guild.voice_client.disconnect()
|
|
await interaction.followup.send("Left the voice channel.")
|
|
else:
|
|
await interaction.followup.send("Not connected to a voice channel.")
|
|
|
|
async def play(self, interaction: discord.Interaction, search: str):
|
|
if not interaction.guild.voice_client:
|
|
await self.join(interaction)
|
|
|
|
info = ytdl.extract_info(f"ytsearch:{search}", download=False)['entries'][0]
|
|
url = info['url']
|
|
|
|
if interaction.guild.voice_client.is_playing():
|
|
self.queue.append((url, info['title']))
|
|
await interaction.followup.send(f'Queued: {info["title"]}')
|
|
else:
|
|
self.current_song = (url, info['title'])
|
|
source = discord.PCMVolumeTransformer(discord.FFmpegPCMAudio(url, **FFMPEG_OPTIONS), volume=self.volume)
|
|
interaction.guild.voice_client.play(source, after=lambda e: self.bot.loop.create_task(self.play_next(interaction)))
|
|
self.is_playing = True
|
|
await interaction.followup.send(f'Playing: {info["title"]}')
|
|
|
|
async def play_next(self, interaction: discord.Interaction):
|
|
if self.loop and self.current_song: # If loop is active, repeat the current song
|
|
url, title = self.current_song
|
|
source = discord.PCMVolumeTransformer(discord.FFmpegPCMAudio(url, **FFMPEG_OPTIONS), volume=self.volume)
|
|
interaction.guild.voice_client.play(source, after=lambda e: self.bot.loop.create_task(self.play_next(interaction)))
|
|
await interaction.followup.send(f'Repeating: {title}')
|
|
elif self.queue:
|
|
url, title = self.queue.pop(0)
|
|
self.current_song = (url, title)
|
|
source = discord.PCMVolumeTransformer(discord.FFmpegPCMAudio(url, **FFMPEG_OPTIONS), volume=self.volume)
|
|
interaction.guild.voice_client.play(source, after=lambda e: self.bot.loop.create_task(self.play_next(interaction)))
|
|
await interaction.followup.send(f'Playing next: {title}')
|
|
else:
|
|
self.is_playing = False
|
|
|
|
async def pause(self, interaction: discord.Interaction):
|
|
if interaction.guild.voice_client.is_playing():
|
|
interaction.guild.voice_client.pause()
|
|
await interaction.followup.send("Paused the song.")
|
|
else:
|
|
await interaction.followup.send("No song is currently playing.")
|
|
|
|
async def resume(self, interaction: discord.Interaction):
|
|
if interaction.guild.voice_client.is_paused():
|
|
interaction.guild.voice_client.resume()
|
|
await interaction.followup.send("Resumed the song.")
|
|
else:
|
|
await interaction.followup.send("The song is not paused.")
|
|
|
|
async def stop(self, interaction: discord.Interaction):
|
|
if interaction.guild.voice_client.is_playing():
|
|
interaction.guild.voice_client.stop()
|
|
self.queue = []
|
|
self.current_song = None
|
|
await interaction.followup.send("Stopped the song.")
|
|
else:
|
|
await interaction.followup.send("No song is currently playing.")
|
|
|
|
async def set_volume(self, interaction: discord.Interaction, volume: float):
|
|
self.volume = volume
|
|
if interaction.guild.voice_client and interaction.guild.voice_client.source:
|
|
interaction.guild.voice_client.source.volume = self.volume
|
|
await interaction.followup.send(f"Volume set to {volume * 100}%.")
|
|
else:
|
|
await interaction.followup.send("No audio source found.")
|
|
|
|
async def toggle_loop(self, interaction: discord.Interaction):
|
|
self.loop = not self.loop # Toggle the loop state
|
|
state = "enabled" if self.loop else "disabled"
|
|
await interaction.followup.send(f"Loop has been {state}.")
|
|
|
|
def setup(self, tree: discord.app_commands.CommandTree):
|
|
@tree.command(name="join", description="Join the voice channel")
|
|
async def join_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.join(interaction)
|
|
|
|
@tree.command(name="leave", description="Leave the voice channel")
|
|
async def leave_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.leave(interaction)
|
|
|
|
@tree.command(name="play", description="Play a song from YouTube")
|
|
async def play_command(interaction: discord.Interaction, search: str):
|
|
await interaction.response.defer()
|
|
await self.play(interaction, search)
|
|
|
|
@tree.command(name="pause", description="Pause the current song")
|
|
async def pause_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.pause(interaction)
|
|
|
|
@tree.command(name="resume", description="Resume the paused song")
|
|
async def resume_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.resume(interaction)
|
|
|
|
@tree.command(name="stop", description="Stop the current song")
|
|
async def stop_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.stop(interaction)
|
|
|
|
@tree.command(name="volume", description="Set the volume (0 to 1)")
|
|
async def volume_command(interaction: discord.Interaction, volume: float):
|
|
await interaction.response.defer()
|
|
await self.set_volume(interaction, volume)
|
|
|
|
@tree.command(name="loop", description="Toggle loop for the current song")
|
|
async def loop_command(interaction: discord.Interaction):
|
|
await interaction.response.defer()
|
|
await self.toggle_loop(interaction)
|
|
|
|
|
|
def setup(bot):
|
|
music = Music(bot)
|
|
music.setup(bot.tree)
|