mirror of
https://github.com/9001/copyparty.git
synced 2026-04-06 07:47:56 +10:00
11661 lines
408 KiB
JavaScript
11661 lines
408 KiB
JavaScript
"use strict";
|
||
|
||
var XHR = XMLHttpRequest,
|
||
img_re = /\.(a?png|avif|bmp|gif|heif|jpe?g|jfif|svg|webp|webm|mkv|mp4|m4v|mov)(\?|$)/i;
|
||
|
||
var Ls = {
|
||
"eng": {
|
||
"tt": "English",
|
||
|
||
"cols": {
|
||
"c": "action buttons",
|
||
"dur": "duration",
|
||
"q": "quality / bitrate",
|
||
"Ac": "audio codec",
|
||
"Vc": "video codec",
|
||
"Fmt": "format / container",
|
||
"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": "vertical resolution",
|
||
"chs": "audio channels",
|
||
"hz": "sample rate"
|
||
},
|
||
|
||
"hks": [
|
||
[
|
||
"misc",
|
||
["ESC", "close various things"],
|
||
|
||
"file-manager",
|
||
["G", "toggle list / grid view"],
|
||
["T", "toggle thumbnails / icons"],
|
||
["⇧ A/D", "thumbnail size"],
|
||
["ctrl-K", "delete selected"],
|
||
["ctrl-X", "cut selection to clipboard"],
|
||
["ctrl-C", "copy selection to clipboard"],
|
||
["ctrl-V", "paste (move/copy) here"],
|
||
["Y", "download selected"],
|
||
["F2", "rename selected"],
|
||
|
||
"file-list-sel",
|
||
["space", "toggle file selection"],
|
||
["↑/↓", "move selection cursor"],
|
||
["ctrl ↑/↓", "move cursor and viewport"],
|
||
["⇧ ↑/↓", "select prev/next file"],
|
||
["ctrl-A", "select all files / folders"],
|
||
], [
|
||
"navigation",
|
||
["B", "toggle breadcrumbs / navpane"],
|
||
["I/K", "prev/next folder"],
|
||
["M", "parent folder (or unexpand current)"],
|
||
["V", "toggle folders / textfiles in navpane"],
|
||
["A/D", "navpane size"],
|
||
], [
|
||
"audio-player",
|
||
["J/L", "prev/next song"],
|
||
["U/O", "skip 10sec back/fwd"],
|
||
["0..9", "jump to 0%..90%"],
|
||
["P", "play/pause (also initiates)"],
|
||
["S", "select playing song"],
|
||
["Y", "download song"],
|
||
], [
|
||
"image-viewer",
|
||
["J/L, ←/→", "prev/next pic"],
|
||
["Home/End", "first/last pic"],
|
||
["F", "fullscreen"],
|
||
["R", "rotate clockwise"],
|
||
["⇧ R", "rotate ccw"],
|
||
["S", "select pic"],
|
||
["Y", "download pic"],
|
||
], [
|
||
"video-player",
|
||
["U/O", "skip 10sec back/fwd"],
|
||
["P/K/Space", "play/pause"],
|
||
["C", "continue playing next"],
|
||
["V", "loop"],
|
||
["M", "mute"],
|
||
["[ and ]", "set loop interval"],
|
||
], [
|
||
"textfile-viewer",
|
||
["I/K", "prev/next file"],
|
||
["M", "close textfile"],
|
||
["E", "edit textfile"],
|
||
["S", "select file (for cut/copy/rename)"],
|
||
]
|
||
],
|
||
|
||
"m_ok": "OK",
|
||
"m_ng": "Cancel",
|
||
|
||
"enable": "Enable",
|
||
"danger": "DANGER",
|
||
"clipped": "copied to clipboard",
|
||
|
||
"ht_s1": "second",
|
||
"ht_s2": "seconds",
|
||
"ht_m1": "minute",
|
||
"ht_m2": "minutes",
|
||
"ht_h1": "hour",
|
||
"ht_h2": "hours",
|
||
"ht_d1": "day",
|
||
"ht_d2": "days",
|
||
"ht_and": " and ",
|
||
|
||
"goh": "control-panel",
|
||
"gop": 'previous sibling">prev',
|
||
"gou": 'parent folder">up',
|
||
"gon": 'next folder">next',
|
||
"logout": "Logout ",
|
||
"access": " access",
|
||
"ot_close": "close submenu",
|
||
"ot_search": "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> = start with «yana» and be an «opus» file$N<code>"try unite"</code> = contain exactly «try unite»$N$Nthe date format is iso-8601, like$N<code>2009-12-31</code> or <code>2020-09-12 23:30:00</code>",
|
||
"ot_unpost": "unpost: delete your recent uploads, or abort unfinished ones",
|
||
"ot_bup": "bup: basic uploader, even supports netscape 4.0",
|
||
"ot_mkdir": "mkdir: create a new directory",
|
||
"ot_md": "new-md: create a new markdown document",
|
||
"ot_msg": "msg: send a message to the server log",
|
||
"ot_mp": "media player options",
|
||
"ot_cfg": "configuration options",
|
||
"ot_u2i": 'up2k: upload files (if you have write-access) or toggle into the search-mode to see if they exist somewhere on the server$N$Nuploads are resumable, multithreaded, and file timestamps are preserved, but it uses more CPU than [🎈] (the basic uploader)<br /><br />during uploads, this icon becomes a progress indicator!',
|
||
"ot_u2w": 'up2k: upload files with resume support (close your browser and drop the same files in later)$N$Nmultithreaded, and file timestamps are preserved, but it uses more CPU than [🎈] (the basic uploader)<br /><br />during uploads, this icon becomes a progress indicator!',
|
||
"ot_noie": 'Please use Chrome / Firefox / Edge',
|
||
|
||
"ab_mkdir": "make directory",
|
||
"ab_mkdoc": "new markdown doc",
|
||
"ab_msg": "send msg to srv log",
|
||
|
||
"ay_path": "skip to folders",
|
||
"ay_files": "skip to files",
|
||
|
||
"wt_ren": "rename selected items$NHotkey: F2",
|
||
"wt_del": "delete selected items$NHotkey: ctrl-K",
|
||
"wt_cut": "cut selected items <small>(then paste somewhere else)</small>$NHotkey: ctrl-X",
|
||
"wt_cpy": "copy selected items to clipboard$N(to paste them somewhere else)$NHotkey: ctrl-C",
|
||
"wt_pst": "paste a previously cut / copied selection$NHotkey: ctrl-V",
|
||
"wt_selall": "select all files$NHotkey: ctrl-A (when file focused)",
|
||
"wt_selinv": "invert selection",
|
||
"wt_zip1": "download this folder as archive",
|
||
"wt_selzip": "download selection as archive",
|
||
"wt_seldl": "download selection as separate files$NHotkey: Y",
|
||
"wt_npirc": "copy irc-formatted track info",
|
||
"wt_nptxt": "copy plaintext track info",
|
||
"wt_m3ua": "add to m3u playlist (click <code>📻copy</code> later)",
|
||
"wt_m3uc": "copy m3u playlist to clipboard",
|
||
"wt_grid": "toggle grid / list view$NHotkey: G",
|
||
"wt_prev": "previous track$NHotkey: J",
|
||
"wt_play": "play / pause$NHotkey: P",
|
||
"wt_next": "next track$NHotkey: L",
|
||
|
||
"ul_par": "parallel uploads:",
|
||
"ut_rand": "randomize filenames",
|
||
"ut_u2ts": "copy the last-modified timestamp$Nfrom your filesystem to the server\">📅",
|
||
"ut_ow": "overwrite existing files on the server?$N🛡️: never (will generate a new filename instead)$N🕒: overwrite if server-file is older than yours$N♻️: always overwrite if the files are different",
|
||
"ut_mt": "continue hashing other files while uploading$N$Nmaybe disable if your CPU or HDD is a bottleneck",
|
||
"ut_ask": 'ask for confirmation before upload starts">💭',
|
||
"ut_pot": "improve upload speed on slow devices$Nby making the UI less complex",
|
||
"ut_srch": "don't actually upload, instead check if the files already $N exist on the server (will scan all folders you can read)",
|
||
"ut_par": "pause uploads by setting it to 0$N$Nincrease if your connection is slow / high latency$N$Nkeep it 1 on LAN or if the server HDD is a bottleneck",
|
||
"ul_btn": "drop files / folders<br>here (or click me)",
|
||
"ul_btnu": "U P L O A D",
|
||
"ul_btns": "S E A R C H",
|
||
|
||
"ul_hash": "hash",
|
||
"ul_send": "send",
|
||
"ul_done": "done",
|
||
"ul_idle1": "no uploads are queued yet",
|
||
"ut_etah": "average <em>hashing</em> speed, and estimated time until finish",
|
||
"ut_etau": "average <em>upload</em> speed and estimated time until finish",
|
||
"ut_etat": "average <em>total</em> speed and estimated time until finish",
|
||
|
||
"uct_ok": "completed successfully",
|
||
"uct_ng": "no-good: failed / rejected / not-found",
|
||
"uct_done": "ok and ng combined",
|
||
"uct_bz": "hashing or uploading",
|
||
"uct_q": "idle, pending",
|
||
|
||
"utl_name": "filename",
|
||
"utl_ulist": "list",
|
||
"utl_ucopy": "copy",
|
||
"utl_links": "links",
|
||
"utl_stat": "status",
|
||
"utl_prog": "progress",
|
||
|
||
// keep short:
|
||
"utl_404": "404",
|
||
"utl_err": "ERROR",
|
||
"utl_oserr": "OS-error",
|
||
"utl_found": "found",
|
||
"utl_defer": "defer",
|
||
"utl_yolo": "YOLO",
|
||
"utl_done": "done",
|
||
|
||
"ul_flagblk": "the files were added to the queue</b><br>however there is a busy up2k in another browser tab,<br>so waiting for that to finish first",
|
||
"ul_btnlk": "the server configuration has locked this switch into this state",
|
||
|
||
"udt_up": "Upload",
|
||
"udt_srch": "Search",
|
||
"udt_drop": "drop it here",
|
||
|
||
"u_nav_m": '<h6>aight, what do you have?</h6><code>Enter</code> = Files (one or more)\n<code>ESC</code> = One folder (including subfolders)',
|
||
"u_nav_b": '<a href="#" id="modal-ok">Files</a><a href="#" id="modal-ng">One folder</a>',
|
||
|
||
"cl_opts": "switches",
|
||
"cl_themes": "theme",
|
||
"cl_langs": "language",
|
||
"cl_ziptype": "folder download",
|
||
"cl_uopts": "up2k switches",
|
||
"cl_favico": "favicon",
|
||
"cl_bigdir": "big dirs",
|
||
"cl_hsort": "#sort",
|
||
"cl_keytype": "key notation",
|
||
"cl_hiddenc": "hidden columns",
|
||
"cl_hidec": "hide",
|
||
"cl_reset": "reset",
|
||
"cl_hpick": "tap on column headers to hide in the table below",
|
||
"cl_hcancel": "column hiding aborted",
|
||
|
||
"ct_grid": '田 the grid',
|
||
"ct_ttips": '◔ ◡ ◔">ℹ️ tooltips',
|
||
"ct_thumb": 'in grid-view, toggle icons or thumbnails$NHotkey: T">🖼️ thumbs',
|
||
"ct_csel": 'use CTRL and SHIFT for file selection in grid-view">sel',
|
||
"ct_ihop": 'when the image viewer is closed, scroll down to the last viewed file">g⮯',
|
||
"ct_dots": 'show hidden files (if server permits)">dotfiles',
|
||
"ct_qdel": 'when deleting files, only ask for confirmation once">qdel',
|
||
"ct_dir1st": 'sort folders before files">📁 first',
|
||
"ct_nsort": 'natural sort (for filenames with leading digits)">nsort',
|
||
"ct_readme": 'show README.md in folder listings">📜 readme',
|
||
"ct_idxh": 'show index.html instead of folder listing">htm',
|
||
"ct_sbars": 'show scrollbars">⟊',
|
||
|
||
"cut_umod": "if a file already exists on the server, update the server's last-modified timestamp to match your local file (requires write+delete permissions)\">re📅",
|
||
|
||
"cut_turbo": "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",
|
||
|
||
"cut_datechk": "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",
|
||
|
||
"cut_u2sz": "size (in MiB) of each upload chunk; big values fly better across the atlantic. Try low values on very unreliable connections",
|
||
|
||
"cut_flag": "ensure only one tab is uploading at a time $N -- other tabs must have this enabled too $N -- only affects tabs on the same domain",
|
||
|
||
"cut_az": "upload files in alphabetical order, rather than smallest-file-first$N$Nalphabetical order can make it easier to eyeball if something went wrong on the server, but it makes uploading slightly slower on fiber / LAN",
|
||
|
||
"cut_nag": "OS notification when upload completes$N(only if the browser or tab is not active)",
|
||
"cut_sfx": "audible alert when upload completes$N(only if the browser or tab is not active)",
|
||
|
||
"cut_mt": "use multithreading to accelerate file hashing$N$Nthis uses web-workers and requires$Nmore RAM (up to 512 MiB extra)$N$Nmakes https 30% faster, http 4.5x faster\">mt",
|
||
|
||
"cut_wasm": "use wasm instead of the browser's built-in hasher; improves speed on chrome-based browsers but increases CPU load, and many older versions of chrome have bugs which makes the browser consume all RAM and crash if this is enabled\">wasm",
|
||
|
||
"cft_text": "favicon text (blank and refresh to disable)",
|
||
"cft_fg": "foreground color",
|
||
"cft_bg": "background color",
|
||
|
||
"cdt_lim": "max number of files to show in a folder",
|
||
"cdt_ask": "when scrolling to the bottom,$Ninstead of loading more files,$Nask what to do",
|
||
"cdt_hsort": "how many sorting rules (<code>,sorthref</code>) to include in media-URLs. Setting this to 0 will also ignore sorting-rules included in media links when clicking them",
|
||
|
||
"tt_entree": "show navpane (directory tree sidebar)$NHotkey: B",
|
||
"tt_detree": "show breadcrumbs$NHotkey: B",
|
||
"tt_visdir": "scroll to selected folder",
|
||
"tt_ftree": "toggle folder-tree / textfiles$NHotkey: V",
|
||
"tt_pdock": "show parent folders in a docked pane at the top",
|
||
"tt_dynt": "autogrow as tree expands",
|
||
"tt_wrap": "word wrap",
|
||
"tt_hover": "reveal overflowing lines on hover$N( breaks scrolling unless mouse $N cursor is in the left gutter )",
|
||
|
||
"ml_pmode": "at end of folder...",
|
||
"ml_btns": "cmds",
|
||
"ml_tcode": "transcode",
|
||
"ml_tcode2": "transcode to",
|
||
"ml_tint": "tint",
|
||
"ml_eq": "audio equalizer",
|
||
"ml_drc": "dynamic range compressor",
|
||
|
||
"mt_loop": "loop/repeat one song\">🔁",
|
||
"mt_one": "stop after one song\">1️⃣",
|
||
"mt_shuf": "shuffle the songs in each folder\">🔀",
|
||
"mt_aplay": "autoplay if there is a song-ID in the link you clicked to access the server$N$Ndisabling this will also stop the page URL from being updated with song-IDs when playing music, to prevent autoplay if these settings are lost but the URL remains\">a▶",
|
||
"mt_preload": "start loading the next song near the end for gapless playback\">preload",
|
||
"mt_prescan": "go to the next folder before the last song$Nends, keeping the webbrowser happy$Nso it doesn't stop the playback\">nav",
|
||
"mt_fullpre": "try to preload the entire song;$N✅ enable on <b>unreliable</b> connections,$N❌ <b>disable</b> on slow connections probably\">full",
|
||
"mt_fau": "on phones, prevent music from stopping if the next song doesn't preload fast enough (can make tags display glitchy)\">☕️",
|
||
"mt_waves": "waveform seekbar:$Nshow audio amplitude in the scrubber\">~s",
|
||
"mt_npclip": "show buttons for clipboarding the currently playing song\">/np",
|
||
"mt_m3u_c": "show buttons for clipboarding the$Nselected songs as m3u8 playlist entries\">📻",
|
||
"mt_octl": "os integration (media hotkeys / osd)\">os-ctl",
|
||
"mt_oseek": "allow seeking through os integration$N$Nnote: on some devices (iPhones),$Nthis replaces the next-song button\">seek",
|
||
"mt_oscv": "show album cover in osd\">art",
|
||
"mt_follow": "keep the playing track scrolled into view\">🎯",
|
||
"mt_compact": "compact controls\">⟎",
|
||
"mt_uncache": "clear cache (try this if your browser cached$Na broken copy of a song so it refuses to play)\">uncache",
|
||
"mt_mloop": "loop the open folder\">🔁 loop",
|
||
"mt_mnext": "load the next folder and continue\">📂 next",
|
||
"mt_mstop": "stop playback\">⏸ stop",
|
||
"mt_cflac": "convert flac / wav to opus\">flac",
|
||
"mt_caac": "convert aac / m4a to opus\">aac",
|
||
"mt_coth": "convert all others (not mp3) to opus\">oth",
|
||
"mt_c2opus": "best choice for desktops, laptops, android\">opus",
|
||
"mt_c2owa": "opus-weba, for iOS 17.5 and newer\">owa",
|
||
"mt_c2caf": "opus-caf, for iOS 11 through 17\">caf",
|
||
"mt_c2mp3": "use this on very old devices\">mp3",
|
||
"mt_c2ok": "nice, good choice",
|
||
"mt_c2nd": "that's not the recommended output format for your device, but that's fine",
|
||
"mt_c2ng": "your device does not seem to support this output format, but let's try anyways",
|
||
"mt_xowa": "there are bugs in iOS preventing background playback using this format; please use caf or mp3 instead",
|
||
"mt_tint": "background level (0-100) on the seekbar$Nto make buffering less distracting",
|
||
"mt_eq": "enables the equalizer and gain control;$N$Nboost <code>0</code> = standard 100% volume (unmodified)$N$Nwidth <code>1 </code> = standard stereo (unmodified)$Nwidth <code>0.5</code> = 50% left-right crossfeed$Nwidth <code>0 </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = vocal removal :^)$N$Nenabling the equalizer makes gapless albums fully gapless, so leave it on with all the values at zero (except width = 1) if you care about that",
|
||
"mt_drc": "enables the dynamic range compressor (volume flattener / brickwaller); will also enable EQ to balance the spaghetti, so set all EQ fields except for 'width' to 0 if you don't want it$N$Nlowers the volume of audio above THRESHOLD dB; for every RATIO dB past THRESHOLD there is 1 dB of output, so default values of tresh -24 and ratio 12 means it should never get louder than -22 dB and it is safe to increase the equalizer boost to 0.8, or even 1.8 with ATK 0 and a huge RLS like 90 (only works in firefox; RLS is max 1 in other browsers)$N$N(see wikipedia, they explain it much better)",
|
||
|
||
"mb_play": "play",
|
||
"mm_hashplay": "play this audio file?",
|
||
"mm_m3u": "press <code>Enter/OK</code> to Play\npress <code>ESC/Cancel</code> to Edit",
|
||
"mp_breq": "need firefox 82+ or chrome 73+ or iOS 15+",
|
||
"mm_bload": "now loading...",
|
||
"mm_bconv": "converting to {0}, please wait...",
|
||
"mm_opusen": "your browser cannot play aac / m4a files;\ntranscoding to opus is now enabled",
|
||
"mm_playerr": "playback failed: ",
|
||
"mm_eabrt": "The playback attempt was cancelled",
|
||
"mm_enet": "Your internet connection is wonky",
|
||
"mm_edec": "This file is supposedly corrupted??",
|
||
"mm_esupp": "Your browser does not understand this audio format",
|
||
"mm_eunk": "Unknown Errol",
|
||
"mm_e404": "Could not play audio; error 404: File not found.",
|
||
"mm_e403": "Could not play audio; error 403: Access denied.\n\nTry pressing F5 to reload, maybe you got logged out",
|
||
"mm_e500": "Could not play audio; error 500: Check server logs.",
|
||
"mm_e5xx": "Could not play audio; server error ",
|
||
"mm_nof": "not finding any more audio files nearby",
|
||
"mm_prescan": "Looking for music to play next...",
|
||
"mm_scank": "Found the next song:",
|
||
"mm_uncache": "cache cleared; all songs will redownload on next playback",
|
||
"mm_hnf": "that song no longer exists",
|
||
|
||
"im_hnf": "that image no longer exists",
|
||
|
||
"f_empty": 'this folder is empty',
|
||
"f_chide": 'this will hide the column «{0}»\n\nyou can unhide columns in the settings tab',
|
||
"f_bigtxt": "this file is {0} MiB large -- really view as text?",
|
||
"f_bigtxt2": "view just the end of the file instead? this will also enable following/tailing, showing newly added lines of text in real time",
|
||
"fbd_more": '<div id="blazy">showing <code>{0}</code> of <code>{1}</code> files; <a href="#" id="bd_more">show {2}</a> or <a href="#" id="bd_all">show all</a></div>',
|
||
"fbd_all": '<div id="blazy">showing <code>{0}</code> of <code>{1}</code> files; <a href="#" id="bd_all">show all</a></div>',
|
||
"f_anota": "only {0} of the {1} items were selected;\nto select the full folder, first scroll to the bottom",
|
||
|
||
"f_dls": 'the file links in the current folder have\nbeen changed into download links',
|
||
|
||
"f_partial": "To safely download a file which is currently being uploaded, please click the file which has the same filename, but without the <code>.PARTIAL</code> file extension. Please press CANCEL or Escape to do this.\n\nPressing OK / Enter will ignore this warning and continue downloading the <code>.PARTIAL</code> scratchfile instead, which will almost definitely give you corrupted data.",
|
||
|
||
"ft_paste": "paste {0} items$NHotkey: ctrl-V",
|
||
"fr_eperm": 'cannot rename:\nyou do not have “move” permission in this folder',
|
||
"fd_eperm": 'cannot delete:\nyou do not have “delete” permission in this folder',
|
||
"fc_eperm": 'cannot cut:\nyou do not have “move” permission in this folder',
|
||
"fp_eperm": 'cannot paste:\nyou do not have “write” permission in this folder',
|
||
"fr_emore": "select at least one item to rename",
|
||
"fd_emore": "select at least one item to delete",
|
||
"fc_emore": "select at least one item to cut",
|
||
"fcp_emore": "select at least one item to copy to clipboard",
|
||
|
||
"fs_sc": "share the folder you're in",
|
||
"fs_ss": "share the selected files",
|
||
"fs_just1d": "you cannot select more than one folder,\nor mix files and folders in one selection",
|
||
"fs_abrt": "❌ abort",
|
||
"fs_rand": "🎲 rand.name",
|
||
"fs_go": "✅ create share",
|
||
"fs_name": "name",
|
||
"fs_src": "source",
|
||
"fs_pwd": "passwd",
|
||
"fs_exp": "expiry",
|
||
"fs_tmin": "min",
|
||
"fs_thrs": "hours",
|
||
"fs_tdays": "days",
|
||
"fs_never": "eternal",
|
||
"fs_pname": "optional link name; will be random if blank",
|
||
"fs_tsrc": "the file or folder to share",
|
||
"fs_ppwd": "optional password",
|
||
"fs_w8": "creating share...",
|
||
"fs_ok": "press <code>Enter/OK</code> to Clipboard\npress <code>ESC/Cancel</code> to Close",
|
||
|
||
"frt_dec": "may fix some cases of broken filenames\">url-decode",
|
||
"frt_rst": "reset modified filenames back to the original ones\">↺ reset",
|
||
"frt_abrt": "abort and close this window\">❌ cancel",
|
||
"frb_apply": "APPLY RENAME",
|
||
"fr_adv": "batch / metadata / pattern renaming\">advanced",
|
||
"fr_case": "case-sensitive regex\">case",
|
||
"fr_win": "windows-safe names; replace <code><>:"\\|?*</code> with japanese fullwidth characters\">win",
|
||
"fr_slash": "replace <code>/</code> with a character that doesn't cause new folders to be created\">no /",
|
||
"fr_re": "regex search pattern to apply to original filenames; capturing groups can be referenced in the format field below like <code>(1)</code> and <code>(2)</code> and so on",
|
||
"fr_fmt": "inspired by foobar2000:$N<code>(title)</code> is replaced by song title,$N<code>[(artist) - ](title)</code> skips [this] part if artist is blank$N<code>$lpad((tn),2,0)</code> pads tracknumber to 2 digits",
|
||
"fr_pdel": "delete",
|
||
"fr_pnew": "save as",
|
||
"fr_pname": "provide a name for your new preset",
|
||
"fr_aborted": "aborted",
|
||
"fr_lold": "old name",
|
||
"fr_lnew": "new name",
|
||
"fr_tags": "tags for the selected files (read-only, just for reference):",
|
||
"fr_busy": "renaming {0} items...\n\n{1}",
|
||
"fr_efail": "rename failed:\n",
|
||
"fr_nchg": "{0} of the new names were altered due to <code>win</code> and/or <code>no /</code>\n\nOK to continue with these altered new names?",
|
||
|
||
"fd_ok": "delete OK",
|
||
"fd_err": "delete failed:\n",
|
||
"fd_none": "nothing was deleted; maybe blocked by server config (xbd)?",
|
||
"fd_busy": "deleting {0} items...\n\n{1}",
|
||
"fd_warn1": "DELETE these {0} items?",
|
||
"fd_warn2": "<b>Last chance!</b> No way to undo. Delete?",
|
||
|
||
"fc_ok": "cut {0} items",
|
||
"fc_warn": 'cut {0} items\n\nbut: only <b>this</b> browser-tab can paste them\n(since the selection is so absolutely massive)',
|
||
|
||
"fcc_ok": "copied {0} items to clipboard",
|
||
"fcc_warn": 'copied {0} items to clipboard\n\nbut: only <b>this</b> browser-tab can paste them\n(since the selection is so absolutely massive)',
|
||
|
||
"fp_apply": "use these names",
|
||
"fp_ecut": "first cut or copy some files / folders to paste / move\n\nnote: you can cut / paste across different browser tabs",
|
||
"fp_ename": "{0} items cannot be moved here because the names are already taken. Give them new names below to continue, or blank the name to skip them:",
|
||
"fcp_ename": "{0} items cannot be copied here because the names are already taken. Give them new names below to continue, or blank the name to skip them:",
|
||
"fp_emore": "there are still some filename collisions left to fix",
|
||
"fp_ok": "move OK",
|
||
"fcp_ok": "copy OK",
|
||
"fp_busy": "moving {0} items...\n\n{1}",
|
||
"fcp_busy": "copying {0} items...\n\n{1}",
|
||
"fp_err": "move failed:\n",
|
||
"fcp_err": "copy failed:\n",
|
||
"fp_confirm": "move these {0} items here?",
|
||
"fcp_confirm": "copy these {0} items here?",
|
||
"fp_etab": 'failed to read clipboard from other browser tab',
|
||
"fp_name": "uploading a file from your device. Give it a name:",
|
||
"fp_both_m": '<h6>choose what to paste</h6><code>Enter</code> = Move {0} files from «{1}»\n<code>ESC</code> = Upload {2} files from your device',
|
||
"fcp_both_m": '<h6>choose what to paste</h6><code>Enter</code> = Copy {0} files from «{1}»\n<code>ESC</code> = Upload {2} files from your device',
|
||
"fp_both_b": '<a href="#" id="modal-ok">Move</a><a href="#" id="modal-ng">Upload</a>',
|
||
"fcp_both_b": '<a href="#" id="modal-ok">Copy</a><a href="#" id="modal-ng">Upload</a>',
|
||
|
||
"mk_noname": "type a name into the text field on the left before you do that :p",
|
||
|
||
"tv_load": "Loading text document:\n\n{0}\n\n{1}% ({2} of {3} MiB loaded)",
|
||
"tv_xe1": "could not load textfile:\n\nerror ",
|
||
"tv_xe2": "404, file not found",
|
||
"tv_lst": "list of textfiles in",
|
||
"tvt_close": "return to folder view$NHotkey: M (or Esc)\">❌ close",
|
||
"tvt_dl": "download this file$NHotkey: Y\">💾 download",
|
||
"tvt_prev": "show previous document$NHotkey: i\">⬆ prev",
|
||
"tvt_next": "show next document$NHotkey: K\">⬇ next",
|
||
"tvt_sel": "select file ( for cut / copy / delete / ... )$NHotkey: S\">sel",
|
||
"tvt_edit": "open file in text editor$NHotkey: E\">✏️ edit",
|
||
"tvt_tail": "monitor file for changes; show new lines in real time\">📡 follow",
|
||
"tvt_wrap": "word-wrap\">↵",
|
||
"tvt_atail": "lock scroll to bottom of page\">⚓",
|
||
"tvt_ctail": "decode terminal colors (ansi escape codes)\">🌈",
|
||
"tvt_ntail": "scrollback limit (how many bytes of text to keep loaded)",
|
||
|
||
"m3u_add1": "song added to m3u playlist",
|
||
"m3u_addn": "{0} songs added to m3u playlist",
|
||
"m3u_clip": "m3u playlist now copied to clipboard\n\nyou should create a new textfile named something.m3u and paste the playlist in that document; this will make it playable",
|
||
|
||
"gt_vau": "don't show videos, just play the audio\">🎧",
|
||
"gt_msel": "enable file selection; ctrl-click a file to override$N$N<em>when active: doubleclick a file / folder to open it</em>$N$NHotkey: S\">multiselect",
|
||
"gt_crop": "center-crop thumbnails\">crop",
|
||
"gt_3x": "hi-res thumbnails\">3x",
|
||
"gt_zoom": "zoom",
|
||
"gt_chop": "chop",
|
||
"gt_sort": "sort by",
|
||
"gt_name": "name",
|
||
"gt_sz": "size",
|
||
"gt_ts": "date",
|
||
"gt_ext": "type",
|
||
"gt_c1": "truncate filenames more (show less)",
|
||
"gt_c2": "truncate filenames less (show more)",
|
||
|
||
"sm_w8": "searching...",
|
||
"sm_prev": "search results below are from a previous query:\n ",
|
||
"sl_close": "close search results",
|
||
"sl_hits": "showing {0} hits",
|
||
"sl_moar": "load more",
|
||
|
||
"s_sz": "size",
|
||
"s_dt": "date",
|
||
"s_rd": "path",
|
||
"s_fn": "name",
|
||
"s_ta": "tags",
|
||
"s_ua": "up@",
|
||
"s_ad": "adv.",
|
||
"s_s1": "minimum MiB",
|
||
"s_s2": "maximum MiB",
|
||
"s_d1": "min. iso8601",
|
||
"s_d2": "max. iso8601",
|
||
"s_u1": "uploaded after",
|
||
"s_u2": "and/or before",
|
||
"s_r1": "path contains (space-separated)",
|
||
"s_f1": "name contains (negate with -nope)",
|
||
"s_t1": "tags contains (^=start, end=$)",
|
||
"s_a1": "specific metadata properties",
|
||
|
||
"md_eshow": "cannot render ",
|
||
"md_off": "[📜<em>readme</em>] disabled in [⚙️] -- document hidden",
|
||
|
||
"badreply": "Failed to parse reply from server",
|
||
|
||
"xhr403": "403: Access denied\n\ntry pressing F5, maybe you got logged out",
|
||
"xhr0": "unknown (probably lost connection to server, or server is offline)",
|
||
"cf_ok": "sorry about that -- DD" + wah + "oS protection kicked in\n\nthings should resume in about 30 sec\n\nif nothing happens, hit F5 to reload the page",
|
||
"tl_xe1": "could not list subfolders:\n\nerror ",
|
||
"tl_xe2": "404: Folder not found",
|
||
"fl_xe1": "could not list files in folder:\n\nerror ",
|
||
"fl_xe2": "404: Folder not found",
|
||
"fd_xe1": "could not create subfolder:\n\nerror ",
|
||
"fd_xe2": "404: Parent folder not found",
|
||
"fsm_xe1": "could not send message:\n\nerror ",
|
||
"fsm_xe2": "404: Parent folder not found",
|
||
"fu_xe1": "failed to load unpost list from server:\n\nerror ",
|
||
"fu_xe2": "404: File not found??",
|
||
|
||
"fz_tar": "uncompressed gnu-tar file (linux / mac)",
|
||
"fz_pax": "uncompressed pax-format tar (slower)",
|
||
"fz_targz": "gnu-tar with gzip level 3 compression$N$Nthis is usually very slow, so$Nuse uncompressed tar instead",
|
||
"fz_tarxz": "gnu-tar with xz level 1 compression$N$Nthis is usually very slow, so$Nuse uncompressed tar instead",
|
||
"fz_zip8": "zip with utf8 filenames (maybe wonky on windows 7 and older)",
|
||
"fz_zipd": "zip with traditional cp437 filenames, for really old software",
|
||
"fz_zipc": "cp437 with crc32 computed early,$Nfor MS-DOS PKZIP v2.04g (october 1993)$N(takes longer to process before download can start)",
|
||
|
||
"un_m1": "you can delete your recent uploads (or abort unfinished ones) below",
|
||
"un_upd": "refresh",
|
||
"un_m4": "or share the files visible below:",
|
||
"un_ulist": "show",
|
||
"un_ucopy": "copy",
|
||
"un_flt": "optional filter: URL must contain",
|
||
"un_fclr": "clear filter",
|
||
"un_derr": 'unpost-delete failed:\n',
|
||
"un_f5": 'something broke, please try a refresh or hit F5',
|
||
"un_uf5": "sorry but you have to refresh the page (for example by pressing F5 or CTRL-R) before this upload can be aborted",
|
||
"un_nou": '<b>warning:</b> server too busy to show unfinished uploads; click the "refresh" link in a bit',
|
||
"un_noc": '<b>warning:</b> unpost of fully uploaded files is not enabled/permitted in server config',
|
||
"un_max": "showing first 2000 files (use the filter)",
|
||
"un_avail": "{0} recent uploads can be deleted<br />{1} unfinished ones can be aborted",
|
||
"un_m2": "sorted by upload time; most recent first:",
|
||
"un_no1": "sike! no uploads are sufficiently recent",
|
||
"un_no2": "sike! no uploads matching that filter are sufficiently recent",
|
||
"un_next": "delete the next {0} files below",
|
||
"un_abrt": "abort",
|
||
"un_del": "delete",
|
||
"un_m3": "loading your recent uploads...",
|
||
"un_busy": "deleting {0} files...",
|
||
"un_clip": "{0} links copied to clipboard",
|
||
|
||
"u_https1": "you should",
|
||
"u_https2": "switch to https",
|
||
"u_https3": "for better performance",
|
||
"u_ancient": 'your browser is impressively ancient -- maybe you should <a href="#" onclick="goto(\'bup\')">use bup instead</a>',
|
||
"u_nowork": "need firefox 53+ or chrome 57+ or iOS 11+",
|
||
"tail_2old": "need firefox 105+ or chrome 71+ or iOS 14.5+",
|
||
"u_nodrop": 'your browser is too old for drag-and-drop uploading',
|
||
"u_notdir": "that's not a folder!\n\nyour browser is too old,\nplease try dragdrop instead",
|
||
"u_uri": "to dragdrop images from other browser windows,\nplease drop it onto the big upload button",
|
||
"u_enpot": 'switch to <a href="#">potato UI</a> (may improve upload speed)',
|
||
"u_depot": 'switch to <a href="#">fancy UI</a> (may reduce upload speed)',
|
||
"u_gotpot": 'switching to the potato UI for improved upload speed,\n\nfeel free to disagree and switch back!',
|
||
"u_pott": "<p>files: <b>{0}</b> finished, <b>{1}</b> failed, <b>{2}</b> busy, <b>{3}</b> queued</p>",
|
||
"u_ever": "this is the basic uploader; up2k needs at least<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1",
|
||
"u_su2k": 'this is the basic uploader; <a href="#" id="u2yea">up2k</a> is better',
|
||
"u_uput": 'optimize for speed (skip checksum)',
|
||
"u_ewrite": 'you do not have write-access to this folder',
|
||
"u_eread": 'you do not have read-access to this folder',
|
||
"u_enoi": 'file-search is not enabled in server config',
|
||
"u_enoow": "overwrite will not work here; need Delete-permission",
|
||
"u_badf": 'These {0} files (of {1} total) were skipped, possibly due to filesystem permissions:\n\n',
|
||
"u_blankf": 'These {0} files (of {1} total) are blank / empty; upload them anyways?\n\n',
|
||
"u_applef": 'These {0} files (of {1} total) are probably undesirable;\nPress <code>OK/Enter</code> to SKIP the following files,\nPress <code>Cancel/ESC</code> to NOT exclude, and UPLOAD those as well:\n\n',
|
||
"u_just1": '\nMaybe it works better if you select just one file',
|
||
"u_ff_many": "if you're using <b>Linux / MacOS / Android,</b> then this amount of files <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>may</em> crash Firefox!</a>\nif that happens, please try again (or use Chrome).",
|
||
"u_up_life": "This upload will be deleted from the server\n{0} after it completes",
|
||
"u_asku": 'upload these {0} files to <code>{1}</code>',
|
||
"u_unpt": "you can undo / delete this upload using the top-left 🧯",
|
||
"u_bigtab": 'about to show {0} files\n\nthis may crash your browser, are you sure?',
|
||
"u_scan": 'Scanning files...',
|
||
"u_dirstuck": 'directory iterator got stuck trying to access the following {0} items; will skip:',
|
||
"u_etadone": 'Done ({0}, {1} files)',
|
||
"u_etaprep": '(preparing to upload)',
|
||
"u_hashdone": 'hashing done',
|
||
"u_hashing": 'hash',
|
||
"u_hs": 'handshaking...',
|
||
"u_started": "the files are now being uploaded; see [🚀]",
|
||
"u_dupdefer": "duplicate; will be processed after all other files",
|
||
"u_actx": "click this text to prevent loss of<br />performance when switching to other windows/tabs",
|
||
"u_fixed": "OK! Fixed it 👍",
|
||
"u_cuerr": "failed to upload chunk {0} of {1};\nprobably harmless, continuing\n\nfile: {2}",
|
||
"u_cuerr2": "server rejected upload (chunk {0} of {1});\nwill retry later\n\nfile: {2}\n\nerror ",
|
||
"u_ehstmp": "will retry; see bottom-right",
|
||
"u_ehsfin": "server rejected the request to finalize upload; retrying...",
|
||
"u_ehssrch": "server rejected the request to perform search; retrying...",
|
||
"u_ehsinit": "server rejected the request to initiate upload; retrying...",
|
||
"u_eneths": "network error while performing upload handshake; retrying...",
|
||
"u_enethd": "network error while testing target existence; retrying...",
|
||
"u_cbusy": "waiting for server to trust us again after a network glitch...",
|
||
"u_ehsdf": "server ran out of disk space!\n\nwill keep retrying, in case someone\nfrees up enough space to continue",
|
||
"u_emtleak1": "it looks like your webbrowser may have a memory leak;\nplease",
|
||
"u_emtleak2": ' <a href="{0}">switch to https (recommended)</a> or ',
|
||
"u_emtleak3": ' ',
|
||
"u_emtleakc": 'try the following:\n<ul><li>hit <code>F5</code> to refresh the page</li><li>then disable the <code>mt</code> button in the <code>⚙️ settings</code></li><li>and try that upload again</li></ul>Uploads will be a bit slower, but oh well.\nSorry for the trouble !\n\nPS: chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">has a bugfix</a> for this',
|
||
"u_emtleakf": 'try the following:\n<ul><li>hit <code>F5</code> to refresh the page</li><li>then enable <code>🥔</code> (potato) in the upload UI<li>and try that upload again</li></ul>\nPS: firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">will hopefully have a bugfix</a> at some point',
|
||
"u_s404": "not found on server",
|
||
"u_expl": "explain",
|
||
"u_maxconn": "most browsers limit this to 6, but firefox lets you raise it with <code>connections-per-server</code> in <code>about:config</code>",
|
||
"u_tu": '<p class="warn">WARNING: turbo enabled, <span> client may not detect and resume incomplete uploads; see turbo-button tooltip</span></p>',
|
||
"u_ts": '<p class="warn">WARNING: turbo enabled, <span> search results can be incorrect; see turbo-button tooltip</span></p>',
|
||
"u_turbo_c": "turbo is disabled in server config",
|
||
"u_turbo_g": "disabling turbo because you don't have\ndirectory listing privileges within this volume",
|
||
"u_life_cfg": 'autodelete after <input id="lifem" p="60" /> min (or <input id="lifeh" p="3600" /> hours)',
|
||
"u_life_est": 'upload will be deleted <span id="lifew" tt="local time">---</span>',
|
||
"u_life_max": 'this folder enforces a\nmax lifetime of {0}',
|
||
"u_unp_ok": 'unpost is allowed for {0}',
|
||
"u_unp_ng": 'unpost will NOT be allowed',
|
||
"ue_ro": 'your access to this folder is Read-Only\n\n',
|
||
"ue_nl": 'you are currently not logged in',
|
||
"ue_la": 'you are currently logged in as "{0}"',
|
||
"ue_sr": 'you are currently in file-search mode\n\nswitch to upload-mode by clicking the magnifying glass 🔎 (next to the big SEARCH button), and try uploading again\n\nsorry',
|
||
"ue_ta": 'try uploading again, it should work now',
|
||
"ue_ab": "this file is already being uploaded into another folder, and that upload must be completed before the file can be uploaded elsewhere.\n\nYou can abort and forget the initial upload using the top-left 🧯",
|
||
"ur_1uo": "OK: File uploaded successfully",
|
||
"ur_auo": "OK: All {0} files uploaded successfully",
|
||
"ur_1so": "OK: File found on server",
|
||
"ur_aso": "OK: All {0} files found on server",
|
||
"ur_1un": "Upload failed, sorry",
|
||
"ur_aun": "All {0} uploads failed, sorry",
|
||
"ur_1sn": "File was NOT found on server",
|
||
"ur_asn": "The {0} files were NOT found on server",
|
||
"ur_um": "Finished;\n{0} uploads OK,\n{1} uploads failed, sorry",
|
||
"ur_sm": "Finished;\n{0} files found on server,\n{1} files NOT found on server",
|
||
|
||
"lang_set": "refresh to make the change take effect?",
|
||
},
|
||
"nor": {
|
||
"tt": "Norsk",
|
||
|
||
"cols": {
|
||
"c": "handlingsknapper",
|
||
"dur": "varighet",
|
||
"q": "kvalitet / bitrate",
|
||
"Ac": "lyd-format",
|
||
"Vc": "video-format",
|
||
"Fmt": "format / innpakning",
|
||
"Ahash": "lyd-kontrollsum",
|
||
"Vhash": "video-kontrollsum",
|
||
"Res": "oppløsning",
|
||
"T": "filtype",
|
||
"aq": "lydkvalitet / bitrate",
|
||
"vq": "videokvalitet / bitrate",
|
||
"pixfmt": "fargekoding / detaljenivå",
|
||
"resw": "horisontal oppløsning",
|
||
"resh": "vertikal oppløsning",
|
||
"chs": "lydkanaler",
|
||
"hz": "lyd-oppløsning"
|
||
},
|
||
|
||
"hks": [
|
||
[
|
||
"ymse",
|
||
["ESC", "lukk saker og ting"],
|
||
|
||
"filbehandler",
|
||
["G", "listevisning eller ikoner"],
|
||
["T", "miniatyrbilder på/av"],
|
||
["⇧ A/D", "ikonstørrelse"],
|
||
["ctrl-K", "slett valgte"],
|
||
["ctrl-X", "klipp ut valgte"],
|
||
["ctrl-C", "kopiér til utklippstavle"],
|
||
["ctrl-V", "lim inn (flytt/kopiér)"],
|
||
["Y", "last ned valgte"],
|
||
["F2", "endre navn på valgte"],
|
||
|
||
"filmarkering",
|
||
["space", "marker fil"],
|
||
["↑/↓", "flytt markør"],
|
||
["ctrl ↑/↓", "flytt markør og scroll"],
|
||
["⇧ ↑/↓", "velg forr./neste fil"],
|
||
["ctrl-A", "velg alle filer / mapper"],
|
||
], [
|
||
"navigering",
|
||
["B", "mappehierarki eller filsti"],
|
||
["I/K", "forr./neste mappe"],
|
||
["M", "ett nivå opp (eller lukk)"],
|
||
["V", "vis mapper eller tekstfiler"],
|
||
["A/D", "panelstørrelse"],
|
||
], [
|
||
"musikkspiller",
|
||
["J/L", "forr./neste sang"],
|
||
["U/O", "hopp 10sek bak/frem"],
|
||
["0..9", "hopp til 0%..90%"],
|
||
["P", "pause, eller start / fortsett"],
|
||
["S", "marker spillende sang"],
|
||
["Y", "last ned sang"],
|
||
], [
|
||
"bildeviser",
|
||
["J/L, ←/→", "forr./neste bilde"],
|
||
["Home/End", "første/siste bilde"],
|
||
["F", "fullskjermvisning"],
|
||
["R", "rotere mot høyre"],
|
||
["⇧ R", "rotere mot venstre"],
|
||
["S", "marker bilde"],
|
||
["Y", "last ned bilde"],
|
||
], [
|
||
"videospiller",
|
||
["U/O", "hopp 10sek bak/frem"],
|
||
["P/K/Space", "pause / fortsett"],
|
||
["C", "fortsett til neste fil"],
|
||
["V", "gjenta avspilling"],
|
||
["M", "lyd av/på"],
|
||
["[ og ]", "gjentaksintervall"],
|
||
], [
|
||
"dokumentviser",
|
||
["I/K", "forr./neste fil"],
|
||
["M", "lukk tekstdokument"],
|
||
["E", "rediger tekstdokument"],
|
||
["S", "marker fil (for F2/ctrl-x/...)"],
|
||
["Y", "last ned tekstfil"],
|
||
]
|
||
],
|
||
|
||
"m_ok": "OK",
|
||
"m_ng": "Avbryt",
|
||
|
||
"enable": "Aktiv",
|
||
"danger": "VARSKU",
|
||
"clipped": "kopiert til utklippstavlen",
|
||
|
||
"ht_s1": "sekund",
|
||
"ht_s2": "sekunder",
|
||
"ht_m1": "minutt",
|
||
"ht_m2": "minutter",
|
||
"ht_h1": "time",
|
||
"ht_h2": "timer",
|
||
"ht_d1": "dag",
|
||
"ht_d2": "dager",
|
||
"ht_and": " og ",
|
||
|
||
"goh": "kontrollpanel",
|
||
"gop": 'naviger til mappen før denne">forr.',
|
||
"gou": 'naviger ett nivå opp">opp',
|
||
"gon": 'naviger til mappen etter denne">neste',
|
||
"logout": "Logg ut ",
|
||
"access": " tilgang",
|
||
"ot_close": "lukk verktøy",
|
||
"ot_search": "søk etter filer ved å angi filnavn, mappenavn, tid, størrelse, eller metadata som sangtittel / artist / osv.$N$N<code>foo bar</code> = inneholder både «foo» og «bar»,$N<code>foo -bar</code> = inneholder «foo» men ikke «bar»,$N<code>^yana .opus$</code> = starter med «yana», filtype «opus»$N<code>"try unite"</code> = «try unite» eksakt$N$Ndatoformat er iso-8601, så f.eks.$N<code>2009-12-31</code> eller <code>2020-09-12 23:30:00</code>",
|
||
"ot_unpost": "unpost: slett filer som du nylig har lastet opp; «angre-knappen»",
|
||
"ot_bup": "bup: tradisjonell / primitiv filopplastning,$N$Nfungerer i omtrent samtlige nettlesere",
|
||
"ot_mkdir": "mkdir: lag en ny mappe",
|
||
"ot_md": "new-md: lag et nytt markdown-dokument",
|
||
"ot_msg": "msg: send en beskjed til serverloggen",
|
||
"ot_mp": "musikkspiller-instillinger",
|
||
"ot_cfg": "andre innstillinger",
|
||
"ot_u2i": 'up2k: last opp filer (hvis du har skrivetilgang) eller bytt til søkemodus for å sjekke om filene finnes et-eller-annet sted på serveren$N$Nopplastninger kan gjenopptas etter avbrudd, skjer stykkevis for potensielt høyere ytelse, og ivaretar datostempling -- men bruker litt mer prosessorkraft enn [🎈] (den primitive opplasteren "bup")<br /><br />mens opplastninger foregår så vises fremdriften her oppe!',
|
||
"ot_u2w": 'up2k: filopplastning med støtte for å gjenoppta avbrutte opplastninger -- steng ned nettleseren og dra de samme filene inn i nettleseren igjen for å plukke opp igjen der du slapp$N$Nopplastninger skjer stykkevis for potensielt høyere ytelse, og ivaretar datostempling -- men bruker litt mer prosessorkraft enn [🎈] (den primitive opplasteren "bup")<br /><br />mens opplastninger foregår så vises fremdriften her oppe!',
|
||
"ot_noie": 'Fungerer mye bedre i Chrome / Firefox / Edge',
|
||
|
||
"ab_mkdir": "lag mappe",
|
||
"ab_mkdoc": "nytt dokument",
|
||
"ab_msg": "send melding",
|
||
|
||
"ay_path": "gå videre til mapper",
|
||
"ay_files": "gå videre til filer",
|
||
|
||
"wt_ren": "gi nye navn til de valgte filene$NSnarvei: F2",
|
||
"wt_del": "slett de valgte filene$NSnarvei: ctrl-K",
|
||
"wt_cut": "klipp ut de valgte filene <small>(for å lime inn et annet sted)</small>$NSnarvei: ctrl-X",
|
||
"wt_cpy": "kopiér de valgte filene til utklippstavlen$N(for å lime inn et annet sted)$NSnarvei: ctrl-C",
|
||
"wt_pst": "lim inn filer (som tidligere ble klippet ut / kopiert et annet sted)$NSnarvei: ctrl-V",
|
||
"wt_selall": "velg alle filer$NSnarvei: ctrl-A (mens fokus er på en fil)",
|
||
"wt_selinv": "inverter utvalg",
|
||
"wt_zip1": "last ned denne mappen som et arkiv",
|
||
"wt_selzip": "last ned de valgte filene som et arkiv",
|
||
"wt_seldl": "last ned de valgte filene$NSnarvei: Y",
|
||
"wt_npirc": "kopiér sang-info (irc-formatert)",
|
||
"wt_nptxt": "kopiér sang-info",
|
||
"wt_m3ua": "legg til sang i m3u-spilleliste$N(husk å klikke på <code>📻copy</code> senere)",
|
||
"wt_m3uc": "kopiér m3u-spillelisten til utklippstavlen",
|
||
"wt_grid": "bytt mellom ikoner og listevisning$NSnarvei: G",
|
||
"wt_prev": "forrige sang$NSnarvei: J",
|
||
"wt_play": "play / pause$NSnarvei: P",
|
||
"wt_next": "neste sang$NSnarvei: L",
|
||
|
||
"ul_par": "samtidige handl.:",
|
||
"ut_rand": "finn opp nye tilfeldige filnavn",
|
||
"ut_u2ts": "gi filen på serveren samme$Ntidsstempel som lokalt hos deg\">📅",
|
||
"ut_ow": "overskrive eksisterende filer på serveren?$N🛡️: aldri (finner på et nytt filnavn istedenfor)$N🕒: overskriv hvis serverens fil er eldre$N♻️: alltid, gitt at innholdet er forskjellig",
|
||
"ut_mt": "fortsett å befare køen mens opplastning foregår$N$Nskru denne av dersom du har en$Ntreg prosessor eller harddisk",
|
||
"ut_ask": 'bekreft filutvalg før opplastning starter">💭',
|
||
"ut_pot": "forbedre ytelsen på trege enheter ved å$Nforenkle brukergrensesnittet",
|
||
"ut_srch": "utfør søk istedenfor å laste opp --$Nleter igjennom alle mappene du har lov til å se",
|
||
"ut_par": "sett til 0 for å midlertidig stanse opplastning$N$Nhøye verdier (4 eller 8) kan gi bedre ytelse,$Nspesielt på trege internettlinjer$N$Nbør ikke være høyere enn 1 på LAN$Neller hvis serveren sin harddisk er treg",
|
||
"ul_btn": "slipp filer / mapper<br>her (eller klikk meg)",
|
||
"ul_btnu": "L A S T O P P",
|
||
"ul_btns": "F I L S Ø K",
|
||
|
||
"ul_hash": "befar",
|
||
"ul_send": " send",
|
||
"ul_done": "total",
|
||
"ul_idle1": "ingen handlinger i køen",
|
||
"ut_etah": "snitthastighet for <em>befaring</em> samt gjenstående tid",
|
||
"ut_etau": "snitthastighet for <em>opplastning</em> samt gjenstående tid",
|
||
"ut_etat": "<em>total</em> snitthastighet og gjenstående tid",
|
||
|
||
"uct_ok": "fullført uten problemer",
|
||
"uct_ng": "fullført under tvil (duplikat, ikke funnet, ...)",
|
||
"uct_done": "fullført (enten <em>ok</em> eller <em>ng</em>)",
|
||
"uct_bz": "aktive handlinger (befaring / opplastning)",
|
||
"uct_q": "køen",
|
||
|
||
"utl_name": "filnavn",
|
||
"utl_ulist": "vis",
|
||
"utl_ucopy": "kopiér",
|
||
"utl_links": "lenker",
|
||
"utl_stat": "status",
|
||
"utl_prog": "fremdrift",
|
||
|
||
// må være korte:
|
||
"utl_404": "404",
|
||
"utl_err": "FEIL!",
|
||
"utl_oserr": "OS-feil",
|
||
"utl_found": "funnet",
|
||
"utl_defer": "senere",
|
||
"utl_yolo": "YOLO",
|
||
"utl_done": "ferdig",
|
||
|
||
"ul_flagblk": "filene har blitt lagt i køen</b><br>men det er en annen nettleserfane som holder på med befaring eller opplastning akkurat nå,<br>så venter til den er ferdig først",
|
||
"ul_btnlk": "bryteren har blitt låst til denne tilstanden i serverens konfigurasjon",
|
||
|
||
"udt_up": "Last opp",
|
||
"udt_srch": "Søk",
|
||
"udt_drop": "Slipp filene her",
|
||
|
||
"u_nav_m": '<h6>hva har du?</h6><code>Enter</code> = Filer (én eller flere)\n<code>ESC</code> = Én mappe (inkludert undermapper)',
|
||
"u_nav_b": '<a href="#" id="modal-ok">Filer</a><a href="#" id="modal-ng">Én mappe</a>',
|
||
|
||
"cl_opts": "brytere",
|
||
"cl_themes": "utseende",
|
||
"cl_langs": "språk",
|
||
"cl_ziptype": "nedlastning av mapper",
|
||
"cl_uopts": "up2k-brytere",
|
||
"cl_favico": "favicon",
|
||
"cl_bigdir": "store mapper",
|
||
"cl_hsort": "#sort",
|
||
"cl_keytype": "notasjon for musikalsk dur",
|
||
"cl_hiddenc": "skjulte kolonner",
|
||
"cl_hidec": "skjul",
|
||
"cl_reset": "nullstill",
|
||
"cl_hpick": "klikk på overskriften til kolonnene du ønsker å skjule i tabellen nedenfor",
|
||
"cl_hcancel": "kolonne-skjuling avbrutt",
|
||
|
||
"ct_grid": '田 ikoner',
|
||
"ct_ttips": 'hvis hjelpetekst ved å holde musen over ting">ℹ️ tips',
|
||
"ct_thumb": 'vis miniatyrbilder istedenfor ikoner$NSnarvei: T">🖼️ bilder',
|
||
"ct_csel": 'bruk tastene CTRL og SHIFT for markering av filer i ikonvisning">merk',
|
||
"ct_ihop": 'bla ned til sist viste bilde når bildeviseren lukkes">g⮯',
|
||
"ct_dots": 'vis skjulte filer (gitt at serveren tillater det)">.synlig',
|
||
"ct_qdel": 'sletteknappen spør bare én gang om bekreftelse">hurtig🗑️',
|
||
"ct_dir1st": 'sorter slik at mapper kommer foran filer">📁 først',
|
||
"ct_nsort": 'naturlig sortering (forstår tall i filnavn)">nsort',
|
||
"ct_readme": 'vis README.md nedenfor filene">📜 readme',
|
||
"ct_idxh": 'vis index.html istedenfor fil-liste">htm',
|
||
"ct_sbars": 'vis rullgardiner / skrollefelt">⟊',
|
||
|
||
"cut_umod": 'i tilfelle en fil du laster opp allerede finnes på serveren, så skal serverens tidsstempel oppdateres slik at det stemmer overens med din lokale fil (krever rettighetene write+delete)">re📅',
|
||
|
||
"cut_turbo": "forenklet befaring ved opplastning; bør sannsynlig <em>ikke</em> skrus på:$N$Nnyttig dersom du var midt i en svær opplastning som måtte restartes av en eller annen grunn, og du vil komme igang igjen så raskt som overhodet mulig.$N$Nnår denne er skrudd på så forenkles befaringen kraftig; istedenfor å utføre en trygg sjekk på om filene finnes på serveren i god stand, så sjekkes kun om <em>filstørrelsen</em> stemmer. Så dersom en korrupt fil skulle befinne seg på serveren allerede, på samme sted med samme størrelse og navn, så blir det <em>ikke oppdaget</em>.$N$Ndet anbefales å kun benytte denne funksjonen for å komme seg raskt igjennom selve opplastningen, for så å skru den av, og til slutt "laste opp" de samme filene én gang til -- slik at integriteten kan verifiseres\">turbo",
|
||
|
||
"cut_datechk": "har ingen effekt dersom turbo er avslått$N$Ngjør turbo bittelitt tryggere ved å sjekke datostemplingen på filene (i tillegg til filstørrelse)$N$N<em>burde</em> oppdage og gjenoppta de fleste ufullstendige opplastninger, men er <em>ikke</em> en fullverdig erstatning for å deaktivere turbo og gjøre en skikkelig sjekk\">date-chk",
|
||
|
||
"cut_u2sz": "størrelse i megabyte for hvert bruddstykke for opplastning. Store verdier flyr bedre over atlanteren. Små verdier kan være bedre på særdeles ustabile forbindelser",
|
||
|
||
"cut_flag": "samkjører nettleserfaner slik at bare én $N kan holde på med befaring / opplastning $N -- andre faner må også ha denne skrudd på $N -- fungerer kun innenfor samme domene",
|
||
|
||
"cut_az": "last opp filer i alfabetisk rekkefølge, istedenfor minste-fil-først$N$Nalfabetisk kan gjøre det lettere å anslå om alt gikk bra, men er bittelitt tregere på fiber / LAN",
|
||
|
||
"cut_nag": "meldingsvarsel når opplastning er ferdig$N(kun on nettleserfanen ikke er synlig)",
|
||
"cut_sfx": "lydvarsel når opplastning er ferdig$N(kun on nettleserfanen ikke er synlig)",
|
||
|
||
"cut_mt": "raskere befaring ved å bruke hele CPU'en$N$Ndenne funksjonen anvender web-workers$Nog krever mer RAM (opptil 512 MiB ekstra)$N$Ngjør https 30% raskere, http 4.5x raskere\">mt",
|
||
|
||
"cut_wasm": "bruk wasm istedenfor nettleserens sha512-funksjon; gir bedre ytelse på chrome-baserte nettlesere, men bruker mere CPU, og eldre versjoner av chrome tåler det ikke (spiser opp all RAM og krasjer)\">wasm",
|
||
|
||
"cft_text": "ikontekst (blank ut og last siden på nytt for å deaktivere)",
|
||
"cft_fg": "farge",
|
||
"cft_bg": "bakgrunnsfarge",
|
||
|
||
"cdt_lim": "maks antall filer å vise per mappe",
|
||
"cdt_ask": "vis knapper for å laste flere filer nederst på siden istedenfor å gradvis laste mer av mappen når man scroller ned",
|
||
"cdt_hsort": "antall sorterings-regler (<code>,sorthref</code>) som skal inkluderes når media-URL'er genereres. Hvis denne er 0 så vil sorterings-regler i URL'er hverken bli generert eller lest",
|
||
|
||
"tt_entree": "bytt til mappehierarki$NSnarvei: B",
|
||
"tt_detree": "bytt til tradisjonell sti-visning$NSnarvei: B",
|
||
"tt_visdir": "bla ned til den åpne mappen",
|
||
"tt_ftree": "bytt mellom filstruktur og tekstfiler$NSnarvei: V",
|
||
"tt_pdock": "vis de overordnede mappene i et panel",
|
||
"tt_dynt": "øk bredden på panelet ettersom treet utvider seg",
|
||
"tt_wrap": "linjebryting",
|
||
"tt_hover": "vis hele mappenavnet når musepekeren treffer mappen$N( gjør dessverre at scrollhjulet fusker dersom musepekeren ikke befinner seg i grøfta )",
|
||
|
||
"ml_pmode": "ved enden av mappen",
|
||
"ml_btns": "knapper",
|
||
"ml_tcode": "konvertering",
|
||
"ml_tcode2": "konverter til",
|
||
"ml_tint": "tint",
|
||
"ml_eq": "audio equalizer (tonejustering)",
|
||
"ml_drc": "compressor (volum-utjevning)",
|
||
|
||
"mt_loop": "spill den samme sangen om og om igjen\">🔁",
|
||
"mt_one": "spill kun én sang\">1️⃣",
|
||
"mt_shuf": "sangene i hver mappe$Nspilles i tilfeldig rekkefølge\">🔀",
|
||
"mt_aplay": "forsøk å starte avspilling hvis linken du klikket på for å åpne nettsiden inneholder en sang-ID$N$Nhvis denne deaktiveres så vil heller ikke nettside-URLen bli oppdatert med sang-ID'er når musikk spilles, i tilfelle innstillingene skulle gå tapt og nettsiden lastes på ny\">a▶",
|
||
"mt_preload": "hent ned litt av neste sang i forkant,$Nslik at pausen i overgangen blir mindre\">forles",
|
||
"mt_prescan": "ved behov, bla til neste mappe$Nslik at nettleseren lar oss$Nfortsette å spille musikk\">bla",
|
||
"mt_fullpre": "hent ned hele neste sang, ikke bare litt:$N✅ skru på hvis nettet ditt er <b>ustabilt</b>,$N❌ skru av hvis nettet ditt er <b>tregt</b>\">full",
|
||
"mt_fau": "for telefoner: forhindre at avspilling stopper hvis nettet er for tregt til å laste neste sang i tide. Hvis påskrudd, kan forårsake at sang-info ikke vises korrekt i OS'et\">☕️",
|
||
"mt_waves": "waveform seekbar:$Nvis volumkurve i avspillingsfeltet\">~s",
|
||
"mt_npclip": "vis knapper for å kopiere info om sangen du hører på\">/np",
|
||
"mt_m3u_c": "vis knapper for å kopiere de valgte$Nsangene som innslag i en m3u8 spilleliste\">📻",
|
||
"mt_octl": "integrering med operativsystemet (fjernkontroll, info-skjerm)\">os-ctl",
|
||
"mt_oseek": "tillat spoling med fjernkontroll$N$Nmerk: på noen enheter (iPhones) så vil$Ndette erstatte knappen for neste sang\">spoling",
|
||
"mt_oscv": "vis album-cover på infoskjermen\">bilde",
|
||
"mt_follow": "bla slik at sangen som spilles alltid er synlig\">🎯",
|
||
"mt_compact": "tettpakket avspillerpanel\">⟎",
|
||
"mt_uncache": "prøv denne hvis en sang ikke spiller riktig\">oppfrisk",
|
||
"mt_mloop": "repeter hele mappen\">🔁 gjenta",
|
||
"mt_mnext": "hopp til neste mappe og fortsett\">📂 neste",
|
||
"mt_mstop": "stopp avspilling\">⏸ stopp",
|
||
"mt_cflac": "konverter flac / wav-filer til opus\">flac",
|
||
"mt_caac": "konverter aac / m4a-filer til to opus\">aac",
|
||
"mt_coth": "konverter alt annet (men ikke mp3) til opus\">andre",
|
||
"mt_c2opus": "det beste valget for alle PCer og Android\">opus",
|
||
"mt_c2owa": "opus-weba, for iOS 17.5 og nyere\">owa",
|
||
"mt_c2caf": "opus-caf, for iOS 11 tilogmed 17\">caf",
|
||
"mt_c2mp3": "bra valg for steinalder-utstyr (slår aldri feil)\">mp3",
|
||
"mt_c2ok": "bra valg!",
|
||
"mt_c2nd": "ikke det foretrukne valget for din enhet, men funker sikkert greit",
|
||
"mt_c2ng": "ser virkelig ikke ut som enheten din takler dette formatet... men ok, vi prøver",
|
||
"mt_xowa": "iOS har fortsatt problemer med avspilling av owa-musikk i bakgrunnen. Bruk caf eller mp3 istedenfor",
|
||
"mt_tint": "nivå av bakgrunnsfarge på søkestripa (0-100),$Ngjør oppdateringer mindre distraherende",
|
||
"mt_eq": "aktiver tonekontroll og forsterker;$N$Nboost <code>0</code> = normal volumskala$N$Nwidth <code>1 </code> = normal stereo$Nwidth <code>0.5</code> = 50% blanding venstre-høyre$Nwidth <code>0 </code> = mono$N$Nboost <code>-0.8</code> & width <code>10</code> = instrumental :^)$N$Nreduserer også dødtid imellom sangfiler",
|
||
"mt_drc": "aktiver volum-utjevning (dynamic range compressor); vil også aktivere tonejustering, så sett alle EQ-feltene bortsett fra 'width' til 0 hvis du ikke vil ha noe EQ$N$Nfilteret vil dempe volumet på alt som er høyere enn TRESH dB; for hver RATIO dB over grensen er det 1dB som treffer høyttalerne, så standardverdiene tresh -24 og ratio 12 skal bety at volumet ikke går høyere enn -22 dB, slik at man trygt kan øke boost-verdien i equalizer'n til rundt 0.8, eller 1.8 kombinert med ATK 0 og RLS 90 (bare mulig i firefox; andre nettlesere tar ikke høyere RLS enn 1)$N$Nwikipedia forklarer dette mye bedre forresten",
|
||
|
||
"mb_play": "lytt",
|
||
"mm_hashplay": "spill denne sangen?",
|
||
"mm_m3u": "trykk <code>Enter/OK</code> for å spille\ntrykk <code>ESC/Avbryt</code> for å redigere",
|
||
"mp_breq": "krever firefox 82+, chrome 73+, eller iOS 15+",
|
||
"mm_bload": "laster inn...",
|
||
"mm_bconv": "konverterer til {0}, vent litt...",
|
||
"mm_opusen": "nettleseren din forstår ikke aac / m4a;\nkonvertering til opus er nå aktivert",
|
||
"mm_playerr": "avspilling feilet: ",
|
||
"mm_eabrt": "Avspillingsforespørselen ble avbrutt",
|
||
"mm_enet": "Nettet ditt er ustabilt",
|
||
"mm_edec": "Noe er galt med musikkfilen",
|
||
"mm_esupp": "Nettleseren din forstår ikke filtypen",
|
||
"mm_eunk": "Ukjent feil",
|
||
"mm_e404": "Avspilling feilet: Fil ikke funnet.",
|
||
"mm_e403": "Avspilling feilet: Tilgang nektet.\n\nKanskje du ble logget ut?\nPrøv å trykk F5 for å laste siden på nytt.",
|
||
"mm_e500": "Avspilling feilet: Rusk i maskineriet, sjekk serverloggen.",
|
||
"mm_e5xx": "Avspilling feilet: ",
|
||
"mm_nof": "finner ikke flere sanger i nærheten",
|
||
"mm_prescan": "Leter etter neste sang...",
|
||
"mm_scank": "Fant neste sang:",
|
||
"mm_uncache": "alle sanger vil lastes på nytt ved neste avspilling",
|
||
"mm_hnf": "sangen finnes ikke lenger",
|
||
|
||
"im_hnf": "bildet finnes ikke lenger",
|
||
|
||
"f_empty": 'denne mappen er tom',
|
||
"f_chide": 'dette vil skjule kolonnen «{0}»\n\nfanen for "andre innstillinger" lar deg vise kolonnen igjen',
|
||
"f_bigtxt": "denne filen er hele {0} MiB -- vis som tekst?",
|
||
"f_bigtxt2": "vil du se bunnen av filen istedenfor? du vil da også se nye linjer som blir lagt til på slutten av filen i sanntid",
|
||
"fbd_more": '<div id="blazy">viser <code>{0}</code> av <code>{1}</code> filer; <a href="#" id="bd_more">vis {2}</a> eller <a href="#" id="bd_all">vis alle</a></div>',
|
||
"fbd_all": '<div id="blazy">viser <code>{0}</code> av <code>{1}</code> filer; <a href="#" id="bd_all">vis alle</a></div>',
|
||
"f_anota": "kun {0} av totalt {1} elementer ble markert;\nfor å velge alt må du bla til bunnen av mappen først",
|
||
|
||
"f_dls": 'linkene i denne mappen er nå\nomgjort til nedlastningsknapper',
|
||
|
||
"f_partial": "For å laste ned en fil som enda ikke er ferdig opplastet, klikk på filen som har samme filnavn som denne, men uten <code>.PARTIAL</code> på slutten. Da vil serveren passe på at nedlastning går bra. Derfor anbefales det sterkt å trykke AVBRYT eller Escape-tasten.\n\nHvis du virkelig ønsker å laste ned denne <code>.PARTIAL</code>-filen på en ukontrollert måte, trykk OK / Enter for å ignorere denne advarselen. Slik vil du høyst sannsynlig motta korrupt data.",
|
||
|
||
"ft_paste": "Lim inn {0} filer$NSnarvei: ctrl-V",
|
||
"fr_eperm": 'kan ikke endre navn:\ndu har ikke “move”-rettigheten i denne mappen',
|
||
"fd_eperm": 'kan ikke slette:\ndu har ikke “delete”-rettigheten i denne mappen',
|
||
"fc_eperm": 'kan ikke klippe ut:\ndu har ikke “move”-rettigheten i denne mappen',
|
||
"fp_eperm": 'kan ikke lime inn:\ndu har ikke “write”-rettigheten i denne mappen',
|
||
"fr_emore": "velg minst én fil som skal få nytt navn",
|
||
"fd_emore": "velg minst én fil som skal slettes",
|
||
"fc_emore": "velg minst én fil som skal klippes ut",
|
||
"fcp_emore": "velg minst én fil som skal kopieres til utklippstavlen",
|
||
|
||
"fs_sc": "del mappen du er i nå",
|
||
"fs_ss": "del de valgte filene",
|
||
"fs_just1d": "du kan ikke markere flere mapper samtidig,\neller kombinere mapper og filer",
|
||
"fs_abrt": "❌ avbryt",
|
||
"fs_rand": "🎲 tilfeldig navn",
|
||
"fs_go": "✅ opprett deling",
|
||
"fs_name": "navn",
|
||
"fs_src": "kilde",
|
||
"fs_pwd": "passord",
|
||
"fs_exp": "varighet",
|
||
"fs_tmin": "min",
|
||
"fs_thrs": "timer",
|
||
"fs_tdays": "dager",
|
||
"fs_never": "for evig",
|
||
"fs_pname": "frivillig navn (blir noe tilfeldig ellers)",
|
||
"fs_tsrc": "fil/mappe som skal deles",
|
||
"fs_ppwd": "frivillig passord",
|
||
"fs_w8": "oppretter deling...",
|
||
"fs_ok": "trykk <code>Enter/OK</code> for å kopiere linken (for CTRL-V)\ntrykk <code>ESC/Avbryt</code> for å bare bekrefte",
|
||
|
||
"frt_dec": "kan korrigere visse ødelagte filnavn\">url-decode",
|
||
"frt_rst": "nullstiller endringer (tilbake til de originale filnavnene)\">↺ reset",
|
||
"frt_abrt": "avbryt og lukk dette vinduet\">❌ avbryt",
|
||
"frb_apply": "IVERKSETT",
|
||
"fr_adv": "automasjon basert på metadata<br>og / eller mønster (regulære uttrykk)\">avansert",
|
||
"fr_case": "versalfølsomme uttrykk\">Aa",
|
||
"fr_win": "bytt ut bokstavene <code><>:"\\|?*</code> med$Ntilsvarende som windows ikke får panikk av\">win",
|
||
"fr_slash": "bytt ut bokstaven <code>/</code> slik at den ikke forårsaker at nye mapper opprettes\">ikke /",
|
||
"fr_re": "regex-mønster som kjøres på hvert filnavn. Grupper kan leses ut i format-feltet nedenfor, f.eks. <code>(1)</code> og <code>(2)</code> osv.",
|
||
"fr_fmt": "inspirert av foobar2000:$N<code>(title)</code> byttes ut med sangtittel,$N<code>[(artist) - ](title)</code> dropper [dette] hvis artist er blank$N<code>$lpad((tn),2,0)</code> viser sangnr. med 2 siffer",
|
||
"fr_pdel": "slett",
|
||
"fr_pnew": "lagre som",
|
||
"fr_pname": "gi innstillingene dine et navn",
|
||
"fr_aborted": "avbrutt",
|
||
"fr_lold": "gammelt navn",
|
||
"fr_lnew": "nytt navn",
|
||
"fr_tags": "metadata for de valgte filene (kun for referanse):",
|
||
"fr_busy": "endrer navn på {0} filer...\n\n{1}",
|
||
"fr_efail": "endring av navn feilet:\n",
|
||
"fr_nchg": "{0} av navnene ble justert pga. <code>win</code> og/eller <code>ikke /</code>\n\nvil du fortsette med de nye navnene som ble valgt?",
|
||
|
||
"fd_ok": "sletting OK",
|
||
"fd_err": "sletting feilet:\n",
|
||
"fd_none": "ingenting ble slettet; kanskje avvist av serverkonfigurasjon (xbd)?",
|
||
"fd_busy": "sletter {0} filer...\n\n{1}",
|
||
"fd_warn1": "SLETT disse {0} filene?",
|
||
"fd_warn2": "<b>Siste sjanse!</b> Dette kan ikke angres. Slett?",
|
||
|
||
"fc_ok": "klippet ut {0} filer",
|
||
"fc_warn": 'klippet ut {0} filer\n\nmen: kun <b>denne</b> nettleserfanen har mulighet til å lime dem inn et annet sted, siden antallet filer er helt hinsides',
|
||
|
||
"fcc_ok": "kopierte {0} filer til utklippstavlen",
|
||
"fcc_warn": 'kopierte {0} filer til utklippstavlen\n\nmen: kun <b>denne</b> nettleserfanen har mulighet til å lime dem inn et annet sted, siden antallet filer er helt hinsides',
|
||
|
||
"fp_apply": "bekreft og lim inn nå",
|
||
"fp_ecut": "du må klippe ut eller kopiere noen filer / mapper først\n\nmerk: du kan gjerne jobbe på kryss av nettleserfaner; klippe ut i én fane, lime inn i en annen",
|
||
"fp_ename": "{0} filer kan ikke flyttes til målmappen fordi det allerede finnes filer med samme navn. Gi dem nye navn nedenfor, eller gi dem et blankt navn for å hoppe over dem:",
|
||
"fcp_ename": "{0} filer kan ikke kopieres til målmappen fordi det allerede finnes filer med samme navn. Gi dem nye navn nedenfor, eller gi dem et blankt navn for å hoppe over dem:",
|
||
"fp_emore": "det er fortsatt flere navn som må endres",
|
||
"fp_ok": "flytting OK",
|
||
"fcp_ok": "kopiering OK",
|
||
"fp_busy": "flytter {0} filer...\n\n{1}",
|
||
"fcp_busy": "kopierer {0} filer...\n\n{1}",
|
||
"fp_err": "flytting feilet:\n",
|
||
"fcp_err": "kopiering feilet:\n",
|
||
"fp_confirm": "flytt disse {0} filene hit?",
|
||
"fcp_confirm": "kopiér disse {0} filene hit?",
|
||
"fp_etab": 'kunne ikke lese listen med filer ifra den andre nettleserfanen',
|
||
"fp_name": "Laster opp én fil fra enheten din. Velg filnavn:",
|
||
"fp_both_m": '<h6>hva skal limes inn her?</h6><code>Enter</code> = Flytt {0} filer fra «{1}»\n<code>ESC</code> = Last opp {2} filer fra enheten din',
|
||
"fcp_both_m": '<h6>hva skal limes inn her?</h6><code>Enter</code> = Kopiér {0} filer fra «{1}»\n<code>ESC</code> = Last opp {2} filer fra enheten din',
|
||
"fp_both_b": '<a href="#" id="modal-ok">Flytt</a><a href="#" id="modal-ng">Last opp</a>',
|
||
"fcp_both_b": '<a href="#" id="modal-ok">Kopiér</a><a href="#" id="modal-ng">Last opp</a>',
|
||
|
||
"mk_noname": "skriv inn et navn i tekstboksen til venstre først :p",
|
||
|
||
"tv_load": "Laster inn tekstfil:\n\n{0}\n\n{1}% ({2} av {3} MiB lastet ned)",
|
||
"tv_xe1": "kunne ikke laste tekstfil:\n\nfeil ",
|
||
"tv_xe2": "404, Fil ikke funnet",
|
||
"tv_lst": "tekstfiler i mappen",
|
||
"tvt_close": "gå tilbake til mappen$NSnarvei: M (eller Esc)\">❌ lukk",
|
||
"tvt_dl": "last ned denne filen$NSnarvei: Y\">💾 last ned",
|
||
"tvt_prev": "vis forrige dokument$NSnarvei: i\">⬆ forr.",
|
||
"tvt_next": "vis neste dokument$NSnarvei: K\">⬇ neste",
|
||
"tvt_sel": "markér filen ( for utklipp / sletting / ... )$NSnarvei: S\">merk",
|
||
"tvt_edit": "redigér filen$NSnarvei: E\">✏️ endre",
|
||
"tvt_tail": "overvåk filen for endringer og vis nye linjer i sanntid\">📡 følg",
|
||
"tvt_wrap": "tekstbryting\">↵",
|
||
"tvt_atail": "hold de nyeste linjene synlig (lås til bunnen av siden)\">⚓",
|
||
"tvt_ctail": "forstå og vis terminalfarger (ansi-sekvenser)\">🌈",
|
||
"tvt_ntail": "maks-grense for antall bokstaver som skal vises i vinduet",
|
||
|
||
"m3u_add1": "sangen ble lagt til i m3u-spillelisten",
|
||
"m3u_addn": "{0} sanger ble lagt til i m3u-spillelisten",
|
||
"m3u_clip": "m3u-spillelisten ble kopiert til utklippstavlen\n\nneste steg er å opprette et tekstdokument med filnavn som slutter på <code>.m3u</code> og lime inn spillelisten der",
|
||
|
||
"gt_vau": "ikke vis videofiler, bare spill lyden\">🎧",
|
||
"gt_msel": "markér filer istedenfor å åpne dem; ctrl-klikk filer for å overstyre$N$N<em>når aktiv: dobbelklikk en fil / mappe for å åpne</em>$N$NSnarvei: S\">markering",
|
||
"gt_crop": "beskjær ikonene så de passer bedre\">✂",
|
||
"gt_3x": "høyere oppløsning på ikoner\">3x",
|
||
"gt_zoom": "zoom",
|
||
"gt_chop": "trim",
|
||
"gt_sort": "sorter",
|
||
"gt_name": "navn",
|
||
"gt_sz": "størr.",
|
||
"gt_ts": "dato",
|
||
"gt_ext": "type",
|
||
"gt_c1": "reduser maks-lengde på filnavn",
|
||
"gt_c2": "øk maks-lengde på filnavn",
|
||
|
||
"sm_w8": "søker...",
|
||
"sm_prev": "søkeresultatene er fra et tidligere søk:\n ",
|
||
"sl_close": "lukk søkeresultater",
|
||
"sl_hits": "viser {0} treff",
|
||
"sl_moar": "hent flere",
|
||
|
||
"s_sz": "størr.",
|
||
"s_dt": "dato",
|
||
"s_rd": "sti",
|
||
"s_fn": "navn",
|
||
"s_ta": "meta",
|
||
"s_ua": "up@",
|
||
"s_ad": "avns.",
|
||
"s_s1": "større enn ↓ MiB",
|
||
"s_s2": "mindre enn ↓ MiB",
|
||
"s_d1": "nyere enn <dato>",
|
||
"s_d2": "eldre enn",
|
||
"s_u1": "lastet opp etter",
|
||
"s_u2": "og/eller før",
|
||
"s_r1": "mappenavn inneholder",
|
||
"s_f1": "filnavn inneholder",
|
||
"s_t1": "sang-info inneholder",
|
||
"s_a1": "konkrete egenskaper",
|
||
|
||
"md_eshow": "viser forenklet ",
|
||
"md_off": "[📜<em>readme</em>] er avskrudd i [⚙️] -- dokument skjult",
|
||
|
||
"badreply": "Ugyldig svar ifra serveren",
|
||
|
||
"xhr403": "403: Tilgang nektet\n\nkanskje du ble logget ut? prøv å trykk F5",
|
||
"xhr0": "ukjent (enten nettverksproblemer eller serverkrasj)",
|
||
"cf_ok": "beklager -- liten tilfeldig kontroll, alt OK\n\nting skal fortsette om ca. 30 sekunder\n\nhvis ikkeno skjer, trykk F5 for å laste siden på nytt",
|
||
"tl_xe1": "kunne ikke hente undermapper:\n\nfeil ",
|
||
"tl_xe2": "404: Mappen finnes ikke",
|
||
"fl_xe1": "kunne ikke hente filer i mappen:\n\nfeil ",
|
||
"fl_xe2": "404: Mappen finnes ikke",
|
||
"fd_xe1": "kan ikke opprette ny mappe:\n\nfeil ",
|
||
"fd_xe2": "404: Den overordnede mappen finnes ikke",
|
||
"fsm_xe1": "kunne ikke sende melding:\n\nfeil ",
|
||
"fsm_xe2": "404: Den overordnede mappen finnes ikke",
|
||
"fu_xe1": "kunne ikke hente listen med nylig opplastede filer ifra serveren:\n\nfeil ",
|
||
"fu_xe2": "404: Filen finnes ikke??",
|
||
|
||
"fz_tar": "ukomprimert gnu-tar arkiv, for linux og mac",
|
||
"fz_pax": "ukomprimert pax-tar arkiv, litt tregere",
|
||
"fz_targz": "gnu-tar pakket med gzip (nivå 3)$N$NNB: denne er veldig treg;$Nukomprimert tar er bedre",
|
||
"fz_tarxz": "gnu-tar pakket med xz (nivå 1)$N$NNB: denne er veldig treg;$Nukomprimert tar er bedre",
|
||
"fz_zip8": "zip med filnavn i utf8 (noe problematisk på windows 7 og eldre)",
|
||
"fz_zipd": "zip med filnavn i cp437, for høggamle maskiner",
|
||
"fz_zipc": "cp437 med tidlig crc32,$Nfor MS-DOS PKZIP v2.04g (oktober 1993)$N(øker behandlingstid på server)",
|
||
|
||
"un_m1": "nedenfor kan du angre / slette filer som du nylig har lastet opp, eller avbryte ufullstendige opplastninger",
|
||
"un_upd": "oppdater",
|
||
"un_m4": "eller hvis du vil dele nedlastnings-lenkene:",
|
||
"un_ulist": "vis",
|
||
"un_ucopy": "kopiér",
|
||
"un_flt": "valgfritt filter: filnavn / filsti må inneholde",
|
||
"un_fclr": "nullstill filter",
|
||
"un_derr": 'unpost-sletting feilet:\n',
|
||
"un_f5": 'noe gikk galt, prøv å oppdatere listen eller trykk F5',
|
||
"un_uf5": "beklager, men du må laste siden på nytt (f.eks. ved å trykke F5 eller CTRL-R) før denne opplastningen kan avbrytes",
|
||
"un_nou": '<b>advarsel:</b> kan ikke vise ufullstendige opplastninger akkurat nå; klikk på oppdater-linken om litt',
|
||
"un_noc": '<b>advarsel:</b> angring av fullførte opplastninger er deaktivert i serverkonfigurasjonen',
|
||
"un_max": "viser de første 2000 filene (bruk filteret for å innsnevre)",
|
||
"un_avail": "{0} nylig opplastede filer kan slettes<br />{1} ufullstendige opplastninger kan avbrytes",
|
||
"un_m2": "sortert etter opplastningstid; nyeste først:",
|
||
"un_no1": "men nei, her var det jaggu ikkeno som slettes kan",
|
||
"un_no2": "men nei, her var det jaggu ingenting som passet overens med filteret",
|
||
"un_next": "slett de neste {0} filene nedenfor",
|
||
"un_abrt": "avbryt",
|
||
"un_del": "slett",
|
||
"un_m3": "henter listen med nylig opplastede filer...",
|
||
"un_busy": "sletter {0} filer...",
|
||
"un_clip": "{0} lenker kopiert til utklippstavlen",
|
||
|
||
"u_https1": "du burde",
|
||
"u_https2": "bytte til https",
|
||
"u_https3": "for høyere hastighet",
|
||
"u_ancient": 'nettleseren din er prehistorisk -- mulig du burde <a href="#" onclick="goto(\'bup\')">bruke bup istedenfor</a>',
|
||
"u_nowork": "krever firefox 53+, chrome 57+, eller iOS 11+",
|
||
"tail_2old": "krever firefox 105+, chrome 71+, eller iOS 14.5+",
|
||
"u_nodrop": 'nettleseren din er for gammel til å laste opp filer ved å dra dem inn i vinduet',
|
||
"u_notdir": "mottok ikke mappen!\n\nnettleseren din er for gammel,\nprøv å dra mappen inn i vinduet istedenfor",
|
||
"u_uri": "for å laste opp bilder ifra andre nettleservinduer,\nslipp bildet rett på den store last-opp-knappen",
|
||
"u_enpot": 'bytt til <a href="#">enkelt UI</a> (gir sannsynlig raskere opplastning)',
|
||
"u_depot": 'bytt til <a href="#">snæsent UI</a> (gir sannsynlig tregere opplastning)',
|
||
"u_gotpot": 'byttet til et enklere UI for å laste opp raskere,\n\ndu kan gjerne bytte tilbake altså!',
|
||
"u_pott": "<p>filer: <b>{0}</b> ferdig, <b>{1}</b> feilet, <b>{2}</b> behandles, <b>{3}</b> i kø</p>",
|
||
"u_ever": "dette er den primitive opplasteren; up2k krever minst:<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1",
|
||
"u_su2k": 'dette er den primitive opplasteren; <a href="#" id="u2yea">up2k</a> er bedre',
|
||
"u_uput": 'litt raskere (uten sha512)',
|
||
"u_ewrite": 'du har ikke skrivetilgang i denne mappen',
|
||
"u_eread": 'du har ikke lesetilgang i denne mappen',
|
||
"u_enoi": 'filsøk er deaktivert i serverkonfigurasjonen',
|
||
"u_enoow": "kan ikke overskrive filer her (Delete-rettigheten er nødvendig)",
|
||
"u_badf": 'Disse {0} filene (av totalt {1}) kan ikke leses, kanskje pga rettighetsproblemer i filsystemet på datamaskinen din:\n\n',
|
||
"u_blankf": 'Disse {0} filene (av totalt {1}) er blanke / uten innhold; ønsker du å laste dem opp uansett?\n\n',
|
||
"u_applef": 'Disse {0} filene (av totalt {1}) er antagelig uønskede;\nTrykk <code>OK/Enter</code> for å HOPPE OVER disse filene,\nTrykk <code>Avbryt/ESC</code> for å LASTE OPP disse filene også:\n\n',
|
||
"u_just1": '\nFunker kanskje bedre hvis du bare tar én fil om gangen',
|
||
"u_ff_many": 'Hvis du bruker <b>Linux / MacOS / Android,</b> så kan dette antallet filer<br /><a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank"><em>kanskje</em> krasje Firefox!</a> Hvis det skjer, så prøv igjen (eller bruk Chrome).',
|
||
"u_up_life": "Filene slettes fra serveren {0}\netter at opplastningen er fullført",
|
||
"u_asku": 'Laste opp disse {0} filene til <code>{1}</code>',
|
||
"u_unpt": "Du kan angre / slette opplastningen med 🧯 oppe til venstre",
|
||
"u_bigtab": 'Vil nå vise {0} filer...\n\nDette kan krasje nettleseren din. Fortsette?',
|
||
"u_scan": 'Leser mappene...',
|
||
"u_dirstuck": 'Nettleseren din fikk ikke tilgang til å lese følgende {0} filer/mapper, så de blir hoppet over:',
|
||
"u_etadone": 'Ferdig ({0}, {1} filer)',
|
||
"u_etaprep": '(forbereder opplastning)',
|
||
"u_hashdone": 'befaring ferdig',
|
||
"u_hashing": 'les',
|
||
"u_hs": 'serveren tenker...',
|
||
"u_started": "filene blir nå lastet opp 🚀",
|
||
"u_dupdefer": "duplikat; vil bli håndtert til slutt",
|
||
"u_actx": "klikk her for å forhindre tap av<br />ytelse ved bytte til andre vinduer/faner",
|
||
"u_fixed": "OK! Løste seg 👍",
|
||
"u_cuerr": "kunne ikke laste opp del {0} av {1};\nsikkert greit, fortsetter\n\nfil: {2}",
|
||
"u_cuerr2": "server nektet opplastningen (del {0} av {1});\nprøver igjen senere\n\nfil: {2}\n\nerror ",
|
||
"u_ehstmp": "prøver igjen; se mld nederst",
|
||
"u_ehsfin": "server nektet forespørselen om å ferdigstille filen; prøver igjen...",
|
||
"u_ehssrch": "server nektet forespørselen om å utføre søk; prøver igjen...",
|
||
"u_ehsinit": "server nektet forespørselen om å begynne en ny opplastning; prøver igjen...",
|
||
"u_eneths": "et problem med nettverket gjorde at avtale om opplastning ikke kunne inngås; prøver igjen...",
|
||
"u_enethd": "et problem med nettverket gjorde at filsjekk ikke kunne utføres; prøver igjen...",
|
||
"u_cbusy": "venter på klarering ifra server etter et lite nettverksglipp...",
|
||
"u_ehsdf": "serveren er full!\n\nprøver igjen regelmessig,\ni tilfelle noen rydder litt...",
|
||
"u_emtleak1": "uff, det er mulig at nettleseren din har en minnelekkasje...\nForeslår",
|
||
"u_emtleak2": ' helst at du <a href="{0}">bytter til https</a>, eller ',
|
||
"u_emtleak3": ' at du ',
|
||
"u_emtleakc": 'prøver følgende:\n<ul><li>trykk F5 for å laste siden på nytt</li><li>så skru av <code>mt</code> bryteren under <code>⚙️ innstillinger</code></li><li>og forsøk den samme opplastningen igjen</li></ul>Opplastning vil gå litt tregere, men det får så være.\nBeklager bryderiet !\n\nPS: feilen <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">skal være fikset</a> i chrome v107',
|
||
"u_emtleakf": 'prøver følgende:\n<ul><li>trykk F5 for å laste siden på nytt</li><li>så skru på <code>🥔</code> ("enkelt UI") i opplasteren</li><li>og forsøk den samme opplastningen igjen</li></ul>\nPS: Firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">fikser forhåpentligvis feilen</a> en eller annen gang',
|
||
"u_s404": "ikke funnet på serveren",
|
||
"u_expl": "forklar",
|
||
"u_maxconn": "de fleste nettlesere tillater ikke mer enn 6, men firefox lar deg øke grensen med <code>connections-per-server</code> i <code>about:config</code>",
|
||
"u_tu": '<p class="warn">ADVARSEL: turbo er på, <span> avbrutte opplastninger vil muligens ikke oppdages og gjenopptas; hold musepekeren over turbo-knappen for mer info</span></p>',
|
||
"u_ts": '<p class="warn">ADVARSEL: turbo er på, <span> søkeresultater kan være feil; hold musepekeren over turbo-knappen for mer info</span></p>',
|
||
"u_turbo_c": "turbo er deaktivert i serverkonfigurasjonen",
|
||
"u_turbo_g": 'turbo ble deaktivert fordi du ikke har\ntilgang til å se mappeinnhold i dette volumet',
|
||
"u_life_cfg": 'slett opplastning etter <input id="lifem" p="60" /> min (eller <input id="lifeh" p="3600" /> timer)',
|
||
"u_life_est": 'opplastningen slettes <span id="lifew" tt="lokal tid">---</span>',
|
||
"u_life_max": 'denne mappen tillater ikke å \noppbevare filer i mer enn {0}',
|
||
"u_unp_ok": 'opplastning kan angres i {0}',
|
||
"u_unp_ng": 'opplastning kan IKKE angres',
|
||
"ue_ro": 'du har ikke skrivetilgang i denne mappen\n\n',
|
||
"ue_nl": 'du er ikke logget inn',
|
||
"ue_la": 'du er logget inn som "{0}"',
|
||
"ue_sr": 'du er i filsøk-modus\n\nbytt til opplastning ved å klikke på forstørrelsesglasset 🔎 (ved siden av den store FILSØK-knappen) og prøv igjen\n\nsorry',
|
||
"ue_ta": 'prøv å laste opp igjen, det burde funke nå',
|
||
"ue_ab": "den samme filen er allerede under opplastning til en annen mappe, og den må fullføres der før filen kan lastes opp andre steder.\n\nDu kan avbryte og glemme den påbegynte opplastningen ved hjelp av 🧯 oppe til venstre",
|
||
"ur_1uo": "OK: Filen ble lastet opp",
|
||
"ur_auo": "OK: Alle {0} filene ble lastet opp",
|
||
"ur_1so": "OK: Filen ble funnet på serveren",
|
||
"ur_aso": "OK: Alle {0} filene ble funnet på serveren",
|
||
"ur_1un": "Opplastning feilet!",
|
||
"ur_aun": "Alle {0} opplastningene gikk feil!",
|
||
"ur_1sn": "Filen finnes IKKE på serveren",
|
||
"ur_asn": "Fant INGEN av de {0} filene på serveren",
|
||
"ur_um": "Ferdig;\n{0} opplastninger gikk bra,\n{1} opplastninger gikk feil",
|
||
"ur_sm": "Ferdig;\n{0} filer ble funnet,\n{1} filer finnes IKKE på serveren",
|
||
|
||
"lang_set": "passer det å laste siden på nytt?",
|
||
},
|
||
|
||
"chi": {
|
||
// 以 //m 结尾的行是未经验证的机器翻译
|
||
"tt": "中文",
|
||
"cols": {
|
||
"c": "操作按钮",
|
||
"dur": "持续时间",
|
||
"q": "质量 / 比特率",
|
||
"Ac": "音频编码",
|
||
"Vc": "视频编码",
|
||
"Fmt": "格式 / 容器",
|
||
"Ahash": "音频校验和",
|
||
"Vhash": "视频校验和",
|
||
"Res": "分辨率",
|
||
"T": "文件类型",
|
||
"aq": "音频质量 / 比特率",
|
||
"vq": "视频质量 / 比特率",
|
||
"pixfmt": "子采样 / 像素结构",
|
||
"resw": "水平分辨率",
|
||
"resh": "垂直分辨率",
|
||
"chs": "音频频道",
|
||
"hz": "采样率"
|
||
},
|
||
|
||
"hks": [
|
||
[
|
||
"misc",
|
||
["ESC", "关闭各种窗口"],
|
||
|
||
"file-manager",
|
||
["G", "切换列表 / 网格视图"],
|
||
["T", "切换缩略图 / 图标"],
|
||
["⇧ A/D", "缩略图大小"],
|
||
["ctrl-K", "删除选中项"],
|
||
["ctrl-X", "剪切选中项"],
|
||
["ctrl-C", "复制选中项"], //m
|
||
["ctrl-V", "粘贴到文件夹"],
|
||
["Y", "下载选中项"],
|
||
["F2", "重命名选中项"],
|
||
|
||
"file-list-sel",
|
||
["space", "切换文件选择"],
|
||
["↑/↓", "移动选择光标"],
|
||
["ctrl ↑/↓", "移动光标和视图"],
|
||
["⇧ ↑/↓", "选择上一个/下一个文件"],
|
||
["ctrl-A", "选择所有文件 / 文件夹"]
|
||
], [
|
||
"navigation",
|
||
["B", "切换面包屑导航 / 导航窗格"],
|
||
["I/K", "前一个/下一个文件夹"],
|
||
["M", "父文件夹(或折叠当前文件夹)"],
|
||
["V", "切换导航窗格中的文件夹 / 文本文件"],
|
||
["A/D", "导航窗格大小"]
|
||
], [
|
||
"audio-player",
|
||
["J/L", "上一首/下一首歌曲"],
|
||
["U/O", "跳过10秒向前/向后"],
|
||
["0..9", "跳转到0%..90%"],
|
||
["P", "播放/暂停(也可以启动)"],
|
||
["S", "选择正在播放的歌曲"], //m
|
||
["Y", "下载歌曲"]
|
||
], [
|
||
"image-viewer",
|
||
["J/L, ←/→", "上一张/下一张图片"],
|
||
["Home/End", "第一张/最后一张图片"],
|
||
["F", "全屏"],
|
||
["R", "顺时针旋转"],
|
||
["⇧ R", "逆时针旋转"],
|
||
["S", "选择图片"], //m
|
||
["Y", "下载图片"]
|
||
], [
|
||
"video-player",
|
||
["U/O", "跳过10秒向前/向后"],
|
||
["P/K/Space", "播放/暂停"],
|
||
["C", "继续播放下一段"],
|
||
["V", "循环"],
|
||
["M", "静音"],
|
||
["[ and ]", "设置循环区间"]
|
||
], [
|
||
"textfile-viewer",
|
||
["I/K", "前一个/下一个文件"],
|
||
["M", "关闭文本文件"],
|
||
["E", "编辑文本文件"],
|
||
["S", "选择文件(用于剪切/重命名)"]
|
||
]
|
||
],
|
||
|
||
"m_ok": "确定",
|
||
"m_ng": "取消",
|
||
|
||
"enable": "启用",
|
||
"danger": "危险",
|
||
"clipped": "已复制到剪贴板",
|
||
|
||
"ht_s1": "秒",
|
||
"ht_s2": "秒",
|
||
"ht_m1": "分",
|
||
"ht_m2": "分",
|
||
"ht_h1": "时",
|
||
"ht_h2": "时",
|
||
"ht_d1": "天",
|
||
"ht_d2": "天",
|
||
"ht_and": " 和 ",
|
||
|
||
"goh": "控制面板",
|
||
"gop": '前一项">pre',
|
||
"gou": '顶部">up',
|
||
"gon": '下一项">next',
|
||
"logout": " 登出",
|
||
"access": " 访问",
|
||
"ot_close": "关闭子菜单",
|
||
"ot_search": "按属性、路径/名称、音乐标签或上述内容的任意组合搜索文件$N$N<code>foo bar</code> = 必须包含 «foo» 和 «bar»,$N<code>foo -bar</code> = 包含 «foo» 而不包含 «bar»,$N<code>^yana .opus$</code> = 以 «yama» 为开头的 «opus» 文件$N<code>"try unite"</code> = 正好包含 «try unite»$N$N时间格式为 iso-8601, 比如:$N<code>2009-12-31</code> or <code>2020-09-12 23:30:00</code>",
|
||
"ot_unpost": "取消发布:删除最近上传的内容,或中止未完成的内容",
|
||
"ot_bup": "bup:基础上传器,甚至支持 Netscape 4.0",
|
||
"ot_mkdir": "mkdir:创建新目录",
|
||
"ot_md": "new-md:创建新 Markdown 文档",
|
||
"ot_msg": "msg:向服务器日志发送消息",
|
||
"ot_mp": "媒体播放器选项",
|
||
"ot_cfg": "配置选项",
|
||
"ot_u2i": 'up2k:上传文件(如果你有写入权限),或切换到搜索模式以查看文件是否存在于服务器上,$N$N上传是可恢复的,多线程的,保留文件时间戳,但比 [🎈] (基础上传器)占用 更多的CPU<br /><br />上传过程中,此图标会变成进度指示器!',
|
||
"ot_u2w": 'up2k:带有恢复支持的文件上传(关闭浏览器后,重新上传相同文件)$N$N多线程的,文件时间戳得以保留,但比 [🎈] (基础上传器)使用更多CPU<br /><br />上传过程中,这个图标会变成进度指示器!',
|
||
"ot_noie": '请使用 Chrome / Firefox / Edge',
|
||
|
||
"ab_mkdir": "创建目录",
|
||
"ab_mkdoc": "新建 Markdown 文档",
|
||
"ab_msg": "发送消息到服务器日志",
|
||
|
||
"ay_path": "跳转到文件夹",
|
||
"ay_files": "跳转到文件",
|
||
|
||
"wt_ren": "重命名选中的项目$N快捷键: F2",
|
||
"wt_del": "删除选中的项目$N快捷键: ctrl-K",
|
||
"wt_cut": "剪切选中的项目<small>(然后粘贴到其他地方)</small>$N快捷键: ctrl-X",
|
||
"wt_cpy": "将选中的项目复制到剪贴板<small>(然后粘贴到其他地方)</small>$N快捷键: ctrl-C", //m
|
||
"wt_pst": "粘贴之前剪切/复制的选择$N快捷键: ctrl-V",
|
||
"wt_selall": "选择所有文件$N快捷键: ctrl-A(当文件被聚焦时)",
|
||
"wt_selinv": "反转选择",
|
||
"wt_zip1": "将此文件夹下载为归档文件", //m
|
||
"wt_selzip": "将选择下载为归档文件",
|
||
"wt_seldl": "将选择下载为单独的文件$N快捷键: Y",
|
||
"wt_npirc": "复制 IRC 格式的曲目信息",
|
||
"wt_nptxt": "复制纯文本格式的曲目信息",
|
||
"wt_m3ua": "添加到 m3u 播放列表(稍后点击 <code>📻copy</code>)", //m
|
||
"wt_m3uc": "复制 m3u 播放列表到剪贴板", //m
|
||
"wt_grid": "切换网格/列表视图$N快捷键: G",
|
||
"wt_prev": "上一曲$N快捷键: J",
|
||
"wt_play": "播放/暂停$N快捷键: P",
|
||
"wt_next": "下一曲$N快捷键: L",
|
||
|
||
"ul_par": "并行上传:",
|
||
"ut_rand": "随机化文件名",
|
||
"ut_u2ts": "将最后修改的时间戳$N从你的文件系统复制到服务器\">📅",
|
||
"ut_ow": "覆盖服务器上的现有文件?$N🛡️: 从不(会生成一个新文件名)$N🕒: 服务器文件较旧则覆盖$N♻️: 总是覆盖,如果文件内容不同", //m
|
||
"ut_mt": "在上传时继续哈希其他文件$N$N如果你的 CPU 或硬盘是瓶颈,可能需要禁用",
|
||
"ut_ask": '上传开始前询问确认">💭',
|
||
"ut_pot": "通过简化 UI 来$N提高慢设备上的上传速度",
|
||
"ut_srch": "实际不上传,而是检查文件是否$N已经存在于服务器上(将扫描你可以读取的所有文件夹)",
|
||
"ut_par": "通过将其设置为 0 来暂停上传$N$N如果你的连接很慢/延迟高,$N$N请增加在局域网或服务器硬盘是瓶颈时保持为 1",
|
||
"ul_btn": "将文件/文件夹拖放到这里(或点击我)",
|
||
"ul_btnu": "上 传",
|
||
"ul_btns": "搜 索",
|
||
|
||
"ul_hash": "哈希",
|
||
"ul_send": "发送",
|
||
"ul_done": "完成",
|
||
"ul_idle1": "没有排队的上传任务",
|
||
"ut_etah": "平均 <em>hashing</em> 速度和估计完成时间",
|
||
"ut_etau": "平均 <em>上传</em> 速度和估计完成时间",
|
||
"ut_etat": "平均 <em>总</em> 速度和估计完成时间",
|
||
|
||
"uct_ok": "成功完成",
|
||
"uct_ng": "失败/拒绝/未找到",
|
||
"uct_done": "成功和失败的组合",
|
||
"uct_bz": "正在哈希或上传",
|
||
"uct_q": "空闲,待处理",
|
||
|
||
"utl_name": "文件名",
|
||
"utl_ulist": "列表",
|
||
"utl_ucopy": "复制",
|
||
"utl_links": "链接",
|
||
"utl_stat": "状态",
|
||
"utl_prog": "进度",
|
||
|
||
// 保持简短:
|
||
"utl_404": "404",
|
||
"utl_err": "错误",
|
||
"utl_oserr": "OS错误",
|
||
"utl_found": "已找到",
|
||
"utl_defer": "延期",
|
||
"utl_yolo": "加速",
|
||
"utl_done": "完成",
|
||
|
||
"ul_flagblk": "文件已添加到队列</b><br>但另一个浏览器标签中有一个繁忙的 up2k,<br>因此等待它完成",
|
||
"ul_btnlk": "服务器配置已将此开关锁定到此状态",
|
||
|
||
"udt_up": "上传",
|
||
"udt_srch": "搜索",
|
||
"udt_drop": "将文件拖放到这里",
|
||
|
||
"u_nav_m": '<h6>好的,你有什么?</h6><code>Enter</code> = 文件(一个或多个)\n<code>ESC</code> = 一个文件夹(包括子文件夹)',
|
||
"u_nav_b": '<a href="#" id="modal-ok">文件</a><a href="#" id="modal-ng">一个文件夹</a>',
|
||
|
||
"cl_opts": "开关选项",
|
||
"cl_themes": "主题",
|
||
"cl_langs": "语言",
|
||
"cl_ziptype": "文件夹下载",
|
||
"cl_uopts": "up2k 开关",
|
||
"cl_favico": "网站图标",
|
||
"cl_bigdir": "最大目录数",
|
||
"cl_hsort": "#sort", //m
|
||
"cl_keytype": "键位符号",
|
||
"cl_hiddenc": "隐藏列",
|
||
"cl_hidec": "隐藏",
|
||
"cl_reset": "重置",
|
||
"cl_hpick": "点击列标题以在下表中隐藏",
|
||
"cl_hcancel": "列隐藏已取消",
|
||
|
||
"ct_grid": '网格视图',
|
||
"ct_ttips": '◔ ◡ ◔">ℹ️ 工具提示',
|
||
"ct_thumb": '在网格视图中,切换图标或缩略图$N快捷键: T">🖼️ 缩略图',
|
||
"ct_csel": '在网格视图中使用 CTRL 和 SHIFT 进行文件选择">CTRL',
|
||
"ct_ihop": '当图像查看器关闭时,滚动到最后查看的文件">滚动',
|
||
"ct_dots": '显示隐藏文件(如果服务器允许)">隐藏文件',
|
||
"ct_qdel": '删除文件时,只需确认一次">快删', //m
|
||
"ct_dir1st": '在文件之前排序文件夹">📁 排序',
|
||
"ct_nsort": '正确排序以数字开头的文件名">数字排序', //m
|
||
"ct_readme": '在文件夹列表中显示 README.md">📜 readme',
|
||
"ct_idxh": '显示 index.html 代替文件夹列表">htm',
|
||
"ct_sbars": '显示滚动条">⟊',
|
||
|
||
"cut_umod": "如果文件已存在于服务器上,将服务器的最后修改时间戳更新为与你的本地文件匹配(需要写入和删除权限)\">re📅",
|
||
|
||
"cut_turbo": "YOLO 按钮,你可能不想启用这个:$N$N如果你上传了大量文件并且由于某些原因需要重新启动,$N并且想要尽快继续上传,使用此选项$N$N这会用简单的 <em>"服务器上的文件大小是否相同?"</em> 替代哈希检查,$N因此如果文件内容不同,它将不会被上传$N$N上传完成后,你应该关闭此选项,$N然后重新"上传"相同的文件以让客户端验证它们\">加速",
|
||
|
||
"cut_datechk": "除非启用「加速」按钮,否则没有效果$N$N略微减少 YOLO 因素;检查服务器上的文件时间戳是否与你的一致$N$N<em>理论上</em> 应该能捕捉到大多数未完成/损坏的上传,$N但不能替代之后禁用「加速」进行的验证\">日期检查",
|
||
|
||
"cut_u2sz": "每个上传块的大小(以 MiB 为单位);较大的值跨大西洋传输效果更好。在非常不可靠的连接上尝试较小的值",
|
||
|
||
"cut_flag": "确保一次只有一个标签页在上传$N -- 其他标签页也必须启用此选项$N -- 仅影响同一域名下的标签页",
|
||
|
||
"cut_az": "按字母顺序上传文件,而不是按最小文件优先$N$N按字母顺序可以更容易地查看服务器上是否出现了问题,但在光纤/局域网上传稍微慢一些",
|
||
|
||
"cut_nag": "上传完成时的操作系统通知$N(仅当浏览器或标签页不活跃时)",
|
||
"cut_sfx": "上传完成时的声音警报$N(仅当浏览器或标签页不活跃时)",
|
||
|
||
"cut_mt": "使用多线程加速文件哈希$N$N这使用 Web Worker 并且需要更多内存(额外最多 512 MiB)$N$N这使得 https 快 30%,http 快 4.5 倍\">mt",
|
||
|
||
"cut_wasm": "使用基于 WASM 的哈希计算器代替浏览器内置的哈希功能;这可以提升在基于 Chrome 的浏览器上的速度,但会增加 CPU 使用率,而且许多旧版本的 Chrome 存在漏洞,启用此功能会导致浏览器占用所有内存并崩溃。\">wasm", //m
|
||
|
||
"cft_text": "网站图标文本(为空并刷新以禁用)",
|
||
"cft_fg": "前景色",
|
||
"cft_bg": "背景色",
|
||
|
||
"cdt_lim": "文件夹中显示的最大文件数",
|
||
"cdt_ask": "滚动到底部时,$N不会加载更多文件,$N而是询问你该怎么做",
|
||
"cdt_hsort": "包含在媒体 URL 中的排序规则 (<code>,sorthref</code>) 数量。将其设置为 0 时,点击媒体链接时也会忽略排序规则。", //m
|
||
|
||
"tt_entree": "显示导航面板(目录树侧边栏)$N快捷键: B",
|
||
"tt_detree": "显示面包屑导航$N快捷键: B",
|
||
"tt_visdir": "滚动到选定的文件夹",
|
||
"tt_ftree": "切换文件夹树 / 文本文件$N快捷键: V",
|
||
"tt_pdock": "在顶部的停靠窗格中显示父文件夹",
|
||
"tt_dynt": "随着树的展开自动增长",
|
||
"tt_wrap": "自动换行",
|
||
"tt_hover": "悬停时显示溢出的行$N(当鼠标光标在左侧边栏中时,滚动可能会中断)",
|
||
|
||
"ml_pmode": "在文件夹末尾时...",
|
||
"ml_btns": "命令",
|
||
"ml_tcode": "转码",
|
||
"ml_tcode2": "转换为", //m
|
||
"ml_tint": "透明度",
|
||
"ml_eq": "音频均衡器",
|
||
"ml_drc": "动态范围压缩器",
|
||
|
||
"mt_loop": "循环播放当前的歌曲\">🔁", //m
|
||
"mt_one": "只播放一首歌后停止\">1️⃣", //m
|
||
"mt_shuf": "在每个文件夹中随机播放歌曲\">🔀",
|
||
"mt_aplay": "如果链接中有歌曲 ID,则自动播放,禁用此选项将停止在播放音乐时更新页面 URL 中的歌曲 ID,以防止在设置丢失但 URL 保留时自动播放\">自动播放▶",
|
||
"mt_preload": "在歌曲快结束时开始加载下一首歌,以实现无缝播放\">预加载",
|
||
"mt_prescan": "在最后一首歌结束之前切换到下一个文件夹$N保持网页浏览器活跃$N以免停止播放\">自动切换",
|
||
"mt_fullpre": "尝试预加载整首歌;$N✅ 在 <b>不可靠</b> 连接上启用,$N❌ 可能在慢速连接上禁用\">加载整首歌",
|
||
"mt_fau": "在手机上,如果下一首歌未能快速预加载,防止音乐停止(可能导致标签显示异常)\">☕️",
|
||
"mt_waves": "波形进度条:$N显示音频幅度\">进度条",
|
||
"mt_npclip": "显示当前播放歌曲的剪贴板按钮\">♪剪切板",
|
||
"mt_m3u_c": "显示按钮以将所选歌曲$N复制为 m3u8 播放列表条目\">📻", //m
|
||
"mt_octl": "操作系统集成(媒体快捷键 / OSD)\">OSD",
|
||
"mt_oseek": "允许通过操作系统集成进行跳转$N$N注意:在某些设备(如 iPhone)上,$N这将替代下一首歌按钮\">seek",
|
||
"mt_oscv": "在 OSD 中显示专辑封面\">封面",
|
||
"mt_follow": "保持正在播放的曲目滚动到视图中\">🎯",
|
||
"mt_compact": "紧凑的控制按钮\">⟎",
|
||
"mt_uncache": "清除缓存 $N(如果你的浏览器缓存了一个损坏的歌曲副本而拒绝播放,请尝试此操作)\">uncache",
|
||
"mt_mloop": "循环打开的文件夹\">🔁 循环",
|
||
"mt_mnext": "加载下一个文件夹并继续\">📂 下一首",
|
||
"mt_mstop": "停止播放\">⏸ 停止", //m
|
||
"mt_cflac": "将 flac / wav 转换为 opus\">flac",
|
||
"mt_caac": "将 aac / m4a 转换为 opus\">aac",
|
||
"mt_coth": "将所有其他(不是 mp3)转换为 opus\">oth",
|
||
"mt_c2opus": "适合桌面电脑、笔记本电脑和安卓设备的最佳选择\">opus", //m
|
||
"mt_c2owa": "opus-weba(适用于 iOS 17.5 及更新版本)\">owa", //m
|
||
"mt_c2caf": "opus-caf(适用于 iOS 11 到 iOS 17)\">caf", //m
|
||
"mt_c2mp3": "适用于非常旧的设备\">mp3", //m
|
||
"mt_c2ok": "不错的选择!", //m
|
||
"mt_c2nd": "这不是您的设备推荐的输出格式,但应该没问题。", //m
|
||
"mt_c2ng": "您的设备似乎不支持此输出格式,不过我们还是试试看吧。", //m
|
||
"mt_xowa": "iOS 系统仍存在无法后台播放 owa 音乐的错误,请改用 caf 或 mp3 格式。", //m
|
||
"mt_tint": "在进度条上设置背景级别(0-100)",
|
||
"mt_eq": "启用均衡器和增益控制;$N$Nboost <code>0</code> = 标准 100% 音量(默认)$N$Nwidth <code>1 </code> = 标准立体声(默认)$Nwidth <code>0.5</code> = 50% 左右交叉反馈$Nwidth <code>0 </code> = 单声道$N$Nboost <code>-0.8</code> & width <code>10</code> = 人声移除 )$N$N启用均衡器使无缝专辑完全无缝,所以如果你在乎这一点,请保持启用,所有值设为零(除了宽度 = 1)",
|
||
"mt_drc": "启用动态范围压缩器(音量平滑器 / 限幅器);还会启用均衡器以平衡音频,因此如果你不想要它,请将均衡器字段除了 '宽度' 外的所有字段设置为 0$N$N降低 THRESHOLD dB 以上的音频的音量;每超过 THRESHOLD dB 的 RATIO 会有 1 dB 输出,所以默认值 tresh -24 和 ratio 12 意味着它的音量不应超过 -22 dB,可以安全地将均衡器增益提高到 0.8,甚至在 ATK 0 和 RLS 如 90 的情况下提高到 1.8(仅在 Firefox 中有效;其他浏览器中 RLS 最大为 1)$N$N(见维基百科,他们解释得更好)",
|
||
|
||
"mb_play": "播放",
|
||
"mm_hashplay": "播放这个音频文件?",
|
||
"mm_m3u": "按 <code>Enter/确定</code> 播放\n按 <code>ESC/取消</code> 编辑", //m
|
||
"mp_breq": "需要 Firefox 82+ 或 Chrome 73+ 或 iOS 15+",
|
||
"mm_bload": "正在加载...",
|
||
"mm_bconv": "正在转换为 {0},请稍等...",
|
||
"mm_opusen": "你的浏览器无法播放 aac / m4a 文件;\n现在启用转码为 opus",
|
||
"mm_playerr": "播放失败:",
|
||
"mm_eabrt": "播放尝试已取消",
|
||
"mm_enet": "你的互联网连接有问题",
|
||
"mm_edec": "这个文件可能已损坏??",
|
||
"mm_esupp": "你的浏览器不支持这个音频格式",
|
||
"mm_eunk": "未知错误",
|
||
"mm_e404": "无法播放音频;错误 404:文件未找到。",
|
||
"mm_e403": "无法播放音频;错误 403:访问被拒绝。\n\n尝试按 F5 重新加载,也许你已被注销",
|
||
"mm_e500": "无法播放音频;错误 500:检查服务器日志。", //m
|
||
"mm_e5xx": "无法播放音频;服务器错误",
|
||
"mm_nof": "附近找不到更多音频文件",
|
||
"mm_prescan": "正在寻找下一首音乐...",
|
||
"mm_scank": "找到下一首歌:",
|
||
"mm_uncache": "缓存已清除;所有歌曲将在下次播放时重新下载",
|
||
"mm_hnf": "那首歌不再存在",
|
||
|
||
"im_hnf": "那张图片不再存在",
|
||
|
||
"f_empty": '该文件夹为空',
|
||
"f_chide": '隐藏列 «{0}»\n\n你可以在设置选项卡中重新显示列',
|
||
"f_bigtxt": "这个文件大小为 {0} MiB -- 真的以文本形式查看?",
|
||
"f_bigtxt2": " 你想查看文件的结尾部分吗?这也将启用实时跟踪功能,能够实时显示新添加的文本行。", //m
|
||
"fbd_more": '<div id="blazy">显示 <code>{0}</code> 个文件中的 <code>{1}</code> 个;<a href="#" id="bd_more">显示 {2}</a> 或 <a href="#" id="bd_all">显示全部</a></div>',
|
||
"fbd_all": '<div id="blazy">显示 <code>{0}</code> 个文件中的 <code>{1}</code> 个;<a href="#" id="bd_all">显示全部</a></div>',
|
||
"f_anota": "仅选择了 {0} 个项目,共 {1} 个;\n要选择整个文件夹,请先滚动到底部", //m
|
||
|
||
"f_dls": '当前文件夹中的文件链接已\n更改为下载链接',
|
||
|
||
"f_partial": "要安全下载正在上传的文件,请点击没有 <code>.PARTIAL</code> 文件扩展名的同名文件。请按取消或 Escape 执行此操作。\n\n按 OK / Enter 将忽略此警告并继续下载 <code>.PARTIAL</code> 临时文件,这几乎肯定会导致数据损坏。",
|
||
|
||
"ft_paste": "粘贴 {0} 项$N快捷键: ctrl-V",
|
||
"fr_eperm": '无法重命名:\n你在此文件夹中没有 “移动” 权限',
|
||
"fd_eperm": '无法删除:\n你在此文件夹中没有 “删除” 权限',
|
||
"fc_eperm": '无法剪切:\n你在此文件夹中没有 “移动” 权限',
|
||
"fp_eperm": '无法粘贴:\n你在此文件夹中没有 “写入” 权限',
|
||
"fr_emore": "选择至少一个项目以重命名",
|
||
"fd_emore": "选择至少一个项目以删除",
|
||
"fc_emore": "选择至少一个项目以剪切",
|
||
"fcp_emore": "选择至少一个要复制到剪贴板的项目", //m
|
||
|
||
"fs_sc": "分享你所在的文件夹",
|
||
"fs_ss": "分享选定的文件",
|
||
"fs_just1d": "你不能同时选择多个文件夹,也不能同时选择文件夹和文件",
|
||
"fs_abrt": "❌ 取消",
|
||
"fs_rand": "🎲 随机名称",
|
||
"fs_go": "✅ 创建分享",
|
||
"fs_name": "名称",
|
||
"fs_src": "源",
|
||
"fs_pwd": "密码",
|
||
"fs_exp": "过期",
|
||
"fs_tmin": "分",
|
||
"fs_thrs": "时",
|
||
"fs_tdays": "天",
|
||
"fs_never": "永久",
|
||
"fs_pname": "链接名称可选;如果为空则随机",
|
||
"fs_tsrc": "共享的文件或文件夹",
|
||
"fs_ppwd": "密码可选",
|
||
"fs_w8": "正在创建文件共享...",
|
||
"fs_ok": "按 <code>Enter/确定</code> 复制到剪贴板\n按 <code>ESC/取消</code> 关闭",
|
||
|
||
"frt_dec": "可能修复一些损坏的文件名\">url-decode",
|
||
"frt_rst": "将修改后的文件名重置为原始文件名\">↺ 重置",
|
||
"frt_abrt": "中止并关闭此窗口\">❌ 取消",
|
||
"frb_apply": "应用重命名",
|
||
"fr_adv": "批量 / 元数据 / 模式重命名\">高级",
|
||
"fr_case": "区分大小写的正则表达式\">case",
|
||
"fr_win": "Windows 安全名称;将 <code><>:"\\|?*</code> 替换为日文全角字符\">win",
|
||
"fr_slash": "将 <code>/</code> 替换为不会导致新文件夹创建的字符\">不使用 /",
|
||
"fr_re": "正则表达式搜索模式应用于原始文件名;$N可以在下面的格式字段中引用捕获组,如<code>(1)</code>和<code>(2)</code>等等。",
|
||
"fr_fmt": "受到 foobar2000 的启发:$N<code>(title)</code> 被歌曲名称替换,$N<code>[(artist) - ](title)</code> 仅当歌曲艺术家不为空时才包含<code>[此]</code>部分$N<code>$lpad((tn),2,0)</code> 将曲目编号填充为 2 位数字",
|
||
"fr_pdel": "删除",
|
||
"fr_pnew": "另存为",
|
||
"fr_pname": "为你的新预设提供一个名称",
|
||
"fr_aborted": "已中止",
|
||
"fr_lold": "旧名称",
|
||
"fr_lnew": "新名称",
|
||
"fr_tags": "选定文件的标签(只读,仅供参考):",
|
||
"fr_busy": "正在重命名 {0} 项...\n\n{1}",
|
||
"fr_efail": "重命名失败:\n",
|
||
"fr_nchg": "{0} 个新名称由于 <code>win</code> 和/或 <code>不使用 /</code> 被更改\n\n确定继续使用这些更改的新名称?",
|
||
|
||
"fd_ok": "删除成功",
|
||
"fd_err": "删除失败:\n",
|
||
"fd_none": "没有文件被删除;可能被服务器配置(xbd)阻止?",
|
||
"fd_busy": "正在删除 {0} 项...\n\n{1}",
|
||
"fd_warn1": "删除这 {0} 项?",
|
||
"fd_warn2": "<b>最后机会!</b> 无法撤销。删除?",
|
||
|
||
"fc_ok": "剪切 {0} 项",
|
||
"fc_warn": '剪切 {0} 项\n\n但:只有 <b>这个</b> 浏览器标签页可以粘贴它们\n(因为选择非常庞大)',
|
||
|
||
"fcc_ok": "已将 {0} 项复制到剪贴板", //m
|
||
"fcc_warn": '已将 {0} 项复制到剪贴板\n\n但:只有 <b>这个</b> 浏览器标签页可以粘贴它们\n(因为选择非常庞大)', //m
|
||
|
||
"fp_apply": "确认并立即粘贴", //m
|
||
"fp_ecut": "首先剪切或复制一些文件/文件夹以粘贴/移动\n\n注意:你可以在不同的浏览器标签页之间剪切/粘贴", //m
|
||
"fp_ename": "{0} 项不能移动到这里,因为名称已被占用。请在下方输入新名称以继续,或将名称留空以跳过这些项:", //m
|
||
"fcp_ename": "{0} 项不能复制到这里,因为名称已被占用。请在下方输入新名称以继续,或将名称留空以跳过这些项:", //m
|
||
"fp_emore": "还有一些文件名冲突需要解决", //m
|
||
"fp_ok": "移动成功",
|
||
"fcp_ok": "复制成功", //m
|
||
"fp_busy": "正在移动 {0} 项...\n\n{1}",
|
||
"fcp_busy": "正在复制 {0} 项...\n\n{1}", //m
|
||
"fp_err": "移动失败:\n",
|
||
"fcp_err": "复制失败:\n", //m
|
||
"fp_confirm": "将这些 {0} 项移动到这里?",
|
||
"fcp_confirm": "将这些 {0} 项复制到这里?", //m
|
||
"fp_etab": '无法从其他浏览器标签页读取剪贴板',
|
||
"fp_name": "从你的设备上传一个文件。给它一个名字:",
|
||
"fp_both_m": '<h6>选择粘贴内容</h6><code>Enter</code> = 从 «{1}» 移动 {0} 个文件\n<code>ESC</code> = 从你的设备上传 {2} 个文件',
|
||
"fcp_both_m": '<h6>选择粘贴内容</h6><code>Enter</code> = 从 «{1}» 复制 {0} 个文件\n<code>ESC</code> = 从你的设备上传 {2} 个文件', //m
|
||
"fp_both_b": '<a href="#" id="modal-ok">移动</a><a href="#" id="modal-ng">上传</a>',
|
||
"fcp_both_b": '<a href="#" id="modal-ok">复制</a><a href="#" id="modal-ng">上传</a>', //m
|
||
|
||
"mk_noname": "在左侧文本框中输入名称,然后再执行此操作 :p",
|
||
|
||
"tv_load": "加载文本文件:\n\n{0}\n\n{1}% ({2} 的 {3} MiB 已加载)",
|
||
"tv_xe1": "无法加载文本文件:\n\n错误 ",
|
||
"tv_xe2": "404,文件未找到",
|
||
"tv_lst": "文本文件列表",
|
||
"tvt_close": "返回到文件夹视图$N快捷键: M(或 Esc)\">❌ 关闭",
|
||
"tvt_dl": "下载此文件$N快捷键: Y\">💾 下载",
|
||
"tvt_prev": "显示上一个文档$N快捷键: i\">⬆ 上一个",
|
||
"tvt_next": "显示下一个文档$N快捷键: K\">⬇ 下一个",
|
||
"tvt_sel": "选择文件 (用于剪切/删除/...)$N快捷键: S\">选择",
|
||
"tvt_edit": "在文本编辑器中打开文件$N快捷键: E\">✏️ 编辑",
|
||
"tvt_tail": "监视文件更改,并实时显示新增的行\">📡 跟踪", //m
|
||
"tvt_wrap": "自动换行\">↵", //m
|
||
"tvt_atail": "锁定到底部,显示最新内容\">⚓", //m
|
||
"tvt_ctail": "解析终端颜色(ANSI 转义码)\">🌈", //m
|
||
"tvt_ntail": "滚动历史上限(保留多少字节的文本)", //m
|
||
|
||
"m3u_add1": "歌曲已添加到 m3u 播放列表", //m
|
||
"m3u_addn": "已添加 {0} 首歌曲到 m3u 播放列表", //m
|
||
"m3u_clip": "m3u 播放列表已复制到剪贴板\n\n请创建一个以 <code>.m3u</code> 结尾的文本文件,\n并将播放列表粘贴到该文件中;\n这样就可以播放了", //m
|
||
|
||
"gt_vau": "不显示视频,仅播放音频\">🎧",
|
||
"gt_msel": "启用文件选择;按住 ctrl 键点击文件以覆盖$N$N<em>当启用时:双击文件/文件夹以打开它</em>$N$N快捷键:S\">多选",
|
||
"gt_crop": "中心裁剪缩略图\">裁剪",
|
||
"gt_3x": "高分辨率缩略图\">3x",
|
||
"gt_zoom": "缩放",
|
||
"gt_chop": "剪裁",
|
||
"gt_sort": "排序依据",
|
||
"gt_name": "名称",
|
||
"gt_sz": "大小",
|
||
"gt_ts": "日期",
|
||
"gt_ext": "类型",
|
||
"gt_c1": "截断文件名更多(显示更少)",
|
||
"gt_c2": "截断文件名更少(显示更多)",
|
||
|
||
"sm_w8": "正在搜索...",
|
||
"sm_prev": "上次查询的搜索结果:\n ",
|
||
"sl_close": "关闭搜索结果",
|
||
"sl_hits": "显示 {0} 个结果",
|
||
"sl_moar": "加载更多",
|
||
|
||
"s_sz": "大小",
|
||
"s_dt": "日期",
|
||
"s_rd": "路径",
|
||
"s_fn": "名称",
|
||
"s_ta": "标签",
|
||
"s_ua": "上传于",
|
||
"s_ad": "高级",
|
||
"s_s1": "最小 MiB",
|
||
"s_s2": "最大 MiB",
|
||
"s_d1": "最早 iso8601",
|
||
"s_d2": "最晚 iso8601",
|
||
"s_u1": "上传后",
|
||
"s_u2": "和/或之前",
|
||
"s_r1": "路径包含 (空格分隔)",
|
||
"s_f1": "名称包含 (用 -nope 否定)",
|
||
"s_t1": "标签包含 (^=开头,$=结尾)",
|
||
"s_a1": "特定元数据属性",
|
||
|
||
"md_eshow": "无法渲染 ",
|
||
"md_off": "[📜<em>readme</em>] 在 [⚙️] 中禁用 -- 文档隐藏",
|
||
|
||
"badreply": "解析服务器回复失败",
|
||
|
||
"xhr403": "403: 访问被拒绝\n\n尝试按 F5 可能会重新登录",
|
||
"xhr0": "未知(可能丢失连接到服务器,或服务器离线)",
|
||
"cf_ok": "抱歉 -- DD" + wah + "oS 保护启动\n\n事情应该在大约 30 秒后恢复\n\n如果没有任何变化,按 F5 重新加载页面",
|
||
"tl_xe1": "无法列出子文件夹:\n\n错误 ",
|
||
"tl_xe2": "404: 文件夹未找到",
|
||
"fl_xe1": "无法列出文件夹中的文件:\n\n错误 ",
|
||
"fl_xe2": "404: 文件夹未找到",
|
||
"fd_xe1": "无法创建子文件夹:\n\n错误 ",
|
||
"fd_xe2": "404: 父文件夹未找到",
|
||
"fsm_xe1": "无法发送消息:\n\n错误 ",
|
||
"fsm_xe2": "404: 父文件夹未找到",
|
||
"fu_xe1": "无法从服务器加载未发布列表:\n\n错误 ",
|
||
"fu_xe2": "404: 文件未找到??",
|
||
|
||
"fz_tar": "未压缩的 gnu-tar 文件(linux / mac)",
|
||
"fz_pax": "未压缩的 pax 格式 tar(较慢)",
|
||
"fz_targz": "gnu-tar 带 gzip 级别 3 压缩$N$N通常非常慢,所以$N建议使用未压缩的 tar",
|
||
"fz_tarxz": "gnu-tar 带 xz 级别 1 压缩$N$N通常非常慢,所以$N建议使用未压缩的 tar",
|
||
"fz_zip8": "zip 带 utf8 文件名(在 windows 7 及更早版本上可能会出现问题)",
|
||
"fz_zipd": "zip 带传统 cp437 文件名,适用于非常旧的软件",
|
||
"fz_zipc": "cp437 带 crc32 提前计算,$N适用于 MS-DOS PKZIP v2.04g(1993 年 10 月)$N(处理时间较长,在下载开始之前)",
|
||
|
||
"un_m1": "你可以删除下面的近期上传(或中止未完成的上传)",
|
||
"un_upd": "刷新",
|
||
"un_m4": "或分享下面可见的文件:",
|
||
"un_ulist": "显示",
|
||
"un_ucopy": "复制",
|
||
"un_flt": "可选过滤器: URL 必须包含",
|
||
"un_fclr": "清除过滤器",
|
||
"un_derr": '未发布删除失败:\n',
|
||
"un_f5": '出现问题,请尝试刷新或按 F5',
|
||
"un_uf5": "抱歉,你必须刷新页面(例如,按 F5 或 CTRL-R),然后才能中止此上传",
|
||
"un_nou": '<b>警告:</b> 服务器太忙,无法显示未完成的上传;稍后点击“刷新”链接',
|
||
"un_noc": '<b>警告:</b> 服务器配置中未启用/允许完全上传文件的取消发布',
|
||
"un_max": "显示前 2000 个文件(使用过滤器)",
|
||
"un_avail": "{0} 个近期上传可以被删除<br />{1} 个未完成的上传可以被中止",
|
||
"un_m2": "按上传时间排序;最新的在前:",
|
||
"un_no1": "哎呀!没有足够新的上传",
|
||
"un_no2": "哎呀!没有符合该过滤器的足够新的上传",
|
||
"un_next": "删除下面的下一个 {0} 个文件",
|
||
"un_abrt": "中止",
|
||
"un_del": "删除",
|
||
"un_m3": "正在加载你的近期上传...",
|
||
"un_busy": "正在删除 {0} 个文件...",
|
||
"un_clip": "{0} 个链接已复制到剪贴板",
|
||
|
||
"u_https1": "你应该",
|
||
"u_https2": "切换到 https",
|
||
"u_https3": "以获得更好的性能",
|
||
"u_ancient": '你的浏览器非常古老 -- 也许你应该 <a href="#" onclick="goto(\'bup\')">改用 bup</a>',
|
||
"u_nowork": "需要 Firefox 53+ 或 Chrome 57+ 或 iOS 11+",
|
||
"tail_2old": "需要 Firefox 105+ 或 Chrome 71+ 或 iOS 14.5+",
|
||
"u_nodrop": '浏览器版本低,不支持通过拖动文件到窗口来上传文件',
|
||
"u_notdir": "不是文件夹!\n\n您的浏览器太旧;\n请尝试将文件夹拖入窗口",
|
||
"u_uri": "要从其他浏览器窗口拖放图片,\n请将其拖放到大的上传按钮上",
|
||
"u_enpot": '切换到 <a href="#">简约 UI</a>(可能提高上传速度)',
|
||
"u_depot": '切换到 <a href="#">精美 UI</a>(可能降低上传速度)',
|
||
"u_gotpot": '切换到简化UI以提高上传速度,\n\n随时可以不同意并切换回去!',
|
||
"u_pott": "<p>个文件: <b>{0}</b> 已完成, <b>{1}</b> 失败, <b>{2}</b> 正在处理, <b>{3}</b> 排队中</p>",
|
||
"u_ever": "这是基本的上传工具; up2k 需要至少<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1",
|
||
"u_su2k": '这是基本的上传工具;<a href="#" id="u2yea">up2k</a> 更好',
|
||
"u_uput": '提高速度(跳过校验和)',
|
||
"u_ewrite": '你对这个文件夹没有写入权限',
|
||
"u_eread": '你对这个文件夹没有读取权限',
|
||
"u_enoi": '文件搜索在服务器配置中未启用',
|
||
"u_enoow": "无法覆盖此处的文件;需要删除权限", //m
|
||
"u_badf": '这些 {0} 个文件(共 {1} 个)被跳过,可能是由于文件系统权限:\n\n',
|
||
"u_blankf": '这些 {0} 个文件(共 {1} 个)是空白的;是否仍然上传?\n\n',
|
||
"u_applef": "这些 {0} 个文件(共 {1} 个)可能是不需要的;\n按 <code>确定/Enter</code> 跳过以下文件,\n按 <code>取消/ESC</code> 取消排除,并上传这些文件:\n\n", //m
|
||
"u_just1": '\n也许如果你只选择一个文件会更好',
|
||
"u_ff_many": "如果你使用的是 <b>Linux / MacOS / Android,</b> 那么这个文件数量 <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>可能</em> 崩溃 Firefox!</a>\n如果发生这种情况,请再试一次(或使用 Chrome)。",
|
||
"u_up_life": "此上传将在 {0} 后从服务器删除",
|
||
"u_asku": '将这些 {0} 个文件上传到 <code>{1}</code>',
|
||
"u_unpt": "你可以使用左上角的 🧯 撤销/删除此上传",
|
||
"u_bigtab": '将显示 {0} 个文件,可能会导致您的浏览器崩溃。您确定吗?',
|
||
"u_scan": '正在扫描文件...',
|
||
"u_dirstuck": '您的浏览器无法访问以下 {0} 个文件/文件夹,它们将被跳过:',
|
||
"u_etadone": '完成 ({0}, {1} 个文件)',
|
||
"u_etaprep": '(准备上传)',
|
||
"u_hashdone": '哈希完成',
|
||
"u_hashing": '哈希',
|
||
"u_hs": '正在等待服务器...',
|
||
"u_started": "文件现在正在上传 🚀", //m
|
||
"u_dupdefer": "这是一个重复文件。它将在所有其他文件上传后进行处理",
|
||
"u_actx": "单击此文本以防止切换到其他窗口/选项卡时性能下降",
|
||
"u_fixed": "好! 已修复 👍",
|
||
"u_cuerr": "上传块 {0} 的 {1} 失败;\n可能无害,继续中\n\n文件:{2}",
|
||
"u_cuerr2": "服务器拒绝上传(块 {0} 的 {1});\n稍后重试\n\n文件:{2}\n\n错误 ",
|
||
"u_ehstmp": "将重试;见右下角",
|
||
"u_ehsfin": "服务器拒绝了最终上传请求;正在重试...",
|
||
"u_ehssrch": "服务器拒绝了搜索请求;正在重试...",
|
||
"u_ehsinit": "服务器拒绝了启动上传请求;正在重试...",
|
||
"u_eneths": "进行上传握手时的网络错误;正在重试...",
|
||
"u_enethd": "测试目标存在时的网络错误;正在重试...",
|
||
"u_cbusy": "等待服务器在网络故障后再次信任我们...",
|
||
"u_ehsdf": "服务器磁盘空间不足!\n\n将继续重试,以防有人\n释放足够的空间以继续",
|
||
"u_emtleak1": "看起来你的网页浏览器可能有内存泄漏;\n请",
|
||
"u_emtleak2": ' <a href="{0}">切换到 https(推荐)</a> 或 ',
|
||
"u_emtleak3": ' ',
|
||
"u_emtleakc": '尝试以下操作:\n<ul><li>按 <code>F5</code> 刷新页面</li><li>然后在 <code>⚙️ 设置</code> 中禁用 <code>mt</code> 按钮</li><li>然后再次尝试上传</li></ul>上传会稍微慢一些,不过没关系。\n抱歉带来麻烦!\n\nPS:chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">已修复</a>此问题',
|
||
"u_emtleakf": '尝试以下操作:\n<ul><li>按 <code>F5</code> 刷新页面</li><li>然后在上传 UI 中启用 <code>🥔</code>(土豆)<li>然后再次尝试上传</li></ul>\nPS: firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">希望会在某个时点修复此问题</a>',
|
||
"u_s404": "在服务器上未找到",
|
||
"u_expl": "解释",
|
||
"u_maxconn": "大多数浏览器限制为 6,但 Firefox 允许你通过 <code>connections-per-server</code> 在 <code>about:config</code> 中提高限制",
|
||
"u_tu": '<p class="warn">警告:启用了 turbo,<span> 客户端可能无法检测和恢复不完整的上传;查看 turbo 按钮工具提示</span></p>',
|
||
"u_ts": '<p class="warn">警告:启用了 turbo,<span> 搜索结果可能不正确;查看 turbo 按钮工具提示</span></p>',
|
||
"u_turbo_c": "服务器配置中禁用了 turbo",
|
||
"u_turbo_g": "禁用 turbo,因为你在此卷中没有\n目录列表权限",
|
||
"u_life_cfg": '自动删除时间为 <input id="lifem" p="60" /> 分钟(或 <input id="lifeh" p="3600" /> 小时)',
|
||
"u_life_est": '上传将在 <span id="lifew" tt="本地时间">---</span> 删除',
|
||
"u_life_max": '此文件夹强制执行\n最大寿命为 {0}',
|
||
"u_unp_ok": '允许取消发布 {0}',
|
||
"u_unp_ng": '取消发布将不被允许',
|
||
"ue_ro": '你对这个文件夹的访问是只读的\n\n',
|
||
"ue_nl": '你当前未登录',
|
||
"ue_la": '你当前以 "{0}" 登录',
|
||
"ue_sr": '你当前处于文件搜索模式\n\n通过点击大搜索按钮旁边的放大镜 🔎 切换到上传模式,然后重试上传\n\n抱歉',
|
||
"ue_ta": '尝试再次上传,现在应该能正常工作',
|
||
"ue_ab": "这份文件正在上传到另一个文件夹,必须完成该上传后,才能将文件上传到其他位置。\n\n您可以通过左上角的🧯中止并忘记该上传。", //m
|
||
"ur_1uo": "成功:文件上传成功",
|
||
"ur_auo": "成功:所有 {0} 个文件上传成功",
|
||
"ur_1so": "成功:文件在服务器上找到",
|
||
"ur_aso": "成功:所有 {0} 个文件在服务器上找到",
|
||
"ur_1un": "上传失败,抱歉",
|
||
"ur_aun": "所有 {0} 个上传失败,抱歉",
|
||
"ur_1sn": "文件未在服务器上找到",
|
||
"ur_asn": "这些 {0} 个文件未在服务器上找到",
|
||
"ur_um": "完成;\n{0} 个上传成功,\n{1} 个上传失败,抱歉",
|
||
"ur_sm": "完成;\n{0} 个文件在服务器上找到,\n{1} 个文件未在服务器上找到",
|
||
|
||
"lang_set": "刷新以使更改生效?",
|
||
},
|
||
"deu": {
|
||
"tt": "Deutsch",
|
||
|
||
"cols": {
|
||
"c": "Aktionen",
|
||
"dur": "Dauer",
|
||
"q": "Qualität / Bitrate",
|
||
"Ac": "Audiocodec",
|
||
"Vc": "Videocodec",
|
||
"Fmt": "Format / Container",
|
||
"Ahash": "Audio Checksumme",
|
||
"Vhash": "Video Checksumme",
|
||
"Res": "Auflösung",
|
||
"T": "Dateityp",
|
||
"aq": "Audioqualität / Bitrate",
|
||
"vq": "Videoqualität / Bitrate",
|
||
"pixfmt": "Subsampling / Pixelstruktur",
|
||
"resw": "horizontale Auflösung",
|
||
"resh": "vertikale Auflösung",
|
||
"chs": "Audiokanäle",
|
||
"hz": "Abtastrate"
|
||
},
|
||
|
||
"hks": [
|
||
[
|
||
"misc",
|
||
["ESC", "Dinge schliessen"],
|
||
|
||
"file-manager",
|
||
["G", "zwischen Liste und Gitter wechseln"],
|
||
["T", "zwischen Vorschaubildern und Symbolen wechseln"],
|
||
["⇧ A/D", "Vorschaubildergrösse ändern"],
|
||
["STRG-K", "Auswahl löschen"],
|
||
["STRG-X", "Auswahl ausschneiden"],
|
||
["STRG-C", "Auswahl in Zwischenablage kopieren"],
|
||
["STRG-V", "Zwischenablage hier einfügen"],
|
||
["Y", "Auswahl herunterladen"],
|
||
["F2", "Auswahl umbenennen"],
|
||
|
||
"file-list-sel",
|
||
["LEER", "Dateiauswahl aktivieren"],
|
||
["↑/↓", "Cursor verschieben"],
|
||
["STRG ↑/↓", "Cursor und Bildschirm verschieben"],
|
||
["⇧ ↑/↓", "Vorherige / nächste Datei auswählen"],
|
||
["STRG-A", "Alle Dateien / Ordner auswählen"],
|
||
], [
|
||
"navigation",
|
||
["B", "Zwischen Brotkrumen und Navpane wechseln"],
|
||
["I/K", "vorheriger / nächster Ordner"],
|
||
["M", "übergeordneter Ordner (oder Vorherigen einklappen)"],
|
||
["V", "Zwischen Textdateien und Navpane wechseln"],
|
||
["A/D", "Grösse der Navpane ändern"],
|
||
], [
|
||
"audio-player",
|
||
["J/L", "Vorheriger / nächster Song"],
|
||
["U/O", "10 Sek. vor- / zurückspringen"],
|
||
["0..9", "zu 0%..90% springen"],
|
||
["P", "Wiedergabe / Pause"],
|
||
["S", "aktuell abgespielten Song auswählen"],
|
||
["Y", "Sing herunterladen"],
|
||
], [
|
||
"image-viewer",
|
||
["J/L, ←/→", "vorheriges / nächstes Bild"],
|
||
["Pos1/Ende", "erstes / letztes Bild"],
|
||
["F", "Vollbild"],
|
||
["R", "im Uhrzeigersinn drehen"],
|
||
["⇧ R", "gegen den Uhrzeigensinn drehen"],
|
||
["S", "Bild auswählen"],
|
||
["Y", "Bild herunterladen"],
|
||
], [
|
||
"video-player",
|
||
["U/O", "10 Sek. vor- / zurückspringen"],
|
||
["P/K/LEER", "Wiedergabe / Pause"],
|
||
["C", "continue playing next"],
|
||
["V", "Wiederholungs-Wiedergabe (Loop)"],
|
||
["M", "Stummschalten"],
|
||
["[ und ]", "Loop-Interval einstellen"],
|
||
], [
|
||
"textfile-viewer",
|
||
["I/K", "vorherige / nächste Datei"],
|
||
["M", "Textdatei schliessen"],
|
||
["E", "Textdatei bearbeiten"],
|
||
["S", "Textdatei auswählen (für Ausschneiden / Kopieren / Umbenennen)"],
|
||
]
|
||
],
|
||
|
||
"m_ok": "OK",
|
||
"m_ng": "Abbrechen",
|
||
|
||
"enable": "Aktivieren",
|
||
"danger": "ACHTUNG",
|
||
"clipped": "in Zwischenablage kopiert",
|
||
|
||
"ht_s1": "Sekunde",
|
||
"ht_s2": "Sekunden",
|
||
"ht_m1": "Minute",
|
||
"ht_m2": "Minuten",
|
||
"ht_h1": "Stunde",
|
||
"ht_h2": "Stunden",
|
||
"ht_d1": "Tag",
|
||
"ht_d2": "Tage",
|
||
"ht_and": " und ",
|
||
|
||
"goh": "Einstellungen",
|
||
"gop": 'zum vorherigen Ordner springen">vorh.',
|
||
"gou": 'zum übergeordneter Ordner springen">hoch',
|
||
"gon": 'zum nächsten Ordner springen">nächst.',
|
||
"logout": "Abmelden ",
|
||
"access": " Zugriff",
|
||
"ot_close": "Submenu schliessen",
|
||
"ot_search": "Dateien nach Attributen, Pfad/Name, Musiktags oder beliebiger Kombination suchen$N$N<code>foo bar</code> = muss «foo» und «bar» enthalten,$N<code>foo -bar</code> = muss «foo» aber nicht «bar» enthalten,$N<code>^yana .opus$</code> = beginnt mit «yana» und ist «opus»-Datei$N<code>"try unite"</code> = genau «try unite» enthalten$N$NDatumsformat ist iso-8601, z.B.$N<code>2009-12-31</code> oder <code>2020-09-12 23:30:00</code>",
|
||
"ot_unpost": "unpost: lösche deine letzten Uploads oder breche unvollständige ab",
|
||
"ot_bup": "bup: Basic Uploader, unterstützt sogar Neuheiten wie Netscape 4.0",
|
||
"ot_mkdir": "mkdir: Neuen Ordner erstellen",
|
||
"ot_md": "new-md: Neues Markdown-Dokument erstellen",
|
||
"ot_msg": "msg: Eine Nachricht an das Server-Log schicken",
|
||
"ot_mp": "Media Player-Optionen",
|
||
"ot_cfg": "Konfigurationsoptionen",
|
||
"ot_u2i": 'up2k: Dateien hochladen (wenn du Schreibrechte hast) oder in den Suchmodus wechseln, um zu prüfen, ob sie bereits auf dem Server existieren$N$NUploads sind fortsetzbar, multithreaded und behalten Dateizeitstempel, verbrauchen aber mehr CPU als [🎈] (der einfache Uploader)<br /><br />während Uploads wird dieses Symbol zu einem Fortschrittsanzeiger!',
|
||
"ot_u2w": 'up2k: Dateien mit Wiederaufnahme-Unterstützung hochladen (Browser schließen und später dieselben Dateien erneut hochladen)$N$Nmultithreaded, behält Dateizeitstempel, verbraucht aber mehr CPU als [🎈] (der einfache Uploader)<br /><br />während Uploads wird dieses Symbol zu einem Fortschrittsanzeiger!',
|
||
"ot_noie": 'Bitte benutze Chrome / Firefox / Edge',
|
||
|
||
"ab_mkdir": "Ordner erstellen",
|
||
"ab_mkdoc": "Markdown Doc erstellen",
|
||
"ab_msg": "Nachricht an Server Log senden",
|
||
|
||
"ay_path": "zu Ordnern springen",
|
||
"ay_files": "zu Dateien springen",
|
||
|
||
"wt_ren": "ausgewählte Elemente umbenennen$NHotkey: F2",
|
||
"wt_del": "ausgewählte Elemente löschen$NHotkey: STRG-K",
|
||
"wt_cut": "ausgewählte Elemente ausschneiden <small>(um sie dann irgendwo anders einzufügen)</small>$NHotkey: STRG-X",
|
||
"wt_cpy": "ausgewählte Elemente in Zwischenablage kopieren$N(um sie dann irgendwo anders einzufügen)$NHotkey: ctrl-C",
|
||
"wt_pst": "zuvor ausgeschnittenen / kopierte Elemente einfügen$NHotkey: STRG-V",
|
||
"wt_selall": "alle Dateien auswählen$NHotkey: STRG-A (wenn Datei fokusiert)",
|
||
"wt_selinv": "Auswahl invertieren",
|
||
"wt_zip1": "Diesen Ordner als Archiv herunterladen",
|
||
"wt_selzip": "Auswahl als Archiv herunterladen",
|
||
"wt_seldl": "Auswahl als separate Dateien herunterladen$NHotkey: Y",
|
||
"wt_npirc": "kopiere Titelinfo als IRC-formattierten Text",
|
||
"wt_nptxt": "kopiere Titelinfo als Text",
|
||
"wt_m3ua": "Zu M3U-Wiedergabeliste hinzufügen (wähle später <code>📻copy</code>)",
|
||
"wt_m3uc": "M3U-Wiedergabeliste in Zwischenablage kopieren",
|
||
"wt_grid": "Zwischen Gitter und Liste wechseln$NHotkey: G",
|
||
"wt_prev": "Vorheriger Titel$NHotkey: J",
|
||
"wt_play": "Wiedergabe / Pause$NHotkey: P",
|
||
"wt_next": "Nächster Titel$NHotkey: L",
|
||
|
||
"ul_par": "Parallele Uploads:",
|
||
"ut_rand": "Zufällige Dateinamen",
|
||
"ut_u2ts": "Zuletzt geändert-Zeitstempel von$Ndeinem Dateisystem auf den Server übertragen\">📅",
|
||
"ut_ow": "Existierende Dateien auf dem Server überschreiben?$N🛡️: Nie (generiert einen neuen Dateinamen)$N🕒: Überschreiben, wenn Server-Datei älter ist als meine$N♻️: Überschreiben, wenn der Dateiinhalt anders ist",
|
||
"ut_mt": "Andere Dateien während des Uploads hashen$N$Nsolltest du deaktivieren, falls deine CPU oder Festplatte zum Flaschenhals werden könnte",
|
||
"ut_ask": 'Vor dem Upload nach Bestätigung fragen">💭',
|
||
"ut_pot": "Verbessert Upload-Geschwindigkeit$Nindem das UI weniger komplex gemacht wird",
|
||
"ut_srch": "nicht wirklich hochladen, stattdessen prüfen ob Datei bereits auf dem Server existiert (scannt alle Ordner, die du lesen kannst)",
|
||
"ut_par": "setze auf 0 zum Pausieren$N$Nerhöhe, wenn deine Verbindung langsam / instabil ist$N$lass auf 1 im LAN oder wenn die Festplatte auf dem Server ein Flaschenhals ist",
|
||
"ul_btn": "Dateien / Ordner hier<br>ablegen (oder klick mich)",
|
||
"ul_btnu": "U P L O A D",
|
||
"ul_btns": "S U C H E N",
|
||
|
||
"ul_hash": "hash",
|
||
"ul_send": "senden",
|
||
"ul_done": "fertig",
|
||
"ul_idle1": "keine Uploads in der Warteschlange",
|
||
"ut_etah": "durchschnittl. <em>hashing</em> Geschw. & gesch. Restzeit",
|
||
"ut_etau": "durchschnittl. <em>upload</em> Geschw. & gesch. Restzeit",
|
||
"ut_etat": "durchschnittl. <em>total</em> Geschw. & gesch. Restzeit",
|
||
|
||
"uct_ok": "Erfolgreich abgeschlossen",
|
||
"uct_ng": "no-good: fehlgeschlagen / abgelehnt / nicht gefunden",
|
||
"uct_done": "ok and ng zusammen",
|
||
"uct_bz": "wird gehasht oder hochgeladen",
|
||
"uct_q": "ausstehend",
|
||
|
||
"utl_name": "Dateiname",
|
||
"utl_ulist": "Liste",
|
||
"utl_ucopy": "kopieren",
|
||
"utl_links": "Links",
|
||
"utl_stat": "Status",
|
||
"utl_prog": "Fortschritt",
|
||
|
||
// keep short:
|
||
"utl_404": "404",
|
||
"utl_err": "Fehler",
|
||
"utl_oserr": "OS-Fehler",
|
||
"utl_found": "gefunden",
|
||
"utl_defer": "zurückstellen",
|
||
"utl_yolo": "YOLO",
|
||
"utl_done": "fertig",
|
||
|
||
"ul_flagblk": "Die Dateien wurden zur Warteschlange hinzugefügt</b><br>jedoch ist up2k gerade in einem anderen Browsertab aktiv.<br>Ich warte, bis der Upload abgeschlossen ist.",
|
||
"ul_btnlk": "Die Serverkonfiguration hat diese Einstellung gesperrt",
|
||
|
||
"udt_up": "Upload",
|
||
"udt_srch": "Suchen",
|
||
"udt_drop": "hier ablegen",
|
||
|
||
"u_nav_m": '<h6>okay, was gibts??</h6><code>Eingabe</code> = Dateien (1 oder mehr)\n<code>ESC</code> = 1 Ordner (inkl. Unterordner)',
|
||
"u_nav_b": '<a href="#" id="modal-ok">Dateien</a><a href="#" id="modal-ng">1 Ordner</a>',
|
||
|
||
"cl_opts": "Schalter",
|
||
"cl_themes": "Themes",
|
||
"cl_langs": "Sprache",
|
||
"cl_ziptype": "Ordner Download",
|
||
"cl_uopts": "up2k Schalter",
|
||
"cl_favico": "Favicon",
|
||
"cl_bigdir": "grosse Ordner",
|
||
"cl_hsort": "#sort",
|
||
"cl_keytype": "Schlüsselnotation",
|
||
"cl_hiddenc": "Spalten verstecken",
|
||
"cl_hidec": "verstecken",
|
||
"cl_reset": "zurücksetzen",
|
||
"cl_hpick": "zum Verstecken, tippe auf Spaltenüberschriften in der Tabelle unten",
|
||
"cl_hcancel": "Spaltenbearbeitung abgebrochen",
|
||
|
||
"ct_grid": '田 Das Raster™',
|
||
"ct_ttips": '◔ ◡ ◔">ℹ️ Tooltips',
|
||
"ct_thumb": 'In Raster-Ansicht, zwischen Icons und Vorschau wechseln$NHotkey: T">🖼️ Vorschaubilder',
|
||
"ct_csel": 'Benutze STRG und UMSCHALT für Dateiauswahl in Raster-Ansicht">sel',
|
||
"ct_ihop": 'Wenn die Bildanzeige geschlossen ist, scrolle runter zu den zuletzt angesehenen Dateien">g⮯',
|
||
"ct_dots": 'Verstecke Dateien anzeigen (wenn erlaubt durch Server)">dotfiles',
|
||
"ct_qdel": 'Nur einmal fragen, wenn mehrere Dateien gelöscht werden">qdel',
|
||
"ct_dir1st": 'Ordner vor Dateien sortieren">📁 zuerst',
|
||
"ct_nsort": 'Natürliche Sortierung (für Dateinamen mit führenden Ziffern)">nsort',
|
||
"ct_readme": 'README.md in Dateiliste anzeigen">📜 readme',
|
||
"ct_idxh": 'index.html anstelle von Dateiliste anzeigen">htm',
|
||
"ct_sbars": 'Scrollbars zeigen">⟊',
|
||
|
||
"cut_umod": "Sollte die Datei bereits auf dem Server existieren, den 'Zuletzt geändert'-Zeitstempel an deine lokale Datei anpassen (benötigt Lese- und Löschrechte)\">re📅",
|
||
|
||
"cut_turbo": "der YOLO-Knopf, den du wahrscheinlich NICHT aktivieren willst:$N$NBenutze ihn, falls du ne Menge Zeug hochladen wolltest und aus irgendeinem Grund neustarten musstest und du so schnell wie möglich weitermachen willst.$N$Ndies ersetzt den Hash-Check mit einem einfachen <em>"Ist die Datei auf dem Server gleich gross?"</em>, wenn die Datei also anderen Inhalt hat, wird sie NICHT nochmal hochgeladen!$N$NDu solltest dieses Feature ausschalten, sobald der Upload fertig ist und dann die gleichen Dateien nochmal "hochladen", damit der Client sie verifizieren kann.\">turbo",
|
||
|
||
"cut_datechk": "Funktioniert nur in kombination mit dem Turbo-Knopf$N$NReduziert den YOLO-Faktor ein bisschen; prüft, ob der Zeitstempel deiner Datei mit dem auf dem Server übereinstimmt$N$Nsollte <em>theoretisch</em> die meisten unfertigen / korrupten Uploads erwischen, ist aber nicht zu gebrauchen, um einen Prüfdurchgang nach einem Turbo-Upload zu machen\">date-chk",
|
||
|
||
"cut_u2sz": "Grösse (in MiB) für jeden Upload-Chunk; mit grossen Werten fliegen die Bits besser über den Atlantik. Versuche kleine Werte, wenn du eine schlechte Verbindung hast (z.B. du benutzt mobile Daten in Deutschland)",
|
||
|
||
"cut_flag": "Stelle sicher, dass nur ein Tab auf einmal Dateien hochlädt$N -- andere Tabs müssen diese Funktion auch aktiviert haben $N -- funktioniert nur bei Tabs mit der gleichen Domäne",
|
||
|
||
"cut_az": "Lädt Dateien in alphabetischer Reihenfolge hoch, anstatt nach Dateigrösse$N$NAlphabethische Reihenfolge kann es einfacher machen, Server-Fehler mit naktem Auge zu erkennen, macht aber Uploads über Glassfaser / LAN etwas langsamer",
|
||
|
||
"cut_nag": "Benachrichtigung über das Betriebssystem abgeben, wenn Upload fertig ist$N(nur wenn Browser oder Tab nicht im Vordergrund ist)",
|
||
"cut_sfx": "Spielt ein Ton ab, wenn Upload fertig ist$N(nur wenn Browser oder Tab nicht im Vordergrund ist)",
|
||
|
||
"cut_mt": "Multithreading benutzen um Datei-Hashing zu beschleunigen$N$NDies nutzt Web-Workers und benötigt$Nmehr RAM (bis zu 512 MiB extra)$N$Nbeschleunigt HTTPS 30% schneller, HTTP um 4.5x\">mt",
|
||
|
||
"cut_wasm": "benutzt WASM anstelle des Browser-eigenen Hashers; verbessert Geschwindigkeit auf Chromium-basierten Browsern, erhöht aber die CPU-Auslastung. Viele ältere Versionen von Chrome haben Memory-Leaks, die den gesamten RAM verbrauchen und dann crashen, wenn diese Funktion aktiviert ist.\">wasm",
|
||
|
||
"cft_text": "Favicon Text (leer lassen und neuladen zum Deaktivieren)",
|
||
"cft_fg": "Vordergrundfarbe",
|
||
"cft_bg": "Hintergrundfarbe",
|
||
|
||
"cdt_lim": "max. Anz. Dateien, die in einem Ordner gezeigt werden sollen",
|
||
"cdt_ask": "beim Runterscrollen nach $NAktion fragen statt mehr,$NDateien zu laden",
|
||
"cdt_hsort": "Menge an Sortierregeln (<code>,sorthref</code>) in Media-URLs enthalten sein sollen. Ein Wert von 0 sorgt dafür, dass Sortierregeln in Media-URLs ignoriert werden",
|
||
|
||
"tt_entree": "Navpane anzeigen (Ordnerbaum Sidebar)$NHotkey: B",
|
||
"tt_detree": "Breadcrumbs anzeigen$NHotkey: B",
|
||
"tt_visdir": "zu ausgewähltem Ordner scrollen",
|
||
"tt_ftree": "zw. Ordnerbaum / Textdateien wechseln$NHotkey: V",
|
||
"tt_pdock": "übergeordnete Ordner in einem angedockten Fenster oben anzeigen",
|
||
"tt_dynt": "autom. wachsen wenn Baum wächst",
|
||
"tt_wrap": "Zeilenumbruch",
|
||
"tt_hover": "Beim Hovern überlange Zeilen anzeigen$N(Scrollen funktioniert nicht ausser $N Cursor ist im linken Gutter)",
|
||
|
||
"ml_pmode": "am Ende des Ordners...",
|
||
"ml_btns": "cmds",
|
||
"ml_tcode": "transcodieren",
|
||
"ml_tcode2": "transcodieren zu",
|
||
"ml_tint": "färben",
|
||
"ml_eq": "Audio Equalizer",
|
||
"ml_drc": "Dynamic Range Compressor",
|
||
|
||
"mt_loop": "Song wiederholen\">🔁",
|
||
"mt_one": "Wiedergabe nach diesem Song beenden\">1️⃣",
|
||
"mt_shuf": "Zufällige Wiedergabe im Ordner\">🔀",
|
||
"mt_aplay": "automatisch abspielen, wenn der Link, mit dem du auf den Server zugreifst, eine Titel-ID enthält$N$NDeaktivieren verhindert auch, dass die Seiten-URL bei Musikwiedergabe mit Titel-IDs aktualisiert wird, um Autoplay zu verhindern, falls diese Einstellungen verloren gehen, die URL aber bestehen bleibt\">a▶",
|
||
"mt_preload": "nächsten Titel gegen Ende vorladen für nahtlose Wiedergabe\">Vorladen",
|
||
"mt_prescan": "vor Ende des letzten Titels zum nächsten Ordner wechseln,$Ndamit der Browser die$NWiedergabe nicht stoppt\">Navigation",
|
||
"mt_fullpre": "versuchen, den gesamten Titel vorzuladen;$N✅ bei <b>unzuverlässiger</b> Verbindung aktivieren,$N❌ bei langsamer Verbindung deaktivieren\">vollständig",
|
||
"mt_fau": "auf Handys verhindern, dass Musik stoppt, wenn der nächste Titel nicht schnell genug vorlädt (kann zu fehlerhafter Tag-Anzeige führen)\">☕️",
|
||
"mt_waves": "Wellenform-Suchleiste:$NAudio-Amplitude in der Leiste anzeigen\">~s",
|
||
"mt_npclip": "Buttons zum Kopieren des aktuellen Titels anzeigen\">/np",
|
||
"mt_m3u_c": "Buttons zum Kopieren der$Nausgewählten Titel als m3u8-Wiedergabeliste anzeigen\">📻",
|
||
"mt_octl": "OS-Integration (Media-Hotkeys/OSD)\">os-ctl",
|
||
"mt_oseek": "Suchen via OS-Integration erlauben$N$NHinweis: auf einigen Geräten (iPhones)$Nersetzt dies den nächsten-Titel-Button\">Suchen",
|
||
"mt_oscv": "Albumcover in OSD anzeigen\">Cover",
|
||
"mt_follow": "den spielenden Titel im Blick behalten\">🎯",
|
||
"mt_compact": "kompakte Steuerelemente\">⟎",
|
||
"mt_uncache": "Cache leeren (probier das, wenn dein Browser$Neine defekte Kopie eines Titels zwischenspeichert und sich weigert, ihn abzuspielen)\">Cache leeren",
|
||
"mt_mloop": "offenen Ordner wiederholen\">🔁 Schleife",
|
||
"mt_mnext": "nächsten Ordner laden und fortfahren\">📂 nächster",
|
||
"mt_mstop": "Wiedergabe beenden\">⏸ Stop",
|
||
"mt_cflac": "FLAC / WAV zu OPUS konvertierebn\">flac",
|
||
"mt_caac": "AAC / M4A zu OPUS konvertieren\">aac",
|
||
"mt_coth": "Convertiere alle Dateien (die nicht MP3 sind) zu OPUS\">oth",
|
||
"mt_c2opus": "Beste Wahl für Desktops, Laptops, Android\">opus",
|
||
"mt_c2owa": "opus-weba, für iOS 17.5 und neuer\">owa",
|
||
"mt_c2caf": "opus-caf, für iOS 11 bis 17\">caf",
|
||
"mt_c2mp3": "benutze dieses Format für ältere Geräte\">mp3",
|
||
"mt_c2ok": "Gute Wahl, Chef!",
|
||
"mt_c2nd": "Das ist nicht das empfohlene Ausgabeformat für dein Gerät, aber passt schon",
|
||
"mt_c2ng": "Dein Gerät scheint dieses Ausgabeformat nicht zu unterstützen, aber lass trotzdem mal probieren",
|
||
"mt_xowa": "Es gibt Bugs in iOS, die die Hintergrund-Wiedergabe mit diesem Format verhindern; bitte nutze caf oder mp3 stattdessen",
|
||
"mt_tint": "Hintergrundlevel (0-100) auf der Seekbar$Num Buffern weniger ablenkend zu machen",
|
||
"mt_eq": "Aktiviert Equalizer und Lautstärkeregelung;$N$Nboost <code>0</code> = Standard 100% Lautstärke (unverändert)$N$Nwidth <code>1 </code> = Standard Stereo (unverändert)$Nwidth <code>0.5</code> = 50% Links-Rechts-Crossfeed$Nwidth <code>0 </code> = Mono$N$Nboost <code>-0.8</code> & width <code>10</code> = Gesangsentfernung :^)$N$NDer Equalizer macht nahtlose Alben vollständig nahtlos, also lass' ihn mit allen Werten auf Null (außer width = 1) aktiviert, wenn dir das wichtig ist",
|
||
"mt_drc": "Aktiviert den Dynamic Range Compressor (Lautstärkeglättung/-begrenzung); aktiviert auch den Equalizer zum Ausgleich, setze alle EQ-Felder außer 'width' auf 0, wenn du das nicht willst$N$Nsenkt die Lautstärke von Audio über SCHWELLENWERT dB; für jedes VERHÄLTNIS dB über SCHWELLENWERT gibt es 1 dB Ausgabe, also bedeuten Standardwerte von tresh -24 und ratio 12, dass es nie lauter als -22 dB werden sollte und der Equalizer-Boost sicher auf 0.8 oder sogar 1.8 mit ATK 0 und einem großen RLS wie 90 erhöht werden kann (funktioniert nur in Firefox; in anderen Browsern ist RLS max. 1)$N$N(siehe Wikipedia, dort wird es viel besser erklärt)",
|
||
|
||
"mb_play": "Abspielen",
|
||
"mm_hashplay": "Diese Audiodatei abspielen?",
|
||
"mm_m3u": "Drücke <code>Eingabe/OK</code> zum Abspielen\nDrücke <code>ESC/Abbrechen</code> zum Bearbeiten",
|
||
"mp_breq": "Benötigt Firefox 82+ oder Chrome 73+ oder iOS 15+",
|
||
"mm_bload": "Lädt...",
|
||
"mm_bconv": "Konvertiere zu {0}, bitte warte...",
|
||
"mm_opusen": "Dein Browser kann AAC- / M4A-Dateien nicht abspielen;\nUmwandlung zu Opus ist jetzt aktiv",
|
||
"mm_playerr": "Wiedergabefehler: ",
|
||
"mm_eabrt": "Der Wiedergabeversuch wurde abgebrochen",
|
||
"mm_enet": "Dein Internet läuft auf Edge, wa?",
|
||
"mm_edec": "Die Datei scheint beschädigt zu sein??",
|
||
"mm_esupp": "Dein Browser versteht dieses Audioformat nicht",
|
||
"mm_eunk": "Unbekannter Fehler",
|
||
"mm_e404": "Konnte Datei nicht abspielen; Fehler 404: Datei nicht gefunden.",
|
||
"mm_e403": "Konnte Datei nicht abspielen; Fehler 403: Zugriff verweigert.\n\nDrücke F5 zum Neuladen, vielleicht wurdest du abgemeldet",
|
||
"mm_e500": "Konnte Datei nicht abspielen; Fehler 500: Prüfe die Serverlogs.",
|
||
"mm_e5xx": "Konnte Datei nicht abspielen; Server Fehler ",
|
||
"mm_nof": "finde keine weiteren Audiodateien in der Nähe",
|
||
"mm_prescan": "Suche nach Musik zum Abspielen...",
|
||
"mm_scank": "Nächster Song gefunden:",
|
||
"mm_uncache": "Cache geleert; Alle Songs werden beim nächsten Abspielversuch neu heruntergeladen",
|
||
"mm_hnf": "dieser Song existiert nicht mehr",
|
||
|
||
"im_hnf": "dieses Bild existiert nicht mehr",
|
||
|
||
"f_empty": 'Dieser Ordner ist leer',
|
||
"f_chide": 'Dies blendet die Spalte «{0}» aus\n\nDu kannst Spalten in den Einstellungen wieder einblenden.',
|
||
"f_bigtxt": "Diese Datei ist {0} MiB gross -- Sicher, dass du sie als Text anzeigen willst?",
|
||
"f_bigtxt2": "Möchtest du stattdessen nur das Ende der Datei anzeigen? Das aktiviert ausserdem die Folgen- und Verfolgen-Funktion, welche neu hinzugefügte Textzeilen in Echtzeit anzeigt",
|
||
"fbd_more": '<div id="blazy">zeige <code>{0}</code> von <code>{1}</code> Dateien; <a href="#" id="bd_more">{2} anzeigen</a> oder <a href="#" id="bd_all">alle anzeigen</a></div>',
|
||
"fbd_all": '<div id="blazy">zeige <code>{0}</code> von <code>{1}</code> Dateien; <a href="#" id="bd_all">alle anzeigen</a></div>',
|
||
"f_anota": "nur {0} der {1} Elemente wurden ausgewählt;\num den gesamten Ordner auszuwählen, zuerst nach unten scrollen",
|
||
|
||
"f_dls": 'die Dateilinks im aktuellen Ordner wurden\nin Downloadlinks geändert',
|
||
|
||
"f_partial": "Um eine Datei sicher herunterzuladen, die gerade hochgeladen wird, klicke bitte die Datei mit dem gleichen Namen, aber ohne die <code>.PARTIAL</code>-Endung. Bitte drücke Abbrechen oder Escape, um dies zu tun.\n\nWenn du auf OK / Eingabe drückst, ignorierst du diese Warnung und lädst die <code>.PARTIAL</code>-Datei herunter, die ziemlich sicher beschädigte Daten enthält.",
|
||
|
||
"ft_paste": "{0} Elemente einfügen$NHotkey: STRG-V",
|
||
"fr_eperm": 'Umbenennen fehlgeschlagen:\nDir fehlt die "Verschieben"-Berechtigung in diesem Ordner',
|
||
"fd_eperm": 'Löschen fehlgeschlagen:\nDir fehlt die "Löschen"-Berechtigung in diesem Ordner',
|
||
"fc_eperm": 'Ausschneiden fehlgeschlagen:\nDir fehlt die "Verschieben"-Berechtigung in diesem Ordner',
|
||
"fp_eperm": 'Einfügen fehlgeschlagen:\nDir fehlt die "Schreiben"-Berechtigung in diesem Ordner',
|
||
"fr_emore": "Wähle mindestens ein Element zum Umbenennen aus",
|
||
"fd_emore": "Wähle mindestens ein Element zum Löschen aus",
|
||
"fc_emore": "Wähle mindestens ein Element zum Ausschneiden aus",
|
||
"fcp_emore": "Wähle mindestens ein Element aus, um es in die Zwischenablage zu kopieren",
|
||
|
||
"fs_sc": "Teile diesen Ordner",
|
||
"fs_ss": "Teile die ausgewählten Dateien",
|
||
"fs_just1d": "Du kannst nicht mehrere Ordner auswählen \noder Dateien und Ordner in der Auswahl mischen.",
|
||
"fs_abrt": "❌ Abbrechen",
|
||
"fs_rand": "🎲 Zufallsname",
|
||
"fs_go": "✅ Share erstellen",
|
||
"fs_name": "Name",
|
||
"fs_src": "Quelle",
|
||
"fs_pwd": "Passwort",
|
||
"fs_exp": "Ablauf",
|
||
"fs_tmin": "Minuten",
|
||
"fs_thrs": "Stunden",
|
||
"fs_tdays": "Tage",
|
||
"fs_never": "nie",
|
||
"fs_pname": "optionaler Linkname; zufällig wenn leer",
|
||
"fs_tsrc": "zu teilende Datei oder Ordner",
|
||
"fs_ppwd": "optionales Passwort",
|
||
"fs_w8": "erstelle Share...",
|
||
"fs_ok": "drücke <code>Eingabe/OK</code> für Zwischenablage\ndrücke <code>ESC/Abbrechen</code> zum Schliessen",
|
||
|
||
"frt_dec": "Kann Fälle von beschädigten Dateien beheben\">url-decode",
|
||
"frt_rst": "Geänderte Dateinamen auf Orginale zurücksetzen\">↺ zurücksetzen",
|
||
"frt_abrt": "Abbrechen und dieses Fenster schliessen\">❌ abbrechen",
|
||
"frb_apply": "ÜBERNEHMEN",
|
||
"fr_adv": "Stapel-/Metadaten-/Musterumbenennung\">erweitert",
|
||
"fr_case": "Groß-/Kleinschreibung beachten (Regex)\">Großschreibung",
|
||
"fr_win": "Windows-kompatible Namen; ersetzt <code><>:"\\|?*</code> durch japanische Fullwidth-Zeichen\">win",
|
||
"fr_slash": "Ersetzt <code>/</code> durch ein Zeichen, das keine neuen Ordner erstellt\">no /",
|
||
"fr_re": "Regex-Suchmuster für Originaldateinamen; Erfassungsgruppen können im Formatfeld unten als <code>(1)</code> und <code>(2)</code> usw. referenziert werden",
|
||
"fr_fmt": "inspiriert von foobar2000:$N<code>(title)</code> wird durch Songtitel ersetzt,$N<code>[(artist) - ](title)</code> überspringt [diesen] Teil falls Interpret leer$N<code>$lpad((tn),2,0)</code> füllt die Titelnummer auf 2 Ziffern auf",
|
||
"fr_pdel": "Löschen",
|
||
"fr_pnew": "Speichern als",
|
||
"fr_pname": "Gib der Vorlage einen Namen",
|
||
"fr_aborted": "Abgebrochen",
|
||
"fr_lold": "Alter Name",
|
||
"fr_lnew": "Neuer Name",
|
||
"fr_tags": "Tags für die ausgewählten Dateien (liest nur, als Referenz):",
|
||
"fr_busy": "Benenne {0} Elemente um...\n\n{1}",
|
||
"fr_efail": "Umbenennen fehlgeschlagen:\n",
|
||
"fr_nchg": "{0} der neuen Namen wurden angepasst durch <code>win</code> und/oder <code>no /</code>\n\nMöchtest du mit diesen geänderten Namen fortfahren?",
|
||
|
||
"fd_ok": "Löschen OK",
|
||
"fd_err": "Löschen fehlgeschlagen:\n",
|
||
"fd_none": "Nichts würde gelöscht; vielleicht durch die Serverkonfiguration blockiert (xbd)?",
|
||
"fd_busy": "Lösche {0} Elemente...\n\n{1}",
|
||
"fd_warn1": "Diese {0} Elemente LÖSCHEN?",
|
||
"fd_warn2": "<b>Ich frage das letzte Mal!</b> Was weg ist, ist weg. Keine Chance, das rückgängig zu machen. Löschen?",
|
||
|
||
"fc_ok": "{0} Elemente ausgeschnitten",
|
||
"fc_warn": '{0} Elemente in die Zwischenablage kopiert\n\nAber: nur <b>dieses</b> Browsertab kann sie einfügen\n(da deine Auswahl so abartig riesig war)',
|
||
|
||
"fcc_ok": "{0} Elemente in die Zwischenablage kopiert",
|
||
"fcc_warn": '{0} Elemente in die Zwischenablage kopiert\n\nAber: nur <b>dieses</b> Browsertab kann sie einfügen\n(da deine Auswahl so abartig riesig war)',
|
||
|
||
"fp_apply": "Diese Namen verwenden",
|
||
"fp_ecut": "Kopiere erst ein paar Dateien / Ordner, um sie einzufügen\n\nTipp: Ausschneiden und Kopieren funktioniert über Browsertabs hinweg",
|
||
"fp_ename": "{0} Elemente konnten nicht verschoben werden, weil bereits andere Dateien mit diesen Namen existieren. Gib ihnen unten neue Namen um fortzufahren, oder lass das Feld leer zum Überspringen:",
|
||
"fcp_ename": "{0} Elemente konnten nicht kopiert werden, weil bereits andere Dateien mit diesen Namen existieren. Gib ihnen unten neue Namen um fortzufahren, oder lass das Feld leer zum Überspringen:",
|
||
"fp_emore": "Es gibt noch ein paar Dateinamen, die geändert werden müssen",
|
||
"fp_ok": "Verschieben OK",
|
||
"fcp_ok": "Kopieren OK",
|
||
"fp_busy": "Verschiebe {0} Elemente...\n\n{1}",
|
||
"fcp_busy": "Kopiere {0} Elemente...\n\n{1}",
|
||
"fp_err": "Verschieben fehlgeschlagen:\n",
|
||
"fcp_err": "Kopieren fehlgeschlagen:\n",
|
||
"fp_confirm": "Diese {0} Elemente hierher verschieben?",
|
||
"fcp_confirm": "Diese {0} Elemente hierher kopieren?",
|
||
"fp_etab": 'Konnte die Zwischenablage nicht vom anderen Browsertab lesen',
|
||
"fp_name": "Lade Datei von deinem Gerät hoch. Gib ihr einen Namen:",
|
||
"fp_both_m": '<h6>Wähle, was eingefügt werden soll</h6><code>Eingabe</code> = {0} Dateien von «{1}» verschieben\n<code>ESC</code> = {2} Dateien von deinem Gerät hochladen',
|
||
"fcp_both_m": '<h6>Wähle, was eingefügt werden soll</h6><code>Eingabe</code> = {0} Dateien von «{1}» kopieren\n<code>ESC</code> = {2} Dateien von deinem Gerät hochladen',
|
||
"fp_both_b": '<a href="#" id="modal-ok">Verschieben</a><a href="#" id="modal-ng">Hochladen</a>',
|
||
"fcp_both_b": '<a href="#" id="modal-ok">Kopieren</a><a href="#" id="modal-ng">Hochladen</a>',
|
||
|
||
"mk_noname": "Tipp' mal vorher lieber einen Namen in das Textfeld links, bevor du das machst :p",
|
||
|
||
"tv_load": "Textdatei wird geladen:\n\n{0}\n\n{1}% ({2} von {3} MiB geladen)",
|
||
"tv_xe1": "Konnte Textdatei nicht laden:\n\nFehler ",
|
||
"tv_xe2": "404, Datei nicht gefunden",
|
||
"tv_lst": "Liste der Textdateien in",
|
||
"tvt_close": "Zu Ordneransicht zurück$NHotkey: M (oder Esc)\">❌ Schliessen",
|
||
"tvt_dl": "Diese Datei herunterladen$NHotkey: Y\">💾 Herunterladen",
|
||
"tvt_prev": "Vorheriges Dokument zeigen$NHotkey: i\">⬆ vorh.",
|
||
"tvt_next": "Nächstes Dokument zeigen$NHotkey: K\">⬇ nächst.",
|
||
"tvt_sel": "Wählt diese Datei aus ( zum Ausschneiden / Kopieren / Löschen / ... )$NHotkey: S\">ausw.",
|
||
"tvt_edit": "Datei im Texteditor zum Bearbeiten öffnen$NHotkey: E\">✏️ bearb.",
|
||
"tvt_tail": "Datei auf Veränderungen überwachen; Neue Zeilen werden in Echtzeit angezeigt\">📡 folgen",
|
||
"tvt_wrap": "Zeilenumbruch\">↵",
|
||
"tvt_atail": "Automatisch nach unten scrollen\">⚓",
|
||
"tvt_ctail": "Terminal-Farben dekodieren (ANSI Escape Codes)\">🌈",
|
||
"tvt_ntail": "Scrollback limitieren (Menge an Bytes an Text, die geladen bleiben sollen)",
|
||
|
||
"m3u_add1": "Song wurde zur M3U-Playlist hinzugefügt",
|
||
"m3u_addn": "{0} Songs zur M3U-Playlist hinzugefügt",
|
||
"m3u_clip": "M3U-Playlist in die Zwischenablage kopiert\n\nDu solltest eine neue Datei mit dem Namen something.m3u erstellen und die Playlist da rein kopieren; damit wird die Playlist abspielbar",
|
||
|
||
"gt_vau": "nur Ton abspielen, kein Video zeigen\">🎧",
|
||
"gt_msel": "Dateiauswahl aktivieren; STRG-klicke eine Datei zum überschreiben$N$N<em>wenn aktiv: Datei / Ordner doppelklicken zum Öffnen</em>$N$NHotkey: S\">multiselect",
|
||
"gt_crop": "Vorschaubilder mittig zuschneiden\">crop",
|
||
"gt_3x": "hochauflösende Vorschaubilder\">3x",
|
||
"gt_zoom": "zoom",
|
||
"gt_chop": "kürzen",
|
||
"gt_sort": "sortieren nach",
|
||
"gt_name": "Name",
|
||
"gt_sz": "Grösse",
|
||
"gt_ts": "Datum",
|
||
"gt_ext": "Typ",
|
||
"gt_c1": "Dateinamen mehr kürzen (weniger zeigen)",
|
||
"gt_c2": "Dateinamen weniger kürzen (mehr zeigen)",
|
||
|
||
"sm_w8": "Suche ...",
|
||
"sm_prev": "Die Suchresultate gehören zu einer vorherigen Suchanfrage:\n ",
|
||
"sl_close": "Suchresultate schliessen",
|
||
"sl_hits": "Zeige {0} Treffer",
|
||
"sl_moar": "Mehr laden",
|
||
|
||
"s_sz": "Grösse",
|
||
"s_dt": "Datum",
|
||
"s_rd": "Pfad",
|
||
"s_fn": "Name",
|
||
"s_ta": "Tags",
|
||
"s_ua": "up@",
|
||
"s_ad": "adv.",
|
||
"s_s1": "minimum MiB",
|
||
"s_s2": "maximum MiB",
|
||
"s_d1": "min. iso8601",
|
||
"s_d2": "max. iso8601",
|
||
"s_u1": "hochgeladen nach",
|
||
"s_u2": "und/oder vor",
|
||
"s_r1": "Pfad enthält (Leerzeichen-separiert)",
|
||
"s_f1": "Name enthält (negieren mit -nope)",
|
||
"s_t1": "Tags enthält (^=start, end=$)",
|
||
"s_a1": "spezifische Metadaten-Eigenschaften",
|
||
|
||
"md_eshow": "Kann nicht rendern ",
|
||
"md_off": "[📜<em>readme</em>] deaktiviert in [⚙️] -- Dokument versteckt",
|
||
|
||
"badreply": "Hab die Antwort vom Server nicht verstanden. (badreply)",
|
||
|
||
"xhr403": "403: Zugriff verweigert\n\nVersuche, F5 zu drücken. Vielleicht wurdest du abgemeldet.",
|
||
"xhr0": "Unbekannt (wahrschenlich Verbindung zum Server verloren oder der Server ist offline)",
|
||
"cf_ok": "Sorry dafür -- Der DD" + wah + "oS-Schutz hat angeschlagen.\n\nEs sollte in etwa 30 Sekunden weitergehen.\n\nFalls nichts passiert, drück' F5, um die Seite neuzuladen",
|
||
"tl_xe1": "Konnte Unterordner nicht auflisten:\n\nFehler ",
|
||
"tl_xe2": "404: Ordner nicht gefunden",
|
||
"fl_xe1": "Konnte Dateien in Ordner nicht auflisten:\n\nFehler ",
|
||
"fl_xe2": "404: Ordner nicht gefunden",
|
||
"fd_xe1": "Konnte Unterordner nicht erstellen:\n\nFehler ",
|
||
"fd_xe2": "404: Übergeordneter Ordner nicht gefunden",
|
||
"fsm_xe1": "Konnte Nachricht nicht senden:\n\nFehler ",
|
||
"fsm_xe2": "404: Übergeordneter Ordner nicht gefunden",
|
||
"fu_xe1": "Konnte unpost-Liste nicht laden:\n\nFehler ",
|
||
"fu_xe2": "404: Datei nicht gefunden??",
|
||
|
||
"fz_tar": "Unkomprimierte GNU TAR-Datei (Linux / Mac)",
|
||
"fz_pax": "Unkomprimierte pax-format TAR-Datei (etwas langsamer)", //m
|
||
"fz_targz": "GNU-TAR mit gzip Level 3 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR",
|
||
"fz_tarxz": "GNU-TAR mit xz level 1 Kompression$N$Nüblicherweise recht langsam,$Nbenutze stattdessen ein unkomprimiertes TAR",
|
||
"fz_zip8": "ZIP mit UTF8-Dateinamen (könnte kaputt gehen auf Windows 7 oder älter)",
|
||
"fz_zipd": "ZIP mit traditionellen CP437-Dateinamen, für richtig alte Software",
|
||
"fz_zipc": "CP437 mit CRC32 früh berechnet,$Nfür MS-DOS PKZIP v2.04g (Oktober 1993)$N(braucht länger zum Verarbeiten, bevor der Download starten kann)",
|
||
|
||
"un_m1": "Unten kannst du deine neusten Uploads löschen (oder Unvollständige abbrechen)",
|
||
"un_upd": "Neu laden",
|
||
"un_m4": "Oder die unten sichtbaren Dateien teilen:",
|
||
"un_ulist": "Anzeigen",
|
||
"un_ucopy": "Kopieren",
|
||
"un_flt": "Optionale Filter: URL muss enthalten",
|
||
"un_fclr": "Filter löschen",
|
||
"un_derr": 'unpost-delete fehlgeschlagen:\n',
|
||
"un_f5": 'Etwas ist kaputt gegangen, versuche die Seite neuzuladen (drücke dazu F5)',
|
||
"un_uf5": "Sorry, aber du musst die Seite neuladen (z.B. in dem du F5 oder STRG-R drückst) bevor zu diesen Upload abbrechen kannst",
|
||
"un_nou": '<b>Warnung:</b> Der Server ist grade zu beschäftigt, um unvollständige Uploads anzuzeigen; Drücke den "Neu laden"-Link in ein paar Sekunden',
|
||
"un_noc": '<b>Warnung:</b> unpost von vollständig hochgeladenen Dateien ist über die Serverkonfiguration gesperrt',
|
||
"un_max": "Zeige die ersten 2000 Dateien (benutze Filter, um die gewünschten Dateien zu finden)",
|
||
"un_avail": "{0} zuletzt hochgeladene Dateien können gelöscht werden<br />{1} Unvollständige können abgebrochen werden",
|
||
"un_m2": "Sortiert nach Upload-Zeitpunkt; neuste zuerst:",
|
||
"un_no1": "Hoppala! Es gibt keine ausreichend aktuellen Uploads.",
|
||
"un_no2": "Pech gehabt! Kein Upload, der zu dem Filter passen würde, ist neu genug",
|
||
"un_next": "Lösche die nächsten {0} Dateien",
|
||
"un_abrt": "Abbrechen",
|
||
"un_del": "Löschen",
|
||
"un_m3": "Deine letzten Uploads werden geladen ...",
|
||
"un_busy": "Lösche {0} Dateien ...",
|
||
"un_clip": "{0} Links in die Zwischenablage kopiert",
|
||
|
||
"u_https1": "für bessere Performance solltest du",
|
||
"u_https2": "auf HTTPS wechseln",
|
||
"u_https3": "",
|
||
"u_ancient": 'Dein Browser ist verdammt antik -- vielleicht solltest du <a href="#" onclick="goto(\'bup\')">stattdessen bup benutzen</a>',
|
||
"u_nowork": "Benötigt Firefox 53+ oder Chrome 57+ oder iOS 11+",
|
||
"tail_2old": "Benötigt Firefox 105+ oder Chrome 71+ oder iOS 14.5+",
|
||
"u_nodrop": 'Dein Browser ist zu alt für Drag-and-Drop Uploads',
|
||
"u_notdir": "Das ist kein Ordner!\n\nDein Browser ist zu alt,\nversuch stattdessen dragdrop",
|
||
"u_uri": "Um Bilder per Drag-and-Drop aus anderen Browserfenstern hochzuladen,\nlass' sie bitte über dem grossen Upload-Button fallen",
|
||
"u_enpot": 'Zu <a href="#">Potato UI</a> wechseln (kann Upload-Geschw. verbessern)',
|
||
"u_depot": 'Zu <a href="#">fancy UI</a> wechseln (kann Upload-Geschw. verschlechtern)',
|
||
"u_gotpot": 'Wechsle zu Potato UI für verbesserte Upload-Geschwindigkeit,\n\nwenn du anderer Meinung bist, kannst du gerne zurück wechseln',
|
||
"u_pott": "<p>Dateien: <b>{0}</b> fertig, <b>{1}</b> fehlgeschlagen, <b>{2}</b> in Bearbeitung, <b>{3}</b> ausstehend</p>",
|
||
"u_ever": "Dies ist der Basic Uploader; up2k benötigt mind. <br>Chrome 21 // Firefox 13 // Edge 12 // Opera 12 // Safari 5.1",
|
||
"u_su2k": 'Dies ist der Basic Uploader; <a href="#" id="u2yea">up2k</a> ist besser',
|
||
"u_uput": 'Für Geschwindigkeit optimieren (Checksum überspringen)',
|
||
"u_ewrite": 'Du hast kein Schreibzugriff auf diesen Ordner',
|
||
"u_eread": 'Du hast kein Lesezugriff auf diesen Ordner',
|
||
"u_enoi": 'file-search ist in der Serverkonfiguration nicht aktiviert',
|
||
"u_enoow": "Überschreiben wird hier nicht funktionieren; benötige Lösch-Berechtigung",
|
||
"u_badf": 'Diese {0} Dateien (von insgesammt {1}) wurden übersprungen, wahrscheinlich wegen Dateisystem-Berechtigungen:\n\n',
|
||
"u_blankf": 'Diese {0} Dateien (von insgesammt {1}) sind leer; trotzdem hochladen?\n\n',
|
||
"u_applef": 'Diese {0} Dateien (von insgesammt {1}) sind möglicherweise unerwünscht;\n<code>OK/Eingabe</code> drücken, um die folgenden Dateien zu überspringen.\nDrücke <code>Abbrechen/ESC</code> um sie NICHT zu überspringen und diese AUCH HOCHZULADEN:\n\n',
|
||
"u_just1": '\nFunktioniert vielleicht besser, wenn du nur eine Datei auswählst',
|
||
"u_ff_many": "Falls du <b>Linux / MacOS / Android</b> benutzt, <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>könnte</em> Firefox mit dieser Menge an Dateien crashen!</a>\nFalls das passiert, probier nochmal (oder benutz Chrome).",
|
||
"u_up_life": "Dieser Upload wird vom Server gelöscht\n{0} nachdem er abgeschlossen ist",
|
||
"u_asku": 'Diese {0} Dateien nach <code>{1}</code> hochladen',
|
||
"u_unpt": "Du kannst diesen Upload rückgängig machen mit dem 🧯 oben-links",
|
||
"u_bigtab": 'Versuche {0} Dateien anzuzeigen.\n\nDas könnte dein Browser crashen, bist du dir wirklich sicher?',
|
||
"u_scan": 'Scanne Dateien...',
|
||
"u_dirstuck": 'Ordner-Iterator blieb hängen beim Versuch, diese {0} Einträge zu lesen; überspringe:',
|
||
"u_etadone": 'Fertig ({0}, {1} Dateien)',
|
||
"u_etaprep": '(Upload wird vorbereitet)',
|
||
"u_hashdone": 'Hashing vollständig',
|
||
"u_hashing": 'Hash',
|
||
"u_hs": 'Wir schütteln uns die Hände ("handshaking")...',
|
||
"u_started": "Dateien werden hochgeladen; siehe [🚀]",
|
||
"u_dupdefer": "Duplikat; wird nach allen anderen Dateien verarbeitet",
|
||
"u_actx": "Klicke diesen Text um Performance-<br />Einbusen zu Vermeiden beim Wechsel auf andere Fenster/Tabs",
|
||
"u_fixed": "OK! Habs repariert 👍",
|
||
"u_cuerr": "failed to upload chunk {0} of {1};\nprobably harmless, continuing\n\nfile: {2}",
|
||
"u_cuerr2": "server rejected upload (chunk {0} of {1});\nwill retry later\n\nfile: {2}\n\nerror ",
|
||
"u_ehstmp": "versuche nochmal; siehe unten-rechts",
|
||
"u_ehsfin": "Der Server hat die Anfrage zum Abschluss des Uploads abgelehnt; versuche nochmal...",
|
||
"u_ehssrch": "Der Server hat die Anfrage zur Suche abgelehnt; versuche nochmal...",
|
||
"u_ehsinit": "Der Server hat die Anfrage zum Start des Uploads abgelehnt; versuche nochmal...",
|
||
"u_eneths": "Netzwerkfehler beim Upload-Handshake; versuche nochmal...",
|
||
"u_enethd": "Netzwerkfehler beim Testen der Existenz des Ziels; versuche nochmal...",
|
||
"u_cbusy": "Der Server mag uns grade nicht mehr nach einem Netzwerkglitch, warte einen Moment...",
|
||
"u_ehsdf": "Server hat kein Speicherplatz mehr!\n\nwerde es erneut versuchen, falls jemand\ngenug Platz schafft um fortzufahren",
|
||
"u_emtleak1": "scheint, als ob dein Browser ein Memory Leak hätte;\nbitte",
|
||
"u_emtleak2": ' <a href="{0}">wechsle auf HTTPS (empfohlen)</a> oder ',
|
||
"u_emtleak3": '',
|
||
"u_emtleakc": 'versuche folgendes:\n<ul><li>drücke <code>F5</code> um die Seite neu zu laden</li><li>deaktivere dann den <code>mt</code> Button in den <code>⚙️ Einstellungen</code></li><li>und versuche den Upload nochmal.</li></ul>Uploads werden etwas langsamer sein, aber man kann ja nicht alles haben.\nSorry für die Umstände !\n\nPS: Chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">hat ein Bugfix</a> dafür',
|
||
"u_emtleakf": 'versuche folgendes:\n<ul><li>drücke <code>F5</code> um die Seite neu zu laden</li><li>aktivere dann <code>🥔</code> (potato) im Upload UI<li>und versuche den Upload nochmal</li></ul>\nPS: Firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">hat hoffentlich irgendwann ein Bugfix</a>',
|
||
"u_s404": "nicht auf dem Server gefunden",
|
||
"u_expl": "erklären",
|
||
"u_maxconn": "die meisten Browser limitieren dies auf 6, aber Firefox lässt mehr zu unter <code>connections-per-server</code> in <code>about:config</code>",
|
||
"u_tu": '<p class="warn">WARNUNG: Turbo aktiviert, <span> Client könnte unvollständige Uploads verpassen und nicht wiederholen; siehe Turbo-Button Tooltip</span></p>',
|
||
"u_ts": '<p class="warn">WARNUNG: Turbo aktiviert, <span> Suchresultate können inkorrekt sein; siehe Turbo-Button Tooltip</span></p>',
|
||
"u_turbo_c": "Turbo deaktiviert in der Serverkonfiguration",
|
||
"u_turbo_g": "Turbo deaktiviert, da du keine Listen-Berechtigung\nauf diesem Volume hast",
|
||
"u_life_cfg": 'Autodelete nach <input id="lifem" p="60" /> min (or <input id="lifeh" p="3600" /> h)',
|
||
"u_life_est": 'Upload wird gelöscht <span id="lifew" tt="local time">---</span>',
|
||
"u_life_max": 'Dieser Ordner erzwingt eine\nmax Lebensdauer von {0}',
|
||
"u_unp_ok": 'unpost ist erlaubt für {0}',
|
||
"u_unp_ng": 'unpost wird NICHT erlaubt',
|
||
"ue_ro": 'Du hast nur Lese-Zugriff auf diesen Ordner\n\n',
|
||
"ue_nl": 'Du bist nicht angemeldet',
|
||
"ue_la": 'Du bist angemeldet als "{0}"',
|
||
"ue_sr": 'Du bist derzeit im Suchmodus\n\nWechsle zum Upload-Modus indem du auf die Lupe 🔎 klickst (neben dem grossen SUCHEN Button), und versuche den Upload nochmal.\n\nSorry',
|
||
"ue_ta": 'Versuche den Upload nochmal, sollte jetzt klappen',
|
||
"ue_ab": "Diese Datei wird gerade in einem anderen Ordner hochgeladen, dieser Upload muss zuerst abgeschlossen werden, bevor die Datei woanders hochgeladen werden kann.\n\nDu kannst den Upload abbrechen und vergessen mit dem 🧯 oben-links",
|
||
"ur_1uo": "OK: Datei erfolgreich hochgeladen",
|
||
"ur_auo": "OK: Alle {0} Dateien erfolgreich hochgeladen",
|
||
"ur_1so": "OK: Datei auf dem Server gefunden",
|
||
"ur_aso": "OK: Alle {0} Dateien auf dem Server gefunden",
|
||
"ur_1un": "Upload fehlgeschlagen, sorry",
|
||
"ur_aun": "Alle {0} Uploads fehlgeschlagen, sorry",
|
||
"ur_1sn": "Datei wurde NICHT auf dem Server gefunden",
|
||
"ur_asn": "Die {0} Dateien wurden NICHT auf dem Server gefunden",
|
||
"ur_um": "Fertig;\n{0} Uploads OK,\n{1} Uploads fehlgeschlagen, sorry",
|
||
"ur_sm": "Fertig;\n{0} Uploads gefunden auf dem Server,\n{1} Dateien NICHT gefunden auf dem Server",
|
||
|
||
"lang_set": "Neuladen um Änderungen anzuwenden?",
|
||
},
|
||
"rus": {
|
||
"tt": "Русский",
|
||
|
||
"cols": {
|
||
"c": "кнопки действий",
|
||
"dur": "длительность",
|
||
"q": "качество / битрейт",
|
||
"Ac": "аудио кодек",
|
||
"Vc": "видео кодек",
|
||
"Fmt": "формат / контейнер",
|
||
"Ahash": "контрольная сумма аудио",
|
||
"Vhash": "контрольная сумма видео",
|
||
"Res": "разрешение",
|
||
"T": "тип файла",
|
||
"aq": "качество аудио / битрейт",
|
||
"vq": "качество видео / битрейт",
|
||
"pixfmt": "сабсемплинг / пиксельный формат",
|
||
"resw": "горизонтальное разрешение",
|
||
"resh": "вертикальное разрешение",
|
||
"chs": "аудио каналы",
|
||
"hz": "частота дискретизации"
|
||
},
|
||
|
||
"hks": [
|
||
[
|
||
"разное",
|
||
["ESC", "закрыть всякие штуки"],
|
||
|
||
"файловый менеджер",
|
||
["G", "переключиться между списком / плиткой"],
|
||
["T", "переключиться между миниатюрами / иконками"],
|
||
["⇧ A/D", "размер миниатюры"],
|
||
["ctrl-K", "удалить выделенное"],
|
||
["ctrl-X", "вырезать выделенное в буфер"],
|
||
["ctrl-C", "копировать выделенное в буфер"],
|
||
["ctrl-V", "вставить (переместить/копировать) сюда"],
|
||
["Y", "скачать выделенное"],
|
||
["F2", "переименовать выделенное"],
|
||
|
||
"выделение файлов",
|
||
["пробел", "выделить/снять выделение с текущего файла"],
|
||
["↑/↓", "двигать курсор"],
|
||
["ctrl ↑/↓", "двигать курсор и скроллить"],
|
||
["⇧ ↑/↓", "выделить предыдущий/следующий файл"],
|
||
["ctrl-A", "выделить все файлы и папки"],
|
||
], [
|
||
"навигация",
|
||
["B", "показать/скрыть панель навигации"],
|
||
["I/K", "предыдущая/следующая папка"],
|
||
["M", "перейти на уровень выше (или свернуть текущую папку)"],
|
||
["V", "переключиться между папками / текстовыми файлами в панели навигации"],
|
||
["A/D", "уменьшить/увеличить панель навигации"],
|
||
], [
|
||
"аудиоплеер",
|
||
["J/L", "предыдущий/следующий трек"],
|
||
["U/O", "перемотать на 10 секунд назад/вперёд"],
|
||
["0..9", "перемотать на 0%..90%"],
|
||
["P", "играть/пауза (или подгрузить трек)"],
|
||
["S", "выделить текущий трек"],
|
||
["Y", "скачать трек"],
|
||
], [
|
||
"просмотрщик изображений",
|
||
["J/L, ←/→", "предыдущее/следующее изображение"],
|
||
["Home/End", "первое/последнее изображение"],
|
||
["F", "развернуть на полный экран"],
|
||
["R", "повернуть по часовой стрелке"],
|
||
["⇧ R", "повернуть против часовой стрелки"],
|
||
["S", "выделить изображение"],
|
||
["Y", "скачать изображение"],
|
||
], [
|
||
"видеоплеер",
|
||
["U/O", "перемотать на 10 секунд назад/вперёд"],
|
||
["P/K/Space", "играть/пауза"],
|
||
["C", "продолжить проигрывать следующее"],
|
||
["V", "повтор"],
|
||
["M", "выключить звук"],
|
||
["[ and ]", "задать интервал повтора"],
|
||
], [
|
||
"просмотрщик текстовых файлов",
|
||
["I/K", "предыдущий/следующий файл"],
|
||
["M", "закрыть файл"],
|
||
["E", "отредактировать файл"],
|
||
["S", "выделить файл"],
|
||
]
|
||
],
|
||
|
||
"m_ok": "OK",
|
||
"m_ng": "Отмена",
|
||
|
||
"enable": "Включить",
|
||
"danger": "ВНИМАНИЕ",
|
||
"clipped": "скопировано в буфер обмена",
|
||
|
||
"ht_s1": "секунда",
|
||
"ht_s2": "секунды",
|
||
"ht_m1": "минута",
|
||
"ht_m2": "минуты",
|
||
"ht_h1": "час",
|
||
"ht_h2": "часы",
|
||
"ht_d1": "день",
|
||
"ht_d2": "дни",
|
||
"ht_and": " и ",
|
||
|
||
"goh": "панель управления",
|
||
"gop": 'предыдущая папка">пред',
|
||
"gou": 'родительская папка">вверх',
|
||
"gon": 'следующая папка">след',
|
||
"logout": "Выйти ",
|
||
"access": " доступ",
|
||
"ot_close": "закрыть подменю",
|
||
"ot_search": "искать файлы по атрибутам, пути / имени, музыкальным тегам или любой другой комбинации из следующих конструкций$N$N<code>foo bar</code> = обязано содержать «foo» И «bar»,$N<code>foo -bar</code> = обязано содержать «foo», но не «bar»,$N<code>^yana .opus$</code> = начинается с «yana» и имеет расширение «opus»$N<code>"try unite"</code> = содержит именно «try unite»$N$Nформат времени задаётся по стандарту iso-8601, например$N<code>2009-12-31</code> или <code>2020-09-12 23:30:00</code>",
|
||
"ot_unpost": "unpost: удалить ваши недавние загрузки и отменить незавершённые",
|
||
"ot_bup": "bup: легковесный загрузчик файлов, поддерживает даже netscape 4.0",
|
||
"ot_mkdir": "mkdir: создать новую папку",
|
||
"ot_md": "new-md: создать новый markdown-документ",
|
||
"ot_msg": "msg: отправить сообщение в лог сервера",
|
||
"ot_mp": "настройка медиаплеера",
|
||
"ot_cfg": "остальные настройки",
|
||
"ot_u2i": 'up2k: загрузить файлы (если имеется доступ к записи) или переключиться в режим поиска$N$Nзагрузки являются возобновляемыми и многопоточными, а даты изменения файлов сохраняются в процессе, но этот метод использует больше ресурсов процессора, чем [🎈] (легковесный загрузчик)<br /><br />во время загрузки эта иконка превращается в индикатор!',
|
||
"ot_u2w": 'up2k: загрузить файлы с поддержкой возобновления (закиньте те же файлы после перезапуска браузера)$N$Nподдерживается многопоточность, даты изменения файлов сохраняются в процессе, но этот метод использует больше ресурсов процессора, чем [🎈] (легковесный загрузчик)<br /><br />во время загрузки эта иконка превращается в индикатор!',
|
||
"ot_noie": 'Пожалуйста, используйте Chrome / Firefox / Edge',
|
||
|
||
"ab_mkdir": "создать папку",
|
||
"ab_mkdoc": "создать markdown-документ",
|
||
"ab_msg": "отправить сообщение в лог сервера",
|
||
|
||
"ay_path": "перейти к папкам",
|
||
"ay_files": "перейти к файлам",
|
||
|
||
"wt_ren": "переименовать выделенные файлы$NГорячая клавиша: F2",
|
||
"wt_del": "удалить выделенные файлы$NГорячая клавиша: ctrl-K",
|
||
"wt_cut": "вырезать выделенные файлы <small>(затем вставить куда-то в другое место)</small>$NГорячая клавиша: ctrl-X",
|
||
"wt_cpy": "копировать выделенные файлы в буфер$N(чтобы вставить их куда-то ещё)$NГорячая клавиша: ctrl-C",
|
||
"wt_pst": "вставить ранее вырезанный / скопированный файл$NГорячая клавиша: ctrl-V",
|
||
"wt_selall": "выделить все файлы$NГорячая клавиша: ctrl-A (когда выделен хотя бы один файл)",
|
||
"wt_selinv": "инвертировать выделение",
|
||
"wt_zip1": "скачать эту папку как архив",
|
||
"wt_selzip": "скачать выделенные файлы как архив",
|
||
"wt_seldl": "скачать выделенные файлы по-отдельности$NГорячая клавиша: Y",
|
||
"wt_npirc": "копировать информацию о треке в формате irc",
|
||
"wt_nptxt": "копировать информацию о треке обычным текстом",
|
||
"wt_m3ua": "добавить в плейлист m3u (нажмите <code>📻коп.</code> в конце)",
|
||
"wt_m3uc": "копировать плейлист m3u в буфер обмена",
|
||
"wt_grid": "переключить между сеткой / списком$NГорячая клавиша: G",
|
||
"wt_prev": "предыдущий трек$NГорячая клавиша: J",
|
||
"wt_play": "играть / пауза$NГорячая клавиша: P",
|
||
"wt_next": "следующий трек$NГорячая клавиша: L",
|
||
|
||
"ul_par": "параллельные загрузки:",
|
||
"ut_rand": "случайные имена файлов",
|
||
"ut_u2ts": "копировать время последнего изменения$Nиз вашей файловой системы на сервер\">📅",
|
||
"ut_ow": "перезаписывать существующие файлы на сервере?$N🛡️: нет (для повторяющихся файлов будут создаваться новые имена)$N🕒: перезаписать файлы с датой изменения старее, чем у загружаемых$N♻️: всегда перезаписывать (если файлы различаются по содержанию)",
|
||
"ut_mt": "продолжать хешировать другие файлы во время загрузки$N$Nесть смысл отключить при медленном диске или процессоре",
|
||
"ut_ask": 'требовать подтверждения перед началом загрузки">💭',
|
||
"ut_pot": "улучшить скорость загрузки на слабых устройства$Nс помощью упрощения интерфейса",
|
||
"ut_srch": "не загружать, а проверять, существуют ли данные файлы $N на сервере (проверка всех доступных вам папок)",
|
||
"ut_par": "при 0 загрузка встанет на паузу$N$Nследует повысить, если ваше подключение медленное$N$Nоставьте 1, если используется локальная сеть или диск сервера медленный",
|
||
"ul_btn": "отпустите файлы / папки<br>здесь (или нажмите)",
|
||
"ul_btnu": "З А Г Р У З И Т Ь",
|
||
"ul_btns": "И С К А Т Ь",
|
||
|
||
"ul_hash": "хеш",
|
||
"ul_send": "отправка",
|
||
"ul_done": "готово",
|
||
"ul_idle1": "нет загрузок в очереди",
|
||
"ut_etah": "средняя скорость <em>хеширования</em> и примерное время до завершения",
|
||
"ut_etau": "средняя скорость <em>загрузки</em> и примерное время до завершения",
|
||
"ut_etat": "средняя <em>общая</em> скорость и примерное время до завершения",
|
||
|
||
"uct_ok": "успешно завершены",
|
||
"uct_ng": "ошибки / отказы / не найдены",
|
||
"uct_done": "готово (ok и ng вместе)",
|
||
"uct_bz": "хешируются или загружаются",
|
||
"uct_q": "в очереди",
|
||
|
||
"utl_name": "имя файла",
|
||
"utl_ulist": "список",
|
||
"utl_ucopy": "копировать",
|
||
"utl_links": "ссылки",
|
||
"utl_stat": "статус",
|
||
"utl_prog": "прогресс",
|
||
|
||
// keep short:
|
||
"utl_404": "404",
|
||
"utl_err": "ОШИБКА",
|
||
"utl_oserr": "ошибка ОС",
|
||
"utl_found": "найдено",
|
||
"utl_defer": "отложить",
|
||
"utl_yolo": "турбо",
|
||
"utl_done": "готово",
|
||
|
||
"ul_flagblk": "файлы были добавлены в очередь</b>,<br>однако в другой вкладке уже есть активная загрузка через up2k,<br>поэтому ожидаем её завершения",
|
||
"ul_btnlk": "настройки сервера запрещают изменение состояния этой опции",
|
||
|
||
"udt_up": "Загрузить",
|
||
"udt_srch": "Поиск",
|
||
"udt_drop": "отпустите здесь",
|
||
|
||
"u_nav_m": '<h6>лады, что там у вас?</h6><code>Enter</code> = Файлы (один или больше)\n<code>ESC</code> = Одна папка (с учётом подпапок)',
|
||
"u_nav_b": '<a href="#" id="modal-ok">Файлы</a><a href="#" id="modal-ng">Одна папка</a>',
|
||
|
||
"cl_opts": "переключатели",
|
||
"cl_themes": "тема",
|
||
"cl_langs": "язык",
|
||
"cl_ziptype": "архивация папок",
|
||
"cl_uopts": "опции up2k",
|
||
"cl_favico": "иконка",
|
||
"cl_bigdir": "бол. папки",
|
||
"cl_hsort": "#сорт.",
|
||
"cl_keytype": "схема горячих клавиш",
|
||
"cl_hiddenc": "скрытые столбцы",
|
||
"cl_hidec": "скрыть",
|
||
"cl_reset": "сбросить",
|
||
"cl_hpick": "нажмите на заголовки столбцов, чтобы скрыть их в таблице ниже",
|
||
"cl_hcancel": "скрытие столбца отменено",
|
||
|
||
"ct_grid": '田 сетка',
|
||
"ct_ttips": '◔ ◡ ◔">ℹ️ подсказки',
|
||
"ct_thumb": 'переключение между иконками и миниатюрами в режиме сетки$NГорячая клавиша: T">🖼️ миниат.',
|
||
"ct_csel": 'держите CTRL или SHIFT для выделения файлов в режиме сетки">выбор',
|
||
"ct_ihop": 'показывать последний открытый файл после закрытия просмотрщика изображений">g⮯',
|
||
"ct_dots": 'показывать скрытые файлы (если есть доступ)">скрыт.',
|
||
"ct_qdel": 'спрашивать подтверждение только один раз перед удалением файлов">быстр. удал.',
|
||
"ct_dir1st": 'разместить папки над файлами">📁 сверху',
|
||
"ct_nsort": 'сортировка по числам$N(например, файл с <code>2</code> в начале названия идёт перед <code>11</code>)">нат. сорт.',
|
||
"ct_readme": 'показывать содержимое README.md в описании папки">📜 ридми',
|
||
"ct_idxh": 'показывать страницу index.html в текущей папке вместо интерфейса">htm',
|
||
"ct_sbars": 'показывать полосы прокрутки">⟊',
|
||
|
||
"cut_umod": "если файл уже существует на сервере, обновить время последнего изменения на сервере в соответствии с локальным файлом (требуются права write+delete)\">перенос 📅",
|
||
|
||
"cut_turbo": "используйте эту функцию С ОСТОРОЖНОСТЬЮ:$N$Nпригодится в случае, если была прервана загрузка большого количества файлов, и вы хотите возобновить её как можно быстрее$N$Nпроверка файлов по хешу заменяется на простой алгоритм: <em>"если размер файла отличается - тогда загрузить"</em>, но содержание файлов не сравнивается$N$Nследует отключить эту функцию после окончания загрузки, а затем "загрузить" те же файлы заново, чтобы валидировать их\">турбо",
|
||
|
||
"cut_datechk": "работает только при включённой кнопке "турбо"$N$Nчуть-чуть повышает надёжность турбо-загрузок с помощью сверки дат изменений между файлами на сервере и вашими$N$N<em>в теории</em> достаточно для проверки большинства незавершённых / повреждённых загрузок, но не является альтернативой валидации файлов после турбо-загрузки\">провер. дат",
|
||
|
||
"cut_u2sz": "размер (в МиБ) каждой загружаемой части; большие значения показывают лучшие результаты для дальних соединений. Если подключение нестабильное, попробуйте значение пониже",
|
||
|
||
"cut_flag": "разрешить одновременную загрузку только из одной вкладки за раз $N -- обязательно включить эту опцию в остальных вкладках $N -- работает только в пределах одного домена",
|
||
|
||
"cut_az": "загружать файлы в алфавитном порядке вместо "от меньшего к большему"$N$Nэто позволит проще отследить проблемы во время загрузки, но скорость слегка ниже на очень быстрых соединениях (например, в локальной сети)",
|
||
|
||
"cut_nag": "системное уведомление по завершении загрузки$N(только при неактивной вкладке браузера)",
|
||
"cut_sfx": "звуковое уведомление по завершении загрузки$N(только при неактивной вкладке браузера)",
|
||
|
||
"cut_mt": "использовать многопоточность для ускорения хеширования$N$Nиспользует Web Worker'ы и требует больше памяти (до 512 МиБ)$N$Nускоряет https на 30%, http - в 4,5 раз\">мп",
|
||
|
||
"cut_wasm": "использовать модуль WASM вместо встроенной в браузер функции хеширования; ускоряет процесс в браузерах на основе Chromium, но увеличивает нагрузку на процессор. Старые версии Chrome содержат баги, которые заполняют всю оперативную память и крашат браузер, когда включена эта опция\">wasm",
|
||
|
||
"cft_text": "текст для иконки (очистите поле и перезагрузите страницу для применения)",
|
||
"cft_fg": "цвет текста",
|
||
"cft_bg": "цвет фона",
|
||
|
||
"cdt_lim": "максимальное количество файлов для показа в папке",
|
||
"cdt_ask": "внизу страницы спрашивать о действии вместо автоматической загрузки следующих файлов",
|
||
"cdt_hsort": "сколько правил сортировки (<code>,sorthref</code>) включать в адрес страницы. Если значение равно 0, по нажатии на ссылки будут игнорироваться правила, включённые в них",
|
||
|
||
"tt_entree": "показать панель навигации$NГорячая клавиша: B",
|
||
"tt_detree": "скрыть панель навигации$NГорячая клавиша: B",
|
||
"tt_visdir": "прокрутить до выделенной папки",
|
||
"tt_ftree": "переключить между иерархией и списком текстовых файлов$NГорячая клавиша: V",
|
||
"tt_pdock": "закрепить родительские папки сверху панели",
|
||
"tt_dynt": "автоматическое расширение панели",
|
||
"tt_wrap": "перенос слов",
|
||
"tt_hover": "раскрывать обрезанные строки при наведении$N( ломает скроллинг, если $N курсор не в пустоте слева )",
|
||
|
||
"ml_pmode": "в конце папки...",
|
||
"ml_btns": "команды",
|
||
"ml_tcode": "транскодировать",
|
||
"ml_tcode2": "транскод. в",
|
||
"ml_tint": "затемн.",
|
||
"ml_eq": "эквалайзер",
|
||
"ml_drc": "компрессор",
|
||
|
||
"mt_loop": "повторять один трек\">🔁",
|
||
"mt_one": "остановить после этого трека\">1️⃣",
|
||
"mt_shuf": "перемешать треки во всех папках\">🔀",
|
||
"mt_aplay": "автоматически играть треки по нажатии на ссылки с их ID$N$Nпри отключении адрес сайта также перестанет обновляться в соответствии с текущим треком\">a▶",
|
||
"mt_preload": "подгружать следующий трек перед концом текущего для бесшовного переключения\">предзагр.",
|
||
"mt_prescan": "переходить в следующую папку перед окончанием последнего трека$Nне даёт браузеру прервать следующий плейлист\">нав.",
|
||
"mt_fullpre": "подгружать следующий трек целиком;$N✅ полезно при <b>нестабильном</b> подключении,$N❌ при медленной скорости лучше <b>выключить</b>\">цел.",
|
||
"mt_fau": "для телефонов: начинать следующий трек сразу, даже если он не успел подгрузиться целиком (может сломать отображение тегов)\">☕️",
|
||
"mt_waves": "визуализация:$Nпоказывать волну громкости на полосе воспроизведения\">~",
|
||
"mt_npclip": "показать кнопки копирования для текущего трека\">/np",
|
||
"mt_m3u_c": "показать кнопки копирования для выделенных треков$Nв формате плейлистов m3u8\">📻",
|
||
"mt_octl": "интеграция с ОС (поддержка медиа-клавиш и музыкальных виджетов)\">интегр.",
|
||
"mt_oseek": "позволить перематывать треки через системные виджеты$N$Nвнимание: на некоторых устройствах (iPhone)$Nэто заменит кнопку следующего трека\">перемотка",
|
||
"mt_oscv": "показывать картинки альбомов в виджетах\">арт",
|
||
"mt_follow": "держать фокус на играющем треке\">🎯",
|
||
"mt_compact": "компактный плеер\">⟎",
|
||
"mt_uncache": "очистить кеш (если браузер кешировал повреждённый$Nтрек и отказывается его запускать)\">уд. кеш",
|
||
"mt_mloop": "повторять треки в папке\">🔁 цикл",
|
||
"mt_mnext": "загрузить следующую папку и продолжить в ней\">📂 след.",
|
||
"mt_mstop": "приостановить воспроизведение\">⏸ стоп",
|
||
"mt_cflac": "конвертировать flac / wav в opus\">flac",
|
||
"mt_caac": "конвертировать aac / m4a в opus\">aac",
|
||
"mt_coth": "конвертировать всё остальное (кроме mp3) в opus\">др.",
|
||
"mt_c2opus": "лучший вариант для компьютеров и устройств на Android\">opus",
|
||
"mt_c2owa": "opus-weba, для iOS 17.5 и выше\">owa",
|
||
"mt_c2caf": "opus-caf, для iOS 11-17\">caf",
|
||
"mt_c2mp3": "для очень старых устройств\">mp3",
|
||
"mt_c2ok": "хороший выбор",
|
||
"mt_c2nd": "это не рекомендованный вариант формата для вашего устройства, но сойдёт",
|
||
"mt_c2ng": "не похоже, что ваше устройство поддерживает этот формат, но давайте попробуем и узнаем наверняка",
|
||
"mt_xowa": "в iOS есть баги, препятствующие фоновому воспроизведению этого формата. Пожалуйста, используйте caf или mp3",
|
||
"mt_tint": "непрозрачность фона (0-100) на полосе воспроизведения$N$Nделает буферизацию менее отвлекающей",
|
||
"mt_eq": "включить эквалайзер$N$Nboost <code>0</code> = стандартная громкость$N$Nwidth <code>1 </code> = обычное стерео$Nwidth <code>0.5</code> = микширование левого и правого каналов на 50%$Nwidth <code>0 </code> = моно$N$Nboost <code>-0.8</code> и width <code>10</code> = удаление голоса :^)$N$Nвключённый эквалайзер полностью убирает задержку между треками, поэтому следует его включить со всеми значениями на 0 (кроме width = 1), если вам нужно бесшовное воспроизведение",
|
||
"mt_drc": "включить компрессор; также включит эквалайзер для баланса вселенной, так что выставьте всё на 0 кроме width, если он вам не нужен$N$Nпонижает громкость при волне выше значения dB в tresh; каждый dB в ratio равен одному dB на выходе, так что стандартные значения tresh = -24 и ratio = 12 сделают так, что звук никогда не будет громче -22 dB. При таком раскладе можно поставить boost в эквалайзере на 0.8 или даже 1.8 при значении atk = 0 и огромном rls вроде 90 (работает только в Firefox, rls не может быть выше 1 в других браузерах)$N$N(загляните в википедию, там всё объяснено подробнее)",
|
||
|
||
"mb_play": "играть",
|
||
"mm_hashplay": "воспроизвести этот музыкальный файл?",
|
||
"mm_m3u": "нажмите <code>Enter/OK</code>, чтобы играть\nнажмите <code>ESC/Отмена</code>, чтобы редактировать",
|
||
"mp_breq": "требуется Firefox 82+, Chrome 73+ или iOS 15+",
|
||
"mm_bload": "загружаю...",
|
||
"mm_bconv": "конвертирую в {0}, подождите...",
|
||
"mm_opusen": "ваш браузер не может воспроизводить файлы aac / m4a;\nвключено транскодирование в opus",
|
||
"mm_playerr": "ошибка воспроизведения: ",
|
||
"mm_eabrt": "Попытка воспроизведения была отменена",
|
||
"mm_enet": "Ваше подключение нестабильно",
|
||
"mm_edec": "Этот файл, возможно, повреждён??",
|
||
"mm_esupp": "Ваш браузер не распознаёт этот аудио-формат",
|
||
"mm_eunk": "Неопознанная ошибка",
|
||
"mm_e404": "Не удалось воспроизвести аудио; ошибка 404: Файл не найден.",
|
||
"mm_e403": "Не удалось воспроизвести аудио; ошибка 403: Доступ запрещён.\n\nПопробуйте перезагрузить страницу, возможно, ваша сессия истекла",
|
||
"mm_e500": "Не удалось воспроизвести аудио; ошибка 500: Проверьте логи сервера.",
|
||
"mm_e5xx": "Не удалось воспроизвести аудио; ошибка сервера ",
|
||
"mm_nof": "больше аудио-файлов не найдено",
|
||
"mm_prescan": "Поиск музыки для воспроизведения дальше...",
|
||
"mm_scank": "Найден следующий трек:",
|
||
"mm_uncache": "кеш очищен; все треки будут загружены заново при воспроизведении",
|
||
"mm_hnf": "это трек больше не существует",
|
||
|
||
"im_hnf": "это изображение больше не существует",
|
||
|
||
"f_empty": 'эта папка пуста',
|
||
"f_chide": 'это скроет столбец «{0}»\n\nвы можете показать скрытые столбцы в настройках',
|
||
"f_bigtxt": "объём данного файла - {0} МиБ. точно открыть как текст?",
|
||
"f_bigtxt2": "просмотреть только конец файла? это также включит обновление в реальном времени, показывая новые строки сразу после их добавления",
|
||
"fbd_more": '<div id="blazy">показано <code>{0}</code> из <code>{1}</code> файлов; <a href="#" id="bd_more">показать {2}</a> или <a href="#" id="bd_all">показать всё</a></div>',
|
||
"fbd_all": '<div id="blazy">показано <code>{0}</code> из <code>{1}</code> файлов; <a href="#" id="bd_all">показать всё</a></div>',
|
||
"f_anota": "только {0} из {1} файлов было выделено;\nчтобы выделить всё папку, отмотайте до низа",
|
||
|
||
"f_dls": 'ссылки на файлы в данной папке были\nзаменены ссылками на скачивание',
|
||
|
||
"f_partial": "Чтобы безопасно скачать файл, который в текущий момент загружается, нажмите на файл с таким же названием, но без расширения <code>.PARTIAL</code>. Пожалуйста, нажмите Отмена или ESC, чтобы сделать это.\n\nПри нажатии OK / Enter, вы скачаете этот временный файл, который с огромной вероятностью содержит лишь неполные данные.",
|
||
|
||
"ft_paste": "вставить {0} файлов$NГорячая клавиша: ctrl-V",
|
||
"fr_eperm": 'не удалось переименовать:\nу вас нет разрешения “move” в этой папке',
|
||
"fd_eperm": 'не удалось удалить:\nу вас нет разрешения “delete” в этой папке',
|
||
"fc_eperm": 'не удалось вырезать:\nу вас нет разрешения “move” в этой папке',
|
||
"fp_eperm": 'не удалось вставить:\nу вас нет разрешения “write” в этой папке',
|
||
"fr_emore": "выделите хотя бы один файл, чтобы переименовать",
|
||
"fd_emore": "выделите хотя бы один файл, чтобы удалить",
|
||
"fc_emore": "выделите хотя бы один файл, чтобы вырезать",
|
||
"fcp_emore": "выделите хотя бы один файл, чтобы скопировать в буфер",
|
||
|
||
"fs_sc": "поделиться текущей папкой",
|
||
"fs_ss": "поделиться выделенными файлами",
|
||
"fs_just1d": "вы не можете выбрать больше одной папки\nили смешивать файлы с папками при выделении",
|
||
"fs_abrt": "❌ отменить",
|
||
"fs_rand": "🎲 случ. имя",
|
||
"fs_go": "✅ создать доступ",
|
||
"fs_name": "имя",
|
||
"fs_src": "путь",
|
||
"fs_pwd": "пароль",
|
||
"fs_exp": "срок",
|
||
"fs_tmin": "мин",
|
||
"fs_thrs": "часов",
|
||
"fs_tdays": "дней",
|
||
"fs_never": "вечно",
|
||
"fs_pname": "имя ссылки; генерируется случайно если не указано",
|
||
"fs_tsrc": "путь к файлу или папке, которыми нужно поделиться",
|
||
"fs_ppwd": "пароль (необязательно)",
|
||
"fs_w8": "создаю доступ...",
|
||
"fs_ok": "нажмите <code>Enter/OK</code>, чтобы скопировать\nнажмите <code>ESC/Cancel</code>, чтобы закрыть",
|
||
|
||
"frt_dec": "может исправить некоторые случаи с некорректными именами файлов\">декодировать url",
|
||
"frt_rst": "сбросить изменённые имена обратно к оригинальным\">↺ сброс",
|
||
"frt_abrt": "отменить операцию и закрыть это окно\">❌ отмена",
|
||
"frb_apply": "ПЕРЕИМЕНОВАТЬ",
|
||
"fr_adv": "переименование массовое / метаданных / по шаблону\">эксперт",
|
||
"fr_case": "чувствительный к регистру regex\">РеГиСтР",
|
||
"fr_win": "совместимые с windows имена; заменяет <code><>:"\\|?*</code> японскими полноширинными символами\">win",
|
||
"fr_slash": "заменяет <code>/</code> символом, который не создаёт новые папки\">без /",
|
||
"fr_re": "поиск по шаблону regex, применяемый к оригинальным именам; группы захвата могут применяться в поле форматирования с помощью <code>(1)</code> и <code>(2)</code> и так далее",
|
||
"fr_fmt": "вдохновлено foobar2000:$N<code>(title)</code> заменяется названием трека,$N<code>[(artist) - ](title)</code> пропускает [эту] часть, если композитор не указан$N<code>$lpad((tn),2,0)</code> добавляет ведущие нули до двух цифр",
|
||
"fr_pdel": "удалить",
|
||
"fr_pnew": "сохранить как",
|
||
"fr_pname": "предоставьте название для нового шаблона",
|
||
"fr_aborted": "прервано",
|
||
"fr_lold": "старое имя",
|
||
"fr_lnew": "новое имя",
|
||
"fr_tags": "теги для выделенных файлов (не редактируется, это для инструкции):",
|
||
"fr_busy": "переименовываю {0} файлов...\n\n{1}",
|
||
"fr_efail": "ошибка переименования:\n",
|
||
"fr_nchg": "{0} новых имён были модифицированы для соответствия опциям <code>win</code> и/или <code>без /</code>\n\nХотите использовать эти имена?",
|
||
|
||
"fd_ok": "успешно удалено",
|
||
"fd_err": "ошибка удаления:\n",
|
||
"fd_none": "ничего не удалено; возможно, не позволяет конфигурация сервера (xbd)?",
|
||
"fd_busy": "удалено {0} файлов...\n\n{1}",
|
||
"fd_warn1": "УДАЛИТЬ эти {0} файлов?",
|
||
"fd_warn2": "<b>Внимание!</b> Это необратимый процесс. Удалить?",
|
||
|
||
"fc_ok": "вырезано {0} файлов",
|
||
"fc_warn": 'вырезано {0} файлов\n\nно только <b>эта</b> вкладка браузера может их вставить\n(поскольку выделение оказалось настолько огромным)',
|
||
|
||
"fcc_ok": "скопировано {0} файлов в буфер",
|
||
"fcc_warn": 'скопировано {0} файлов в буфер\n\nно только <b>эта</b> вкладка браузера может их вставить\n(поскольку выделение оказалось настолько огромным)',
|
||
|
||
"fp_apply": "использовать эти имена",
|
||
"fp_ecut": "сначала вырезать или скопировать только некоторые файлы / папки\n\nучтите: вы можете вырезать / вставлять файлы между вкладками",
|
||
"fp_ename": "{0} файлов невозможно перенести сюда, потому что их имена уже заняты. Введите имена ниже, чтобы продолжить, или оставьте поля пустыми, чтобы пропустить:",
|
||
"fcp_ename": "{0} файлов невозможно скопировать сюда, потому что их имена уже заняты. Введите имена ниже, чтобы продолжить, или оставьте поля пустыми, чтобы пропустить:",
|
||
"fp_emore": "есть ещё коллизии имён, которые требуется исправить",
|
||
"fp_ok": "успешно перенесено",
|
||
"fcp_ok": "успешно скопировано",
|
||
"fp_busy": "перемещаю {0} файлов...\n\n{1}",
|
||
"fcp_busy": "копирую {0} файлов...\n\n{1}",
|
||
"fp_err": "ошибка перемещения:\n",
|
||
"fcp_err": "ошибка копирования:\n",
|
||
"fp_confirm": "переместить эти {0} файлов сюда?",
|
||
"fcp_confirm": "скопировать эти {0} файлов сюда?",
|
||
"fp_etab": 'ошибка чтения буфера обмена из другой вкладки браузера',
|
||
"fp_name": "загружаю файл с вашего устройства. Назовите его:",
|
||
"fp_both_m": '<h6>выберите, что вставить</h6><code>Enter</code> = Перенести {0} файлов из «{1}»\n<code>ESC</code> = Загрузить {2} файлов с вашего устройства',
|
||
"fcp_both_m": '<h6>выберите, что вставить</h6><code>Enter</code> = Скопировать {0} файлов из «{1}»\n<code>ESC</code> = Загрузить {2} файлов с вашего устройства',
|
||
"fp_both_b": '<a href="#" id="modal-ok">Переместить</a><a href="#" id="modal-ng">Загрузить</a>',
|
||
"fcp_both_b": '<a href="#" id="modal-ok">Скопировать</a><a href="#" id="modal-ng">Загрузить</a>',
|
||
|
||
"mk_noname": "введите имя в текстовое поле слева перед тем, как это делать :p",
|
||
|
||
"tv_load": "Загружаю текстовый документ:\n\n{0}\n\n{1}% ({2} из {3} МиБ загружено)",
|
||
"tv_xe1": "не удалось загрузить текстовый файл:\n\nошибка ",
|
||
"tv_xe2": "404, файл не найден",
|
||
"tv_lst": "список текстовых файлов в",
|
||
"tvt_close": "вернуться в обзор папки$NГорячая клавиша: M (или Esc)\">❌ закрыть",
|
||
"tvt_dl": "скачать этот файл$NГорячая клавиша: Y\">💾 скачать",
|
||
"tvt_prev": "показать предыдущий документ$NГорячая клавиша: i\">⬆ пред",
|
||
"tvt_next": "показать следующий документ$NГорячая клавиша: K\">⬇ след",
|
||
"tvt_sel": "выбрать документ ( для вырезания / копирования / удаления / ... )$NГорячая клавиша: S\">выд",
|
||
"tvt_edit": "открыть документ в текстовом редакторе$NГорячая клавиша: E\">✏️ изменить",
|
||
"tvt_tail": "проверять файл на изменения; показывать новые строки в реальном времени\">📡 обновлять",
|
||
"tvt_wrap": "перенос слов\">↵",
|
||
"tvt_atail": "прикрепить вид к низу страницы\">⚓",
|
||
"tvt_ctail": "декодировать цвета терминала (ansi escape codes)\">🌈",
|
||
"tvt_ntail": "лимит прокрутки (как много байт текста держать в памяти)",
|
||
|
||
"m3u_add1": "трек добавлен в плейлист m3u",
|
||
"m3u_addn": "{0} треков добавлено в плейлист m3u",
|
||
"m3u_clip": "плейлист m3u скопирован в буфер\n\nсоздайте файл с расширением <code>.m3u</code> и вставьте текст туда, чтобы сделать из него плейлист",
|
||
|
||
"gt_vau": "не показывать видео, только воспроизводить аудио\">🎧",
|
||
"gt_msel": "включить режим выделения; держите ctrl при нажатии для инвертации действия$N$N<em>когда активно: дважды кликните на файле / папке, чтобы открыть их</em>$N$NГорячая клавиша: S\">выделение",
|
||
"gt_crop": "обрезать миниатюры\">обрезка",
|
||
"gt_3x": "миниатюры высокого разрешения\">3x",
|
||
"gt_zoom": "размер",
|
||
"gt_chop": "длина имён",
|
||
"gt_sort": "сортировать по",
|
||
"gt_name": "имени",
|
||
"gt_sz": "размеру",
|
||
"gt_ts": "дате",
|
||
"gt_ext": "типу",
|
||
"gt_c1": "укоротить названия файлов",
|
||
"gt_c2": "удлинить названия файлов",
|
||
|
||
"sm_w8": "ищем...",
|
||
"sm_prev": "результаты поиска ниже - из предыдущего запроса:\n ",
|
||
"sl_close": "закрыть результаты поиска",
|
||
"sl_hits": "показ {0} совпадений",
|
||
"sl_moar": "загрузить больше",
|
||
|
||
"s_sz": "размер",
|
||
"s_dt": "дата",
|
||
"s_rd": "путь",
|
||
"s_fn": "имя",
|
||
"s_ta": "теги",
|
||
"s_ua": "дата⬆️",
|
||
"s_ad": "другое",
|
||
"s_s1": "минимум МиБ",
|
||
"s_s2": "максимум МиБ",
|
||
"s_d1": "мин. iso8601",
|
||
"s_d2": "макс. iso8601",
|
||
"s_u1": "загружено после",
|
||
"s_u2": "и/или до",
|
||
"s_r1": "путь содержит (разделить пробелами)",
|
||
"s_f1": "имя содержит (для исключения писать -nope)",
|
||
"s_t1": "теги содержат (^=начало, конец=$)",
|
||
"s_a1": "свойства метаданных",
|
||
|
||
"md_eshow": "не удалось показать ",
|
||
"md_off": "[📜<em>ридми</em>] отключён в [⚙️] -- документ скрыт",
|
||
|
||
"badreply": "Ошибка обработки ответа сервера",
|
||
|
||
"xhr403": "403: Доступ запрещён\n\nпопробуйте перезагрузить страницу, возможно, ваша сессия истекла",
|
||
"xhr0": "неизвестно (возможно, потеряно соединение с сервером, либо он отключён)",
|
||
"cf_ok": "просим прощения -- сработала защита от DD" + wah + "oS\n\nвсё должно вернуться в норму через 30 сек\n\nесли ничего не происходит - перезагрузите страницу",
|
||
"tl_xe1": "не удалось показать подпапки:\n\nошибка ",
|
||
"tl_xe2": "404: Папка не найдена",
|
||
"fl_xe1": "не удалось показать файлы:\n\nошибка ",
|
||
"fl_xe2": "404: Папка не найдена",
|
||
"fd_xe1": "не удалось создать подпапку:\n\nошибка ",
|
||
"fd_xe2": "404: Родительская папка не найдена",
|
||
"fsm_xe1": "не удалось отправить сообщение:\n\nошибка ",
|
||
"fsm_xe2": "404: Родительская папка не найдена",
|
||
"fu_xe1": "не удалось удалить список с сервера:\n\nошибка ",
|
||
"fu_xe2": "404: Файл не найден??",
|
||
|
||
"fz_tar": "несжатый файл gnu-tar (linux / mac)",
|
||
"fz_pax": "несжатый pax-форматированный tar (медленнее)",
|
||
"fz_targz": "gnu-tar с 3 уровнем сжатия gzip$N$Nобычно это очень медленно,$Nлучше использовать несжатый tar",
|
||
"fz_tarxz": "gnu-tar с 1 уровнем сжатия xz$N$Nобычно это очень медленно,$Nлучше использовать несжатый tar",
|
||
"fz_zip8": "zip с именами по utf8 (может работать криво на windows 7 и ниже)",
|
||
"fz_zipd": "zip с именами по cp437, для очень старого софта",
|
||
"fz_zipc": "cp437 с предварительным вычислением crc32,$N для MS-DOS PKZIP v2.04g (октябрь 1993)$N(требует больше времени для обработки перед скачиванием)",
|
||
|
||
"un_m1": "вы можете удалить ваши недавние загрузки (или отменить незавершённые) ниже",
|
||
"un_upd": "обновить",
|
||
"un_m4": "или поделиться файлами снизу:",
|
||
"un_ulist": "показать",
|
||
"un_ucopy": "копировать",
|
||
"un_flt": "опциональный фильтр: адрес должен содержать",
|
||
"un_fclr": "очистить фильтр",
|
||
"un_derr": 'ошибка удаления:\n',
|
||
"un_f5": 'что-то сломалось, пожалуйста перезагрузите страницу',
|
||
"un_uf5": "извините, но вам нужно перезагрузить страницу (F5 или Ctrl+R) перед тем, как отменить эту загрузку",
|
||
"un_nou": '<b>внимание:</b> сервер слишком нагружен, чтобы показать незавершённые загрузки; нажмите на ссылку "обновления" через пару секунд',
|
||
"un_noc": '<b>внимание:</b> удаление уже загруженных файлов запрещено конфигурацией сервера',
|
||
"un_max": "показаны первые 2000 файлов (используйте фильтр)",
|
||
"un_avail": "{0} недавних загрузок может быть удалено<br />{1} незавершённых может быть отменено",
|
||
"un_m2": "отсортировано по времени загрузки; сначала самые последние:",
|
||
"un_no1": "ха, поверил! достаточно свежих загрузок ещё нет",
|
||
"un_no2": "ха, поверил! достаточно свежих загрузок, соответствующих фильтру, ещё нет",
|
||
"un_next": "удалить следующие {0} файлов ниже",
|
||
"un_abrt": "отменить",
|
||
"un_del": "удалить",
|
||
"un_m3": "загружаю ваши недавние загрузки...",
|
||
"un_busy": "удаляю {0} файлов...",
|
||
"un_clip": "{0} ссылок скопировано в буфер",
|
||
|
||
"u_https1": "вам стоит",
|
||
"u_https2": "включить https",
|
||
"u_https3": "для лучшей производительности",
|
||
"u_ancient": 'у вас действительно антикварный браузер -- возможно, стоит <a href="#" onclick="goto(\'bup\')">использовать bup</a>',
|
||
"u_nowork": "требуется firefox 53+, chrome 57+ или iOS 11+",
|
||
"tail_2old": "требуется firefox 105+, chrome 71+ или iOS 14.5+",
|
||
"u_nodrop": 'ваш браузер слишком старый для загрузки через перетаскивание',
|
||
"u_notdir": "это не папка!\n\nваш браузер слишком старый,\nиспользуйте перетаскивание",
|
||
"u_uri": "чтобы перетащить картинку из других окон браузера,\nотпустите её на большую кнопку загрузки",
|
||
"u_enpot": 'переключиться на <a href="#">простой интерфейс</a> (может ускорить загрузку)',
|
||
"u_depot": 'переключиться на <a href="#">модный интерфейс</a> (может замедлить загрузку)',
|
||
"u_gotpot": 'переключаюсь на простой интерфес для ускорения загрузки,\n\nможете переключиться обратно, если хотите!',
|
||
"u_pott": "<p>файлы: <b>{0}</b> завершено, <b>{1}</b> ошибок, <b>{2}</b> загружаются, <b>{3}</b> в очереди</p>",
|
||
"u_ever": "это упрощённый загрузчик; up2k требует хотя бы<br>chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1",
|
||
"u_su2k": 'это упрощённый загрузчик; <a href="#" id="u2yea">up2k</a> лучше',
|
||
"u_uput": 'увеличить скорость (пропуск подсчёта контрольных сумм)',
|
||
"u_ewrite": 'у вас нет прав на запись в эту папку',
|
||
"u_eread": 'у вас нет прав на чтение из этой папки',
|
||
"u_enoi": 'поиск файлов выключен настройками сервера',
|
||
"u_enoow": "перезапись здесь не работает; требуются права на удаление",
|
||
"u_badf": 'Эти {0} из {1} файлов были пропущены, вероятно, из-за настроек доступа в файловой системе:\n\n',
|
||
"u_blankf": 'Эти {0} из {1} файлов пустые; всё равно загрузить?\n\n',
|
||
"u_applef": 'Эти {0} из {1} файлов, вероятно, нежелательны;\nНажмите <code>OK/Enter</code>, чтобы ПРОПУСТИТЬ их,\nНажмите <code>Отмена/ESC</code>, чтобы проигнорировать сообщение и ЗАГРУЗИТЬ их:\n\n',
|
||
"u_just1": '\nВозможно, будет лучше, если вы выберете только один файл',
|
||
"u_ff_many": "если вы используете <b>Linux / MacOS / Android,</b> тогда такое количество файлов <a href=\"https://bugzilla.mozilla.org/show_bug.cgi?id=1790500\" target=\"_blank\"><em>может</em> крашнуть Firefox!</a>\nв таком случае попробуйте снова (или используйте Chrome).",
|
||
"u_up_life": "Эта загрузка будет удалена с сервера\nчерез {0} после её завершения",
|
||
"u_asku": 'загрузить эти {0} файлов в <code>{1}</code>',
|
||
"u_unpt": "вы можете отменить / удалить эту загрузку с помощью 🧯 сверху слева",
|
||
"u_bigtab": 'будет показано {0} файлов\n\nэто может крашнуть браузер, вы уверены?',
|
||
"u_scan": 'Сканирую файлы...',
|
||
"u_dirstuck": 'сканер папки завис при попытке получить доступ к следующим {0} файлам; будет пропущено:',
|
||
"u_etadone": 'Готово ({0}, {1} файлов)',
|
||
"u_etaprep": '(подготавливаю загрузку)',
|
||
"u_hashdone": 'хеширование выполнено',
|
||
"u_hashing": 'хеш',
|
||
"u_hs": 'подготовка к хешированию...',
|
||
"u_started": "файлы загружаются; подробнее в [🚀]",
|
||
"u_dupdefer": "дубликат; будет обработан после всех остальных файлов",
|
||
"u_actx": "нажмите на этот текст, чтобы предотвратить<br />падение производительности при просмотре других вкладок / окон",
|
||
"u_fixed": "Окей! Исправлено 👍",
|
||
"u_cuerr": "не удалось загрузить фрагмент {0} из {1};\nвероятно, не критично - продолжаю\n\nфайл: {2}",
|
||
"u_cuerr2": "отказ в загрузке (фрагмент {0} из {1});\nпопытаюсь повторить позже\n\nфайл: {2}\n\nошибка ",
|
||
"u_ehstmp": "попытаюсь повторить позже; подробнее снизу справа",
|
||
"u_ehsfin": "отказ в запросе завершения загрузки; повторяю...",
|
||
"u_ehssrch": "отказ в запросе поиска; повторяю...",
|
||
"u_ehsinit": "отказ в запросе начала загрузки; повторяю...",
|
||
"u_eneths": "ошибка подключения во время подготовки загрузки; повторяю...",
|
||
"u_enethd": "ошибка подключения во время проверки существования целевого файла; повторяю...",
|
||
"u_cbusy": "ожидаю возвращения доступа к серверу после ошибки подключения...",
|
||
"u_ehsdf": "на сервере закончилось место!\n\nбуду пытаться повторить, на случай если\nместо будет освобождено",
|
||
"u_emtleak1": "кажется, у вашего браузера может быть утечка памяти;\nпожалуйста,",
|
||
"u_emtleak2": ' <a href="{0}">перейдите на https (рекомендовано)</a> или ',
|
||
"u_emtleak3": ' ',
|
||
"u_emtleakc": 'попробуйте сделать так:\n<ul><li>нажмите <code>F5</code> для обновления страницы</li><li>затем отключите опцию <code>мп</code> в <code>⚙️ настройках</code></li><li>и повторите попытку загрузки</li></ul>Она будет чуть медленнее, но что поделать.\nИзвините за неудобства!\n\nPS: в chrome v107 <a href="https://bugs.chromium.org/p/chromium/issues/detail?id=1354816" target="_blank">это исправили</a>',
|
||
"u_emtleakf": 'попробуйте сделать так:\n<ul><li>нажмите <code>F5</code> для обновления страницы</li><li>затем включите опцию <code>🥔</code> (простой интерфейс) во вкладке загрузок<li>и повторите попытку</li></ul>\nPS: firefox <a href="https://bugzilla.mozilla.org/show_bug.cgi?id=1790500" target="_blank">скоро должны починить</a> в этом аспекте',
|
||
"u_s404": "не найдено на сервере",
|
||
"u_expl": "объяснить",
|
||
"u_maxconn": "в большинстве браузеров это нельзя поднять выше 6, но firefox позволяет увеличить лимит с помощью <code>connections-per-server</code> в <code>about:config</code>",
|
||
"u_tu": '<p class="warn">ВНИМАНИЕ: активен режим турбо, <span> клиент может игнорировать незавершённые загрузки; подробнее при наведении на кнопку турбо</span></p>',
|
||
"u_ts": '<p class="warn">ВНИМАНИЕ: активен режим турбо, <span> результаты поиска могут быть некорректными; подробнее при наведении на кнопку турбо</span></p>',
|
||
"u_turbo_c": "режим турбо отключён сервером",
|
||
"u_turbo_g": "отключаю турбо, поскольку у вас нет прав\nна просмотр папок в этом хранилище",
|
||
"u_life_cfg": 'автоудаление через <input id="lifem" p="60" /> мин (или <input id="lifeh" p="3600" /> часов)',
|
||
"u_life_est": 'загрузка будет удалена <span id="lifew" tt="local time">---</span>',
|
||
"u_life_max": 'эта папка требует\nавтоудаления файлов через {0}',
|
||
"u_unp_ok": 'удаление разрешено для {0}',
|
||
"u_unp_ng": 'удаление НЕ будет разрешено',
|
||
"ue_ro": 'ваш доступ к данной папке - только чтение\n\n',
|
||
"ue_nl": 'на данный момент вы не авторизованы',
|
||
"ue_la": 'вы авторизованы как "{0}"',
|
||
"ue_sr": 'сейчас вы в режиме поиска файлов\n\nперейдите в режим загрузки, нажав на лупу 🔎 (рядом с огромной кнопкой ИСКАТЬ), и попробуйте снова\n\nизвините',
|
||
"ue_ta": 'попробуйте загрузить снова, теперь должно сработать',
|
||
"ue_ab": "этот файл уже загружают в другую папку, та загрузка должна быть завершена перед тем, как загрузить этот же файл в другое место.\n\nВы можете отменить свою загрузку через 🧯 сверху слева",
|
||
"ur_1uo": "OK: Файл успешно загружен",
|
||
"ur_auo": "OK: Все {0} файлов успешно загружены",
|
||
"ur_1so": "OK: Файл найден на сервере",
|
||
"ur_aso": "OK: Все {0} файлов найдены на сервере",
|
||
"ur_1un": "Загрузка не удалась, извините",
|
||
"ur_aun": "Все {0} загрузок не удались, извините",
|
||
"ur_1sn": "Файл НЕ был найден на сервере",
|
||
"ur_asn": "Все {0} файлов НЕ было найдено на сервере",
|
||
"ur_um": "Завершено;\n{0} успешно,\n{1} ошибок, извините",
|
||
"ur_sm": "Завершено;\n{0} файлов найдено на сервере,\n{1} файлов НЕ найдено на сервере",
|
||
|
||
"lang_set": "перезагрузить страницу, чтобы применить изменения?",
|
||
},
|
||
};
|
||
|
||
var LANGS = ["eng", "nor", "chi", "deu", "rus"];
|
||
|
||
if (window.langmod)
|
||
langmod();
|
||
|
||
for (var a = LANGS.length; a > 0;)
|
||
if (!Ls[LANGS[--a]])
|
||
LANGS.splice(a, 1);
|
||
|
||
var L = Ls[sread("cpp_lang", LANGS) || lang] ||
|
||
Ls.eng || Ls.nor || Ls.chi;
|
||
|
||
for (var a = 0; a < LANGS.length; a++) {
|
||
for (var b = a + 1; b < LANGS.length; b++) {
|
||
var i1 = Object.keys(Ls[LANGS[a]]).length > Object.keys(Ls[LANGS[b]]).length ? a : b,
|
||
i2 = i1 == a ? b : a,
|
||
t1 = Ls[LANGS[i1]],
|
||
t2 = Ls[LANGS[i2]];
|
||
|
||
for (var k in t1)
|
||
if (!t2[k]) {
|
||
console.log("E missing TL", LANGS[i2], k);
|
||
t2[k] = t1[k];
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!has(LANGS, lang))
|
||
alert('unsupported --lang "' + lang + '" specified in server args;\nplease use one of these: ' + LANGS);
|
||
|
||
modal.load();
|
||
|
||
|
||
// toolbar
|
||
ebi('ops').innerHTML = (
|
||
'<a href="#" id="opa_x" data-dest="" tt="' + L.ot_close + '">--</a>' +
|
||
'<a href="#" id="opa_srch" data-perm="read" data-dep="idx" data-dest="search" tt="' + L.ot_search + '">🔎</a>' +
|
||
(have_del ? '<a href="#" id="opa_del" data-dest="unpost" tt="' + L.ot_unpost + '">🧯</a>' : '') +
|
||
'<a href="#" id="opa_up" data-dest="up2k">🚀</a>' +
|
||
'<a href="#" id="opa_bup" data-perm="write" data-dest="bup" tt="' + L.ot_bup + '">🎈</a>' +
|
||
'<a href="#" id="opa_mkd" data-perm="write" data-dest="mkdir" tt="' + L.ot_mkdir + '">📂</a>' +
|
||
'<a href="#" id="opa_md" data-perm="read write" data-dest="new_md" tt="' + L.ot_md + '">📝</a>' +
|
||
'<a href="#" id="opa_msg" data-dest="msg" tt="' + L.ot_msg + '">📟</a>' +
|
||
'<a href="#" id="opa_auc" data-dest="player" tt="' + L.ot_mp + '">🎺</a>' +
|
||
'<a href="#" id="opa_cfg" data-dest="cfg" tt="' + L.ot_cfg + '">⚙️</a>' +
|
||
(IE ? '<span id="noie">' + L.ot_noie + '</span>' : '') +
|
||
'<div id="opdesc"></div>'
|
||
);
|
||
|
||
|
||
// media player
|
||
ebi('widget').innerHTML = (
|
||
'<div id="wtoggle">' +
|
||
'<span id="wfs"></span>' +
|
||
'<span id="wfm"><a' +
|
||
' href="#" id="fshr" tt="' + L.wt_shr + '">📨<span>share</span></a><a' +
|
||
' href="#" id="fren" tt="' + L.wt_ren + '">✎<span>name</span></a><a' +
|
||
' href="#" id="fdel" tt="' + L.wt_del + '">⌫<span>del.</span></a><a' +
|
||
' href="#" id="fcut" tt="' + L.wt_cut + '">✂<span>cut</span></a><a' +
|
||
' href="#" id="fcpy" tt="' + L.wt_cpy + '">⧉<span>copy</span></a><a' +
|
||
' href="#" id="fpst" tt="' + L.wt_pst + '">📋<span>paste</span></a>' +
|
||
'</span><span id="wzip1"><a' +
|
||
' href="#" id="zip1" tt="' + L.wt_zip1 + '">📦<span>zip</span></a>' +
|
||
'</span><span id="wzip"><a' +
|
||
' href="#" id="selall" tt="' + L.wt_selall + '">sel.<br />all</a><a' +
|
||
' href="#" id="selinv" tt="' + L.wt_selinv + '">sel.<br />inv.</a><a' +
|
||
' href="#" id="selzip" class="l1" tt="' + L.wt_selzip + '">zip</a><a' +
|
||
' href="#" id="seldl" class="l1" tt="' + L.wt_seldl + '">dl</a>' +
|
||
'</span><span id="wnp"><a' +
|
||
' href="#" id="npirc" tt="' + L.wt_npirc + '">📋<span>irc</span></a><a' +
|
||
' href="#" id="nptxt" tt="' + L.wt_nptxt + '">📋<span>txt</span></a>' +
|
||
'</span><span id="wm3u"><a' +
|
||
' href="#" id="m3ua" tt="' + L.wt_m3ua + '">📻<span>add</span></a><a' +
|
||
' href="#" id="m3uc" tt="' + L.wt_m3uc + '">📻<span>copy</span></a>' +
|
||
'</span><a' +
|
||
' href="#" id="wtgrid" tt="' + L.wt_grid + '">田</a><a' +
|
||
' href="#" id="wtico">♫</a>' +
|
||
'</div>' +
|
||
'<div id="widgeti">' +
|
||
' <div id="pctl"><a href="#" id="bprev" tt="' + L.wt_prev + '">⏮</a><a href="#" id="bplay" tt="' + L.wt_play + '">▶</a><a href="#" id="bnext" tt="' + L.wt_next + '">⏭</a></div>' +
|
||
' <canvas id="pvol" width="288" height="38"></canvas>' +
|
||
' <canvas id="barpos"></canvas>' +
|
||
' <canvas id="barbuf"></canvas>' +
|
||
'</div>' +
|
||
'<div id="np_inf">' +
|
||
' <img id="np_img" />' +
|
||
' <span id="np_url"></span>' +
|
||
' <span id="np_circle"></span>' +
|
||
' <span id="np_album"></span>' +
|
||
' <span id="np_tn"></span>' +
|
||
' <span id="np_artist"></span>' +
|
||
' <span id="np_title"></span>' +
|
||
' <span id="np_pos"></span>' +
|
||
' <span id="np_dur"></span>' +
|
||
'</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 class="c" data-perm="read"><br />' + L.ul_par + '</td>\n' +
|
||
' <td class="c" rowspan="2">\n' +
|
||
' <input type="checkbox" id="multitask" />\n' +
|
||
' <label for="multitask" tt="' + L.ut_mt + '">🏃</label>\n' +
|
||
' </td>\n' +
|
||
' <td class="c" rowspan="2">\n' +
|
||
' <input type="checkbox" id="potato" />\n' +
|
||
' <label for="potato" tt="' + L.ut_pot + '">🥔</label>\n' +
|
||
' </td>\n' +
|
||
' <td class="c" rowspan="2">\n' +
|
||
' <input type="checkbox" id="u2rand" />\n' +
|
||
' <label for="u2rand" tt="' + L.ut_rand + '">🎲</label>\n' +
|
||
' </td>\n' +
|
||
' <td class="c" rowspan="2">\n' +
|
||
' <input type="checkbox" id="u2ow" />\n' +
|
||
' <label for="u2ow" tt="' + L.ut_ow + '">?</a>\n' +
|
||
' </td>\n' +
|
||
' <td class="c" data-perm="read" data-dep="idx" rowspan="2">\n' +
|
||
' <input type="checkbox" id="fsearch" />\n' +
|
||
' <label for="fsearch" tt="' + L.ut_srch + '">🔎</label>\n' +
|
||
' </td>\n' +
|
||
' <td data-perm="read" rowspan="2" id="u2btn_cw"></td>\n' +
|
||
' <td data-perm="read" rowspan="2" id="u2c3w"></td>\n' +
|
||
' </tr>\n' +
|
||
' <tr>\n' +
|
||
' <td class="c" data-perm="read">\n' +
|
||
' <a href="#" class="b" id="nthread_sub">–</a><input\n' +
|
||
' class="txtbox" id="nthread" value="" tt="' + L.ut_par + '"/><a\n' +
|
||
' href="#" class="b" 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" tabindex="0">\n' +
|
||
' <span id="u2bm"></span>\n' + L.ul_btn +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
|
||
'<div id="u2c3t">\n' +
|
||
|
||
'<div id="u2etaw"><div id="u2etas"><div class="o">\n' +
|
||
L.ul_hash + ': <span id="u2etah" tt="' + L.ut_etah + '">(' + L.ul_idle1 + ')</span><br />\n' +
|
||
L.ul_send + ': <span id="u2etau" tt="' + L.ut_etau + '">(' + L.ul_idle1 + ')</span><br />\n' +
|
||
' </div><span class="o">' +
|
||
L.ul_done + ': </span><span id="u2etat" tt="' + L.ut_etat + '">(' + L.ul_idle1 + ')</span>\n' +
|
||
'</div></div>\n' +
|
||
|
||
'<div id="u2cards">\n' +
|
||
' <a href="#" act="ok" tt="' + L.uct_ok + '">ok <span>0</span></a><a\n' +
|
||
' href="#" act="ng" tt="' + L.uct_ng + '">ng <span>0</span></a><a\n' +
|
||
' href="#" act="done" tt="' + L.uct_done + '">done <span>0</span></a><a\n' +
|
||
' href="#" act="bz" tt="' + L.uct_bz + '" class="act">busy <span>0</span></a><a\n' +
|
||
' href="#" act="q" tt="' + L.uct_q + '">que <span>0</span></a>\n' +
|
||
'</div>\n' +
|
||
|
||
'</div>\n' +
|
||
|
||
'<div id="u2tabw" class="na"><table id="u2tab">\n' +
|
||
' <thead>\n' +
|
||
' <tr>\n' +
|
||
' <td>' + L.utl_name + ' (<a href="#" id="luplinks">' + L.utl_ulist + '</a>/<a href="#" id="cuplinks">' + L.utl_ucopy + '</a>' + L.utl_links + ')</td>\n' +
|
||
' <td>' + L.utl_stat + '</td>\n' +
|
||
' <td>' + L.utl_prog + '</td>\n' +
|
||
' </tr>\n' +
|
||
' </thead>\n' +
|
||
' <tbody></tbody>\n' +
|
||
'</table><div id="u2mu"></div></div>\n' +
|
||
|
||
'<p id="u2flagblock"><b>' + L.ul_flagblk + '</p>\n' +
|
||
'<div id="u2life"></div>' +
|
||
'<div id="u2foot"></div>'
|
||
);
|
||
|
||
|
||
ebi('wrap').insertBefore(mknod('div', 'lazy'), ebi('epi'));
|
||
|
||
var x = ebi('bbsw');
|
||
x.parentNode.insertBefore(mknod('div', null,
|
||
'<input type="checkbox" id="uput" name="uput"><label for="uput">' + L.u_uput + '</label>'), x);
|
||
|
||
|
||
(function () {
|
||
var o = mknod('div');
|
||
o.innerHTML = (
|
||
'<div id="drops">\n' +
|
||
' <div class="dropdesc" id="up_zd"><div>🚀 ' + L.udt_up + '<br /><span></span><div>🚀<b>' + L.udt_up + '</b></div><div><b>' + L.udt_up + '</b>🚀</div></div></div>\n' +
|
||
' <div class="dropdesc" id="srch_zd"><div>🔎 ' + L.udt_srch + '<br /><span></span><div>🔎<b>' + L.udt_srch + '</b></div><div><b>' + L.udt_srch + '</b>🔎</div></div></div>\n' +
|
||
' <div class="dropzone" id="up_dz" v="up_zd"></div>\n' +
|
||
' <div class="dropzone" id="srch_dz" v="srch_zd"></div>\n' +
|
||
'</div>'
|
||
);
|
||
document.body.appendChild(o);
|
||
})();
|
||
|
||
|
||
// config panel
|
||
ebi('op_cfg').innerHTML = (
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_opts + '</h3>\n' +
|
||
' <div>\n' +
|
||
' <a id="tooltips" class="tgl btn" href="#" tt="' + L.ct_ttips + '</a>\n' +
|
||
' <a id="griden" class="tgl btn" href="#" tt="' + L.wt_grid + '">' + L.ct_grid + '</a>\n' +
|
||
' <a id="thumbs" class="tgl btn" href="#" tt="' + L.ct_thumb + '</a>\n' +
|
||
' <a id="csel" class="tgl btn" href="#" tt="' + L.ct_csel + '</a>\n' +
|
||
' <a id="ihop" class="tgl btn" href="#" tt="' + L.ct_ihop + '</a>\n' +
|
||
' <a id="dotfiles" class="tgl btn" href="#" tt="' + L.ct_dots + '</a>\n' +
|
||
' <a id="qdel" class="tgl btn" href="#" tt="' + L.ct_qdel + '</a>\n' +
|
||
' <a id="dir1st" class="tgl btn" href="#" tt="' + L.ct_dir1st + '</a>\n' +
|
||
' <a id="nsort" class="tgl btn" href="#" tt="' + L.ct_nsort + '</a>\n' +
|
||
' <a id="ireadme" class="tgl btn" href="#" tt="' + L.ct_readme + '</a>\n' +
|
||
' <a id="idxh" class="tgl btn" href="#" tt="' + L.ct_idxh + '</a>\n' +
|
||
' <a id="sbars" class="tgl btn" href="#" tt="' + L.ct_sbars + '</a>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_themes + '</h3>\n' +
|
||
' <div id="themes">\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_langs + '</h3>\n' +
|
||
' <div id="langs">\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
(have_zip ? (
|
||
'<div><h3>' + L.cl_ziptype + '</h3><div id="arc_fmt"></div></div>\n'
|
||
) : '') +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_uopts + '</h3>\n' +
|
||
' <div>\n' +
|
||
' <a id="ask_up" class="tgl btn" href="#" tt="' + L.ut_ask + '</a>\n' +
|
||
' <a id="u2ts" class="tgl btn" href="#" tt="' + L.ut_u2ts + '</a>\n' +
|
||
' <a id="umod" class="tgl btn" href="#" tt="' + L.cut_umod + '</a>\n' +
|
||
' <a id="hashw" class="tgl btn" href="#" tt="' + L.cut_mt + '</a>\n' +
|
||
' <a id="nosubtle" class="tgl btn" href="#" tt="' + L.cut_wasm + '</a>\n' +
|
||
' <a id="u2turbo" class="tgl btn ttb" href="#" tt="' + L.cut_turbo + '</a>\n' +
|
||
' <a id="u2tdate" class="tgl btn ttb" href="#" tt="' + L.cut_datechk + '</a>\n' +
|
||
' <input type="text" id="u2szg" value="" ' + NOAC + ' style="width:3em" tt="' + L.cut_u2sz + '" />' +
|
||
' <a id="flag_en" class="tgl btn" href="#" tt="' + L.cut_flag + '">💤</a>\n' +
|
||
' <a id="u2sort" class="tgl btn" href="#" tt="' + L.cut_az + '">az</a>\n' +
|
||
' <a id="upnag" class="tgl btn" href="#" tt="' + L.cut_nag + '">🔔</a>\n' +
|
||
' <a id="upsfx" class="tgl btn" href="#" tt="' + L.cut_sfx + '">🔊</a>\n' +
|
||
' </td>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_favico + ' <span id="ico1">🎉</span></h3>\n' +
|
||
' <div>\n' +
|
||
' <input type="text" id="icot" value="" ' + NOAC + ' style="width:1.3em" tt="' + L.cft_text + '" />' +
|
||
' <input type="text" id="icof" value="" ' + NOAC + ' style="width:2em" tt="' + L.cft_fg + '" />' +
|
||
' <input type="text" id="icob" value="" ' + NOAC + ' style="width:2em" tt="' + L.cft_bg + '" />' +
|
||
' </td>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_bigdir + '</h3>\n' +
|
||
' <div>\n' +
|
||
' <input type="text" id="bd_lim" value="250" ' + NOAC + ' style="width:4em" tt="' + L.cdt_lim + '" />' +
|
||
' <a id="bd_ask" class="tgl btn" href="#" tt="' + L.cdt_ask + '">ask</a>\n' +
|
||
' </td>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div>\n' +
|
||
' <h3>' + L.cl_hsort + '</h3>\n' +
|
||
' <div>\n' +
|
||
' <input type="text" id="hsortn" value="" ' + NOAC + ' style="width:3em" tt="' + L.cdt_hsort + '" />' +
|
||
' </td>\n' +
|
||
' </div>\n' +
|
||
'</div>\n' +
|
||
'<div><h3>' + L.cl_keytype + '</h3><div id="key_notation"></div></div>\n' +
|
||
'<div><h3>' + L.cl_hiddenc + ' ' + (MOBILE ? '<a href="#" id="hcolsh">' + L.cl_hidec + '</a> / ' : '') + '<a href="#" id="hcolsr">' + L.cl_reset + '</a></h3><div id="hcols"></div></div>'
|
||
);
|
||
|
||
|
||
// navpane
|
||
ebi('tree').innerHTML = (
|
||
'<div id="treeh">\n' +
|
||
' <a href="#" id="detree" tt="' + L.tt_detree + '">🍞...</a>\n' +
|
||
' <a href="#" class="btn" step="2" id="twobytwo" tt="Hotkey: D">+</a>\n' +
|
||
' <a href="#" class="btn" step="-2" id="twig" tt="Hotkey: A">–</a>\n' +
|
||
' <a href="#" class="btn" id="visdir" tt="' + L.tt_visdir + '">🎯</a>\n' +
|
||
' <a href="#" class="tgl btn" id="filetree" tt="' + L.tt_ftree + '">📃</a>\n' +
|
||
' <a href="#" class="tgl btn" id="parpane" tt="' + L.tt_pdock + '">📌</a>\n' +
|
||
' <a href="#" class="tgl btn" id="dyntree" tt="' + L.tt_dynt + '">a</a>\n' +
|
||
' <a href="#" class="tgl btn" id="wraptree" tt="' + L.tt_wrap + '">↵</a>\n' +
|
||
' <a href="#" class="tgl btn" id="hovertree" tt="' + L.tt_hover + '">👀</a>\n' +
|
||
'</div>\n' +
|
||
'<ul id="docul"></ul>\n' +
|
||
'<ul class="ntree" id="treepar"></ul>\n' +
|
||
'<ul class="ntree" id="treeul"></ul>\n' +
|
||
'<div id="thx_ff"> </div>'
|
||
);
|
||
clmod(ebi('tree'), 'sbar', 1);
|
||
ebi('entree').setAttribute('tt', L.tt_entree);
|
||
ebi('goh').textContent = L.goh;
|
||
QS('#op_mkdir input[type="submit"]').value = L.ab_mkdir;
|
||
QS('#op_new_md input[type="submit"]').value = L.ab_mkdoc;
|
||
QS('#op_msg input[type="submit"]').value = L.ab_msg;
|
||
|
||
|
||
(function () {
|
||
var ops = QSA('#ops>a');
|
||
for (var a = 0; a < ops.length; a++) {
|
||
ops[a].onclick = opclick;
|
||
var v = ops[a].getAttribute('data-dest');
|
||
if (v)
|
||
ops[a].href = '#v=' + v;
|
||
}
|
||
})();
|
||
|
||
|
||
function opclick(e) {
|
||
var dest = this.getAttribute('data-dest');
|
||
if (QS('#op_' + dest + '.act'))
|
||
dest = '';
|
||
|
||
swrite('opmode', dest || null);
|
||
if (ctrl(e))
|
||
return;
|
||
|
||
ev(e);
|
||
goto(dest);
|
||
|
||
var input = QS('.opview.act input:not([type="hidden"])')
|
||
if (input && !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 lnk = QS('#ops>a[data-dest=' + dest + ']'),
|
||
nps = lnk.getAttribute('data-perm');
|
||
|
||
nps = nps && nps.length ? nps.split(' ') : [];
|
||
|
||
if (perms.length)
|
||
for (var a = 0; a < nps.length; a++)
|
||
if (!has(perms, nps[a]))
|
||
return;
|
||
|
||
if (!has(perms, 'read') && !has(perms, 'write') && (dest == 'up2k'))
|
||
return;
|
||
|
||
clmod(ebi('op_' + dest), 'act', 1);
|
||
clmod(lnk, 'act', 1);
|
||
|
||
var fn = window['goto_' + dest];
|
||
if (fn)
|
||
fn();
|
||
}
|
||
|
||
clmod(document.documentElement, 'op_open', dest);
|
||
|
||
if (treectl)
|
||
treectl.onscroll();
|
||
}
|
||
|
||
|
||
var m = SPINNER.split(','),
|
||
SPINNER_CSS = SPINNER.slice(1 + m[0].length);
|
||
SPINNER = m[0];
|
||
|
||
|
||
var SBW, SBH; // scrollbar size
|
||
function read_sbw() {
|
||
var el = mknod('div');
|
||
el.style.cssText = 'overflow:scroll;width:100px;height:100px;position:absolute;top:0;left:0';
|
||
document.body.appendChild(el);
|
||
SBW = el.offsetWidth - el.clientWidth;
|
||
SBH = el.offsetHeight - el.clientHeight;
|
||
document.body.removeChild(el);
|
||
setcvar('--sbw', SBW + 'px');
|
||
setcvar('--sbh', SBH + 'px');
|
||
}
|
||
onresize100.add(read_sbw, true);
|
||
|
||
|
||
var have_webp = sread('have_webp');
|
||
(function () {
|
||
if (have_webp !== null)
|
||
return;
|
||
|
||
var img = new Image();
|
||
img.onload = function () {
|
||
have_webp = img.width > 0 && img.height > 0;
|
||
swrite('have_webp', 'ya');
|
||
};
|
||
img.onerror = function () {
|
||
have_webp = false;
|
||
swrite('have_webp', '');
|
||
};
|
||
img.src = "data:image/webp;base64,UklGRhoAAABXRUJQVlA4TA0AAAAvAAAAEAcQERGIiP4HAA==";
|
||
})();
|
||
|
||
|
||
function set_files_html(html) {
|
||
var files = ebi('files');
|
||
try {
|
||
files.innerHTML = html;
|
||
return files;
|
||
}
|
||
catch (e) {
|
||
var par = files.parentNode;
|
||
par.removeChild(files);
|
||
files = mknod('div');
|
||
files.innerHTML = '<table id="files">' + html + '</table>';
|
||
par.insertBefore(files.childNodes[0], ebi('lazy'));
|
||
files = ebi('files');
|
||
return files;
|
||
}
|
||
}
|
||
|
||
|
||
// actx breaks background album playback on ios
|
||
var ACtx = !IPHONE && (window.AudioContext || window.webkitAudioContext),
|
||
ACB = sread('au_cbv') || 1,
|
||
hash0 = location.hash,
|
||
sloc0 = '' + location,
|
||
noih = /[?&]v\b/.exec(sloc0),
|
||
rtt = null,
|
||
ldks = [],
|
||
dks = {},
|
||
dk, mp;
|
||
|
||
|
||
if (window.og_fn) {
|
||
hash0 = 1;
|
||
hist_replace(vsplit(get_evpath())[0]);
|
||
}
|
||
|
||
|
||
var hsortn = ebi('hsortn').value = icfg_get('hsortn', dhsortn);
|
||
ebi('hsortn').oninput = function (e) {
|
||
var n = parseInt(this.value);
|
||
swrite('hsortn', hsortn = (isNum(n) ? n : dhsortn));
|
||
};
|
||
(function() {
|
||
var args = ('' + hash0).split(/,sort/g);
|
||
if (args.length < 2)
|
||
return;
|
||
|
||
var ret = [];
|
||
for (var a = 1; a < args.length; a++) {
|
||
var t = '', n = 1, z = args[a].split(',')[0];
|
||
if (z.startsWith('-')) {
|
||
z = z.slice(1);
|
||
n = -1;
|
||
}
|
||
if (z == "sz" || z.indexOf('/.') + 1)
|
||
t = "int";
|
||
ret.push([z, n, t]);
|
||
}
|
||
n = Math.min(ret.length, hsortn);
|
||
if (n) {
|
||
var cmp = jread('fsort', []);
|
||
if (JSON.stringify(ret.slice(0, n) !=
|
||
JSON.stringify(cmp.slice(0, n))))
|
||
jwrite('fsort', ret);
|
||
}
|
||
})();
|
||
|
||
|
||
var mpl = (function () {
|
||
var have_mctl = 'mediaSession' in navigator && window.MediaMetadata;
|
||
|
||
ebi('op_player').innerHTML = (
|
||
'<div><h3>' + L.cl_opts + '</h3><div>' +
|
||
'<a href="#" class="tgl btn" id="au_loop" tt="' + L.mt_loop + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_one" tt="' + L.mt_one + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_shuf" tt="' + L.mt_shuf + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_aplay" tt="' + L.mt_aplay + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_preload" tt="' + L.mt_preload + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_prescan" tt="' + L.mt_prescan + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_fullpre" tt="' + L.mt_fullpre + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_fau" tt="' + L.mt_fau + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_waves" tt="' + L.mt_waves + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_npclip" tt="' + L.mt_npclip + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_m3u_c" tt="' + L.mt_m3u_c + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_os_ctl" tt="' + L.mt_octl + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_os_seek" tt="' + L.mt_oseek + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_osd_cv" tt="' + L.mt_oscv + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_follow" tt="' + L.mt_follow + '</a>' +
|
||
'<a href="#" class="tgl btn" id="au_compact" tt="' + L.mt_compact + '</a>' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3>' + L.ml_btns + '</h3><div>' +
|
||
'<a href="#" class="btn" id="au_uncache" tt="' + L.mt_uncache + '</a>' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3>' + L.ml_pmode + '</h3><div id="pb_mode">' +
|
||
'<a href="#" class="tgl btn" m="loop" tt="' + L.mt_mloop + '</a>' +
|
||
'<a href="#" class="tgl btn" m="next" tt="' + L.mt_mnext + '</a>' +
|
||
'<a href="#" class="tgl btn" m="stop" tt="' + L.mt_mstop + '</a>' +
|
||
'</div></div>' +
|
||
|
||
(have_acode ? (
|
||
'<div><h3>' + L.ml_tcode + '</h3><div>' +
|
||
'<a href="#" id="ac_flac" class="tgl btn" tt="' + L.mt_cflac + '</a>' +
|
||
'<a href="#" id="ac_aac" class="tgl btn" tt="' + L.mt_caac + '</a>' +
|
||
'<a href="#" id="ac_oth" class="tgl btn" tt="' + L.mt_coth + '</a>' +
|
||
'</div></div>' +
|
||
'<div><h3>' + L.ml_tcode2 + '</h3><div>' +
|
||
'<a href="#" id="ac2opus" class="tgl btn" tt="' + L.mt_c2opus + '</a>' +
|
||
'<a href="#" id="ac2owa" class="tgl btn" tt="' + L.mt_c2owa + '</a>' +
|
||
'<a href="#" id="ac2caf" class="tgl btn" tt="' + L.mt_c2caf + '</a>' +
|
||
'<a href="#" id="ac2mp3" class="tgl btn" tt="' + L.mt_c2mp3 + '</a>' +
|
||
'</div></div>'
|
||
) : '') +
|
||
|
||
'<div><h3>' + L.ml_tint + '</h3><div>' +
|
||
'<input type="text" id="pb_tint" value="0" ' + NOAC + ' style="width:2.4em" tt="' + L.mt_tint + '" />' +
|
||
'</div></div>' +
|
||
|
||
'<div><h3 id="h_drc">' + L.ml_drc + '</h3><div id="audio_drc"></div></div>' +
|
||
'<div><h3>' + L.ml_eq + '</h3><div id="audio_eq"></div></div>' +
|
||
'');
|
||
|
||
var r = {
|
||
"pb_mode": (sread('pb_mode', ['loop', 'next', 'stop']) || 'next').split('-')[0],
|
||
"os_ctl": bcfg_get('au_os_ctl', have_mctl) && have_mctl,
|
||
'traversals': 0,
|
||
'm3ut': '#EXTM3U\n',
|
||
};
|
||
bcfg_bind(r, 'one', 'au_one', false, function (v) {
|
||
if (mp.au)
|
||
mp.au.loop = !v && r.loop;
|
||
});
|
||
bcfg_bind(r, 'loop', 'au_loop', false, function (v) {
|
||
if (mp.au)
|
||
mp.au.loop = v;
|
||
});
|
||
bcfg_bind(r, 'shuf', 'au_shuf', false, function () {
|
||
mp.read_order(); // don't bind
|
||
});
|
||
bcfg_bind(r, 'aplay', 'au_aplay', true);
|
||
bcfg_bind(r, 'preload', 'au_preload', true);
|
||
bcfg_bind(r, 'prescan', 'au_prescan', true);
|
||
bcfg_bind(r, 'fullpre', 'au_fullpre', false);
|
||
bcfg_bind(r, 'fau', 'au_fau', MOBILE && !IPHONE, function (v) {
|
||
mp.nopause();
|
||
if (mp.fau) {
|
||
mp.fau.pause();
|
||
mp.fau = mpo.fau = null;
|
||
console.log('stop fau');
|
||
}
|
||
mp.init_fau();
|
||
});
|
||
bcfg_bind(r, 'waves', 'au_waves', true, function (v) {
|
||
if (!v) pbar.unwave();
|
||
});
|
||
bcfg_bind(r, 'os_seek', 'au_os_seek', !IPHONE, announce);
|
||
bcfg_bind(r, 'osd_cv', 'au_osd_cv', true, announce);
|
||
bcfg_bind(r, 'clip', 'au_npclip', false, function (v) {
|
||
clmod(ebi('wtoggle'), 'np', v && mp.au);
|
||
});
|
||
bcfg_bind(r, 'm3uen', 'au_m3u_c', false, function (v) {
|
||
clmod(ebi('wtoggle'), 'm3u', v && (mp.au || msel.getsel().length));
|
||
});
|
||
bcfg_bind(r, 'follow', 'au_follow', false, setaufollow);
|
||
bcfg_bind(r, 'ac_flac', 'ac_flac', true);
|
||
bcfg_bind(r, 'ac_aac', 'ac_aac', false);
|
||
bcfg_bind(r, 'ac_oth', 'ac_oth', true, reload_mp);
|
||
if (!have_acode)
|
||
r.ac_flac = r.ac_aac = r.ac_oth = false;
|
||
|
||
if (IPHONE) {
|
||
ebi('au_fullpre').style.display = 'none';
|
||
r.fullpre = false;
|
||
}
|
||
|
||
ebi('au_uncache').onclick = function (e) {
|
||
ev(e);
|
||
ACB = (Date.now() % 46656).toString(36);
|
||
swrite('au_cbv', ACB);
|
||
reload_mp();
|
||
toast.inf(5, L.mm_uncache);
|
||
};
|
||
|
||
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, L.mp_breq);
|
||
};
|
||
|
||
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].getAttribute("m") == r.pb_mode);
|
||
btns[a].onclick = set_pb_mode;
|
||
}
|
||
}
|
||
draw_pb_mode();
|
||
|
||
function set_pb_mode(e) {
|
||
ev(e);
|
||
r.pb_mode = this.getAttribute('m');
|
||
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.acode = function (url) {
|
||
var c = true,
|
||
cs = url.split('?')[0];
|
||
|
||
if (!have_acode)
|
||
c = false;
|
||
else if (/\.(wav|flac)$/i.exec(cs))
|
||
c = r.ac_flac;
|
||
else if (/\.(aac|m4a)$/i.exec(cs))
|
||
c = r.ac_aac;
|
||
else if (/\.(oga|ogg|opus)$/i.exec(cs) && (!can_ogg || mpl.ac2 == 'mp3'))
|
||
c = true;
|
||
else if (re_au_native.exec(cs))
|
||
c = false;
|
||
|
||
if (!c)
|
||
return url;
|
||
|
||
return addq(url, 'th=' + r.ac2);
|
||
};
|
||
|
||
r.set_ac2 = function () {
|
||
r.init_ac2(this.getAttribute('id').split('ac2')[1]);
|
||
};
|
||
|
||
r.init_ac2 = function (v) {
|
||
if (!window.have_acode) {
|
||
r.ac2 = 'opus';
|
||
return;
|
||
}
|
||
|
||
var dv = can_ogg ? 'opus' : can_caf ? 'caf' : 'mp3',
|
||
fmts = ['opus', 'owa', 'caf', 'mp3'],
|
||
btns = [];
|
||
|
||
if (v === dv)
|
||
toast.ok(5, L.mt_c2ok);
|
||
else if (v)
|
||
toast.inf(10, L.mt_c2nd);
|
||
|
||
if ((v == 'opus' && !can_ogg) ||
|
||
(v == 'caf' && !can_caf) ||
|
||
(v == 'owa' && !can_owa))
|
||
toast.warn(15, L.mt_c2ng);
|
||
|
||
if (v == 'owa' && IPHONE)
|
||
toast.err(30, L.mt_xowa);
|
||
|
||
for (var a = 0; a < fmts.length; a++) {
|
||
var btn = ebi('ac2' + fmts[a]);
|
||
if (!btn)
|
||
return console.log('!btn', fmts[a]);
|
||
btn.onclick = r.set_ac2;
|
||
btns.push(btn);
|
||
}
|
||
if (!IPHONE)
|
||
btns[1].style.display = btns[2].style.display = 'none';
|
||
|
||
if (v)
|
||
swrite('acode2', v);
|
||
else
|
||
v = dv;
|
||
|
||
v = sread('acode2', fmts) || v;
|
||
for (var a = 0; a < fmts.length; a++)
|
||
clmod(btns[a], 'on', fmts[a] == v)
|
||
|
||
r.ac2 = v;
|
||
};
|
||
|
||
r.pp = function () {
|
||
var adur, apos, playing = mp.au && !mp.au.paused;
|
||
|
||
clearTimeout(mpl.t_eplay);
|
||
|
||
clmod(ebi('np_inf'), 'playing', playing);
|
||
|
||
if (mp.au && isNum(adur = mp.au.duration) && isNum(apos = mp.au.currentTime) && apos >= 0)
|
||
ebi('np_pos').textContent = s2ms(apos);
|
||
|
||
if (!r.os_ctl)
|
||
return;
|
||
|
||
navigator.mediaSession.playbackState = playing ? "playing" : "paused";
|
||
};
|
||
|
||
function setaufollow() {
|
||
window[(r.follow ? "add" : "remove") + "EventListener"]("resize", scroll2playing);
|
||
}
|
||
setaufollow();
|
||
|
||
function announce() {
|
||
if (!r.os_ctl || !mp.au)
|
||
return;
|
||
|
||
var np = get_np()[0],
|
||
fns = np.file.split(' - '),
|
||
artist = (np.circle && np.circle != np.artist ? np.circle + ' // ' : '') + (np.artist || (fns.length > 1 ? fns[0] : '')),
|
||
title = np.title || fns.pop(),
|
||
cover = '',
|
||
tags = { title: title };
|
||
|
||
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)$/i.test(files[a].textContent)) {
|
||
cover = files[a].getAttribute('href');
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (cover) {
|
||
cover = addq(cover, 'th=j');
|
||
tags.artwork = [{ "src": cover, type: "image/jpeg" }];
|
||
}
|
||
}
|
||
|
||
ebi('np_circle').textContent = np.circle || '';
|
||
ebi('np_album').textContent = np.album || '';
|
||
ebi('np_tn').textContent = np['.tn'] || '';
|
||
ebi('np_artist').textContent = np.artist || (fns.length > 1 ? fns[0] : '');
|
||
ebi('np_title').textContent = np.title || '';
|
||
ebi('np_dur').textContent = np['.dur'] || '';
|
||
ebi('np_url').textContent = uricom_dec(get_evpath()) + np.file.split('?')[0];
|
||
if (!MOBILE && cover)
|
||
ebi('np_img').setAttribute('src', cover);
|
||
else
|
||
ebi('np_img').removeAttribute('src');
|
||
|
||
navigator.mediaSession.metadata = new MediaMetadata(tags);
|
||
navigator.mediaSession.setActionHandler('play', mplay);
|
||
navigator.mediaSession.setActionHandler('pause', mpause);
|
||
navigator.mediaSession.setActionHandler('seekbackward', r.os_seek ? function () { seek_au_rel(-10); } : null);
|
||
navigator.mediaSession.setActionHandler('seekforward', r.os_seek ? function () { seek_au_rel(10); } : null);
|
||
navigator.mediaSession.setActionHandler('previoustrack', prev_song);
|
||
navigator.mediaSession.setActionHandler('nexttrack', next_song);
|
||
r.pp();
|
||
}
|
||
r.announce = announce;
|
||
|
||
r.stop = function () {
|
||
if (!r.os_ctl)
|
||
return;
|
||
|
||
// dead code; left for debug
|
||
navigator.mediaSession.metadata = null;
|
||
navigator.mediaSession.playbackState = "paused";
|
||
|
||
var hs = 'play pause seekbackward seekforward previoustrack nexttrack'.split(/ /g);
|
||
for (var a = 0; a < hs.length; a++)
|
||
navigator.mediaSession.setActionHandler(hs[a], null);
|
||
|
||
navigator.mediaSession.setPositionState();
|
||
};
|
||
|
||
r.unbuffer = function (url) {
|
||
if (mp.au2 && (!url || mp.au2.rsrc == url)) {
|
||
mp.au2.src = mp.au2.rsrc = '';
|
||
mp.au2.ld = 0; //owa
|
||
mp.au2.load();
|
||
}
|
||
if (!url)
|
||
mpl.preload_url = null;
|
||
}
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
var za,
|
||
can_ogg = true,
|
||
can_owa = false,
|
||
can_caf = APPLE && !/ OS ([1-9]|1[01])_/.test(UA);
|
||
try {
|
||
za = new Audio();
|
||
can_ogg = za.canPlayType('audio/ogg; codecs=opus') === 'probably';
|
||
can_owa = za.canPlayType('audio/webm; codecs=opus') === 'probably';
|
||
can_caf = za.canPlayType('audio/x-caf') && can_caf; //'maybe'
|
||
}
|
||
catch (ex) { }
|
||
za = null;
|
||
|
||
if (can_owa && APPLE && / OS ([1-9]|1[0-7])_/.test(UA))
|
||
can_owa = false;
|
||
|
||
mpl.init_ac2();
|
||
|
||
|
||
var re_m3u = /\.(m3u8?)$/i;
|
||
var re_au_native = (can_ogg || have_acode) ? /\.(aac|flac|m4a|mp3|oga|ogg|opus|wav)$/i : /\.(aac|flac|m4a|mp3|wav)$/i,
|
||
re_au_all = /\.(aac|ac3|aif|aiff|alac|alaw|amr|ape|au|dfpwm|dts|flac|gsm|it|itgz|itxz|itz|m4a|mdgz|mdxz|mdz|mo3|mod|mp2|mp3|mpc|mptm|mt2|mulaw|oga|ogg|okt|opus|ra|s3m|s3gz|s3xz|s3z|tak|tta|ulaw|wav|wma|wv|xm|xmgz|xmxz|xmz|xpk|3gp|asf|avi|flv|m4v|mkv|mov|mp4|mpeg|mpeg2|mpegts|mpg|mpg2|nut|ogm|ogv|rm|ts|vob|webm|wmv)$/i;
|
||
|
||
|
||
// extract songs + add play column
|
||
var mpo = { "au": null, "au2": null, "acs": null, "fau": null };
|
||
function MPlayer() {
|
||
var r = this;
|
||
r.id = Date.now();
|
||
r.au = mpo.au;
|
||
r.au2 = mpo.au2;
|
||
r.acs = mpo.acs;
|
||
r.fau = mpo.fau;
|
||
r.tracks = {};
|
||
r.order = [];
|
||
r.cd_pause = 0;
|
||
|
||
var re_audio = have_acode && mpl.ac_oth ? re_au_all : re_au_native,
|
||
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'),
|
||
fn = url.split('?')[0];
|
||
|
||
if (re_audio.exec(fn)) {
|
||
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">' + L.mb_play + '</a></td>';
|
||
ebi('a' + tid).onclick = ev_play;
|
||
clmod(trs[a], 'au', 1);
|
||
}
|
||
else if (re_m3u.exec(fn)) {
|
||
var tid = link.getAttribute('id');
|
||
tds[0].innerHTML = '<a id="a' + tid + '" href="#a' + tid + '" class="play">' + L.mb_play + '</a></td>';
|
||
ebi('a' + tid).onclick = ev_load_m3u;
|
||
}
|
||
}
|
||
|
||
r.vol = clamp(fcfg_get('vol', IPHONE ? 1 : dvol / 100), 0, 1);
|
||
|
||
r.expvol = function (v) {
|
||
return 0.5 * v + 0.5 * v * v;
|
||
};
|
||
|
||
r.setvol = function (vol) {
|
||
r.vol = clamp(vol, 0, 1);
|
||
swrite('vol', vol);
|
||
r.stopfade(true);
|
||
|
||
if (r.au)
|
||
r.au.volume = r.expvol(r.vol);
|
||
};
|
||
|
||
r.shuffle = function () {
|
||
if (!mpl.shuf)
|
||
return;
|
||
|
||
// durstenfeld
|
||
for (var a = r.order.length - 1; a > 0; a--) {
|
||
var b = Math.floor(Math.random() * (a + 1)),
|
||
c = r.order[a];
|
||
r.order[a] = r.order[b];
|
||
r.order[b] = c;
|
||
}
|
||
};
|
||
r.shuffle();
|
||
|
||
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.shuffle();
|
||
};
|
||
|
||
r.fdir = 0;
|
||
r.fvol = -1;
|
||
r.ftid = -1;
|
||
r.ftimer = null;
|
||
r.fade_in = function () {
|
||
r.nopause();
|
||
r.fvol = 0;
|
||
r.fdir = 0.025 * r.vol * (CHROME ? 1.5 : 1);
|
||
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.vol * (CHROME ? 2 : 1);
|
||
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 / (r.fdir < 0 && r.fvol < r.vol / 4 ? 2 : 1);
|
||
if (r.fvol < 0) {
|
||
r.fvol = 0;
|
||
r.au.pause();
|
||
mpl.pp();
|
||
|
||
var t = r.au.currentTime - 0.8;
|
||
if (isNum(t))
|
||
r.au.currentTime = Math.max(t, 0);
|
||
}
|
||
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, full) {
|
||
var t0 = Date.now(),
|
||
fname = uricom_dec(url.split('/').pop().split('?')[0]);
|
||
|
||
url = addq(mpl.acode(url), 'cache=987&_=' + ACB);
|
||
mpl.preload_url = full ? url : null;
|
||
|
||
if (mpl.waves)
|
||
fetch(url.replace(/\bth=(opus|mp3)&/, '') + '&th=p').then(function (x) {
|
||
x.body.getReader().read();
|
||
});
|
||
|
||
if (full)
|
||
return fetch(url).then(function (x) {
|
||
var rd = x.body.getReader(), n = 0;
|
||
function spd() {
|
||
return humansize(n / ((Date.now() + 1 - t0) / 1000)) + '/s';
|
||
}
|
||
function drop(x) {
|
||
if (x && x.done)
|
||
return console.log('xhr-preload finished, ' + spd());
|
||
|
||
if (x && x.value && x.value.length)
|
||
n += x.value.length;
|
||
|
||
if (mpl.preload_url !== url || n >= 128 * 1024 * 1024) {
|
||
console.log('xhr-preload aborted at ' + Math.floor(n / 1024) + ' KiB, ' + spd() + ' for ' + url);
|
||
return rd.cancel();
|
||
}
|
||
|
||
return rd.read().then(drop);
|
||
}
|
||
drop();
|
||
});
|
||
|
||
r.nopause();
|
||
r.au2.ld = 0; //owa
|
||
r.au2.onloadeddata = r.au2.onloadedmetadata = r.onpreload;
|
||
r.au2.preload = "auto";
|
||
r.au2.src = r.au2.rsrc = url;
|
||
|
||
if (mpl.prescan_evp) {
|
||
mpl.prescan_evp = null;
|
||
toast.ok(7, L.mm_scank + "\n" + esc(fname));
|
||
}
|
||
console.log("preloading " + fname);
|
||
};
|
||
|
||
r.nopause = function () {
|
||
r.cd_pause = Date.now();
|
||
};
|
||
|
||
r.onpreload = function () {
|
||
r.nopause();
|
||
this.ld++;
|
||
};
|
||
|
||
r.init_fau = function () {
|
||
if (r.fau || !mpl.fau)
|
||
return;
|
||
|
||
// breaks touchbar-macs
|
||
console.log('init fau');
|
||
r.fau = new Audio(SR + '/.cpr/deps/busy.mp3?_=' + TS);
|
||
r.fau.loop = true;
|
||
r.fau.play();
|
||
};
|
||
|
||
r.set_ev = function () {
|
||
mp.au.onended = evau_end;
|
||
mp.au.onerror = evau_error;
|
||
mp.au.onprogress = pbar.drawpos;
|
||
mp.au.onplaying = mpui.progress_updater;
|
||
mp.au.onloadeddata = mp.au.onloadedmetadata = mp.nopause;
|
||
};
|
||
}
|
||
|
||
|
||
function ft2dict(tr, skip) {
|
||
var th = ebi('files').tHead.rows[0].cells,
|
||
rv = [],
|
||
rh = [],
|
||
ra = [],
|
||
rt = {};
|
||
|
||
skip = skip || {};
|
||
|
||
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('/').pop().toLowerCase(),
|
||
vis = th[a].className.indexOf('min') === -1;
|
||
|
||
if (!tv || skip[tk])
|
||
continue;
|
||
|
||
(vis ? rv : rh).push(tk);
|
||
ra.push(tk);
|
||
rt[tk] = tv;
|
||
}
|
||
return [rt, rv, rh, ra];
|
||
}
|
||
|
||
|
||
function get_np() {
|
||
var tr = QS('#files tr.play');
|
||
return ft2dict(tr, { 'up_ip': 1 });
|
||
};
|
||
|
||
|
||
// toggle player widget
|
||
var widget = (function () {
|
||
var r = {},
|
||
widget = ebi('widget'),
|
||
wtico = ebi('wtico'),
|
||
nptxt = ebi('nptxt'),
|
||
npirc = ebi('npirc'),
|
||
m3ua = ebi('m3ua'),
|
||
m3uc = ebi('m3uc'),
|
||
touchmode = false,
|
||
was_paused = true;
|
||
|
||
r.open = function () {
|
||
return r.set(true);
|
||
};
|
||
r.close = function () {
|
||
return r.set(false);
|
||
};
|
||
r.set = function (is_open) {
|
||
if (r.is_open == is_open)
|
||
return false;
|
||
|
||
clmod(document.documentElement, 'np_open', is_open);
|
||
clmod(widget, 'open', is_open);
|
||
bcfg_set('au_open', r.is_open = is_open);
|
||
if (vbar) {
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
}
|
||
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 ? '▶' : '⏸';
|
||
}
|
||
};
|
||
r.setvis = function () {
|
||
widget.style.display = !has(perms, "read") || showfile.abrt ? 'none' : '';
|
||
};
|
||
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]).replace(/^\./, '') + '(' + cv + np[npk[a]] + ck + ') // ';
|
||
|
||
m += '[' + cv + s2ms(mp.au.currentTime) + ck + '/' + cv + s2ms(mp.au.duration) + ck + ']';
|
||
|
||
cliptxt(m, function () {
|
||
toast.ok(1, L.clipped, null, 'top');
|
||
});
|
||
};
|
||
m3ua.onclick = function (e) {
|
||
ev(e);
|
||
var el,
|
||
files = [],
|
||
sel = msel.getsel();
|
||
|
||
for (var a = 0; a < sel.length; a++) {
|
||
el = ebi(sel[a].id).closest('tr');
|
||
if (clgot(el, 'au'))
|
||
files.push(el);
|
||
}
|
||
el = QS('#files tr.play');
|
||
if (!sel.length && el)
|
||
files.push(el);
|
||
|
||
for (var a = 0; a < files.length; a++) {
|
||
var md = ft2dict(files[a])[0],
|
||
dur = md['.dur'] || '1',
|
||
tag = '';
|
||
|
||
if (md.artist && md.title)
|
||
tag = md.artist + ' - ' + md.title;
|
||
else if (md.artist)
|
||
tag = md.artist + ' - ' + md.file;
|
||
else if (md.title)
|
||
tag = md.title;
|
||
|
||
if (dur.indexOf(':') > 0) {
|
||
dur = dur.split(':');
|
||
dur = 60 * parseInt(dur[0]) + parseInt(dur[1]);
|
||
}
|
||
else dur = parseInt(dur);
|
||
|
||
mpl.m3ut += '#EXTINF:' + dur + ',' + tag + '\n' + uricom_dec(get_evpath()) + md.file + '\n';
|
||
}
|
||
toast.ok(2, files.length == 1 ? L.m3u_add1 : L.m3u_addn.format(files.length), null, 'top');
|
||
};
|
||
m3uc.onclick = function (e) {
|
||
ev(e);
|
||
cliptxt(mpl.m3ut, function () {
|
||
toast.ok(15, L.m3u_clip, null, 'top');
|
||
});
|
||
};
|
||
r.set(sread('au_open') == 1);
|
||
setTimeout(function () {
|
||
clmod(widget, 'anim', 1);
|
||
}, 10);
|
||
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 = {},
|
||
bau = null,
|
||
html_txt = 'a',
|
||
lastmove = 0,
|
||
mousepos = 0,
|
||
t_redraw = 0,
|
||
gradh = -1,
|
||
grad;
|
||
|
||
r.onresize = function () {
|
||
if (!widget.is_open && r.buf)
|
||
return;
|
||
|
||
r.buf = canvas_cfg(ebi('barbuf'));
|
||
r.pos = canvas_cfg(ebi('barpos'));
|
||
r.buf.ctx.font = '.5em sans-serif';
|
||
r.pos.ctx.font = '.9em sans-serif';
|
||
r.pos.ctx.strokeStyle = 'rgba(24,56,0,0.5)';
|
||
r.drawbuf();
|
||
r.drawpos();
|
||
if (!r.pos.can.onmouseleave)
|
||
mleave();
|
||
};
|
||
|
||
r.loadwaves = function (url) {
|
||
r.wurl = url;
|
||
var img = new Image();
|
||
img.onload = function () {
|
||
if (r.wurl != url)
|
||
return;
|
||
|
||
r.wimg = img;
|
||
r.onresize();
|
||
};
|
||
img.src = url;
|
||
};
|
||
|
||
r.unwave = function () {
|
||
r.wurl = r.wimg = null;
|
||
}
|
||
|
||
function mmove(e) {
|
||
var adur;
|
||
if (e.buttons || !mp || !mp.au || !isNum(adur = mp.au.duration))
|
||
return;
|
||
|
||
var rect = r.pos.can.getBoundingClientRect(),
|
||
x = e.clientX - rect.left,
|
||
mul = x * 1.0 / rect.width;
|
||
|
||
mousepos = adur * mul;
|
||
lastmove = Date.now();
|
||
r.drawpos();
|
||
}
|
||
function menter() {
|
||
r.pos.can.onmousemove = mmove;
|
||
r.pos.can.onmouseleave = mleave;
|
||
}
|
||
function mleave() {
|
||
r.pos.can.onmousemove = null;
|
||
r.pos.can.onmouseleave = null;
|
||
r.pos.can.onmouseenter = menter;
|
||
if (lastmove) {
|
||
lastmove = 0;
|
||
r.drawpos();
|
||
}
|
||
}
|
||
|
||
r.drawbuf = function () {
|
||
var bc = r.buf,
|
||
pc = r.pos,
|
||
bctx = bc.ctx,
|
||
apos, adur;
|
||
|
||
if (!widget.is_open)
|
||
return;
|
||
|
||
bctx.clearRect(0, 0, bc.w, bc.h);
|
||
|
||
if (!mp || !mp.au || !isNum(adur = mp.au.duration) || !isNum(apos = mp.au.currentTime) || apos < 0 || adur < apos)
|
||
return; // not-init || unsupp-codec
|
||
|
||
bau = mp.au;
|
||
|
||
var sm = bc.w * 1.0 / mp.au.duration,
|
||
gk = bc.h + '/' + themen,
|
||
dz = themen == 'dz',
|
||
dy = themen == 'dy';
|
||
|
||
if (gradh != gk) {
|
||
gradh = gk;
|
||
grad = glossy_grad(bc, dz ? 120 : 85,
|
||
dy ? [0, 0, 0, 0] : [35, 40, 37, 35],
|
||
dy ? [20, 24, 22, 20] : 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);
|
||
}
|
||
if (r.wimg) {
|
||
bctx.globalAlpha = 0.6;
|
||
bctx.filter = light ? '' : 'invert(1)';
|
||
bctx.drawImage(r.wimg, 0, 0, bc.w, bc.h);
|
||
bctx.filter = 'invert(0)';
|
||
bctx.globalAlpha = 1;
|
||
}
|
||
|
||
var step = sm > 1 ? 1 : sm > 0.4 ? 3 : sm > 0.05 ? 30 : 720;
|
||
bctx.fillStyle = light && !dy ? 'rgba(0,64,0,0.15)' : 'rgba(204,255,128,0.15)';
|
||
for (var p = step, mins = adur / 10; p <= mins; p += step)
|
||
bctx.fillRect(Math.floor(sm * p * 10), 0, 2, pc.h);
|
||
|
||
step = sm > 0.15 ? 1 : sm > 0.05 ? 10 : 360;
|
||
bctx.fillStyle = light && !dy ? 'rgba(0,64,0,0.5)' : 'rgba(192,255,96,0.5)';
|
||
for (var p = step, mins = adur / 60; p <= mins; p += step)
|
||
bctx.fillRect(Math.floor(sm * p * 60), 0, 2, pc.h);
|
||
|
||
step = sm > 0.33 ? 1 : sm > 0.15 ? 5 : sm > 0.05 ? 10 : sm > 0.01 ? 60 : 720;
|
||
bctx.fillStyle = dz ? '#0f0' : dy ? '#999' : light ? 'rgba(0,64,0,0.9)' : 'rgba(192,255,96,1)';
|
||
for (var p = step, mins = adur / 60; p <= mins; p += step) {
|
||
bctx.fillText(p, Math.floor(sm * p * 60 + 3), pc.h / 3);
|
||
}
|
||
|
||
step = sm > 0.2 ? 10 : sm > 0.1 ? 30 : sm > 0.01 ? 60 : sm > 0.005 ? 720 : 1440;
|
||
bctx.fillStyle = light ? 'rgba(0,0,0,1)' : 'rgba(255,255,255,1)';
|
||
for (var p = step, mins = adur / 60; p <= mins; p += step)
|
||
bctx.fillRect(Math.floor(sm * p * 60), 0, 2, pc.h);
|
||
};
|
||
|
||
r.drawpos = function () {
|
||
var bc = r.buf,
|
||
pc = r.pos,
|
||
pctx = pc.ctx,
|
||
w = 8,
|
||
apos, adur;
|
||
|
||
if (t_redraw) {
|
||
clearTimeout(t_redraw);
|
||
t_redraw = 0;
|
||
}
|
||
pctx.clearRect(0, 0, pc.w, pc.h);
|
||
|
||
if (!mp || !mp.au)
|
||
return; // not-init
|
||
|
||
if (!isNum(adur = mp.au.duration) || !isNum(apos = mp.au.currentTime) || apos < 0 || adur < apos) {
|
||
if (Date.now() - mp.au.pt0 < 500)
|
||
return;
|
||
|
||
pctx.fillStyle = light ? 'rgba(0,0,0,0.5)' : 'rgba(255,255,255,0.5)';
|
||
var m = /[?&]th=(opus|owa|caf|mp3)/.exec('' + mp.au.rsrc),
|
||
txt = mp.au.ded ? L.mm_playerr.replace(':', ' ;_;') :
|
||
m ? L.mm_bconv.format(m[1]) : L.mm_bload;
|
||
|
||
pctx.fillText(txt, 16, pc.h / 1.5);
|
||
return; // not-init || unsupp-codec
|
||
}
|
||
|
||
if (bau != mp.au)
|
||
r.drawbuf();
|
||
|
||
if (Date.now() - lastmove < 400) {
|
||
apos = mousepos;
|
||
w = 0;
|
||
}
|
||
|
||
var sm = bc.w * 1.0 / adur,
|
||
t1 = s2ms(adur),
|
||
t2 = s2ms(apos),
|
||
x = sm * apos;
|
||
|
||
if (w && html_txt != t2) {
|
||
ebi('np_pos').textContent = html_txt = t2;
|
||
if (mpl.os_ctl)
|
||
navigator.mediaSession.setPositionState({
|
||
'duration': adur,
|
||
'position': apos,
|
||
'playbackRate': 1
|
||
});
|
||
}
|
||
|
||
if (!widget.is_open)
|
||
return;
|
||
|
||
pctx.fillStyle = '#573'; pctx.fillRect((x - w / 2) - 1, 0, w + 2, pc.h);
|
||
pctx.fillStyle = '#dfc'; pctx.fillRect((x - w / 2), 0, w, pc.h);
|
||
|
||
pctx.lineWidth = 2.5;
|
||
pctx.fillStyle = '#fff';
|
||
|
||
var m1 = pctx.measureText(t1),
|
||
m1b = pctx.measureText(t1 + ":88"),
|
||
m2 = pctx.measureText(t2),
|
||
yt = pc.h * 0.94,
|
||
xt1 = pc.w - (m1.width + 12),
|
||
xt2 = x < m1.width * 1.4 ? (x + 12) : (Math.min(pc.w - m1b.width, x - 12) - m2.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);
|
||
|
||
if (sm > 10)
|
||
t_redraw = setTimeout(r.drawpos, sm > 50 ? 20 : 50);
|
||
};
|
||
|
||
onresize100.add(r.onresize, true);
|
||
return r;
|
||
})();
|
||
|
||
|
||
// volume bar
|
||
var vbar = (function () {
|
||
var r = {},
|
||
gradh = -1,
|
||
lastv = -1,
|
||
untext = -1,
|
||
can, ctx, w, h, grad1, grad2;
|
||
|
||
r.onresize = function () {
|
||
if (!widget.is_open && r.can)
|
||
return;
|
||
|
||
r.can = canvas_cfg(ebi('pvol'));
|
||
can = r.can.can;
|
||
ctx = r.can.ctx;
|
||
ctx.font = '.7em sans-serif';
|
||
ctx.fontVariantCaps = 'small-caps';
|
||
w = r.can.w;
|
||
h = r.can.h;
|
||
r.draw();
|
||
}
|
||
|
||
r.draw = function () {
|
||
if (!mp)
|
||
return;
|
||
|
||
var gh = h + '' + light,
|
||
dz = themen == 'dz',
|
||
dy = themen == 'dy';
|
||
|
||
if (gradh != gh) {
|
||
gradh = gh;
|
||
grad1 = glossy_grad(r.can, dz ? 120 : 50,
|
||
dy ? [0, 0, 0, 0] : light ? [50, 55, 52, 48] : [45, 52, 47, 43],
|
||
dy ? [20, 24, 22, 20] : light ? [54, 60, 52, 47] : [42, 51, 47, 42]);
|
||
grad2 = glossy_grad(r.can, dz ? 120 : 205,
|
||
dz ? [100, 100, 100, 100] : dy ? [0, 0, 0, 0] : [10, 15, 13, 10],
|
||
dz ? [10, 14, 12, 10] : dy ? [90, 90, 90, 90] : [16, 20, 18, 16]);
|
||
}
|
||
ctx.fillStyle = grad2; ctx.fillRect(0, 0, w, h);
|
||
ctx.fillStyle = grad1; ctx.fillRect(0, 0, w * mp.vol, h);
|
||
|
||
var vt = 'volume ' + Math.floor(mp.vol * 100),
|
||
tw = ctx.measureText(vt).width,
|
||
x = w * mp.vol - tw - 8,
|
||
li = dy;
|
||
|
||
if (mp.vol < 0.5) {
|
||
x += tw + 16;
|
||
li = !li;
|
||
}
|
||
|
||
ctx.fillStyle = li ? '#fff' : '#210';
|
||
ctx.fillText(vt, x, h / 3 * 2);
|
||
|
||
clearTimeout(untext);
|
||
untext = setTimeout(r.draw, 1000);
|
||
};
|
||
onresize100.add(r.onresize, true);
|
||
|
||
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;
|
||
|
||
lastv = Date.now();
|
||
mp.setvol(mul);
|
||
r.draw();
|
||
|
||
setTimeout(function () {
|
||
if (IPHONE && mp.au && mul < 0.9 && mp.au.volume == 1)
|
||
toast.inf(6, 'volume doesnt work because <a href="https://developer.apple.com/library/archive/documentation/AudioVideo/Conceptual/Using_HTML5_Audio_Video/Device-SpecificConsiderations/Device-SpecificConsiderations.html#//apple_ref/doc/uid/TP40009523-CH5-SW11" target="_blank">apple says no</a>');
|
||
}, 1);
|
||
}
|
||
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 (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 (!isNum(seek))
|
||
return;
|
||
|
||
mp.nopause();
|
||
mp.au.currentTime = seek;
|
||
|
||
if (mp.au.paused)
|
||
mp.fade_in();
|
||
|
||
mpui.progress_updater();
|
||
}
|
||
|
||
|
||
function song_skip(n, dirskip) {
|
||
var tid = mp.au && mp.au.evp == get_evpath() ? mp.au.tid : null,
|
||
ofs = tid ? mp.order.indexOf(tid) : -1;
|
||
|
||
if (dirskip && ofs + 1 && ofs > mp.order.length - 2) {
|
||
toast.inf(10, L.mm_nof);
|
||
console.log("mm_nof1");
|
||
mpl.traversals = 0;
|
||
return;
|
||
}
|
||
|
||
if (tid && !dirskip)
|
||
play(ofs + n);
|
||
else
|
||
play(mp.order[n == -1 ? mp.order.length - 1 : 0]);
|
||
}
|
||
function next_song(e) {
|
||
ev(e);
|
||
if (mp.order.length) {
|
||
var dirskip = mpl.traversals;
|
||
mpl.traversals = 0;
|
||
return song_skip(1, dirskip);
|
||
}
|
||
if (mpl.traversals++ < 5) {
|
||
treectl.ls_cb = next_song;
|
||
return tree_neigh(1);
|
||
}
|
||
toast.inf(10, L.mm_nof);
|
||
console.log("mm_nof2");
|
||
mpl.traversals = 0;
|
||
}
|
||
function last_song(e) {
|
||
ev(e);
|
||
if (mp.order.length) {
|
||
mpl.traversals = 0;
|
||
return song_skip(-1, true);
|
||
}
|
||
if (mpl.traversals++ < 5) {
|
||
treectl.ls_cb = last_song;
|
||
return tree_neigh(-1);
|
||
}
|
||
toast.inf(10, L.mm_nof);
|
||
console.log("mm_nof2");
|
||
mpl.traversals = 0;
|
||
}
|
||
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 dl_song() {
|
||
if (!mp || !mp.au) {
|
||
var o = QSA('#files a[id]');
|
||
for (var a = 0; a < o.length; a++)
|
||
o[a].setAttribute('download', '');
|
||
|
||
return toast.inf(10, L.f_dls);
|
||
}
|
||
|
||
var url = addq(mp.au.osrc, 'cache=987&_=' + ACB);
|
||
dl_file(url);
|
||
}
|
||
function sel_song() {
|
||
var o = QS('#files tr.play');
|
||
if (!o)
|
||
return;
|
||
clmod(o, 'sel', 't');
|
||
msel.origin_tr(o);
|
||
msel.selui();
|
||
}
|
||
|
||
|
||
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();
|
||
};
|
||
|
||
|
||
function mplay(e) {
|
||
if (mp.au && !mp.au.paused)
|
||
return;
|
||
|
||
playpause(e);
|
||
}
|
||
|
||
|
||
function mpause(e) {
|
||
if (mp.cd_pause > Date.now() - 100)
|
||
return;
|
||
|
||
if (mp.au && mp.au.paused)
|
||
return;
|
||
|
||
playpause(e);
|
||
}
|
||
|
||
|
||
// 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 (!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);
|
||
};
|
||
ebi('pvol').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;
|
||
|
||
dist *= -1;
|
||
mp.setvol(mp.vol + dist / 500);
|
||
vbar.draw();
|
||
ev(e);
|
||
};
|
||
}
|
||
})();
|
||
|
||
|
||
// periodic tasks
|
||
var mpui = (function () {
|
||
var r = {},
|
||
nth = 0,
|
||
preloaded = null,
|
||
fpreloaded = null;
|
||
|
||
r.progress_updater = function () {
|
||
//console.trace();
|
||
timer.add(updater_impl, true);
|
||
};
|
||
|
||
function repreload() {
|
||
preloaded = fpreloaded = null;
|
||
}
|
||
|
||
function updater_impl() {
|
||
if (!mp.au) {
|
||
widget.paused(true);
|
||
timer.rm(updater_impl);
|
||
return;
|
||
}
|
||
|
||
var paint = !MOBILE || document.hasFocus();
|
||
|
||
var pos = mp.au.currentTime;
|
||
if (!isNum(pos))
|
||
pos = 0;
|
||
|
||
// indicate playback state in ui
|
||
widget.paused(mp.au.paused);
|
||
|
||
if (paint && ++nth > 69) {
|
||
// android-chrome breaks aspect ratio with unannounced viewport changes
|
||
nth = 0;
|
||
if (MOBILE) {
|
||
nth = 1;
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
}
|
||
}
|
||
else if (paint) {
|
||
// 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.one && mpl.preload && preloaded != mp.au.rsrc) {
|
||
var len = mp.au.duration,
|
||
rem = pos > 1 ? len - pos : 999,
|
||
full = null;
|
||
|
||
if (rem < 7 || (!mpl.fullpre && (rem < 40 || (rem < 90 && pos > 10)))) {
|
||
preloaded = fpreloaded = mp.au.rsrc;
|
||
full = false;
|
||
}
|
||
else if (rem < 60 && mpl.fullpre && fpreloaded != mp.au.rsrc) {
|
||
fpreloaded = mp.au.rsrc;
|
||
full = true;
|
||
}
|
||
|
||
if (full !== null)
|
||
try {
|
||
var oi = mp.order.indexOf(mp.au.tid) + 1,
|
||
evp = get_evpath();
|
||
|
||
if (oi >= mp.order.length && (
|
||
mpl.one ||
|
||
mpl.pb_mode != 'next' ||
|
||
mp.au.evp != evp ||
|
||
ebi('unsearch'))
|
||
)
|
||
oi = 0;
|
||
|
||
if (oi >= mp.order.length) {
|
||
if (!mpl.prescan)
|
||
throw "prescan disabled";
|
||
|
||
if (mpl.prescan_evp == evp)
|
||
throw "evp match";
|
||
|
||
if (mpl.traversals++ > 4) {
|
||
mpl.prescan_evp = null;
|
||
toast.inf(10, L.mm_nof);
|
||
throw L.mm_nof;
|
||
}
|
||
|
||
mpl.prescan_evp = evp;
|
||
toast.inf(10, L.mm_prescan);
|
||
treectl.ls_cb = repreload;
|
||
tree_neigh(1);
|
||
}
|
||
else
|
||
mp.preload(mp.tracks[mp.order[oi]], full);
|
||
}
|
||
catch (ex) {
|
||
console.log("preload failed", ex);
|
||
}
|
||
}
|
||
|
||
if (mp.au.paused)
|
||
timer.rm(updater_impl);
|
||
}
|
||
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;
|
||
}
|
||
|
||
|
||
var actx = null;
|
||
|
||
function start_actx() {
|
||
// bonus: speedhack for unfocused file hashing (removes 1sec delay on subtle.digest resolves)
|
||
if (!actx) {
|
||
if (!ACtx)
|
||
return;
|
||
|
||
actx = new ACtx();
|
||
console.log('actx created');
|
||
}
|
||
try {
|
||
if (actx.state == 'suspended') {
|
||
actx.resume();
|
||
setTimeout(function () {
|
||
console.log('actx is ' + actx.state);
|
||
}, 500);
|
||
}
|
||
}
|
||
catch (ex) {
|
||
console.log('actx start failed; ' + ex);
|
||
}
|
||
}
|
||
|
||
var afilt = (function () {
|
||
var r = {
|
||
"eqen": false,
|
||
"drcen": 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],
|
||
"drcv": [-24, 30, 12, 0.01, 0.25],
|
||
"drch": ['tresh', 'knee', 'ratio', 'atk', 'rls'],
|
||
"drck": ['threshold', 'knee', 'ratio', 'attack', 'release'],
|
||
"drcn": null,
|
||
"filters": [],
|
||
"filterskip": [],
|
||
"plugs": [],
|
||
"amp": 0,
|
||
"chw": 1,
|
||
"last_au": null,
|
||
"acst": {}
|
||
};
|
||
|
||
function setvis(vis) {
|
||
ebi('audio_eq').parentNode.style.display = ebi('audio_drc').parentNode.style.display = (vis ? '' : 'none');
|
||
}
|
||
|
||
setvis(ACtx);
|
||
|
||
r.init = function () {
|
||
start_actx();
|
||
if (r.cfg)
|
||
return;
|
||
|
||
setvis(actx);
|
||
|
||
// some browsers have insane high-frequency boost
|
||
// (or rather the actual problem is Q but close enough)
|
||
r.cali = (function () {
|
||
try {
|
||
var fi = actx.createBiquadFilter(),
|
||
freqs = new Float32Array(1),
|
||
mag = new Float32Array(1),
|
||
phase = new Float32Array(1);
|
||
|
||
freqs[0] = 14000;
|
||
fi.type = 'peaking';
|
||
fi.frequency.value = 18000;
|
||
fi.Q.value = 0.8;
|
||
fi.gain.value = 1;
|
||
fi.getFrequencyResponse(freqs, mag, phase);
|
||
|
||
return mag[0]; // 1.0407 good, 1.0563 bad
|
||
}
|
||
catch (ex) {
|
||
return 0;
|
||
}
|
||
})();
|
||
console.log('eq cali: ' + r.cali);
|
||
|
||
var e1 = r.cali < 1.05;
|
||
|
||
r.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, e1 ? 1.24 : 1.2],
|
||
[16000 * 0.89, 0.7, e1 ? 1.26 : 1.2], // peak
|
||
[16000 * 1.13, 0.82, e1 ? 1.09 : 0.75], // peak
|
||
[16000 * 1.205, 0, e1 ? 1.9 : 1.85] // shelf
|
||
];
|
||
};
|
||
|
||
try {
|
||
r.amp = fcfg_get('au_eq_amp', r.amp);
|
||
r.chw = fcfg_get('au_eq_chw', r.chw);
|
||
var gains = jread('au_eq_gain', r.gains);
|
||
if (r.gains.length == gains.length)
|
||
r.gains = gains;
|
||
|
||
r.drcv = jread('au_drcv', r.drcv);
|
||
}
|
||
catch (ex) { }
|
||
|
||
r.draw = function () {
|
||
jwrite('au_eq_gain', r.gains);
|
||
swrite('au_eq_amp', r.amp);
|
||
swrite('au_eq_chw', r.chw);
|
||
|
||
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;
|
||
QS('input.eq_gain[band="chw"]').value = r.chw;
|
||
};
|
||
|
||
r.stop = function () {
|
||
if (r.filters.length)
|
||
for (var a = 0; a < r.filters.length; a++)
|
||
r.filters[a].disconnect();
|
||
|
||
r.filters = [];
|
||
r.filterskip = [];
|
||
|
||
for (var a = 0; a < r.plugs.length; a++)
|
||
r.plugs[a].unload();
|
||
|
||
if (!mp)
|
||
return;
|
||
|
||
if (mp.acs)
|
||
mp.acs.disconnect();
|
||
|
||
mp.acs = mpo.acs = null;
|
||
};
|
||
|
||
r.apply = function (v, au) {
|
||
r.init();
|
||
r.draw();
|
||
|
||
if (!actx) {
|
||
bcfg_set('au_eq', r.eqen = false);
|
||
bcfg_set('au_drc', r.drcen = false);
|
||
}
|
||
else if (v === true && r.drcen && !r.eqen)
|
||
bcfg_set('au_eq', r.eqen = true);
|
||
else if (v === false && !r.eqen)
|
||
bcfg_set('au_drc', r.drcen = false);
|
||
|
||
r.drcn = null;
|
||
|
||
var plug = false;
|
||
for (var a = 0; a < r.plugs.length; a++)
|
||
if (r.plugs[a].en)
|
||
plug = true;
|
||
|
||
au = au || (mp && mp.au);
|
||
if (!actx || !au || (!r.eqen && !plug && !mp.acs))
|
||
return;
|
||
|
||
r.stop();
|
||
au.id = au.id || Date.now();
|
||
mp.acs = r.acst[au.id] = r.acst[au.id] || actx.createMediaElementSource(au);
|
||
|
||
if (r.eqen)
|
||
add_eq();
|
||
|
||
for (var a = 0; a < r.plugs.length; a++)
|
||
if (r.plugs[a].en)
|
||
r.plugs[a].load();
|
||
|
||
for (var a = 0; a < r.filters.length; a++)
|
||
if (!has(r.filterskip, a))
|
||
r.filters[a].connect(a ? r.filters[a - 1] : actx.destination);
|
||
|
||
mp.acs.connect(r.filters.length ?
|
||
r.filters[r.filters.length - 1] : actx.destination);
|
||
}
|
||
|
||
function add_eq() {
|
||
var min, max;
|
||
min = max = r.gains[0];
|
||
for (var a = 1; a < r.gains.length; a++) {
|
||
min = Math.min(min, r.gains[a]);
|
||
max = Math.max(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 < r.cfg.length && min != max; a++) {
|
||
var fi = actx.createBiquadFilter(), c = r.cfg[a];
|
||
fi.frequency.value = c[0];
|
||
fi.gain.value = c[2] * gains[a];
|
||
fi.Q.value = c[1];
|
||
fi.type = a == 0 ? 'lowshelf' : a == r.cfg.length - 1 ? 'highshelf' : 'peaking';
|
||
r.filters.push(fi);
|
||
}
|
||
|
||
// pregain, keep first in chain
|
||
fi = actx.createGain();
|
||
fi.gain.value = r.amp + 0.94; // +.137 dB measured; now -.25 dB and almost bitperfect
|
||
r.filters.push(fi);
|
||
|
||
// wait nevermind, drc goes first
|
||
timer.rm(showdrc);
|
||
if (r.drcen) {
|
||
fi = r.drcn = actx.createDynamicsCompressor();
|
||
for (var a = 0; a < r.drcv.length; a++)
|
||
fi[r.drck[a]].value = r.drcv[a];
|
||
|
||
if (r.drcv[3] < 0.02) {
|
||
// avoid static at decode start
|
||
fi.attack.value = 0.02;
|
||
setTimeout(function () {
|
||
try {
|
||
fi.attack.value = r.drcv[3];
|
||
}
|
||
catch (ex) { }
|
||
}, 200);
|
||
}
|
||
|
||
r.filters.push(fi);
|
||
timer.add(showdrc);
|
||
}
|
||
|
||
if (Math.round(r.chw * 25) != 25) {
|
||
var split = actx.createChannelSplitter(2),
|
||
merge = actx.createChannelMerger(2),
|
||
lg1 = actx.createGain(),
|
||
lg2 = actx.createGain(),
|
||
rg1 = actx.createGain(),
|
||
rg2 = actx.createGain(),
|
||
vg1 = 1 - (1 - r.chw) / 2,
|
||
vg2 = 1 - vg1;
|
||
|
||
console.log('chw', vg1, vg2);
|
||
|
||
merge.connect(r.filters[r.filters.length - 1]);
|
||
lg1.gain.value = rg2.gain.value = vg1;
|
||
lg2.gain.value = rg1.gain.value = vg2;
|
||
lg1.connect(merge, 0, 0);
|
||
rg1.connect(merge, 0, 0);
|
||
lg2.connect(merge, 0, 1);
|
||
rg2.connect(merge, 0, 1);
|
||
|
||
split.connect(lg1, 0);
|
||
split.connect(lg2, 0);
|
||
split.connect(rg1, 1);
|
||
split.connect(rg2, 1);
|
||
r.filterskip.push(r.filters.length);
|
||
r.filters.push(split);
|
||
mp.acs.channelCountMode = 'explicit';
|
||
}
|
||
}
|
||
|
||
function eq_step(e) {
|
||
ev(e);
|
||
var sb = this.getAttribute('band'),
|
||
band = parseInt(sb),
|
||
step = parseFloat(this.getAttribute('step'));
|
||
|
||
if (sb == 'amp')
|
||
r.amp = Math.round((r.amp + step * 0.2) * 100) / 100;
|
||
else if (sb == 'chw')
|
||
r.chw = Math.round((r.chw + step * 0.2) * 100) / 100;
|
||
else
|
||
r.gains[band] += step;
|
||
|
||
r.apply();
|
||
}
|
||
|
||
function adj_band(that, step) {
|
||
var err = false;
|
||
try {
|
||
var sb = that.getAttribute('band'),
|
||
band = parseInt(sb),
|
||
vs = that.value,
|
||
v = parseFloat(vs);
|
||
|
||
if (!isNum(v) || v + '' != vs)
|
||
throw new Error('inval band');
|
||
|
||
if (sb == 'amp')
|
||
r.amp = Math.round((v + step * 0.2) * 100) / 100;
|
||
else if (sb == 'chw')
|
||
r.chw = 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 adj_drc() {
|
||
var err = false;
|
||
try {
|
||
var n = this.getAttribute('k'),
|
||
ov = r.drcv[n],
|
||
vs = this.value,
|
||
v = parseFloat(vs);
|
||
|
||
if (!isNum(v) || v + '' != vs)
|
||
throw new Error('inval v');
|
||
|
||
if (v == ov)
|
||
return;
|
||
|
||
r.drcv[n] = v;
|
||
jwrite('au_drcv', r.drcv);
|
||
if (r.drcn)
|
||
r.drcn[r.drck[n]].value = v;
|
||
}
|
||
catch (ex) {
|
||
err = true;
|
||
}
|
||
clmod(this, '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);
|
||
}
|
||
|
||
function showdrc() {
|
||
if (!r.drcn)
|
||
return timer.rm(showdrc);
|
||
|
||
ebi('h_drc').textContent = f2f(r.drcn.reduction, 1);
|
||
}
|
||
|
||
var html = ['<table><tr><td rowspan="4">',
|
||
'<a id="au_eq" class="tgl btn" href="#" tt="' + L.mt_eq + '">' + L.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]);
|
||
vs.push(["chw", "width", r.chw]);
|
||
|
||
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" ' + NOAC + ' 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;
|
||
|
||
h2 = [];
|
||
html = ['<table><tr><td rowspan="2">',
|
||
'<a id="au_drc" class="tgl btn" href="#" tt="' + L.mt_drc + '">' + L.enable + '</a></td>'];
|
||
|
||
for (var a = 0; a < r.drch.length; a++) {
|
||
html.push('<td>' + r.drch[a] + '</td>');
|
||
h2.push('<td><input type="text" class="drc_v" ' + NOAC + ' k="' + a + '" value="' + r.drcv[a] + '" /></td>');
|
||
}
|
||
html = html.join('\n') + '</tr><tr>';
|
||
html += h2.join('\n') + '</tr><table>';
|
||
ebi('audio_drc').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;
|
||
}
|
||
txt = QSA('input.drc_v');
|
||
for (var a = 0; a < txt.length; a++)
|
||
txt[a].oninput = txt[a].onkeydown = adj_drc;
|
||
|
||
bcfg_bind(r, 'eqen', 'au_eq', false, r.apply);
|
||
bcfg_bind(r, 'drcen', 'au_drc', false, r.apply);
|
||
|
||
r.draw();
|
||
return r;
|
||
})();
|
||
|
||
|
||
// plays the tid'th audio file on the page
|
||
function play(tid, is_ev, seek) {
|
||
clearTimeout(mpl.t_eplay);
|
||
if (mp.order.length == 0)
|
||
return console.log('no audio found wait what');
|
||
|
||
if (crashed)
|
||
return;
|
||
|
||
mpl.preload_url = null;
|
||
mp.nopause();
|
||
mp.stopfade(true);
|
||
|
||
var tn = tid;
|
||
if ((tn + '').indexOf('f-') === 0) {
|
||
tn = mp.order.indexOf(tn);
|
||
if (tn < 0)
|
||
return toast.warn(10, L.mm_hnf);
|
||
}
|
||
|
||
if (tn >= mp.order.length) {
|
||
if (mpl.pb_mode == 'stop')
|
||
return;
|
||
|
||
if (mpl.pb_mode == 'loop' || ebi('unsearch')) {
|
||
tn = 0;
|
||
}
|
||
else if (mpl.pb_mode == 'next') {
|
||
treectl.ls_cb = next_song;
|
||
return tree_neigh(1);
|
||
}
|
||
}
|
||
|
||
if (tn < 0) {
|
||
if (mpl.pb_mode == 'loop') {
|
||
tn = mp.order.length - 1;
|
||
}
|
||
else if (mpl.pb_mode == 'next') {
|
||
treectl.ls_cb = last_song;
|
||
return tree_neigh(-1);
|
||
}
|
||
}
|
||
|
||
tid = mp.order[tn];
|
||
|
||
if (mp.au) {
|
||
mp.au.pause();
|
||
var el = ebi('a' + mp.au.tid);
|
||
if (el)
|
||
clmod(el, 'act');
|
||
}
|
||
else {
|
||
mp.au = new Audio();
|
||
mp.au2 = new Audio();
|
||
mp.set_ev();
|
||
widget.open();
|
||
}
|
||
mp.init_fau();
|
||
|
||
var url = addq(mpl.acode(mp.tracks[tid]), 'cache=987&_=' + ACB);
|
||
|
||
if (mp.au.rsrc == url)
|
||
mp.au.currentTime = 0;
|
||
else if (mp.au2.rsrc == url) {
|
||
var t = mp.au;
|
||
mp.au = mp.au2;
|
||
mp.au2 = t;
|
||
t.onerror = t.onprogress = t.onended = t.loop = null;
|
||
t.ld = 0; //owa
|
||
mp.set_ev();
|
||
t = mp.au.currentTime;
|
||
if (isNum(t) && t > 0.1)
|
||
mp.au.currentTime = 0;
|
||
}
|
||
else {
|
||
console.log('get ' + url.split('/').pop());
|
||
mp.au.src = mp.au.rsrc = url;
|
||
}
|
||
|
||
mp.au.osrc = mp.tracks[tid];
|
||
afilt.apply();
|
||
|
||
setTimeout(function () {
|
||
mpl.unbuffer(url);
|
||
}, 500);
|
||
|
||
mp.au.ded = 0;
|
||
mp.au.tid = tid;
|
||
mp.au.pt0 = Date.now();
|
||
mp.au.evp = get_evpath();
|
||
mp.au.volume = mp.expvol(mp.vol);
|
||
var trs = QSA('#files tr.play');
|
||
for (var a = 0, aa = trs.length; a < aa; a++)
|
||
clmod(trs[a], 'play');
|
||
|
||
var oid = 'a' + tid;
|
||
clmod(ebi(oid), 'act', 1);
|
||
clmod(ebi(oid).closest('tr'), 'play', 1);
|
||
clmod(ebi('wtoggle'), 'np', mpl.clip);
|
||
clmod(ebi('wtoggle'), 'm3u', mpl.m3uen);
|
||
if (thegrid)
|
||
thegrid.loadsel();
|
||
|
||
if (mpl.follow)
|
||
scroll2playing();
|
||
|
||
try {
|
||
mp.nopause();
|
||
mp.au.loop = mpl.loop && !mpl.one;
|
||
if (mpl.aplay || is_ev !== -1)
|
||
mp.au.play();
|
||
|
||
if (mp.au.paused)
|
||
autoplay_blocked(seek);
|
||
else if (seek) {
|
||
seek_au_sec(seek);
|
||
}
|
||
|
||
if (!seek && !ebi('unsearch')) {
|
||
var o = ebi(oid);
|
||
o.setAttribute('id', 'thx_js');
|
||
if (mpl.aplay)
|
||
sethash(oid + getsort());
|
||
o.setAttribute('id', oid);
|
||
}
|
||
|
||
pbar.unwave();
|
||
if (mpl.waves)
|
||
pbar.loadwaves(url.replace(/\bth=(opus|mp3)&/, '') + '&th=p');
|
||
|
||
mpui.progress_updater();
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
mpl.announce();
|
||
return true;
|
||
}
|
||
catch (ex) {
|
||
toast.err(0, esc(L.mm_playerr + basenames(ex)));
|
||
}
|
||
clmod(ebi(oid), 'act');
|
||
mpl.t_eplay = setTimeout(next_song, 5000);
|
||
}
|
||
|
||
|
||
function scroll2playing() {
|
||
try {
|
||
QS((!thegrid || !thegrid.en) ?
|
||
'tr.play' : '#ggrid a.play').scrollIntoView();
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
|
||
|
||
function evau_end(e) {
|
||
if (mpl.one)
|
||
return;
|
||
if (!mpl.loop)
|
||
return next_song(e);
|
||
ev(e);
|
||
mp.au.currentTime = 0;
|
||
mp.au.play();
|
||
}
|
||
|
||
|
||
// event from the audio object if something breaks
|
||
function evau_error(e) {
|
||
var err = '',
|
||
eplaya = (e && e.target) || (window.event && window.event.srcElement);
|
||
|
||
eplaya.ded = 1;
|
||
|
||
switch (eplaya.error.code) {
|
||
case eplaya.error.MEDIA_ERR_ABORTED:
|
||
err = L.mm_eabrt;
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_NETWORK:
|
||
if (IPHONE && eplaya.ld === 1 && mpl.ac2 == 'owa' && !eplaya.paused && !eplaya.currentTime) {
|
||
eplaya.ded = 0;
|
||
if (!mpl.owaw) {
|
||
mpl.owaw = 1;
|
||
console.log('ignored iOS bug; spurious error sent in parallel with preloaded songs starting to play just fine');
|
||
}
|
||
return;
|
||
}
|
||
err = L.mm_enet;
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_DECODE:
|
||
err = L.mm_edec;
|
||
break;
|
||
case eplaya.error.MEDIA_ERR_SRC_NOT_SUPPORTED:
|
||
err = L.mm_esupp;
|
||
if (/\.(aac|m4a)(\?|$)/i.exec(eplaya.rsrc) && !mpl.ac_aac) {
|
||
try {
|
||
ebi('ac_aac').click();
|
||
QS('a.play.act').click();
|
||
toast.warn(10, L.mm_opusen);
|
||
return;
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
break;
|
||
default:
|
||
err = L.mm_eunk;
|
||
break;
|
||
}
|
||
var em = '' + eplaya.error.message,
|
||
mfile = '\n\nFile: «' + uricom_dec(eplaya.src.split('/').pop()) + '»',
|
||
e500 = L.mm_e500,
|
||
e404 = L.mm_e404,
|
||
e403 = L.mm_e403;
|
||
|
||
if (em)
|
||
err += '\n\n' + em;
|
||
|
||
if (em.startsWith('403: '))
|
||
err = e403;
|
||
|
||
if (em.startsWith('404: '))
|
||
err = e404;
|
||
|
||
if (em.startsWith('500: '))
|
||
err = e500;
|
||
|
||
toast.warn(15, esc(basenames(err + mfile)));
|
||
console.log(basenames(err + mfile));
|
||
|
||
if (em.startsWith('MEDIA_ELEMENT_ERROR:')) {
|
||
// chromish for 40x
|
||
var xhr = new XHR();
|
||
xhr.open('HEAD', eplaya.src, true);
|
||
xhr.onload = xhr.onerror = function () {
|
||
if (this.status < 400)
|
||
return;
|
||
|
||
err = this.status == 403 ? e403 :
|
||
this.status == 404 ? e404 :
|
||
this.status == 500 ? e500 :
|
||
L.mm_e5xx + this.status;
|
||
|
||
toast.warn(15, esc(basenames(err + mfile)));
|
||
};
|
||
xhr.send();
|
||
return;
|
||
}
|
||
|
||
mpl.t_eplay = setTimeout(next_song, 15000);
|
||
}
|
||
|
||
|
||
// show ui to manually start playback of a linked song
|
||
function autoplay_blocked(seek) {
|
||
var tid = mp.au.tid,
|
||
fn = mp.tracks[tid].split(/\//).pop();
|
||
|
||
fn = uricom_dec(fn.replace(/\+/g, ' ').split('?')[0]);
|
||
|
||
modal.confirm('<h6>' + L.mm_hashplay + '</h6>\n«' + esc(fn) + '»', function () {
|
||
// chrome 91 may permanently taint on a failed play()
|
||
// depending on win10 settings or something? idk
|
||
mp.au = mpo.au = null;
|
||
|
||
play(tid, true, seek);
|
||
mp.fade_in();
|
||
}, function () {
|
||
sethash('');
|
||
clmod(QS('#files tr.play'), 'play');
|
||
return reload_mp();
|
||
});
|
||
}
|
||
|
||
|
||
function scan_hash(v) {
|
||
if (!v)
|
||
return null;
|
||
|
||
var m = /^#([ag])(f-[0-9a-f]{8,16})(&.+)?/.exec(v + '');
|
||
if (!m)
|
||
return null;
|
||
|
||
var mtype = m[1],
|
||
id = m[2],
|
||
ts = null;
|
||
|
||
if (m.length > 3) {
|
||
var tm = /^&[Tt=0]*([0-9]+[Mm:])?0*([0-9\.]+)[Ss]?$/.exec(m[3]);
|
||
if (tm) {
|
||
ts = parseInt(tm[1] || 0) * 60 + parseFloat(tm[2] || 0);
|
||
}
|
||
tm = /^&[Tt=0]*([0-9\.]+)-([0-9\.]+)$/.exec(m[3]);
|
||
if (tm) {
|
||
ts = '' + tm[1] + '-' + tm[2];
|
||
}
|
||
}
|
||
|
||
return [mtype, id, ts];
|
||
}
|
||
|
||
|
||
function eval_hash() {
|
||
if (!window.hotkeys_attached) {
|
||
window.hotkeys_attached = true;
|
||
document.onkeydown = ahotkeys;
|
||
window.onpopstate = treectl.onpopfun;
|
||
}
|
||
|
||
if (hash0 && window.og_fn) {
|
||
var all = msel.getall(), mi;
|
||
for (var a = 0; a < all.length; a++)
|
||
if (og_fn == uricom_dec(vsplit(all[a].vp)[1].split('?')[0])) {
|
||
mi = all[a];
|
||
break;
|
||
}
|
||
|
||
var ch = !mi ? '' :
|
||
img_re.exec(og_fn) ? 'g' :
|
||
ebi('a' + mi.id) ? 'a' :
|
||
'';
|
||
|
||
hash0 = ch ? ('#' + ch + mi.id) : '';
|
||
}
|
||
|
||
var v = hash0;
|
||
hash0 = null;
|
||
if (!v)
|
||
return;
|
||
|
||
var media = scan_hash(v);
|
||
if (media) {
|
||
var mtype = media[0],
|
||
id = media[1],
|
||
ts = media[2];
|
||
|
||
if (mtype == 'a') {
|
||
if (!ts)
|
||
return play(id, -1);
|
||
|
||
return play(id, -1, ts);
|
||
}
|
||
|
||
if (mtype == 'g') {
|
||
if (!thegrid.en)
|
||
ebi('griden').click();
|
||
|
||
var t = setInterval(function () {
|
||
if (!thegrid.bbox)
|
||
return;
|
||
|
||
clearInterval(t);
|
||
baguetteBox.urltime(ts);
|
||
var im = QS('#ggrid a[ref="' + id + '"]');
|
||
if (!im)
|
||
return toast.warn(10, L.im_hnf);
|
||
|
||
if (thegrid.sel)
|
||
setTimeout(function () {
|
||
thegrid.sel = true;
|
||
}, 1);
|
||
|
||
thegrid.sel = false;
|
||
im.click();
|
||
im.scrollIntoView();
|
||
}, 50);
|
||
}
|
||
}
|
||
|
||
if (v.startsWith('#q=')) {
|
||
goto('search');
|
||
var i = ebi('q_raw');
|
||
i.value = uricom_dec(v.slice(3));
|
||
return i.onkeydown({ 'key': 'Enter' });
|
||
}
|
||
|
||
if (v.startsWith('#v=')) {
|
||
goto(v.slice(3));
|
||
return;
|
||
}
|
||
|
||
if (v.startsWith("#m3u=")) {
|
||
load_m3u(v.slice(5));
|
||
return;
|
||
}
|
||
}
|
||
|
||
|
||
(function () {
|
||
var props = {};
|
||
|
||
// a11y jump-to-content
|
||
for (var a = 0; a < 2; a++)
|
||
(function (a) {
|
||
var d = mknod('a');
|
||
d.setAttribute('href', '#');
|
||
d.setAttribute('class', 'ayjump');
|
||
d.innerHTML = a ? L.ay_path : L.ay_files;
|
||
document.body.insertBefore(d, ebi('ops'));
|
||
d.onclick = function (e) {
|
||
ev(e);
|
||
if (a)
|
||
QS(treectl.hidden ? '#path a:nth-last-child(2)' : '#treeul a.hl').focus();
|
||
else
|
||
QS(thegrid.en ? '#ggrid a' : '#files tbody tr[tabindex]').focus();
|
||
};
|
||
})(a);
|
||
|
||
// account-info label
|
||
var d = mknod('div', 'acc_info');
|
||
document.body.insertBefore(d, ebi('ops'));
|
||
|
||
// folder nav
|
||
ebi('goh').parentElement.appendChild(mknod('span', null,
|
||
'<a href="#" id="gop" tt="' + L.gop + '</a>/<a href="#" id="gou" tt="' + L.gou + '</a>/<a href="#" id="gon" tt="' + L.gon + '</a>'));
|
||
ebi('gop').onclick = function () { tree_neigh(-1); }
|
||
ebi('gon').onclick = function () { tree_neigh(1); }
|
||
ebi('gou').onclick = function () { tree_up(true); }
|
||
|
||
// show/hide scrollbars
|
||
function setsb() {
|
||
clmod(document.documentElement, 'noscroll', !props.sbars);
|
||
}
|
||
bcfg_bind(props, 'sbars', 'sbars', true, setsb);
|
||
setsb();
|
||
|
||
// compact media player
|
||
function setacmp() {
|
||
clmod(ebi('widget'), 'cmp', props.mcmp);
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
}
|
||
bcfg_bind(props, 'mcmp', 'au_compact', false, setacmp);
|
||
setacmp();
|
||
|
||
// toggle bup checksums
|
||
ebi('uput').onchange = function() {
|
||
QS('#op_bup input[name="act"]').value = this.checked ? 'uput' : 'bput';
|
||
};
|
||
})();
|
||
|
||
|
||
function read_dsort(txt) {
|
||
dnsort = dnsort ? 1 : 0;
|
||
ENATSORT = NATSORT && (sread('nsort') || dnsort) == 1;
|
||
clmod(ebi('nsort'), 'on', ENATSORT);
|
||
try {
|
||
var zt = (('' + txt).trim() || 'href').split(/,+/g);
|
||
dsort = [];
|
||
for (var a = 0; a < zt.length; a++) {
|
||
var z = zt[a].trim(), n = 1, t = "";
|
||
if (z.startsWith("-")) {
|
||
z = z.slice(1);
|
||
n = -1;
|
||
}
|
||
if (z == "sz" || z.indexOf('/.') + 1)
|
||
t = "int";
|
||
|
||
dsort.push([z, n, t]);
|
||
}
|
||
}
|
||
catch (ex) {
|
||
toast.warn(10, 'failed to apply default sort order [' + esc('' + txt) + ']:\n' + ex);
|
||
dsort = [['href', 1, '']];
|
||
}
|
||
}
|
||
read_dsort(dsort);
|
||
|
||
|
||
function getsort() {
|
||
var ret = '',
|
||
sopts = jread('fsort');
|
||
|
||
sopts = sopts && sopts.length ? sopts : dsort;
|
||
|
||
for (var a = 0; a < Math.min(hsortn, sopts.length); a++)
|
||
ret += ',sort' + (sopts[a][1] < 0 ? '-' : '') + sopts[a][0];
|
||
|
||
return ret;
|
||
}
|
||
|
||
|
||
function sortfiles(nodes) {
|
||
if (!nodes.length)
|
||
return nodes;
|
||
|
||
var sopts = jread('fsort'),
|
||
dir1st = sread('dir1st') !== '0';
|
||
|
||
sopts = sopts && sopts.length ? sopts : jcp(dsort);
|
||
|
||
try {
|
||
var is_srch = false;
|
||
if (nodes[0]['rp']) {
|
||
is_srch = true;
|
||
for (var b = 0, bb = nodes.length; b < bb; b++)
|
||
nodes[b].ext = nodes[b].rp.split('.').pop();
|
||
for (var b = 0; b < sopts.length; b++)
|
||
if (sopts[b][0] == 'href')
|
||
sopts[b][0] = 'rp';
|
||
}
|
||
for (var a = sopts.length - 1; a >= 0; a--) {
|
||
var name = sopts[a][0], rev = sopts[a][1], typ = sopts[a][2];
|
||
if (!name)
|
||
continue;
|
||
|
||
name = name.toLowerCase();
|
||
|
||
if (name == 'ts')
|
||
typ = 'int';
|
||
|
||
if (name.indexOf('tags/') === 0) {
|
||
name = name.slice(5);
|
||
for (var b = 0, bb = nodes.length; b < bb; b++)
|
||
nodes[b]._sv = nodes[b].tags[name];
|
||
}
|
||
else {
|
||
for (var b = 0, bb = nodes.length; b < bb; b++) {
|
||
var v = nodes[b][name];
|
||
|
||
if ((v + '').indexOf('<a ') === 0)
|
||
v = v.split('>')[1];
|
||
else if (name == "href" && v)
|
||
v = uricom_dec(v);
|
||
|
||
nodes[b]._sv = v
|
||
}
|
||
}
|
||
|
||
var onodes = nodes.map(function (x) { return x; });
|
||
nodes.sort(function (n1, n2) {
|
||
var v1 = n1._sv,
|
||
v2 = n2._sv;
|
||
|
||
if (v1 === undefined) {
|
||
if (v2 === undefined) {
|
||
return onodes.indexOf(n1) - onodes.indexOf(n2);
|
||
}
|
||
return -1 * rev;
|
||
}
|
||
if (v2 === undefined) return 1 * rev;
|
||
|
||
var ret = rev * (typ == 'int' ? (v1 - v2) :
|
||
ENATSORT ? NATSORT.compare(v1, v2) :
|
||
v1.localeCompare(v2));
|
||
|
||
if (ret === 0)
|
||
ret = onodes.indexOf(n1) - onodes.indexOf(n2);
|
||
|
||
return ret;
|
||
});
|
||
}
|
||
for (var b = 0, bb = nodes.length; b < bb; b++) {
|
||
delete nodes[b]._sv;
|
||
if (is_srch)
|
||
delete nodes[b].ext;
|
||
}
|
||
if (dir1st) {
|
||
var r1 = [], r2 = [];
|
||
for (var b = 0, bb = nodes.length; b < bb; b++)
|
||
(nodes[b].href.split('?')[0].slice(-1) == '/' ? r1 : r2).push(nodes[b]);
|
||
|
||
nodes = r1.concat(r2);
|
||
}
|
||
}
|
||
catch (ex) {
|
||
console.log("failed to apply sort config: " + ex);
|
||
console.log("resetting fsort " + sread('fsort'));
|
||
sdrop('fsort');
|
||
}
|
||
return nodes;
|
||
}
|
||
|
||
|
||
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'),
|
||
bcpy = ebi('fcpy'),
|
||
bpst = ebi('fpst'),
|
||
bshr = ebi('fshr'),
|
||
t_paste,
|
||
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', []).slice(1);
|
||
r.ccp = r.clip.length && r.clip[0] == '//c';
|
||
if (r.ccp)
|
||
r.clip.shift();
|
||
}
|
||
|
||
var sel = msel.getsel(),
|
||
nsel = sel.length,
|
||
enren = nsel,
|
||
endel = nsel,
|
||
encut = nsel,
|
||
encpy = nsel,
|
||
enpst = r.clip && r.clip.length,
|
||
hren = !(have_mv && has(perms, 'write') && has(perms, 'move')),
|
||
hdel = !(have_del && has(perms, 'delete')),
|
||
hcut = !(have_mv && has(perms, 'move')),
|
||
hpst = !(have_mv && has(perms, 'write')),
|
||
hshr = !(have_shr && acct != '*' && (has(perms, 'read') || has(perms, 'write')));
|
||
|
||
if (!(enren || endel || encut || enpst))
|
||
hren = hdel = hcut = hpst = true;
|
||
|
||
clmod(bren, 'en', enren);
|
||
clmod(bdel, 'en', endel);
|
||
clmod(bcut, 'en', encut);
|
||
clmod(bcpy, 'en', encpy);
|
||
clmod(bpst, 'en', enpst);
|
||
clmod(bshr, 'en', 1);
|
||
|
||
clmod(bren, 'hide', hren);
|
||
clmod(bdel, 'hide', hdel);
|
||
clmod(bcut, 'hide', hcut);
|
||
clmod(bpst, 'hide', hpst);
|
||
clmod(bshr, 'hide', hshr);
|
||
|
||
clmod(ebi('wfm'), 'act', QS('#wfm a.en:not(.hide)'));
|
||
clmod(ebi('wtoggle'), 'm3u', mpl.m3uen && (nsel || (mp && mp.au)));
|
||
|
||
var wfs = ebi('wfs'), h = '';
|
||
try {
|
||
wfs.innerHTML = h = r.fsi(sel);
|
||
}
|
||
catch (ex) { }
|
||
clmod(wfs, 'act', h);
|
||
|
||
bpst.setAttribute('tt', L.ft_paste.format(r.clip.length));
|
||
bshr.setAttribute('tt', nsel ? L.fs_ss : L.fs_sc);
|
||
};
|
||
|
||
r.fsi = function (sel) {
|
||
if (!sel.length)
|
||
return '';
|
||
|
||
var lf = treectl.lsc.files,
|
||
nf = 0,
|
||
sz = 0,
|
||
dur = 0,
|
||
ntab = new Set();
|
||
|
||
for (var a = 0; a < sel.length; a++)
|
||
ntab.add(sel[a].vp.split('/').pop());
|
||
|
||
for (var a = 0; a < lf.length; a++) {
|
||
if (!ntab.has(lf[a].href.split('?')[0]))
|
||
continue;
|
||
|
||
var f = lf[a];
|
||
nf++;
|
||
sz += f.sz;
|
||
if (f.tags && f.tags['.dur'])
|
||
dur += f.tags['.dur']
|
||
}
|
||
|
||
if (!nf)
|
||
return '';
|
||
|
||
var ret = '{0}<br />{1}<small>F</small>'.format(humansize(sz), nf);
|
||
|
||
if (dur)
|
||
ret += ' ' + s2ms(dur);
|
||
|
||
return ret;
|
||
};
|
||
|
||
r.share = function (e) {
|
||
ev(e);
|
||
|
||
var vp = uricom_dec(get_evpath()),
|
||
sel = msel.getsel(),
|
||
fns = [];
|
||
|
||
for (var a = 0; a < sel.length; a++)
|
||
fns.push(uricom_dec(noq_href(ebi(sel[a].id))));
|
||
|
||
if (fns.length == 1 && fns[0].endsWith('/'))
|
||
vp = fns.pop();
|
||
|
||
for (var a = 0; a < fns.length; a++)
|
||
if (fns[a].endsWith('/'))
|
||
return toast.err(10, L.fs_just1d);
|
||
|
||
var shui = ebi('shui');
|
||
if (!shui) {
|
||
shui = mknod('div', 'shui');
|
||
document.body.appendChild(shui);
|
||
}
|
||
shui.style.display = 'block';
|
||
|
||
var html = [
|
||
'<div>',
|
||
'<table>',
|
||
'<tr><td colspan="2">',
|
||
'<button id="sh_abrt">' + L.fs_abrt + '</button>',
|
||
'<button id="sh_rand">' + L.fs_rand + '</button>',
|
||
'<button id="sh_apply">' + L.fs_go + '</button>',
|
||
'</td></tr>',
|
||
'<tr><td>' + L.fs_name + '</td><td><input type="text" id="sh_k" ' + NOAC + ' placeholder=" ' + L.fs_pname + '" /></td></tr>',
|
||
'<tr><td>' + L.fs_src + '</td><td><input type="text" id="sh_vp" ' + NOAC + ' readonly tt="' + L.fs_tsrc + '" /></td></tr>',
|
||
'<tr><td>' + L.fs_pwd + '</td><td><input type="text" id="sh_pw" ' + NOAC + ' placeholder=" ' + L.fs_ppwd + '" /></td></tr>',
|
||
'<tr><td>' + L.fs_exp + '</td><td class="exs">',
|
||
'<input type="text" id="sh_exm" ' + NOAC + ' /> ' + L.fs_tmin + ' / ',
|
||
'<input type="text" id="sh_exh" ' + NOAC + ' /> ' + L.fs_thrs + ' / ',
|
||
'<input type="text" id="sh_exd" ' + NOAC + ' /> ' + L.fs_tdays + ' / ',
|
||
'<button id="sh_noex">' + L.fs_never + '</button>',
|
||
'</td></tr>',
|
||
'<tr><td>perms</td><td class="sh_axs">',
|
||
];
|
||
for (var a = 0; a < perms.length; a++)
|
||
if (!has(['admin', 'move'], perms[a]))
|
||
html.push('<a href="#" class="tgl btn">' + perms[a] + '</a>');
|
||
|
||
if (has(perms, 'write'))
|
||
html.push('<a href="#" class="btn">write-only</a>');
|
||
|
||
html.push('</td></tr></div');
|
||
shui.innerHTML = html.join('\n');
|
||
|
||
var sh_rand = ebi('sh_rand'),
|
||
sh_abrt = ebi('sh_abrt'),
|
||
sh_apply = ebi('sh_apply'),
|
||
sh_noex = ebi('sh_noex'),
|
||
exm = ebi('sh_exm'),
|
||
exh = ebi('sh_exh'),
|
||
exd = ebi('sh_exd'),
|
||
sh_k = ebi('sh_k'),
|
||
sh_vp = ebi('sh_vp'),
|
||
sh_pw = ebi('sh_pw');
|
||
|
||
function setexp(a, b) {
|
||
a = parseFloat(a);
|
||
if (!isNum(a))
|
||
return;
|
||
|
||
var v = a * b;
|
||
swrite('fsh_exp', v);
|
||
|
||
if (exm.value != v) exm.value = Math.round(v * 10) / 10; v /= 60;
|
||
if (exh.value != v) exh.value = Math.round(v * 10) / 10; v /= 24;
|
||
if (exd.value != v) exd.value = Math.round(v * 10) / 10;
|
||
}
|
||
function setdef() {
|
||
setexp(icfg_get('fsh_exp', 60 * 24), 1);
|
||
}
|
||
setdef();
|
||
|
||
exm.oninput = function () { setexp(this.value, 1); };
|
||
exh.oninput = function () { setexp(this.value, 60); };
|
||
exd.oninput = function () { setexp(this.value, 60 * 24); };
|
||
exm.onfocus = exh.onfocus = exd.onfocus = function () {
|
||
this.value = '';
|
||
};
|
||
sh_noex.onclick = function () {
|
||
setexp(0, 1);
|
||
};
|
||
exm.onblur = exh.onblur = exd.onblur = setdef;
|
||
|
||
exm.onkeydown = exh.onkeydown = exd.onkeydown =
|
||
sh_k.onkeydown = sh_pw.onkeydown = function (e) {
|
||
var kc = (e.code || e.key) + '';
|
||
if (kc.endsWith('Enter'))
|
||
sh_apply.click();
|
||
};
|
||
|
||
sh_abrt.onclick = function () {
|
||
shui.parentNode.removeChild(shui);
|
||
};
|
||
sh_rand.onclick = function () {
|
||
sh_k.value = randstr(12).replace(/l/g, 'n');
|
||
};
|
||
tt.att(shui);
|
||
|
||
var pbtns = QSA('#shui .sh_axs a');
|
||
for (var a = 0; a < pbtns.length; a++)
|
||
pbtns[a].onclick = shspf;
|
||
|
||
function shspf() {
|
||
clmod(this, 'on', 't');
|
||
if (this.textContent == 'write-only')
|
||
for (var a = 0; a < pbtns.length; a++)
|
||
clmod(pbtns[a], 'on', pbtns[a].textContent == 'write');
|
||
}
|
||
clmod(pbtns[0], 'on', 1);
|
||
|
||
var vpt = vp;
|
||
if (fns.length) {
|
||
vpt = fns.length + ' files in ' + vp + ' '
|
||
for (var a = 0; a < fns.length; a++)
|
||
vpt += '「' + fns[a].split('/').pop() + '」';
|
||
}
|
||
sh_vp.value = vpt;
|
||
|
||
sh_k.oninput = function (e) {
|
||
var v = this.value,
|
||
v2 = v.replace(/[^0-9a-zA-Z-]/g, '_');
|
||
|
||
if (v != v2)
|
||
this.value = v2;
|
||
};
|
||
|
||
function shr_cb() {
|
||
toast.hide();
|
||
var surl = this.responseText;
|
||
if (this.status !== 201 || !/^created share:/.exec(surl)) {
|
||
shui.style.display = 'block';
|
||
var msg = unpre(surl);
|
||
toast.err(9, msg);
|
||
return;
|
||
}
|
||
surl = surl.slice(15).trim();
|
||
var txt = esc(surl) + '<img class="b64" width="100" height="100" src="' + surl + '?qr" />';
|
||
modal.confirm(txt + L.fs_ok, function() {
|
||
cliptxt(surl, function () {
|
||
toast.ok(2, L.clipped);
|
||
});
|
||
}, null);
|
||
}
|
||
|
||
sh_apply.onclick = function () {
|
||
if (!sh_k.value)
|
||
sh_rand.click();
|
||
|
||
var plist = [];
|
||
for (var a = 0; a < pbtns.length; a++)
|
||
if (clgot(pbtns[a], 'on'))
|
||
plist.push(pbtns[a].textContent);
|
||
|
||
shui.style.display = 'none';
|
||
toast.inf(30, L.fs_w8);
|
||
|
||
var body = {
|
||
"k": sh_k.value,
|
||
"vp": fns.length ? fns : [sh_vp.value],
|
||
"pw": sh_pw.value,
|
||
"exp": exm.value,
|
||
"perms": plist,
|
||
};
|
||
var xhr = new XHR();
|
||
xhr.open('POST', SR + '/?share', true);
|
||
xhr.setRequestHeader('Content-Type', 'text/plain');
|
||
xhr.onload = xhr.onerror = shr_cb;
|
||
xhr.send(JSON.stringify(body));
|
||
};
|
||
|
||
setTimeout(sh_pw.focus.bind(sh_pw), 1);
|
||
};
|
||
|
||
r.rename = function (e) {
|
||
ev(e);
|
||
var sel = msel.getsel();
|
||
if (!sel.length)
|
||
return toast.err(3, L.fr_emore);
|
||
|
||
if (clgot(bren, 'hide'))
|
||
return toast.err(3, L.fr_eperm);
|
||
|
||
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, esc('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] = (md[k] + '').replace(/[\/\\]/g, '-');
|
||
|
||
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]),
|
||
"md": vars[0],
|
||
"ok": true
|
||
});
|
||
}
|
||
|
||
var rui = ebi('rui');
|
||
if (!rui) {
|
||
rui = mknod('div', 'rui');
|
||
document.body.appendChild(rui);
|
||
}
|
||
|
||
var html = sel.length > 1 ? ['<div>'] : [
|
||
'<div>',
|
||
'<button class="rn_dec" id="rn_dec_0" tt="' + L.frt_dec + '</button>',
|
||
'//',
|
||
'<button class="rn_reset" id="rn_reset_0" tt="' + L.frt_rst + '</button>'
|
||
];
|
||
|
||
html = html.concat([
|
||
'<button id="rn_cancel" tt="' + L.frt_abrt + '</button>',
|
||
'<button id="rn_apply">✅ ' + L.frb_apply + '</button>',
|
||
'<a id="rn_adv" class="tgl btn" href="#" tt="' + L.fr_adv + '</a>',
|
||
'<a id="rn_case" class="tgl btn" href="#" tt="' + L.fr_case + '</a>',
|
||
'<a id="rn_win" class="tgl btn" href="#" tt="' + L.fr_win + '</a>',
|
||
'<a id="rn_slash" class="tgl btn" href="#" tt="' + L.fr_slash + '</a>',
|
||
'</div>',
|
||
'<div id="rn_vadv"><table>',
|
||
'<tr><td>regex</td><td><input type="text" id="rn_re" ' + NOAC + ' tt="' + L.fr_re + '" placeholder="^[0-9]+[\\. ]+(.*) - (.*)" /></td></tr>',
|
||
'<tr><td>format</td><td><input type="text" id="rn_fmt" ' + NOAC + ' tt="' + L.fr_fmt + '" placeholder="[(artist) - ](title).(ext)" /></td></tr>',
|
||
'<tr><td>preset</td><td><select id="rn_pre"></select>',
|
||
'<button id="rn_pdel">❌ ' + L.fr_pdel + '</button>',
|
||
'<button id="rn_pnew">💾 ' + L.fr_pnew + '</button>',
|
||
'</td></tr>',
|
||
'</table></div>'
|
||
]);
|
||
|
||
var cheap = f.length > 500,
|
||
t_rst = L.frt_rst.split('>').pop();
|
||
|
||
if (sel.length == 1)
|
||
html.push(
|
||
'<div><table id="rn_f">\n' +
|
||
'<tr><td>old:</td><td><input type="text" id="rn_old_0" readonly /></td></tr>\n' +
|
||
'<tr><td>new:</td><td><input type="text" id="rn_new_0" /></td></tr>');
|
||
else {
|
||
html.push(
|
||
'<div><table id="rn_f" class="m">' +
|
||
'<tr><td></td><td>' + L.fr_lnew + '</td><td>' + L.fr_lold + '</td></tr>');
|
||
for (var a = 0; a < f.length; a++)
|
||
html.push(
|
||
'<tr><td>' +
|
||
(cheap ? '</td>' :
|
||
'<button class="rn_dec" id="rn_dec_' + a + '">decode</button>' +
|
||
'<button class="rn_reset" id="rn_reset_' + a + '">' + t_rst + '</button></td>') +
|
||
'<td><input type="text" id="rn_new_' + a + '" /></td>' +
|
||
'<td><input type="text" id="rn_old_' + a + '" readonly /></td></tr>');
|
||
}
|
||
html.push('</table></div>');
|
||
|
||
if (sel.length == 1) {
|
||
html.push('<div><p style="margin:.6em 0">' + L.fr_tags + '</p><table>');
|
||
for (var a = 0; a < mkeys.length; a++)
|
||
html.push('<tr><td>' + esc(mkeys[a]) + '</td><td><input type="text" readonly value="' + esc(f[0].md[mkeys[a]]) + '" /></td></tr>');
|
||
|
||
html.push('</table></div>');
|
||
}
|
||
|
||
rui.innerHTML = html.join('\n');
|
||
for (var a = 0; a < f.length; a++) {
|
||
f[a].iold = ebi('rn_old_' + a);
|
||
f[a].inew = ebi('rn_new_' + a);
|
||
f[a].inew.value = f[a].iold.value = f[a].ofn;
|
||
|
||
if (!cheap)
|
||
(function (a) {
|
||
f[a].inew.onkeydown = function (e) {
|
||
rn_ok(a, true);
|
||
var kc = (e.code || e.key) + '';
|
||
if (kc.endsWith('Enter'))
|
||
return rn_apply();
|
||
};
|
||
ebi('rn_dec_' + a).onclick = function (e) {
|
||
ev(e);
|
||
f[a].inew.value = uricom_dec(f[a].inew.value);
|
||
};
|
||
ebi('rn_reset_' + a).onclick = function (e) {
|
||
ev(e);
|
||
rn_reset(a);
|
||
};
|
||
})(a);
|
||
}
|
||
rn_reset(0);
|
||
tt.att(rui);
|
||
|
||
function sadv() {
|
||
ebi('rn_vadv').style.display = ebi('rn_case').style.display = r.adv ? '' : 'none';
|
||
}
|
||
bcfg_bind(r, 'adv', 'rn_adv', false, sadv);
|
||
bcfg_bind(r, 'cs', 'rn_case', false);
|
||
bcfg_bind(r, 'win', 'rn_win', true);
|
||
bcfg_bind(r, 'slash', 'rn_slash', true);
|
||
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(e) {
|
||
ev(e);
|
||
rui.parentNode.removeChild(rui);
|
||
}
|
||
|
||
ebi('rn_cancel').onclick = rn_cancel;
|
||
ebi('rn_apply').onclick = rn_apply;
|
||
|
||
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)';
|
||
|
||
ire.value = sread('cpp_rn_re') || '';
|
||
ifmt.value = sread('cpp_rn_fmt') || '';
|
||
|
||
var presets = {};
|
||
presets[defp] = ['', defp];
|
||
presets = jread("rn_pre", presets);
|
||
|
||
function spresets() {
|
||
var keys = Object.keys(presets), o;
|
||
keys.sort();
|
||
ipre.innerHTML = '<option value=""></option>';
|
||
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 (e) {
|
||
ev(e);
|
||
modal.prompt(L.fr_pname, ifmt.value, function (name) {
|
||
if (!name)
|
||
return toast.warn(3, L.fr_aborted);
|
||
|
||
presets[name] = [ire.value, ifmt.value];
|
||
jwrite('rn_pre', presets);
|
||
spresets();
|
||
ipre.value = name;
|
||
});
|
||
};
|
||
idel.onclick = function (e) {
|
||
ev(e);
|
||
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) {
|
||
var k = (e.code || e.key) + '';
|
||
|
||
if (k == 'Escape' || k == 'Esc')
|
||
return rn_cancel();
|
||
|
||
if (k.endsWith('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, r.cs ? 'i' : '');
|
||
}
|
||
catch (ex) {
|
||
return toast.err(5, esc('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(e) {
|
||
ev(e);
|
||
swrite('cpp_rn_re', ire.value);
|
||
swrite('cpp_rn_fmt', ifmt.value);
|
||
if (r.win || r.slash) {
|
||
var changed = 0;
|
||
for (var a = 0; a < f.length; a++) {
|
||
var ov = f[a].inew.value,
|
||
nv = namesan(ov, r.win, r.slash);
|
||
|
||
if (ov != nv) {
|
||
f[a].inew.value = nv;
|
||
changed++;
|
||
}
|
||
}
|
||
if (changed)
|
||
return modal.confirm(L.fr_nchg.format(changed), rn_apply_loop, null);
|
||
}
|
||
rn_apply_loop();
|
||
}
|
||
|
||
function rn_apply_loop() {
|
||
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();
|
||
return rn_cancel();
|
||
}
|
||
|
||
toast.show('inf r', 0, esc(L.fr_busy.format(f.length, f[0].ofn)));
|
||
var dst = base + uricom_enc(f[0].inew.value, false);
|
||
|
||
function rename_cb() {
|
||
if (this.status !== 201) {
|
||
var msg = unpre(this.responseText);
|
||
toast.err(9, L.fr_efail + msg);
|
||
return;
|
||
}
|
||
|
||
f.shift().inew.value = '( OK )';
|
||
return rn_apply_loop();
|
||
}
|
||
|
||
var xhr = new XHR();
|
||
xhr.open('POST', f[0].src + '?move=' + dst, true);
|
||
xhr.onload = xhr.onerror = rename_cb;
|
||
xhr.send();
|
||
}
|
||
};
|
||
|
||
r.delete = function (e) {
|
||
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, L.fd_emore);
|
||
|
||
ev(e);
|
||
|
||
if (clgot(bdel, 'hide'))
|
||
return toast.err(3, L.fd_eperm);
|
||
|
||
function deleter(err) {
|
||
var xhr = new XHR(),
|
||
vp = vps.shift();
|
||
|
||
if (!vp) {
|
||
if (err !== 'xbd')
|
||
toast.ok(2, L.fd_ok);
|
||
|
||
treectl.goto();
|
||
return;
|
||
}
|
||
toast.show('inf r', 0, esc(L.fd_busy.format(vps.length + 1, vp)), 'r');
|
||
|
||
xhr.open('POST', vp + '?delete', true);
|
||
xhr.onload = xhr.onerror = delete_cb;
|
||
xhr.send();
|
||
}
|
||
function delete_cb() {
|
||
if (this.status !== 200) {
|
||
var msg = unpre(this.responseText);
|
||
toast.err(9, L.fd_err + msg);
|
||
return;
|
||
}
|
||
if (this.responseText.indexOf('deleted 0 files (and 0') + 1) {
|
||
toast.err(9, L.fd_none);
|
||
return deleter('xbd');
|
||
}
|
||
deleter();
|
||
}
|
||
|
||
var asks = r.qdel ? 1 : 2;
|
||
if (dqdel === 0)
|
||
asks -= 1;
|
||
|
||
if (!asks)
|
||
return deleter();
|
||
|
||
modal.confirm('<h6 style="color:#900">' + L.danger + '</h6>\n<b>' + L.fd_warn1.format(vps.length) + '</b><ul>' + uricom_adec(vps, true).join('') + '</ul>', function () {
|
||
if (asks === 1)
|
||
return deleter();
|
||
modal.confirm(L.fd_warn2, deleter, null);
|
||
}, null);
|
||
};
|
||
|
||
r.cut = function (e) {
|
||
var sel = msel.getsel(),
|
||
stamp = Date.now(),
|
||
vps = [stamp];
|
||
|
||
if (!sel.length)
|
||
return toast.err(3, L.fc_emore);
|
||
|
||
ev(e);
|
||
|
||
if (clgot(bcut, 'hide'))
|
||
return toast.err(3, L.fc_eperm);
|
||
|
||
var els = [], griden = thegrid.en;
|
||
for (var a = 0; a < sel.length; a++) {
|
||
vps.push(sel[a].vp);
|
||
if (sel.length < 100)
|
||
try {
|
||
if (griden)
|
||
els.push(QS('#ggrid>a[ref="' + sel[a].id + '"]'));
|
||
else
|
||
els.push(ebi(sel[a].id).closest('tr'));
|
||
|
||
clmod(els[a], 'fcut');
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
|
||
setTimeout(function () {
|
||
try {
|
||
for (var a = 0; a < els.length; a++)
|
||
clmod(els[a], 'fcut', 1);
|
||
}
|
||
catch (ex) { }
|
||
}, 1);
|
||
|
||
r.ccp = false;
|
||
r.clip = vps.slice(1);
|
||
|
||
try {
|
||
vps = JSON.stringify(vps);
|
||
if (vps.length > 1024 * 1024)
|
||
throw 'a';
|
||
|
||
swrite('fman_clip', vps);
|
||
r.tx(stamp);
|
||
if (sel.length)
|
||
toast.inf(1.5, L.fc_ok.format(sel.length));
|
||
}
|
||
catch (ex) {
|
||
toast.warn(30, L.fc_warn.format(sel.length));
|
||
}
|
||
};
|
||
|
||
r.cpy = function (e) {
|
||
var sel = msel.getsel(),
|
||
stamp = Date.now(),
|
||
vps = [stamp, '//c'];
|
||
|
||
if (!sel.length)
|
||
return toast.err(3, L.fcp_emore);
|
||
|
||
ev(e);
|
||
|
||
var els = [], griden = thegrid.en;
|
||
for (var a = 0; a < sel.length; a++) {
|
||
vps.push(sel[a].vp);
|
||
if (sel.length < 100)
|
||
try {
|
||
if (griden)
|
||
els.push(QS('#ggrid>a[ref="' + sel[a].id + '"]'));
|
||
else
|
||
els.push(ebi(sel[a].id).closest('tr'));
|
||
|
||
clmod(els[a], 'fcut');
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
|
||
setTimeout(function () {
|
||
try {
|
||
for (var a = 0; a < els.length; a++)
|
||
clmod(els[a], 'fcut', 1);
|
||
}
|
||
catch (ex) { }
|
||
}, 1);
|
||
|
||
if (vps.length < 3)
|
||
vps.pop();
|
||
|
||
r.ccp = true;
|
||
r.clip = vps.slice(2);
|
||
|
||
try {
|
||
vps = JSON.stringify(vps);
|
||
if (vps.length > 1024 * 1024)
|
||
throw 'a';
|
||
|
||
swrite('fman_clip', vps);
|
||
r.tx(stamp);
|
||
if (sel.length)
|
||
toast.inf(1.5, L.fcc_ok.format(sel.length));
|
||
}
|
||
catch (ex) {
|
||
toast.warn(30, L.fcc_warn.format(sel.length));
|
||
}
|
||
};
|
||
|
||
document.onpaste = function (e) {
|
||
var xfer = e.clipboardData || window.clipboardData;
|
||
if (!xfer || !xfer.files || !xfer.files.length)
|
||
return;
|
||
|
||
var files = [];
|
||
for (var a = 0, aa = xfer.files.length; a < aa; a++)
|
||
files.push(xfer.files[a]);
|
||
|
||
clearTimeout(t_paste);
|
||
|
||
if (!r.clip.length)
|
||
return r.clip_up(files);
|
||
|
||
var src = r.clip.length == 1 ? r.clip[0] : vsplit(r.clip[0])[0],
|
||
msg = (r.ccp ? L.fcp_both_m : L.fp_both_m).format(r.clip.length, src, files.length);
|
||
|
||
modal.confirm(msg, r.paste, function () { r.clip_up(files); }, null, (r.ccp ? L.fcp_both_b : L.fp_both_b));
|
||
};
|
||
|
||
r.clip_up = function (files) {
|
||
goto_up2k();
|
||
var good = [], nil = [], bad = [];
|
||
for (var a = 0, aa = files.length; a < aa; a++) {
|
||
var fobj = files[a], dst = good;
|
||
try {
|
||
if (fobj.size < 1)
|
||
dst = nil;
|
||
}
|
||
catch (ex) {
|
||
dst = bad;
|
||
}
|
||
dst.push([fobj, fobj.name]);
|
||
}
|
||
var doit = function (is_img) {
|
||
jwrite('fman_clip', [Date.now()]);
|
||
r.clip = [];
|
||
|
||
var x = up2k.uc.ask_up;
|
||
if (is_img)
|
||
up2k.uc.ask_up = false;
|
||
|
||
up2k.gotallfiles[0](good, nil, bad, up2k.gotallfiles.slice(1));
|
||
up2k.uc.ask_up = x;
|
||
};
|
||
if (good.length != 1)
|
||
return doit();
|
||
|
||
var fn = good[0][1],
|
||
ofs = fn.lastIndexOf('.');
|
||
|
||
// stop linux-chrome from adding the fs-path into the <input>
|
||
setTimeout(function () {
|
||
modal.prompt(L.fp_name, fn, function (v) {
|
||
good[0][1] = v;
|
||
doit(true);
|
||
}, null, null, 0, ofs > 0 ? ofs : undefined);
|
||
}, 1);
|
||
};
|
||
|
||
r.d_paste = function () {
|
||
// gets called before onpaste; defer
|
||
clearTimeout(t_paste);
|
||
t_paste = setTimeout(r.paste, 50);
|
||
};
|
||
|
||
r.paste = function () {
|
||
if (!r.clip.length)
|
||
return toast.err(5, L.fp_ecut);
|
||
|
||
if (clgot(bpst, 'hide'))
|
||
return toast.err(3, L.fp_eperm);
|
||
|
||
var html = [
|
||
'<div>',
|
||
'<button id="rn_cancel" tt="' + L.frt_abrt + '</button>',
|
||
'<button id="rn_apply">✅ ' + L.fp_apply + '</button>',
|
||
' src: ' + esc(r.clip[0].replace(/[^/]+$/, '')),
|
||
'</div>',
|
||
'<p id="cnmt"></p>',
|
||
'<div><table id="rn_f" class="m">',
|
||
'<tr><td>' + L.fr_lnew + '</td><td>' + L.fr_lold + '</td></tr>',
|
||
],
|
||
ui = false,
|
||
f = [],
|
||
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(uricom_dec(vsplit(noq_href(links[a]))[1]));
|
||
|
||
for (var a = 0; a < r.clip.length; a++) {
|
||
var t = {
|
||
'ok': true,
|
||
'src': r.clip[a],
|
||
'dst': uricom_dec(r.clip[a].split('/').pop()),
|
||
};
|
||
f.push(t);
|
||
|
||
for (var b = 0; b < indir.length; b++)
|
||
if (t.dst == indir[b]) {
|
||
t.ok = false;
|
||
ui = true;
|
||
}
|
||
|
||
html.push('<tr' + (!t.ok ? ' class="ng"' : '') + '><td><input type="text" id="rn_new_' + a + '" value="' + esc(t.dst) + '" /></td><td><input type="text" id="rn_old_' + a + '" value="' + esc(t.dst) + '" readonly /></td></tr>');
|
||
}
|
||
|
||
function paster() {
|
||
var t = f.shift();
|
||
if (!t) {
|
||
toast.ok(2, r.ccp ? L.fcp_ok : L.fp_ok);
|
||
treectl.goto();
|
||
r.tx(srcdir);
|
||
return;
|
||
}
|
||
if (!t.dst)
|
||
return paster();
|
||
|
||
toast.show('inf r', 0, esc((r.ccp ? L.fcp_busy : L.fp_busy).format(f.length + 1, uricom_dec(t.src))));
|
||
|
||
var xhr = new XHR(),
|
||
act = r.ccp ? '?copy=' : '?move=',
|
||
dst = get_evpath() + uricom_enc(t.dst);
|
||
|
||
xhr.open('POST', t.src + act + dst, true);
|
||
xhr.onload = xhr.onerror = paste_cb;
|
||
xhr.send();
|
||
}
|
||
function paste_cb() {
|
||
if (this.status !== 201) {
|
||
var msg = unpre(this.responseText);
|
||
toast.err(9, (r.ccp ? L.fcp_err : L.fp_err) + msg);
|
||
return;
|
||
}
|
||
paster();
|
||
}
|
||
function okgo() {
|
||
paster();
|
||
jwrite('fman_clip', [Date.now()]);
|
||
}
|
||
|
||
if (!ui) {
|
||
var src = [];
|
||
for (var a = 0; a < f.length; a++)
|
||
src.push(f[a].src);
|
||
|
||
return modal.confirm((r.ccp ? L.fcp_confirm : L.fp_confirm).format(f.length) + '<ul>' + uricom_adec(src, true).join('') + '</ul>', okgo, null);
|
||
}
|
||
|
||
var rui = ebi('rui');
|
||
if (!rui) {
|
||
rui = mknod('div', 'rui');
|
||
document.body.appendChild(rui);
|
||
}
|
||
html.push('</table>');
|
||
rui.innerHTML = html.join('\n');
|
||
tt.att(rui);
|
||
|
||
function rn_apply(e) {
|
||
for (var a = 0; a < f.length; a++)
|
||
if (!f[a].ok) {
|
||
toast.err(30, L.fp_emore);
|
||
return setcnmt(true);
|
||
}
|
||
rn_cancel(e);
|
||
okgo();
|
||
}
|
||
function rn_cancel(e) {
|
||
ev(e);
|
||
rui.parentNode.removeChild(rui);
|
||
}
|
||
ebi('rn_cancel').onclick = rn_cancel;
|
||
ebi('rn_apply').onclick = rn_apply;
|
||
|
||
var first_bad = 0;
|
||
function setcnmt(sel) {
|
||
var nbad = 0;
|
||
for (var a = 0; a < f.length; a++) {
|
||
if (f[a].ok)
|
||
continue;
|
||
if (!nbad)
|
||
first_bad = a;
|
||
nbad += 1;
|
||
}
|
||
ebi('cnmt').innerHTML = (r.ccp ? L.fcp_ename : L.fp_ename).format(nbad);
|
||
if (sel && nbad) {
|
||
var el = ebi('rn_new_' + first_bad);
|
||
el.focus();
|
||
el.setSelectionRange(0, el.value.lastIndexOf('.'), "forward");
|
||
}
|
||
}
|
||
setcnmt(true);
|
||
|
||
for (var a = 0; a < f.length; a++)
|
||
(function (a) {
|
||
var inew = ebi('rn_new_' + a);
|
||
inew.onkeydown = function (e) {
|
||
if (((e.code || e.key) + '').endsWith('Enter'))
|
||
return rn_apply();
|
||
};
|
||
inew.oninput = function (e) {
|
||
f[a].dst = this.value;
|
||
f[a].ok = true;
|
||
if (f[a].dst)
|
||
for (var b = 0; b < indir.length; b++)
|
||
if (indir[b] == this.value)
|
||
f[a].ok = false;
|
||
clmod(this.closest('tr'), 'ng', !f[a].ok);
|
||
setcnmt();
|
||
};
|
||
})(a);
|
||
}
|
||
|
||
function onmsg(msg) {
|
||
r.clip = null;
|
||
var n = parseInt('' + msg), tries = 0;
|
||
var fun = function () {
|
||
if (n == msg && n > 1 && r.clip === null) {
|
||
var fc = jread('fman_clip', []);
|
||
if (!fc || !fc.length || fc[0] != n) {
|
||
if (++tries > 10)
|
||
return modal.alert(L.fp_etab);
|
||
|
||
return setTimeout(fun, 100);
|
||
}
|
||
}
|
||
r.render();
|
||
if (msg == get_evpath())
|
||
treectl.goto(msg);
|
||
};
|
||
fun();
|
||
}
|
||
|
||
if (r.bus)
|
||
r.bus.onmessage = function (e) {
|
||
onmsg(e ? e.data : 1)
|
||
};
|
||
|
||
r.tx = function (msg) {
|
||
if (!r.bus)
|
||
return onmsg(msg);
|
||
|
||
r.bus.postMessage(msg);
|
||
r.bus.onmessage();
|
||
};
|
||
|
||
bcfg_bind(r, 'qdel', 'qdel', dqdel == 1);
|
||
|
||
bren.onclick = r.rename;
|
||
bdel.onclick = r.delete;
|
||
bcut.onclick = r.cut;
|
||
bcpy.onclick = r.cpy;
|
||
bpst.onclick = r.paste;
|
||
bshr.onclick = r.share;
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
var showfile = (function () {
|
||
var r = {
|
||
'nrend': 0,
|
||
};
|
||
r.map = {
|
||
'.ahk': 'autohotkey',
|
||
'.bas': 'basic',
|
||
'.bat': 'batch',
|
||
'.cxx': 'cpp',
|
||
'.diz': 'ans',
|
||
'.ex': 'elixir',
|
||
'.exs': 'elixir',
|
||
'.frag': 'glsl',
|
||
'.h': 'c',
|
||
'.hpp': 'cpp',
|
||
'.htm': 'html',
|
||
'.hxx': 'cpp',
|
||
'.log': 'ans',
|
||
'.m': 'matlab',
|
||
'.moon': 'moonscript',
|
||
'.patch': 'diff',
|
||
'.ps1': 'powershell',
|
||
'.psm1': 'powershell',
|
||
'.pl': 'perl',
|
||
'.rs': 'rust',
|
||
'.sh': 'bash',
|
||
'.service': 'systemd',
|
||
'.vb': 'vbnet',
|
||
'.v': 'verilog',
|
||
'.vert': 'glsl',
|
||
'.vh': 'verilog',
|
||
'.yml': 'yaml'
|
||
};
|
||
r.nmap = {
|
||
'cmakelists.txt': 'cmake',
|
||
'dockerfile': 'docker'
|
||
};
|
||
var x = txt_ext + ' ans c cfg conf cpp cs css diff glsl go html ini java js json jsx kt kts latex less lisp lua makefile md nim py r rss rb ruby sass scss sql svg swift tex toml ts vhdl xml yaml zig';
|
||
x = x.split(/ +/g);
|
||
for (var a = 0; a < x.length; a++)
|
||
r.map["." + x[a]] = x[a];
|
||
|
||
r.sname = function (srch) {
|
||
return srch.split(/[?&]doc=/)[1].split('&')[0];
|
||
};
|
||
|
||
if (window.og_fn) {
|
||
var ext = og_fn.split(/\./g).pop();
|
||
if (r.map['.' + ext])
|
||
hist_replace(get_evpath() + '?doc=' + og_fn);
|
||
}
|
||
|
||
window.Prism = { 'manual': true };
|
||
var em = QS('#bdoc>pre');
|
||
if (em)
|
||
em = [r.sname(location.search), location.hash, em.textContent];
|
||
else {
|
||
var m = /[?&]doc=([^&]+)/.exec(location.search);
|
||
if (m) {
|
||
setTimeout(function () {
|
||
r.show(uricom_dec(m[1]), true);
|
||
}, 1);
|
||
}
|
||
}
|
||
|
||
r.setstyle = function () {
|
||
if (window.no_prism)
|
||
return;
|
||
|
||
qsr('#prism_css');
|
||
var el = mknod('link', 'prism_css');
|
||
el.rel = 'stylesheet';
|
||
el.href = SR + '/.cpr/deps/prism' + (light ? '' : 'd') + '.css?_=' + TS;
|
||
document.head.appendChild(el);
|
||
};
|
||
|
||
r.active = function () {
|
||
return !!/[?&]doc=/.exec(location.search);
|
||
};
|
||
|
||
r.getlang = function (fn) {
|
||
fn = fn.toLowerCase();
|
||
var ext = fn.slice(fn.lastIndexOf('.'));
|
||
return r.map[ext] || r.nmap[fn];
|
||
}
|
||
|
||
r.addlinks = function () {
|
||
r.files = [];
|
||
var links = msel.getall();
|
||
for (var a = 0; a < links.length; a++) {
|
||
var link = links[a],
|
||
fn = link.vp.split('/').pop(),
|
||
lang = r.getlang(fn);
|
||
|
||
if (!lang)
|
||
continue;
|
||
|
||
r.files.push({ 'id': link.id, 'name': uricom_dec(fn) });
|
||
|
||
var ah = ebi(link.id),
|
||
td = ah.closest('tr').getElementsByTagName('td')[0];
|
||
|
||
if (ah.textContent.endsWith('/'))
|
||
continue;
|
||
|
||
if (lang == 'ts' || (lang == 'md' && td.textContent != '-'))
|
||
continue;
|
||
|
||
td.innerHTML = '<a href="#" id="t' +
|
||
link.id + '" class="doc bri" hl="' +
|
||
link.id + '" rel="nofollow">-txt-</a>';
|
||
|
||
td.getElementsByTagName('a')[0].setAttribute('href', '?doc=' + fn);
|
||
}
|
||
r.mktree();
|
||
if (em) {
|
||
if (r.taildoc)
|
||
r.show(em[0], true);
|
||
else
|
||
render(em);
|
||
em = null;
|
||
}
|
||
};
|
||
|
||
r.tail = function (url, no_push) {
|
||
r.abrt = new AbortController();
|
||
widget.setvis();
|
||
render([url, '', ''], no_push);
|
||
var me = r.tail_id = Date.now(),
|
||
wfp = ebi('wfp'),
|
||
edoc = ebi('doc'),
|
||
txt = '';
|
||
|
||
url = addq(url, 'tail=-' + r.tailnb);
|
||
fetch(url, {'signal': r.abrt.signal}).then(function(rsp) {
|
||
var ro = rsp.body.pipeThrough(
|
||
new TextDecoderStream('utf-8', {'fatal': false}),
|
||
{'signal': r.abrt.signal}).getReader();
|
||
|
||
var rf = function() {
|
||
ro.read().then(function(v) {
|
||
if (r.tail_id != me)
|
||
return;
|
||
var vt = v.done ? '\n*** lost connection to copyparty ***' : v.value;
|
||
if (vt == '\x00')
|
||
return rf();
|
||
txt += vt;
|
||
var ofs = txt.length - r.tailnb;
|
||
if (ofs > 0) {
|
||
var ofs2 = txt.indexOf('\n', ofs);
|
||
if (ofs2 >= ofs && ofs - ofs2 < 512)
|
||
ofs = ofs2;
|
||
txt = txt.slice(ofs);
|
||
}
|
||
var html = esc(txt);
|
||
if (r.tailansi)
|
||
html = r.ansify(html);
|
||
edoc.innerHTML = html;
|
||
if (r.tail2end)
|
||
window.scrollTo(0, wfp.offsetTop - window.innerHeight);
|
||
if (!v.done)
|
||
rf();
|
||
});
|
||
};
|
||
if (r.tail_id == me)
|
||
rf();
|
||
});
|
||
};
|
||
|
||
r.untail = function () {
|
||
if (!r.abrt)
|
||
return;
|
||
r.abrt.abort();
|
||
r.abrt = null;
|
||
r.tail_id = -1;
|
||
widget.setvis();
|
||
};
|
||
|
||
r.show = function (url, no_push) {
|
||
r.untail();
|
||
var xhr = new XHR(),
|
||
m = /[?&](k=[^&#]+)/.exec(url);
|
||
|
||
url = url.split('?')[0] + (m ? '?' + m[1] : '');
|
||
assert_vp(url);
|
||
if (r.taildoc)
|
||
return r.tail(url, no_push);
|
||
|
||
xhr.url = url;
|
||
xhr.fname = uricom_dec(url.split('/').pop());
|
||
xhr.no_push = no_push;
|
||
xhr.ts = Date.now();
|
||
xhr.open('GET', url, true);
|
||
xhr.onprogress = loading;
|
||
xhr.onload = xhr.onerror = load_cb;
|
||
xhr.send();
|
||
};
|
||
|
||
function loading(e) {
|
||
if (e.total < 1024 * 256)
|
||
return;
|
||
|
||
var m = L.tv_load.format(
|
||
esc(this.fname),
|
||
f2f(e.loaded * 100 / e.total, 1),
|
||
f2f(e.loaded / 1024 / 1024, 1),
|
||
f2f(e.total / 1024 / 1024, 1))
|
||
|
||
if (!this.toasted) {
|
||
this.toasted = 1;
|
||
return toast.inf(573, m);
|
||
}
|
||
ebi('toastb').innerHTML = lf2br(m);
|
||
}
|
||
|
||
function load_cb(e) {
|
||
if (this.toasted)
|
||
toast.hide();
|
||
|
||
if (!xhrchk(this, L.tv_xe1, L.tv_xe2))
|
||
return;
|
||
|
||
render([this.url, '', this.responseText], this.no_push);
|
||
}
|
||
|
||
function render(doc, no_push) {
|
||
r.q = null;
|
||
r.nrend++;
|
||
var url = r.url = doc[0],
|
||
lnh = doc[1],
|
||
txt = doc[2],
|
||
name = url.split('?')[0].split('/').pop(),
|
||
tname = uricom_dec(name),
|
||
lang = r.getlang(name),
|
||
is_md = lang == 'md';
|
||
|
||
ebi('files').style.display = ebi('gfiles').style.display = ebi('lazy').style.display = ebi('pro').style.display = ebi('epi').style.display = 'none';
|
||
ebi('dldoc').setAttribute('href', url);
|
||
ebi('editdoc').setAttribute('href', addq(url, 'edit'));
|
||
ebi('editdoc').style.display = (has(perms, 'write') && (is_md || has(perms, 'delete'))) ? '' : 'none';
|
||
|
||
var wr = ebi('bdoc'),
|
||
nrend = r.nrend,
|
||
defer = !Prism.highlightElement;
|
||
|
||
var fun = function (el) {
|
||
if (r.nrend != nrend)
|
||
return;
|
||
|
||
try {
|
||
if (lnh.slice(0, 5) == '#doc.')
|
||
sethash(lnh.slice(1));
|
||
|
||
el = el || QS('#doc>code');
|
||
Prism.highlightElement(el);
|
||
if (el.className == 'language-ans' || (!lang && /\x1b\[[0-9;]{0,16}m/.exec(txt.slice(0, 4096))))
|
||
el.innerHTML = r.ansify(el.innerHTML);
|
||
}
|
||
catch (ex) { }
|
||
}
|
||
|
||
var skip_prism = !txt || txt.length > 1024 * 256;
|
||
if (skip_prism) {
|
||
fun = function (el) { };
|
||
is_md = false;
|
||
}
|
||
|
||
qsr('#doc');
|
||
var el = mknod('pre', 'doc');
|
||
el.setAttribute('tabindex', '0');
|
||
clmod(ebi('wrap'), 'doc', !is_md);
|
||
if (is_md) {
|
||
show_md(txt, name, el);
|
||
}
|
||
else {
|
||
el.textContent = txt;
|
||
el.innerHTML = '<code>' + el.innerHTML + '</code>';
|
||
if (!window.no_prism && !skip_prism) {
|
||
if ((lang == 'conf' || lang == 'cfg') && ('\n' + txt).indexOf('\n# -*- mode: yaml -*-') + 1)
|
||
lang = 'yaml';
|
||
|
||
el.className = 'prism linkable-line-numbers line-numbers language-' + lang;
|
||
if (!defer)
|
||
fun(el.firstChild);
|
||
else
|
||
import_js(SR + '/.cpr/deps/prism.js', function () { fun(); });
|
||
}
|
||
if (!txt && r.wrap)
|
||
el.className = 'wrap';
|
||
}
|
||
|
||
wr.appendChild(el);
|
||
wr.style.display = '';
|
||
set_tabindex();
|
||
|
||
wintitle(tname + ' \u2014 ');
|
||
document.documentElement.scrollTop = 0;
|
||
var hfun = no_push ? hist_replace : hist_push;
|
||
hfun(get_evpath() + '?doc=' + name); // can't dk: server wants dk and js needs fk
|
||
|
||
qsr('#docname');
|
||
el = mknod('span', 'docname');
|
||
el.textContent = tname;
|
||
ebi('path').appendChild(el);
|
||
|
||
r.updtree();
|
||
treectl.textmode(true);
|
||
tree_scrollto();
|
||
}
|
||
|
||
r.ansify = function (html) {
|
||
var ctab = (light ?
|
||
'bfbfbf d30253 497600 b96900 006fbb a50097 288276 2d2d2d 9f9f9f 943b55 3a5600 7f4f00 00507d 683794 004343 000000' :
|
||
'404040 f03669 b8e346 ffa402 02a2ff f65be3 3da698 d2d2d2 606060 c75b79 c8e37e ffbe4a 71cbff b67fe3 9cf0ed ffffff').split(/ /g),
|
||
src = html.split(/\x1b\[/g),
|
||
out = ['<span>'], fg = 7, bg = null, bfg = 0, bbg = 0, inv = 0, bold = 0;
|
||
|
||
for (var a = 0; a < src.length; a++) {
|
||
var m = /^([0-9;]+)m/.exec(src[a]);
|
||
if (!m) {
|
||
if (a)
|
||
out.push('\x1b[');
|
||
|
||
out.push(src[a]);
|
||
continue;
|
||
}
|
||
|
||
var cs = m[1].split(/;/g),
|
||
txt = src[a].slice(m[1].length + 1);
|
||
|
||
for (var b = 0; b < cs.length; b++) {
|
||
var c = parseInt(cs[b]);
|
||
if (c == 0) {
|
||
fg = 7;
|
||
bg = null;
|
||
bfg = bbg = bold = inv = 0;
|
||
}
|
||
if (c == 1) bfg = bold = 1;
|
||
if (c == 7) inv = 1;
|
||
if (c == 22) bfg = bold = 0;
|
||
if (c == 27) inv = 0;
|
||
if (c >= 30 && c <= 37) fg = c - 30;
|
||
if (c >= 40 && c <= 47) bg = c - 40;
|
||
if (c >= 90 && c <= 97) {
|
||
fg = c - 90;
|
||
bfg = 1;
|
||
}
|
||
if (c >= 100 && c <= 107) {
|
||
bg = c - 100;
|
||
bbg = 1;
|
||
}
|
||
}
|
||
|
||
var cfg = fg, cbg = bg;
|
||
if (inv) {
|
||
cbg = fg;
|
||
cfg = bg || 0;
|
||
}
|
||
|
||
var s = '</span><span style="color:#' + ctab[cfg + bfg * 8];
|
||
if (cbg !== null)
|
||
s += ';background:#' + ctab[cbg + bbg * 8];
|
||
if (bold)
|
||
s += ';font-weight:bold';
|
||
|
||
out.push(s + '">' + txt);
|
||
}
|
||
return out.join('');
|
||
};
|
||
|
||
r.mktree = function () {
|
||
var crumbs = linksplit(get_evpath()).join('<span>/</span>'),
|
||
html = ['<li class="bn">' + L.tv_lst + '<br />' + crumbs + '</li>'];
|
||
for (var a = 0; a < r.files.length; a++) {
|
||
var file = r.files[a];
|
||
html.push('<li><a href="?doc=' +
|
||
uricom_enc(file.name) + '" hl="' + file.id +
|
||
'">' + esc(file.name) + '</a>');
|
||
}
|
||
ebi('docul').innerHTML = html.join('\n');
|
||
};
|
||
|
||
r.updtree = function () {
|
||
var fn = QS('#path span:last-child'),
|
||
lis = QSA('#docul li a'),
|
||
sels = msel.getsel(),
|
||
actsel = false;
|
||
|
||
fn = fn ? fn.textContent : '';
|
||
for (var a = 0, aa = lis.length; a < aa; a++) {
|
||
var lin = lis[a].textContent,
|
||
sel = false;
|
||
|
||
for (var b = 0; b < sels.length; b++)
|
||
if (vsplit(sels[b].vp)[1] == lin)
|
||
sel = true;
|
||
|
||
clmod(lis[a], 'hl', lin == fn);
|
||
clmod(lis[a], 'sel', sel);
|
||
if (lin == fn && sel)
|
||
actsel = true;
|
||
}
|
||
clmod(ebi('seldoc'), 'sel', actsel);
|
||
};
|
||
|
||
r.tglsel = function () {
|
||
var fn = ebi('docname').textContent;
|
||
for (var a = 0; a < r.files.length; a++)
|
||
if (r.files[a].name == fn)
|
||
clmod(ebi(r.files[a].id).closest('tr'), 'sel', 't');
|
||
|
||
msel.selui();
|
||
};
|
||
|
||
r.tgltail = function () {
|
||
if (!window.TextDecoderStream) {
|
||
bcfg_set('taildoc', r.taildoc = false);
|
||
return toast.err(10, L.tail_2old);
|
||
}
|
||
r.show(r.url, true);
|
||
};
|
||
|
||
r.tglwrap = function () {
|
||
r.show(r.url, true);
|
||
};
|
||
|
||
var bdoc = ebi('bdoc');
|
||
bdoc.className = 'line-numbers';
|
||
bdoc.innerHTML = (
|
||
'<div id="hdoc" class="ghead">\n' +
|
||
'<a href="#" class="btn" id="xdoc" tt="' + L.tvt_close + '</a>\n' +
|
||
'<a href="#" class="btn" id="dldoc" tt="' + L.tvt_dl + '</a>\n' +
|
||
'<a href="#" class="btn" id="prevdoc" tt="' + L.tvt_prev + '</a>\n' +
|
||
'<a href="#" class="btn" id="nextdoc" tt="' + L.tvt_next + '</a>\n' +
|
||
'<a href="#" class="btn" id="seldoc" tt="' + L.tvt_sel + '</a>\n' +
|
||
'<a href="#" class="btn" id="editdoc" tt="' + L.tvt_edit + '</a>\n' +
|
||
'<a href="#" class="btn tgl" id="taildoc" tt="' + L.tvt_tail + '</a>\n' +
|
||
'<div id="tailbtns">\n' +
|
||
'<a href="#" class="btn tgl" id="wrapdoc" tt="' + L.tvt_wrap + '</a>\n' +
|
||
'<a href="#" class="btn tgl" id="tail2end" tt="' + L.tvt_atail + '</a>\n' +
|
||
'<a href="#" class="btn tgl" id="tailansi" tt="' + L.tvt_ctail + '</a>\n' +
|
||
'<input type="text" id="tailnb" value="" ' + NOAC + ' style="width:4em" tt="' + L.tvt_ntail + '" />' +
|
||
'</div>\n' +
|
||
'</div>'
|
||
);
|
||
ebi('xdoc').onclick = function () {
|
||
r.untail();
|
||
thegrid.setvis(true);
|
||
bcfg_bind(r, 'taildoc', 'taildoc', false, r.tgltail);
|
||
};
|
||
ebi('dldoc').setAttribute('download', '');
|
||
ebi('prevdoc').onclick = function () { tree_neigh(-1); };
|
||
ebi('nextdoc').onclick = function () { tree_neigh(1); };
|
||
ebi('seldoc').onclick = r.tglsel;
|
||
bcfg_bind(r, 'wrap', 'wrapdoc', true, r.tglwrap);
|
||
bcfg_bind(r, 'taildoc', 'taildoc', false, r.tgltail);
|
||
bcfg_bind(r, 'tail2end', 'tail2end', true);
|
||
bcfg_bind(r, 'tailansi', 'tailansi', false, r.tgltail);
|
||
|
||
r.tailnb = ebi('tailnb').value = icfg_get('tailnb', 131072);
|
||
ebi('tailnb').oninput = function (e) {
|
||
swrite('tailnb', r.tailnb = this.value);
|
||
};
|
||
|
||
if (/[?&]tail\b/.exec(sloc0)) {
|
||
clmod(ebi('taildoc'), 'on', 1);
|
||
r.taildoc = true;
|
||
}
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
var thegrid = (function () {
|
||
var lfiles = ebi('files'),
|
||
gfiles = mknod('div', 'gfiles');
|
||
|
||
gfiles.style.display = 'none';
|
||
gfiles.innerHTML = (
|
||
'<div id="ghead" class="ghead">' +
|
||
'<a href="#" class="tgl btn" id="gridvau" tt="' + L.gt_vau + '</a> ' +
|
||
'<a href="#" class="tgl btn" id="gridsel" tt="' + L.gt_msel + '</a> ' +
|
||
'<a href="#" class="tgl btn" id="gridcrop" tt="' + L.gt_crop + '</a> ' +
|
||
'<a href="#" class="tgl btn" id="grid3x" tt="' + L.gt_3x + '</a> ' +
|
||
'<span>' + L.gt_zoom + ': ' +
|
||
'<a href="#" class="btn" z="-1.1" tt="Hotkey: shift-A">–</a> ' +
|
||
'<a href="#" class="btn" z="1.1" tt="Hotkey: shift-D">+</a></span> <span>' + L.gt_chop + ': ' +
|
||
'<a href="#" class="btn" l="-1" tt="' + L.gt_c1 + '">–</a> ' +
|
||
'<a href="#" class="btn" l="1" tt="' + L.gt_c2 + '">+</a></span> <span>' + L.gt_sort + ': ' +
|
||
'<a href="#" s="href">' + L.gt_name + '</a> ' +
|
||
'<a href="#" s="sz">' + L.gt_sz + '</a> ' +
|
||
'<a href="#" s="ts">' + L.gt_ts + '</a> ' +
|
||
'<a href="#" s="ext">' + L.gt_ext + '</a>' +
|
||
'</span></div>' +
|
||
'<div id="ggrid"></div>'
|
||
);
|
||
lfiles.parentNode.insertBefore(gfiles, lfiles);
|
||
var ggrid = ebi('ggrid');
|
||
|
||
var r = {
|
||
'sz': clamp(fcfg_get('gridsz', 10), 4, 80),
|
||
'ln': clamp(icfg_get('gridln', 3), 1, 7),
|
||
'isdirty': true,
|
||
'bbox': null
|
||
};
|
||
|
||
var btnclick = function (e) {
|
||
ev(e);
|
||
var s = this.getAttribute('s'),
|
||
z = this.getAttribute('z'),
|
||
l = this.getAttribute('l');
|
||
|
||
if (z)
|
||
return setsz(z > 0 ? r.sz * z : r.sz / (-z));
|
||
|
||
if (l)
|
||
return setln(parseInt(l));
|
||
|
||
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;
|
||
|
||
r.setvis = function (force) {
|
||
if (showfile.active()) {
|
||
if (!force)
|
||
return;
|
||
|
||
hist_push(get_evpath() + (dk ? '?k=' + dk : ''));
|
||
wintitle();
|
||
}
|
||
|
||
lfiles = ebi('files');
|
||
gfiles = ebi('gfiles');
|
||
ggrid = ebi('ggrid');
|
||
|
||
var vis = has(perms, "read");
|
||
gfiles.style.display = vis && r.en ? '' : 'none';
|
||
lfiles.style.display = vis && !r.en ? '' : 'none';
|
||
clmod(ggrid, 'crop', r.crop);
|
||
clmod(ggrid, 'nocrop', !r.crop);
|
||
ebi('pro').style.display = ebi('epi').style.display = ebi('lazy').style.display = ebi('treeul').style.display = ebi('treepar').style.display = '';
|
||
ebi('bdoc').style.display = 'none';
|
||
clmod(ebi('wrap'), 'doc');
|
||
qsr('#docname');
|
||
if (treectl)
|
||
treectl.textmode(false);
|
||
|
||
if (filecols)
|
||
filecols.uivis();
|
||
|
||
aligngriditems();
|
||
restore_scroll();
|
||
};
|
||
|
||
r.setdirty = function () {
|
||
r.dirty = true;
|
||
if (r.en)
|
||
loadgrid();
|
||
else
|
||
r.setvis();
|
||
};
|
||
|
||
function setln(v) {
|
||
if (v) {
|
||
r.ln += v;
|
||
if (r.ln < 1) r.ln = 1;
|
||
if (r.ln > 7) r.ln = v < 0 ? 7 : 99;
|
||
swrite('gridln', r.ln);
|
||
setTimeout(r.tippen, 20);
|
||
}
|
||
setcvar('--grid-ln', r.ln);
|
||
}
|
||
setln();
|
||
|
||
function setsz(v) {
|
||
if (v !== undefined) {
|
||
r.sz = clamp(v, 4, 80);
|
||
swrite('gridsz', r.sz);
|
||
setTimeout(r.tippen, 20);
|
||
}
|
||
setcvar('--grid-sz', r.sz + 'em');
|
||
aligngriditems();
|
||
}
|
||
setsz();
|
||
|
||
function gclick1(e) {
|
||
if (ctrl(e) && !treectl.csel && !r.sel)
|
||
return true;
|
||
|
||
return gclick.call(this, e, false);
|
||
}
|
||
|
||
function gclick2(e) {
|
||
if (ctrl(e) || !r.sel)
|
||
return true;
|
||
|
||
return gclick.call(this, e, true);
|
||
}
|
||
|
||
function gclick(e, dbl) {
|
||
var oth = ebi(this.getAttribute('ref')),
|
||
href = noq_href(this),
|
||
fid = oth.getAttribute('id'),
|
||
aplay = ebi('a' + fid),
|
||
atext = ebi('t' + fid),
|
||
is_txt = atext && !/\.ts$/.test(href) && showfile.getlang(href),
|
||
is_img = img_re.test(href),
|
||
is_dir = href.endsWith('/'),
|
||
is_srch = !!ebi('unsearch'),
|
||
in_tree = is_dir && treectl.find(oth.textContent.slice(0, -1)),
|
||
have_sel = QS('#files tr.sel'),
|
||
td = oth.closest('td').nextSibling,
|
||
tr = td.parentNode;
|
||
|
||
if (!is_srch && ((r.sel && !dbl && !ctrl(e)) || (treectl.csel && (e.shiftKey || ctrl(e))))) {
|
||
td.onclick.call(td, e);
|
||
if (e.shiftKey)
|
||
return r.loadsel();
|
||
clmod(this, 'sel', clgot(tr, 'sel'));
|
||
}
|
||
else if (in_tree && !have_sel)
|
||
in_tree.click();
|
||
|
||
else if (oth.hasAttribute('download'))
|
||
oth.click();
|
||
|
||
else if (aplay && (r.vau || !is_img))
|
||
aplay.click();
|
||
|
||
else if (is_dir && !have_sel)
|
||
treectl.reqls(href, true);
|
||
|
||
else if (is_txt && !has(['md', 'htm', 'html'], is_txt))
|
||
atext.click();
|
||
|
||
else if (!is_img && have_sel)
|
||
window.open(href, '_blank');
|
||
|
||
else {
|
||
if (!dbl)
|
||
return true;
|
||
|
||
setTimeout(function () {
|
||
r.sel = true;
|
||
}, 1);
|
||
r.sel = false;
|
||
this.click();
|
||
}
|
||
ev(e);
|
||
}
|
||
|
||
r.imshow = function (url) {
|
||
var sel = '#ggrid>a'
|
||
if (!thegrid.en) {
|
||
thegrid.bagit('#files');
|
||
sel = '#files a[id]';
|
||
}
|
||
var ims = QSA(sel);
|
||
for (var a = 0, aa = ims.length; a < aa; a++) {
|
||
var iu = ims[a].getAttribute('href').split('?')[0].split('/').slice(-1)[0];
|
||
if (iu == url)
|
||
return ims[a].click();
|
||
}
|
||
baguetteBox.hide();
|
||
};
|
||
|
||
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'),
|
||
cl = tr.className || '';
|
||
|
||
if (noq_href(ths[a]).endsWith('/'))
|
||
cl += ' dir';
|
||
|
||
ths[a].className = cl;
|
||
}
|
||
|
||
var sp = ['unsearch', 'moar'];
|
||
for (var a = 0; a < sp.length; a++)
|
||
(function (a) {
|
||
var o = QS('#ggrid a[ref="' + sp[a] + '"]');
|
||
if (o)
|
||
o.onclick = function (e) {
|
||
ev(e);
|
||
ebi(sp[a]).click();
|
||
};
|
||
})(a);
|
||
};
|
||
|
||
r.tippen = function () {
|
||
var els = QSA('#ggrid>a>span'),
|
||
aa = els.length;
|
||
|
||
if (!aa)
|
||
return;
|
||
|
||
var cs = window.getComputedStyle(els[0]),
|
||
fs = parseFloat(cs.lineHeight),
|
||
pad = parseFloat(cs.paddingTop),
|
||
pels = [],
|
||
todo = [];
|
||
|
||
for (var a = 0; a < aa; a++) {
|
||
var vis = Math.round((els[a].offsetHeight - pad) / fs),
|
||
all = Math.round((els[a].scrollHeight - pad) / fs),
|
||
par = els[a].parentNode;
|
||
|
||
pels.push(par);
|
||
todo.push(vis < all ? par.getAttribute('ttt') : null);
|
||
}
|
||
|
||
for (var a = 0; a < todo.length; a++) {
|
||
if (todo[a])
|
||
pels[a].setAttribute('tt', todo[a]);
|
||
else
|
||
pels[a].removeAttribute('tt');
|
||
}
|
||
|
||
tt.att(ggrid);
|
||
};
|
||
|
||
function loadgrid() {
|
||
if (have_webp === null)
|
||
return setTimeout(loadgrid, 50);
|
||
|
||
r.setvis();
|
||
if (!r.dirty)
|
||
return r.loadsel();
|
||
|
||
if (dcrop.startsWith('f') || !sread('gridcrop'))
|
||
bcfg_upd_ui('gridcrop', r.crop = ('y' == dcrop.slice(-1)));
|
||
|
||
if (dth3x.startsWith('f') || !sread('grid3x'))
|
||
bcfg_upd_ui('grid3x', r.x3 = ('y' == dth3x.slice(-1)));
|
||
|
||
var html = [],
|
||
svgs = new Set(),
|
||
max_svgs = CHROME ? 500 : 5000,
|
||
need_ext = !r.thumbs || !!ext_th,
|
||
use_ext_th = r.thumbs && ext_th,
|
||
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],
|
||
ohref = esc(ao.getAttribute('href')),
|
||
href = ohref.split('?')[0],
|
||
ext = '',
|
||
ext0 = '',
|
||
name = uricom_dec(vsplit(href)[1]),
|
||
ref = ao.getAttribute('id'),
|
||
isdir = href.endsWith('/'),
|
||
ac = isdir ? ' class="dir"' : '',
|
||
ihref = ohref;
|
||
|
||
if (need_ext && href != "#") {
|
||
var ar = href.split('.');
|
||
if (ar.length > 1)
|
||
ar.shift();
|
||
|
||
ar.reverse();
|
||
ext0 = ar[0];
|
||
for (var b = 0; b < Math.min(2, ar.length); b++) {
|
||
if (ar[b].length > 7)
|
||
break;
|
||
|
||
ext = ext ? (ar[b] + '.' + ext) : ar[b];
|
||
}
|
||
if (!ext)
|
||
ext = 'unk';
|
||
}
|
||
|
||
if (use_ext_th && (ext_th[ext] || ext_th[ext0])) {
|
||
ihref = ext_th[ext] || ext_th[ext0];
|
||
}
|
||
else if (r.thumbs) {
|
||
ihref = addq(ihref, 'th=' + (have_webp ? 'w' : 'j'));
|
||
if (!r.crop)
|
||
ihref += 'f';
|
||
if (r.x3)
|
||
ihref += '3';
|
||
if (href == "#")
|
||
ihref = SR + '/.cpr/ico/' + (ref == 'moar' ? '++' : 'exit');
|
||
}
|
||
else if (isdir) {
|
||
ihref = SR + '/.cpr/ico/folder';
|
||
}
|
||
else {
|
||
if (!svgs.has(ext)) {
|
||
if (svgs.size < max_svgs)
|
||
svgs.add(ext);
|
||
else
|
||
ext = "unk";
|
||
}
|
||
ihref = SR + '/.cpr/ico/' + ext;
|
||
}
|
||
ihref = addq(ihref, 'cache=i&_=' + ACB + TS);
|
||
if (CHROME)
|
||
ihref += "&raster";
|
||
|
||
html.push('<a href="' + ohref + '" ref="' + ref +
|
||
'"' + ac + ' ttt="' + esc(name) + '"><img style="height:' +
|
||
(r.sz / 1.25) + 'em" loading="lazy" onload="th_onload(this)" src="' +
|
||
ihref + '" /><span' + ac + '>' + ao.innerHTML + '</span></a>');
|
||
}
|
||
ggrid.innerHTML = html.join('\n');
|
||
clmod(ggrid, 'crop', r.crop);
|
||
clmod(ggrid, 'nocrop', !r.crop);
|
||
|
||
var srch = ebi('unsearch'),
|
||
gsel = ebi('gridsel');
|
||
|
||
gsel.style.display = srch ? 'none' : '';
|
||
if (srch && r.sel)
|
||
gsel.click();
|
||
|
||
var ths = QSA('#ggrid>a');
|
||
for (var a = 0, aa = ths.length; a < aa; a++) {
|
||
ths[a].ondblclick = gclick2;
|
||
ths[a].onclick = gclick1;
|
||
}
|
||
|
||
r.dirty = false;
|
||
r.bagit('#ggrid');
|
||
r.loadsel();
|
||
aligngriditems();
|
||
setTimeout(r.tippen, 20);
|
||
}
|
||
|
||
r.bagit = function (isrc) {
|
||
if (!window.baguetteBox)
|
||
return;
|
||
|
||
if (r.bbox)
|
||
baguetteBox.destroy();
|
||
|
||
var br = baguetteBox.run(isrc, {
|
||
noScrollbars: true,
|
||
duringHide: r.onhide,
|
||
afterShow: function () {
|
||
r.bbox_opts.refocus = true;
|
||
},
|
||
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('/').pop())) + '</a>';
|
||
},
|
||
onChange: function (i) {
|
||
sethash('g' + r.bbox[i].imageElement.getAttribute('ref') + getsort());
|
||
}
|
||
});
|
||
r.bbox = br[0][0];
|
||
r.bbox_opts = br[1];
|
||
};
|
||
|
||
r.onhide = function () {
|
||
afilt.apply();
|
||
|
||
if (!thegrid.ihop)
|
||
return;
|
||
|
||
try {
|
||
var el = QS('#ggrid a[ref="' + location.hash.slice(2) + '"]'),
|
||
f = function () {
|
||
try {
|
||
el.focus();
|
||
}
|
||
catch (ex) { }
|
||
};
|
||
|
||
f();
|
||
setTimeout(f, 10);
|
||
setTimeout(f, 100);
|
||
setTimeout(f, 200);
|
||
// thx fullscreen api
|
||
|
||
if (ANIM) {
|
||
clmod(el, 'glow', 1);
|
||
setTimeout(function () {
|
||
try {
|
||
clmod(el, 'glow');
|
||
}
|
||
catch (ex) { }
|
||
}, 600);
|
||
}
|
||
r.bbox_opts.refocus = false;
|
||
}
|
||
catch (ex) {
|
||
console.log('ihop:', ex);
|
||
}
|
||
};
|
||
|
||
r.set_crop = function (en) {
|
||
if (!dcrop.startsWith('f'))
|
||
return r.setdirty();
|
||
|
||
r.crop = dcrop.endsWith('y');
|
||
bcfg_upd_ui('gridcrop', r.crop);
|
||
if (r.crop != en)
|
||
toast.warn(10, L.ul_btnlk);
|
||
};
|
||
|
||
r.set_x3 = function (en) {
|
||
if (!dth3x.startsWith('f'))
|
||
return r.setdirty();
|
||
|
||
r.x3 = dth3x.endsWith('y');
|
||
bcfg_upd_ui('grid3x', r.x3);
|
||
if (r.x3 != en)
|
||
toast.warn(10, L.ul_btnlk);
|
||
};
|
||
|
||
if (/[?&]grid\b/.exec(sloc0))
|
||
swrite('griden', /[?&]grid=0\b/.exec(sloc0) ? 0 : 1)
|
||
|
||
if (/[?&]thumb\b/.exec(sloc0))
|
||
swrite('thumbs', /[?&]thumb=0\b/.exec(sloc0) ? 0 : 1)
|
||
|
||
if (/[?&]imgs\b/.exec(sloc0)) {
|
||
var n = /[?&]imgs=0\b/.exec(sloc0) ? 0 : 1;
|
||
swrite('griden', n);
|
||
if (n)
|
||
swrite('thumbs', 1);
|
||
}
|
||
|
||
bcfg_bind(r, 'thumbs', 'thumbs', true, r.setdirty);
|
||
bcfg_bind(r, 'ihop', 'ihop', true);
|
||
bcfg_bind(r, 'vau', 'gridvau', false);
|
||
bcfg_bind(r, 'crop', 'gridcrop', !dcrop.endsWith('n'), r.set_crop);
|
||
bcfg_bind(r, 'x3', 'grid3x', dth3x.endsWith('y'), r.set_x3);
|
||
bcfg_bind(r, 'sel', 'gridsel', false, r.loadsel);
|
||
bcfg_bind(r, 'en', 'griden', dgrid, function (v) {
|
||
v ? loadgrid() : r.setvis(true);
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
});
|
||
ebi('wtgrid').onclick = ebi('griden').onclick;
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
function th_onload(el) {
|
||
el.style.height = '';
|
||
}
|
||
|
||
|
||
function tree_scrollto(e) {
|
||
ev(e);
|
||
tree_scrolltoo('#treeul a.hl');
|
||
tree_scrolltoo('#docul a.hl');
|
||
}
|
||
|
||
|
||
function tree_scrolltoo(q) {
|
||
var act = QS(q),
|
||
ul = act ? act.offsetParent : null;
|
||
|
||
if (!ul)
|
||
return;
|
||
|
||
var ctr = ebi('tree'),
|
||
em = parseFloat(getComputedStyle(act).fontSize),
|
||
top = act.offsetTop + ul.offsetTop,
|
||
min = top - 20 * em,
|
||
max = top - (ctr.offsetHeight - 16 * 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(showfile.active() || treectl.texts ? '#docul li>a' : '#treeul li>a+a');
|
||
if (!links.length) {
|
||
treectl.dir_cb = function () {
|
||
tree_neigh(n);
|
||
treectl.detree();
|
||
};
|
||
treectl.entree(null, true);
|
||
return;
|
||
}
|
||
var act = -1;
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
if (clgot(links[a], 'hl')) {
|
||
act = a;
|
||
break;
|
||
}
|
||
}
|
||
if (act == -1 && !treectl.texts)
|
||
return;
|
||
|
||
act += n;
|
||
if (act < 0)
|
||
act = links.length - 1;
|
||
if (act >= links.length)
|
||
act = 0;
|
||
|
||
if (showfile.active())
|
||
links[act].click();
|
||
else
|
||
treectl.treego.call(links[act]);
|
||
}
|
||
|
||
|
||
function tree_up(justgo) {
|
||
if (showfile.active())
|
||
return thegrid.setvis(true);
|
||
|
||
var act = QS('#treeul a.hl');
|
||
if (!act) {
|
||
treectl.dir_cb = function () {
|
||
tree_up(justgo);
|
||
treectl.detree();
|
||
};
|
||
treectl.entree(null, true);
|
||
return;
|
||
}
|
||
if (act.previousSibling.textContent == '-') {
|
||
act.previousSibling.click();
|
||
if (!justgo)
|
||
return;
|
||
}
|
||
var a = act.parentNode.parentNode.parentNode.getElementsByTagName('a')[1];
|
||
if (a.parentNode.tagName == 'LI')
|
||
a.click();
|
||
}
|
||
|
||
|
||
function hkhelp() {
|
||
var html = [];
|
||
for (var ic = 0; ic < L.hks.length; ic++) {
|
||
var c = L.hks[ic];
|
||
html.push('<table>');
|
||
for (var a = 0; a < c.length; a++)
|
||
try {
|
||
if (c[a].length != 2)
|
||
html.push('<tr><th colspan="2">' + esc(c[a]) + '</th></tr>');
|
||
else {
|
||
var t1 = c[a][0].replace('⇧', '<b>⇧</b>');
|
||
html.push('<tr><td>{0}</td><td>{1}</td></tr>'.format(t1, c[a][1]));
|
||
}
|
||
}
|
||
catch (ex) {
|
||
html.push(">>> " + c[a]);
|
||
}
|
||
|
||
html.push('</table>');
|
||
}
|
||
qsr('#hkhelp');
|
||
var o = mknod('div', 'hkhelp');
|
||
o.innerHTML = html.join('\n');
|
||
document.body.appendChild(o);
|
||
}
|
||
|
||
|
||
var fselgen, fselctr;
|
||
function fselfunw(e, ae, d, rem) {
|
||
fselctr = 0;
|
||
var gen = fselgen = Date.now();
|
||
if (rem)
|
||
rem *= window.innerHeight;
|
||
|
||
var selfun = function () {
|
||
var el = ae[d + 'ElementSibling'];
|
||
if (!el || gen != fselgen)
|
||
return;
|
||
|
||
el.focus();
|
||
var elh = el.offsetHeight;
|
||
if (ctrl(e))
|
||
document.documentElement.scrollTop += (d == 'next' ? 1 : -1) * elh;
|
||
|
||
if (e.shiftKey) {
|
||
clmod(el, 'sel', 't');
|
||
msel.origin_tr(el);
|
||
msel.selui();
|
||
}
|
||
|
||
rem -= elh;
|
||
if (rem > 0) {
|
||
ae = document.activeElement;
|
||
if (++fselctr % 5 && rem > elh * (FIREFOX ? 5 : 2))
|
||
selfun();
|
||
else
|
||
setTimeout(selfun, 1);
|
||
}
|
||
}
|
||
selfun();
|
||
}
|
||
var ahotkeys = function (e) {
|
||
if (e.altKey || e.isComposing)
|
||
return;
|
||
|
||
if (QS('#bbox-overlay.visible') || modal.busy)
|
||
return;
|
||
|
||
var k = (e.code || e.key) + '', pos = -1, n,
|
||
ae = document.activeElement,
|
||
aet = ae && ae != document.body ? ae.nodeName.toLowerCase() : '';
|
||
|
||
if (k == 'Escape' || k == 'Esc') {
|
||
ae && ae.blur();
|
||
tt.hide();
|
||
|
||
if (ebi('hkhelp'))
|
||
return qsr('#hkhelp');
|
||
|
||
if (toast.visible)
|
||
return toast.hide();
|
||
|
||
if (ebi('rn_cancel'))
|
||
return ebi('rn_cancel').click();
|
||
|
||
if (ebi('sh_abrt'))
|
||
return ebi('sh_abrt').click();
|
||
|
||
if (QS('.opview.act'))
|
||
return QS('#ops>a').click();
|
||
|
||
if (widget.is_open)
|
||
return widget.close();
|
||
|
||
if (showfile.active())
|
||
return thegrid.setvis(true);
|
||
|
||
if (!treectl.hidden)
|
||
return treectl.detree();
|
||
|
||
if (QS('#unsearch'))
|
||
return QS('#unsearch').click();
|
||
|
||
if (thegrid.en)
|
||
return ebi('griden').click();
|
||
}
|
||
|
||
var in_ftab = (aet == 'tr' || aet == 'td') && ae.closest('#files');
|
||
if (in_ftab) {
|
||
var d = '', rem = 0;
|
||
if (aet == 'td') ae = ae.closest('tr'); //ie11
|
||
if (k == 'ArrowUp' || k == 'Up') d = 'previous';
|
||
if (k == 'ArrowDown' || k == 'Down') d = 'next';
|
||
if (k == 'PageUp') { d = 'previous'; rem = 0.6; }
|
||
if (k == 'PageDown') { d = 'next'; rem = 0.6; }
|
||
if (d) {
|
||
fselfunw(e, ae, d, rem);
|
||
return ev(e);
|
||
}
|
||
if (k == 'Space' || k == 'Spacebar') {
|
||
clmod(ae, 'sel', 't');
|
||
msel.origin_tr(ae);
|
||
msel.selui();
|
||
return ev(e);
|
||
}
|
||
}
|
||
if (in_ftab || !aet || (ae && ae.closest('#ggrid'))) {
|
||
if ((k == 'KeyA' || k == 'a') && ctrl(e)) {
|
||
var ntot = treectl.lsc.files.length + treectl.lsc.dirs.length,
|
||
sel = msel.getsel(),
|
||
all = msel.getall();
|
||
|
||
msel.evsel(e, sel.length < all.length);
|
||
msel.origin_id(null);
|
||
if (ntot > all.length)
|
||
toast.warn(10, L.f_anota.format(all.length, ntot), L.f_anota);
|
||
else if (toast.tag == L.f_anota)
|
||
toast.hide();
|
||
return ev(e);
|
||
}
|
||
}
|
||
|
||
if (ae && ae.closest('pre')) {
|
||
if ((k == 'KeyA' || k == 'a') && ctrl(e)) {
|
||
var sel = document.getSelection(),
|
||
ran = document.createRange();
|
||
|
||
sel.removeAllRanges();
|
||
ran.selectNode(ae.closest('pre'));
|
||
sel.addRange(ran);
|
||
return ev(e);
|
||
}
|
||
}
|
||
|
||
if (k.endsWith('Enter') && ae && (ae.onclick || ae.hasAttribute('tabIndex')))
|
||
return ev(e) && ae.click() || true;
|
||
|
||
if (aet && aet != 'a' && aet != 'tr' && aet != 'td' && aet != 'div' && aet != 'pre')
|
||
return;
|
||
|
||
if (e.key == '?')
|
||
return hkhelp();
|
||
|
||
if (!e.shiftKey && ctrl(e)) {
|
||
var sel = window.getSelection && window.getSelection() || {};
|
||
sel = sel && !sel.isCollapsed && sel.direction != 'none';
|
||
|
||
if (k == 'KeyX' || k == 'x')
|
||
return fileman.cut(e);
|
||
|
||
if ((k == 'KeyC' || k == 'c') && !sel)
|
||
return fileman.cpy(e);
|
||
|
||
if (k == 'KeyV' || k == 'v')
|
||
return fileman.d_paste(e);
|
||
|
||
if (k == 'KeyK' || k == 'k')
|
||
return fileman.delete(e);
|
||
|
||
return;
|
||
}
|
||
|
||
if (e.shiftKey && k != 'KeyA' && k != 'KeyD' && k != 'A' && k != 'D')
|
||
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' || k == 'j')
|
||
return prev_song() || true;
|
||
|
||
if (k == 'KeyL' || k == 'l')
|
||
return next_song() || true;
|
||
|
||
if (k == 'KeyP' || k == 'p')
|
||
return playpause() || true;
|
||
|
||
n = (k == 'KeyU' || k == 'u') ? -10 :
|
||
(k == 'KeyO' || k == 'o') ? 10 : 0;
|
||
if (n !== 0)
|
||
return seek_au_rel(n) || true;
|
||
|
||
if (k == 'KeyY')
|
||
return msel.getsel().length ? ebi('seldl').click() :
|
||
showfile.active() ? ebi('dldoc').click() :
|
||
dl_song();
|
||
|
||
n = (k == 'KeyI' || k == 'i') ? -1 :
|
||
(k == 'KeyK' || k == 'k') ? 1 : 0;
|
||
if (n !== 0)
|
||
return tree_neigh(n);
|
||
|
||
if (k == 'KeyM' || k == 'm')
|
||
return tree_up();
|
||
|
||
if (k == 'KeyB' || k == 'b')
|
||
return treectl.hidden ? treectl.entree() : treectl.detree();
|
||
|
||
if (k == 'KeyG' || k == 'g')
|
||
return ebi('griden').click();
|
||
|
||
if (k == 'KeyT' || k == 't')
|
||
return ebi('thumbs').click();
|
||
|
||
if (k == 'KeyV' || k == 'v')
|
||
return ebi('filetree').click();
|
||
|
||
if (k == 'F2')
|
||
return fileman.rename();
|
||
|
||
if (!treectl.hidden && (!e.shiftKey || !thegrid.en)) {
|
||
if (k == 'KeyA' || k == 'a')
|
||
return QS('#twig').click();
|
||
|
||
if (k == 'KeyD' || k == 'd')
|
||
return QS('#twobytwo').click();
|
||
}
|
||
|
||
if (showfile.active()) {
|
||
if (k == 'KeyS' || k == 's')
|
||
showfile.tglsel();
|
||
if ((k == 'KeyE' || k == 'e') && ebi('editdoc').style.display != 'none')
|
||
ebi('editdoc').click();
|
||
}
|
||
|
||
if (mp && mp.au && !mp.au.paused) {
|
||
if (k == 'KeyS')
|
||
return sel_song();
|
||
}
|
||
|
||
if (thegrid.en) {
|
||
if (k == 'KeyS' || k == 's')
|
||
return ebi('gridsel').click();
|
||
|
||
if (k == 'KeyA' || k == 'a')
|
||
return QSA('#ghead a[z]')[0].click();
|
||
|
||
if (k == 'KeyD' || k == 'd')
|
||
return QSA('#ghead a[z]')[1].click();
|
||
}
|
||
};
|
||
|
||
|
||
// search
|
||
var search_ui = (function () {
|
||
var sconf = [
|
||
[
|
||
L.s_sz,
|
||
["szl", "sz_min", L.s_s1, "14", ""],
|
||
["szu", "sz_max", L.s_s2, "14", ""]
|
||
],
|
||
[
|
||
L.s_dt,
|
||
["dtl", "dt_min", L.s_d1, "14", "1997-08-15, 01:00"],
|
||
["dtu", "dt_max", L.s_d2, "14", "2020"]
|
||
],
|
||
[
|
||
L.s_rd,
|
||
["path", "path", L.s_r1, "30", "windows -system32"]
|
||
],
|
||
[
|
||
L.s_fn,
|
||
["name", "name", L.s_f1, "30", ".exe$"]
|
||
],
|
||
[
|
||
L.s_ta,
|
||
["tags", "tags", L.s_t1, "30", "^irui$"]
|
||
],
|
||
[
|
||
L.s_ad,
|
||
["adv", "adv", L.s_a1, "30", "key>=1A key<=2B .bpm>165"]
|
||
],
|
||
[
|
||
L.s_ua,
|
||
["utl", "ut_min", L.s_u1, "14", "2007-04-08"],
|
||
["utu", "ut_max", L.s_u2, "14", "2038-01-19"]
|
||
]
|
||
];
|
||
|
||
var r = {},
|
||
trs = [],
|
||
orig_url = null,
|
||
orig_html = null,
|
||
cap = 125;
|
||
|
||
for (var a = 0; a < sconf.length; a++) {
|
||
var html = ['<tr id="tsrch_' + sconf[a][1][0] + '"><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" style="width:' + sconf[a][b][3] +
|
||
'em" name="' + sconf[a][b][1] + '" placeholder="' + sconf[a][b][4] + '" /></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" ' + NOAC + ' placeholder="( tags like *nhato* or tags like *taishi* ) and ( not tags like *nhato* or not tags like *taishi* )" /></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;
|
||
o[a].onkeydown = ev_search_keydown;
|
||
}
|
||
|
||
function srch_msg(err, txt) {
|
||
var o = ebi('srch_q');
|
||
o.textContent = txt;
|
||
clmod(o, 'err', err);
|
||
}
|
||
|
||
var search_timeout,
|
||
defer_timeout,
|
||
search_in_progress = 0;
|
||
|
||
function ev_search_input() {
|
||
var v = unsmart(this.value),
|
||
id = this.getAttribute('id'),
|
||
is_txt = id.slice(-1) == 'v',
|
||
is_chk = id.slice(-1) == 'c';
|
||
|
||
if (is_txt) {
|
||
var chk = ebi(id.slice(0, -1) + 'c');
|
||
chk.checked = ((v + '').length > 0);
|
||
}
|
||
|
||
if (id != "q_raw")
|
||
encode_query();
|
||
|
||
set_vq();
|
||
cap = 125;
|
||
|
||
clearTimeout(defer_timeout);
|
||
if (is_chk)
|
||
return do_search();
|
||
|
||
defer_timeout = setTimeout(try_search, 2000);
|
||
try_search(v);
|
||
}
|
||
|
||
function ev_search_keydown(e) {
|
||
if ((e.key + '').endsWith('Enter'))
|
||
do_search();
|
||
}
|
||
|
||
function try_search(v) {
|
||
if (Date.now() - search_in_progress > 30 * 1000) {
|
||
clearTimeout(defer_timeout);
|
||
clearTimeout(search_timeout);
|
||
search_timeout = setTimeout(do_search,
|
||
v && v.length < (MOBILE ? 4 : 3) ? 1000 : 500);
|
||
}
|
||
}
|
||
|
||
function set_vq() {
|
||
if (search_in_progress)
|
||
return;
|
||
|
||
var q = unsmart(ebi('q_raw').value),
|
||
vq = ebi('files').getAttribute('q_raw');
|
||
|
||
srch_msg(false, (q == vq) ? '' : L.sm_prev + (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',
|
||
vs = unsmart(ebi('srch_' + k + 'v').value),
|
||
tvs = [];
|
||
|
||
if (a == 1)
|
||
vs = vs.trim().replace(/ +/, 'T');
|
||
|
||
while (vs) {
|
||
vs = vs.trim();
|
||
if (!vs)
|
||
break;
|
||
|
||
var v = '';
|
||
if (vs.startsWith('"')) {
|
||
var vp = vs.slice(1).split(/"(.*)/);
|
||
v = vp[0];
|
||
vs = vp[1] || '';
|
||
while (v.endsWith('\\')) {
|
||
vp = vs.split(/"(.*)/);
|
||
v = v.slice(0, -1) + '"' + vp[0];
|
||
vs = vp[1] || '';
|
||
}
|
||
}
|
||
else {
|
||
var vp = vs.split(/ +(.*)/);
|
||
v = vp[0].replace(/\\"/g, '"');
|
||
vs = vp[1] || '';
|
||
}
|
||
tvs.push(v);
|
||
}
|
||
|
||
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(/l$/, ' >= ').replace(/u$/, ' <= ').replace(/^sz/, 'size').replace(/^dt/, 'date').replace(/^ut/, 'up_at') + 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 += '*';
|
||
}
|
||
|
||
if (tv.indexOf(' ') + 1) {
|
||
tv = '"' + tv + '"';
|
||
}
|
||
|
||
q += not + k + ' like ' + tv;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
ebi('q_raw').value = q.slice(5);
|
||
}
|
||
|
||
function do_search() {
|
||
search_in_progress = Date.now();
|
||
srch_msg(false, L.sm_w8);
|
||
clearTimeout(search_timeout);
|
||
|
||
var xhr = new XHR();
|
||
xhr.open('POST', SR + '/?srch', true);
|
||
xhr.setRequestHeader('Content-Type', 'text/plain');
|
||
xhr.onload = xhr.onerror = xhr_search_results;
|
||
xhr.ts = Date.now();
|
||
xhr.q_raw = unsmart(ebi('q_raw').value);
|
||
xhr.send(JSON.stringify({ "q": xhr.q_raw, "n": cap }));
|
||
}
|
||
|
||
function xhr_search_results() {
|
||
if (this.status !== 200) {
|
||
var msg = hunpre(this.responseText);
|
||
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);
|
||
r.render(res, this, true);
|
||
}
|
||
|
||
r.render = function (res, xhr, sort) {
|
||
var tagord = res.tag_order;
|
||
|
||
srch_msg(false, '');
|
||
if (sort)
|
||
sortfiles(res.hits);
|
||
|
||
var ofiles = ebi('files');
|
||
if (xhr && ofiles.getAttribute('ts') > xhr.ts)
|
||
return;
|
||
|
||
treectl.hide();
|
||
thegrid.setvis(true);
|
||
|
||
var html = mk_files_header(tagord), seen = {};
|
||
html.push('<tbody>');
|
||
html.push('<tr class="srch_hdr"><td>-</td><td><a href="#" id="unsearch"><big style="font-weight:bold">[❌] ' + L.sl_close + '</big></a> -- ' + L.sl_hits.format(res.hits.length) + (res.trunc ? ' -- <a href="#" id="moar">' + L.sl_moar + '</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(uricom_dec(r.rp + '')),
|
||
ext = rp.lastIndexOf('.') > 0 ? rp.split('.').pop().split('?')[0] : '%',
|
||
id = 'f-' + ('00000000' + crc32(rp)).slice(-8);
|
||
|
||
while (seen[id])
|
||
id += 'a';
|
||
seen[id] = 1;
|
||
|
||
if (ext.length > 8)
|
||
ext = '%';
|
||
|
||
var links = linksplit(r.rp + '', id).join('<span>/</span>'),
|
||
nodes = ['<tr><td>-</td><td><div>' + links + '</div>', sz];
|
||
|
||
for (var b = 0; b < tagord.length; b++) {
|
||
var k = esc(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(esc('' + 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 = set_files_html(html.join('\n'));
|
||
ofiles.setAttribute("ts", xhr ? xhr.ts : 1);
|
||
ofiles.setAttribute("q_raw", xhr ? xhr.q_raw : 'playlist');
|
||
set_vq();
|
||
mukey.render();
|
||
reload_browser();
|
||
filecols.set_style(['File Name']);
|
||
|
||
if (xhr)
|
||
sethash('q=' + uricom_enc(xhr.q_raw));
|
||
|
||
ebi('unsearch').onclick = unsearch;
|
||
var m = ebi('moar');
|
||
if (m)
|
||
m.onclick = moar;
|
||
};
|
||
|
||
function unsearch(e) {
|
||
ev(e);
|
||
treectl.show();
|
||
set_files_html(orig_html);
|
||
ebi('files').removeAttribute('q_raw');
|
||
orig_html = null;
|
||
sethash('');
|
||
reload_browser();
|
||
}
|
||
|
||
function moar(e) {
|
||
ev(e);
|
||
cap *= 2;
|
||
do_search();
|
||
}
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
function ev_load_m3u(e) {
|
||
ev(e);
|
||
var id = this.getAttribute('id').slice(1),
|
||
url = ebi(id).getAttribute('href').split('?')[0];
|
||
|
||
modal.confirm(L.mm_m3u,
|
||
function () { load_m3u(url); },
|
||
function () {
|
||
if (has(perms, 'write') && has(perms, 'delete'))
|
||
window.location = url + '?edit';
|
||
else
|
||
showfile.show(url);
|
||
}
|
||
);
|
||
return false;
|
||
}
|
||
function load_m3u(url) {
|
||
assert_vp(url);
|
||
var xhr = new XHR();
|
||
xhr.open('GET', url, true);
|
||
xhr.onload = render_m3u;
|
||
xhr.url = url;
|
||
xhr.send();
|
||
return false;
|
||
}
|
||
function render_m3u() {
|
||
if (!xhrchk(this, L.tv_xe1, L.tv_xe2))
|
||
return;
|
||
|
||
var evp = get_evpath(),
|
||
m3u = this.responseText,
|
||
xtd = m3u.slice(0, 12).indexOf('#EXTM3U') + 1,
|
||
lines = m3u.replace(/\r/g, '\n').split('\n'),
|
||
dur = 1,
|
||
artist = '',
|
||
title = '',
|
||
ret = {'hits': [], 'tag_order': ['artist', 'title', '.dur'], 'trunc': false};
|
||
|
||
for (var a = 0; a < lines.length; a++) {
|
||
var ln = lines[a].trim();
|
||
if (xtd && ln.startsWith('#')) {
|
||
var m = /^#EXTINF:([0-9]+)[, ](.*)/.exec(ln);
|
||
if (m) {
|
||
dur = m[1];
|
||
title = m[2];
|
||
var ofs = title.indexOf(' - ');
|
||
if (ofs > 0) {
|
||
artist = title.slice(0, ofs);
|
||
title = title.slice(ofs + 3);
|
||
}
|
||
}
|
||
continue;
|
||
}
|
||
if (ln.indexOf('.') < 0)
|
||
continue;
|
||
|
||
var n = ret.hits.length + 1,
|
||
url = ln;
|
||
|
||
if (url.indexOf(':\\')) // C:\
|
||
url = url.split(/\\/g).pop();
|
||
|
||
url = url.replace(/\\/g, '/');
|
||
url = uricom_enc(url).replace(/%2f/gi, '/')
|
||
|
||
if (!url.startsWith('/'))
|
||
url = vjoin(evp, url);
|
||
|
||
ret.hits.push({
|
||
"ts": 946684800 + n,
|
||
"sz": 100000 + n,
|
||
"rp": url,
|
||
"tags": {".dur": dur, "artist": artist, "title": title}
|
||
});
|
||
dur = 1;
|
||
artist = title = '';
|
||
}
|
||
|
||
search_ui.render(ret, null, false);
|
||
sethash('m3u=' + this.url.split('?')[0].split('/').pop());
|
||
goto();
|
||
|
||
var el = QS('#files>tbody>tr.au>td>a.play');
|
||
if (el)
|
||
el.click();
|
||
}
|
||
|
||
|
||
function aligngriditems() {
|
||
if (!treectl)
|
||
return;
|
||
|
||
var ggrid = ebi('ggrid'),
|
||
em2px = parseFloat(getComputedStyle(ggrid).fontSize),
|
||
gridsz = 10;
|
||
try {
|
||
gridsz = cprop('--grid-sz').slice(0, -2);
|
||
}
|
||
catch (ex) { }
|
||
var gridwidth = ggrid.clientWidth,
|
||
griditemcount = ggrid.children.length,
|
||
totalgapwidth = em2px * griditemcount;
|
||
|
||
if (/b/.test(themen + ''))
|
||
totalgapwidth *= 2.8;
|
||
|
||
var val, st = ggrid.style;
|
||
|
||
if (((griditemcount * em2px) * gridsz) + totalgapwidth < gridwidth) {
|
||
val = 'left';
|
||
} else {
|
||
val = treectl.hidden ? 'center' : 'space-between';
|
||
}
|
||
if (st.justifyContent != val)
|
||
st.justifyContent = val;
|
||
}
|
||
onresize100.add(aligngriditems);
|
||
|
||
|
||
var filecolwidth = (function () {
|
||
var lastwidth = -1;
|
||
|
||
return function () {
|
||
var vw = window.innerWidth / parseFloat(getComputedStyle(document.body)['font-size']),
|
||
w = Math.floor(vw - 2);
|
||
|
||
if (w == lastwidth)
|
||
return;
|
||
|
||
lastwidth = w;
|
||
setcvar('--file-td-w', w + 'em');
|
||
}
|
||
})();
|
||
onresize100.add(filecolwidth, true);
|
||
|
||
|
||
var treectl = (function () {
|
||
var r = {
|
||
"hidden": true,
|
||
"sb_msg": false,
|
||
"ls_cb": null,
|
||
"dir_cb": tree_scrollto,
|
||
"pdir": []
|
||
},
|
||
entreed = false,
|
||
fixedpos = false,
|
||
prev_atop = null,
|
||
prev_winh = null,
|
||
mentered = null,
|
||
treesz = clamp(icfg_get('treesz', 16), 10, 50);
|
||
|
||
var resort = function () {
|
||
ENATSORT = NATSORT && clgot(ebi('nsort'), 'on');
|
||
treectl.gentab(get_evpath(), treectl.lsc);
|
||
};
|
||
bcfg_bind(r, 'ireadme', 'ireadme', true);
|
||
bcfg_bind(r, 'idxh', 'idxh', idxh, setidxh);
|
||
bcfg_bind(r, 'dyn', 'dyntree', true, onresize);
|
||
bcfg_bind(r, 'csel', 'csel', dgsel);
|
||
bcfg_bind(r, 'dots', 'dotfiles', see_dots, function (v) {
|
||
r.goto();
|
||
var xhr = new XHR();
|
||
xhr.open('GET', SR + '/?setck=dots=' + (v ? 'y' : ''), true);
|
||
xhr.send();
|
||
});
|
||
bcfg_bind(r, 'nsort', 'nsort', dnsort, resort);
|
||
bcfg_bind(r, 'dir1st', 'dir1st', true, resort);
|
||
setwrap(bcfg_bind(r, 'wtree', 'wraptree', true, setwrap));
|
||
setwrap(bcfg_bind(r, 'parpane', 'parpane', true, onscroll));
|
||
bcfg_bind(r, 'htree', 'hovertree', false, reload_tree);
|
||
bcfg_bind(r, 'ask', 'bd_ask', MOBILE && FIREFOX);
|
||
ebi('bd_lim').value = r.lim = icfg_get('bd_lim');
|
||
ebi('bd_lim').oninput = function (e) {
|
||
var n = parseInt(this.value);
|
||
swrite('bd_lim', r.lim = (isNum(n) ? n : 0) || 1000);
|
||
};
|
||
r.nvis = r.lim;
|
||
|
||
ldks = jread('dks', []);
|
||
for (var a = ldks.length - 1; a >= 0; a--) {
|
||
var s = ldks[a],
|
||
o = s.lastIndexOf('?');
|
||
|
||
dks[s.slice(0, o)] = s.slice(o + 1);
|
||
}
|
||
|
||
function setwrap(v) {
|
||
clmod(ebi('tree'), 'nowrap', !v);
|
||
reload_tree();
|
||
}
|
||
setwrap(r.wtree);
|
||
|
||
function setidxh(v) {
|
||
if (!v == !/\bidxh=y\b/.exec('' + document.cookie))
|
||
return;
|
||
|
||
var xhr = new XHR();
|
||
xhr.open('GET', SR + '/?setck=idxh=' + (v ? 'y' : 'n'), true);
|
||
xhr.send();
|
||
}
|
||
setidxh(r.idxh);
|
||
|
||
r.entree = function (e, nostore) {
|
||
ev(e);
|
||
entreed = true;
|
||
if (!nostore)
|
||
swrite('entreed', 'tree');
|
||
|
||
get_tree("", get_evpath(), true);
|
||
r.show();
|
||
}
|
||
|
||
r.show = function () {
|
||
r.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();
|
||
aligngriditems();
|
||
};
|
||
|
||
r.detree = function (e) {
|
||
ev(e);
|
||
entreed = false;
|
||
swrite('entreed', 'na');
|
||
|
||
r.hide();
|
||
ebi('path').style.display = '';
|
||
}
|
||
|
||
r.hide = function () {
|
||
r.hidden = true;
|
||
ebi('path').style.display = 'none';
|
||
ebi('tree').style.display = 'none';
|
||
ebi('wrap').style.marginLeft = '';
|
||
window.removeEventListener('resize', onresize);
|
||
window.removeEventListener('scroll', onscroll);
|
||
aligngriditems();
|
||
}
|
||
|
||
function unmenter() {
|
||
if (mentered) {
|
||
mentered.style.position = '';
|
||
mentered = null;
|
||
}
|
||
}
|
||
|
||
r.textmode = function (ya) {
|
||
var chg = !r.texts != !ya;
|
||
r.texts = ya;
|
||
ebi('docul').style.display = ya ? '' : 'none';
|
||
ebi('treeul').style.display = ebi('treepar').style.display = ya ? 'none' : '';
|
||
clmod(ebi('filetree'), 'on', ya);
|
||
if (chg)
|
||
tree_scrollto();
|
||
};
|
||
ebi('filetree').onclick = function (e) {
|
||
ev(e);
|
||
r.textmode(!r.texts);
|
||
};
|
||
r.textmode(false);
|
||
|
||
function onscroll() {
|
||
unmenter();
|
||
onscroll2();
|
||
}
|
||
|
||
function onscroll2() {
|
||
if (!entreed || r.hidden || document.visibilityState == 'hidden')
|
||
return;
|
||
|
||
var tree = ebi('tree'),
|
||
wrap = ebi('wrap'),
|
||
wraptop = null,
|
||
atop = wrap.getBoundingClientRect().top,
|
||
winh = window.innerHeight,
|
||
parp = ebi('treepar'),
|
||
y = tree.scrollTop,
|
||
w = tree.offsetWidth;
|
||
|
||
if (atop !== prev_atop || winh !== prev_winh)
|
||
wraptop = Math.floor(wrap.offsetTop);
|
||
|
||
if (r.parpane && r.pdir.length && w != r.pdirw) {
|
||
r.pdirw = w;
|
||
compy();
|
||
}
|
||
|
||
if (!r.parpane || !r.pdir.length || y >= r.pdir.slice(-1)[0][0] || y <= r.pdir[0][0]) {
|
||
clmod(parp, 'off', 1);
|
||
r.pdirh = null;
|
||
}
|
||
else {
|
||
var h1 = [], h2 = [], els = [];
|
||
for (var a = 0; a < r.pdir.length; a++) {
|
||
if (r.pdir[a][0] > y)
|
||
break;
|
||
|
||
var e2 = r.pdir[a][1], e1 = e2.previousSibling;
|
||
h1.push('<li>' + e1.outerHTML + e2.outerHTML + '<ul>');
|
||
h2.push('</ul></li>');
|
||
els.push([e1, e2]);
|
||
}
|
||
h1 = h1.join('\n') + h2.join('\n');
|
||
if (h1 != r.pdirh) {
|
||
r.pdirh = h1;
|
||
parp.innerHTML = h1;
|
||
clmod(parp, 'off');
|
||
var els = QSA('#treepar a');
|
||
for (var a = 0, aa = els.length; a < aa; a++)
|
||
els[a].onclick = bad_proxy;
|
||
}
|
||
y = ebi('treeh').offsetHeight;
|
||
if (!fixedpos)
|
||
y += tree.offsetTop - yscroll();
|
||
|
||
y = (y - 3) + 'px';
|
||
if (parp.style.top != y)
|
||
parp.style.top = y;
|
||
}
|
||
|
||
if (wraptop === null)
|
||
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, wraptop),
|
||
treeh = winh - atop;
|
||
|
||
tree.style.top = top + 'px';
|
||
tree.style.height = treeh < 10 ? '' : Math.floor(treeh) + 'px';
|
||
}
|
||
}
|
||
timer.add(onscroll2, true);
|
||
|
||
function onresize(e) {
|
||
if (!entreed || r.hidden)
|
||
return;
|
||
|
||
var q = '#tree',
|
||
nq = -3;
|
||
|
||
while (r.dyn) {
|
||
nq++;
|
||
q += '>ul>li';
|
||
if (!QS(q))
|
||
break;
|
||
}
|
||
nq = Math.max(nq, get_evpath().split('/').length - 2);
|
||
var iw = (treesz + Math.max(0, nq)),
|
||
w = iw + 'em',
|
||
w2 = (iw + 2) + 'em';
|
||
|
||
setcvar('--nav-sz', w);
|
||
ebi('tree').style.width = w;
|
||
ebi('wrap').style.marginLeft = w2;
|
||
onscroll();
|
||
}
|
||
|
||
r.find = function (txt) {
|
||
var ta = QSA('#treeul a.hl+ul>li>a+a');
|
||
for (var a = 0, aa = ta.length; a < aa; a++)
|
||
if (ta[a].textContent == txt)
|
||
return ta[a];
|
||
};
|
||
|
||
r.goto = function (url, push, back) {
|
||
if (!url || !url.startsWith('/'))
|
||
url = get_evpath() + (url || '');
|
||
|
||
get_tree("", url, true);
|
||
r.reqls(url, push, back);
|
||
};
|
||
|
||
function get_tree(top, dst, rst) {
|
||
var xhr = new XHR(),
|
||
m = /[?&](k=[^&#]+)/.exec(dst),
|
||
k = m ? '&' + m[1] : dk ? '&k=' + dk : '';
|
||
|
||
xhr.top = top;
|
||
xhr.dst = dst;
|
||
xhr.rst = rst;
|
||
xhr.ts = Date.now();
|
||
xhr.open('GET', addq(dst, 'tree=' + top + (r.dots ? '&dots' : '') + k), true);
|
||
xhr.onload = xhr.onerror = r.recvtree;
|
||
xhr.send();
|
||
enspin('t');
|
||
}
|
||
|
||
r.recvtree = function () {
|
||
if (!xhrchk(this, L.tl_xe1, L.tl_xe2))
|
||
return;
|
||
|
||
try {
|
||
var res = JSON.parse(this.responseText);
|
||
}
|
||
catch (ex) {
|
||
return toast.err(30, "bad <code>?tree</code> reply;\nexpected json, got this:\n\n" + esc(this.responseText + ''));
|
||
}
|
||
r.rendertree(res, this.ts, this.top, this.dst, this.rst);
|
||
|
||
if (r.lsc && r.lsc.unlist)
|
||
r.prunetree(r.lsc);
|
||
};
|
||
|
||
r.prunetree = function (res) {
|
||
var ptn = new RegExp(res.unlist);
|
||
var els = QSA('#treeul li>a+a');
|
||
for (var a = els.length - 1; a >= 0; a--)
|
||
if (ptn.exec(els[a].textContent) && !els[a].className)
|
||
els[a].closest('ul').removeChild(els[a].closest('li'));
|
||
};
|
||
|
||
r.rendertree = function (res, ts, top0, dst, rst) {
|
||
var cur = ebi('treeul').getAttribute('ts');
|
||
if (cur && parseInt(cur) > ts + 20 && QS('#treeul>li>a+a')) {
|
||
console.log("reject tree; " + cur + " / " + (ts - cur));
|
||
return;
|
||
}
|
||
ebi('treeul').setAttribute('ts', ts);
|
||
|
||
var top = (top0 == '.' ? dst : top0).split('?')[0],
|
||
name = uricom_dec(top.split('/').slice(-2)[0]),
|
||
rtop = top.replace(/^\/+/, ""),
|
||
html = parsetree(res, rtop);
|
||
|
||
if (!top0) {
|
||
html = '<li><a href="#">-</a><a href="/">[root]</a>\n<ul>' + html;
|
||
if (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').split('?')[0] == top) {
|
||
var o = links[a].parentNode;
|
||
if (!o.getElementsByTagName('li').length)
|
||
o.innerHTML = html;
|
||
}
|
||
}
|
||
}
|
||
qsr('#dlt_t');
|
||
|
||
try {
|
||
QS('#treeul>li>a+a').textContent = '[root]';
|
||
}
|
||
catch (ex) {
|
||
console.log('got no root yet');
|
||
r.dir_cb = null;
|
||
return;
|
||
}
|
||
|
||
reload_tree();
|
||
var fun = r.dir_cb;
|
||
if (fun) {
|
||
r.dir_cb = null;
|
||
try {
|
||
fun();
|
||
}
|
||
catch (ex) {
|
||
console.log("dir_cb failed", ex);
|
||
}
|
||
}
|
||
};
|
||
|
||
function reload_tree() {
|
||
var cevp = get_evpath(),
|
||
cdir = r.nextdir || uricom_dec(cevp),
|
||
links = QSA('#treeul a+a'),
|
||
nowrap = QS('#tree.nowrap') && QS('#hovertree.on'),
|
||
act = null;
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
var qhref = links[a].getAttribute('href'),
|
||
ehref = qhref.split('?')[0],
|
||
href = uricom_dec(ehref),
|
||
cl = '';
|
||
|
||
if (dk && ehref == cevp && !/[?&]k=/.exec(qhref))
|
||
links[a].setAttribute('href', addq(qhref, 'k=' + dk));
|
||
|
||
if (href == cdir) {
|
||
act = links[a];
|
||
cl = 'hl';
|
||
}
|
||
else if (cdir.startsWith(href)) {
|
||
cl = 'par';
|
||
}
|
||
|
||
links[a].className = cl;
|
||
links[a].onclick = r.treego;
|
||
links[a].onmouseenter = nowrap ? menter : null;
|
||
links[a].onmouseleave = nowrap ? mleave : null;
|
||
}
|
||
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;
|
||
}
|
||
ebi('tree').onscroll = nowrap ? unmenter : null;
|
||
r.pdir = [];
|
||
try {
|
||
while (act) {
|
||
r.pdir.unshift([-1, act]);
|
||
act = act.parentNode.parentNode.closest('li').querySelector('a:first-child+a');
|
||
}
|
||
}
|
||
catch (ex) { }
|
||
r.pdir.shift();
|
||
r.pdirw = -1;
|
||
onresize();
|
||
}
|
||
|
||
function compy() {
|
||
for (var a = 0; a < r.pdir.length; a++)
|
||
r.pdir[a][0] = r.pdir[a][1].offsetTop;
|
||
|
||
var ofs = 0;
|
||
for (var a = 0; a < r.pdir.length - 1; a++) {
|
||
ofs += r.pdir[a][1].offsetHeight + 1;
|
||
r.pdir[a + 1][0] -= ofs;
|
||
}
|
||
}
|
||
|
||
function menter(e) {
|
||
var p = this.offsetParent,
|
||
pp = p.offsetParent,
|
||
ppy = pp.offsetTop,
|
||
y = this.offsetTop + p.offsetTop + ppy - p.scrollTop - pp.scrollTop - (ppy ? document.documentElement.scrollTop : 0);
|
||
|
||
this.style.top = y + 'px';
|
||
this.style.position = 'fixed';
|
||
mentered = this;
|
||
}
|
||
|
||
function mleave(e) {
|
||
this.style.position = '';
|
||
mentered = null;
|
||
}
|
||
|
||
function bad_proxy(e) {
|
||
if (ctrl(e))
|
||
return true;
|
||
|
||
ev(e);
|
||
var dst = this.getAttribute('dst'),
|
||
k = dst ? 'dst' : 'href',
|
||
v = dst ? dst : this.getAttribute('href'),
|
||
els = QSA('#treeul a');
|
||
|
||
for (var a = 0, aa = els.length; a < aa; a++)
|
||
if (els[a].getAttribute(k) === v)
|
||
return els[a].click();
|
||
}
|
||
|
||
r.treego = function (e) {
|
||
if (ctrl(e))
|
||
return true;
|
||
|
||
ev(e);
|
||
if (this.className == 'hl' &&
|
||
this.previousSibling.textContent == '-') {
|
||
treegrow.call(this.previousSibling, e);
|
||
return;
|
||
}
|
||
var href = this.getAttribute('href');
|
||
if (R && !href.startsWith(SR)) {
|
||
location = href;
|
||
return;
|
||
}
|
||
r.reqls(href, true);
|
||
r.dir_cb = tree_scrollto;
|
||
thegrid.setvis(true);
|
||
clmod(this, 'ld', 1);
|
||
}
|
||
|
||
r.reqls = function (url, hpush, back, hydrate) {
|
||
if (IE && !history.pushState)
|
||
return window.location = url;
|
||
|
||
var xhr = new XHR(),
|
||
m = /[?&](k=[^&#]+)/.exec(url),
|
||
k = m ? '&' + m[1] : dk ? '&k=' + dk : '',
|
||
uq = (r.dots ? '&dots' : '') + k;
|
||
|
||
if (rtt !== null)
|
||
uq += '&rtt=' + rtt;
|
||
|
||
xhr.top = url.split('?')[0];
|
||
xhr.back = back
|
||
xhr.hpush = hpush;
|
||
xhr.hydrate = hydrate;
|
||
xhr.ts = Date.now();
|
||
xhr.open('GET', xhr.top + '?ls' + uq, true);
|
||
xhr.onload = xhr.onerror = recvls;
|
||
xhr.send();
|
||
|
||
r.nvis = r.lim;
|
||
r.sb_msg = false;
|
||
r.nextdir = xhr.top;
|
||
clearTimeout(mpl.t_eplay);
|
||
enspin('t');
|
||
enspin('f');
|
||
window.removeEventListener('scroll', r.tscroll);
|
||
}
|
||
|
||
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 (!xhrchk(this, L.fl_xe1, L.fl_xe2))
|
||
return;
|
||
|
||
rtt = Date.now() - this.ts;
|
||
|
||
r.nextdir = null;
|
||
var cdir = get_evpath(),
|
||
lfiles = ebi('files'),
|
||
cur = lfiles.getAttribute('ts');
|
||
|
||
if (cur && parseInt(cur) > this.ts) {
|
||
console.log("reject ls");
|
||
return;
|
||
}
|
||
lfiles.setAttribute('ts', this.ts);
|
||
|
||
try {
|
||
var res = JSON.parse(this.responseText);
|
||
Object.assign(res, res.cfg);
|
||
}
|
||
catch (ex) {
|
||
if (r.ls_cb) {
|
||
r.ls_cb = null;
|
||
return toast.inf(10, L.mm_nof);
|
||
}
|
||
|
||
if (!this.hydrate) {
|
||
location = this.top;
|
||
return;
|
||
}
|
||
|
||
return toast.err(30, "bad <code>?ls</code> reply;\nexpected json, got this:\n\n" + esc(this.responseText + ''));
|
||
}
|
||
|
||
if (r.chk_index_html(this.top, res))
|
||
return;
|
||
|
||
for (var a = 0; a < res.files.length; a++)
|
||
if (res.files[a].tags === undefined)
|
||
res.files[a].tags = {};
|
||
|
||
dnsort = res.dnsort;
|
||
read_dsort(res.dsort);
|
||
dcrop = res.dcrop;
|
||
dth3x = res.dth3x;
|
||
dk = res.dk;
|
||
|
||
srvinf = res.srvinf;
|
||
if (rtt !== null)
|
||
srvinf += (srvinf ? '</span> // <span>rtt: ' : 'rtt: ') + rtt;
|
||
|
||
var o = ebi('srv_info2');
|
||
if (o)
|
||
o.innerHTML = ebi('srv_info').innerHTML = '<span>' + srvinf + '</span>';
|
||
|
||
if (this.hpush && !showfile.active())
|
||
hist_push(this.top + (dk ? '?k=' + dk : ''));
|
||
|
||
if (!this.back) {
|
||
var dirs = [];
|
||
for (var a = 0; a < res.dirs.length; a++) {
|
||
var dh = res.dirs[a].href,
|
||
dn = dh.split('/')[0].split('?')[0],
|
||
m = /[?&](k=[^&#]+)/.exec(dh);
|
||
|
||
if (m)
|
||
dn += '?' + m[1];
|
||
|
||
dirs.push(dn);
|
||
}
|
||
|
||
r.rendertree({ "a": dirs }, this.ts, ".", get_evpath() + (dk ? '?k=' + dk : ''));
|
||
if (res.unlist)
|
||
r.prunetree(res);
|
||
}
|
||
|
||
r.gentab(this.top, res);
|
||
qsr('#dlt_t');
|
||
qsr('#dlt_f');
|
||
|
||
var lg0 = res.logues ? res.logues[0] || "" : "",
|
||
lg1 = res.logues ? res.logues[1] || "" : "",
|
||
mds = res.readmes && treectl.ireadme,
|
||
md0 = mds ? res.readmes[0] || "" : "",
|
||
md1 = mds ? res.readmes[1] || "" : "",
|
||
dirchg = get_evpath() != cdir;
|
||
|
||
if (lg1 === Ls.eng.f_empty)
|
||
lg1 = L.f_empty;
|
||
|
||
sandbox(ebi('pro'), sb_lg, sba_lg,'', lg0);
|
||
if (dirchg)
|
||
sandbox(ebi('epi'), sb_lg, sba_lg, '', lg1);
|
||
|
||
clmod(ebi('pro'), 'mdo');
|
||
clmod(ebi('epi'), 'mdo');
|
||
|
||
if (md0)
|
||
show_readme(md0, 0);
|
||
|
||
if (md1)
|
||
show_readme(md1, 1);
|
||
else if (!dirchg)
|
||
sandbox(ebi('epi'), sb_lg, sba_lg, '', lg1);
|
||
|
||
if (this.hpush && !this.back) {
|
||
var ofs = ebi('wrap').offsetTop;
|
||
if (document.documentElement.scrollTop > ofs)
|
||
document.documentElement.scrollTop = ofs;
|
||
}
|
||
|
||
wintitle();
|
||
var fun = r.ls_cb;
|
||
if (fun) {
|
||
r.ls_cb = null;
|
||
fun();
|
||
}
|
||
|
||
if (window.have_shr && QS('#op_unpost.act') && (cdir.startsWith(SR + have_shr) || get_evpath().startsWith(SR + have_shr)))
|
||
goto('unpost');
|
||
}
|
||
|
||
r.chk_index_html = function (top, res) {
|
||
if (!r.idxh || !res || !res.files || noih)
|
||
return;
|
||
|
||
for (var a = 0; a < res.files.length; a++)
|
||
if (/^index.html?(\?|$)/i.exec(res.files[a].href)) {
|
||
location = vjoin(top, res.files[a].href);
|
||
return true;
|
||
}
|
||
};
|
||
|
||
r.gentab = function (top, res) {
|
||
showfile.untail();
|
||
var nodes = res.dirs.concat(res.files),
|
||
html = mk_files_header(res.taglist),
|
||
sel = msel.hist[top],
|
||
ae = document.activeElement,
|
||
cid = null,
|
||
plain = [],
|
||
seen = {};
|
||
|
||
if (ae && /^tr$/i.exec(ae.nodeName))
|
||
if (ae = ae.querySelector('a[id]'))
|
||
cid = ae.getAttribute('id');
|
||
|
||
var m = /[?&]k=([^&]+)/.exec(location.search);
|
||
if (m)
|
||
memo_dk(top, m[1]);
|
||
|
||
r.lsc = res;
|
||
if (res.unlist) {
|
||
var ptn = new RegExp(res.unlist);
|
||
for (var a = nodes.length - 1; a >= 0; a--)
|
||
if (ptn.exec(uricom_dec(nodes[a].href.split('?')[0])))
|
||
nodes.splice(a, 1);
|
||
}
|
||
nodes = sortfiles(nodes);
|
||
window.removeEventListener('scroll', r.tscroll);
|
||
r.trunc = nodes.length > r.nvis && location.hash.length < 2;
|
||
if (r.trunc) {
|
||
for (var a = r.lim; a < nodes.length; a++) {
|
||
var tn = nodes[a],
|
||
tns = Object.keys(tn.tags || {});
|
||
|
||
plain.push(uricom_dec(tn.href.split('?')[0]));
|
||
|
||
for (var b = 0; b < tns.length; b++)
|
||
if (has(res.taglist, tns[b]))
|
||
plain.push(tn.tags[tns[b]]);
|
||
}
|
||
nodes = nodes.slice(0, r.nvis);
|
||
}
|
||
|
||
showfile.files = [];
|
||
html.push('<tbody>');
|
||
for (var a = 0; a < nodes.length; a++) {
|
||
var tn = nodes[a],
|
||
bhref = tn.href.split('?')[0],
|
||
fname = uricom_dec(bhref),
|
||
hname = esc(fname),
|
||
id = 'f-' + ('00000000' + crc32(fname)).slice(-8),
|
||
lang = showfile.getlang(fname);
|
||
|
||
while (seen[id]) // ejyefs ev69gg y9j8sg .opus
|
||
id += 'a';
|
||
seen[id] = 1;
|
||
|
||
if (lang) {
|
||
showfile.files.push({ 'id': id, 'name': fname });
|
||
if (lang == 'md')
|
||
tn.href = addq(tn.href, 'v');
|
||
}
|
||
|
||
if (tn.lead == '-')
|
||
tn.lead = '<a href="?doc=' + bhref + '" id="t' + id +
|
||
'" rel="nofollow" class="doc' + (lang ? ' bri' : '') +
|
||
'" hl="' + id + '" name="' + hname + '">-txt-</a>';
|
||
|
||
var cl = /\.PARTIAL$/.exec(fname) ? ' class="fade"' : '',
|
||
ln = ['<tr' + cl + '><td>' + tn.lead + '</td><td><a href="' +
|
||
top + tn.href + '" id="' + id + '">' + hname + '</a>', tn.sz];
|
||
|
||
for (var b = 0; b < res.taglist.length; b++) {
|
||
var k = esc(res.taglist[b]),
|
||
v = (tn.tags || {})[k] || "",
|
||
sv = null;
|
||
|
||
if (k == ".dur")
|
||
sv = v ? s2ms(v) : "";
|
||
else if (k == ".up_at")
|
||
sv = v ? unix2iso(v) : "";
|
||
else {
|
||
ln.push(esc('' + v));
|
||
continue;
|
||
}
|
||
ln[ln.length - 1] += '</td><td sortv="' + v + '">' + sv;
|
||
}
|
||
ln = ln.concat([tn.ext, unix2iso(tn.ts)]).join('</td><td>');
|
||
html.push(ln + '</td></tr>');
|
||
}
|
||
html.push('</tbody>');
|
||
html = html.join('\n');
|
||
set_files_html(html);
|
||
if (r.trunc) {
|
||
r.setlazy(plain);
|
||
if (!r.ask) {
|
||
window.addEventListener('scroll', r.tscroll);
|
||
setTimeout(r.tscroll, 100);
|
||
}
|
||
}
|
||
else ebi('lazy').innerHTML = '';
|
||
|
||
function asdf() {
|
||
showfile.mktree();
|
||
mukey.render();
|
||
reload_tree();
|
||
reload_browser();
|
||
tree_scrollto();
|
||
if (res.acct) {
|
||
acct = res.acct;
|
||
have_up2k_idx = res.idx;
|
||
have_tags_idx = res.itag;
|
||
lifetime = res.lifetime;
|
||
apply_perms(res);
|
||
fileman.render();
|
||
}
|
||
msel.loadsel(top, sel);
|
||
|
||
if (cid) try {
|
||
ebi(cid).closest('tr').focus();
|
||
} catch (ex) { }
|
||
|
||
setTimeout(eval_hash, 1);
|
||
}
|
||
|
||
var m = scan_hash(hash0),
|
||
url = null;
|
||
|
||
if (m) {
|
||
url = ebi(m[1]);
|
||
if (url) {
|
||
url = url.href;
|
||
var mt = m[0] == 'a' ? 'audio' : /\.(webm|mkv)($|\?)/i.exec(url) ? 'video' : 'image'
|
||
if (mt == 'image') {
|
||
url = addq(url, 'cache');
|
||
console.log(url);
|
||
new Image().src = url;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (url) setTimeout(asdf, 1); else asdf();
|
||
}
|
||
|
||
r.hydrate = function () {
|
||
qsr('#bbsw');
|
||
if (ls0 === null) {
|
||
var xhr = new XHR();
|
||
xhr.open('GET', SR + '/?setck=js=y', true);
|
||
xhr.send();
|
||
|
||
r.ls_cb = showfile.addlinks;
|
||
return r.reqls(get_evpath(), false, undefined, true);
|
||
}
|
||
ls0.unlist = unlist0;
|
||
|
||
var top = get_evpath();
|
||
if (r.chk_index_html(top, ls0))
|
||
return;
|
||
|
||
r.gentab(top, ls0);
|
||
pbar.onresize();
|
||
vbar.onresize();
|
||
showfile.addlinks();
|
||
setTimeout(eval_hash, 1);
|
||
};
|
||
|
||
function memo_dk(vp, k) {
|
||
dks[vp] = k;
|
||
var lv = vp + "?" + k;
|
||
if (has(ldks, lv))
|
||
return;
|
||
|
||
ldks.unshift(lv);
|
||
if (ldks.length > 32) {
|
||
var keep = [], evp = get_evpath();
|
||
for (var a = 0; a < ldks.length; a++) {
|
||
var s = ldks[a];
|
||
if (evp.startsWith(s.replace(/\?[^?]+$/, '')))
|
||
keep.push(s);
|
||
}
|
||
var lim = 32 - keep.length;
|
||
for (var a = 0; a < lim; a++) {
|
||
if (!has(keep, ldks[a]))
|
||
keep.push(ldks[a])
|
||
}
|
||
ldks = keep;
|
||
}
|
||
jwrite('dks', ldks);
|
||
}
|
||
|
||
r.setlazy = function (plain) {
|
||
var html = ['<div id="plazy">', esc(plain.join(' ')), '</div>'],
|
||
all = r.lsc.files.length + r.lsc.dirs.length,
|
||
nxt = r.nvis * 4;
|
||
|
||
if (r.ask)
|
||
html.push((nxt >= all ? L.fbd_all : L.fbd_more).format(r.nvis, all, nxt));
|
||
|
||
ebi('lazy').innerHTML = html.join('\n');
|
||
|
||
try {
|
||
ebi('bd_all').onclick = function (e) {
|
||
ev(e);
|
||
r.showmore(all);
|
||
};
|
||
ebi('bd_more').onclick = function (e) {
|
||
ev(e);
|
||
r.showmore(nxt);
|
||
};
|
||
}
|
||
catch (ex) { }
|
||
};
|
||
|
||
r.showmore = function (n) {
|
||
window.removeEventListener('scroll', r.tscroll);
|
||
console.log('nvis {0} -> {1}'.format(r.nvis, n));
|
||
r.nvis = n;
|
||
ebi('lazy').innerHTML = '';
|
||
ebi('wrap').style.opacity = 0.4;
|
||
document.documentElement.scrollLeft = 0;
|
||
setTimeout(function () {
|
||
r.gentab(get_evpath(), r.lsc);
|
||
ebi('wrap').style.opacity = CLOSEST ? 'unset' : 1;
|
||
}, 1);
|
||
};
|
||
|
||
r.tscroll = function () {
|
||
var el = r.trunc ? ebi('plazy') : null;
|
||
if (!el || ebi('lazy').style.display || ebi('unsearch'))
|
||
return;
|
||
|
||
var sy = yscroll() + window.innerHeight,
|
||
ty = el.offsetTop;
|
||
|
||
if (sy <= ty)
|
||
return;
|
||
|
||
window.removeEventListener('scroll', r.tscroll);
|
||
|
||
var all = r.lsc.files.length + r.lsc.dirs.length;
|
||
if (r.nvis * 16 <= all) {
|
||
console.log("{0} ({1} * 16) <= {2}".format(r.nvis * 16, r.nvis, all));
|
||
r.showmore(r.nvis * 4);
|
||
}
|
||
else {
|
||
console.log("{0} ({1} * 16) > {2}".format(r.nvis * 16, r.nvis, all));
|
||
r.showmore(all);
|
||
}
|
||
};
|
||
|
||
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);
|
||
for (var a = 0; a < keys.length; a++)
|
||
keys[a] = [uricom_dec(keys[a]), keys[a]];
|
||
|
||
if (ENATSORT)
|
||
keys.sort(function (a, b) { return NATSORT.compare(a[0], b[0]); });
|
||
else
|
||
keys.sort(function (a, b) { return a[0].localeCompare(b[0]); });
|
||
|
||
for (var a = 0; a < keys.length; a++) {
|
||
var kk = keys[a][1],
|
||
m = /(\?k=[^\n]+)/.exec(kk),
|
||
kdk = m ? m[1] : '',
|
||
ks = kk.replace(kdk, '').slice(1),
|
||
ded = ks.endsWith('\n'),
|
||
k = uricom_sdec(ded ? ks.replace(/\n$/, '') : 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 + kdk + '">' + 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 += (ded ? '<li class="offline">' : '<li>') + link + '</li>\n';
|
||
}
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
function scaletree(e) {
|
||
ev(e);
|
||
treesz += parseInt(this.getAttribute("step"));
|
||
if (!isNum(treesz))
|
||
treesz = 16;
|
||
|
||
treesz = clamp(treesz, 2, 120);
|
||
swrite('treesz', treesz);
|
||
onresize();
|
||
}
|
||
|
||
ebi('entree').onclick = r.entree;
|
||
ebi('detree').onclick = r.detree;
|
||
ebi('visdir').onclick = tree_scrollto;
|
||
ebi('twig').onclick = scaletree;
|
||
ebi('twobytwo').onclick = scaletree;
|
||
|
||
var cs = sread('entreed'),
|
||
vw = window.innerWidth / parseFloat(getComputedStyle(document.body)['font-size']);
|
||
|
||
if (cs == 'tree' || (cs != 'na' && vw >= 60))
|
||
r.entree(null, true);
|
||
|
||
r.onpopfun = function (e) {
|
||
console.log("h-pop " + e.state);
|
||
if (!e.state)
|
||
return;
|
||
|
||
var url = new URL(e.state, "https://" + location.host),
|
||
req = url.pathname,
|
||
hbase = req,
|
||
cbase = location.pathname,
|
||
mdoc = /[?&]doc=/.exec('' + url),
|
||
mdk = /[?&](k=[^&#]+)/.exec('' + url);
|
||
|
||
if (mdoc && hbase == cbase)
|
||
return showfile.show(hbase + showfile.sname(url.search), true);
|
||
|
||
if (mdk)
|
||
req += '?' + mdk[1];
|
||
|
||
r.goto(req, false, true);
|
||
};
|
||
|
||
var evp = get_evpath() + (dk ? '?k=' + dk : '');
|
||
hist_replace(evp + location.hash);
|
||
r.onscroll = onscroll;
|
||
return r;
|
||
})();
|
||
|
||
|
||
function enspin(i) {
|
||
i = 'dlt_' + i;
|
||
if (ebi(i))
|
||
return;
|
||
var d = mknod('div', i, SPINNER);
|
||
d.className = 'dumb_loader_thing';
|
||
if (SPINNER_CSS)
|
||
d.style.cssText = SPINNER_CSS;
|
||
document.body.appendChild(d);
|
||
}
|
||
|
||
|
||
var wfp_debounce = (function () {
|
||
var r = { 'n': 0, 't': 0 };
|
||
|
||
r.hide = function () {
|
||
if (!sb_lg && !sb_md)
|
||
return;
|
||
|
||
if (++r.n <= 1) {
|
||
r.n = 1;
|
||
clearTimeout(r.t);
|
||
r.t = setTimeout(r.reset, 300);
|
||
ebi('wfp').style.opacity = 0.1;
|
||
}
|
||
};
|
||
r.show = function () {
|
||
if (!sb_lg && !sb_md)
|
||
return;
|
||
|
||
if (--r.n <= 0) {
|
||
r.n = 0;
|
||
clearTimeout(r.t);
|
||
ebi('wfp').style.opacity = CLOSEST ? 'unset' : 1;
|
||
}
|
||
};
|
||
r.reset = function () {
|
||
r.n = 0;
|
||
r.show();
|
||
};
|
||
return r;
|
||
})();
|
||
|
||
|
||
function apply_perms(res) {
|
||
perms = res.perms || [];
|
||
|
||
var a = QS('#ops a[data-dest="up2k"]');
|
||
if (have_up2k_idx) {
|
||
a.removeAttribute('data-perm');
|
||
a.setAttribute('tt', L.ot_u2i);
|
||
}
|
||
else {
|
||
a.setAttribute('data-perm', 'write');
|
||
a.setAttribute('tt', L.ot_u2w);
|
||
}
|
||
clmod(ebi('srch_form'), 'tags', have_tags_idx);
|
||
|
||
a.style.display = '';
|
||
tt.att(QS('#ops'));
|
||
|
||
var axs = [],
|
||
aclass = '>',
|
||
chk = ['read', 'write', 'move', 'delete', 'get', 'admin'];
|
||
|
||
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 id="srv_info2"><span>' + srvinf +
|
||
'</span></span><span' + aclass + axs + L.access + '</span>' + (acct != '*' ?
|
||
'<form id="flogout" method="post" enctype="multipart/form-data"><input type="hidden" name="act" value="logout" /><input id="blogout" type="submit" value="' + (window.is_idp ? '' : L.logout) + acct + '"></form>' :
|
||
'<a href="?h">Login</a>');
|
||
|
||
var o = QSA('#ops>a[data-perm]');
|
||
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 o = QSA('#ops>a[data-dep], #u2conf td[data-dep]');
|
||
for (var a = 0; a < o.length; a++)
|
||
o[a].style.display = (
|
||
o[a].getAttribute('data-dep') != 'idx' || have_up2k_idx
|
||
) ? '' : 'none';
|
||
|
||
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 (res.frand)
|
||
ebi('u2rand').parentNode.style.display = 'none';
|
||
|
||
u2ts = res.u2ts;
|
||
if (up2k)
|
||
up2k.set_fsearch();
|
||
|
||
widget.setvis();
|
||
thegrid.setvis();
|
||
if (!have_read && have_write)
|
||
goto('up2k');
|
||
}
|
||
|
||
|
||
function tr2id(tr) {
|
||
try {
|
||
return tr.cells[1].querySelector('a[id]').getAttribute('id');
|
||
}
|
||
catch (ex) {
|
||
return null;
|
||
}
|
||
}
|
||
|
||
|
||
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].className || '').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 = esc(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 r = { 'picking': false };
|
||
var hidden = jread('filecols', []);
|
||
|
||
r.add_btns = function () {
|
||
var ths = QSA('#files>thead th>span');
|
||
for (var a = 0, aa = ths.length; a < aa; a++) {
|
||
var th = ths[a].parentElement,
|
||
toh = ths[a].outerHTML, // !ff10
|
||
ttv = L.cols[ths[a].textContent];
|
||
|
||
ttv = (ttv ? ttv + '; ' : '') + 'id=<code>' + th.getAttribute('name') + '</code>';
|
||
if (!MOBILE && toh) {
|
||
th.innerHTML = '<div class="cfg"><a href="#">-</a></div>' + toh;
|
||
th.getElementsByTagName('a')[0].onclick = ev_row_tgl;
|
||
}
|
||
if (ttv) {
|
||
th.setAttribute("tt", ttv);
|
||
th.setAttribute("ttd", "u");
|
||
th.setAttribute("ttm", "12");
|
||
}
|
||
}
|
||
};
|
||
|
||
function hcols_click(e) {
|
||
ev(e);
|
||
var t = e.target;
|
||
if (t.tagName != 'A')
|
||
return;
|
||
|
||
r.toggle(t.textContent);
|
||
}
|
||
|
||
r.uivis = function () {
|
||
var hcols = ebi('hcols');
|
||
hcols.previousSibling.style.display = hcols.style.display = ((!thegrid || !thegrid.en) && (hidden.length || MOBILE)) ? 'block' : 'none';
|
||
};
|
||
|
||
r.set_style = function (unhide) {
|
||
hidden.sort();
|
||
|
||
if (!unhide)
|
||
unhide = [];
|
||
|
||
var html = [],
|
||
hcols = ebi('hcols');
|
||
|
||
for (var a = 0; a < hidden.length; a++) {
|
||
var ttv = L.cols[hidden[a]],
|
||
tta = ttv ? ' tt="' + ttv + '">' : '>';
|
||
|
||
html.push('<a href="#" class="btn"' + tta + esc(hidden[a]) + '</a>');
|
||
}
|
||
hcols.innerHTML = html.join('\n');
|
||
hcols.onclick = hcols_click;
|
||
r.uivis();
|
||
r.add_btns();
|
||
|
||
var ohidden = [],
|
||
ths = QSA('#files>thead 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].className = cls;
|
||
}
|
||
if (tt) {
|
||
tt.att(ebi('hcols'));
|
||
tt.att(QS('#files>thead'));
|
||
}
|
||
};
|
||
|
||
r.setvis = function (name, vis) {
|
||
var ofs = hidden.indexOf(name);
|
||
if (ofs !== -1 && vis != 0)
|
||
hidden.splice(ofs, 1);
|
||
else if (vis != 1) {
|
||
if (!sread("chide_ok")) {
|
||
return modal.confirm(L.f_chide.format(name), function () {
|
||
swrite("chide_ok", 1);
|
||
r.toggle(name);
|
||
}, null);
|
||
}
|
||
hidden.push(name);
|
||
}
|
||
jwrite("filecols", hidden);
|
||
r.set_style();
|
||
};
|
||
r.show = function (name) { r.setvis(name, 1); };
|
||
r.hide = function (name) { r.setvis(name, 0); };
|
||
r.toggle = function (name) { r.setvis(name, -1); };
|
||
|
||
ebi('hcolsr').onclick = function (e) {
|
||
ev(e);
|
||
r.reset(true);
|
||
};
|
||
|
||
if (MOBILE)
|
||
ebi('hcolsh').onclick = function (e) {
|
||
ev(e);
|
||
if (r.picking)
|
||
return r.unpick();
|
||
|
||
var lbs = QSA('#files>thead th');
|
||
for (var a = 0; a < lbs.length; a++) {
|
||
lbs[a].onclick = function (e) {
|
||
ev(e);
|
||
if (toast.tag == 'pickhide')
|
||
toast.hide();
|
||
|
||
r.hide(e.target.textContent);
|
||
};
|
||
};
|
||
r.picking = true;
|
||
clmod(ebi('files'), 'hhpick', 1);
|
||
toast.inf(0, L.cl_hpick, 'pickhide');
|
||
};
|
||
|
||
r.unpick = function () {
|
||
r.picking = false;
|
||
toast.inf(5, L.cl_hcancel);
|
||
|
||
clmod(ebi('files'), 'hhpick');
|
||
|
||
var lbs = QSA('#files>thead th');
|
||
for (var a = 0; a < lbs.length; a++)
|
||
lbs[a].onclick = null;
|
||
};
|
||
|
||
r.reset = function (force) {
|
||
if (force || JSON.stringify(def_hcols) != sread('hfilecols')) {
|
||
console.log("applying default hidden-cols");
|
||
hidden = [];
|
||
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);
|
||
}
|
||
r.set_style();
|
||
}
|
||
r.reset();
|
||
|
||
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 r;
|
||
})();
|
||
|
||
|
||
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 "
|
||
]
|
||
},
|
||
defnot = 'rekobo_alnum';
|
||
|
||
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("cpp_keynot", 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("cpp_keynot") || defnot;
|
||
if (!maps[notation])
|
||
notation = defnot;
|
||
|
||
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
|
||
};
|
||
})();
|
||
|
||
|
||
var light, theme, themen;
|
||
var settheme = (function () {
|
||
var r = {},
|
||
ax = 'abcdefghijklmnopqrstuvwx',
|
||
tre = '🌲',
|
||
chldr = !SPINNER_CSS && SPINNER == tre;
|
||
|
||
r.ldr = {
|
||
'4':['🌴'],
|
||
'5':['🌭', 'padding:0 0 .7em .7em;filter:saturate(3)'],
|
||
'6':['📞', 'padding:0;filter:brightness(2) sepia(1) saturate(3) hue-rotate(60deg)'],
|
||
'7':['▲', 'font-size:3em'], //cp437
|
||
};
|
||
|
||
theme = sread('cpp_thm') || 'a';
|
||
if (!/^[a-x][yz]/.exec(theme))
|
||
theme = dtheme;
|
||
|
||
themen = theme.split(/ /)[0];
|
||
light = !!(theme.indexOf('y') + 1);
|
||
|
||
function freshen() {
|
||
var cl = document.documentElement.className;
|
||
cl = cl.replace(/\b(light|dark|[a-z]{1,2})\b/g, '').replace(/ +/g, ' ');
|
||
document.documentElement.className = cl + ' ' + theme + ' ';
|
||
|
||
pbar.drawbuf();
|
||
pbar.drawpos();
|
||
vbar.draw();
|
||
showfile.setstyle();
|
||
bchrome();
|
||
|
||
var html = [], itheme = ax.indexOf(theme[0]) * 2 + (light ? 1 : 0),
|
||
names = ['classic dark', 'classic light', 'pm-monokai', 'flat light', 'vice', 'hotdog stand', 'hacker', 'hi-con'];
|
||
|
||
for (var a = 0; a < themes; a++)
|
||
html.push('<a href="#" class="btn tgl' + (a == itheme ? ' on' : '') +
|
||
'" tt="' + (names[a] || 'custom') + '">' + a + '</a>');
|
||
|
||
ebi('themes').innerHTML = html.join('');
|
||
var btns = QSA('#themes a');
|
||
for (var a = 0; a < themes; a++)
|
||
btns[a].onclick = r.go;
|
||
|
||
if (chldr) {
|
||
var x = r.ldr[itheme] || [tre];
|
||
SPINNER = x[0];
|
||
SPINNER_CSS = x[1];
|
||
}
|
||
|
||
bcfg_set('light', light);
|
||
tt.att(ebi('themes'));
|
||
}
|
||
|
||
r.go = function (e) {
|
||
var i = e;
|
||
try { ev(e); i = e.target.textContent; } catch (ex) { }
|
||
light = i % 2 == 1;
|
||
var c = ax[Math.floor(i / 2)],
|
||
l = light ? 'y' : 'z';
|
||
theme = c + l + ' ' + c + ' ' + l;
|
||
themen = c + l;
|
||
swrite('cpp_thm', theme);
|
||
freshen();
|
||
}
|
||
|
||
freshen();
|
||
return r;
|
||
})();
|
||
|
||
|
||
(function () {
|
||
function freshen() {
|
||
lang = sread("cpp_lang", LANGS) || lang;
|
||
var k, html = [];
|
||
for (var a = 0; a < LANGS.length; a++) {
|
||
k = LANGS[a];
|
||
html.push('<a href="#" class="btn tgl' + (k == lang ? ' on' : '') +
|
||
'" tt="' + Ls[k].tt + '">' + k + '</a>');
|
||
}
|
||
ebi('langs').innerHTML = html.join('');
|
||
var btns = QSA('#langs a');
|
||
for (var a = 0, aa = btns.length; a < aa; a++)
|
||
btns[a].onclick = setlang;
|
||
}
|
||
|
||
function setlang(e) {
|
||
ev(e);
|
||
var t = L.lang_set;
|
||
L = Ls[this.textContent];
|
||
swrite("cpp_lang", this.textContent);
|
||
freshen();
|
||
modal.confirm(L.lang_set + "\n\n" + t, location.reload.bind(location), null);
|
||
};
|
||
|
||
freshen();
|
||
})();
|
||
|
||
|
||
var arcfmt = (function () {
|
||
if (!ebi('arc_fmt'))
|
||
return { "render": function () { } };
|
||
|
||
var html = [],
|
||
fmts = [
|
||
["tar", "tar", L.fz_tar],
|
||
["pax", "tar=pax", L.fz_pax],
|
||
["tgz", "tar=gz", L.fz_targz],
|
||
["txz", "tar=xz", L.fz_tarxz],
|
||
["zip", "zip", L.fz_zip8],
|
||
["zip_dos", "zip=dos", L.fz_zipd],
|
||
["zip_crc", "zip=crc", L.fz_zipc]
|
||
];
|
||
|
||
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 (!/^(zip|tar|pax|tgz|txz)$/.exec(txt))
|
||
continue;
|
||
|
||
var m = /(.*[?&])(tar|zip)([^&#]*)(.*)$/.exec(href);
|
||
if (!m)
|
||
throw new Error('missing arg in url');
|
||
|
||
o.setAttribute("href", m[1] + arg + m[4]);
|
||
o.textContent = fmt.split('_')[0];
|
||
}
|
||
ebi('selzip').textContent = fmt.split('_')[0];
|
||
ebi('selzip').setAttribute('fmt', arg);
|
||
|
||
QS('#zip1 span').textContent = fmt.split('_')[0];
|
||
ebi('zip1').setAttribute("href",
|
||
get_evpath() + (dk ? '?k=' + dk + '&': '?') + arg);
|
||
|
||
if (!have_zip) {
|
||
ebi('zip1').style.display = 'none';
|
||
ebi('selzip').style.display = 'none';
|
||
}
|
||
}
|
||
|
||
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.hist = {};
|
||
r.so = null; // selection origin
|
||
r.pr = null; // previous range
|
||
|
||
r.load = function (reset) {
|
||
if (r.sel && !reset)
|
||
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 = clgot(ebi(ao.id).closest('tr'), 'sel');
|
||
if (ao.sel)
|
||
r.sel.push(ao);
|
||
}
|
||
if (!reset)
|
||
return;
|
||
}
|
||
|
||
r.all = [];
|
||
var links = QSA('#files tbody td:nth-child(2) a:last-child'),
|
||
is_srch = !!ebi('unsearch'),
|
||
vbase = get_evpath();
|
||
|
||
for (var a = 0, aa = links.length; a < aa; a++) {
|
||
var qhref = links[a].getAttribute('href'),
|
||
href = qhref.split('?')[0].replace(/\/$/, ""),
|
||
item = {};
|
||
|
||
item.id = links[a].getAttribute('id');
|
||
item.sel = clgot(links[a].closest('tr'), 'sel');
|
||
item.vp = href.indexOf('/') !== -1 ? href : vbase + href;
|
||
|
||
if (dk) {
|
||
var m = /[?&](k=[^&#]+)/.exec(qhref);
|
||
item.q = m ? '?' + m[1] : '';
|
||
}
|
||
else item.q = '';
|
||
|
||
r.all.push(item);
|
||
if (item.sel)
|
||
r.sel.push(item);
|
||
|
||
if (!is_srch)
|
||
links[a].closest('tr').setAttribute('tabindex', '0');
|
||
}
|
||
};
|
||
|
||
r.loadsel = function (vp, sel) {
|
||
if (!sel || !r.so || !ebi(r.so))
|
||
r.so = r.pr = null;
|
||
|
||
if (!sel)
|
||
return r.origin_id(null);
|
||
|
||
r.hist[vp] = sel;
|
||
r.sel = [];
|
||
r.load();
|
||
|
||
var vsel = new Set();
|
||
for (var a = 0; a < sel.length; a++)
|
||
vsel.add(sel[a].vp);
|
||
|
||
for (var a = 0; a < r.all.length; a++)
|
||
if (vsel.has(r.all[a].vp))
|
||
clmod(ebi(r.all[a].id).closest('tr'), 'sel', 1);
|
||
|
||
r.selui();
|
||
};
|
||
|
||
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();
|
||
showfile.updtree();
|
||
|
||
if (r.sel.length)
|
||
r.hist[get_evpath()] = r.sel;
|
||
else
|
||
delete r.hist[get_evpath()];
|
||
};
|
||
r.seltgl = function (e) {
|
||
ev(e);
|
||
var tr = this.parentNode,
|
||
id = tr2id(tr);
|
||
|
||
if ((treectl.csel || !thegrid.en || thegrid.sel) && e.shiftKey && r.so && id && r.so != id) {
|
||
var o1 = -1, o2 = -1;
|
||
for (a = 0; a < r.all.length; a++) {
|
||
var ai = r.all[a].id;
|
||
if (ai == r.so)
|
||
o1 = a;
|
||
if (ai == id)
|
||
o2 = a;
|
||
}
|
||
var st = r.all[o1].sel;
|
||
if (o1 > o2)
|
||
o2 = [o1, o1 = o2][0];
|
||
|
||
if (r.pr) {
|
||
// invert previous range, in case it was narrowed
|
||
for (var a = r.pr[0]; a <= r.pr[1]; a++)
|
||
clmod(ebi(r.all[a].id).closest('tr'), 'sel', !st);
|
||
|
||
// and invert current selection if repeated
|
||
if (r.pr[0] === o1 && r.pr[1] === o2)
|
||
st = !st;
|
||
}
|
||
|
||
for (var a = o1; a <= o2; a++)
|
||
clmod(ebi(r.all[a].id).closest('tr'), 'sel', st);
|
||
|
||
r.pr = [o1, o2];
|
||
|
||
if (window.getSelection)
|
||
window.getSelection().removeAllRanges();
|
||
}
|
||
else {
|
||
clmod(tr, 'sel', 't');
|
||
r.origin_tr(tr);
|
||
}
|
||
r.selui();
|
||
};
|
||
r.origin_tr = function (tr) {
|
||
r.so = tr2id(tr);
|
||
r.pr = null;
|
||
};
|
||
r.origin_id = function (id) {
|
||
r.so = id;
|
||
r.pr = null;
|
||
};
|
||
r.evsel = function (e, fun) {
|
||
ev(e);
|
||
r.so = r.pr = null;
|
||
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 = [];
|
||
|
||
if (dk)
|
||
arg += '&k=' + dk;
|
||
|
||
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 = '<input name="act" value="zip" />' +
|
||
'<textarea name="files" id="ziptxt"></textarea>';
|
||
frm.style.display = 'none';
|
||
|
||
qsr('#widgeti>form');
|
||
ebi('widgeti').appendChild(frm);
|
||
var obj = ebi('ziptxt');
|
||
obj.value = txt;
|
||
console.log(txt);
|
||
frm.submit();
|
||
};
|
||
ebi('seldl').onclick = function (e) {
|
||
ev(e);
|
||
var sel = r.getsel();
|
||
for (var a = 0; a < sel.length; a++)
|
||
dl_file(sel[a].vp + sel[a].q);
|
||
};
|
||
r.render = function () {
|
||
var tds = QSA('#files tbody td+td+td'),
|
||
is_srch = !!ebi('unsearch');
|
||
|
||
if (!is_srch)
|
||
for (var a = 0, aa = tds.length; a < aa; a++)
|
||
tds[a].onclick = r.seltgl;
|
||
|
||
r.selui(true);
|
||
arcfmt.render();
|
||
fileman.render();
|
||
|
||
var zipvis = (is_srch || !have_zip) ? 'none' : '';
|
||
ebi('selzip').style.display = zipvis;
|
||
ebi('zip1').style.display = zipvis;
|
||
}
|
||
return r;
|
||
})();
|
||
|
||
|
||
(function () {
|
||
if (!FormData)
|
||
return;
|
||
|
||
var form = QS('#op_new_md>form'),
|
||
tb = QS('#op_new_md input[name="name"]');
|
||
|
||
form.onsubmit = function (e) {
|
||
if (tb.value) {
|
||
if (toast.tag == L.mk_noname)
|
||
toast.hide();
|
||
|
||
return true;
|
||
}
|
||
|
||
ev(e);
|
||
toast.err(10, L.mk_noname, L.mk_noname);
|
||
return false;
|
||
};
|
||
})();
|
||
|
||
|
||
(function () {
|
||
if (!FormData)
|
||
return;
|
||
|
||
var form = QS('#op_mkdir>form'),
|
||
tb = QS('#op_mkdir input[name="name"]'),
|
||
sf = mknod('div');
|
||
|
||
clmod(sf, 'msg', 1);
|
||
form.parentNode.appendChild(sf);
|
||
|
||
form.onsubmit = function (e) {
|
||
ev(e);
|
||
var dn = tb.value;
|
||
if (!dn) {
|
||
toast.err(10, L.mk_noname, L.mk_noname);
|
||
return false;
|
||
}
|
||
|
||
if (toast.tag == L.mk_noname || toast.tag == L.fd_xe1)
|
||
toast.hide();
|
||
|
||
clmod(sf, 'vis', 1);
|
||
sf.textContent = 'creating "' + dn + '"...';
|
||
|
||
var fd = new FormData();
|
||
fd.append("act", "mkdir");
|
||
fd.append("name", dn);
|
||
|
||
var xhr = new XHR();
|
||
xhr.vp = get_evpath();
|
||
xhr.dn = dn;
|
||
xhr.open('POST', dn.startsWith('/') ? (SR || '/') : xhr.vp, true);
|
||
xhr.onload = xhr.onerror = cb;
|
||
xhr.responseType = 'text';
|
||
xhr.send(fd);
|
||
|
||
return false;
|
||
};
|
||
|
||
function cb() {
|
||
if (this.vp !== get_evpath()) {
|
||
sf.textContent = 'aborted due to location change';
|
||
return;
|
||
}
|
||
|
||
xhrchk(this, L.fd_xe1, L.fd_xe2);
|
||
|
||
if (this.status !== 201) {
|
||
sf.textContent = 'error: ' + hunpre(this.responseText);
|
||
return;
|
||
}
|
||
|
||
tb.value = '';
|
||
clmod(sf, 'vis');
|
||
sf.textContent = '';
|
||
|
||
var dn = this.getResponseHeader('X-New-Dir');
|
||
dn = dn ? '/' + dn + '/' : uricom_enc(this.dn);
|
||
treectl.goto(dn, true);
|
||
tree_scrollto();
|
||
}
|
||
})();
|
||
|
||
|
||
(function () {
|
||
var form = QS('#op_msg>form'),
|
||
tb = QS('#op_msg input[name="msg"]'),
|
||
sf = mknod('div');
|
||
|
||
clmod(sf, 'msg', 1);
|
||
form.parentNode.appendChild(sf);
|
||
|
||
form.onsubmit = function (e) {
|
||
ev(e);
|
||
clmod(sf, 'vis', 1);
|
||
sf.textContent = 'sending...';
|
||
|
||
var xhr = new XHR(),
|
||
ct = 'application/x-www-form-urlencoded;charset=UTF-8';
|
||
|
||
xhr.msg = tb.value;
|
||
xhr.open('POST', get_evpath(), true);
|
||
xhr.responseType = 'text';
|
||
xhr.onload = xhr.onerror = cb;
|
||
xhr.setRequestHeader('Content-Type', ct);
|
||
if (xhr.overrideMimeType)
|
||
xhr.overrideMimeType('Content-Type', ct);
|
||
|
||
xhr.send('msg=' + uricom_enc(xhr.msg));
|
||
return false;
|
||
};
|
||
|
||
function cb() {
|
||
xhrchk(this, L.fsm_xe1, L.fsm_xe2);
|
||
|
||
if (this.status < 200 || this.status > 202) {
|
||
sf.textContent = 'error: ' + hunpre(this.responseText);
|
||
return;
|
||
}
|
||
|
||
tb.value = '';
|
||
clmod(sf, 'vis');
|
||
var txt = 'sent: <code>' + esc(this.msg) + '</code>';
|
||
if (this.status == 202)
|
||
txt += '<br /> got: <code>' + esc(this.responseText) + '</code>';
|
||
|
||
sf.innerHTML = txt;
|
||
setTimeout(function () {
|
||
treectl.goto();
|
||
}, 100);
|
||
}
|
||
})();
|
||
|
||
|
||
var globalcss = (function () {
|
||
var ret = '';
|
||
return function () {
|
||
if (ret)
|
||
return ret;
|
||
|
||
var dcs = document.styleSheets;
|
||
for (var a = 0; a < dcs.length; a++) {
|
||
var ds, base = '';
|
||
try {
|
||
base = dcs[a].href;
|
||
if (!base)
|
||
continue;
|
||
|
||
ds = dcs[a].cssRules;
|
||
base = base.replace(/[^/]+$/, '');
|
||
for (var b = 0; b < ds.length; b++) {
|
||
var css = ds[b].cssText.split(/\burl\(/g);
|
||
ret += css[0];
|
||
for (var c = 1; c < css.length; c++) {
|
||
var m = /(^ *["']?)(.*)/.exec(css[c]),
|
||
delim = m[1],
|
||
ctxt = m[2],
|
||
is_abs = /^\/|[^)/:]+:\/\//.exec(ctxt);
|
||
|
||
ret += 'url(' + delim + (is_abs ? '' : base) + ctxt;
|
||
}
|
||
ret += '\n';
|
||
}
|
||
if (ret.indexOf('\n@import') + 1) {
|
||
var c0 = ret.split('\n'),
|
||
c1 = [],
|
||
c2 = [];
|
||
|
||
for (var a = 0; a < c0.length; a++)
|
||
(c0[a].startsWith('@import') ? c1 : c2).push(c0[a]);
|
||
|
||
ret = c1.concat(c2).join('\n');
|
||
}
|
||
}
|
||
catch (ex) {
|
||
console.log('could not read css', a, base);
|
||
}
|
||
}
|
||
return ret;
|
||
};
|
||
})();
|
||
|
||
var sandboxjs = (function () {
|
||
var ret = '',
|
||
busy = false,
|
||
url = SR + '/.cpr/util.js?_=' + TS,
|
||
tag = '<script src="' + url + '"></script>';
|
||
|
||
return function () {
|
||
if (ret || busy)
|
||
return ret || tag;
|
||
|
||
var xhr = new XHR();
|
||
xhr.open('GET', url, true);
|
||
xhr.onload = function () {
|
||
if (this.status == 200)
|
||
ret = '<script>' + this.responseText + '</script>';
|
||
};
|
||
xhr.send();
|
||
busy = true;
|
||
return tag;
|
||
};
|
||
})();
|
||
|
||
|
||
function show_md(md, name, div, url, depth) {
|
||
var errmsg = L.md_eshow + name + ':\n\n',
|
||
now = get_evpath();
|
||
|
||
url = url || now;
|
||
if (url != now)
|
||
return;
|
||
|
||
wfp_debounce.hide();
|
||
if (!marked) {
|
||
if (depth) {
|
||
clmod(div, 'raw', 1);
|
||
div.textContent = "--[ " + name + " ]---------\r\n" + md;
|
||
return toast.warn(10, errmsg + (WebAssembly ? 'failed to load marked.js' : 'your browser is too old'));
|
||
}
|
||
|
||
wfp_debounce.n--;
|
||
return import_js(SR + '/.cpr/deps/marked.js', function () {
|
||
show_md(md, name, div, url, 1);
|
||
});
|
||
}
|
||
|
||
md_plug = {}
|
||
md = load_md_plug(md, 'pre');
|
||
md = load_md_plug(md, 'post', sb_md);
|
||
|
||
var marked_opts = {
|
||
headerPrefix: 'md-',
|
||
breaks: true,
|
||
gfm: true
|
||
};
|
||
var ext = md_plug.pre;
|
||
if (ext)
|
||
Object.assign(marked_opts, ext[0]);
|
||
|
||
try {
|
||
clmod(div, 'mdo', 1);
|
||
|
||
var md_html = marked.parse(md, marked_opts);
|
||
if (!have_emp)
|
||
md_html = DOMPurify.sanitize(md_html);
|
||
|
||
if (sandbox(div, sb_md, sba_md, 'mdo', md_html))
|
||
return;
|
||
|
||
ext = md_plug.post;
|
||
ext = ext ? [ext[0].render, ext[0].render2] : [];
|
||
for (var a = 0; a < ext.length; a++)
|
||
if (ext[a])
|
||
try {
|
||
ext[a](div);
|
||
}
|
||
catch (ex) {
|
||
console.log(ex);
|
||
}
|
||
|
||
var els = QSA('#epi a');
|
||
for (var a = 0, aa = els.length; a < aa; a++) {
|
||
var href = els[a].getAttribute('href');
|
||
if (!href.startsWith('#') || href.startsWith('#md-'))
|
||
continue;
|
||
|
||
els[a].setAttribute('href', '#md-' + href.slice(1));
|
||
}
|
||
md_th_set();
|
||
set_tabindex();
|
||
var hash = location.hash;
|
||
if (hash.startsWith('#md-'))
|
||
setTimeout(function () {
|
||
try {
|
||
QS(hash).scrollIntoView();
|
||
}
|
||
catch (ex) { }
|
||
}, 1);
|
||
}
|
||
catch (ex) {
|
||
toast.warn(10, errmsg + ex);
|
||
}
|
||
wfp_debounce.show();
|
||
}
|
||
|
||
|
||
function set_tabindex() {
|
||
var els = QSA('pre');
|
||
for (var a = 0, aa = els.length; a < aa; a++)
|
||
els[a].setAttribute('tabindex', '0');
|
||
}
|
||
|
||
|
||
function show_readme(md, n) {
|
||
var tgt = ebi(n ? 'epi' : 'pro');
|
||
|
||
if (!treectl.ireadme)
|
||
return sandbox(tgt, '', '', '', 'a');
|
||
|
||
show_md(md, n ? 'README.md' : 'PREADME.md', tgt);
|
||
}
|
||
for (var a = 0; a < readmes.length; a++)
|
||
if (readmes[a])
|
||
show_readme(readmes[a], a);
|
||
|
||
|
||
function sandbox(tgt, rules, allow, cls, html) {
|
||
if (!treectl.ireadme) {
|
||
tgt.innerHTML = html ? L.md_off : '';
|
||
return;
|
||
}
|
||
if (!rules || (html || '').indexOf('<') == -1) {
|
||
tgt.innerHTML = html;
|
||
clmod(tgt, 'sb');
|
||
return false;
|
||
}
|
||
if (!CLOSEST) {
|
||
tgt.textContent = html;
|
||
clmod(tgt, 'sb');
|
||
return false;
|
||
}
|
||
clmod(tgt, 'sb', 1);
|
||
|
||
var tid = tgt.getAttribute('id'),
|
||
hash = location.hash,
|
||
want = '';
|
||
|
||
if (!cls)
|
||
wfp_debounce.hide();
|
||
|
||
if (hash.startsWith('#md-'))
|
||
want = hash.slice(1);
|
||
|
||
var env = '', tags = QSA('script');
|
||
for (var a = 0; a < tags.length; a++) {
|
||
var js = tags[a].innerHTML;
|
||
if (js && js.indexOf('have_up2k_idx') + 1)
|
||
env = js.split(/\blogues *=/)[0] + 'a;';
|
||
}
|
||
|
||
html = '<html class="iframe ' + document.documentElement.className +
|
||
'"><head><style>html{background:#eee;color:#000}</style><style>' + globalcss() +
|
||
'</style><base target="_parent"></head><body id="b" class="logue ' + cls + '">' + html +
|
||
'<script>' + env + '</script>' + sandboxjs() +
|
||
'<script>var d=document.documentElement,TS="' + TS + '",' +
|
||
'loc=new URL("' + location.href.split('?')[0] + '");' +
|
||
'function say(m){window.parent.postMessage(m,"*")};' +
|
||
'setTimeout(function(){var its=0,pih=-1,f=function(){' +
|
||
'var ih=2+Math.min(parseInt(getComputedStyle(d).height),d.scrollHeight);' +
|
||
'if(ih!=pih&&!isNaN(ih)){pih=ih;say("iheight #' + tid + ' "+ih,"*")}' +
|
||
'if(++its<20)return setTimeout(f,20);if(its==20)setInterval(f,200)' +
|
||
'};f();' +
|
||
'window.onfocus=function(){say("igot #' + tid + '")};' +
|
||
'window.onblur=function(){say("ilost #' + tid + '")};' +
|
||
'window.treectl={"goto":function(a){say("goto #' + tid + ' "+(a||""))}};' +
|
||
'var el="' + want + '"&&ebi("' + want + '");' +
|
||
'if(el)say("iscroll #' + tid + ' "+el.offsetTop);' +
|
||
'md_th_set();' +
|
||
(cls == 'mdo' && md_plug.post ?
|
||
'const x={' + md_plug.post + '};' +
|
||
'if(x.render)x.render(ebi("b"));' +
|
||
'if(x.render2)x.render2(ebi("b"));' : '') +
|
||
'},1)</script></body></html>';
|
||
|
||
var fr = mknod('iframe');
|
||
fr.setAttribute('title', 'folder ' + tid + 'logue');
|
||
fr.setAttribute('sandbox', rules ? 'allow-' + rules.replace(/ /g, ' allow-') : '');
|
||
fr.setAttribute('allow', allow);
|
||
fr.setAttribute('srcdoc', html);
|
||
tgt.appendChild(fr);
|
||
treectl.sb_msg = true;
|
||
return true;
|
||
}
|
||
window.addEventListener("message", function (e) {
|
||
if (!treectl.sb_msg)
|
||
return;
|
||
|
||
try {
|
||
console.log('msg:' + e.data);
|
||
var t = e.data.split(/ /g);
|
||
if (t[0] == 'iheight') {
|
||
var el = QSA(t[1] + '>iframe');
|
||
el = el[el.length - 1];
|
||
if (wfp_debounce.n)
|
||
while (el.previousSibling)
|
||
el.parentNode.removeChild(el.previousSibling);
|
||
|
||
el.style.height = (parseInt(t[2]) + SBH) + 'px';
|
||
el.style.visibility = CLOSEST ? 'unset' : 'block';
|
||
wfp_debounce.show();
|
||
}
|
||
else if (t[0] == 'iscroll') {
|
||
var y1 = QS(t[1]).offsetTop,
|
||
y2 = parseInt(t[2]);
|
||
console.log(y1, y2);
|
||
document.documentElement.scrollTop = y1 + y2;
|
||
}
|
||
else if (t[0] == 'igot' || t[0] == 'ilost') {
|
||
clmod(QS(t[1] + '>iframe'), 'focus', t[0] == 'igot');
|
||
}
|
||
else if (t[0] == 'imshow') {
|
||
thegrid.imshow(e.data.slice(7));
|
||
}
|
||
else if (t[0] == 'goto') {
|
||
var t = e.data.replace(/^[^ ]+ [^ ]+ /, '').split(/[?&]/)[0];
|
||
treectl.goto(t, !!t);
|
||
}
|
||
} catch (ex) {
|
||
console.log('msg-err: ' + ex);
|
||
}
|
||
}, false);
|
||
|
||
|
||
if (sb_lg && logues.length) {
|
||
if (logues[1] === Ls.eng.f_empty)
|
||
logues[1] = L.f_empty;
|
||
|
||
sandbox(ebi('pro'), sb_lg, sba_lg, '', logues[0]);
|
||
sandbox(ebi('epi'), sb_lg, sba_lg, '', logues[1]);
|
||
}
|
||
|
||
|
||
(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 = noq_href(ao),
|
||
isdir = href.endsWith('/'),
|
||
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 = (
|
||
L.un_m1 + ' – <a id="unpost_refresh" href="#">' + L.un_upd + '</a>' +
|
||
'<p>' + L.un_m4 + ' <a id="unpost_ulist" href="#">' + L.un_ulist + '</a> / <a id="unpost_ucopy" href="#">' + L.un_ucopy + '</a>' +
|
||
'<p>' + L.un_flt + ' <input type="text" id="unpost_filt" size="20" placeholder="documents/passwords" /><a id="unpost_nofilt" href="#">' + L.un_fclr + '</a></p>' +
|
||
'<div id="unpost"></div>'
|
||
);
|
||
|
||
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 (!xhrchk(this, L.fu_xe1, L.fu_xe2))
|
||
return ebi('op_unpost').innerHTML = L.fu_xe1;
|
||
|
||
try {
|
||
var ores = JSON.parse(this.responseText);
|
||
}
|
||
catch (ex) {
|
||
return ebi('op_unpost').innerHTML = '<p>' + L.badreply + ':</p>' + unpre(this.responseText);
|
||
}
|
||
|
||
if (ores.u.length == 1 && ores.u[0].timeout) {
|
||
html.push('<p>' + L.un_nou + '</p>');
|
||
ores.u = [];
|
||
}
|
||
|
||
if (ores.c.length == 1 && ores.c[0].kinshi) {
|
||
html.push('<p>' + L.un_noc + '</p>');
|
||
ores.c = [];
|
||
}
|
||
|
||
for (var a = 0; a < ores.u.length; a++)
|
||
ores.u[a].k = 'u';
|
||
|
||
for (var a = 0; a < ores.c.length; a++)
|
||
ores.c[a].k = 'c';
|
||
|
||
var res = ores.u.concat(ores.c);
|
||
|
||
if (res.length) {
|
||
if (res.length == 2000)
|
||
html.push("<p>" + L.un_max);
|
||
else
|
||
html.push("<p>" + L.un_avail.format(ores.c.length, ores.u.length));
|
||
|
||
html.push("<br />" + L.un_m2 + "</p>");
|
||
html.push("<table><thead><tr><td></td><td>time</td><td>size</td><td>done</td><td>file</td></tr></thead><tbody>");
|
||
}
|
||
else
|
||
html.push('-- <em>' + (filt.value ? L.un_no2 : L.un_no1) + '</em>');
|
||
|
||
var mods = [10, 100, 1000];
|
||
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(
|
||
'<tr><td></td><td colspan="3" style="padding:.5em">' +
|
||
'<a me="' + me + '" class="n' + a + '" n2="' + (a + mods[b]) +
|
||
'" href="#">' + L.un_next.format(Math.min(mods[b], res.length - a)) + '</a></td></tr>');
|
||
|
||
var done = res[a].k == 'c';
|
||
html.push(
|
||
'<tr><td><a me="' + me + '" class="n' + a + '" href="#">' + (done ? L.un_del : L.un_abrt) + '</a></td>' +
|
||
'<td>' + unix2iso(res[a].at) + '</td>' +
|
||
'<td>' + ('' + res[a].sz).replace(/\B(?=(\d{3})+(?!\d))/g, " ") + '</td>' +
|
||
(done ? '<td>100%</td>' : '<td>' + res[a].pd + '%</td>') +
|
||
'<td>' + linksplit(res[a].vp).join('<span> / </span>') + '</td></tr>');
|
||
}
|
||
|
||
html.push("</tbody></table>");
|
||
ct.innerHTML = html.join('\n');
|
||
r.files = res;
|
||
r.me = me;
|
||
}
|
||
|
||
var q = get_evpath() + '?ups';
|
||
if (filt.value)
|
||
q += '&filter=' + uricom_enc(filt.value, true);
|
||
|
||
var xhr = new XHR();
|
||
xhr.open('GET', q, true);
|
||
xhr.onload = xhr.onerror = unpost_load_cb;
|
||
xhr.send();
|
||
|
||
ct.innerHTML = "<p><em>" + L.un_m3 + "</em></p>";
|
||
};
|
||
|
||
function linklist() {
|
||
var ret = [],
|
||
base = location.origin.replace(/\/$/, '');
|
||
|
||
for (var a = 0; a < r.files.length; a++)
|
||
ret.push(base + r.files[a].vp);
|
||
|
||
return ret.join('\r\n');
|
||
}
|
||
|
||
function unpost_delete_cb() {
|
||
if (this.status !== 200) {
|
||
var msg = unpre(this.responseText);
|
||
toast.err(9, L.un_derr + 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]'))
|
||
goto_unpost();
|
||
|
||
var fi = window.up2k && up2k.st.files;
|
||
if (fi && fi.length < 9) {
|
||
for (var a = 0; a < fi.length; a++) {
|
||
var f = fi[a];
|
||
if (!f.done && (f.rechecks || f.want_recheck) &&
|
||
!has(up2k.st.todo.handshake, f) &&
|
||
!has(up2k.st.busy.handshake, f)
|
||
) {
|
||
up2k.st.todo.handshake.push(f);
|
||
up2k.ui.seth(f.n, 2, L.u_hashdone);
|
||
up2k.ui.seth(f.n, 1, '📦 wait');
|
||
up2k.ui.move(f.n, 'bz');
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
ct.onclick = function (e) {
|
||
var tgt = e.target.closest('a[me]');
|
||
if (!tgt)
|
||
return;
|
||
|
||
if (!tgt.getAttribute('href'))
|
||
return;
|
||
|
||
ev(e);
|
||
var ame = tgt.getAttribute('me');
|
||
if (ame != r.me)
|
||
return toast.err(0, L.un_f5);
|
||
|
||
var n = parseInt(tgt.className.slice(1)),
|
||
n2 = parseInt(tgt.getAttribute('n2') || n + 1),
|
||
req = [];
|
||
|
||
for (var a = n; a < n2; a++) {
|
||
var links = QSA('#op_unpost a.n' + a);
|
||
if (!links.length)
|
||
continue;
|
||
|
||
var f = r.files[a];
|
||
if (f.k == 'u') {
|
||
var vp = vsplit(f.vp.split('?')[0]),
|
||
dfn = uricom_dec(vp[1]);
|
||
for (var iu = 0; iu < up2k.st.files.length; iu++) {
|
||
var uf = up2k.st.files[iu];
|
||
if (uf.name == dfn && uf.purl == vp[0])
|
||
return modal.alert(L.un_uf5);
|
||
}
|
||
}
|
||
req.push(uricom_dec(f.vp.split('?')[0]));
|
||
for (var b = 0; b < links.length; b++) {
|
||
links[b].removeAttribute('href');
|
||
links[b].innerHTML = '[busy]';
|
||
}
|
||
}
|
||
|
||
toast.show('inf r', 0, L.un_busy.format(req.length));
|
||
|
||
var xhr = new XHR();
|
||
xhr.n = n;
|
||
xhr.n2 = n2;
|
||
xhr.open('POST', SR + '/?delete&unpost&lim=' + req.length, true);
|
||
xhr.onload = xhr.onerror = 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 (e) {
|
||
ev(e);
|
||
filt.value = '';
|
||
r.load();
|
||
};
|
||
|
||
ebi('unpost_refresh').onclick = function (e) {
|
||
ev(e);
|
||
goto('unpost');
|
||
};
|
||
|
||
ebi('unpost_ulist').onclick = function (e) {
|
||
ev(e);
|
||
modal.alert(linklist());
|
||
};
|
||
|
||
ebi('unpost_ucopy').onclick = function (e) {
|
||
ev(e);
|
||
var txt = linklist();
|
||
cliptxt(txt + '\n', function () {
|
||
toast.inf(5, L.un_clip.format(txt.split('\n').length));
|
||
});
|
||
};
|
||
|
||
return r;
|
||
})();
|
||
|
||
|
||
function goto_unpost(e) {
|
||
unpost.load();
|
||
}
|
||
|
||
|
||
function wintitle(txt, noname) {
|
||
if (txt === undefined)
|
||
txt = '';
|
||
|
||
if (s_name && !noname)
|
||
txt = s_name + ' ' + txt;
|
||
|
||
txt += uricom_dec(get_evpath()).slice(1, -1).split('/').pop();
|
||
|
||
document.title = txt;
|
||
}
|
||
|
||
|
||
ebi('path').onclick = function (e) {
|
||
if (ctrl(e))
|
||
return true;
|
||
|
||
var a = e.target.closest('a[href]');
|
||
if (!a || !(a = a.getAttribute('href') + '') || !a.endsWith('/'))
|
||
return;
|
||
|
||
thegrid.setvis(true);
|
||
treectl.reqls(a, true);
|
||
return ev(e);
|
||
};
|
||
|
||
|
||
var scroll_y = -1;
|
||
var scroll_vp = '\n';
|
||
var scroll_obj = null;
|
||
function persist_scroll() {
|
||
var obj = scroll_obj;
|
||
if (!obj) {
|
||
var o1 = document.getElementsByTagName('html')[0];
|
||
var o2 = document.body;
|
||
obj = o1.scrollTop > o2.scrollTop ? o1 : o2;
|
||
}
|
||
var y = obj.scrollTop;
|
||
if (y > 0)
|
||
scroll_obj = obj;
|
||
|
||
scroll_y = y;
|
||
scroll_vp = get_evpath();
|
||
}
|
||
function restore_scroll() {
|
||
if (get_evpath() == scroll_vp && scroll_obj && scroll_obj.scrollTop < 1)
|
||
scroll_obj.scrollTop = scroll_y;
|
||
}
|
||
|
||
|
||
ebi('files').onclick = ebi('docul').onclick = function (e) {
|
||
if (!treectl.csel && e && (ctrl(e) || e.shiftKey))
|
||
return true;
|
||
|
||
if (!showfile.active())
|
||
persist_scroll();
|
||
|
||
var tgt = e.target.closest('a[id]');
|
||
if (tgt && tgt.getAttribute('id').indexOf('f-') === 0 && tgt.textContent.endsWith('/')) {
|
||
var el = treectl.find(tgt.textContent.slice(0, -1));
|
||
if (el) {
|
||
el.click();
|
||
return ev(e);
|
||
}
|
||
treectl.reqls(tgt.getAttribute('href'), true);
|
||
return ev(e);
|
||
}
|
||
if (tgt && /\.PARTIAL(\?|$)/.exec('' + tgt.getAttribute('href')) && !window.partdlok) {
|
||
ev(e);
|
||
modal.confirm(L.f_partial, function () {
|
||
window.partdlok = 1;
|
||
tgt.click();
|
||
}, null);
|
||
}
|
||
|
||
tgt = e.target.closest('a[hl]');
|
||
if (tgt) {
|
||
var a = ebi(tgt.getAttribute('hl')),
|
||
href = a.getAttribute('href'),
|
||
fun = function () {
|
||
showfile.show(href, tgt.getAttribute('lang'));
|
||
},
|
||
tfun = function () {
|
||
bcfg_set('taildoc', showfile.taildoc = true);
|
||
fun();
|
||
},
|
||
szs = ft2dict(a.closest('tr'))[0].sz,
|
||
sz = parseInt(szs.replace(/[, ]/g, ''));
|
||
|
||
if (sz < 1024 * 1024 || showfile.taildoc)
|
||
fun();
|
||
else
|
||
modal.confirm(L.f_bigtxt.format(f2f(sz / 1024 / 1024, 1)), fun, function() {
|
||
modal.confirm(L.f_bigtxt2, tfun, null)});
|
||
|
||
return ev(e);
|
||
}
|
||
|
||
tgt = e.target.closest('a');
|
||
if (tgt && tgt.closest('li.bn')) {
|
||
thegrid.setvis(true);
|
||
treectl.goto(tgt.getAttribute('href'), true);
|
||
return ev(e);
|
||
}
|
||
};
|
||
|
||
|
||
function reload_mp() {
|
||
if (mp && mp.au) {
|
||
mpo.au = mp.au;
|
||
mpo.au2 = mp.au2;
|
||
mpo.acs = mp.acs;
|
||
mpo.fau = mp.fau;
|
||
mpl.unbuffer();
|
||
}
|
||
var plays = QSA('tr>td:first-child>a.play');
|
||
for (var a = plays.length - 1; a >= 0; a--)
|
||
plays[a].parentNode.innerHTML = '-';
|
||
|
||
mp = new MPlayer();
|
||
if (mp.au && mp.au.tid && mp.au.evp == get_evpath()) {
|
||
var el = QS('a#a' + mp.au.tid);
|
||
if (el)
|
||
clmod(el, 'act', 1);
|
||
|
||
el = el && el.closest('tr');
|
||
if (el)
|
||
clmod(el, 'play', 1);
|
||
}
|
||
|
||
setTimeout(pbar.onresize, 1);
|
||
}
|
||
|
||
|
||
function reload_browser() {
|
||
filecols.set_style();
|
||
|
||
var parts = get_evpath().split('/'),
|
||
rm = ebi('entree'),
|
||
ftab = ebi('files'),
|
||
link = '', o;
|
||
|
||
while (rm.nextSibling)
|
||
rm.parentNode.removeChild(rm.nextSibling);
|
||
|
||
for (var a = 0; a < parts.length - 1; a++) {
|
||
link += parts[a] + '/';
|
||
var link2 = dks[link] ? addq(link, 'k=' + dks[link]) : link;
|
||
|
||
o = mknod('a');
|
||
o.setAttribute('href', link2);
|
||
o.textContent = uricom_dec(parts[a]) || '/';
|
||
ebi('path').appendChild(mknod('i'));
|
||
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;
|
||
}
|
||
|
||
reload_mp();
|
||
try { showsort(ftab); } catch (ex) { }
|
||
makeSortable(ftab, function () {
|
||
msel.origin_id(null);
|
||
msel.load(true);
|
||
thegrid.setdirty();
|
||
mp.read_order();
|
||
});
|
||
|
||
var ns = ['pro', 'epi', 'lazy']
|
||
for (var a = 0; a < ns.length; a++)
|
||
clmod(ebi(ns[a]), 'hidden', ebi('unsearch'));
|
||
|
||
if (up2k)
|
||
up2k.set_fsearch();
|
||
|
||
thegrid.setdirty();
|
||
msel.render();
|
||
}
|
||
treectl.hydrate();
|