mirror of
https://github.com/9001/copyparty.git
synced 2025-08-17 09:02:15 -06:00
3413 lines
78 KiB
JavaScript
3413 lines
78 KiB
JavaScript
"use strict";
|
||
|
||
window.onerror = vis_exh;
|
||
|
||
function dbg(msg) {
|
||
ebi('path').innerHTML = msg;
|
||
}
|
||
|
||
|
||
// toolbar
|
||
ebi('ops').innerHTML = (
|
||
'<a href="#" data-dest="" tt="close submenu">---</a>\n' +
|
||
(have_up2k_idx ? (
|
||
'<a href="#" data-perm="read" data-dest="search" tt="search for files by attributes, path/name, music tags, or any combination of those.$N$N<code>foo bar</code> = must contain both foo and bar,$N<code>foo -bar</code> = must contain foo but not bar,$N<code>^yana .opus$</code> = must start with yana and have the opus extension">🔎</a>\n' +
|
||
'<a href="#" data-dest="up2k" tt="up2k: upload files (if you have write-access) or toggle into the search-mode and drag files onto the search button to see if they exist somewhere on the server">🚀</a>\n'
|
||
) : (
|
||
'<a href="#" data-perm="write" data-dest="up2k" tt="up2k: upload files with resume support (close your browser and drop the same files in later)">🚀</a>\n'
|
||
)) +
|
||
'<a href="#" data-perm="write" data-dest="bup" tt="bup: basic uploader, even supports netscape 4.0">🎈</a>\n' +
|
||
'<a href="#" data-perm="write" data-dest="mkdir" tt="mkdir: create a new directory">📂</a>\n' +
|
||
'<a href="#" data-perm="read write" data-dest="new_md" tt="new-md: create a new markdown document">📝</a>\n' +
|
||
'<a href="#" data-perm="write" data-dest="msg" tt="msg: send a message to the server log">📟</a>\n' +
|
||
'<a href="#" data-dest="player" tt="media player options">🎺</a>\n' +
|
||
'<a href="#" data-dest="cfg" tt="configuration options">⚙️</a>\n' +
|
||
'<div id="opdesc"></div>'
|
||
);
|
||
|
||
|
||
// media player
|
||
ebi('widget').innerHTML = (
|
||
'<div id="wtoggle">' +
|
||
'<span id="wfm"><a' +
|
||
' href="#" id="fren" tt="rename selected item$NHotkey: F2">✎<span>name</span></a><a' +
|
||
' href="#" id="fdel" tt="delete selected items$NHotkey: ctrl-K">⌫<span>delete</span></a><a' +
|
||
' href="#" id="fcut" tt="cut selected items <small>(then paste somewhere else)</small>$NHotkey: ctrl-X">✂<span>cut</span></a><a' +
|
||
' href="#" id="fpst" tt="paste a previously cut/copied selection$NHotkey: ctrl-V">📋<span>paste</span></a>' +
|
||
'</span><span id="wzip"><a' +
|
||
' href="#" id="selall" tt="select all files">sel.<br />all</a><a' +
|
||
' href="#" id="selinv" tt="invert selection">sel.<br />inv.</a><a' +
|
||
' href="#" id="selzip" tt="download selection as archive">zip</a>' +
|
||
'</span><span id="wnp"><a' +
|
||
' href="#" id="npirc" tt="copy irc-formatted track info">📋irc</a><a' +
|
||
' href="#" id="nptxt" tt="copy plaintext track info">📋txt</a>' +
|
||
'</span><a' +
|
||
' href="#" id="wtgrid" tt="toggle grid/list view">田</a><a' +
|
||
' href="#" id="wtico">♫</a>' +
|
||
'</div>' +
|
||
'<div id="widgeti">' +
|
||
' <div id="pctl"><a href="#" id="bprev" tt="previous track$NHotkey: J">⏮</a><a href="#" id="bplay" tt="play/pause$NHotkey: P">▶</a><a href="#" id="bnext" tt="next track$NHotkey: L">⏭</a></div>' +
|
||
' <canvas id="pvol" width="288" height="38"></canvas>' +
|
||
' <canvas id="barpos"></canvas>' +
|
||
' <canvas id="barbuf"></canvas>' +
|
||
'</div>'
|
||
);
|
||
|
||
|
||
// up2k ui
|
||
ebi('op_up2k').innerHTML = (
|
||
'<form id="u2form" method="post" enctype="multipart/form-data" onsubmit="return false;"></form>\n' +
|
||
|
||
'<table id="u2conf">\n' +
|
||
' <tr>\n' +
|
||
' <td><br />parallel uploads:</td>\n' +
|
||
' <td rowspan="2">\n' +
|
||
' <input type="checkbox" id="multitask" />\n' +
|
||
' <label for="multitask" tt="continue hashing other files while uploading">🏃</label>\n' +
|
||
' </td>\n' +
|
||
' <td rowspan="2">\n' +
|
||
' <input type="checkbox" id="ask_up" />\n' +
|
||
' <label for="ask_up" tt="ask for confirmation before upload starts">💭</label>\n' +
|
||
' </td>\n' +
|
||
' <td rowspan="2">\n' +
|
||
' <input type="checkbox" id="flag_en" />\n' +
|
||
' <label for="flag_en" tt="ensure only one tab is uploading at a time $N (other tabs must have this enabled too)">💤</label>\n' +
|
||
' </td>\n' +
|
||
(have_up2k_idx ? (
|
||
' <td data-perm="read" rowspan="2">\n' +
|
||
' <input type="checkbox" id="fsearch" />\n' +
|
||
' <label for="fsearch" tt="don\'t actually upload, instead check if the files already $N exist on the server (will scan all folders you can read)">🔎</label>\n' +
|
||
' </td>\n'
|
||
) : '') +
|
||
' <td data-perm="read" rowspan="2" id="u2btn_cw"></td>\n' +
|
||
' </tr>\n' +
|
||
' <tr>\n' +
|
||
' <td>\n' +
|
||
' <a href="#" id="nthread_sub">–</a><input\n' +
|
||
' class="txtbox" id="nthread" value="2" tt="pause uploads by setting it to 0"/><a\n' +
|
||
' href="#" id="nthread_add">+</a><br /> \n' +
|
||
' </td>\n' +
|
||
' </tr>\n' +
|
||
'</table>\n' +
|
||
|
||
'<div id="u2notbtn"></div>\n' +
|
||
|
||
'<div id="u2btn_ct">\n' +
|
||
' <div id="u2btn">\n' +
|
||
' <span id="u2bm"></span><br />\n' +
|
||
' drag/drop files<br />\n' +
|
||
' and folders here<br />\n' +
|
||
' (or click me)\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
|
||
'<div id="u2cards">\n' +
|
||
' <a href="#" act="ok" tt="completed successfully">ok <span>0</span></a><a\n' +
|
||
' href="#" act="ng" tt="failed / rejected / not-found">ng <span>0</span></a><a\n' +
|
||
' href="#" act="done" tt="ok and ng combined">done <span>0</span></a><a\n' +
|
||
' href="#" act="bz" tt="hashing or uploading" class="act">busy <span>0</span></a><a\n' +
|
||
' href="#" act="q" tt="idle, pending">que <span>0</span></a>\n' +
|
||
'</div>\n' +
|
||
|
||
'<table id="u2tab">\n' +
|
||
' <thead>\n' +
|
||
' <tr>\n' +
|
||
' <td>filename</td>\n' +
|
||
' <td>status</td>\n' +
|
||
' <td>progress<a href="#" id="u2cleanup" tt="remove completed uploads$N(makes it possible to upload a file after searching for it)">cleanup</a></td>\n' +
|
||
' </tr>\n' +
|
||
' </thead>\n' +
|
||
' <tbody></tbody>\n' +
|
||
'</table>\n' +
|
||
|
||
'<p id="u2foot"></p>\n' +
|
||
'<p id="u2footfoot" data-perm="write">( you can use the <a href="#" id="u2nope">basic uploader</a> if you don\'t need lastmod timestamps, resumable uploads, or progress bars )</p>'
|
||
);
|
||
|
||
|
||
// config panel
|
||
ebi('op_cfg').innerHTML = (
|
||
'<div>\n' +
|
||
' <h3>switches</h3>\n' +
|
||
' <div>\n' +
|
||
' <a id="tooltips" class="tgl btn" href="#" tt="◔ ◡ ◔">ℹ️ tooltips</a>\n' +
|
||
' <a id="lightmode" class="tgl btn" href="#">☀️ lightmode</a>\n' +
|
||
' <a id="griden" class="tgl btn" href="#" tt="toggle icons or list-view$NHotkey: G">田 the grid</a>\n' +
|
||
' <a id="thumbs" class="tgl btn" href="#" tt="in icon view, toggle icons or thumbnails$NHotkey: T">🖼️ thumbs</a>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
(have_zip ? (
|
||
'<div><h3>folder download</h3><div id="arc_fmt"></div></div>\n'
|
||
) : '') +
|
||
'<div>\n' +
|
||
' <h3>up2k switches</h3>\n' +
|
||
' <div>\n' +
|
||
' <a id="u2turbo" class="tgl btn ttb" href="#" tt="the yolo button, you probably DO NOT want to enable this:$N$Nuse this if you were uploading a huge amount of files and had to restart for some reason, and want to continue the upload ASAP$N$Nthis replaces the hash-check with a simple <em>"does this have the same filesize on the server?"</em> so if the file contents are different it will NOT be uploaded$N$Nyou should turn this off when the upload is done, and then "upload" the same files again to let the client verify them">turbo</a>\n' +
|
||
' <a id="u2tdate" class="tgl btn ttb" href="#" tt="has no effect unless the turbo button is enabled$N$Nreduces the yolo factor by a tiny amount; checks whether the file timestamps on the server matches yours$N$Nshould <em>theoretically</em> catch most unfinished/corrupted uploads, but is not a substitute for doing a verification pass with turbo disabled afterwards">date-chk</a>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div><h3>key notation</h3><div id="key_notation"></div></div>\n' +
|
||
'<div class="fill"><h3>hidden columns</h3><div id="hcols"></div></div>'
|
||
);
|
||
|
||
|
||
// tree sidebar
|
||
ebi('tree').innerHTML = (
|
||
'<div id="treeh">\n' +
|
||
' <a href="#" id="detree" tt="show breadcrumbs$NHotkey: B">🍞...</a>\n' +
|
||
' <a href="#" class="btn" step="2" id="twobytwo" tt="Hotkey: A">+</a>\n' +
|
||
' <a href="#" class="btn" step="-2" id="twig" tt="Hotkey: D">–</a>\n' +
|
||
' <a href="#" class="tgl btn" id="dyntree" tt="autogrow as tree expands">a</a>\n' +
|
||
'</div>\n' +
|
||
'<ul id="treeul"></ul>\n' +
|
||
'<div id="thx_ff"> </div>'
|
||
);
|
||
|
||
|
||
(function () {
|
||
var ops = QSA('#ops>a');
|
||
for (var a = 0; a < ops.length; a++) {
|
||
ops[a].onclick = opclick;
|
||
}
|
||
})();
|
||
|
||
|
||
function opclick(e) {
|
||
var dest = this.getAttribute('data-dest');
|
||
swrite('opmode', dest || null);
|
||
if (ctrl(e))
|
||
return;
|
||
|
||
ev(e);
|
||
goto(dest);
|
||
|
||
var input = QS('.opview.act input:not([type="hidden"])')
|
||
if (input && !is_touch) {
|
||
tt.skip = true;
|
||
input.focus();
|
||
}
|
||
}
|
||
|
||
|
||
function goto(dest) {
|
||
var obj = QSA('.opview.act');
|
||
for (var a = obj.length - 1; a >= 0; a--)
|
||
clmod(obj[a], 'act');
|
||
|
||
obj = QSA('#ops>a');
|
||
for (var a = obj.length - 1; a >= 0; a--)
|
||
clmod(obj[a], 'act');
|
||
|
||
if (dest) {
|
||
var ui = ebi('op_' + dest);
|
||
clmod(ui, 'act', true);
|
||
QS('#ops>a[data-dest=' + dest + ']').className += " act";
|
||
|
||
var fn = window['goto_' + dest];
|
||
if (fn)
|
||
fn();
|
||
}
|
||
|
||
if (window['treectl'])
|
||
treectl.onscroll();
|
||
}
|
||
|
||
|
||
(function () {
|
||
goto();
|
||
var op = sread('opmode');
|
||
if (op !== null && op !== '.')
|
||
try {
|
||
goto(op);
|
||
}
|
||
catch (ex) { }
|
||
})();
|
||
|
||
|
||
var have_webp = null;
|
||
(function () {
|
||
var img = new Image();
|
||
img.onload = function () {
|
||
have_webp = img.width > 0 && img.height > 0;
|
||
};
|
||
img.onerror = function () {
|
||
have_webp = false;
|
||
};
|
||
img.src = "";
|
||
})();
|
||
|
||
|
||
var mpl = (function () {
|
||
var have_mctl = 'mediaSession' in navigator && window.MediaMetadata;
|
||
|
||
ebi('op_player').innerHTML = (
|
||
'<div><h3>switches</h3><div>' +
|
||
'<a href="#" class="tgl btn" id="au_preload" tt="start loading the next song near the end for gapless playback">preload</a>' +
|
||
'<a href="#" class="tgl btn" id="au_npclip" tt="show buttons for clipboarding the currently playing song">/np clip</a>' +
|
||
'<a href="#" class="tgl btn" id="au_os_ctl" tt="os integration (media hotkeys / osd)">os-ctl</a>' +
|
||
'<a href="#" class="tgl btn" id="au_osd_cv" tt="show album cover in osd">osd-cv</a>' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3>playback mode</h3><div id="pb_mode">' +
|
||
'<a href="#" class="tgl btn" tt="loop the open folder">🔁 loop-folder</a>' +
|
||
'<a href="#" class="tgl btn" tt="load the next folder and continue">📂 next-folder</a>' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3>tint</h3><div>' +
|
||
'<input type="text" id="pb_tint" size="3" value="0" tt="background level (0-100) on the seekbar$Nto make buffering less distracting" />' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3>audio equalizer</h3><div id="audio_eq"></div></div>');
|
||
|
||
var r = {
|
||
"pb_mode": sread('pb_mode') || 'loop-folder',
|
||
"preload": bcfg_get('au_preload', true),
|
||
"clip": bcfg_get('au_npclip', false),
|
||
"os_ctl": bcfg_get('au_os_ctl', have_mctl) && have_mctl,
|
||
"osd_cv": bcfg_get('au_osd_cv', true),
|
||
};
|
||
|
||
ebi('au_preload').onclick = function (e) {
|
||
ev(e);
|
||
r.preload = !r.preload;
|
||
bcfg_set('au_preload', r.preload);
|
||
};
|
||
|
||
ebi('au_npclip').onclick = function (e) {
|
||
ev(e);
|
||
r.clip = !r.clip;
|
||
bcfg_set('au_npclip', r.clip);
|
||
clmod(ebi('wtoggle'), 'np', r.clip && mp.au);
|
||
};
|
||
|
||
ebi('au_os_ctl').onclick = function (e) {
|
||
ev(e);
|
||
r.os_ctl = !r.os_ctl && have_mctl;
|
||
bcfg_set('au_os_ctl', r.os_ctl);
|
||
if (!have_mctl)
|
||
alert('need firefox 82+ or chrome 73+');
|
||
};
|
||
|
||
ebi('au_osd_cv').onclick = function (e) {
|
||
ev(e);
|
||
r.osd_cv = !r.osd_cv;
|
||
bcfg_set('au_osd_cv', r.osd_cv);
|
||
};
|
||
|
||
function draw_pb_mode() {
|
||
var btns = QSA('#pb_mode>a');
|
||
for (var a = 0, aa = btns.length; a < aa; a++) {
|
||
clmod(btns[a], 'on', btns[a].textContent.indexOf(r.pb_mode) != -1);
|
||
btns[a].onclick = set_pb_mode;
|
||
}
|
||
}
|
||
draw_pb_mode();
|
||
|
||
function set_pb_mode(e) {
|
||
ev(e);
|
||
r.pb_mode = this.textContent.split(' ').slice(-1)[0];
|
||
swrite('pb_mode', r.pb_mode);
|
||
draw_pb_mode();
|
||
}
|
||
|
||
function set_tint() {
|
||
var tint = icfg_get('pb_tint', 0);
|
||
if (!tint)
|
||
ebi('barbuf').style.removeProperty('background');
|
||
else
|
||
ebi('barbuf').style.background = 'rgba(126,163,75,' + (tint / 100.0) + ')';
|
||
}
|
||
ebi('pb_tint').oninput = function (e) {
|
||
swrite('pb_tint', this.value);
|
||
set_tint();
|
||
};
|
||
set_tint();
|
||
|
||
r.pp = function () {
|
||
if (!r.os_ctl)
|
||
return;
|
||
|
||
navigator.mediaSession.playbackState = mp.au && !mp.au.paused ? "playing" : "paused";
|
||
};
|
||
|
||
r.announce = function () {
|
||
if (!r.os_ctl)
|
||
return;
|
||
|
||
var np = get_np()[0],
|
||
fns = np.file.split(' - '),
|
||
artist = (np.circle ? np.circle + ' // ' : '') + (np.artist || (fns.length > 1 ? fns[0] : '')),
|
||
tags = {
|
||
title: np.title || fns.slice(-1)[0]
|
||
};
|
||
|
||
if (artist)
|
||
tags.artist = artist;
|
||
|
||
if (np.album)
|
||
tags.album = np.album;
|
||
|
||
if (r.osd_cv) {
|
||
var files = QSA("#files tr>td:nth-child(2)>a[id]"),
|
||
cover = null;
|
||
|
||
for (var a = 0, aa = files.length; a < aa; a++) {
|
||
if (/^(cover|folder)\.(jpe?g|png|gif)$/.test(files[a].textContent)) {
|
||
cover = files[a].getAttribute('href');
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (cover) {
|
||
cover += (cover.indexOf('?') === -1 ? '?' : '&') + 'th=j';
|
||
|
||
var pwd = get_pwd();
|
||
if (pwd)
|
||
cover += '&pw=' + uricom_enc(pwd);
|
||
|
||
tags.artwork = [{ "src": cover, type: "image/jpeg" }];
|
||
}
|
||
}
|
||
|
||
navigator.mediaSession.metadata = new MediaMetadata(tags);
|
||
navigator.mediaSession.setActionHandler('play', playpause);
|
||
navigator.mediaSession.setActionHandler('pause', playpause);
|
||
navigator.mediaSession.setActionHandler('seekbackward', function () { seek_au_rel(-10); });
|
||
navigator.mediaSession.setActionHandler('seekforward', function () { seek_au_rel(10); });
|
||
navigator.mediaSession.setActionHandler('previoustrack', prev_song);
|
||
navigator.mediaSession.setActionHandler('nexttrack', next_song);
|
||
r.pp();
|
||
};
|
||
|
||
r.stop = function () {
|
||
if (!r.os_ctl || !navigator.mediaSession.metadata)
|
||
return;
|
||
|
||
navigator.mediaSession.metadata = null;
|
||
navigator.mediaSession.playbackState = "paused";
|
||
};
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
// extract songs + add play column
|
||
function MPlayer() {
|
||
var r = this;
|
||
r.id = Date.now();
|
||
r.au = null;
|
||
r.au_native = null;
|
||
r.au_native2 = null;
|
||
r.au_ogvjs = null;
|
||
r.au_ogvjs2 = null;
|
||
r.tracks = {};
|
||
r.order = [];
|
||
|
||
var re_audio = /\.(opus|ogg|m4a|aac|mp3|wav|flac)$/i,
|
||
trs = QSA('#files tbody tr');
|
||
|
||
for (var a = 0, aa = trs.length; a < aa; a++) {
|
||
var tds = trs[a].getElementsByTagName('td'),
|
||
link = tds[1].getElementsByTagName('a');
|
||
|
||
link = link[link.length - 1];
|
||
var url = link.getAttribute('href'),
|
||
m = re_audio.exec(url);
|
||
|
||
if (m) {
|
||
var tid = link.getAttribute('id');
|
||
r.order.push(tid);
|
||
r.tracks[tid] = url;
|
||
tds[0].innerHTML = '<a id="a' + tid + '" href="#a' + tid + '" class="play">play</a></td>';
|
||
ebi('a' + tid).onclick = ev_play;
|
||
}
|
||
}
|
||
|
||
r.vol = sread('vol');
|
||
if (r.vol !== null)
|
||
r.vol = parseFloat(r.vol);
|
||
else
|
||
r.vol = 0.5;
|
||
|
||
r.expvol = function (v) {
|
||
return 0.5 * v + 0.5 * v * v;
|
||
};
|
||
|
||
r.setvol = function (vol) {
|
||
r.vol = Math.max(Math.min(vol, 1), 0);
|
||
swrite('vol', vol);
|
||
r.stopfade(true);
|
||
|
||
if (r.au)
|
||
r.au.volume = r.expvol(r.vol);
|
||
};
|
||
|
||
r.read_order = function () {
|
||
var order = [],
|
||
links = QSA('#files>tbody>tr>td:nth-child(1)>a');
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
var tid = links[a].getAttribute('id');
|
||
if (!tid || tid.indexOf('af-') !== 0)
|
||
continue;
|
||
|
||
order.push(tid.slice(1));
|
||
}
|
||
r.order = order;
|
||
};
|
||
|
||
r.fdir = 0;
|
||
r.fvol = -1;
|
||
r.ftid = -1;
|
||
r.ftimer = null;
|
||
r.fade_in = function () {
|
||
r.fvol = 0;
|
||
r.fdir = 0.025;
|
||
if (r.au) {
|
||
r.ftid = r.au.tid;
|
||
r.au.play();
|
||
mpl.pp();
|
||
fader();
|
||
}
|
||
};
|
||
r.fade_out = function () {
|
||
r.fvol = r.vol;
|
||
r.fdir = -0.05;
|
||
r.ftid = r.au.tid;
|
||
fader();
|
||
};
|
||
r.stopfade = function (hard) {
|
||
clearTimeout(r.ftimer);
|
||
if (hard)
|
||
r.ftid = -1;
|
||
}
|
||
function fader() {
|
||
r.stopfade();
|
||
if (!r.au || r.au.tid !== r.ftid)
|
||
return;
|
||
|
||
var done = true;
|
||
r.fvol += r.fdir;
|
||
if (r.fvol < 0) {
|
||
r.fvol = 0;
|
||
r.au.pause();
|
||
mpl.pp();
|
||
}
|
||
else if (r.fvol > r.vol)
|
||
r.fvol = r.vol;
|
||
else
|
||
done = false;
|
||
|
||
r.au.volume = r.expvol(r.fvol);
|
||
if (!done)
|
||
setTimeout(fader, 10);
|
||
}
|
||
|
||
r.preload = function (url) {
|
||
var au = null;
|
||
if (need_ogv_for(url)) {
|
||
au = mp.au_ogvjs2;
|
||
if (!au && window['OGVPlayer']) {
|
||
au = new OGVPlayer();
|
||
au.preload = "auto";
|
||
r.au_ogvjs2 = au;
|
||
}
|
||
} else {
|
||
au = mp.au_native2;
|
||
if (!au) {
|
||
au = new Audio();
|
||
au.preload = "auto";
|
||
r.au_native2 = au;
|
||
}
|
||
}
|
||
if (au) {
|
||
au.src = url + (url.indexOf('?') < 0 ? '?cache' : '&cache');
|
||
}
|
||
};
|
||
}
|
||
|
||
addcrc();
|
||
var mp = new MPlayer();
|
||
makeSortable(ebi('files'), mp.read_order.bind(mp));
|
||
|
||
|
||
function get_np() {
|
||
var th = ebi('files').tHead.rows[0].cells,
|
||
tr = QS('#files tr.play').cells,
|
||
rv = [],
|
||
ra = [],
|
||
rt = {};
|
||
|
||
for (var a = 1, aa = th.length; a < aa; a++) {
|
||
var tv = tr[a].textContent,
|
||
tk = a == 1 ? 'file' : th[a].getAttribute('name').split('/').slice(-1)[0],
|
||
vis = th[a].className.indexOf('min') === -1;
|
||
|
||
if (!tv)
|
||
continue;
|
||
|
||
(vis ? rv : ra).push(tk);
|
||
rt[tk] = tv;
|
||
}
|
||
return [rt, rv, ra];
|
||
};
|
||
|
||
|
||
// toggle player widget
|
||
var widget = (function () {
|
||
var r = {},
|
||
widget = ebi('widget'),
|
||
wtico = ebi('wtico'),
|
||
nptxt = ebi('nptxt'),
|
||
npirc = ebi('npirc'),
|
||
touchmode = false,
|
||
was_paused = true;
|
||
|
||
r.is_open = false;
|
||
|
||
r.open = function () {
|
||
if (r.is_open)
|
||
return false;
|
||
|
||
widget.className = 'open';
|
||
r.is_open = true;
|
||
return true;
|
||
};
|
||
r.close = function () {
|
||
if (!r.is_open)
|
||
return false;
|
||
|
||
widget.className = '';
|
||
r.is_open = false;
|
||
return true;
|
||
};
|
||
r.toggle = function (e) {
|
||
r.open() || r.close();
|
||
ev(e);
|
||
return false;
|
||
};
|
||
r.paused = function (paused) {
|
||
if (was_paused != paused) {
|
||
was_paused = paused;
|
||
ebi('bplay').innerHTML = paused ? '▶' : '⏸';
|
||
}
|
||
};
|
||
wtico.onclick = function (e) {
|
||
if (!touchmode)
|
||
r.toggle(e);
|
||
|
||
return false;
|
||
};
|
||
npirc.onclick = nptxt.onclick = function (e) {
|
||
ev(e);
|
||
var irc = this.getAttribute('id') == 'npirc',
|
||
ck = irc ? '06' : '',
|
||
cv = irc ? '07' : '',
|
||
m = ck + 'np: ',
|
||
npr = get_np(),
|
||
npk = npr[1],
|
||
np = npr[0];
|
||
|
||
for (var a = 0; a < npk.length; a++)
|
||
m += (npk[a] == 'file' ? '' : npk[a]) + '(' + cv + np[npk[a]] + ck + ') // ';
|
||
|
||
m += '[' + cv + s2ms(mp.au.currentTime) + ck + '/' + cv + s2ms(mp.au.duration) + ck + ']';
|
||
|
||
var o = mknod('input');
|
||
o.style.cssText = 'position:fixed;top:45%;left:48%;padding:1em;z-index:9';
|
||
o.value = m;
|
||
document.body.appendChild(o);
|
||
o.focus();
|
||
o.select();
|
||
document.execCommand("copy");
|
||
o.value = 'copied to clipboard ';
|
||
setTimeout(function () {
|
||
document.body.removeChild(o);
|
||
}, 500);
|
||
};
|
||
return r;
|
||
})();
|
||
|
||
|
||
function canvas_cfg(can) {
|
||
var r = {},
|
||
b = can.getBoundingClientRect(),
|
||
mul = window.devicePixelRatio || 1;
|
||
|
||
r.w = b.width;
|
||
r.h = b.height;
|
||
can.width = r.w * mul;
|
||
can.height = r.h * mul;
|
||
|
||
r.can = can;
|
||
r.ctx = can.getContext('2d');
|
||
r.ctx.scale(mul, mul);
|
||
return r;
|
||
}
|
||
|
||
|
||
function glossy_grad(can, h, s, l) {
|
||
var g = can.ctx.createLinearGradient(0, 0, 0, can.h),
|
||
p = [0, 0.49, 0.50, 1];
|
||
|
||
for (var a = 0; a < p.length; a++)
|
||
g.addColorStop(p[a], 'hsl(' + h + ',' + s[a] + '%,' + l[a] + '%)');
|
||
|
||
return g;
|
||
}
|
||
|
||
|
||
// buffer/position bar
|
||
var pbar = (function () {
|
||
var r = {},
|
||
gradh = -1,
|
||
grad;
|
||
|
||
r.onresize = function () {
|
||
r.buf = canvas_cfg(ebi('barbuf'));
|
||
r.pos = canvas_cfg(ebi('barpos'));
|
||
r.drawbuf();
|
||
r.drawpos();
|
||
}
|
||
|
||
r.drawbuf = function () {
|
||
var bc = r.buf,
|
||
bctx = bc.ctx;
|
||
|
||
bctx.clearRect(0, 0, bc.w, bc.h);
|
||
|
||
if (!mp.au)
|
||
return;
|
||
|
||
var sm = bc.w * 1.0 / mp.au.duration,
|
||
gk = bc.h + '' + light;
|
||
|
||
if (gradh != gk) {
|
||
gradh = gk;
|
||
grad = glossy_grad(bc, 85, [35, 40, 37, 35], light ? [45, 56, 50, 45] : [42, 51, 47, 42]);
|
||
}
|
||
bctx.fillStyle = grad;
|
||
for (var a = 0; a < mp.au.buffered.length; a++) {
|
||
var x1 = sm * mp.au.buffered.start(a),
|
||
x2 = sm * mp.au.buffered.end(a);
|
||
|
||
bctx.fillRect(x1, 0, x2 - x1, bc.h);
|
||
}
|
||
};
|
||
|
||
r.drawpos = function () {
|
||
var bc = r.buf,
|
||
pc = r.pos,
|
||
pctx = pc.ctx;
|
||
|
||
pctx.clearRect(0, 0, pc.w, pc.h);
|
||
|
||
if (!mp.au || isNaN(mp.au.duration) || isNaN(mp.au.currentTime))
|
||
return; // not-init || unsupp-codec
|
||
|
||
var sm = bc.w * 1.0 / mp.au.duration;
|
||
|
||
pctx.fillStyle = light ? 'rgba(0,64,0,0.15)' : 'rgba(204,255,128,0.15)';
|
||
for (var p = 1, mins = mp.au.duration / 10; p <= mins; p++)
|
||
pctx.fillRect(Math.floor(sm * p * 10), 0, 2, pc.h);
|
||
|
||
pctx.fillStyle = light ? 'rgba(0,64,0,0.5)' : 'rgba(192,255,96,0.5)';
|
||
for (var p = 1, mins = mp.au.duration / 60; p <= mins; p++)
|
||
pctx.fillRect(Math.floor(sm * p * 60), 0, 2, pc.h);
|
||
|
||
var w = 8,
|
||
x = sm * mp.au.currentTime;
|
||
|
||
pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, pc.h);
|
||
pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, 8, pc.h);
|
||
|
||
pctx.lineWidth = 2.5;
|
||
pctx.fillStyle = '#fff';
|
||
pctx.strokeStyle = 'rgba(24,56,0,0.4)';
|
||
pctx.font = '1em sans-serif';
|
||
|
||
var m = pctx.measureText.bind(pctx),
|
||
t1 = s2ms(mp.au.duration),
|
||
t2 = s2ms(mp.au.currentTime),
|
||
yt = pc.h / 3 * 2.1,
|
||
xt1 = pc.w - (m(t1).width + 12),
|
||
xt2 = x < pc.w / 2 ? (x + 12) : (Math.min(pc.w - m(t1 + ":88").width, x - 12) - m(t2).width);
|
||
|
||
pctx.strokeText(t1, xt1 + 1, yt + 1);
|
||
pctx.strokeText(t2, xt2 + 1, yt + 1);
|
||
pctx.strokeText(t1, xt1, yt);
|
||
pctx.strokeText(t2, xt2, yt);
|
||
pctx.fillText(t1, xt1, yt);
|
||
pctx.fillText(t2, xt2, yt);
|
||
};
|
||
|
||
window.addEventListener('resize', r.onresize);
|
||
r.onresize();
|
||
return r;
|
||
})();
|
||
|
||
|
||
// volume bar
|
||
var vbar = (function () {
|
||
var r = {},
|
||
gradh = -1,
|
||
can, ctx, w, h, grad1, grad2;
|
||
|
||
r.onresize = function () {
|
||
r.can = canvas_cfg(ebi('pvol'));
|
||
can = r.can.can;
|
||
ctx = r.can.ctx;
|
||
w = r.can.w;
|
||
h = r.can.h;
|
||
r.draw();
|
||
}
|
||
|
||
r.draw = function () {
|
||
var gh = h + '' + light;
|
||
if (gradh != gh) {
|
||
gradh = gh;
|
||
grad1 = glossy_grad(r.can, 50, light ? [50, 55, 52, 48] : [45, 52, 47, 43], light ? [54, 60, 52, 47] : [42, 51, 47, 42]);
|
||
grad2 = glossy_grad(r.can, 205, [10, 15, 13, 10], [16, 20, 18, 16]);
|
||
}
|
||
ctx.fillStyle = grad2; ctx.fillRect(0, 0, w, h);
|
||
ctx.fillStyle = grad1; ctx.fillRect(0, 0, w * mp.vol, h);
|
||
};
|
||
window.addEventListener('resize', r.onresize);
|
||
r.onresize();
|
||
|
||
var rect;
|
||
function mousedown(e) {
|
||
rect = can.getBoundingClientRect();
|
||
mousemove(e);
|
||
}
|
||
function mousemove(e) {
|
||
if (e.changedTouches && e.changedTouches.length > 0) {
|
||
e = e.changedTouches[0];
|
||
}
|
||
else if (e.buttons === 0) {
|
||
can.onmousemove = null;
|
||
return;
|
||
}
|
||
|
||
var x = e.clientX - rect.left,
|
||
mul = x * 1.0 / rect.width;
|
||
|
||
if (mul > 0.98)
|
||
mul = 1;
|
||
|
||
mp.setvol(mul);
|
||
r.draw();
|
||
}
|
||
can.onmousedown = function (e) {
|
||
if (e.button !== 0)
|
||
return;
|
||
|
||
can.onmousemove = mousemove;
|
||
mousedown(e);
|
||
};
|
||
can.onmouseup = function (e) {
|
||
if (e.button === 0)
|
||
can.onmousemove = null;
|
||
};
|
||
if (window.Touch) {
|
||
can.ontouchstart = mousedown;
|
||
can.ontouchmove = mousemove;
|
||
}
|
||
return r;
|
||
})();
|
||
|
||
|
||
function seek_au_mul(mul) {
|
||
if (mp.au)
|
||
seek_au_sec(mp.au.duration * mul);
|
||
}
|
||
|
||
function seek_au_rel(sec) {
|
||
if (mp.au)
|
||
seek_au_sec(mp.au.currentTime + sec);
|
||
}
|
||
|
||
function seek_au_sec(seek) {
|
||
if (!mp.au)
|
||
return;
|
||
|
||
console.log('seek: ' + seek);
|
||
if (!isFinite(seek))
|
||
return;
|
||
|
||
mp.au.currentTime = seek;
|
||
|
||
if (mp.au.paused)
|
||
mp.fade_in();
|
||
|
||
mpui.progress_updater();
|
||
}
|
||
|
||
|
||
function song_skip(n) {
|
||
var tid = null;
|
||
if (mp.au)
|
||
tid = mp.au.tid;
|
||
|
||
if (tid !== null)
|
||
play(mp.order.indexOf(tid) + n);
|
||
else
|
||
play(mp.order[n == -1 ? mp.order.length - 1 : 0]);
|
||
}
|
||
function next_song(e) {
|
||
ev(e);
|
||
return song_skip(1);
|
||
}
|
||
function prev_song(e) {
|
||
ev(e);
|
||
|
||
if (mp.au && !mp.au.paused && mp.au.currentTime > 3)
|
||
return seek_au_sec(0);
|
||
|
||
return song_skip(-1);
|
||
}
|
||
|
||
|
||
function playpause(e) {
|
||
// must be event-chain
|
||
ev(e);
|
||
if (mp.au) {
|
||
if (mp.au.paused)
|
||
mp.fade_in();
|
||
else
|
||
mp.fade_out();
|
||
|
||
mpui.progress_updater();
|
||
}
|
||
else
|
||
play(0, true);
|
||
|
||
mpl.pp();
|
||
};
|
||
|
||
|
||
// hook up the widget buttons
|
||
(function () {
|
||
ebi('bplay').onclick = playpause;
|
||
ebi('bprev').onclick = prev_song;
|
||
ebi('bnext').onclick = next_song;
|
||
|
||
var bar = ebi('barpos');
|
||
|
||
bar.onclick = function (e) {
|
||
if (!mp.au) {
|
||
play(0, true);
|
||
return mp.fade_in();
|
||
}
|
||
|
||
var rect = pbar.buf.can.getBoundingClientRect(),
|
||
x = e.clientX - rect.left;
|
||
|
||
seek_au_mul(x * 1.0 / rect.width);
|
||
};
|
||
|
||
if (!is_touch)
|
||
bar.onwheel = function (e) {
|
||
var dist = Math.sign(e.deltaY) * 10;
|
||
if (Math.abs(e.deltaY) < 30 && !e.deltaMode)
|
||
dist = e.deltaY;
|
||
|
||
if (!dist || !mp.au)
|
||
return true;
|
||
|
||
seek_au_rel(dist);
|
||
ev(e);
|
||
};
|
||
})();
|
||
|
||
|
||
// periodic tasks
|
||
var mpui = (function () {
|
||
var r = {},
|
||
nth = 0,
|
||
timeout = null,
|
||
preloaded = null;
|
||
|
||
r.progress_updater = function () {
|
||
clearTimeout(timeout);
|
||
|
||
if (!mp.au) {
|
||
widget.paused(true);
|
||
return;
|
||
}
|
||
|
||
// indicate playback state in ui
|
||
widget.paused(mp.au.paused);
|
||
|
||
if (++nth > 69) {
|
||
// android-chrome breaks aspect ratio with unannounced viewport changes
|
||
nth = 0;
|
||
if (is_touch) {
|
||
nth = 1;
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
}
|
||
}
|
||
else {
|
||
// draw current position in song
|
||
if (!mp.au.paused)
|
||
pbar.drawpos();
|
||
|
||
// occasionally draw buffered regions
|
||
if (++nth % 5 == 0)
|
||
pbar.drawbuf();
|
||
}
|
||
|
||
// preload next song
|
||
if (mpl.preload && preloaded != mp.au.src) {
|
||
var pos = mp.au.currentTime,
|
||
len = mp.au.duration;
|
||
|
||
if (pos > 0 && pos > len - 10) {
|
||
preloaded = mp.au.src;
|
||
try {
|
||
mp.preload(ebi(mp.order[mp.order.indexOf(mp.au.tid) + 1]).href);
|
||
}
|
||
catch (ex) {
|
||
console.log("preload failed", ex);
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!mp.au.paused)
|
||
timeout = setTimeout(r.progress_updater, 100);
|
||
};
|
||
r.progress_updater();
|
||
return r;
|
||
})();
|
||
|
||
|
||
// event from play button next to a file in the list
|
||
function ev_play(e) {
|
||
ev(e);
|
||
|
||
var fade = !mp.au || mp.au.paused;
|
||
play(this.getAttribute('id').slice(1), true);
|
||
if (fade)
|
||
mp.fade_in();
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
function setclass(id, clas) {
|
||
ebi(id).setAttribute('class', clas);
|
||
}
|
||
|
||
|
||
var need_ogv = true;
|
||
try {
|
||
need_ogv = new Audio().canPlayType('audio/ogg; codecs=opus') !== 'probably';
|
||
|
||
if (/ Edge\//.exec(navigator.userAgent + ''))
|
||
need_ogv = true;
|
||
}
|
||
catch (ex) { }
|
||
|
||
|
||
function need_ogv_for(url) {
|
||
return need_ogv && /\.(ogg|opus)$/i.test(url);
|
||
}
|
||
|
||
|
||
var audio_eq = (function () {
|
||
var r = {
|
||
"en": false,
|
||
"bands": [31.25, 62.5, 125, 250, 500, 1000, 2000, 4000, 8000, 16000],
|
||
"gains": [4, 3, 2, 1, 0, 0, 1, 2, 3, 4],
|
||
"filters": [],
|
||
"amp": 0,
|
||
"last_au": null
|
||
};
|
||
|
||
var cfg = [ // hz, q, g
|
||
[31.25 * 0.88, 0, 1.4], // shelf
|
||
[31.25 * 1.04, 0.7, 0.96], // peak
|
||
[62.5, 0.7, 1],
|
||
[125, 0.8, 1],
|
||
[250, 0.9, 1.03],
|
||
[500, 0.9, 1.1],
|
||
[1000, 0.9, 1.1],
|
||
[2000, 0.9, 1.105],
|
||
[4000, 0.88, 1.05],
|
||
[8000 * 1.006, 0.73, 1.24],
|
||
[16000 * 0.89, 0.7, 1.26], // peak
|
||
[16000 * 1.13, 0.82, 1.09], // peak
|
||
[16000 * 1.205, 0, 1.9] // shelf
|
||
];
|
||
|
||
try {
|
||
r.amp = fcfg_get('au_eq_amp', r.amp);
|
||
var gains = jread('au_eq_gain', r.gains);
|
||
if (r.gains.length == gains.length)
|
||
r.gains = gains;
|
||
}
|
||
catch (ex) { }
|
||
|
||
r.draw = function () {
|
||
jwrite('au_eq_gain', r.gains);
|
||
swrite('au_eq_amp', r.amp);
|
||
|
||
var txt = QSA('input.eq_gain');
|
||
for (var a = 0; a < r.bands.length; a++)
|
||
txt[a].value = r.gains[a];
|
||
|
||
QS('input.eq_gain[band="amp"]').value = r.amp;
|
||
};
|
||
|
||
r.apply = function () {
|
||
r.draw();
|
||
|
||
var Ctx = window.AudioContext || window.webkitAudioContext;
|
||
if (!Ctx)
|
||
bcfg_set('au_eq', false);
|
||
|
||
if (!Ctx || !mp.au)
|
||
return;
|
||
|
||
if (!r.en && !mp.ac)
|
||
return;
|
||
|
||
if (mp.ac) {
|
||
for (var a = 0; a < r.filters.length; a++)
|
||
r.filters[a].disconnect();
|
||
|
||
mp.acs.disconnect();
|
||
}
|
||
|
||
if (!mp.ac || mp.au != r.last_au) {
|
||
if (mp.ac)
|
||
mp.ac.close();
|
||
|
||
r.last_au = mp.au;
|
||
mp.ac = new Ctx();
|
||
mp.acs = mp.ac.createMediaElementSource(mp.au);
|
||
}
|
||
|
||
r.filters = [];
|
||
|
||
if (!r.en) {
|
||
mp.acs.connect(mp.ac.destination);
|
||
return;
|
||
}
|
||
|
||
var max = 0;
|
||
for (var a = 0; a < r.gains.length; a++)
|
||
if (max < r.gains[a])
|
||
max = r.gains[a];
|
||
|
||
var gains = []
|
||
for (var a = 0; a < r.gains.length; a++)
|
||
gains.push(r.gains[a] - max);
|
||
|
||
var t = gains[gains.length - 1];
|
||
gains.push(t);
|
||
gains.push(t);
|
||
gains.unshift(gains[0]);
|
||
|
||
for (var a = 0; a < cfg.length; a++) {
|
||
var fi = mp.ac.createBiquadFilter();
|
||
fi.frequency.value = cfg[a][0];
|
||
fi.gain.value = cfg[a][2] * gains[a];
|
||
fi.Q.value = cfg[a][1];
|
||
fi.type = a == 0 ? 'lowshelf' : a == cfg.length - 1 ? 'highshelf' : 'peaking';
|
||
r.filters.push(fi);
|
||
}
|
||
|
||
// pregain, keep first in chain
|
||
fi = mp.ac.createGain();
|
||
fi.gain.value = r.amp + 0.94; // +.137 dB measured; now -.25 dB and almost bitperfect
|
||
r.filters.push(fi);
|
||
|
||
for (var a = r.filters.length - 1; a >= 0; a--)
|
||
r.filters[a].connect(a > 0 ? r.filters[a - 1] : mp.ac.destination);
|
||
|
||
mp.acs.connect(r.filters[r.filters.length - 1]);
|
||
}
|
||
|
||
function eq_step(e) {
|
||
ev(e);
|
||
var band = parseInt(this.getAttribute('band')),
|
||
step = parseFloat(this.getAttribute('step'));
|
||
|
||
if (isNaN(band))
|
||
r.amp = Math.round((r.amp + step * 0.2) * 100) / 100;
|
||
else
|
||
r.gains[band] += step;
|
||
|
||
r.apply();
|
||
}
|
||
|
||
function adj_band(that, step) {
|
||
var err = false;
|
||
try {
|
||
var band = parseInt(that.getAttribute('band')),
|
||
vs = that.value,
|
||
v = parseFloat(vs);
|
||
|
||
if (isNaN(v) || v + '' != vs)
|
||
throw new Error('inval band');
|
||
|
||
if (isNaN(band))
|
||
r.amp = Math.round((v + step * 0.2) * 100) / 100;
|
||
else
|
||
r.gains[band] = v + step;
|
||
|
||
r.apply();
|
||
}
|
||
catch (ex) {
|
||
err = true;
|
||
}
|
||
clmod(that, 'err', err);
|
||
}
|
||
|
||
function eq_mod(e) {
|
||
ev(e);
|
||
adj_band(this, 0);
|
||
}
|
||
|
||
function eq_keydown(e) {
|
||
var step = e.key == 'ArrowUp' ? 0.25 : e.key == 'ArrowDown' ? -0.25 : 0;
|
||
if (step != 0)
|
||
adj_band(this, step);
|
||
}
|
||
|
||
var html = ['<table><tr><td rowspan="4">',
|
||
'<a id="au_eq" class="tgl btn" href="#" tt="enables the equalizer and gain control;$Nboost 0 = unmodified 100% volume">enable</a></td>'],
|
||
h2 = [], h3 = [], h4 = [];
|
||
|
||
var vs = [];
|
||
for (var a = 0; a < r.bands.length; a++) {
|
||
var hz = r.bands[a];
|
||
if (hz >= 1000)
|
||
hz = (hz / 1000) + 'k';
|
||
|
||
hz = (hz + '').split('.')[0];
|
||
vs.push([a, hz, r.gains[a]]);
|
||
}
|
||
vs.push(["amp", "boost", r.amp]);
|
||
|
||
for (var a = 0; a < vs.length; a++) {
|
||
var b = vs[a][0];
|
||
html.push('<td><a href="#" class="eq_step" step="0.5" band="' + b + '">+</a></td>');
|
||
h2.push('<td>' + vs[a][1] + '</td>');
|
||
h4.push('<td><a href="#" class="eq_step" step="-0.5" band="' + b + '">–</a></td>');
|
||
h3.push('<td><input type="text" class="eq_gain" band="' + b + '" value="' + vs[a][2] + '" /></td>');
|
||
}
|
||
html = html.join('\n') + '</tr><tr>';
|
||
html += h2.join('\n') + '</tr><tr>';
|
||
html += h3.join('\n') + '</tr><tr>';
|
||
html += h4.join('\n') + '</tr><table>';
|
||
ebi('audio_eq').innerHTML = html;
|
||
|
||
var stp = QSA('a.eq_step');
|
||
for (var a = 0, aa = stp.length; a < aa; a++)
|
||
stp[a].onclick = eq_step;
|
||
|
||
var txt = QSA('input.eq_gain');
|
||
for (var a = 0; a < txt.length; a++) {
|
||
txt[a].oninput = eq_mod;
|
||
txt[a].onkeydown = eq_keydown;
|
||
}
|
||
|
||
r.en = bcfg_get('au_eq', false);
|
||
ebi('au_eq').onclick = function (e) {
|
||
ev(e);
|
||
r.en = !r.en;
|
||
bcfg_set('au_eq', r.en);
|
||
r.apply();
|
||
};
|
||
|
||
r.draw();
|
||
return r;
|
||
})();
|
||
|
||
|
||
// plays the tid'th audio file on the page
|
||
function play(tid, is_ev, seek, call_depth) {
|
||
if (mp.order.length == 0)
|
||
return console.log('no audio found wait what');
|
||
|
||
mp.stopfade(true);
|
||
|
||
var tn = tid;
|
||
if ((tn + '').indexOf('f-') === 0) {
|
||
tn = mp.order.indexOf(tn);
|
||
if (tn < 0)
|
||
return;
|
||
}
|
||
|
||
if (tn >= mp.order.length) {
|
||
if (mpl.pb_mode == 'loop-folder') {
|
||
tn = 0;
|
||
}
|
||
else if (mpl.pb_mode == 'next-folder') {
|
||
treectl.ls_cb = next_song;
|
||
return tree_neigh(1);
|
||
}
|
||
}
|
||
|
||
if (tn < 0) {
|
||
if (mpl.pb_mode == 'loop-folder') {
|
||
tn = mp.order.length - 1;
|
||
}
|
||
else if (mpl.pb_mode == 'next-folder') {
|
||
treectl.ls_cb = prev_song;
|
||
return tree_neigh(-1);
|
||
}
|
||
}
|
||
|
||
tid = mp.order[tn];
|
||
|
||
if (mp.au) {
|
||
mp.au.pause();
|
||
setclass('a' + mp.au.tid, 'play');
|
||
}
|
||
|
||
// ogv.js breaks on .play() unless directly user-triggered
|
||
var attempt_play = true;
|
||
|
||
var url = mp.tracks[tid];
|
||
if (need_ogv_for(url)) {
|
||
if (mp.au_ogvjs) {
|
||
mp.au = mp.au_ogvjs;
|
||
}
|
||
else if (window['OGVPlayer']) {
|
||
mp.au = mp.au_ogvjs = new OGVPlayer();
|
||
attempt_play = is_ev;
|
||
mp.au.addEventListener('error', evau_error, true);
|
||
mp.au.addEventListener('progress', pbar.drawpos);
|
||
mp.au.addEventListener('ended', next_song);
|
||
widget.open();
|
||
}
|
||
else {
|
||
if (call_depth !== undefined)
|
||
return alert('failed to load ogv.js');
|
||
|
||
show_modal('<h1>loading ogv.js</h1><h2>thanks apple</h2>');
|
||
|
||
import_js('/.cpr/deps/ogv.js', function () {
|
||
play(tid, false, seek, 1);
|
||
});
|
||
|
||
return;
|
||
}
|
||
}
|
||
else {
|
||
if (!mp.au_native) {
|
||
mp.au = mp.au_native = new Audio();
|
||
mp.au.addEventListener('error', evau_error, true);
|
||
mp.au.addEventListener('progress', pbar.drawpos);
|
||
mp.au.addEventListener('ended', next_song);
|
||
widget.open();
|
||
}
|
||
mp.au = mp.au_native;
|
||
}
|
||
|
||
audio_eq.apply();
|
||
|
||
mp.au.tid = tid;
|
||
mp.au.src = url + (url.indexOf('?') < 0 ? '?cache' : '&cache');
|
||
mp.au.volume = mp.expvol(mp.vol);
|
||
var oid = 'a' + tid;
|
||
setclass(oid, 'play act');
|
||
var trs = ebi('files').getElementsByTagName('tbody')[0].getElementsByTagName('tr');
|
||
for (var a = 0, aa = trs.length; a < aa; a++) {
|
||
clmod(trs[a], 'play');
|
||
}
|
||
ebi(oid).parentElement.parentElement.className += ' play';
|
||
clmod(ebi('wtoggle'), 'np', mpl.clip);
|
||
if (window.thegrid)
|
||
thegrid.loadsel();
|
||
|
||
try {
|
||
if (attempt_play)
|
||
mp.au.play();
|
||
|
||
if (mp.au.paused)
|
||
autoplay_blocked(seek);
|
||
else if (seek) {
|
||
seek_au_sec(seek);
|
||
}
|
||
|
||
if (!seek) {
|
||
var o = ebi(oid);
|
||
o.setAttribute('id', 'thx_js');
|
||
if (window.history && history.replaceState) {
|
||
hist_replace(document.location.pathname + '#' + oid);
|
||
}
|
||
else {
|
||
document.location.hash = oid;
|
||
}
|
||
o.setAttribute('id', oid);
|
||
}
|
||
|
||
mpui.progress_updater();
|
||
pbar.drawbuf();
|
||
mpl.announce();
|
||
return true;
|
||
}
|
||
catch (ex) {
|
||
alert('playback failed: ' + ex);
|
||
}
|
||
setclass(oid, 'play');
|
||
setTimeout(next_song, 500);
|
||
}
|
||
|
||
|
||
// event from the audio object if something breaks
|
||
function evau_error(e) {
|
||
var err = '',
|
||
eplaya = (e && e.target) || (window.event && window.event.srcElement);
|
||
|
||
switch (eplaya.error.code) {
|
||
case eplaya.error.MEDIA_ERR_ABORTED:
|
||
err = "You aborted the playback attempt (how tho)";
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_NETWORK:
|
||
err = "Your internet connection is wonky";
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_DECODE:
|
||
err = "This file is supposedly corrupted??";
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_SRC_NOT_SUPPORTED:
|
||
err = "Your browser does not understand this audio format";
|
||
break;
|
||
default:
|
||
err = 'Unknown Errol';
|
||
break;
|
||
}
|
||
if (eplaya.error.message)
|
||
err += '\n\n' + eplaya.error.message;
|
||
|
||
err += '\n\nFile: «' + uricom_dec(eplaya.src.split('/').slice(-1)[0])[0] + '»';
|
||
|
||
alert(err);
|
||
}
|
||
|
||
|
||
// show a fullscreen message
|
||
function show_modal(html) {
|
||
var body = document.body || document.getElementsByTagName('body')[0],
|
||
div = mknod('div');
|
||
|
||
div.setAttribute('id', 'blocked');
|
||
div.innerHTML = html;
|
||
unblocked();
|
||
body.appendChild(div);
|
||
}
|
||
|
||
|
||
// hide fullscreen message
|
||
function unblocked(e) {
|
||
ev(e);
|
||
var dom = ebi('blocked');
|
||
if (dom)
|
||
dom.parentNode.removeChild(dom);
|
||
}
|
||
|
||
|
||
// show ui to manually start playback of a linked song
|
||
function autoplay_blocked(seek) {
|
||
show_modal(
|
||
'<div id="blk_play"><a href="#" id="blk_go"></a></div>' +
|
||
'<div id="blk_abrt"><a href="#" id="blk_na">Cancel<br />(show file list)</a></div>');
|
||
|
||
var go = ebi('blk_go'),
|
||
na = ebi('blk_na'),
|
||
tid = mp.au.tid,
|
||
fn = mp.tracks[tid].split(/\//).pop();
|
||
|
||
fn = uricom_dec(fn.replace(/\+/g, ' '))[0];
|
||
|
||
go.textContent = 'Play "' + fn + '"';
|
||
go.onclick = function (e) {
|
||
unblocked(e);
|
||
// chrome 91 may permanently taint on a failed play()
|
||
// depending on win10 settings or something? idk
|
||
mp.au_native = mp.au_ogvjs = null;
|
||
play(tid, true, seek);
|
||
mp.fade_in();
|
||
};
|
||
na.onclick = unblocked;
|
||
}
|
||
|
||
|
||
function play_linked() {
|
||
var v = location.hash;
|
||
if (v && v.indexOf('#af-') === 0) {
|
||
var id = v.slice(2).split('&');
|
||
if (id[0].length != 10)
|
||
return;
|
||
|
||
if (id.length == 1)
|
||
return play(id[0]);
|
||
|
||
var m = /^[Tt=0]*([0-9]+[Mm:])?0*([0-9]+)[Ss]?$/.exec(id[1]);
|
||
if (!m)
|
||
return play(id[0]);
|
||
|
||
return play(id[0], false, parseInt(m[1] || 0) * 60 + parseInt(m[2] || 0));
|
||
}
|
||
};
|
||
|
||
|
||
(function () {
|
||
var d = mknod('div');
|
||
d.setAttribute('id', 'acc_info');
|
||
document.body.insertBefore(d, ebi('ops'));
|
||
})();
|
||
|
||
|
||
var fileman = (function () {
|
||
var bren = ebi('fren'),
|
||
bdel = ebi('fdel'),
|
||
bcut = ebi('fcut'),
|
||
bpst = ebi('fpst'),
|
||
r = {};
|
||
|
||
r.clip = null;
|
||
r.bus = new BroadcastChannel("fileman_bus");
|
||
|
||
r.render = function () {
|
||
if (r.clip === null)
|
||
r.clip = jread('fman_clip', []);
|
||
|
||
var sel = msel.getsel();
|
||
clmod(bren, 'en', sel.length == 1);
|
||
clmod(bdel, 'en', sel.length);
|
||
clmod(bcut, 'en', sel.length);
|
||
clmod(bpst, 'en', r.clip && r.clip.length);
|
||
bren.style.display = has(perms, 'write') && has(perms, 'move') ? '' : 'none';
|
||
bdel.style.display = has(perms, 'delete') ? '' : 'none';
|
||
bcut.style.display = has(perms, 'move') ? '' : 'none';
|
||
bpst.style.display = has(perms, 'write') ? '' : 'none';
|
||
bpst.setAttribute('tt', 'paste ' + r.clip.length + ' items$NHotkey: ctrl-V');
|
||
};
|
||
|
||
r.rename = function (e) {
|
||
ev(e);
|
||
var sel = msel.getsel();
|
||
if (sel.length !== 1)
|
||
return alert('select exactly 1 item to rename');
|
||
|
||
var fn = prompt('new filename:', sel[0]);
|
||
if (!fn || fn == sel[0])
|
||
return alert('aborted');
|
||
};
|
||
|
||
r.delete = function (e) {
|
||
ev(e);
|
||
var sel = msel.getsel(),
|
||
vps = [];
|
||
|
||
for (var a = 0; a < sel.length; a++)
|
||
vps.push(sel[a].vp);
|
||
|
||
if (!sel.length)
|
||
return alert('select at least 1 item to delete');
|
||
|
||
if (!confirm('===== DANGER =====\nDELETE these ' + vps.length + ' items?\n\n' + vps.join('\n')))
|
||
return;
|
||
|
||
if (!confirm('Last chance! Delete?'))
|
||
return;
|
||
|
||
function deleter() {
|
||
var xhr = new XMLHttpRequest(),
|
||
vp = vps.shift();
|
||
|
||
if (!vp) {
|
||
toast.ok(2000, 'delete OK');
|
||
treectl.goto(get_evpath());
|
||
return;
|
||
}
|
||
toast.inf(2000, 'deleting ' + (vps.length + 1) + ' items<br /><br />' + vp);
|
||
|
||
xhr.open('GET', vp + '?delete', true);
|
||
xhr.onreadystatechange = delete_cb;
|
||
xhr.send();
|
||
}
|
||
function delete_cb() {
|
||
if (this.readyState != XMLHttpRequest.DONE)
|
||
return;
|
||
|
||
if (this.status !== 200) {
|
||
var msg = this.responseText;
|
||
toast.err(2000, 'delete failed:<br />' + msg);
|
||
return;
|
||
}
|
||
deleter();
|
||
}
|
||
deleter();
|
||
};
|
||
|
||
r.cut = function (e) {
|
||
ev(e);
|
||
var sel = msel.getsel(),
|
||
vps = [];
|
||
|
||
if (!sel.length)
|
||
return alert('select at least 1 item to cut');
|
||
|
||
for (var a = 0; a < sel.length; a++) {
|
||
vps.push(sel[a].vp);
|
||
var cl = ebi(sel[a].id).closest('tr').classList,
|
||
inv = cl.contains('c1');
|
||
|
||
cl.remove(inv ? 'c1' : 'c2');
|
||
cl.add(inv ? 'c2' : 'c1');
|
||
}
|
||
|
||
toast.inf(1000, 'cut ' + sel.length + ' items');
|
||
jwrite('fman_clip', vps);
|
||
r.tx();
|
||
};
|
||
|
||
r.paste = function (e) {
|
||
ev(e);
|
||
if (!r.clip.length)
|
||
return alert('first cut some items to paste\n\nnote: you can cut/paste across different browser tabs');
|
||
|
||
var req = [],
|
||
exists = [],
|
||
indir = [],
|
||
links = QSA('#files tbody td:nth-child(2) a');
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++)
|
||
indir.push(links[a].getAttribute('name'));
|
||
|
||
for (var a = 0; a < r.clip.length; a++) {
|
||
var found = false;
|
||
for (var b = 0; b < indir.length; b++) {
|
||
if (r.clip[a].endsWith('/' + indir[b])) {
|
||
exists.push(r.clip[a]);
|
||
found = true;
|
||
}
|
||
}
|
||
if (!found)
|
||
req.push(r.clip[a]);
|
||
}
|
||
|
||
if (exists.length)
|
||
alert('these ' + exists.length + ' items cannot be pasted here (names already exist):\n\n' + exists.join('\n'));
|
||
|
||
if (!req.length)
|
||
return;
|
||
|
||
if (!confirm('paste these ' + req.length + ' items here?\n\n' + req.join('\n')))
|
||
return;
|
||
|
||
function paster() {
|
||
var xhr = new XMLHttpRequest(),
|
||
vp = req.shift();
|
||
|
||
if (!vp) {
|
||
toast.ok(2000, 'paste OK');
|
||
treectl.goto(get_evpath());
|
||
return;
|
||
}
|
||
toast.inf(2000, 'pasting ' + (req.length + 1) + ' items<br /><br />' + vp);
|
||
|
||
var dst = get_evpath() + vp.split('/').slice(-1)[0];
|
||
|
||
xhr.open('GET', vp + '?move=' + dst, true);
|
||
xhr.onreadystatechange = paste_cb;
|
||
xhr.send();
|
||
}
|
||
function paste_cb() {
|
||
if (this.readyState != XMLHttpRequest.DONE)
|
||
return;
|
||
|
||
if (this.status !== 200) {
|
||
var msg = this.responseText;
|
||
toast.err(2000, 'paste failed:<br />' + msg);
|
||
return;
|
||
}
|
||
paster();
|
||
}
|
||
paster();
|
||
|
||
jwrite('fman_clip', []);
|
||
r.tx();
|
||
};
|
||
|
||
r.bus.onmessage = function () {
|
||
console.log('fman onmsg');
|
||
r.clip = null;
|
||
r.render();
|
||
};
|
||
|
||
r.tx = function () {
|
||
r.bus.postMessage(1);
|
||
r.bus.onmessage();
|
||
};
|
||
|
||
bren.onclick = r.rename;
|
||
bdel.onclick = r.delete;
|
||
bcut.onclick = r.cut;
|
||
bpst.onclick = r.paste;
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
var thegrid = (function () {
|
||
var lfiles = ebi('files'),
|
||
gfiles = mknod('div');
|
||
|
||
gfiles.setAttribute('id', 'gfiles');
|
||
gfiles.style.display = 'none';
|
||
gfiles.innerHTML = (
|
||
'<div id="ghead">' +
|
||
'<a href="#" class="tgl btn" id="gridsel" tt="enable file selection; ctrl-click a file to override$NHotkey: S">multiselect</a> zoom ' +
|
||
'<a href="#" class="btn" z="-1.2" tt="Hotkey: shift-A">–</a> ' +
|
||
'<a href="#" class="btn" z="1.2" tt="Hotkey: shift-D">+</a> sort by: ' +
|
||
'<a href="#" s="href">name</a>, ' +
|
||
'<a href="#" s="sz">size</a>, ' +
|
||
'<a href="#" s="ts">date</a>, ' +
|
||
'<a href="#" s="ext">type</a>' +
|
||
'</div>' +
|
||
'<div id="ggrid"></div>'
|
||
);
|
||
lfiles.parentNode.insertBefore(gfiles, lfiles);
|
||
|
||
var r = {
|
||
'thumbs': bcfg_get('thumbs', true),
|
||
'en': bcfg_get('griden', false),
|
||
'sel': bcfg_get('gridsel', false),
|
||
'sz': fcfg_get('gridsz', 10),
|
||
'isdirty': true,
|
||
'bbox': null
|
||
};
|
||
|
||
ebi('thumbs').onclick = function (e) {
|
||
ev(e);
|
||
r.thumbs = !r.thumbs;
|
||
bcfg_set('thumbs', r.thumbs);
|
||
r.setdirty();
|
||
};
|
||
|
||
ebi('griden').onclick = ebi('wtgrid').onclick = function (e) {
|
||
ev(e);
|
||
r.en = !r.en;
|
||
bcfg_set('griden', r.en);
|
||
if (r.en) {
|
||
loadgrid();
|
||
}
|
||
else {
|
||
lfiles.style.display = '';
|
||
gfiles.style.display = 'none';
|
||
}
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
};
|
||
|
||
var btnclick = function (e) {
|
||
ev(e);
|
||
var s = this.getAttribute('s'),
|
||
z = this.getAttribute('z');
|
||
|
||
if (z)
|
||
return setsz(z > 0 ? r.sz * z : r.sz / (-z));
|
||
|
||
var t = lfiles.tHead.rows[0].cells;
|
||
for (var a = 0; a < t.length; a++)
|
||
if (t[a].getAttribute('name') == s) {
|
||
t[a].click();
|
||
break;
|
||
}
|
||
|
||
r.setdirty();
|
||
};
|
||
|
||
var links = QSA('#ghead>a');
|
||
for (var a = 0; a < links.length; a++)
|
||
links[a].onclick = btnclick;
|
||
|
||
ebi('gridsel').onclick = function (e) {
|
||
ev(e);
|
||
r.sel = !r.sel;
|
||
bcfg_set('gridsel', r.sel);
|
||
r.loadsel();
|
||
};
|
||
|
||
r.setvis = function (vis) {
|
||
(r.en ? gfiles : lfiles).style.display = vis ? '' : 'none';
|
||
}
|
||
|
||
r.setdirty = function () {
|
||
r.dirty = true;
|
||
if (r.en) {
|
||
loadgrid();
|
||
}
|
||
}
|
||
|
||
function setsz(v) {
|
||
if (v !== undefined) {
|
||
r.sz = v;
|
||
swrite('gridsz', r.sz);
|
||
}
|
||
try {
|
||
document.documentElement.style.setProperty('--grid-sz', r.sz + 'em');
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
setsz();
|
||
|
||
function gclick(e) {
|
||
if (ctrl(e))
|
||
return true;
|
||
|
||
var oth = ebi(this.getAttribute('ref')),
|
||
href = this.getAttribute('href'),
|
||
aplay = ebi('a' + oth.getAttribute('id')),
|
||
is_img = /\.(gif|jpe?g|png|webp)(\?|$)/i.test(href),
|
||
in_tree = null,
|
||
have_sel = QS('#files tr.sel'),
|
||
td = oth.closest('td').nextSibling,
|
||
tr = td.parentNode;
|
||
|
||
if (/\/(\?|$)/.test(href)) {
|
||
var ta = QSA('#treeul a.hl+ul>li>a+a'),
|
||
txt = oth.textContent.slice(0, -1);
|
||
|
||
for (var a = 0, aa = ta.length; a < aa; a++) {
|
||
if (ta[a].textContent == txt) {
|
||
in_tree = ta[a];
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (r.sel) {
|
||
td.click();
|
||
this.setAttribute('class', tr.getAttribute('class'));
|
||
}
|
||
else if (widget.is_open && aplay)
|
||
aplay.click();
|
||
|
||
else if (in_tree && !have_sel)
|
||
in_tree.click();
|
||
|
||
else if (!is_img && have_sel)
|
||
window.open(href, '_blank');
|
||
|
||
else return true;
|
||
ev(e);
|
||
}
|
||
|
||
r.loadsel = function () {
|
||
if (r.dirty)
|
||
return;
|
||
|
||
var ths = QSA('#ggrid>a');
|
||
|
||
for (var a = 0, aa = ths.length; a < aa; a++) {
|
||
var tr = ebi(ths[a].getAttribute('ref')).closest('tr');
|
||
ths[a].setAttribute('class', tr.getAttribute('class'));
|
||
}
|
||
var uns = QS('#ggrid a[ref="unsearch"]');
|
||
if (uns)
|
||
uns.onclick = function () {
|
||
ebi('unsearch').click();
|
||
};
|
||
}
|
||
|
||
function loadgrid() {
|
||
if (have_webp === null)
|
||
return setTimeout(loadgrid, 50);
|
||
|
||
lfiles.style.display = 'none';
|
||
gfiles.style.display = 'block';
|
||
|
||
if (!r.dirty)
|
||
return r.loadsel();
|
||
|
||
var html = [];
|
||
var files = QSA('#files>tbody>tr>td:nth-child(2) a[id]');
|
||
for (var a = 0, aa = files.length; a < aa; a++) {
|
||
var ao = files[a],
|
||
href = esc(ao.getAttribute('href')),
|
||
ref = ao.getAttribute('id'),
|
||
isdir = href.split('?')[0].slice(-1)[0] == '/',
|
||
ac = isdir ? ' class="dir"' : '',
|
||
ihref = href;
|
||
|
||
if (r.thumbs) {
|
||
ihref += (ihref.indexOf('?') === -1 ? '?' : '&') + 'th=' + (have_webp ? 'w' : 'j');
|
||
if (href == "#")
|
||
ihref = '/.cpr/ico/⏏️';
|
||
}
|
||
else if (isdir) {
|
||
ihref = '/.cpr/ico/folder';
|
||
}
|
||
else {
|
||
var ar = href.split('?')[0].split('.');
|
||
if (ar.length > 1)
|
||
ar = ar.slice(1);
|
||
|
||
ihref = '';
|
||
ar.reverse();
|
||
for (var b = 0; b < ar.length; b++) {
|
||
if (ar[b].length > 7)
|
||
break;
|
||
|
||
ihref = ar[b] + '.' + ihref;
|
||
}
|
||
if (!ihref) {
|
||
ihref = 'unk.';
|
||
}
|
||
ihref = '/.cpr/ico/' + ihref.slice(0, -1);
|
||
}
|
||
|
||
html.push('<a href="' + href + '" ref="' + ref + '"><img src="' +
|
||
ihref + '" /><span' + ac + '>' + ao.innerHTML + '</span></a>');
|
||
}
|
||
ebi('ggrid').innerHTML = html.join('\n');
|
||
|
||
var ths = QSA('#ggrid>a');
|
||
for (var a = 0, aa = ths.length; a < aa; a++)
|
||
ths[a].onclick = gclick;
|
||
|
||
r.dirty = false;
|
||
r.bagit();
|
||
r.loadsel();
|
||
}
|
||
|
||
r.bagit = function () {
|
||
if (!window.baguetteBox)
|
||
return;
|
||
|
||
if (r.bbox)
|
||
baguetteBox.destroy();
|
||
|
||
r.bbox = baguetteBox.run('#ggrid', {
|
||
captions: function (g) {
|
||
var idx = -1,
|
||
h = '' + g;
|
||
|
||
for (var a = 0; a < r.bbox.length; a++)
|
||
if (r.bbox[a].imageElement == g)
|
||
idx = a;
|
||
|
||
return '<a download href="' + h +
|
||
'">' + (idx + 1) + ' / ' + r.bbox.length + ' -- ' +
|
||
esc(uricom_dec(h.split('/').slice(-1)[0])[0]) + '</a>';
|
||
}
|
||
})[0];
|
||
};
|
||
|
||
setTimeout(function () {
|
||
import_js('/.cpr/baguettebox.js', r.bagit);
|
||
}, 1);
|
||
|
||
if (r.en) {
|
||
loadgrid();
|
||
}
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
function tree_scrollto() {
|
||
var act = QS('#treeul a.hl'),
|
||
ul = act ? act.offsetParent : null;
|
||
|
||
if (!ul)
|
||
return;
|
||
|
||
var ctr = ebi('tree'),
|
||
em = parseFloat(getComputedStyle(act).fontSize),
|
||
top = act.offsetTop + ul.offsetTop,
|
||
min = top - 11 * em,
|
||
max = top - (ctr.offsetHeight - 10 * em);
|
||
|
||
if (ctr.scrollTop > min)
|
||
ctr.scrollTop = Math.floor(min);
|
||
else if (ctr.scrollTop < max)
|
||
ctr.scrollTop = Math.floor(max);
|
||
}
|
||
|
||
|
||
function tree_neigh(n) {
|
||
var links = QSA('#treeul li>a+a');
|
||
if (!links.length) {
|
||
treectl.dir_cb = function () {
|
||
tree_neigh(n);
|
||
};
|
||
treectl.entree();
|
||
return;
|
||
}
|
||
var act = -1;
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
if (links[a].getAttribute('class') == 'hl') {
|
||
act = a;
|
||
break;
|
||
}
|
||
}
|
||
if (act == -1)
|
||
return;
|
||
|
||
act += n;
|
||
if (act < 0)
|
||
act = links.length - 1;
|
||
if (act >= links.length)
|
||
act = 0;
|
||
|
||
treectl.dir_cb = tree_scrollto;
|
||
links[act].click();
|
||
}
|
||
|
||
|
||
function tree_up() {
|
||
var act = QS('#treeul a.hl');
|
||
if (!act) {
|
||
treectl.dir_cb = tree_up;
|
||
treectl.entree();
|
||
return;
|
||
}
|
||
if (act.previousSibling.textContent == '-')
|
||
return act.previousSibling.click();
|
||
|
||
act.parentNode.parentNode.parentNode.getElementsByTagName('a')[1].click();
|
||
}
|
||
|
||
|
||
document.onkeydown = function (e) {
|
||
if (!document.activeElement || document.activeElement != document.body && document.activeElement.nodeName.toLowerCase() != 'a')
|
||
return;
|
||
|
||
if (e.altKey || e.isComposing)
|
||
return;
|
||
|
||
if (QS('#bbox-overlay.visible'))
|
||
return;
|
||
|
||
var k = e.code + '', pos = -1, n;
|
||
|
||
if (ctrl(e)) {
|
||
if (k == 'KeyX')
|
||
return fileman.cut();
|
||
|
||
if (k == 'KeyV')
|
||
return fileman.paste();
|
||
|
||
if (k == 'KeyK')
|
||
return fileman.delete();
|
||
|
||
return;
|
||
}
|
||
|
||
if (e.shiftKey && k != 'KeyA' && k != 'KeyD')
|
||
return;
|
||
|
||
if (k.indexOf('Digit') === 0)
|
||
pos = parseInt(k.slice(-1)) * 0.1;
|
||
|
||
if (pos !== -1)
|
||
return seek_au_mul(pos) || true;
|
||
|
||
if (k == 'KeyJ')
|
||
return prev_song() || true;
|
||
|
||
if (k == 'KeyL')
|
||
return next_song() || true;
|
||
|
||
if (k == 'KeyP')
|
||
return playpause() || true;
|
||
|
||
n = k == 'KeyU' ? -10 : k == 'KeyO' ? 10 : 0;
|
||
if (n !== 0)
|
||
return seek_au_rel(n) || true;
|
||
|
||
n = k == 'KeyI' ? -1 : k == 'KeyK' ? 1 : 0;
|
||
if (n !== 0)
|
||
return tree_neigh(n);
|
||
|
||
if (k == 'KeyM')
|
||
return tree_up();
|
||
|
||
if (k == 'KeyB')
|
||
return treectl.hidden ? treectl.entree() : treectl.detree();
|
||
|
||
if (k == 'KeyG')
|
||
return ebi('griden').click();
|
||
|
||
if (k == 'KeyT')
|
||
return ebi('thumbs').click();
|
||
|
||
if (k == 'F2')
|
||
return fileman.rename();
|
||
|
||
if (!treectl.hidden && (!e.shiftKey || !thegrid.en)) {
|
||
if (k == 'KeyA')
|
||
return QS('#twig').click();
|
||
|
||
if (k == 'KeyD')
|
||
return QS('#twobytwo').click();
|
||
}
|
||
|
||
if (thegrid.en) {
|
||
if (k == 'KeyS')
|
||
return ebi('gridsel').click();
|
||
|
||
if (k == 'KeyA')
|
||
return QSA('#ghead>a[z]')[0].click();
|
||
|
||
if (k == 'KeyD')
|
||
return QSA('#ghead>a[z]')[1].click();
|
||
}
|
||
};
|
||
|
||
|
||
// search
|
||
(function () {
|
||
var sconf = [
|
||
["size",
|
||
["szl", "sz_min", "minimum MiB", ""],
|
||
["szu", "sz_max", "maximum MiB", ""]
|
||
],
|
||
["date",
|
||
["dtl", "dt_min", "min. iso8601", ""],
|
||
["dtu", "dt_max", "max. iso8601", ""]
|
||
],
|
||
["path",
|
||
["path", "path", "path contains (space-separated)", "46"]
|
||
],
|
||
["name",
|
||
["name", "name", "name contains (negate with -nope)", "46"]
|
||
]
|
||
];
|
||
var oldcfg = [];
|
||
|
||
if (QS('#srch_form.tags')) {
|
||
sconf.push(["tags",
|
||
["tags", "tags", "tags contains (^=start, end=$)", "46"]
|
||
]);
|
||
sconf.push(["adv.",
|
||
["adv", "adv", "key>=1A key<=2B .bpm>165", "46"]
|
||
]);
|
||
}
|
||
|
||
var trs = [],
|
||
orig_url = null,
|
||
orig_html = null;
|
||
|
||
for (var a = 0; a < sconf.length; a++) {
|
||
var html = ['<tr><td><br />' + sconf[a][0] + '</td>'];
|
||
for (var b = 1; b < 3; b++) {
|
||
var hn = "srch_" + sconf[a][b][0],
|
||
csp = (sconf[a].length == 2) ? 2 : 1;
|
||
|
||
html.push(
|
||
'<td colspan="' + csp + '"><input id="' + hn + 'c" type="checkbox">\n' +
|
||
'<label for="' + hn + 'c">' + sconf[a][b][2] + '</label>\n' +
|
||
'<br /><input id="' + hn + 'v" type="text" size="' + sconf[a][b][3] +
|
||
'" name="' + sconf[a][b][1] + '" /></td>');
|
||
if (csp == 2)
|
||
break;
|
||
}
|
||
html.push('</tr>');
|
||
trs.push(html);
|
||
}
|
||
var html = [];
|
||
for (var a = 0; a < trs.length; a += 2) {
|
||
html.push('<table>' + (trs[a].concat(trs[a + 1])).join('\n') + '</table>');
|
||
}
|
||
html.push('<table id="tq_raw"><tr><td>raw</td><td><input id="q_raw" type="text" name="q" /></td></tr></table>');
|
||
ebi('srch_form').innerHTML = html.join('\n');
|
||
|
||
var o = QSA('#op_search input');
|
||
for (var a = 0; a < o.length; a++) {
|
||
o[a].oninput = ev_search_input;
|
||
}
|
||
|
||
function srch_msg(err, txt) {
|
||
var o = ebi('srch_q');
|
||
o.textContent = txt;
|
||
o.style.color = err ? '#f09' : '#c90';
|
||
}
|
||
|
||
var search_timeout,
|
||
defer_timeout,
|
||
search_in_progress = 0;
|
||
|
||
function ev_search_input() {
|
||
var v = this.value,
|
||
id = this.getAttribute('id');
|
||
|
||
if (id.slice(-1) == 'v') {
|
||
var chk = ebi(id.slice(0, -1) + 'c');
|
||
chk.checked = ((v + '').length > 0);
|
||
}
|
||
|
||
if (id != "q_raw")
|
||
encode_query();
|
||
|
||
set_vq();
|
||
|
||
clearTimeout(defer_timeout);
|
||
defer_timeout = setTimeout(try_search, 2000);
|
||
try_search();
|
||
}
|
||
|
||
function try_search() {
|
||
if (Date.now() - search_in_progress > 30 * 1000) {
|
||
clearTimeout(defer_timeout);
|
||
clearTimeout(search_timeout);
|
||
search_timeout = setTimeout(do_search, 200);
|
||
}
|
||
}
|
||
|
||
function set_vq() {
|
||
if (search_in_progress)
|
||
return;
|
||
|
||
var q = ebi('q_raw').value,
|
||
vq = ebi('files').getAttribute('q_raw');
|
||
|
||
srch_msg(false, (q == vq) ? '' : 'search results below are from a previous query:\n ' + (vq ? vq : '(*)'));
|
||
}
|
||
|
||
function encode_query() {
|
||
var q = '';
|
||
for (var a = 0; a < sconf.length; a++) {
|
||
for (var b = 1; b < sconf[a].length; b++) {
|
||
var k = sconf[a][b][0],
|
||
chk = 'srch_' + k + 'c',
|
||
tvs = ebi('srch_' + k + 'v').value.split(/ +/g);
|
||
|
||
if (!ebi(chk).checked)
|
||
continue;
|
||
|
||
for (var c = 0; c < tvs.length; c++) {
|
||
var tv = tvs[c];
|
||
if (!tv.length)
|
||
break;
|
||
|
||
q += ' and ';
|
||
|
||
if (k == 'adv') {
|
||
q += tv.replace(/ +/g, " and ").replace(/([=!><]=?)/, " $1 ");
|
||
continue;
|
||
}
|
||
|
||
if (k.length == 3) {
|
||
q += k.replace(/sz/, 'size').replace(/dt/, 'date').replace(/l$/, ' >= ').replace(/u$/, ' <= ') + tv;
|
||
continue;
|
||
}
|
||
|
||
if (k == 'path' || k == 'name' || k == 'tags') {
|
||
var not = ' ';
|
||
if (tv.slice(0, 1) == '-') {
|
||
tv = tv.slice(1);
|
||
not = ' not ';
|
||
}
|
||
|
||
if (tv.slice(0, 1) == '^') {
|
||
tv = tv.slice(1);
|
||
}
|
||
else {
|
||
tv = '*' + tv;
|
||
}
|
||
|
||
if (tv.slice(-1) == '$') {
|
||
tv = tv.slice(0, -1);
|
||
}
|
||
else {
|
||
tv += '*';
|
||
}
|
||
|
||
q += k + not + 'like ' + tv;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
ebi('q_raw').value = q.slice(5);
|
||
}
|
||
|
||
function do_search() {
|
||
search_in_progress = Date.now();
|
||
srch_msg(false, "searching...");
|
||
clearTimeout(search_timeout);
|
||
|
||
var xhr = new XMLHttpRequest();
|
||
xhr.open('POST', '/?srch', true);
|
||
xhr.setRequestHeader('Content-Type', 'text/plain');
|
||
xhr.onreadystatechange = xhr_search_results;
|
||
xhr.ts = Date.now();
|
||
xhr.q_raw = ebi('q_raw').value;
|
||
xhr.send(JSON.stringify({ "q": xhr.q_raw }));
|
||
}
|
||
|
||
function xhr_search_results() {
|
||
if (this.readyState != XMLHttpRequest.DONE)
|
||
return;
|
||
|
||
if (this.status !== 200) {
|
||
var msg = this.responseText;
|
||
if (msg.indexOf('<pre>') === 0)
|
||
msg = msg.slice(5);
|
||
|
||
srch_msg(true, "http " + this.status + ": " + msg);
|
||
search_in_progress = 0;
|
||
return;
|
||
}
|
||
search_in_progress = 0;
|
||
srch_msg(false, '');
|
||
|
||
var res = JSON.parse(this.responseText),
|
||
tagord = res.tag_order;
|
||
|
||
sortfiles(res.hits);
|
||
|
||
var ofiles = ebi('files');
|
||
if (ofiles.getAttribute('ts') > this.ts)
|
||
return;
|
||
|
||
treectl.hide();
|
||
|
||
var html = mk_files_header(tagord);
|
||
html.push('<tbody>');
|
||
html.push('<tr><td>-</td><td colspan="42"><a href="#" id="unsearch">! close search results</a></td></tr>');
|
||
for (var a = 0; a < res.hits.length; a++) {
|
||
var r = res.hits[a],
|
||
ts = parseInt(r.ts),
|
||
sz = esc(r.sz + ''),
|
||
rp = esc(r.rp + ''),
|
||
ext = rp.lastIndexOf('.') > 0 ? rp.split('.').slice(-1)[0] : '%',
|
||
links = linksplit(r.rp + '');
|
||
|
||
if (ext.length > 8)
|
||
ext = '%';
|
||
|
||
links = links.join('');
|
||
var nodes = ['<tr><td>-</td><td><div>' + links + '</div>', sz];
|
||
for (var b = 0; b < tagord.length; b++) {
|
||
var k = tagord[b],
|
||
v = r.tags[k] || "";
|
||
|
||
if (k == ".dur") {
|
||
var sv = v ? s2ms(v) : "";
|
||
nodes[nodes.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
||
continue;
|
||
}
|
||
|
||
nodes.push(v);
|
||
}
|
||
|
||
nodes = nodes.concat([ext, unix2iso(ts)]);
|
||
html.push(nodes.join('</td><td>'));
|
||
html.push('</td></tr>');
|
||
}
|
||
|
||
if (!orig_html || orig_url != get_evpath()) {
|
||
orig_html = ebi('files').innerHTML;
|
||
orig_url = get_evpath();
|
||
}
|
||
|
||
ofiles.innerHTML = html.join('\n');
|
||
ofiles.setAttribute("ts", this.ts);
|
||
ofiles.setAttribute("q_raw", this.q_raw);
|
||
set_vq();
|
||
mukey.render();
|
||
msel.render();
|
||
reload_browser();
|
||
filecols.set_style(['File Name']);
|
||
|
||
ebi('unsearch').onclick = unsearch;
|
||
}
|
||
|
||
function unsearch(e) {
|
||
ev(e);
|
||
treectl.show();
|
||
ebi('files').innerHTML = orig_html;
|
||
ebi('files').removeAttribute('q_raw');
|
||
orig_html = null;
|
||
msel.render();
|
||
reload_browser();
|
||
}
|
||
})();
|
||
|
||
|
||
var treectl = (function () {
|
||
var treectl = {
|
||
"hidden": true,
|
||
"ls_cb": null,
|
||
"dir_cb": null
|
||
},
|
||
entreed = false,
|
||
fixedpos = false,
|
||
prev_atop = null,
|
||
prev_winh = null,
|
||
dyn = bcfg_get('dyntree', true),
|
||
treesz = icfg_get('treesz', 16);
|
||
|
||
treesz = Math.min(Math.max(treesz, 4), 50);
|
||
|
||
treectl.entree = function (e) {
|
||
ev(e);
|
||
entreed = true;
|
||
swrite('entreed', 'tree');
|
||
|
||
get_tree("", get_evpath(), true);
|
||
treectl.show();
|
||
}
|
||
|
||
treectl.show = function () {
|
||
treectl.hidden = false;
|
||
if (!entreed) {
|
||
ebi('path').style.display = 'inline-block';
|
||
return;
|
||
}
|
||
ebi('path').style.display = 'none';
|
||
ebi('tree').style.display = 'block';
|
||
window.addEventListener('scroll', onscroll);
|
||
window.addEventListener('resize', onresize);
|
||
onresize();
|
||
};
|
||
|
||
treectl.detree = function (e) {
|
||
ev(e);
|
||
entreed = false;
|
||
swrite('entreed', 'na');
|
||
|
||
treectl.hide();
|
||
ebi('path').style.display = 'inline-block';
|
||
}
|
||
|
||
treectl.hide = function () {
|
||
treectl.hidden = true;
|
||
ebi('path').style.display = 'none';
|
||
ebi('tree').style.display = 'none';
|
||
ebi('wrap').style.marginLeft = '0';
|
||
window.removeEventListener('resize', onresize);
|
||
window.removeEventListener('scroll', onscroll);
|
||
}
|
||
|
||
function onscroll() {
|
||
if (!entreed || treectl.hidden)
|
||
return;
|
||
|
||
var tree = ebi('tree'),
|
||
wrap = ebi('wrap'),
|
||
atop = wrap.getBoundingClientRect().top,
|
||
winh = window.innerHeight;
|
||
|
||
if (atop === prev_atop && winh === prev_winh)
|
||
return;
|
||
|
||
prev_atop = atop;
|
||
prev_winh = winh;
|
||
|
||
if (fixedpos && atop >= 0) {
|
||
tree.style.position = 'absolute';
|
||
tree.style.bottom = '';
|
||
fixedpos = false;
|
||
}
|
||
else if (!fixedpos && atop < 0) {
|
||
tree.style.position = 'fixed';
|
||
tree.style.height = 'auto';
|
||
fixedpos = true;
|
||
}
|
||
|
||
if (fixedpos) {
|
||
tree.style.top = Math.max(0, parseInt(atop)) + 'px';
|
||
}
|
||
else {
|
||
var top = Math.max(0, parseInt(wrap.offsetTop)),
|
||
treeh = winh - atop;
|
||
|
||
tree.style.top = top + 'px';
|
||
tree.style.height = treeh < 10 ? '' : treeh + 'px';
|
||
}
|
||
}
|
||
|
||
function periodic() {
|
||
onscroll();
|
||
setTimeout(periodic, document.visibilityState ? 100 : 5000);
|
||
}
|
||
periodic();
|
||
|
||
function onresize(e) {
|
||
if (!entreed || treectl.hidden)
|
||
return;
|
||
|
||
var q = '#tree',
|
||
nq = 0;
|
||
|
||
while (dyn) {
|
||
nq++;
|
||
q += '>ul>li';
|
||
if (!QS(q))
|
||
break;
|
||
}
|
||
var w = treesz + nq;
|
||
ebi('tree').style.width = w + 'em';
|
||
ebi('wrap').style.marginLeft = w + 'em';
|
||
onscroll();
|
||
}
|
||
|
||
treectl.goto = function (url, push) {
|
||
get_tree("", url, true);
|
||
reqls(url, push);
|
||
}
|
||
|
||
function get_tree(top, dst, rst) {
|
||
var xhr = new XMLHttpRequest();
|
||
xhr.top = top;
|
||
xhr.dst = dst;
|
||
xhr.rst = rst;
|
||
xhr.ts = Date.now();
|
||
xhr.open('GET', dst + '?tree=' + top, true);
|
||
xhr.onreadystatechange = recvtree;
|
||
xhr.send();
|
||
enspin('#tree');
|
||
}
|
||
|
||
function recvtree() {
|
||
if (this.readyState != XMLHttpRequest.DONE)
|
||
return;
|
||
|
||
if (this.status !== 200) {
|
||
alert("http " + this.status + ": " + this.responseText);
|
||
return;
|
||
}
|
||
|
||
var cur = ebi('treeul').getAttribute('ts');
|
||
if (cur && parseInt(cur) > this.ts) {
|
||
console.log("reject tree");
|
||
return;
|
||
}
|
||
ebi('treeul').setAttribute('ts', this.ts);
|
||
|
||
var top = this.top == '.' ? this.dst : this.top,
|
||
name = uricom_dec(top.split('/').slice(-2)[0])[0],
|
||
rtop = top.replace(/^\/+/, ""),
|
||
res;
|
||
|
||
try {
|
||
res = JSON.parse(this.responseText);
|
||
}
|
||
catch (ex) {
|
||
return;
|
||
}
|
||
var html = parsetree(res, rtop);
|
||
if (!this.top) {
|
||
html = '<li><a href="#">-</a><a href="/">[root]</a>\n<ul>' + html;
|
||
if (this.rst || !ebi('treeul').getElementsByTagName('li').length)
|
||
ebi('treeul').innerHTML = html + '</ul></li>';
|
||
}
|
||
else {
|
||
html = '<a href="#">-</a><a href="' +
|
||
esc(top) + '">' + esc(name) +
|
||
"</a>\n<ul>\n" + html + "</ul>";
|
||
|
||
var links = QSA('#treeul a+a');
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
if (links[a].getAttribute('href') == top) {
|
||
var o = links[a].parentNode;
|
||
if (!o.getElementsByTagName('li').length)
|
||
o.innerHTML = html;
|
||
}
|
||
}
|
||
}
|
||
QS('#treeul>li>a+a').textContent = '[root]';
|
||
despin('#tree');
|
||
reload_tree();
|
||
onresize();
|
||
|
||
var fun = treectl.dir_cb;
|
||
if (fun) {
|
||
treectl.dir_cb = null;
|
||
try {
|
||
fun();
|
||
}
|
||
catch (ex) {
|
||
console.log("dir_cb failed", ex);
|
||
}
|
||
}
|
||
}
|
||
|
||
function reload_tree() {
|
||
var cdir = get_evpath(),
|
||
links = QSA('#treeul a+a');
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
var href = links[a].getAttribute('href');
|
||
links[a].setAttribute('class', href == cdir ? 'hl' : '');
|
||
links[a].onclick = treego;
|
||
}
|
||
links = QSA('#treeul li>a:first-child');
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
links[a].setAttribute('dst', links[a].nextSibling.getAttribute('href'));
|
||
links[a].onclick = treegrow;
|
||
}
|
||
}
|
||
|
||
function treego(e) {
|
||
if (ctrl(e))
|
||
return true;
|
||
|
||
ev(e);
|
||
if (this.getAttribute('class') == 'hl' &&
|
||
this.previousSibling.textContent == '-') {
|
||
treegrow.call(this.previousSibling, e);
|
||
return;
|
||
}
|
||
reqls(this.getAttribute('href'), true);
|
||
}
|
||
|
||
function reqls(url, hpush) {
|
||
var xhr = new XMLHttpRequest();
|
||
xhr.top = url;
|
||
xhr.hpush = hpush;
|
||
xhr.ts = Date.now();
|
||
xhr.open('GET', xhr.top + '?ls', true);
|
||
xhr.onreadystatechange = recvls;
|
||
xhr.send();
|
||
if (hpush)
|
||
get_tree('.', xhr.top);
|
||
|
||
enspin(thegrid.en ? '#gfiles' : '#files');
|
||
}
|
||
|
||
function treegrow(e) {
|
||
ev(e);
|
||
if (this.textContent == '-') {
|
||
while (this.nextSibling.nextSibling) {
|
||
var rm = this.nextSibling.nextSibling;
|
||
rm.parentNode.removeChild(rm);
|
||
}
|
||
this.textContent = '+';
|
||
onresize();
|
||
return;
|
||
}
|
||
var dst = this.getAttribute('dst');
|
||
get_tree('.', dst);
|
||
}
|
||
|
||
function recvls() {
|
||
if (this.readyState != XMLHttpRequest.DONE)
|
||
return;
|
||
|
||
if (this.status !== 200) {
|
||
alert("http " + this.status + ": " + this.responseText);
|
||
return;
|
||
}
|
||
|
||
var cur = ebi('files').getAttribute('ts');
|
||
if (cur && parseInt(cur) > this.ts) {
|
||
console.log("reject ls");
|
||
return;
|
||
}
|
||
ebi('files').setAttribute('ts', this.ts);
|
||
|
||
try {
|
||
var res = JSON.parse(this.responseText);
|
||
}
|
||
catch (ex) {
|
||
window.location = this.top;
|
||
return;
|
||
}
|
||
|
||
ebi('srv_info').innerHTML = '<span>' + res.srvinf + '</span>';
|
||
|
||
var top = this.top,
|
||
nodes = res.dirs.concat(res.files),
|
||
html = mk_files_header(res.taglist);
|
||
|
||
html.push('<tbody>');
|
||
nodes = sortfiles(nodes);
|
||
for (var a = 0; a < nodes.length; a++) {
|
||
var r = nodes[a],
|
||
hname = esc(uricom_dec(r.href)[0]),
|
||
sortv = (r.href.slice(-1) == '/' ? '\t' : '') + hname,
|
||
ln = ['<tr><td>' + r.lead + '</td><td sortv="' + sortv +
|
||
'"><a href="' + top + r.href + '">' + hname + '</a>', r.sz];
|
||
|
||
for (var b = 0; b < res.taglist.length; b++) {
|
||
var k = res.taglist[b],
|
||
v = (r.tags || {})[k] || "";
|
||
|
||
if (k == ".dur") {
|
||
var sv = v ? s2ms(v) : "";
|
||
ln[ln.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
||
continue;
|
||
}
|
||
ln.push(v);
|
||
}
|
||
ln = ln.concat([r.ext, unix2iso(r.ts)]).join('</td><td>');
|
||
html.push(ln + '</td></tr>');
|
||
}
|
||
html.push('</tbody>');
|
||
html = html.join('\n');
|
||
ebi('files').innerHTML = html;
|
||
|
||
if (this.hpush)
|
||
hist_push(this.top);
|
||
|
||
acct = res.acct;
|
||
apply_perms(res.perms);
|
||
despin('#files');
|
||
despin('#gfiles');
|
||
|
||
ebi('pro').innerHTML = res.logues ? res.logues[0] || "" : "";
|
||
ebi('epi').innerHTML = res.logues ? res.logues[1] || "" : "";
|
||
|
||
document.title = '⇆🎉 ' + uricom_dec(document.location.pathname.slice(1, -1))[0];
|
||
|
||
filecols.set_style();
|
||
mukey.render();
|
||
msel.render();
|
||
reload_tree();
|
||
reload_browser();
|
||
|
||
var fun = treectl.ls_cb;
|
||
if (fun) {
|
||
treectl.ls_cb = null;
|
||
fun();
|
||
}
|
||
}
|
||
|
||
function parsetree(res, top) {
|
||
var ret = '';
|
||
for (var a = 0; a < res.a.length; a++) {
|
||
if (res.a[a] !== '')
|
||
res['k' + res.a[a]] = 0;
|
||
}
|
||
delete res['a'];
|
||
var keys = Object.keys(res);
|
||
keys.sort();
|
||
for (var a = 0; a < keys.length; a++) {
|
||
var kk = keys[a],
|
||
ks = kk.slice(1),
|
||
k = uricom_dec(ks),
|
||
hek = esc(k[0]),
|
||
uek = k[1] ? uricom_enc(k[0], true) : k[0],
|
||
url = '/' + (top ? top + uek : uek) + '/',
|
||
sym = res[kk] ? '-' : '+',
|
||
link = '<a href="#">' + sym + '</a><a href="' +
|
||
url + '">' + hek + '</a>';
|
||
|
||
if (res[kk]) {
|
||
var subtree = parsetree(res[kk], url.slice(1));
|
||
ret += '<li>' + link + '\n<ul>\n' + subtree + '</ul></li>\n';
|
||
}
|
||
else {
|
||
ret += '<li>' + link + '</li>\n';
|
||
}
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
function dyntree(e) {
|
||
ev(e);
|
||
dyn = !dyn;
|
||
bcfg_set('dyntree', dyn);
|
||
onresize();
|
||
}
|
||
|
||
function scaletree(e) {
|
||
ev(e);
|
||
treesz += parseInt(this.getAttribute("step"));
|
||
if (isNaN(treesz))
|
||
treesz = 16;
|
||
|
||
swrite('treesz', treesz);
|
||
onresize();
|
||
}
|
||
|
||
ebi('entree').onclick = treectl.entree;
|
||
ebi('detree').onclick = treectl.detree;
|
||
ebi('dyntree').onclick = dyntree;
|
||
ebi('twig').onclick = scaletree;
|
||
ebi('twobytwo').onclick = scaletree;
|
||
if (sread('entreed') == 'tree')
|
||
treectl.entree();
|
||
|
||
window.onpopstate = function (e) {
|
||
console.log("h-pop " + e.state);
|
||
if (!e.state)
|
||
return;
|
||
|
||
var url = new URL(e.state, "https://" + document.location.host);
|
||
treectl.goto(url.pathname);
|
||
};
|
||
|
||
if (window.history && history.pushState) {
|
||
hist_replace(get_evpath() + window.location.hash);
|
||
}
|
||
|
||
treectl.onscroll = onscroll;
|
||
return treectl;
|
||
})();
|
||
|
||
|
||
function enspin(sel) {
|
||
despin(sel);
|
||
var d = mknod('div');
|
||
d.setAttribute('class', 'dumb_loader_thing');
|
||
d.innerHTML = '🌲';
|
||
var tgt = QS(sel);
|
||
tgt.insertBefore(d, tgt.childNodes[0]);
|
||
}
|
||
|
||
|
||
function despin(sel) {
|
||
var o = QSA(sel + '>.dumb_loader_thing');
|
||
for (var a = o.length - 1; a >= 0; a--)
|
||
o[a].parentNode.removeChild(o[a]);
|
||
}
|
||
|
||
|
||
function apply_perms(newperms) {
|
||
perms = newperms || [];
|
||
|
||
var axs = [],
|
||
aclass = '>',
|
||
chk = ['read', 'write', 'rename', 'delete'];
|
||
|
||
for (var a = 0; a < chk.length; a++)
|
||
if (has(perms, chk[a]))
|
||
axs.push(chk[a].slice(0, 1).toUpperCase() + chk[a].slice(1));
|
||
|
||
axs = axs.join('-');
|
||
if (perms.length == 1) {
|
||
aclass = ' class="warn">';
|
||
axs += '-Only';
|
||
}
|
||
|
||
ebi('acc_info').innerHTML = '<span' + aclass + axs + ' access</span>' + (acct != '*' ?
|
||
'<a href="/?pw=x">Logout ' + acct + '</a>' : '<a href="/?h">Login</a>');
|
||
|
||
var o = QSA('#ops>a[data-perm], #u2footfoot');
|
||
for (var a = 0; a < o.length; a++) {
|
||
var display = '';
|
||
var needed = o[a].getAttribute('data-perm').split(' ');
|
||
for (var b = 0; b < needed.length; b++) {
|
||
if (!has(perms, needed[b])) {
|
||
display = 'none';
|
||
}
|
||
}
|
||
o[a].style.display = display;
|
||
}
|
||
|
||
var act = QS('#ops>a.act');
|
||
if (act && act.style.display === 'none')
|
||
goto();
|
||
|
||
document.body.setAttribute('perms', perms.join(' '));
|
||
|
||
var have_write = has(perms, "write"),
|
||
have_read = has(perms, "read"),
|
||
de = document.documentElement,
|
||
tds = QSA('#u2conf td');
|
||
|
||
clmod(de, "read", have_read);
|
||
clmod(de, "write", have_write);
|
||
clmod(de, "nread", !have_read);
|
||
clmod(de, "nwrite", !have_write);
|
||
|
||
for (var a = 0; a < tds.length; a++) {
|
||
tds[a].style.display =
|
||
(have_write || tds[a].getAttribute('data-perm') == 'read') ?
|
||
'table-cell' : 'none';
|
||
}
|
||
|
||
if (window['up2k'])
|
||
up2k.set_fsearch();
|
||
|
||
ebi('widget').style.display = have_read ? '' : 'none';
|
||
thegrid.setvis(have_read);
|
||
if (!have_read)
|
||
goto('up2k');
|
||
}
|
||
|
||
|
||
function find_file_col(txt) {
|
||
var i = -1,
|
||
min = false,
|
||
tds = ebi('files').tHead.getElementsByTagName('th');
|
||
|
||
for (var a = 0; a < tds.length; a++) {
|
||
var spans = tds[a].getElementsByTagName('span');
|
||
if (spans.length && spans[0].textContent == txt) {
|
||
min = (tds[a].getAttribute('class') || '').indexOf('min') !== -1;
|
||
i = a;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (i == -1)
|
||
return;
|
||
|
||
return [i, min];
|
||
}
|
||
|
||
|
||
function mk_files_header(taglist) {
|
||
var html = [
|
||
'<thead><tr>',
|
||
'<th name="lead"><span>c</span></th>',
|
||
'<th name="href"><span>File Name</span></th>',
|
||
'<th name="sz" sort="int"><span>Size</span></th>'
|
||
];
|
||
for (var a = 0; a < taglist.length; a++) {
|
||
var tag = taglist[a],
|
||
c1 = tag.slice(0, 1).toUpperCase();
|
||
|
||
tag = c1 + tag.slice(1);
|
||
if (c1 == '.')
|
||
tag = '<th name="tags/' + tag + '" sort="int"><span>' + tag.slice(1);
|
||
else
|
||
tag = '<th name="tags/' + tag + '"><span>' + tag;
|
||
|
||
html.push(tag + '</span></th>');
|
||
}
|
||
html = html.concat([
|
||
'<th name="ext"><span>T</span></th>',
|
||
'<th name="ts"><span>Date</span></th>',
|
||
'</tr></thead>',
|
||
]);
|
||
return html;
|
||
}
|
||
|
||
|
||
var filecols = (function () {
|
||
var hidden = jread('filecols', []),
|
||
tts = {
|
||
"c": "action buttons",
|
||
"dur": "duration",
|
||
"q": "quality / bitrate",
|
||
"Ac": "audio codec",
|
||
"Vc": "video codec",
|
||
"Res": "resolution",
|
||
"T": "filetype",
|
||
"aq": "audio quality / bitrate",
|
||
"vq": "video quality / bitrate",
|
||
"pixfmt": "subsampling / pixel structure",
|
||
"resw": "horizontal resolution",
|
||
"resh": "veritcal resolution",
|
||
"acs": "audio channels",
|
||
"hz": "sample rate"
|
||
};
|
||
|
||
var add_btns = function () {
|
||
var ths = QSA('#files th>span');
|
||
for (var a = 0, aa = ths.length; a < aa; a++) {
|
||
var th = ths[a].parentElement,
|
||
ttv = tts[ths[a].textContent];
|
||
|
||
th.innerHTML = '<div class="cfg"><a href="#">-</a></div>' + ths[a].outerHTML;
|
||
th.getElementsByTagName('a')[0].onclick = ev_row_tgl;
|
||
if (ttv) {
|
||
th.setAttribute("tt", ttv);
|
||
th.setAttribute("ttd", "u");
|
||
}
|
||
}
|
||
};
|
||
|
||
function hcols_click(e) {
|
||
ev(e);
|
||
var t = e.target;
|
||
if (t.tagName != 'A')
|
||
return;
|
||
|
||
toggle(t.textContent);
|
||
}
|
||
|
||
var set_style = function (unhide) {
|
||
hidden.sort();
|
||
|
||
if (!unhide)
|
||
unhide = [];
|
||
|
||
var html = [],
|
||
hcols = ebi('hcols');
|
||
|
||
for (var a = 0; a < hidden.length; a++) {
|
||
var ttv = tts[hidden[a]],
|
||
tta = ttv ? ' tt="' + ttv + '">' : '>';
|
||
|
||
html.push('<a href="#" class="btn"' + tta + esc(hidden[a]) + '</a>');
|
||
}
|
||
hcols.previousSibling.style.display = html.length ? 'block' : 'none';
|
||
hcols.innerHTML = html.join('\n');
|
||
hcols.onclick = hcols_click;
|
||
|
||
add_btns();
|
||
|
||
var ohidden = [],
|
||
ths = QSA('#files th'),
|
||
ncols = ths.length;
|
||
|
||
for (var a = 0; a < ncols; a++) {
|
||
var span = ths[a].getElementsByTagName('span');
|
||
if (span.length <= 0)
|
||
continue;
|
||
|
||
var name = span[0].textContent,
|
||
cls = false;
|
||
|
||
if (has(hidden, name) && !has(unhide, name)) {
|
||
ohidden.push(a);
|
||
cls = true;
|
||
}
|
||
clmod(ths[a], 'min', cls)
|
||
}
|
||
for (var a = 0; a < ncols; a++) {
|
||
var cls = has(ohidden, a) ? 'min' : '',
|
||
tds = QSA('#files>tbody>tr>td:nth-child(' + (a + 1) + ')');
|
||
|
||
for (var b = 0, bb = tds.length; b < bb; b++)
|
||
tds[b].setAttribute('class', cls);
|
||
}
|
||
if (window['tt']) {
|
||
tt.att(ebi('hcols'));
|
||
tt.att(QS('#files thead'));
|
||
}
|
||
};
|
||
set_style();
|
||
|
||
var toggle = function (name) {
|
||
var ofs = hidden.indexOf(name);
|
||
if (ofs !== -1)
|
||
hidden.splice(ofs, 1);
|
||
else
|
||
hidden.push(name);
|
||
|
||
jwrite("filecols", hidden);
|
||
set_style();
|
||
};
|
||
|
||
try {
|
||
var ci = find_file_col('dur'),
|
||
i = ci[0],
|
||
rows = ebi('files').tBodies[0].rows;
|
||
|
||
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||
var c = rows[a].cells[i];
|
||
if (c && c.textContent)
|
||
c.textContent = s2ms(c.textContent);
|
||
}
|
||
}
|
||
catch (ex) { }
|
||
|
||
return {
|
||
"add_btns": add_btns,
|
||
"set_style": set_style,
|
||
"toggle": toggle,
|
||
};
|
||
})();
|
||
|
||
|
||
var mukey = (function () {
|
||
var maps = {
|
||
"rekobo_alnum": [
|
||
"1B ", "2B ", "3B ", "4B ", "5B ", "6B ", "7B ", "8B ", "9B ", "10B", "11B", "12B",
|
||
"1A ", "2A ", "3A ", "4A ", "5A ", "6A ", "7A ", "8A ", "9A ", "10A", "11A", "12A"
|
||
],
|
||
"rekobo_classic": [
|
||
"B ", "F# ", "Db ", "Ab ", "Eb ", "Bb ", "F ", "C ", "G ", "D ", "A ", "E ",
|
||
"Abm", "Ebm", "Bbm", "Fm ", "Cm ", "Gm ", "Dm ", "Am ", "Em ", "Bm ", "F#m", "Dbm"
|
||
],
|
||
"traktor_musical": [
|
||
"B ", "Gb ", "Db ", "Ab ", "Eb ", "Bb ", "F ", "C ", "G ", "D ", "A ", "E ",
|
||
"Abm", "Ebm", "Bbm", "Fm ", "Cm ", "Gm ", "Dm ", "Am ", "Em ", "Bm ", "Gbm", "Dbm"
|
||
],
|
||
"traktor_sharps": [
|
||
"B ", "F# ", "C# ", "G# ", "D# ", "A# ", "F ", "C ", "G ", "D ", "A ", "E ",
|
||
"G#m", "D#m", "A#m", "Fm ", "Cm ", "Gm ", "Dm ", "Am ", "Em ", "Bm ", "F#m", "C#m"
|
||
],
|
||
"traktor_open": [
|
||
"6d ", "7d ", "8d ", "9d ", "10d", "11d", "12d", "1d ", "2d ", "3d ", "4d ", "5d ",
|
||
"6m ", "7m ", "8m ", "9m ", "10m", "11m", "12m", "1m ", "2m ", "3m ", "4m ", "5m "
|
||
]
|
||
};
|
||
var map = {},
|
||
html = [];
|
||
|
||
for (var k in maps) {
|
||
if (!maps.hasOwnProperty(k))
|
||
continue;
|
||
|
||
html.push(
|
||
'<span><input type="radio" name="keytype" value="' + k + '" id="key_' + k + '">' +
|
||
'<label for="key_' + k + '">' + k + '</label></span>');
|
||
|
||
for (var a = 0; a < 24; a++)
|
||
maps[k][a] = maps[k][a].trim();
|
||
}
|
||
ebi('key_notation').innerHTML = html.join('\n');
|
||
|
||
function set_key_notation(e) {
|
||
ev(e);
|
||
var notation = this.getAttribute('value');
|
||
load_notation(notation);
|
||
try_render();
|
||
}
|
||
|
||
function load_notation(notation) {
|
||
swrite("key_notation", notation);
|
||
map = {};
|
||
var dst = maps[notation];
|
||
for (var k in maps)
|
||
if (k != notation && maps.hasOwnProperty(k))
|
||
for (var a = 0; a < 24; a++)
|
||
if (maps[k][a] != dst[a])
|
||
map[maps[k][a]] = dst[a];
|
||
}
|
||
|
||
function render() {
|
||
var ci = find_file_col('Key');
|
||
if (!ci)
|
||
return;
|
||
|
||
var i = ci[0],
|
||
min = ci[1],
|
||
rows = ebi('files').tBodies[0].rows;
|
||
|
||
if (min)
|
||
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||
var c = rows[a].cells[i];
|
||
if (!c)
|
||
continue;
|
||
|
||
var v = c.getAttribute('html');
|
||
c.setAttribute('html', map[v] || v);
|
||
}
|
||
else
|
||
for (var a = 0, aa = rows.length; a < aa; a++) {
|
||
var c = rows[a].cells[i];
|
||
if (!c)
|
||
continue;
|
||
|
||
var v = c.textContent;
|
||
c.textContent = map[v] || v;
|
||
}
|
||
}
|
||
|
||
function try_render() {
|
||
try {
|
||
render();
|
||
}
|
||
catch (ex) {
|
||
console.log("key notation failed: " + ex);
|
||
}
|
||
}
|
||
|
||
var notation = sread("key_notation") || "rekobo_alnum";
|
||
ebi('key_' + notation).checked = true;
|
||
load_notation(notation);
|
||
|
||
var o = QSA('#key_notation input');
|
||
for (var a = 0; a < o.length; a++) {
|
||
o[a].onchange = set_key_notation;
|
||
}
|
||
|
||
return {
|
||
"render": try_render
|
||
};
|
||
})();
|
||
|
||
|
||
function addcrc() {
|
||
var links = QSA(
|
||
'#files>tbody>tr>td:first-child+td>' + (
|
||
ebi('unsearch') ? 'div>a:last-child' : 'a'));
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++)
|
||
if (!links[a].getAttribute('id')) {
|
||
var crc = crc32(links[a].textContent || links[a].innerText);
|
||
crc = ('00000000' + crc).slice(-8);
|
||
links[a].setAttribute('id', 'f-' + crc);
|
||
}
|
||
}
|
||
|
||
|
||
var light;
|
||
(function () {
|
||
light = bcfg_get('lightmode', false);
|
||
|
||
function freshen() {
|
||
clmod(document.documentElement, "light", light);
|
||
pbar.drawbuf();
|
||
pbar.drawpos();
|
||
vbar.draw();
|
||
}
|
||
|
||
ebi('lightmode').onclick = function (e) {
|
||
ev(e);
|
||
light = !light;
|
||
bcfg_set('lightmode', light);
|
||
freshen();
|
||
};
|
||
|
||
freshen();
|
||
})();
|
||
|
||
|
||
var arcfmt = (function () {
|
||
if (!ebi('arc_fmt'))
|
||
return { "render": function () { } };
|
||
|
||
var html = [],
|
||
fmts = [
|
||
["tar", "tar", "plain gnutar file"],
|
||
["zip", "zip=utf8", "zip with utf8 filenames (maybe wonky on windows 7 and older)"],
|
||
["zip_dos", "zip", "zip with traditional cp437 filenames, for really old software"],
|
||
["zip_crc", "zip=crc", "cp437 with crc32 computed early,$Nfor MS-DOS PKZIP v2.04g (october 1993)$N(takes longer to process before download can start)"]
|
||
];
|
||
|
||
for (var a = 0; a < fmts.length; a++) {
|
||
var k = fmts[a][0];
|
||
html.push(
|
||
'<span><input type="radio" name="arcfmt" value="' + k + '" id="arcfmt_' + k + '" tt="' + fmts[a][2] + '">' +
|
||
'<label for="arcfmt_' + k + '" tt="' + fmts[a][2] + '">' + k + '</label></span>');
|
||
}
|
||
ebi('arc_fmt').innerHTML = html.join('\n');
|
||
|
||
var fmt = sread("arc_fmt");
|
||
if (!ebi('arcfmt_' + fmt))
|
||
fmt = "zip";
|
||
|
||
ebi('arcfmt_' + fmt).checked = true;
|
||
|
||
function render() {
|
||
var arg = null,
|
||
tds = QSA('#files tbody td:first-child a');
|
||
|
||
for (var a = 0; a < fmts.length; a++)
|
||
if (fmts[a][0] == fmt)
|
||
arg = fmts[a][1];
|
||
|
||
for (var a = 0, aa = tds.length; a < aa; a++) {
|
||
var o = tds[a], txt = o.textContent, href = o.getAttribute('href');
|
||
if (txt != 'tar' && txt != 'zip')
|
||
continue;
|
||
|
||
var ofs = href.lastIndexOf('?');
|
||
if (ofs < 0)
|
||
throw new Error('missing arg in url');
|
||
|
||
o.setAttribute("href", href.slice(0, ofs + 1) + arg);
|
||
o.textContent = fmt.split('_')[0];
|
||
}
|
||
ebi('selzip').textContent = fmt.split('_')[0];
|
||
ebi('selzip').setAttribute('fmt', arg);
|
||
}
|
||
|
||
function try_render() {
|
||
try {
|
||
render();
|
||
}
|
||
catch (ex) {
|
||
console.log("arcfmt failed: " + ex);
|
||
}
|
||
}
|
||
|
||
function change_fmt(e) {
|
||
ev(e);
|
||
fmt = this.getAttribute('value');
|
||
swrite("arc_fmt", fmt);
|
||
try_render();
|
||
}
|
||
|
||
var o = QSA('#arc_fmt input');
|
||
for (var a = 0; a < o.length; a++) {
|
||
o[a].onchange = change_fmt;
|
||
}
|
||
|
||
return {
|
||
"render": try_render
|
||
};
|
||
})();
|
||
|
||
|
||
var msel = (function () {
|
||
var r = {};
|
||
r.sel = null;
|
||
r.all = null;
|
||
|
||
r.load = function () {
|
||
if (r.sel)
|
||
return;
|
||
|
||
r.sel = [];
|
||
r.all = [];
|
||
var links = QSA('#files tbody td:nth-child(2) a:last-child'),
|
||
vbase = get_evpath();
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
var href = links[a].getAttribute('href').replace(/\/$/, ""),
|
||
item = {};
|
||
|
||
item.id = links[a].getAttribute('id');
|
||
item.sel = links[a].closest('tr').classList.contains('sel');
|
||
item.vp = href.indexOf('/') !== -1 ? href : vbase + href;
|
||
item.name = href.split('/').slice(-1);
|
||
|
||
r.all.push(item);
|
||
if (item.sel)
|
||
r.sel.push(item);
|
||
|
||
links[a].setAttribute('name', item.name);
|
||
}
|
||
};
|
||
|
||
r.getsel = function () {
|
||
r.load();
|
||
return r.sel;
|
||
};
|
||
r.getall = function () {
|
||
r.load();
|
||
return r.all;
|
||
};
|
||
function selui() {
|
||
r.sel = r.all = null;
|
||
clmod(ebi('wtoggle'), 'sel', r.getsel().length);
|
||
thegrid.loadsel();
|
||
fileman.render();
|
||
}
|
||
function seltgl(e) {
|
||
ev(e);
|
||
var tr = this.parentNode;
|
||
clmod(tr, 'sel', 't');
|
||
selui();
|
||
}
|
||
function evsel(e, fun) {
|
||
ev(e);
|
||
var trs = QSA('#files tbody tr');
|
||
for (var a = 0, aa = trs.length; a < aa; a++)
|
||
clmod(trs[a], 'sel', fun);
|
||
selui();
|
||
}
|
||
ebi('selall').onclick = function (e) {
|
||
evsel(e, "add");
|
||
};
|
||
ebi('selinv').onclick = function (e) {
|
||
evsel(e, "t");
|
||
};
|
||
ebi('selzip').onclick = function (e) {
|
||
ev(e);
|
||
var names = r.getsel(),
|
||
arg = ebi('selzip').getAttribute('fmt'),
|
||
txt = names.join('\n'),
|
||
frm = mknod('form');
|
||
|
||
frm.setAttribute('action', '?' + arg);
|
||
frm.setAttribute('method', 'post');
|
||
frm.setAttribute('target', '_blank');
|
||
frm.setAttribute('enctype', 'multipart/form-data');
|
||
frm.innerHTML = '<input name="act" value="zip" />' +
|
||
'<textarea name="files" id="ziptxt"></textarea>';
|
||
frm.style.display = 'none';
|
||
|
||
var oldform = QS('#widgeti>form');
|
||
if (oldform)
|
||
oldform.parentNode.removeChild(oldform);
|
||
|
||
ebi('widgeti').appendChild(frm);
|
||
var obj = ebi('ziptxt');
|
||
obj.value = txt;
|
||
console.log(txt);
|
||
frm.submit();
|
||
};
|
||
r.render = function () {
|
||
var tds = QSA('#files tbody td+td+td');
|
||
for (var a = 0, aa = tds.length; a < aa; a++) {
|
||
tds[a].onclick = seltgl;
|
||
}
|
||
r.sel = r.all = null;
|
||
arcfmt.render();
|
||
fileman.render();
|
||
ebi('selzip').style.display = ebi('unsearch') ? 'none' : '';
|
||
}
|
||
return r;
|
||
})();
|
||
|
||
|
||
(function () {
|
||
try {
|
||
var tr = ebi('files').tBodies[0].rows;
|
||
for (var a = 0; a < tr.length; a++) {
|
||
var td = tr[a].cells[1],
|
||
ao = td.firstChild,
|
||
href = ao.getAttribute('href'),
|
||
isdir = href.split('?')[0].slice(-1)[0] == '/',
|
||
txt = ao.textContent;
|
||
|
||
td.setAttribute('sortv', (isdir ? '\t' : '') + txt);
|
||
}
|
||
}
|
||
catch (ex) { }
|
||
})();
|
||
|
||
|
||
function ev_row_tgl(e) {
|
||
ev(e);
|
||
filecols.toggle(this.parentElement.parentElement.getElementsByTagName('span')[0].textContent);
|
||
}
|
||
|
||
|
||
function reload_mp() {
|
||
if (mp && mp.au) {
|
||
mp.au.pause();
|
||
mp.au = null;
|
||
}
|
||
mpl.stop();
|
||
widget.close();
|
||
mp = new MPlayer();
|
||
setTimeout(pbar.onresize, 1);
|
||
}
|
||
|
||
|
||
function reload_browser(not_mp) {
|
||
filecols.set_style();
|
||
|
||
var parts = get_evpath().split('/'),
|
||
rm = QSA('#path>a+a+a');
|
||
|
||
for (a = rm.length - 1; a >= 0; a--)
|
||
rm[a].parentNode.removeChild(rm[a]);
|
||
|
||
var link = '/';
|
||
for (var a = 1; a < parts.length - 1; a++) {
|
||
link += parts[a] + '/';
|
||
var o = mknod('a');
|
||
o.setAttribute('href', link);
|
||
o.textContent = uricom_dec(parts[a])[0];
|
||
ebi('path').appendChild(o);
|
||
}
|
||
|
||
var oo = QSA('#files>tbody>tr>td:nth-child(3)');
|
||
for (var a = 0, aa = oo.length; a < aa; a++) {
|
||
var sz = oo[a].textContent.replace(/ +/g, ""),
|
||
hsz = sz.replace(/\B(?=(\d{3})+(?!\d))/g, " ");
|
||
|
||
oo[a].textContent = hsz;
|
||
}
|
||
|
||
if (!not_mp) {
|
||
addcrc();
|
||
reload_mp();
|
||
makeSortable(ebi('files'), mp.read_order.bind(mp));
|
||
}
|
||
|
||
if (window['up2k'])
|
||
up2k.set_fsearch();
|
||
|
||
thegrid.setdirty();
|
||
}
|
||
reload_browser(true);
|
||
mukey.render();
|
||
msel.render();
|
||
play_linked();
|