0
1
mirror of https://github.com/radio95-rnt/RadioPlayer.git synced 2026-02-26 13:52:00 +01:00
Files
RadioPlayer/radioPlaylist.py
2025-06-22 12:23:29 +02:00

557 lines
21 KiB
Python

#!/usr/bin/env python3
import os
import sys
import termios
import tty
import signal
import shutil
import libcache
from datetime import datetime
files_dir = "/home/user/mixes/"
playlists_dir = "/home/user/playlists/"
formats = ('.mp3', '.m4a')
p = ("Polskie", "Dzem")
def get_audio_files(directory) -> list[str]:
audio_files = []
try:
for file in os.listdir(directory):
file: str
if file.lower().endswith(formats):
audio_files.append(file)
return sorted(audio_files)
except FileNotFoundError:
print(f"Error: Directory '{directory}' not found.")
return []
except PermissionError:
print(f"Error: Permission denied for directory '{directory}'.")
return []
def get_char() -> str:
fd = sys.stdin.fileno()
old_settings = termios.tcgetattr(fd)
try:
tty.setraw(sys.stdin.fileno())
ch = sys.stdin.read(1)
finally:
termios.tcsetattr(fd, termios.TCSADRAIN, old_settings)
return ch
def clear_screen() -> None:
print("\033[2J\033[H", end="", flush=True)
def move_cursor(row, col=1):
print(f"\033[{row};{col}H", end="", flush=True)
def clear_line():
print("\033[2K", end="", flush=True)
def hide_cursor():
print("\033[?25l", end="", flush=True)
def show_cursor():
print("\033[?25h", end="", flush=True)
def get_terminal_size() -> os.terminal_size:
return shutil.get_terminal_size()
def get_days_of_week() -> list[str]:
days = ["monday", "tuesday", "wednesday", "thursday", "friday", "saturday", "sunday"]
today = datetime.now().weekday()
return days[today:] + days[:today]
def ensure_playlist_dir(day: str) -> str:
playlist_dir = os.path.expanduser(os.path.join(playlists_dir, day))
if not os.path.exists(playlist_dir):
os.makedirs(playlist_dir)
return playlist_dir
def calculate_category_percentages(playlists, current_day):
category_counts = {'late_night': 0, 'morning': 0, 'day': 0, 'night': 0}
polskie_counts = {'late_night': 0, 'morning': 0, 'day': 0, 'night': 0}
for category in category_counts.keys():
for file in playlists[current_day][category]:
category_counts[category] += 1
if any(element in file for element in p):
polskie_counts[category] += 1
total_count = sum(category_counts.values())
if total_count == 0:
return None
percentages = {
category: (count / total_count) * 100 for category, count in category_counts.items()
}
polskie_percentages = {
category: (polskie_counts[category] / category_counts[category]) * 100 if category_counts[category] > 0 else 0
for category in category_counts
}
return percentages, polskie_percentages, (sum(polskie_counts.values())/sum(category_counts.values()))*100
def update_playlist_file(day: str, period: str, filepath: str, add: bool):
playlist_dir = ensure_playlist_dir(day)
playlist_file = os.path.join(playlist_dir, period)
full_filepath = os.path.join(files_dir, filepath)
if not os.path.exists(playlist_file):
with open(playlist_file, 'w') as f:
pass
with open(playlist_file, 'r') as f:
lines = f.read().splitlines()
if add and full_filepath not in lines:
lines.append(full_filepath)
elif not add and full_filepath in lines:
lines.remove(full_filepath)
with open(playlist_file, 'w') as f:
f.write('\n'.join(lines) + ('\n' if lines else ''))
def load_playlists(days: list[str]):
playlists = {}
for day in days:
playlists[day] = {'late_night': set(), 'morning': set(), 'day': set(), 'night': set()}
playlist_dir = os.path.expanduser(os.path.join(playlists_dir, day))
if os.path.exists(playlist_dir):
for period in ['late_night', 'morning', 'day', 'night']:
playlist_file = os.path.join(playlist_dir, period)
if os.path.exists(playlist_file):
with open(playlist_file, 'r') as f:
for line in f:
line = line.strip()
if line:
filename = os.path.basename(line)
playlists[day][period].add(filename)
return playlists
def copy_day_to_all(playlists: dict, source_day: str, days: list[str]):
periods = ['late_night', 'morning', 'day', 'night']
for target_day in days:
if target_day == source_day:
continue
for period in periods:
target_dir = ensure_playlist_dir(target_day)
target_file = os.path.join(target_dir, period)
filepaths = []
for filename in playlists[source_day][period]:
full_path = os.path.join(files_dir, filename)
filepaths.append(full_path)
with open(target_file, 'w') as f:
f.write('\n'.join(filepaths) + ('\n' if filepaths else ''))
playlists[target_day][period] = set(playlists[source_day][period])
return playlists
def copy_current_file_to_all(playlists: dict, source_day: str, days: list[str], current_file: str):
source_periods = {
'late_night': current_file in playlists[source_day]['late_night'],
'morning': current_file in playlists[source_day]['morning'],
'day': current_file in playlists[source_day]['day'],
'night': current_file in playlists[source_day]['night'],
}
if not any(source_periods.values()):
return playlists, False
for target_day in days:
if target_day == source_day:
continue
for period, is_present in source_periods.items():
target_set = playlists[target_day][period]
full_path = os.path.join(files_dir, current_file)
if is_present:
target_set.add(current_file)
else:
target_set.discard(current_file)
playlist_dir = ensure_playlist_dir(target_day)
playlist_file = os.path.join(playlist_dir, period)
if os.path.exists(playlist_file):
with open(playlist_file, 'r') as f:
lines = [line.strip() for line in f.readlines()]
else:
lines = []
if is_present and full_path not in lines:
lines.append(full_path)
elif not is_present:
while full_path in lines:
lines.remove(full_path)
with open(playlist_file, 'w') as f:
f.write('\n'.join(lines) + ('\n' if lines else ''))
return playlists, True
class InterfaceState:
def __init__(self):
self.last_header = None
self.last_files_display = None
self.last_selected_idx = -1
self.last_current_day_idx = -1
self.last_scroll_offset = -1
self.last_message = None
def draw_header(playlists: dict, current_day: str, current_day_idx: int, days: list[str], term_width: int, force_redraw=False, state=None):
percentages, polskie_percentages, total_pl = calculate_category_percentages(playlists, current_day) or ({}, {}, 0)
category_bar = ""
for category in ['late_night', 'morning', 'day', 'night']:
percent = percentages.get(category, 0)
polskie_percent = polskie_percentages.get(category, 0)
category_bar += f"{category[:4].capitalize()}: {percent:.1f}% (P:{polskie_percent:.1f}%) | "
category_bar += f"TP:{total_pl:0.1f}% | "
assigned_files = set()
categories = ['late_night', 'morning', 'day', 'night']
for day in get_days_of_week():
for category in categories:
assigned_files.update(playlists[day][category])
total_files = len(get_audio_files(files_dir))
assigned_count = len(assigned_files)
unassigned = ((total_files - assigned_count) / total_files) * 100 if total_files > 0 else 0
category_bar += f"UA:{unassigned:0.1f}%"
if len(category_bar) > term_width - 2:
category_bar = category_bar[:term_width - 5] + "..."
day_bar = ""
for i, day in enumerate(days):
if i == current_day_idx:
day_bar += f"\033[1;44m[{day}]\033[0m "
else:
day_bar += f"[{day}] "
header_content = (category_bar, day_bar.strip())
if force_redraw or (state and state.last_header != header_content):
move_cursor(1)
clear_line()
print("\033[1;37mCategory Distribution:\033[0m".center(term_width), end="", flush=True)
move_cursor(2)
clear_line()
print(category_bar.center(term_width), end="", flush=True)
move_cursor(3)
clear_line()
print(day_bar.strip(), end="", flush=True)
if state:
state.last_header = header_content
def draw_files_section(audio_files: list, playlists: dict, selected_idx: int, current_day: str,
scroll_offset: int, term_width: int, term_height: int, force_redraw=False, state=None):
available_lines = term_height - 6
start_idx = max(0, min(scroll_offset, len(audio_files) - available_lines))
end_idx = min(start_idx + available_lines, len(audio_files))
files_display_state = (start_idx, end_idx, selected_idx, current_day)
if force_redraw or (state and (state.last_files_display != files_display_state or
state.last_selected_idx != selected_idx or
state.last_current_day_idx != current_day)):
# Update position info line
move_cursor(5)
clear_line()
if start_idx > 0:
print("", end="")
else:
print(" ", end="")
position_info = f" {current_day.capitalize()} | File {selected_idx + 1}/{len(audio_files)} "
padding = term_width - len(position_info) - 2
print(position_info.center(padding), end="")
if end_idx < len(audio_files):
print("", end="", flush=True)
else:
print(" ", end="", flush=True)
# Update file list
for display_row, idx in enumerate(range(start_idx, end_idx)):
file = audio_files[idx]
line_row = 7 + display_row # Start after header lines
in_late_night = file in playlists[current_day]['late_night']
in_morning = file in playlists[current_day]['morning']
in_day = file in playlists[current_day]['day']
in_night = file in playlists[current_day]['night']
l_color = "\033[1;32m" if in_late_night else "\033[1;30m"
m_color = "\033[1;32m" if in_morning else "\033[1;30m"
d_color = "\033[1;32m" if in_day else "\033[1;30m"
n_color = "\033[1;32m" if in_night else "\033[1;30m"
if idx == selected_idx:
row_highlight = "\033[1;44m"
else:
row_highlight = ""
max_filename_length = term_width - 15
display_file = file
if len(file) > max_filename_length:
display_file = file[:max_filename_length-3] + "..."
move_cursor(line_row)
clear_line()
print(f"{row_highlight}[{l_color}L\033[0m{row_highlight}] [{m_color}M\033[0m{row_highlight}] [{d_color}D\033[0m{row_highlight}] [{n_color}N\033[0m{row_highlight}] {display_file}\033[0m", end="", flush=True)
# Clear any remaining lines if we're showing fewer files
for clear_row in range(7 + (end_idx - start_idx), term_height):
move_cursor(clear_row)
clear_line()
if state:
state.last_files_display = files_display_state
state.last_selected_idx = selected_idx
def draw_interface(audio_files: list, playlists: dict, selected_idx: int, current_day_idx: int,
scroll_offset: int, terminal_size_cache: libcache.Cache, message=None,
force_redraw=False, state=None):
term_width, term_height = terminal_size_cache.getElement("width", False), terminal_size_cache.getElement("height", False)
if term_width is None or term_height is None:
term_width, term_height = get_terminal_size()
terminal_size_cache.saveElement("width", term_width, 5, False, True)
terminal_size_cache.saveElement("height", term_height, 5, False, True)
force_redraw = True
days = get_days_of_week()
current_day = days[current_day_idx]
if force_redraw:
clear_screen()
hide_cursor()
# Draw static elements
move_cursor(4)
print("UP/DOWN: Navigate | D/N/L: Toggle | C: Copy day to all | F: Copy file to all | Q: Quit", end="", flush=True)
# Draw header (only if changed)
draw_header(playlists, current_day, current_day_idx, days, term_width, force_redraw, state)
# Draw files section (only if changed)
draw_files_section(audio_files, playlists, selected_idx, current_day, scroll_offset,
term_width, term_height, force_redraw, state)
# Handle message display
if message != (state.last_message if state else None):
move_cursor(6)
clear_line()
if message:
print(f"\033[1;32m{message}\033[0m", end="", flush=True)
if state:
state.last_message = message
def main():
audio_files = get_audio_files(files_dir)
if not audio_files:
print("No audio files found. Exiting.")
return 1
days_of_week = get_days_of_week()
playlists = load_playlists(days_of_week)
selected_idx = 0
current_day_idx = 0
scroll_offset = 0
flash_message = None
message_timer = 0
def signal_handler(sig, frame):
show_cursor()
clear_screen()
sys.exit(0)
signal.signal(signal.SIGINT, signal_handler)
terminal_size_cache = libcache.Cache()
interface_state = InterfaceState()
# Initial full draw
draw_interface(audio_files, playlists, selected_idx, current_day_idx, scroll_offset,
terminal_size_cache, flash_message, force_redraw=True, state=interface_state)
try:
while True:
term_width, term_height = terminal_size_cache.getElement("width", False), terminal_size_cache.getElement("height", False)
if term_width is None or term_height is None:
term_width, term_height = get_terminal_size()
terminal_size_cache.saveElement("width", term_width, 5, False, True)
terminal_size_cache.saveElement("height", term_height, 5, False, True)
visible_lines = term_height - 6
if selected_idx < scroll_offset:
scroll_offset = selected_idx
elif selected_idx >= scroll_offset + visible_lines:
scroll_offset = selected_idx - visible_lines + 1
# Only redraw if something changed
needs_redraw = (interface_state.last_selected_idx != selected_idx or
interface_state.last_current_day_idx != current_day_idx or
interface_state.last_scroll_offset != scroll_offset or
flash_message != interface_state.last_message)
if needs_redraw:
draw_interface(audio_files, playlists, selected_idx, current_day_idx, scroll_offset,
terminal_size_cache, flash_message, state=interface_state)
interface_state.last_current_day_idx = current_day_idx
interface_state.last_scroll_offset = scroll_offset
if flash_message:
message_timer += 1
if message_timer > 1:
flash_message = None
message_timer = 0
key = get_char()
if key == 'q':
break
elif key == '\x1b':
next_key = get_char()
if next_key == '[':
arrow_key = get_char()
if arrow_key == 'A':
selected_idx = max(0, selected_idx - 1)
elif arrow_key == 'B':
selected_idx = min(len(audio_files) - 1, selected_idx + 1)
elif arrow_key == 'C':
current_day_idx = (current_day_idx + 1) % len(days_of_week)
elif arrow_key == 'D':
current_day_idx = (current_day_idx - 1) % len(days_of_week)
elif arrow_key == '5':
try:
next_key = get_char()
except:
pass
selected_idx = max(0, selected_idx - visible_lines)
elif arrow_key == '6':
try:
next_key = get_char()
except:
pass
selected_idx = min(len(audio_files) - 1, selected_idx + visible_lines)
elif arrow_key == '1':
if get_char() == '~':
selected_idx = 0
elif arrow_key == '4':
if get_char() == '~':
selected_idx = len(audio_files) - 1
elif key == ' ':
selected_idx = min(len(audio_files) - 1, selected_idx + visible_lines)
elif key.lower() == 'm':
current_day = days_of_week[current_day_idx]
file = audio_files[selected_idx]
is_in_playlist = file in playlists[current_day]['morning']
if is_in_playlist:
playlists[current_day]['morning'].remove(file)
else:
playlists[current_day]['morning'].add(file)
update_playlist_file(current_day, 'morning', file, not is_in_playlist)
elif key.lower() == 'd':
current_day = days_of_week[current_day_idx]
file = audio_files[selected_idx]
is_in_playlist = file in playlists[current_day]['day']
if is_in_playlist:
playlists[current_day]['day'].remove(file)
else:
playlists[current_day]['day'].add(file)
update_playlist_file(current_day, 'day', file, not is_in_playlist)
elif key.lower() == 'n':
current_day = days_of_week[current_day_idx]
file = audio_files[selected_idx]
is_in_playlist = file in playlists[current_day]['night']
if is_in_playlist:
playlists[current_day]['night'].remove(file)
else:
playlists[current_day]['night'].add(file)
update_playlist_file(current_day, 'night', file, not is_in_playlist)
elif key.lower() == 'l':
current_day = days_of_week[current_day_idx]
file = audio_files[selected_idx]
is_in_playlist = file in playlists[current_day]['late_night']
if is_in_playlist:
playlists[current_day]['late_night'].remove(file)
else:
playlists[current_day]['late_night'].add(file)
update_playlist_file(current_day, 'late_night', file, not is_in_playlist)
elif key.lower() == 'c':
current_day = days_of_week[current_day_idx]
playlists = copy_day_to_all(playlists, current_day, days_of_week)
flash_message = f"Playlists from {current_day} copied to all other days!"
message_timer = 0
elif key.lower() == 'f':
current_day = days_of_week[current_day_idx]
current_file = audio_files[selected_idx]
playlists, success = copy_current_file_to_all(playlists, current_day, days_of_week, current_file)
if success:
flash_message = f"File '{current_file}' copied to all days!"
else:
flash_message = f"File not in any playlist! Add it first."
message_timer = 0
elif key.isupper() and len(key) == 1 and key.isalpha():
target_letter = key.lower()
found_idx = -1
for i in range(selected_idx + 1, len(audio_files)):
if audio_files[i].lower().startswith(target_letter):
found_idx = i
break
if found_idx == -1:
for i in range(0, selected_idx):
if audio_files[i].lower().startswith(target_letter):
found_idx = i
break
if found_idx != -1:
selected_idx = found_idx
finally:
show_cursor()
clear_screen()
return 0
if __name__ == "__main__":
exit(main())