grzegorz-clients/grzegorz_clients/remi_ui.py

404 lines
13 KiB
Python
Raw Normal View History

import random, os, time, shutil, sys
from threading import Timer
import remi.gui as gui
from remi import App
from .utils import Namespace, call_as_thread, seconds_to_timestamp
from . import api
from .constants import *
#globals:
2018-06-02 13:33:35 +02:00
WIDTH_L = 400
WIDTH_R = 512
WIDTH = WIDTH_L + 40 + WIDTH_R
class RemiApp(App):
def __init__(self, *args):
res_path = os.path.join(os.path.dirname(__file__), 'res')
2019-11-24 16:21:21 +01:00
self.old_playlist = None
super(RemiApp, self).__init__(*args, static_file_path=res_path)
2022-02-20 00:13:19 +01:00
def make_gui_elements(self): # content and behaviour
#logo:
self.logo_image = gui.Image('/res/logo.png')
2019-03-13 12:19:30 +01:00
self.logo_image.attributes["onclick"] = "document.location='https://www.youtube.com/watch?v=t-fcrn1Edik'"
2022-02-20 00:13:19 +01:00
#playback controls
self.playback = Namespace()
2022-02-20 00:13:19 +01:00
self.playback.playing = gui.Label("Now playing: None")# (TODO): update this
2022-02-20 00:13:19 +01:00
2019-03-09 20:02:01 +01:00
self.playback.party = gui.Button(ICON_PARTY)
self.playback.party.attributes["onclick"] = "document.body.classList.toggle('dancing');"
self.playback.party.attributes["title"] = "ENABLE PARTY MODE" # hover text
self.playback.previous = gui.Button(ICON_PREV)
self.playback.previous.set_on_click_listener(self.playback_previous)
self.playback.play = gui.Button(ICON_PLAY)
self.playback.play.set_on_click_listener(self.playback_play)
self.playback.next = gui.Button(ICON_NEXT)
self.playback.next.set_on_click_listener(self.playback_next)
2022-02-20 00:13:19 +01:00
2018-02-28 22:17:44 +01:00
self.playback.volume_label = gui.Label("Volume:")
self.playback.volume_slider = gui.Slider(100, 0, 100, 1)
2018-02-28 22:17:44 +01:00
self.playback.volume_slider.set_oninput_listener(self.change_volume)
2022-02-20 00:13:19 +01:00
self.playback.seek_slider = gui.Slider(0, 0, 100, 1)
2018-02-28 22:17:44 +01:00
self.playback.seek_slider.set_oninput_listener(self.change_seek)
2022-02-20 00:13:19 +01:00
self.playback.timestamp = gui.Label("--:-- - --:--")
2022-02-20 00:13:19 +01:00
#playlist
self.playlist = Namespace()
2022-02-20 00:13:19 +01:00
self.playlist.table = gui.Table()
self.playlist.table.append_from_list([['#', 'Name', "length", "", "", "", ""]], fill_title=True)
2022-02-20 00:13:19 +01:00
2019-03-13 11:37:49 +01:00
self.playlist.looping = gui.CheckBoxLabel("<i><small>loop playlist</small></i>")
self.playlist.looping.set_on_click_listener(self.on_playlist_set_looping)
self.playlist.shuffle = gui.Button("SHUFFLE")
self.playlist.shuffle.set_on_click_listener(self.on_playlist_clear_shuffle)
2022-02-20 00:13:19 +01:00
self.playlist.clear = gui.Button("CLEAR")
self.playlist.clear.set_on_click_listener(self.on_playlist_clear_click)
2022-02-20 00:13:19 +01:00
#input
self.input = Namespace()
self.input.add_song = gui.Label("Add song:")
self.input.field = gui.TextInput(single_line=True)
self.input.field.set_on_enter_listener(self.input_submit)
self.input.submit = gui.Button("Submit!")
self.input.submit.set_on_click_listener(self.input_submit)
2022-02-20 00:13:19 +01:00
def make_gui_container(self):#placement and styling
# Logo image:
2018-06-02 13:33:35 +02:00
self.logo_image.style["width"] = f"100%"
2019-03-09 20:02:01 +01:00
for i in (self.playback.previous, self.playback.play, self.playback.next, self.playback.party):
i.style["margin"] = "3px"
i.style["width"] = "2.8em"
2022-02-20 00:13:19 +01:00
# Playback:
2019-03-09 20:02:01 +01:00
self.playback.party.style["background"] \
= f"linear-gradient(40deg,{COLOR_PINK},{COLOR_TEAL})"
self.playback.play.style["background"] \
= f"linear-gradient(40deg,{COLOR_BLUE},{COLOR_PURPLE})"
2019-03-09 20:02:01 +01:00
2022-02-20 00:13:19 +01:00
self.playback.volume_label.style["font-size"] = "0.8em"
2022-02-20 00:13:19 +01:00
self.playback.volume_slider.style["width"] = "150px"
self.playback.volume_slider.style["margin-left"] = "20px"
self.playback.volume_slider.style["margin-bottom"] = "13px"
2022-02-20 00:13:19 +01:00
self.playback.seek_slider.set_size("90%", "20px")
self.playback.seek_slider.style["margin"] = "10px"
2022-02-20 00:13:19 +01:00
# Playlist:
self.playlist.table.style["width"] = "100%"
self.playlist.table.style["margin"] = "10px"
title_row = self.playlist.table.get_child("title")
title_row.get_child(title_row._render_children_list[1]) \
.style["width"] = "100%"
2022-02-20 00:13:19 +01:00
self.playlist.clear.style["background"] \
= f"linear-gradient(40deg,{COLOR_RED},{COLOR_ORANGE})"
self.playlist.shuffle.style["background"] \
= f"linear-gradient(40deg,{COLOR_TEAL},{COLOR_GREEN})"
for i in (self.playlist.shuffle, self.playlist.clear):
i.style["height"] = "1.8em"
i.style["font-size"] = "0.8em"
i.style["margin-left"] = "0.25em"
i.style["margin-right"] = "0.25em"
2022-02-20 00:13:19 +01:00
# Input field:
self.input.field.style["height"] = "20px"
self.input.field.style["margin"] = "5px"
self.input.field.style["border"] = "1px solid %s" % COLOR_BLUE
self.input.field.style["box-shadow"] = "0px 0px 5px 0px %s" % COLOR_BLUE_SHADOW
2018-06-02 13:33:35 +02:00
self.input.field.style["align-self"] = "flex-start"
2022-02-20 00:13:19 +01:00
2018-06-02 13:33:35 +02:00
self.input.submit.style["margin"] = "0px 5px"
self.input.submit.style["padding"] = "0px 10px"
2022-02-20 00:13:19 +01:00
# Containers:
2018-06-02 13:33:35 +02:00
root_container = gui.HBox()
root_container.style.update({"margin-left": "auto", "margin-right":"auto"})
root_container.style["margin-left"] = "auto"
root_container.style["margin-right"] = "auto"
root_container.style["max-width"] = f"{WIDTH}px"
root_container.style["flex-wrap"] = "wrap"
2022-02-20 00:13:19 +01:00
# LEFT SIDE
left_container = gui.VBox(width=WIDTH_L)
left_container.style["margin-top"] = "0.8em"
left_container.style["align-self"] = "flex-start"
root_container.append(left_container)
2022-02-20 00:13:19 +01:00
left_container.append(self.logo_image)
2022-02-20 00:13:19 +01:00
left_container.append(self.playback.playing)
2022-02-20 00:13:19 +01:00
playback_container = gui.HBox()
2019-03-09 20:02:01 +01:00
playback_container.append(self.playback.party)
playback_container.append(self.playback.previous)
playback_container.append(self.playback.play)
playback_container.append(self.playback.next)
2022-02-20 00:13:19 +01:00
2018-02-28 22:17:44 +01:00
volume_container = gui.VBox()
volume_container.append(self.playback.volume_label)
volume_container.append(self.playback.volume_slider)
playback_container.append(volume_container)
left_container.append(playback_container)
2022-02-20 00:13:19 +01:00
left_container.append(self.playback.seek_slider)
left_container.append(self.playback.timestamp)
2022-02-20 00:13:19 +01:00
# RIGHT SIDE
right_container = gui.VBox(width=WIDTH_R)
2018-06-02 13:33:35 +02:00
right_container.style["margin-top"] = "0.8em"
right_container.style["align-self"] = "flex-start"
root_container.append(right_container)
2022-02-20 00:13:19 +01:00
2018-06-02 13:33:35 +02:00
self.input.add_song.style["line-height"] = "1em"
right_container.append(self.input.add_song)
2022-02-20 00:13:19 +01:00
input_container = gui.HBox(width="100%")
2018-06-02 13:33:35 +02:00
input_container.style["margin"] = "0px"
input_container.append(self.input.field)
input_container.append(self.input.submit)
right_container.append(input_container)
2022-02-20 00:13:19 +01:00
right_container.append(self.playlist.table)
2022-02-20 00:13:19 +01:00
2019-03-13 11:37:49 +01:00
playlist_controls = gui.HBox()
playlist_controls.style["width"] = "100%"
playlist_controls.style["justify-content"] = "space-between"
playlist_controls.style["margin-right"] = "0.25em"
playlist_controls.style["margin-bottom"] = "0.8em"
playlist_controls.append(self.playlist.looping)
2018-03-06 14:51:04 +01:00
playlist_button_container = gui.HBox()
playlist_button_container.append(self.playlist.shuffle)
playlist_button_container.append(self.playlist.clear)
2019-03-13 11:37:49 +01:00
playlist_controls.append(playlist_button_container)
right_container.append(playlist_controls)
2022-02-20 00:13:19 +01:00
return root_container
2022-02-20 00:13:19 +01:00
def main(self):
self.make_gui_elements()
container = self.make_gui_container()
self.mainLoop()
return container
def mainLoop(self):
self.playback_update()
self.volume_update()
self.playlist_update()
2022-02-20 00:13:19 +01:00
2018-02-28 22:17:44 +01:00
Timer(1, self.mainLoop).start()
# events:
@call_as_thread
def playback_previous(self, widget):
api.playlist_previous()
2022-02-20 00:13:19 +01:00
@call_as_thread
def playback_play(self, widget):# toggle playblack
if api.is_playing():
api.set_playing(False)
2018-02-28 22:17:44 +01:00
self.set_playing(False)
else:
api.set_playing(True)
2018-02-28 22:17:44 +01:00
self.set_playing(True)
2022-02-20 00:13:19 +01:00
@call_as_thread
def playback_next(self, widget):
api.playlist_next()
2022-02-20 00:13:19 +01:00
@call_as_thread
def input_submit(self, widget, value=None):
if value is None:
value = self.input.field.get_text()
self.input.field.set_text("")
2022-02-20 00:13:19 +01:00
self.input.field.set_enabled(False)
self.input.submit.set_enabled(False)
try:
#data = get_youtube_metadata(value)
data = None
finally:
self.input.field.set_enabled(True)
self.input.submit.set_enabled(True)
2022-02-20 00:13:19 +01:00
api.load_path(value, data)
2022-02-20 00:13:19 +01:00
2018-02-28 22:17:44 +01:00
@call_as_thread
def change_seek(self, widget, value):
api.seek_percent(value)
2022-02-20 00:13:19 +01:00
2018-02-28 22:17:44 +01:00
@call_as_thread
def change_volume(self, widget, value):
api.set_volume(value)
2022-02-20 00:13:19 +01:00
2018-03-05 21:48:48 +01:00
@call_as_thread
def on_table_row_click(self, row_widget, playlist_item):
print("row", playlist_item)
2022-02-20 00:13:19 +01:00
@call_as_thread
def on_table_item_move_click(self, row_widget, playlist_item, down = True):
index = playlist_item["index"]
dest = index + 2 if down else index-1
api.playlist_move(index, dest)
2022-02-20 00:13:19 +01:00
@call_as_thread
def on_table_item_remove_click(self, row_widget, playlist_item):
api.playlist_remove(playlist_item["index"])
2022-02-20 00:13:19 +01:00
@call_as_thread
2022-02-20 00:13:19 +01:00
def on_table_item_play_item(self, row_widget, playlist_item):
api.playlist_goto(playlist_item["index"])
2022-02-20 00:13:19 +01:00
api.set_playing(True)
self.set_playing(True)
@call_as_thread
2019-03-13 11:37:49 +01:00
def on_playlist_set_looping(self, row_widget):
toggled = not row_widget.get_value()
api.playlist_set_looping(toggled)
row_widget.set_value(toggled)
2022-02-20 00:13:19 +01:00
2019-03-13 11:37:49 +01:00
@call_as_thread
2018-03-06 14:51:04 +01:00
def on_playlist_clear_shuffle(self, row_widget):
api.playlist_shuffle()
2022-02-20 00:13:19 +01:00
2018-03-06 14:51:04 +01:00
@call_as_thread
def on_playlist_clear_click(self, row_widget):
api.playlist_clear()
2022-02-20 00:13:19 +01:00
# gui updaters:
2022-02-20 00:13:19 +01:00
@call_as_thread
def playback_update(self):
2018-02-28 22:17:44 +01:00
is_playing = api.is_playing()
self.set_playing(is_playing)
2019-03-13 11:37:49 +01:00
self.playlist.looping.set_value(api.get_playlist_looping())
if is_playing: # update seekbar and timestamp
try:
playback_pos = api.get_playback_pos()
except api.APIError:
playback_pos = None
2022-02-20 00:13:19 +01:00
if playback_pos:
slider_pos = playback_pos["current"] / playback_pos["total"] * 100
current = seconds_to_timestamp(playback_pos["current"])
total = seconds_to_timestamp(playback_pos["total"])
2022-02-20 00:13:19 +01:00
if self.playback.seek_slider.get_value() != slider_pos:
self.playback.seek_slider.set_value(slider_pos)
2022-02-20 00:13:19 +01:00
self.playback.timestamp.set_text(f"{current} - {total}")
else:
self.playback.timestamp.set_text("--:-- - --:--")
2022-02-20 00:13:19 +01:00
@call_as_thread
def volume_update(self):
volume = api.get_volume()
if volume > 100:
volume = 100
2022-02-20 00:13:19 +01:00
if self.playback.volume_slider.get_value() != volume:
self.playback.volume_slider.set_value(volume)
2022-02-20 00:13:19 +01:00
@call_as_thread
def playlist_update(self):
playlist = api.get_playlist() # json structure
N = len(playlist)
2022-02-20 00:13:19 +01:00
# update playlist table content:
table = []
for playlist_item in playlist:
name = playlist_item["filename"]
length = None
2022-02-20 00:13:19 +01:00
if "data" in playlist_item:
name = playlist_item["data"].get("title", name)
length = playlist_item["data"].get("duration", length)
2022-02-20 00:13:19 +01:00
table.append([
playlist_item["index"],
name,
seconds_to_timestamp(length) if length else "--:--",
ICON_GOTO,
ICON_UP,
ICON_DOWN,
ICON_TRASH,
])
this_playlist = list(zip(table, [i.get("current", False) for i in playlist])) # ew, but it works...
2019-11-24 16:51:15 +01:00
if this_playlist == self.old_playlist: return
self.old_playlist = this_playlist
2019-11-24 16:21:21 +01:00
self.playlist.table.empty(keep_title=True)
self.playlist.table.append_from_list(table)
# styling the new table:
# for each row element:
for row_key, playlist_item in zip(self.playlist.table._render_children_list[1:], playlist):
row_widget = self.playlist.table.get_child(row_key)
row_widget.set_on_click_listener(self.on_table_row_click, playlist_item)
2022-02-20 00:13:19 +01:00
if playlist_item.get("current", False):
self.playback.previous.set_enabled(playlist_item.get("index") != 0)
self.playback.next.set_enabled(playlist_item.get("index") != N-1)
row_widget.style["background-color"] = COLOR_LIGHT_BLUE
else:
row_widget.style["color"] = COLOR_GRAY_DARK
2022-02-20 00:13:19 +01:00
# for each item element in this row:
for item_index, item_key in enumerate(row_widget._render_children_list):
item_widget = row_widget.get_child(item_key)
2022-02-20 00:13:19 +01:00
if item_index == 1 and "failed" in playlist_item.get("data", {}):
item_widget.style["width"] = "1.1em"
item_widget.style["color"] = COLOR_RED
2022-02-20 00:13:19 +01:00
if item_index >= 3:
item_widget.style["width"] = "1.1em"
2022-02-20 00:13:19 +01:00
if item_index == 3: # seek here
2022-02-20 00:13:19 +01:00
item_widget.style["color"] = COLOR_GREEN#COLOR_RED if playlist_item.get("current", False) else
item_widget.set_on_click_listener(self.on_table_item_play_item, playlist_item)
item_widget.attributes["title"] = "Play this item"
if item_index == 4: # move up
item_widget.set_on_click_listener(self.on_table_item_move_click, playlist_item, False)
if playlist_item["index"] == 0:
item_widget.style["color"] = COLOR_GRAY_LIGHT
else:
item_widget.style["color"] = COLOR_TEAL
item_widget.attributes["title"] = "Move this item up the playlist"
if item_index == 5: # move down
item_widget.set_on_click_listener(self.on_table_item_move_click, playlist_item, True)
if playlist_item["index"] == N-1:
item_widget.style["color"] = COLOR_GRAY_LIGHT
else:
item_widget.style["color"] = COLOR_TEAL
item_widget.attributes["title"] = "Move this item down the playlist"
2022-02-20 00:13:19 +01:00
if item_index == 6: # remove from playlist
item_widget.style["color"] = COLOR_RED
item_widget.set_on_click_listener(self.on_table_item_remove_click, playlist_item)
item_widget.attributes["title"] = "Remove this item from the playlist"
#print(index, key, item_widget)
2022-02-20 00:13:19 +01:00
def set_playing(self, is_playing:bool): # Only updates GUI elements!
self.playback.play.set_text(ICON_PAUSE if is_playing else ICON_PLAY)
2018-02-28 22:17:44 +01:00
self.playback.seek_slider.set_enabled(is_playing)