Whitespace

This commit is contained in:
Peder Bergebakken Sundt 2022-02-20 00:13:19 +01:00
parent 5e848cb16f
commit b4bf1d335a
1 changed files with 76 additions and 58 deletions

View File

@ -16,17 +16,17 @@ class RemiApp(App):
res_path = os.path.join(os.path.dirname(__file__), 'res') res_path = os.path.join(os.path.dirname(__file__), 'res')
self.old_playlist = None self.old_playlist = None
super(RemiApp, self).__init__(*args, static_file_path=res_path) super(RemiApp, self).__init__(*args, static_file_path=res_path)
def make_gui_elements(self):#content and behaviour def make_gui_elements(self): # content and behaviour
#logo: #logo:
self.logo_image = gui.Image('/res/logo.png') self.logo_image = gui.Image('/res/logo.png')
self.logo_image.attributes["onclick"] = "document.location='https://www.youtube.com/watch?v=t-fcrn1Edik'" self.logo_image.attributes["onclick"] = "document.location='https://www.youtube.com/watch?v=t-fcrn1Edik'"
#playback controls #playback controls
self.playback = Namespace() self.playback = Namespace()
self.playback.playing = gui.Label("Now playing: None")# (TODO): update this self.playback.playing = gui.Label("Now playing: None")# (TODO): update this
self.playback.party = gui.Button(ICON_PARTY) self.playback.party = gui.Button(ICON_PARTY)
self.playback.party.attributes["onclick"] = "document.body.classList.toggle('dancing');" self.playback.party.attributes["onclick"] = "document.body.classList.toggle('dancing');"
self.playback.party.attributes["title"] = "ENABLE PARTY MODE" # hover text self.playback.party.attributes["title"] = "ENABLE PARTY MODE" # hover text
@ -36,31 +36,31 @@ class RemiApp(App):
self.playback.play.set_on_click_listener(self.playback_play) self.playback.play.set_on_click_listener(self.playback_play)
self.playback.next = gui.Button(ICON_NEXT) self.playback.next = gui.Button(ICON_NEXT)
self.playback.next.set_on_click_listener(self.playback_next) self.playback.next.set_on_click_listener(self.playback_next)
self.playback.volume_label = gui.Label("Volume:") self.playback.volume_label = gui.Label("Volume:")
self.playback.volume_slider = gui.Slider(100, 0, 100, 1) self.playback.volume_slider = gui.Slider(100, 0, 100, 1)
self.playback.volume_slider.set_oninput_listener(self.change_volume) self.playback.volume_slider.set_oninput_listener(self.change_volume)
self.playback.seek_slider = gui.Slider(0, 0, 100, 1) self.playback.seek_slider = gui.Slider(0, 0, 100, 1)
self.playback.seek_slider.set_oninput_listener(self.change_seek) self.playback.seek_slider.set_oninput_listener(self.change_seek)
self.playback.timestamp = gui.Label("--:-- - --:--") self.playback.timestamp = gui.Label("--:-- - --:--")
#playlist #playlist
self.playlist = Namespace() self.playlist = Namespace()
self.playlist.table = gui.Table() self.playlist.table = gui.Table()
self.playlist.table.append_from_list([['#', 'Name', "length", "", "", "", ""]], fill_title=True) self.playlist.table.append_from_list([['#', 'Name', "length", "", "", "", ""]], fill_title=True)
self.playlist.looping = gui.CheckBoxLabel("<i><small>loop playlist</small></i>") 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.looping.set_on_click_listener(self.on_playlist_set_looping)
self.playlist.shuffle = gui.Button("SHUFFLE") self.playlist.shuffle = gui.Button("SHUFFLE")
self.playlist.shuffle.set_on_click_listener(self.on_playlist_clear_shuffle) self.playlist.shuffle.set_on_click_listener(self.on_playlist_clear_shuffle)
self.playlist.clear = gui.Button("CLEAR") self.playlist.clear = gui.Button("CLEAR")
self.playlist.clear.set_on_click_listener(self.on_playlist_clear_click) self.playlist.clear.set_on_click_listener(self.on_playlist_clear_click)
#input #input
self.input = Namespace() self.input = Namespace()
self.input.add_song = gui.Label("Add song:") self.input.add_song = gui.Label("Add song:")
@ -68,14 +68,14 @@ class RemiApp(App):
self.input.field.set_on_enter_listener(self.input_submit) self.input.field.set_on_enter_listener(self.input_submit)
self.input.submit = gui.Button("Submit!") self.input.submit = gui.Button("Submit!")
self.input.submit.set_on_click_listener(self.input_submit) self.input.submit.set_on_click_listener(self.input_submit)
def make_gui_container(self):#placement and styling def make_gui_container(self):#placement and styling
# Logo image: # Logo image:
self.logo_image.style["width"] = f"100%" self.logo_image.style["width"] = f"100%"
for i in (self.playback.previous, self.playback.play, self.playback.next, self.playback.party): for i in (self.playback.previous, self.playback.play, self.playback.next, self.playback.party):
i.style["margin"] = "3px" i.style["margin"] = "3px"
i.style["width"] = "2.8em" i.style["width"] = "2.8em"
# Playback: # Playback:
self.playback.party.style["background"] \ self.playback.party.style["background"] \
= f"linear-gradient(40deg,{COLOR_PINK},{COLOR_TEAL})" = f"linear-gradient(40deg,{COLOR_PINK},{COLOR_TEAL})"
@ -83,23 +83,23 @@ class RemiApp(App):
self.playback.play.style["background"] \ self.playback.play.style["background"] \
= f"linear-gradient(40deg,{COLOR_BLUE},{COLOR_PURPLE})" = f"linear-gradient(40deg,{COLOR_BLUE},{COLOR_PURPLE})"
self.playback.volume_label.style["font-size"] = "0.8em" self.playback.volume_label.style["font-size"] = "0.8em"
self.playback.volume_slider.style["width"] = "150px" self.playback.volume_slider.style["width"] = "150px"
self.playback.volume_slider.style["margin-left"] = "20px" self.playback.volume_slider.style["margin-left"] = "20px"
self.playback.volume_slider.style["margin-bottom"] = "13px" self.playback.volume_slider.style["margin-bottom"] = "13px"
self.playback.seek_slider.set_size("90%", "20px") self.playback.seek_slider.set_size("90%", "20px")
self.playback.seek_slider.style["margin"] = "10px" self.playback.seek_slider.style["margin"] = "10px"
# Playlist: # Playlist:
self.playlist.table.style["width"] = "100%" self.playlist.table.style["width"] = "100%"
self.playlist.table.style["margin"] = "10px" self.playlist.table.style["margin"] = "10px"
title_row = self.playlist.table.get_child("title") title_row = self.playlist.table.get_child("title")
title_row.get_child(title_row._render_children_list[1]) \ title_row.get_child(title_row._render_children_list[1]) \
.style["width"] = "100%" .style["width"] = "100%"
self.playlist.clear.style["background"] \ self.playlist.clear.style["background"] \
= f"linear-gradient(40deg,{COLOR_RED},{COLOR_ORANGE})" = f"linear-gradient(40deg,{COLOR_RED},{COLOR_ORANGE})"
self.playlist.shuffle.style["background"] \ self.playlist.shuffle.style["background"] \
@ -110,17 +110,17 @@ class RemiApp(App):
i.style["font-size"] = "0.8em" i.style["font-size"] = "0.8em"
i.style["margin-left"] = "0.25em" i.style["margin-left"] = "0.25em"
i.style["margin-right"] = "0.25em" i.style["margin-right"] = "0.25em"
# Input field: # Input field:
self.input.field.style["height"] = "20px" self.input.field.style["height"] = "20px"
self.input.field.style["margin"] = "5px" self.input.field.style["margin"] = "5px"
self.input.field.style["border"] = "1px solid %s" % COLOR_BLUE self.input.field.style["border"] = "1px solid %s" % COLOR_BLUE
self.input.field.style["box-shadow"] = "0px 0px 5px 0px %s" % COLOR_BLUE_SHADOW self.input.field.style["box-shadow"] = "0px 0px 5px 0px %s" % COLOR_BLUE_SHADOW
self.input.field.style["align-self"] = "flex-start" self.input.field.style["align-self"] = "flex-start"
self.input.submit.style["margin"] = "0px 5px" self.input.submit.style["margin"] = "0px 5px"
self.input.submit.style["padding"] = "0px 10px" self.input.submit.style["padding"] = "0px 10px"
# Containers: # Containers:
root_container = gui.HBox() root_container = gui.HBox()
root_container.style.update({"margin-left": "auto", "margin-right":"auto"}) root_container.style.update({"margin-left": "auto", "margin-right":"auto"})
@ -128,50 +128,50 @@ class RemiApp(App):
root_container.style["margin-right"] = "auto" root_container.style["margin-right"] = "auto"
root_container.style["max-width"] = f"{WIDTH}px" root_container.style["max-width"] = f"{WIDTH}px"
root_container.style["flex-wrap"] = "wrap" root_container.style["flex-wrap"] = "wrap"
# LEFT SIDE # LEFT SIDE
left_container = gui.VBox(width=WIDTH_L) left_container = gui.VBox(width=WIDTH_L)
left_container.style["margin-top"] = "0.8em" left_container.style["margin-top"] = "0.8em"
left_container.style["align-self"] = "flex-start" left_container.style["align-self"] = "flex-start"
root_container.append(left_container) root_container.append(left_container)
left_container.append(self.logo_image) left_container.append(self.logo_image)
left_container.append(self.playback.playing) left_container.append(self.playback.playing)
playback_container = gui.HBox() playback_container = gui.HBox()
playback_container.append(self.playback.party) playback_container.append(self.playback.party)
playback_container.append(self.playback.previous) playback_container.append(self.playback.previous)
playback_container.append(self.playback.play) playback_container.append(self.playback.play)
playback_container.append(self.playback.next) playback_container.append(self.playback.next)
volume_container = gui.VBox() volume_container = gui.VBox()
volume_container.append(self.playback.volume_label) volume_container.append(self.playback.volume_label)
volume_container.append(self.playback.volume_slider) volume_container.append(self.playback.volume_slider)
playback_container.append(volume_container) playback_container.append(volume_container)
left_container.append(playback_container) left_container.append(playback_container)
left_container.append(self.playback.seek_slider) left_container.append(self.playback.seek_slider)
left_container.append(self.playback.timestamp) left_container.append(self.playback.timestamp)
# RIGHT SIDE # RIGHT SIDE
right_container = gui.VBox(width=WIDTH_R) right_container = gui.VBox(width=WIDTH_R)
right_container.style["margin-top"] = "0.8em" right_container.style["margin-top"] = "0.8em"
right_container.style["align-self"] = "flex-start" right_container.style["align-self"] = "flex-start"
root_container.append(right_container) root_container.append(right_container)
self.input.add_song.style["line-height"] = "1em" self.input.add_song.style["line-height"] = "1em"
right_container.append(self.input.add_song) right_container.append(self.input.add_song)
input_container = gui.HBox(width="100%") input_container = gui.HBox(width="100%")
input_container.style["margin"] = "0px" input_container.style["margin"] = "0px"
input_container.append(self.input.field) input_container.append(self.input.field)
input_container.append(self.input.submit) input_container.append(self.input.submit)
right_container.append(input_container) right_container.append(input_container)
right_container.append(self.playlist.table) right_container.append(self.playlist.table)
playlist_controls = gui.HBox() playlist_controls = gui.HBox()
playlist_controls.style["width"] = "100%" playlist_controls.style["width"] = "100%"
playlist_controls.style["justify-content"] = "space-between" playlist_controls.style["justify-content"] = "space-between"
@ -186,9 +186,9 @@ class RemiApp(App):
playlist_controls.append(playlist_button_container) playlist_controls.append(playlist_button_container)
right_container.append(playlist_controls) right_container.append(playlist_controls)
return root_container return root_container
def main(self): def main(self):
self.make_gui_elements() self.make_gui_elements()
container = self.make_gui_container() container = self.make_gui_container()
@ -198,13 +198,14 @@ class RemiApp(App):
self.playback_update() self.playback_update()
self.volume_update() self.volume_update()
self.playlist_update() self.playlist_update()
Timer(1, self.mainLoop).start() Timer(1, self.mainLoop).start()
# events: # events:
@call_as_thread @call_as_thread
def playback_previous(self, widget): def playback_previous(self, widget):
api.playlist_previous() api.playlist_previous()
@call_as_thread @call_as_thread
def playback_play(self, widget):# toggle playblack def playback_play(self, widget):# toggle playblack
if api.is_playing(): if api.is_playing():
@ -213,15 +214,17 @@ class RemiApp(App):
else: else:
api.set_playing(True) api.set_playing(True)
self.set_playing(True) self.set_playing(True)
@call_as_thread @call_as_thread
def playback_next(self, widget): def playback_next(self, widget):
api.playlist_next() api.playlist_next()
@call_as_thread @call_as_thread
def input_submit(self, widget, value=None): def input_submit(self, widget, value=None):
if value is None: if value is None:
value = self.input.field.get_text() value = self.input.field.get_text()
self.input.field.set_text("") self.input.field.set_text("")
self.input.field.set_enabled(False) self.input.field.set_enabled(False)
self.input.submit.set_enabled(False) self.input.submit.set_enabled(False)
try: try:
@ -230,41 +233,53 @@ class RemiApp(App):
finally: finally:
self.input.field.set_enabled(True) self.input.field.set_enabled(True)
self.input.submit.set_enabled(True) self.input.submit.set_enabled(True)
api.load_path(value, data) api.load_path(value, data)
@call_as_thread @call_as_thread
def change_seek(self, widget, value): def change_seek(self, widget, value):
api.seek_percent(value) api.seek_percent(value)
@call_as_thread @call_as_thread
def change_volume(self, widget, value): def change_volume(self, widget, value):
api.set_volume(value) api.set_volume(value)
@call_as_thread @call_as_thread
def on_table_row_click(self, row_widget, playlist_item): def on_table_row_click(self, row_widget, playlist_item):
print("row", playlist_item) print("row", playlist_item)
@call_as_thread @call_as_thread
def on_table_item_move_click(self, row_widget, playlist_item, down = True): def on_table_item_move_click(self, row_widget, playlist_item, down = True):
index = playlist_item["index"] index = playlist_item["index"]
dest = index + 2 if down else index-1 dest = index + 2 if down else index-1
api.playlist_move(index, dest) api.playlist_move(index, dest)
@call_as_thread @call_as_thread
def on_table_item_remove_click(self, row_widget, playlist_item): def on_table_item_remove_click(self, row_widget, playlist_item):
api.playlist_remove(playlist_item["index"]) api.playlist_remove(playlist_item["index"])
@call_as_thread @call_as_thread
def on_table_item_goto_item(self, row_widget, playlist_item): def on_table_item_play_item(self, row_widget, playlist_item):
api.playlist_goto(playlist_item["index"]) api.playlist_goto(playlist_item["index"])
api.set_playing(True)
self.set_playing(True)
@call_as_thread @call_as_thread
def on_playlist_set_looping(self, row_widget): def on_playlist_set_looping(self, row_widget):
toggled = not row_widget.get_value() toggled = not row_widget.get_value()
api.playlist_set_looping(toggled) api.playlist_set_looping(toggled)
row_widget.set_value(toggled) row_widget.set_value(toggled)
@call_as_thread @call_as_thread
def on_playlist_clear_shuffle(self, row_widget): def on_playlist_clear_shuffle(self, row_widget):
api.playlist_shuffle() api.playlist_shuffle()
@call_as_thread @call_as_thread
def on_playlist_clear_click(self, row_widget): def on_playlist_clear_click(self, row_widget):
api.playlist_clear() api.playlist_clear()
# gui updaters: # gui updaters:
@call_as_thread @call_as_thread
def playback_update(self): def playback_update(self):
is_playing = api.is_playing() is_playing = api.is_playing()
@ -277,41 +292,43 @@ class RemiApp(App):
playback_pos = api.get_playback_pos() playback_pos = api.get_playback_pos()
except api.APIError: except api.APIError:
playback_pos = None playback_pos = None
if playback_pos: if playback_pos:
slider_pos = playback_pos["current"] / playback_pos["total"] * 100 slider_pos = playback_pos["current"] / playback_pos["total"] * 100
current = seconds_to_timestamp(playback_pos["current"]) current = seconds_to_timestamp(playback_pos["current"])
total = seconds_to_timestamp(playback_pos["total"]) total = seconds_to_timestamp(playback_pos["total"])
if self.playback.seek_slider.get_value() != slider_pos: if self.playback.seek_slider.get_value() != slider_pos:
self.playback.seek_slider.set_value(slider_pos) self.playback.seek_slider.set_value(slider_pos)
self.playback.timestamp.set_text(f"{current} - {total}") self.playback.timestamp.set_text(f"{current} - {total}")
else: else:
self.playback.timestamp.set_text("--:-- - --:--") self.playback.timestamp.set_text("--:-- - --:--")
@call_as_thread @call_as_thread
def volume_update(self): def volume_update(self):
volume = api.get_volume() volume = api.get_volume()
if volume > 100: if volume > 100:
volume = 100 volume = 100
if self.playback.volume_slider.get_value() != volume: if self.playback.volume_slider.get_value() != volume:
self.playback.volume_slider.set_value(volume) self.playback.volume_slider.set_value(volume)
@call_as_thread @call_as_thread
def playlist_update(self): def playlist_update(self):
playlist = api.get_playlist() # json structure playlist = api.get_playlist() # json structure
N = len(playlist) N = len(playlist)
# update playlist table content: # update playlist table content:
table = [] table = []
for playlist_item in playlist: for playlist_item in playlist:
name = playlist_item["filename"] name = playlist_item["filename"]
length = None length = None
if "data" in playlist_item: if "data" in playlist_item:
name = playlist_item["data"].get("title", name) name = playlist_item["data"].get("title", name)
length = playlist_item["data"].get("duration", length) length = playlist_item["data"].get("duration", length)
table.append([ table.append([
playlist_item["index"], playlist_item["index"],
name, name,
@ -334,29 +351,29 @@ class RemiApp(App):
for row_key, playlist_item in zip(self.playlist.table._render_children_list[1:], playlist): 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 = self.playlist.table.get_child(row_key)
row_widget.set_on_click_listener(self.on_table_row_click, playlist_item) row_widget.set_on_click_listener(self.on_table_row_click, playlist_item)
if playlist_item.get("current", False): if playlist_item.get("current", False):
self.playback.previous.set_enabled(playlist_item.get("index") != 0) self.playback.previous.set_enabled(playlist_item.get("index") != 0)
self.playback.next.set_enabled(playlist_item.get("index") != N-1) self.playback.next.set_enabled(playlist_item.get("index") != N-1)
row_widget.style["background-color"] = COLOR_LIGHT_BLUE row_widget.style["background-color"] = COLOR_LIGHT_BLUE
else: else:
row_widget.style["color"] = COLOR_GRAY_DARK row_widget.style["color"] = COLOR_GRAY_DARK
# for each item element in this row: # for each item element in this row:
for item_index, item_key in enumerate(row_widget._render_children_list): for item_index, item_key in enumerate(row_widget._render_children_list):
item_widget = row_widget.get_child(item_key) item_widget = row_widget.get_child(item_key)
if item_index == 1 and "failed" in playlist_item.get("data", {}): if item_index == 1 and "failed" in playlist_item.get("data", {}):
item_widget.style["width"] = "1.1em" item_widget.style["width"] = "1.1em"
item_widget.style["color"] = COLOR_RED item_widget.style["color"] = COLOR_RED
if item_index >= 3: if item_index >= 3:
item_widget.style["width"] = "1.1em" item_widget.style["width"] = "1.1em"
if item_index == 3: # seek here if item_index == 3: # seek here
item_widget.style["color"] = COLOR_GREEN#COLOR_RED if playlist_item.get("current", False) else 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_goto_item, playlist_item) item_widget.set_on_click_listener(self.on_table_item_play_item, playlist_item)
item_widget.attributes["title"] = "Play this item" item_widget.attributes["title"] = "Play this item"
if item_index == 4: # move up if item_index == 4: # move up
@ -373,13 +390,14 @@ class RemiApp(App):
else: else:
item_widget.style["color"] = COLOR_TEAL item_widget.style["color"] = COLOR_TEAL
item_widget.attributes["title"] = "Move this item down the playlist" item_widget.attributes["title"] = "Move this item down the playlist"
if item_index == 6: # remove from playlist if item_index == 6: # remove from playlist
item_widget.style["color"] = COLOR_RED item_widget.style["color"] = COLOR_RED
item_widget.set_on_click_listener(self.on_table_item_remove_click, playlist_item) item_widget.set_on_click_listener(self.on_table_item_remove_click, playlist_item)
item_widget.attributes["title"] = "Remove this item from the playlist" item_widget.attributes["title"] = "Remove this item from the playlist"
#print(index, key, item_widget) #print(index, key, item_widget)
def set_playing(self, is_playing:bool): # Updates GUI elements
def set_playing(self, is_playing:bool): # Only updates GUI elements!
self.playback.play.set_text(ICON_PAUSE if is_playing else ICON_PLAY) self.playback.play.set_text(ICON_PAUSE if is_playing else ICON_PLAY)
self.playback.seek_slider.set_enabled(is_playing) self.playback.seek_slider.set_enabled(is_playing)