1 {% macro dirlisting(dir, path='') -%}
2 <ul class="list-group">
3 {% if dir and dir.get_subdirs().items() %}
4 {% for subdirname, subdirobj in dir.get_subdirs().items() %}
5 {% set subdirpath = os.path.relpath(subdirobj.fullpath, music_library.fullpath) %}
6 {% set subdirid = subdirpath.replace("/","-") %}
7 <li class="directory list-group-item list-group-item-primary">
8 <div class="btn-group" role="group">
9 <div class="btn-group" role="group">
10 <button type="button" class="btn btn-success btn-sm"
11 onclick="request('/post', {add_folder : '{{ subdirpath }}'})">
12 <i class="fa fa-plus" aria-hidden="true"></i>
14 <div class="btn-group" role="group">
15 <button id="btnGroupDrop2" type="button" class="btn btn-success btn-sm dropdown-toggle btn-space" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"></button>
16 <div class="dropdown-menu" aria-labelledby="btnGroupDrop2" style="">
17 <a class="dropdown-item"
18 onclick="request('/post', {add_folder : '{{ subdirpath }}'})">
19 <i class="fa fa-folder" aria-hidden="true"></i> Entire folder
21 <a class="dropdown-item"
22 onclick="request('/post', {add_folder_recursively : '{{ subdirpath }}'})">
23 <i class="fa fa-folder" aria-hidden="true"></i> Entire folder and sub-folders
31 <div class="btn-group lead"><div class="btn-space"><i class="fa fa-folder" aria-hidden="true"></i></div><a class="lead" data-toggle="collapse"
32 data-target="#multiCollapse-{{ subdirid }}" aria-expanded="true"
33 aria-controls="multiCollapse-{{ subdirid }}" href="#"> {{ subdirpath }}/</a>
36 <div class="btn-group" style="float: right;">
37 <form action="./download" method="get" class="directory">
38 <input type="text" value="{{ subdirpath }}" name="directory" hidden>
39 <button type="submit" class="btn btn-primary btn-sm btn-space"><i class="fa fa-download" aria-hidden="true"></i></button>
41 <button type="submit" class="btn btn-danger btn-sm btn-space"
42 onclick="request('/post', {delete_folder : '{{ subdirpath }}'}, true)">
43 <i class="fas fa-trash-alt"></i>
47 <div class="collapse multi-collapse" id="multiCollapse-{{ subdirid }}">
48 {{ dirlisting(subdirobj, subdirpath) -}}
52 {% set files = dir.get_files() %}
54 {% for file in files %}
55 {% set filepath = os.path.relpath(os.path.join(dir.fullpath, file), music_library.fullpath) %}
56 <li class="file list-group-item">
57 <div class="btn-group" role="group">
58 <div class="btn-group" role="group">
59 <button type="button" class="btn btn-success btn-sm"
60 onclick="request('/post', {add_file_bottom : '{{ filepath }}'})">
61 <i class="fa fa-plus" aria-hidden="true"></i>
63 <div class="btn-group" role="group">
64 <button id="btnGroupDrop2" type="button" class="btn btn-success btn-sm dropdown-toggle btn-space" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"></button>
65 <div class="dropdown-menu" aria-labelledby="btnGroupDrop2" style="">
66 <a class="dropdown-item"
67 onclick="request('/post', {add_file_bottom : '{{ filepath }}'})">
68 <i class="fa fa-angle-down" aria-hidden="true"></i> To bottom of play list
70 <a class="dropdown-item"
71 onclick="request('/post', {add_file_next : '{{ filepath }}'})">
72 <i class="fa fa-angle-right" aria-hidden="true"></i> After current song
79 <div class="btn-group lead">
80 <div class="btn-space"><i class="fa fa-music" aria-hidden="true"></i></div>
83 {% if tags_lookup[filepath] %}
84 {% for tag in tags_lookup[filepath] %}
85 <span class="badge badge-{{ tags_color_lookup[tag] }}">{{ tag }}</span>
89 <div class="btn-group" style="float: right;">
90 <form action="./download" method="get" class="file file_download">
91 <input type="text" value="{{ filepath }}" name="file" hidden>
92 <button type="submit" class="btn btn-primary btn-sm btn-space"><i class="fa fa-download" aria-hidden="true"></i></button>
94 <button type="submit" class="btn btn-danger btn-sm btn-space"
95 onclick="request('/post', {delete_music_file : '{{ filepath }}'}, true)">
96 <i class="fas fa-trash-alt"></i>
108 <meta charset="UTF-8">
109 <title>Live 📻, don't 🤧</title>
110 <link id="pagestyle" rel="stylesheet" href="static/css/bootstrap.min.css">
111 <link rel="stylesheet" href="static/css/custom.css">
112 <META HTTP-EQUIV="Pragma" CONTENT="no-cache">
113 <META HTTP-EQUIV="Expires" CONTENT="-1">
117 <div class="container">
118 <div class="bs-docs-section">
119 <div class="page-header" id="banner">
120 <h1>Live 📻, don't 🤧</h1>
123 <div class="bs-docs-section">
126 <div id="playlist" class="col-lg-12">
128 <div class="btn-group" style="margin-bottom: 10px;">
129 <button type="button" id="play-btn" class="btn btn-info btn-space"
130 onclick="request('post', {action : 'resume'})" disabled>
131 <i class="fas fa-play" aria-hidden="true"></i>
134 <button type="button" id="pause-btn" class="btn btn-warning btn-space"
135 onclick="request('post', {action : 'pause'})" disabled>
136 <i class="fas fa-pause" aria-hidden="true"></i>
139 <button type="button" id="stop-btn" class="btn btn-danger btn-space"
140 onclick="request('post', {action : 'stop'})" disabled>
141 <i class="fas fa-stop" aria-hidden="true"></i>
143 <span id="current-time" style="line-height: 200%">
147 <div class="btn-group" style="float: right;">
149 <button type="button" id="oneshot-btn" class="btn btn-primary btn-space"
150 title="One-shot Mode"
151 onclick="request('post', {action : 'one-shot'})" disabled>
152 <i class="fas fa-tasks" aria-hidden="true"></i>
154 <button type="button" id="random-btn" class="btn btn-primary btn-space"
156 onclick="request('post', {action : 'randomize'})" disabled>
157 <i class="fas fa-random" aria-hidden="true"></i>
159 <button type="button" id="repeat-btn" class="btn btn-primary btn-space"
161 onclick="request('post', {action : 'repeat'})" disabled>
162 <i class="fas fa-redo" aria-hidden="true"></i>
164 <button type="button" id="autoplay-btn" class="btn btn-primary btn-space"
165 title="Autoplay Mode"
166 onclick="request('post', {action : 'autoplay'})" disabled>
167 <i class="fas fa-robot" aria-hidden="true"></i>
171 <button type="button" class="btn btn-warning btn-space"
172 onclick="request('post', {action : 'volume_down'})">
173 <i class="fa fa-volume-down" aria-hidden="true"></i>
175 <button type="button" class="btn btn-warning btn-space"
176 onclick="request('post', {action : 'volume_up'})">
177 <i class="fa fa-volume-up" aria-hidden="true"></i>
181 <table class="table">
184 <th scope="col">#</th>
185 <th scope="col" class="playlist-title-td">Title</th>
186 <th scope="col">Url/Path</th>
187 <th scope="col">Action</th>
190 <tbody id="playlist-table">
191 <tr class="table-dark">
192 <td colspan="4" class="text-muted" style="text-align:center;"> Fetching playlist .... </td>
197 <div class="btn-group">
198 <button type="button" class="btn btn-danger btn-space"
199 onclick="request('post', {action : 'clear'})">
200 <i class="fas fa-trash-alt" aria-hidden="true"></i> Clear Playlist
209 <div class="bs-docs-section">
212 <div id="browser" class="card">
213 <div class="card-header">
214 <h4 class="card-title">Files</h4>
217 <div class="card-body">
219 <div class="btn-group" style="margin-bottom: 5px;" role="group">
220 <button type="submit" class="btn btn-secondary btn-space"
221 onclick="request('/post', {action : 'rescan'}); location.reload()">
222 <i class="fas fa-sync-alt" aria-hidden="true"></i> Rescan Files
224 <form action="./download" method="get" class="directory form1">
225 <input type="text" value="./" name="directory" hidden>
226 <button type="submit" class="btn btn-secondary btn-space"><i class="fa fa-download" aria-hidden="true"></i> Download All</button>
228 <form method="post" class="directory form3">
229 <input type="text" value="./" name="add_folder_recursively" hidden>
230 <button type="submit" class="btn btn-secondary btn-space"><i class="fa fa-plus" aria-hidden="true"></i> Add All</button>
235 {{ dirlisting(music_library) }}
243 <div id="upload" class="container">
244 <div class="bs-docs-section">
248 <div class="card-header">
249 <h5 class="card-title">Upload File</h5>
251 <div class="card-body">
252 <form action="./upload" method="post" enctype="multipart/form-data">
253 <div class="row" style="margin-bottom: 5px;">
254 <div id="uploadBox" class="col-lg-7 input-group">
255 <div id="uploadField" style="display: flex; width: 100%">
256 <div class="custom-file btn-space">
257 <input type="file" name="file[]" class="custom-file-input" id="uploadSelectFile"
258 aria-describedby="uploadSubmit" value="Browse Music file" multiple/>
259 <label class="custom-file-label" for="uploadSelectFile">Choose file</label>
263 <div class="col-lg-4 input-group-append">
264 <span class="input-group-text">Upload To</span>
265 <input class="form-control btn-space" list="targetdirs" id="targetdir" name="targetdir"
266 placeholder="uploads" />
267 <datalist id="targetdirs">
268 <option value="uploads">
269 {% for dir in music_library.get_subdirs_recursively() %}
270 <option value="{{ dir }}">
274 <button class="btn btn-primary btn-space" type="submit"
275 id="uploadSubmit" style="margin-left: -5px;">Upload!</button>
284 <div class="bs-docs-section" style="margin-bottom: 150px;">
288 <div class="card-header">
289 <h5 class="card-title">Add URL</h5>
291 <div class="card-body">
292 <label>Add Youtube/Soundcloud URL</label>
293 <div class="input-group">
294 <input class="form-control btn-space" type="text" id="add_url_input" placeholder="URL...">
295 <button type="submit" class="btn btn-primary"
296 onclick="var $i = $('#add_url_input')[0]; request('/post', {add_url : $i.value }); $i.value = ''; ">Add URL</button>
303 <div class="card-header">
304 <h5 class="card-title">Add Radio</h5>
306 <div class="card-body">
307 <label>Add Radio URL</label>
308 <div class="input-group">
309 <input class="form-control btn-space" type="text" id="add_radio_input" placeholder="Radio Address...">
310 <button type="submit" class="btn btn-primary"
311 onclick="var $i = $('#add_radio_input')[0]; request('/post', {add_radio : $i.value }); $i.value = '';">Add Radio</button>
320 <div class="floating-button" onclick="switchTheme()"> <i class="fas fa-lightbulb" aria-hidden="true"></i> </div>
322 <script src="static/js/jquery-3.4.1.min.js" crossorigin="anonymous"></script>
323 <script src="static/js/bootstrap.bundle.min.js" crossorigin="anonymous"></script>
324 <script src="static/js/fontawesome.all.js" crossorigin="anonymous"></script>
327 $('#uploadSelectFile').on('change', function () {
329 var fileName = $(this).val().replace('C:\\fakepath\\', " ");
330 //replace the "Choose a file" label
331 $(this).next('.custom-file-label').html(fileName);
333 $('a.a-submit, button.btn-submit').on('click', function (event) {
334 $(event.target).closest('form').submit();
337 var playlist_ver = 0;
339 function format_duration(secs) {
340 return parseInt(secs/60) + ':' + ('0' + parseInt(secs%60)).slice(-2);
343 function request(url, _data, refresh=false){
349 200 : function(data) {
350 if (data.ver !== playlist_ver) {
352 playlist_ver = data.ver;
354 updateControls(data);
363 function displayPlaylist(data){
364 // console.info(data);
365 $("#playlist-table tr").remove();
367 var items = data.items;
368 $.each(items, function(index, item){
369 $("#playlist-table").append(item);
374 function updatePlaylist(){
379 200 : displayPlaylist
384 function updateControls(data) {
385 var empty = data.empty;
386 var play = data.play;
387 var mode = data.mode;
389 $("#play-btn").prop("disabled", true);
390 $("#pause-btn").prop("disabled", true);
391 $("#stop-btn").prop("disabled", true);
394 $("#play-btn").prop("disabled", true);
395 $("#pause-btn").prop("disabled", false);
396 $("#stop-btn").prop("disabled", false);
398 $("#play-btn").prop("disabled", false);
399 $("#pause-btn").prop("disabled", true);
400 $("#stop-btn").prop("disabled", true);
403 if(mode === "one-shot"){
404 $("#oneshot-btn").removeClass("btn-secondary").addClass("btn-primary").prop("disabled", true);
405 $("#repeat-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
406 $("#random-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
407 $("#autoplay-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
408 }else if(mode === "repeat"){
409 $("#oneshot-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
410 $("#repeat-btn").removeClass("btn-secondary").addClass("btn-primary").prop("disabled", true);
411 $("#random-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
412 $("#autoplay-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
413 }else if(mode === "random"){
414 $("#oneshot-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
415 $("#repeat-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
416 $("#random-btn").removeClass("btn-secondary").addClass("btn-primary").prop("disabled", false); // This is a feature.
417 $("#autoplay-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
418 }else if(mode === "autoplay"){
419 $("#oneshot-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
420 $("#repeat-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
421 $("#random-btn").removeClass("btn-primary").addClass("btn-secondary").prop("disabled", false);
422 $("#autoplay-btn").removeClass("btn-secondary").addClass("btn-primary").prop("disabled", true);
424 var current_time = "";
425 if (data.playhead != -1) {
426 var duration = $('#playlist-table .table-active').data('duration');
428 var elapsed = data.playhead;
429 current_time = format_duration(elapsed) + '/' + format_duration(duration);
432 $('#current-time').text(current_time);
435 function themeInit(){
436 var theme = localStorage.getItem("theme");
442 function switchTheme(){
443 var theme = localStorage.getItem("theme");
444 if(theme === "light" || theme === null){
445 setPageTheme("dark");
446 localStorage.setItem("theme", "dark");
448 setPageTheme("light");
449 localStorage.setItem("theme", "light");
453 function setPageTheme(theme) {
454 if(theme === "light")
455 document.getElementById("pagestyle").setAttribute("href", "static/css/bootstrap.min.css");
456 else if(theme === "dark")
457 document.getElementById("pagestyle").setAttribute("href", "static/css/bootstrap.darkly.min.css");
460 // Check the version of playlist to see if update is needed.
461 var update_interval_id = null;
467 200 : function(data){
468 if(data.ver !== playlist_ver){
470 playlist_ver = data.ver;
472 updateControls(data);
474 clearInterval(update_interval_id);
475 update_interval_id = setInterval(update, 1000);
477 clearInterval(update_interval_id);
478 update_interval_id = setInterval(update, 3000);
484 update_interval_id = setInterval(update, 3000);
487 $(document).ready(updatePlaylist);