"use strict"; window.onerror = vis_exh; function dbg(msg) { ebi('path').innerHTML = msg; } // toolbar ebi('ops').innerHTML = ( '---\n' + (have_up2k_idx ? ( '๐Ÿ”Ž\n' + (have_del && have_unpost ? '๐Ÿงฏ\n' : '') + '๐Ÿš€\n' ) : ( '๐Ÿš€\n' )) + '๐ŸŽˆ\n' + '๐Ÿ“‚\n' + '๐Ÿ“\n' + '๐Ÿ“Ÿ\n' + '๐ŸŽบ\n' + 'โš™๏ธ\n' + '
' ); // media player ebi('widget').innerHTML = ( '
' + 'โœŽnameโŒซdeleteโœ‚cut๐Ÿ“‹paste' + 'sel.
allsel.
inv.zip' + '
๐Ÿ“‹irc๐Ÿ“‹txt' + '็”ฐโ™ซ' + '
' + '
' + '
โฎโ–ถโญ
' + ' ' + ' ' + ' ' + '
' ); // up2k ui ebi('op_up2k').innerHTML = ( '
\n' + '\n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + (have_up2k_idx ? ( ' \n' ) : '') + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + '

parallel uploads:
\n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + '
\n' + ' +
 \n' + '
\n' + '
\n' + '
\n' + '
\n' + '
\n' + ' drag/drop files
\n' + ' and folders here
\n' + ' (or click me)\n' + '
\n' + '
\n' + '
\n' + ' ok 0ng 0done 0busy 0que 0\n' + '
\n' + '\n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + '
filenamestatusprogresscleanup
\n' + '

\n' + '

( you can use the basic uploader if you don\'t need lastmod timestamps, resumable uploads, or progress bars )

' ); // config panel ebi('op_cfg').innerHTML = ( '
\n' + '

switches

\n' + '
\n' + ' โ„น๏ธ tooltips\n' + ' โ˜€๏ธ lightmode\n' + ' dotfiles\n' + ' ็”ฐ the grid\n' + ' ๐Ÿ–ผ๏ธ thumbs\n' + '
\n' + '
\n' + (have_zip ? ( '

folder download

\n' ) : '') + '
\n' + '

up2k switches

\n' + '
\n' + ' turbo\n' + ' date-chk\n' + '
\n' + '
\n' + '

key notation

\n' + '

hidden columns

' ); // navpane ebi('tree').innerHTML = ( '
\n' + ' ๐Ÿž...\n' + ' +\n' + ' \n' + ' a\n' + '
\n' + '\n' + '
 
' ); (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(); } 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 = ( '

switches

' + 'preload' + '/np clip' + 'os-ctl' + 'osd-cv' + '
' + '

playback mode

' + '๐Ÿ” loop-folder' + '๐Ÿ“‚ next-folder' + '
' + '

tint

' + '' + '
' + '

audio equalizer

'); 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) toast.err(5, '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 = 'play'; 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 ft2dict(tr) { var th = ebi('files').tHead.rows[0].cells, rv = [], ra = [], rt = {}; for (var a = 1, aa = th.length; a < aa; a++) { var tv = tr.cells[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]; } function get_np() { var tr = QS('#files tr.play'); return ft2dict(tr); }; // 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, preloaded = null; r.progress_updater = function () { timer.add(updater_impl, true); }; function updater_impl() { if (!mp.au) { widget.paused(true); timer.rm(updater_impl); 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) timer.rm(updater_impl); } 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 = [''], 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(''); h2.push(''); h4.push(''); h3.push(''); } html = html.join('\n') + ''; html += h2.join('\n') + ''; html += h3.join('\n') + ''; html += h4.join('\n') + '
', 'enable+' + vs[a][1] + '
'; 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 toast.err(0, 'failed to load ogv.js:\ncannot play ogg/opus in this browser\n(try a non-apple device)'); toast.inf(0, '

loading ogv.js

thanks apple

'); import_js('/.cpr/deps/ogv.js', function () { toast.hide(); 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) { toast.err(0, '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] + 'ยป'; toast.warn(15, 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( '
' + '
Cancel
(show file list)
'); 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')); })(); function fmt_ren(re, md, fmt) { var ptr = 0; function dive(stop_ch) { var ret = '', ng = 0; while (ptr < fmt.length) { var dbg = fmt.slice(ptr), ch = fmt[ptr++]; if (ch == '\\') { ret += fmt[ptr++]; continue; } if (ch == ')' || ch == ']' || ch == stop_ch) return [ng, ret]; if (ch == '[') { var r2 = dive(); if (r2[0] == 0) ret += r2[1]; } else if (ch == '(') { var end = fmt.indexOf(')', ptr); if (end < 0) throw 'the ( was never closed: ' + fmt.slice(0, ptr); var arg = fmt.slice(ptr, end), v = null; ptr = end + 1; if (arg != parseInt(arg)) v = md[arg]; else { arg = parseInt(arg); if (arg >= re.length) throw 'matching group ' + arg + ' exceeds ' + (re.length - 0); v = re[arg]; } if (v !== null && v !== undefined) ret += v; else ng++; } else if (ch == '$') { ch = fmt[ptr++]; var end = fmt.indexOf('(', ptr); if (end < 0) throw 'no function name after the $ here: ' + fmt.slice(0, ptr); var fun = fmt.slice(ptr - 1, end); ptr = end + 1; if (fun == "lpad") { var str = dive(',')[1]; var len = dive(',')[1]; var chr = dive()[1]; if (!len || !chr) throw 'invalid arguments to ' + fun; if (!str.length) ng += 1; while (str.length < len) str = chr + str; ret += str; } else if (fun == "rpad") { var str = dive(',')[1]; var len = dive(',')[1]; var chr = dive()[1]; if (!len || !chr) throw 'invalid arguments to ' + fun; if (!str.length) ng += 1; while (str.length < len) str += chr; ret += str; } else throw 'function not implemented: "' + fun + '"'; } else ret += ch; } return [ng, ret]; } try { return [true, dive()[1]]; } catch (ex) { return [false, ex]; } } var fileman = (function () { var bren = ebi('fren'), bdel = ebi('fdel'), bcut = ebi('fcut'), bpst = ebi('fpst'), r = {}; r.clip = null; try { r.bus = new BroadcastChannel("fileman_bus"); } catch (ex) { } r.render = function () { if (r.clip === null) r.clip = jread('fman_clip', []); var nsel = msel.getsel().length; clmod(bren, 'en', nsel == 1); clmod(bdel, 'en', nsel); clmod(bcut, 'en', nsel); clmod(bpst, 'en', r.clip && r.clip.length); bren.style.display = have_mv && has(perms, 'write') && has(perms, 'move') ? '' : 'none'; bdel.style.display = have_del && has(perms, 'delete') ? '' : 'none'; bcut.style.display = have_mv && has(perms, 'move') ? '' : 'none'; bpst.style.display = have_mv && has(perms, 'write') ? '' : 'none'; bpst.setAttribute('tt', 'paste ' + r.clip.length + ' items$NHotkey: ctrl-V'); ebi('wfm').style.display = QS('#wfm a.en:not([display])') ? '' : 'none'; }; r.rename = function (e) { ev(e); if (bren.style.display) return toast.err(3, 'cannot rename:\nyou do not have โ€œmoveโ€ permission in this folder'); var sel = msel.getsel(); if (!sel.length) return toast.err(3, 'select at least one item to rename'); var f = [], base = vsplit(sel[0].vp)[0], mkeys; for (var a = 0; a < sel.length; a++) { var vp = sel[a].vp; if (vp.endsWith('/')) vp = vp.slice(0, -1); var vsp = vsplit(vp); if (base != vsp[0]) return toast.err(0, 'bug:\n' + base + '\n' + vsp[0]); var vars = ft2dict(ebi(sel[a].id).closest('tr')); mkeys = vars[1].concat(vars[2]); var md = vars[0]; for (var k in md) { if (!md.hasOwnProperty(k)) continue; md[k.toLowerCase()] = md[k]; k = k.toLowerCase(); if (k.startsWith('.')) md[k.slice(1)] = md[k]; } md.t = md.ext; md.date = md.ts; md.size = md.sz; f.push({ "src": vp, "ofn": uricom_dec(vsp[1])[0], "md": vars[0], "ok": true }); } var rui = ebi('rui'); if (!rui) { rui = mknod('div'); rui.setAttribute('id', 'rui'); document.body.appendChild(rui); } var html = sel.length > 1 ? ['
'] : [ '
', '', '//', '' ]; html = html.concat([ '', '', 'advanced', 'case', '
', '
', '', '', '', '
regex
format
preset', '', '', '
' ]); if (sel.length == 1) html.push( '
\n' + '\n' + ''); else { html.push( '
old:
new:
' + ''); for (var a = 0; a < f.length; a++) html.push( '', '' + ''); } html.push('
new nameold name
' + '', '
'); if (sel.length == 1) { html.push('

tags for the selected file (read-only, just for reference):

'); for (var a = 0; a < mkeys.length; a++) html.push(''); html.push('
' + esc(mkeys[a]) + '
'); } rui.innerHTML = html.join('\n'); for (var a = 0; a < f.length; a++) { var k = '[n="' + a + '"]'; f[a].iold = QS('#rn_old' + k); f[a].inew = QS('#rn_new' + k); f[a].inew.value = f[a].iold.value = f[a].ofn; (function (a) { f[a].inew.onkeydown = function (e) { rn_ok(a, true); if (e.key == 'Escape') return rn_cancel(); if (e.key == 'Enter') return rn_apply(); }; QS('.rn_dec' + k).onclick = function () { f[a].inew.value = uricom_dec(f[a].inew.value)[0]; }; QS('.rn_reset' + k).onclick = function () { rn_reset(a); }; })(a); } rn_reset(0); tt.att(rui); var adv = bcfg_get('rn_adv', false), cs = bcfg_get('rn_case', false); function sadv() { ebi('rn_vadv').style.display = ebi('rn_case').style.display = adv ? '' : 'none'; } sadv(); function rn_ok(n, ok) { f[n].ok = ok; clmod(f[n].inew.closest('tr'), 'err', !ok); } function rn_reset(n) { f[n].inew.value = f[n].iold.value = f[n].ofn; f[n].inew.focus(); f[n].inew.setSelectionRange(0, f[n].inew.value.lastIndexOf('.'), "forward"); } function rn_cancel() { rui.parentNode.removeChild(rui); } ebi('rn_cancel').onclick = rn_cancel; ebi('rn_apply').onclick = rn_apply; ebi('rn_adv').onclick = function () { adv = !adv; bcfg_set('rn_adv', adv); sadv(); }; ebi('rn_case').onclick = function () { cs = !cs; bcfg_set('rn_case', cs); }; var ire = ebi('rn_re'), ifmt = ebi('rn_fmt'), ipre = ebi('rn_pre'), idel = ebi('rn_pdel'), inew = ebi('rn_pnew'), defp = '$lpad((tn),2,0). [(artist) - ](title).(ext)'; var presets = {}; presets[defp] = ['', defp]; presets = jread("rn_pre", presets); function spresets() { var keys = Object.keys(presets), o; keys.sort(); ipre.innerHTML = ''; for (var a = 0; a < keys.length; a++) { o = mknod('option'); o.setAttribute('value', keys[a]); o.textContent = keys[a]; ipre.appendChild(o); } } inew.onclick = function () { modal.prompt('provide a name for your new preset', ifmt.value, function (name) { if (!name) return toast.warn(3, 'aborted'); presets[name] = [ire.value, ifmt.value]; jwrite('rn_pre', presets); spresets(); ipre.value = name; }); }; idel.onclick = function () { delete presets[ipre.value]; jwrite('rn_pre', presets); spresets(); }; ipre.oninput = function () { var cfg = presets[ipre.value]; if (cfg) { ire.value = cfg[0]; ifmt.value = cfg[1]; } ifmt.oninput(); }; spresets(); ire.onkeydown = ifmt.onkeydown = function (e) { if (e.key == 'Escape') return rn_cancel(); if (e.key == 'Enter') return rn_apply(); }; ire.oninput = ifmt.oninput = function (e) { var ptn = ire.value, fmt = ifmt.value, re = null; if (!fmt) return; try { if (ptn) re = new RegExp(ptn, cs ? 'i' : ''); } catch (ex) { return toast.err(5, 'invalid regex:\n' + ex); } toast.hide(); for (var a = 0; a < f.length; a++) { var m = re ? re.exec(f[a].ofn) : null, ok, txt = ''; if (re && !m) { txt = 'regex did not match'; ok = false; } else { var ret = fmt_ren(m, f[a].md, fmt); ok = ret[0]; txt = ret[1]; } rn_ok(a, ok); f[a].inew.value = (ok ? '' : 'ERROR: ') + txt; } }; function rn_apply() { while (f.length && (!f[0].ok || f[0].ofn == f[0].inew.value)) f.shift(); if (!f.length) { toast.ok(2, 'rename OK'); treectl.goto(get_evpath()); return rn_cancel(); } toast.inf(0, 'renaming ' + f.length + ' items\n\n' + f[0].ofn); var dst = base + uricom_enc(f[0].inew.value, false); function rename_cb() { if (this.readyState != XMLHttpRequest.DONE) return; if (this.status !== 200) { var msg = this.responseText; toast.err(9, 'rename failed:\n' + msg); return; } f.shift().inew.value = '( OK )'; return rn_apply(); } var xhr = new XMLHttpRequest(); xhr.open('GET', f[0].src + '?move=' + dst, true); xhr.onreadystatechange = rename_cb; xhr.send(); } }; r.delete = function (e) { ev(e); if (bdel.style.display) return toast.err(3, 'cannot delete:\nyou do not have โ€œdeleteโ€ permission in this folder'); var sel = msel.getsel(), vps = []; for (var a = 0; a < sel.length; a++) vps.push(sel[a].vp); if (!sel.length) return toast.err(3, 'select at least 1 item to delete'); function deleter() { var xhr = new XMLHttpRequest(), vp = vps.shift(); if (!vp) { toast.ok(2, 'delete OK'); treectl.goto(get_evpath()); return; } toast.inf(0, 'deleting ' + (vps.length + 1) + ' items\n\n' + 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(9, 'delete failed:\n' + msg); return; } deleter(); } modal.confirm('===== DANGER =====\nDELETE these ' + vps.length + ' items?\n\n' + uricom_adec(vps).join('\n'), function () { modal.confirm('Last chance! Delete?', deleter, null); }, null); }; r.cut = function (e) { ev(e); if (bcut.style.display) return toast.err(3, 'cannot cut:\nyou do not have โ€œmoveโ€ permission in this folder'); var sel = msel.getsel(), vps = []; if (!sel.length) return toast.err(3, 'select at least 1 item to cut'); if (sel.length < 100) 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(1, 'cut ' + sel.length + ' items'); jwrite('fman_clip', vps); r.tx(1); }; r.paste = function (e) { ev(e); if (bpst.style.display) return toast.err(3, 'cannot paste:\nyou do not have โ€œwriteโ€ permission in this folder'); if (!r.clip.length) return toast.err(5, 'first cut some files/folders to paste\n\nnote: you can cut/paste across different browser tabs'); var req = [], exists = [], indir = [], srcdir = vsplit(r.clip[0])[0], links = QSA('#files tbody td:nth-child(2) a'); for (var a = 0, aa = links.length; a < aa; a++) indir.push(vsplit(links[a].getAttribute('href'))[1]); 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) toast.warn(30, 'these ' + exists.length + ' items cannot be pasted here (names already exist):\n\n' + uricom_adec(exists).join('\n')); if (!req.length) return; function paster() { var xhr = new XMLHttpRequest(), vp = req.shift(); if (!vp) { toast.ok(2, 'paste OK'); treectl.goto(get_evpath()); r.tx(srcdir); return; } toast.inf(0, 'pasting ' + (req.length + 1) + ' items\n\n' + uricom_dec(vp)[0]); 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(9, 'paste failed:\n' + msg); return; } paster(); } modal.confirm('paste these ' + req.length + ' items here?\n\n' + uricom_adec(req).join('\n'), function () { paster(); jwrite('fman_clip', []); }, null); }; if (r.bus) r.bus.onmessage = function (e) { r.clip = null; r.render(); var me = get_evpath(); if (e && e.data == me) treectl.goto(e.data); }; r.tx = function (msg) { if (!r.bus) return; r.bus.postMessage(msg); 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 = ( '
' + 'multiselect   zoom ' + ' ' + '+   sort by: ' + 'name, ' + 'size, ' + 'date, ' + 'type' + '
' + '
' ); 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('' + ao.innerHTML + ''); } 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 '' + (idx + 1) + ' / ' + r.bbox.length + ' -- ' + esc(uricom_dec(h.split('/').slice(-1)[0])[0]) + ''; } })[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) { var ae = document.activeElement, aet = ''; if (ae && ae != document.body) aet = ae.nodeName.toLowerCase(); if (e.altKey || e.isComposing) return; if (QS('#bbox-overlay.visible')) return; var k = e.code + '', pos = -1, n; if (aet == 'tr' && ae.closest('#files')) { var d = ''; if (k == 'ArrowUp') d = 'previous'; if (k == 'ArrowDown') d = 'next'; if (d) { var el = ae[d + 'ElementSibling']; if (el) { el.focus(); if (ctrl(e)) document.documentElement.scrollTop += (d == 'next' ? 1 : -1) * el.offsetHeight; if (e.shiftKey) { clmod(el, 'sel', 't'); msel.selui(); } return ev(e); } } if (k == 'Space') { clmod(ae, 'sel', 't'); msel.selui(); return ev(e); } if (k == 'KeyA' && ctrl(e)) { var sel = msel.getsel(), all = msel.getall(); msel.evsel(e, sel.length < all.length); return ev(e); } } if (aet && aet != 'a' && aet != 'tr') return; 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 = ['
' + sconf[a][0] + '']; for (var b = 1; b < 3; b++) { var hn = "srch_" + sconf[a][b][0], csp = (sconf[a].length == 2) ? 2 : 1; html.push( '\n' + '\n' + '
'); if (csp == 2) break; } html.push(''); trs.push(html); } var html = []; for (var a = 0; a < trs.length; a += 2) { html.push('' + (trs[a].concat(trs[a + 1])).join('\n') + '
'); } html.push('
raw
'); 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('
') === 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('');
		html.push('-! close search results');
		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 = ['-
' + links + '
', 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] += '' + sv; continue; } nodes.push(v); } nodes = nodes.concat([ext, unix2iso(ts)]); html.push(nodes.join('')); html.push(''); } 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(); 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; 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), dots = bcfg_get('dotfiles', false), 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 || document.visibilityState == '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'; } } timer.add(onscroll, true); 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 + (dots ? '&dots' : ''), true); xhr.onreadystatechange = recvtree; xhr.send(); enspin('#tree'); } function recvtree() { if (this.readyState != XMLHttpRequest.DONE) return; if (this.status !== 200) { toast.err(0, "recvtree, 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 = '
  • -[root]\n
  • '; } else { html = '-' + esc(name) + "\n"; 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' + (dots ? '&dots' : ''), 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) { toast.err(0, "recvls, 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 = '' + res.srvinf + ''; var top = this.top, nodes = res.dirs.concat(res.files), html = mk_files_header(res.taglist); html.push(''); 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 = ['' + r.lead + '' + hname + '', 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] += '' + sv; continue; } ln.push(v); } ln = ln.concat([r.ext, unix2iso(r.ts)]).join(''); html.push(ln + ''); } html.push(''); 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(); 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 = '' + sym + '' + hek + ''; if (res[kk]) { var subtree = parsetree(res[kk], url.slice(1)); ret += '
  • ' + link + '\n
  • \n'; } else { ret += '
  • ' + link + '
  • \n'; } } return ret; } function tdots(e) { ev(e); dots = !dots; bcfg_set('dotfiles', dots); treectl.goto(get_evpath()); } 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('dotfiles').onclick = tdots; 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', 'move', '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 = '' + (acct != '*' ? 'Logout ' + acct + '' : 'Login'); 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 = [ '', 'c', 'File Name', 'Size' ]; 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 = '' + tag.slice(1); else tag = '' + tag; html.push(tag + ''); } html = html.concat([ 'T', 'Date', '', ]); return html; } var filecols = (function () { var hidden = jread('filecols', []), tts = { "c": "action buttons", "dur": "duration", "q": "quality / bitrate", "Ac": "audio codec", "Vc": "video codec", "Ahash": "audio checksum", "Vhash": "video checksum", "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" }; if (JSON.stringify(def_hcols) != sread('hfilecols')) { console.log("applying default hidden-cols"); jwrite('hfilecols', def_hcols); for (var a = 0; a < def_hcols.length; a++) { var t = def_hcols[a]; t = t.slice(0, 1).toUpperCase() + t.slice(1); if (t.startsWith(".")) t = t.slice(1); if (hidden.indexOf(t) == -1) hidden.push(t); } jwrite("filecols", hidden); } 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 = '' + 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(''); } 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( '' + ''); 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( '' + ''); } 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 = []; if (r.all && r.all.length) { for (var a = 0; a < r.all.length; a++) { var ao = r.all[a]; ao.sel = ebi(ao.id).closest('tr').classList.contains('sel'); if (ao.sel) r.sel.push(ao); } return; } 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; r.all.push(item); if (item.sel) r.sel.push(item); links[a].closest('tr').setAttribute('tabindex', '0'); } }; r.getsel = function () { r.load(); return r.sel; }; r.getall = function () { r.load(); return r.all; }; r.selui = function (reset) { r.sel = null; if (reset) r.all = null; clmod(ebi('wtoggle'), 'sel', r.getsel().length); thegrid.loadsel(); fileman.render(); } r.seltgl = function (e) { ev(e); var tr = this.parentNode; clmod(tr, 'sel', 't'); r.selui(); } r.evsel = function (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); r.selui(); } ebi('selall').onclick = function (e) { r.evsel(e, "add"); }; ebi('selinv').onclick = function (e) { r.evsel(e, "t"); }; ebi('selzip').onclick = function (e) { ev(e); var sel = r.getsel(), arg = ebi('selzip').getAttribute('fmt'), frm = mknod('form'), txt = []; for (var a = 0; a < sel.length; a++) txt.push(vsplit(sel[a].vp)[1]); txt = txt.join('\n'); frm.setAttribute('action', '?' + arg); frm.setAttribute('method', 'post'); frm.setAttribute('target', '_blank'); frm.setAttribute('enctype', 'multipart/form-data'); frm.innerHTML = '' + ''; 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 = r.seltgl; } r.selui(true); 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); } var unpost = (function () { ebi('op_unpost').innerHTML = ( "you can delete your recent uploads below – click the fire-extinguisher icon to refresh" + '

    optional filter:  URL must contain clear filter

    ' + '
    ' ); var r = {}, ct = ebi('unpost'), filt = ebi('unpost_filt'); r.files = []; r.me = null; r.load = function () { var me = Date.now(), html = []; function unpost_load_cb() { if (this.readyState != XMLHttpRequest.DONE) return; if (this.status !== 200) { var msg = this.responseText; toast.err(9, 'unpost-load failed:\n' + msg); ebi('op_unpost').innerHTML = html.join('\n'); return; } var res = JSON.parse(this.responseText); if (res.length) { if (res.length == 2000) html.push("

    showing first 2000 files (use the filter)"); else html.push("

    " + res.length + " uploads can be deleted"); html.push(" – sorted by upload time – most recent first:

    "); html.push(""); } else html.push("

    sike! no uploads " + (filt.value ? 'matching that filter' : '') + " are sufficiently recent

    "); var mods = [1000, 100, 10]; for (var a = 0; a < res.length; a++) { for (var b = 0; b < mods.length; b++) if (a % mods[b] == 0 && res.length > a + mods[b] / 10) html.push( ''); html.push( '' + '' + '' + ''); } html.push("
    timesizefile
    ' + 'delete the next ' + Math.min(mods[b], res.length - a) + ' files below
    delete' + unix2iso(res[a].at) + '' + res[a].sz + '' + linksplit(res[a].vp).join(' ') + '
    "); ct.innerHTML = html.join('\n'); r.files = res; r.me = me; } var q = '/?ups'; if (filt.value) q += '&filter=' + uricom_enc(filt.value, true); var xhr = new XMLHttpRequest(); xhr.open('GET', q, true); xhr.onreadystatechange = unpost_load_cb; xhr.send(); ct.innerHTML = "

    loading your recent uploads...

    "; }; function unpost_delete_cb() { if (this.readyState != XMLHttpRequest.DONE) return; if (this.status !== 200) { var msg = this.responseText; toast.err(9, 'unpost-delete failed:\n' + msg); return; } for (var a = this.n; a < this.n2; a++) { var o = QSA('#op_unpost a.n' + a); for (var b = 0; b < o.length; b++) { var o2 = o[b].closest('tr'); o2.parentNode.removeChild(o2); } } toast.ok(5, this.responseText); if (!QS('#op_unpost a[me]')) ebi(goto_unpost()); } ct.onclick = function (e) { var tgt = e.target.closest('a[me]'); if (!tgt) return; if (!tgt.getAttribute('href')) return; var ame = tgt.getAttribute('me'); if (ame != r.me) return toast.err(0, 'something broke, please try a refresh'); var n = parseInt(tgt.className.slice(1)), n2 = parseInt(tgt.getAttribute('n2') || n + 1), req = []; for (var a = n; a < n2; a++) if (QS('#op_unpost a.n' + a)) req.push(r.files[a].vp); var links = QSA('#op_unpost a.n' + n); for (var a = 0, aa = links.length; a < aa; a++) { links[a].removeAttribute('href'); links[a].innerHTML = '[busy]'; } toast.inf(0, "deleting " + req.length + " files..."); var xhr = new XMLHttpRequest(); xhr.n = n; xhr.n2 = n2; xhr.open('POST', '/?delete', true); xhr.onreadystatechange = unpost_delete_cb; xhr.send(JSON.stringify(req)); }; var tfilt = null; filt.oninput = function () { clearTimeout(tfilt); tfilt = setTimeout(r.load, 250); }; ebi('unpost_nofilt').onclick = function () { filt.value = ''; r.load(); }; return r; })(); function goto_unpost(e) { unpost.load(); } 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(); msel.render(); } reload_browser(true); mukey.render(); play_linked();