mirror of
https://github.com/RafaelSolVargas/Vulkan.git
synced 2025-10-29 16:57:23 +00:00
196 lines
6.0 KiB
Python
196 lines
6.0 KiB
Python
from collections import deque
|
|
from config import config
|
|
import random
|
|
|
|
from vulkan.music.Interfaces import IPlaylist
|
|
from vulkan.music.Song import Song
|
|
|
|
|
|
class Playlist(IPlaylist):
|
|
"""Class to manage and control the songs to play and played"""
|
|
|
|
def __init__(self) -> None:
|
|
self.__queue = deque() # Store the musics to play
|
|
self.__songs_history = deque() # Store the musics played
|
|
|
|
self.__looping_one = False
|
|
self.__looping_all = False
|
|
|
|
self.__current: Song = None
|
|
|
|
@property
|
|
def songs_history(self) -> deque:
|
|
return self.__songs_history
|
|
|
|
@property
|
|
def looping_one(self) -> bool:
|
|
return self.__looping_one
|
|
|
|
@property
|
|
def looping_all(self) -> bool:
|
|
return self.__looping_all
|
|
|
|
@property
|
|
def current(self) -> Song:
|
|
return self.__current
|
|
|
|
@property
|
|
def songs_to_preload(self) -> list:
|
|
return list(self.__queue)[:config.MAX_PRELOAD_SONGS]
|
|
|
|
def __len__(self) -> int:
|
|
return len(self.__queue)
|
|
|
|
def next_song(self) -> Song:
|
|
"""Return the next song to play in a normal playlist flow"""
|
|
if self.__current == None and len(self.__queue) == 0:
|
|
return None
|
|
|
|
played_song = self.__current
|
|
|
|
# Att played song info
|
|
if played_song != None:
|
|
if not self.__looping_one and not self.__looping_all:
|
|
if played_song.problematic == False:
|
|
self.__songs_history.appendleft(played_song)
|
|
|
|
if len(self.__songs_history) > config.MAX_SONGS_HISTORY:
|
|
self.__songs_history.pop() # Remove the older
|
|
|
|
elif self.__looping_one: # Insert the current song to play again
|
|
self.__queue.appendleft(played_song)
|
|
|
|
elif self.__looping_all: # Insert the current song in the end of queue
|
|
self.__queue.append(played_song)
|
|
|
|
# Get the new song
|
|
if len(self.__queue) == 0:
|
|
return None
|
|
|
|
self.__current = self.__queue.popleft()
|
|
|
|
return self.__current
|
|
|
|
def prev_song(self) -> Song:
|
|
"""If playing return it to queue and return the previous song to play"""
|
|
if len(self.__songs_history) == 0:
|
|
return None
|
|
else:
|
|
if self.__current != None:
|
|
self.__queue.appendleft(self.__current)
|
|
|
|
last_song = self.__songs_history.popleft() # Get the last song
|
|
self.__current = last_song
|
|
return self.__current # return the song
|
|
|
|
def add_song(self, identifier: str, requester: str) -> Song:
|
|
"""Create a song object, add to queue and return it"""
|
|
song = Song(identifier=identifier, playlist=self, requester=requester)
|
|
self.__queue.append(song)
|
|
return song
|
|
|
|
def shuffle(self) -> None:
|
|
"""Shuffle the order of the songs to play"""
|
|
random.shuffle(self.__queue)
|
|
|
|
def revert(self) -> None:
|
|
"""Revert the order of the songs to play"""
|
|
self.__queue.reverse()
|
|
|
|
def clear(self) -> None:
|
|
"""Clear the songs to play song history"""
|
|
self.__queue.clear()
|
|
|
|
def loop_one(self) -> str:
|
|
"""Try to start the loop of the current song
|
|
|
|
Return: Embed descrition to show to user
|
|
"""
|
|
if self.__looping_all == True:
|
|
return config.LOOP_ALL_ON
|
|
|
|
elif self.__looping_one == True:
|
|
return config.LOOP_ONE_ALREADY_ON
|
|
else:
|
|
self.__looping_one = True
|
|
return config.LOOP_ONE_ACTIVATE
|
|
|
|
def loop_all(self) -> str:
|
|
"""Try to start the loop of all songs
|
|
|
|
Return: Embed descrition to show to user
|
|
"""
|
|
if self.__looping_one == True:
|
|
return config.LOOP_ONE_ON
|
|
|
|
elif self.__looping_all == True:
|
|
return config.LOOP_ALL_ALREADY_ON
|
|
|
|
else:
|
|
self.__looping_all = True
|
|
return config.LOOP_ALL_ACTIVATE
|
|
|
|
def loop_off(self) -> str:
|
|
"""Disable both types of loop"""
|
|
if self.__looping_all == False and self.__looping_one == False:
|
|
return config.LOOP_ALREADY_DISABLE
|
|
|
|
self.__looping_all = False
|
|
self.__looping_one = False
|
|
return config.LOOP_DISABLE
|
|
|
|
def destroy_song(self, song_destroy: Song) -> None:
|
|
"""Destroy a song object from the queue"""
|
|
for song in self.__queue:
|
|
if song == song_destroy:
|
|
self.__queue.remove(song)
|
|
break
|
|
|
|
def move_songs(self, pos1, pos2) -> str:
|
|
"""Receive two position and try to change the songs in those positions, -1 is the last
|
|
|
|
Positions: First music is 1
|
|
Return (Error bool, string) with the status of the function, to show to user
|
|
"""
|
|
if pos1 == -1:
|
|
pos1 = len(self.__queue)
|
|
if pos2 == -1:
|
|
pos2 = len(self.__queue)
|
|
|
|
if pos2 not in range(1, len(self.__queue) + 1) or pos1 not in range(1, len(self.__queue) + 1):
|
|
return config.LENGTH_ERROR
|
|
|
|
try:
|
|
song1 = self.__queue[pos1-1]
|
|
song2 = self.__queue[pos2-1]
|
|
|
|
self.__queue[pos1-1] = song2
|
|
self.__queue[pos2-1] = song1
|
|
|
|
song1_name = song1.title if song1.title else song1.identifier
|
|
song2_name = song2.title if song2.title else song2.identifier
|
|
|
|
return config.SONG_MOVED_SUCCESSFULLY.format(song1_name, pos1, song2_name, pos2)
|
|
except:
|
|
return config.ERROR_MOVING
|
|
|
|
def remove_song(self, position) -> str:
|
|
if position not in range(1, len(self.__queue) + 1) and position != -1:
|
|
return config.LENGTH_ERROR
|
|
else:
|
|
song = self.__queue[position-1]
|
|
self.__queue.remove(song)
|
|
|
|
song_name = song.title if song.title else song.identifier
|
|
|
|
return config.SONG_REMOVED_SUCCESSFULLY.format(song_name)
|
|
|
|
def history(self) -> list:
|
|
"""Return a list with the song title of all played songs"""
|
|
titles = []
|
|
for song in self.__songs_history:
|
|
title = song.title if song.title else 'Unknown'
|
|
titles.append(title)
|
|
|
|
return titles
|