diff --git a/copyparty/__main__.py b/copyparty/__main__.py index 1bd0ec72..4458c00c 100644 --- a/copyparty/__main__.py +++ b/copyparty/__main__.py @@ -563,6 +563,7 @@ def run_argparse(argv, formatter): ap2.add_argument("-mtp", metavar="M=[f,]BIN", type=u, action="append", help="read tag M using program BIN to parse the file") ap2 = ap.add_argument_group('ui options') + ap2.add_argument("--lang", metavar="LANG", type=u, default="eng", help="language") ap2.add_argument("--theme", metavar="NUM", type=int, default=0, help="default theme to use") ap2.add_argument("--themes", metavar="NUM", type=int, default=6, help="number of themes installed") ap2.add_argument("--js-browser", metavar="L", type=u, help="URL to additional JS to include") diff --git a/copyparty/httpcli.py b/copyparty/httpcli.py index 9900b42c..1292f6ec 100644 --- a/copyparty/httpcli.py +++ b/copyparty/httpcli.py @@ -2252,6 +2252,7 @@ class HttpCli(object): "readme": readme, "title": html_escape(self.vpath, crlf=True), "srv_info": srv_info, + "lang": self.args.lang, "dtheme": self.args.theme, "themes": self.args.themes, "turbolvl": self.args.turbo, diff --git a/copyparty/web/browser.css b/copyparty/web/browser.css index af8ec40b..9e9b4cd5 100644 --- a/copyparty/web/browser.css +++ b/copyparty/web/browser.css @@ -1839,27 +1839,29 @@ html.y #bbox-overlay figcaption a { position: absolute; top: 40%; top: calc(50% - .5em); + top: calc(50% - 1.2vw); left: -.8em; + transition: top 0.12s; } .dropdesc>div>div+div { left: auto; right: -.8em; } .dropdesc b { - position: relative; font-size: .5em; font-size: 2vw; - top: -.25em; - top: -.2vw; margin: 0 .8em; margin: 0 1.25vw; transition: font-size 0.12s; } -.dropdesc.hl b { +.dropdesc.hl.ok b { border-bottom: .1em solid #fff; font-size: .6em; font-size: 2.5vw; } +.dropdesc.hl.ok>div>div { + top: calc(50% - 1.7vw); +} .dropzone { z-index: 80386; height: 50%; diff --git a/copyparty/web/browser.html b/copyparty/web/browser.html index 1a385aa1..d32a46ce 100644 --- a/copyparty/web/browser.html +++ b/copyparty/web/browser.html @@ -68,7 +68,7 @@

- 🌲 + 🌲 {%- for n in vpnodes %} {{ n[1] }} {%- endfor %} @@ -120,7 +120,7 @@
{{ logues[1] }}
-

control-panel

+

control-panel

π @@ -138,6 +138,7 @@ themes = {{ themes }}, dtheme = "{{ dtheme }}", srvinf = "{{ srv_info }}", + lang = "{{ lang }}", def_hcols = {{ def_hcols|tojson }}, have_up2k_idx = {{ have_up2k_idx|tojson }}, have_tags_idx = {{ have_tags_idx|tojson }}, diff --git a/copyparty/web/browser.js b/copyparty/web/browser.js index 347768a3..fab2c209 100644 --- a/copyparty/web/browser.js +++ b/copyparty/web/browser.js @@ -1,23 +1,582 @@ "use strict"; -function dbg(msg) { - ebi('path').innerHTML = msg; -} var XHR = XMLHttpRequest; +var Ls = { + "eng": { + "tt": "English", + + "cols": { + "c": "action buttons", + "dur": "duration", + "q": "quality / bitrate", + "Ac": "audio codec", + "Vc": "video codec", + "Ahash": "audio checksum", + "Vhash": "video checksum", + "Res": "resolution", + "T": "filetype", + "aq": "audio quality / bitrate", + "vq": "video quality / bitrate", + "pixfmt": "subsampling / pixel structure", + "resw": "horizontal resolution", + "resh": "veritcal resolution", + "chs": "audio channels", + "hz": "sample rate" + }, + + "goh": "control-panel", + "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»", + "ot_unpost": "unpost: delete your recent uploads", + "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', + "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', + + "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_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_selzip": "download selection as archive", + "wt_npirc": "copy irc-formatted track info", + "wt_nptxt": "copy plaintext track info", + "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_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_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
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": "failed / rejected / not-found", + "uct_done": "ok and ng combined", + "uct_bz": "hashing or uploading", + "uct_q": "idle, pending", + + "utl_name": "filename", + "utl_stat": "status", + "utl_prog": "progress", + + "ul_flagblk": "the files were added to the queue
however there is a busy up2k in another browser tab,
so waiting for that to finish first", + + "udt_up": "Upload", + "udt_srch": "Search", + "udt_drop": "drop it here", + + "cl_opts": "switches", + "cl_themes": "theme", + "cl_langs": "language", + "cl_ziptype": "folder download", + "cl_uopts": "up2k switches", + "cl_favico": "favicon", + "cl_keytype": "key notation", + "cl_hiddenc": "hidden columns", + + "ct_thumb": "in icon view, toggle icons or thumbnails$NHotkey: T", + "ct_dots": "show hidden files (if server permits)", + "ct_readme": "show README.md in folder listings", + + "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 "does this have the same filesize on the server?" 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", + + "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 theoretically catch most unfinished / corrupted uploads, but is not a substitute for doing a verification pass with turbo disabled afterwards", + + "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", + + "cft_text": "favicon text (blank and refresh to disable)", + "cft_fg": "foreground color", + "cft_bg": "background color", + + "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": "playback mode", + "ml_tcode": "transcode", + "ml_tint": "tint", + "ml_eq": "audio equalizer", + + "mt_preload": "start loading the next song near the end for gapless playback\">preload", + "mt_fullpre": "try to preload the entire song;$N✅ enable on unreliable connections,$N❌ disable on slow connections probably\">full", + "mt_npclip": "show buttons for clipboarding the currently playing song\">/np", + "mt_octl": "os integration (media hotkeys / osd)\">os-ctl", + "mt_oseek": "allow seeking through os integration\">seek", + "mt_oscv": "show album cover in osd\">art", + "mt_mloop": "loop the open folder\">🔁 loop", + "mt_mnext": "load the next folder and continue\">📂 next", + "mt_cflac": "convert flac to opus\">flac", + "mt_caac": "convert aac / m4a to opus\">aac", + "mt_coth": "convert all others (not mp3) to opus\">oth", + "mt_tint": "background level (0-100) on the seekbar$Nto make buffering less distracting", + "mt_eq": "enables the equalizer and gain control;$Nboost 0 = unmodified 100% volume$N$Nenabling the equalizer makes gapless albums fully gapless, so leave it on with all the values at zero if you care about that", + + "mb_play": "play", + "mm_hashplay": "play this audio file?", + "mp_breq": "need firefox 82+ or chrome 73+ or iOS 15+", + "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_e5xx": "Could not play audio; server error ", + + "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", + + "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_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", + + "fd_ok": "delete OK", + "fd_err": "delete failed:\n", + "fd_busy": "deleting {0} items...\n\n{1}", + "fd_warn1": "DELETE these {0} items?", + "fd_warn2": "Last chance! No way to undo. Delete?", + + "fc_ok": "cut {0} items", + "fc_warn": 'cut {0} items\n\nbut: only this browser-tab can paste them\n(since the selection is so absolutely massive)', + + "fp_ecut": "first cut some files / folders to paste / move\n\nnote: you can cut / paste across different browser tabs", + "fp_ename": "these {0} items cannot be moved here (names already exist):", + "fp_ok": "move OK", + "fp_busy": "moving {0} items...\n\n{1}", + "fp_err": "move failed:\n", + "fp_confirm": "move these {0} items here?", + "fp_etab": 'failed to read clipboard from other browser tab', + + "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\">❌ close", + "tvt_dl": "download this file\">💾 download", + "tvt_prev": "show previous document$NHotkey: i\">⬆ prev", + "tvt_next": "show next document$NHotkey: K\">⬇ next", + "tvt_sel": "select file   ( for cut / delete / ... )$NHotkey: S\">sel", + + "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_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_prev": "search results below are from a previous query:\n ", + "sl_close": "close search results", + "sl_hits": "showing {0} hits", + "sl_moar": "load more", + + "md_eshow": "cannot show ", + + "xhr403": "403: Access denied\n\ntry pressing F5, maybe you got logged out", + "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": "plain gnutar file (linux / mac)", + "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 below", + "un_upd": "refresh list", + "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_max": "showing first 2000 files (use the filter)", + "un_avail": "{0} uploads can be deleted", + "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_del": "delete", + "un_m3": "loading your recent uploads...", + "un_busy": "deleting {0} files...", + + "u_https1": "you should", + "u_https2": "switch to https", + "u_https3": "for much better performance", + "u_ancient": 'your browser is impressively ancient -- maybe you should use bup instead', + "u_ever": "this is the basic uploader; up2k needs at least
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'this is the basic uploader; up2k is better', + "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_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_just1": '\nMaybe it works better if you select just one file', + "u_asku": 'upload these {0} files?', + 'u_asks': 'search for these {0} files?', + "u_etadone": 'Done ({0}, {1} files)', + "u_etaprep": '(preparing to upload)', + "u_hashdone": 'hashing done', + "u_hashing": 'hash', + "u_upping": 'uploading', + "u_cuerr": "failed to upload chunk {0} of {1};\nprobably harmless, continuing\n\nfile: {2}", + "u_cuerr2": "server rejected upload (chunk {0} of {1});\n\nfile: {2}\n\nerror ", + "u_ehsfin": "server rejected the request to finalize upload", + "u_ehssrch": "server rejected the request to perform search", + "u_ehsinit": "server rejected the request to initiate upload", + + "lang_set": "refresh now to make the change take effect?", + }, + "nor": { + "tt": "Norsk", + + "cols": { + "c": "handlingsknapper", + "dur": "varighet", + "q": "kvalitet / bitrate", + "Ac": "lyd-format", + "Vc": "video-format", + "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" + }, + + "goh": "kontrollpanel", + "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", + "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', + "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"', + + "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_pst": "lim inn filer (som tidligere ble klippet ut 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_selzip": "last ned de valgte filene som et arkiv", + "wt_npirc": "kopier sang-info (irc-formattert)", + "wt_nptxt": "kopier sang-info", + "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_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_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
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_stat": "status", + "utl_prog": "progresjon", + + "ul_flagblk": "filene har blitt lagt i køen
men det er en annen nettleserfane som holder på med befaring eller opplastning akkurat nå,
så venter til den er ferdig først", + + "udt_up": "Last opp", + "udt_srch": "Søk", + "udt_drop": "Slipp filene her", + + "cl_opts": "brytere", + "cl_themes": "utseende", + "cl_langs": "språk", + "cl_ziptype": "nedlastning av mapper", + "cl_uopts": "up2k-brytere", + "cl_favico": "favicon", + "cl_keytype": "notasjon for musikalsk dur", + "cl_hiddenc": "skjulte kolonner", + + "ct_thumb": "vis miniatyrbilder istedenfor ikoner$NSnarvei: T", + "ct_dots": "vis skjulte filer (gitt at serveren tillater det)", + "ct_readme": "vis README.md nedenfor filene", + + "cut_turbo": "forenklet befaring ved opplastning; bør sannsynlig ikke 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 filstørrelsen stemmer. Så dersom en korrupt fil skulle befinne seg på serveren allerede, på samme sted med samme størrelse og navn, så blir det ikke oppdaget.$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", + + "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$Nburde oppdage og gjenoppta de fleste ufullstendige opplastninger, men er ikke en fullverdig erstatning for å deaktivere turbo og gjøre en skikkelig sjekk", + + "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", + + "cft_text": "ikontekst (blank ut og last siden på nytt for å deaktivere)", + "cft_fg": "farge", + "cft_bg": "bakgrunnsfarge", + + "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": "spillemodus", + "ml_tcode": "konvertering", + "ml_tint": "tint", + "ml_eq": "audio equalizer / tonekontroll", + + "mt_preload": "hent ned litt av neste sang i forkant,$Nslik at pausen i overgangen blir mindre\">forles", + "mt_fullpre": "hent ned hele neste sang, ikke bare litt:$N✅ skru på hvis nettet ditt er ustabilt,$N❌ skru av hvis nettet ditt er tregt\">full", + "mt_npclip": "vis knapper for å kopiere info om sangen du hører på\">/np", + "mt_octl": "integrering med operativsystemet (fjernkontroll, info-skjerm)\">os-ctl", + "mt_oseek": "tillat spoling med fjernkontroll\">spoling", + "mt_oscv": "vis album-cover på infoskjermen\">bilde", + "mt_mloop": "repeter hele mappen\">🔁 repeter", + "mt_mnext": "hopp til neste mappe og fortsett\">📂 neste", + "mt_cflac": "konverter flac-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_tint": "nivå av bakgrunnsfarge på søkestripa (0-100),$Ngjør oppdateringer mindre distraherende", + "mt_eq": "aktiver tonekontroll og forsterker;$Nboost 0 = normal volumskala$N$Nreduserer også dødtid imellom sangfiler", + + "mb_play": "lytt", + "mm_hashplay": "spill denne sangen?", + "mp_breq": "krever firefox 82+, chrome 73+, eller iOS 15+", + "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_e5xx": "Avspilling feilet: ", + + "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", + + "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
og / eller mønster (regulære uttrykk)\">avansert", + "fr_case": "versalfølsomme uttrykk\">Aa", + "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", + + "fd_ok": "sletting OK", + "fd_err": "sletting feilet:\n", + "fd_busy": "sletter {0} filer...\n\n{1}", + "fd_warn1": "SLETT disse {0} filene?", + "fd_warn2": "Siste sjanse! Dette kan ikke angres. Slett?", + + "fc_ok": "klippet ut {0} filer", + "fc_warn": 'klippet ut {0} filer\n\nmen: kun denne nettleserfanen har mulighet til å lime dem inn et annet sted, siden antallet filer er helt hinsides', + + "fp_ecut": "du må klippe ut 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": "disse {0} filene kan ikke flyttes til målmappen fordi det allerede finnes filer med samme navn:", + "fp_ok": "flytting OK", + "fp_busy": "flytter {0} filer...\n\n{1}", + "fp_err": "flytting feilet:\n", + "fp_confirm": "flytt disse {0} filene hit?", + "fp_etab": 'kunne ikke lese listen med filer ifra den andre nettleserfanen', + + "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\">❌ close", + "tvt_dl": "last ned denne filen\">💾 last ned", + "tvt_prev": "vis forrige dokument$NSnarvei: i\">⬆ prev", + "tvt_next": "vis neste dokument$NSnarvei: K\">⬇ next", + "tvt_sel": "markér filen   ( for utklipp / sletting / ... )$NSnarvei: S\">sel", + + "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_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_prev": "søkeresultatene er fra et tidligere søk:\n ", + "sl_close": "lukk søkeresultater", + "sl_hits": "viser {0} treff", + "sl_moar": "hent flere", + + "md_eshow": "kan ikke vise ", + + "xhr403": "403: Tilgang nektet\n\nkanskje du ble logget ut? prøv å trykk F5", + "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_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", + "un_upd": "oppdater listen", + "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_max": "viser de første 2000 filene (bruk filteret for å innsnevre)", + "un_avail": "{0} filer kan slettes", + "un_m2": "sortert etter opplastningstid – nyeste først:", + "un_no1": "men nei, her var det jaggu ingenting", + "un_no2": "men nei, her var det jaggu ingenting som passer overens med filteret", + "un_next": "slett de neste {0} filene nedenfor", + "un_del": "slett", + "un_m3": "henter listen med nylig opplastede filer...", + "un_busy": "sletter {0} filer...", + + "u_https1": "du burde", + "u_https2": "bytte til https", + "u_https3": "for mye høyere hastighet", + "u_ancient": 'nettleseren din er prehistorisk -- mulig du burde bruke bup istedenfor', + "u_ever": "dette er den primitive opplasteren; up2k krever minst:
chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1", + "u_su2k": 'dette er den primitive opplasteren; up2k er bedre', + "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_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_just1": '\nFunker kanskje bedre hvis du bare tar én fil om gangen', + "u_asku": 'Laste opp disse {0} filene?', + 'u_asks': 'Søk etter disse {0} filene?', + "u_etadone": 'Ferdig ({0}, {1} filer)', + "u_etaprep": '(forbereder opplastning)', + "u_hashdone": 'befaring ferdig', + "u_hashing": 'les', + "u_upping": 'sender', + "u_cuerr": "kunne ikke laste opp del {0} av {1};\nsikkert harmløst, fortsetter\n\nfil: {2}", + "u_cuerr2": "server nektet opplastningen (del {0} of {1});\n\nfile: {2}\n\nerror ", + "u_ehsfin": "server nektet forespørselen om å ferdigstille filen", + "u_ehssrch": "server nektet forespørselen om å utføre søk", + "u_ehsinit": "server nektet forespørselen om å begynne en ny opplastning", + + "lang_set": "siden må lastes på nytt for at endringen skal tre i kraft -- passer det nå?", + }, +}; +var L = Ls[sread("lang") || lang]; +if (Ls.eng && L != Ls.eng) { + for (var k in Ls.eng) + if (Ls.eng.hasOwnProperty(k) && !L[k]) + L[k] = Ls.eng[k]; +} // toolbar ebi('ops').innerHTML = ( - '--' + - '🔎' + - (have_del && have_unpost ? '🧯' : '') + + '--' + + '🔎' + + (have_del && have_unpost ? '🧯' : '') + '🚀' + - '🎈' + - '📂' + - '📝' + - '📟' + - '🎺' + - '⚙️' + + '🎈' + + '📂' + + '📝' + + '📟' + + '🎺' + + '⚙️' + '
' ); @@ -26,23 +585,23 @@ ebi('ops').innerHTML = ( ebi('widget').innerHTML = ( '
' + 'namedel.cut📋paste' + + ' href="#" id="fren" tt="' + L.wt_ren + '">✎namedel.cut📋paste' + 'sel.
allsel.
inv.zip' + + ' href="#" id="selall" tt="' + L.wt_selall + '">sel.
allsel.
inv.zip' + '
📋irc📋txt' + + ' href="#" id="npirc" tt="' + L.wt_npirc + '">📋irc📋txt' + '♫' + '
' + '
' + - '
' + + '
' + ' ' + ' ' + ' ' + @@ -56,18 +615,18 @@ ebi('op_up2k').innerHTML = ( '\n' + ' \n' + - ' \n' + + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + ' \n' + @@ -75,7 +634,7 @@ ebi('op_up2k').innerHTML = ( ' \n' + ' \n' + ' \n' + @@ -85,26 +644,25 @@ ebi('op_up2k').innerHTML = ( '
\n' + '
\n' + - ' \n' + - ' drop files / folders
\n' + - ' here (or click me)\n' + + ' \n' + L.ul_btn + '
\n' + '
\n' + '
\n' + '
\n' + - ' hash: (no uploads are queued yet)
\n' + - ' send: (no uploads are queued yet)
\n' + - '
done: (no uploads are queued yet)\n' + + L.ul_hash + ': (' + L.ul_idle1 + ')
\n' + + L.ul_send + ': (' + L.ul_idle1 + ')
\n' + + '
' + + L.ul_done + ': (' + L.ul_idle1 + ')\n' + '
\n' + '
\n' + - ' ok 0ng 0done 0busy 0que 0\n' + + ' ok 0ng 0done 0busy 0que 0\n' + '
\n' + '\n' + @@ -112,15 +670,15 @@ ebi('op_up2k').innerHTML = ( '

parallel uploads:

' + L.ul_par + '
\n' + ' \n' + - ' \n' + + ' \n' + ' \n' + ' \n' + - ' \n' + + ' \n' + ' \n' + ' \n' + - ' \n' + + ' \n' + '
\n' + ' +
 \n' + '
\n' + ' \n' + ' \n' + - ' \n' + - ' \n' + - ' \n' + + ' \n' + + ' \n' + + ' \n' + ' \n' + ' \n' + ' \n' + '
filenamestatusprogress' + L.utl_name + '' + L.utl_stat + '' + L.utl_prog + '
\n' + - '

the files were added to the queue
however there is a busy up2k in another browser tab,
so waiting for that to finish first

\n' + + '

' + L.ul_flagblk + '

\n' + '

' ); @@ -129,8 +687,8 @@ ebi('op_up2k').innerHTML = ( var o = mknod('div'); o.innerHTML = ( '
\n' + - '
🚀 Upload
🚀Upload
Upload🚀
\n' + - '
🔎 Search
🔎Search
Search🔎
\n' + + '
🚀 ' + L.udt_up + '
🚀' + L.udt_up + '
' + L.udt_up + '🚀
\n' + + '
🔎 ' + L.udt_srch + '
🔎' + L.udt_srch + '
' + L.udt_srch + '🔎
\n' + '
\n' + '
\n' + '
' @@ -142,64 +700,71 @@ ebi('op_up2k').innerHTML = ( // config panel ebi('op_cfg').innerHTML = ( '
\n' + - '

switches

\n' + + '

' + L.cl_opts + '

\n' + '
\n' + ' ℹ️ tooltips\n' + - ' 田 the grid\n' + - ' 🖼️ thumbs\n' + - ' dotfiles\n' + - ' 📜 readme\n' + + ' 田 the grid\n' + + ' 🖼️ thumbs\n' + + ' dotfiles\n' + + ' 📜 readme\n' + '
\n' + '
\n' + '
\n' + - '

themes

\n' + + '

' + L.cl_themes + '

\n' + '
\n' + '
\n' + '
\n' + + '
\n' + + '

' + L.cl_langs + '

\n' + + '
\n' + + '
\n' + + '
\n' + (have_zip ? ( - '

folder download

\n' + '

' + L.cl_ziptype + '

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

up2k switches

\n' + + '

' + L.cl_uopts + '

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

favicon 🎉

\n' + + '

' + L.cl_favico + ' 🎉

\n' + '
\n' + - ' ' + - ' ' + - ' ' + + ' ' + + ' ' + + ' ' + ' \n' + '
\n' + '
\n' + - '

key notation

\n' + - '

hidden columns

' + '

' + L.cl_keytype + '

\n' + + '

' + L.cl_hiddenc + '

' ); // navpane ebi('tree').innerHTML = ( '
\n' + - ' 🍞...\n' + + ' 🍞...\n' + ' +\n' + ' \n' + - ' 🎯\n' + - ' 📃\n' + - ' 📌\n' + - ' a\n' + - ' \n' + - ' 👀\n' + + ' 🎯\n' + + ' 📃\n' + + ' 📌\n' + + ' a\n' + + ' \n' + + ' 👀\n' + '
\n' + '\n' + '\n' + '\n' + '
 
' ); +ebi('entree').setAttribute('tt', L.tt_entree); +ebi('goh').textContent = L.goh; (function () { @@ -317,33 +882,33 @@ var mpl = (function () { var have_mctl = 'mediaSession' in navigator && window.MediaMetadata; ebi('op_player').innerHTML = ( - '

switches

' + - 'preload' + - 'full' + - '/np' + - 'os-ctl' + - 'seek' + - 'art' + + '

' + L.cl_opts + '

' + + '' + - '🔁 loop' + - '📂 next' + + '

' + L.ml_pmode + '

' + + 'flac' + - 'aac' + - 'oth' + + '

' + L.ml_tcode + '

' + + '' + + '

' + L.ml_tint + '

' + + '' + '
' + - '

audio equalizer

'); + '

' + L.ml_eq + '

'); var r = { "pb_mode": (sread('pb_mode') || 'loop').split('-')[0], @@ -372,13 +937,13 @@ var mpl = (function () { r.os_ctl = !r.os_ctl && have_mctl; bcfg_set('au_os_ctl', r.os_ctl); if (!have_mctl) - toast.err(5, 'need firefox 82+ or chrome 73+ or iOS 15+'); + 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].textContent.indexOf(r.pb_mode) != -1); + clmod(btns[a], 'on', btns[a].getAttribute("m") == r.pb_mode); btns[a].onclick = set_pb_mode; } } @@ -386,7 +951,7 @@ var mpl = (function () { function set_pb_mode(e) { ev(e); - r.pb_mode = this.textContent.split(' ').pop(); + r.pb_mode = this.getAttribute('m'); swrite('pb_mode', r.pb_mode); draw_pb_mode(); } @@ -541,7 +1106,7 @@ function MPlayer() { var tid = link.getAttribute('id'); r.order.push(tid); r.tracks[tid] = url; - tds[0].innerHTML = '
play'; + tds[0].innerHTML = '' + L.mb_play + ''; ebi('a' + tid).onclick = ev_play; clmod(trs[a], 'au', 1); } @@ -1374,7 +1939,7 @@ var audio_eq = (function () { } var html = [''], + 'enable'], h2 = [], h3 = [], h4 = []; var vs = []; @@ -1531,7 +2096,7 @@ function play(tid, is_ev, seek) { return true; } catch (ex) { - toast.err(0, esc('playback failed: ' + basenames(ex))); + toast.err(0, esc(L.mm_playerr + basenames(ex))); } clmod(ebi(oid), 'act'); setTimeout(next_song, 5000); @@ -1545,34 +2110,34 @@ function evau_error(e) { switch (eplaya.error.code) { case eplaya.error.MEDIA_ERR_ABORTED: - err = "You aborted the playback attempt (how tho)"; + err = L.mm_eabrt; break; case eplaya.error.MEDIA_ERR_NETWORK: - err = "Your internet connection is wonky"; + err = L.mm_enet; break; case eplaya.error.MEDIA_ERR_DECODE: - err = "This file is supposedly corrupted??"; + err = L.mm_edec; break; case eplaya.error.MEDIA_ERR_SRC_NOT_SUPPORTED: - err = "Your browser does not understand this audio format"; + 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, 'your browser cannot play aac/m4a files;\ntranscoding to opus is now enabled'); + toast.warn(10, L.mm_opusen); return; } catch (ex) { } } break; default: - err = 'Unknown Errol'; + err = L.mm_eunk; break; } var em = '' + eplaya.error.message, mfile = '\n\nFile: «' + uricom_dec(eplaya.src.split('/').pop())[0] + '»', - e404 = 'Could not play audio; error 404: File not found.', - e403 = 'Could not play audio; error 403: Access denied.\n\nTry pressing F5 to reload, maybe you got logged out'; + e404 = L.mm_e404, + e403 = L.mm_e403; if (em) err += '\n\n' + em; @@ -1594,7 +2159,7 @@ function evau_error(e) { return; err = this.status == 403 ? e403 : this.status == 404 ? e404 : - 'Could not play audio; server error ' + this.status; + L.mm_e5xx + this.status; toast.warn(15, esc(basenames(err + mfile))); }; @@ -1610,7 +2175,7 @@ function autoplay_blocked(seek) { fn = uricom_dec(fn.replace(/\+/g, ' '))[0]; - modal.confirm('
play this audio file?
\n«' + esc(fn) + '»', function () { + modal.confirm('
' + L.mm_hashplay + '
\n«' + esc(fn) + '»', function () { // chrome 91 may permanently taint on a failed play() // depending on win10 settings or something? idk mp.au = null; @@ -1912,17 +2477,17 @@ var fileman = (function () { clmod(bpst, 'hide', !(have_mv && has(perms, 'write'))); clmod(ebi('wfm'), 'act', QS('#wfm a.en:not(.hide)')); - bpst.setAttribute('tt', 'paste ' + r.clip.length + ' items$NHotkey: ctrl-V'); + bpst.setAttribute('tt', L.ft_paste.format(r.clip.length)); }; r.rename = function (e) { ev(e); if (clgot(bren, 'hide')) - return toast.err(3, 'cannot rename:\nyou do not have “move” permission in this folder'); + return toast.err(3, L.fr_eperm); var sel = msel.getsel(); if (!sel.length) - return toast.err(3, 'select at least one item to rename'); + return toast.err(3, L.fr_emore); var f = [], base = vsplit(sel[0].vp)[0], @@ -1969,23 +2534,23 @@ var fileman = (function () { var html = sel.length > 1 ? ['
'] : [ '
', - '', + '' + '', - '', - 'advanced', - 'case', + '', + '
', - 'enable
', '', '', '', '
regex
format
preset', - '', - '', + '', + '', '
' ]); @@ -1999,7 +2564,7 @@ var fileman = (function () { else { html.push( '
' + - ''); + ''); for (var a = 0; a < f.length; a++) html.push( '
new nameold name
' + L.fr_lnew + '' + L.fr_lold + '
' + @@ -2012,7 +2577,7 @@ var fileman = (function () { html.push('
'); if (sel.length == 1) { - html.push('

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

'); + html.push('

' + L.fr_tags + '

'); for (var a = 0; a < mkeys.length; a++) html.push(''); @@ -2095,9 +2660,9 @@ var fileman = (function () { } inew.onclick = function (e) { ev(e); - modal.prompt('provide a name for your new preset', ifmt.value, function (name) { + modal.prompt(L.fr_pname, ifmt.value, function (name) { if (!name) - return toast.warn(3, 'aborted'); + return toast.warn(3, L.fr_aborted); presets[name] = [ire.value, ifmt.value]; jwrite('rn_pre', presets); @@ -2175,13 +2740,13 @@ var fileman = (function () { return rn_cancel(); } - toast.inf(0, esc('renaming ' + f.length + ' items\n\n' + f[0].ofn)); + 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 !== 200) { var msg = this.responseText; - toast.err(9, 'rename failed:\n' + msg); + toast.err(9, L.fr_efail + msg); return; } @@ -2199,7 +2764,7 @@ var fileman = (function () { r.delete = function (e) { ev(e); if (clgot(bdel, 'hide')) - return toast.err(3, 'cannot delete:\nyou do not have “delete” permission in this folder'); + return toast.err(3, L.fd_eperm); var sel = msel.getsel(), vps = []; @@ -2208,18 +2773,18 @@ var fileman = (function () { vps.push(sel[a].vp); if (!sel.length) - return toast.err(3, 'select at least 1 item to delete'); + return toast.err(3, L.fd_emore); function deleter() { var xhr = new XHR(), vp = vps.shift(); if (!vp) { - toast.ok(2, 'delete OK'); + toast.ok(2, L.fd_ok); treectl.goto(get_evpath()); return; } - toast.inf(0, esc('deleting ' + (vps.length + 1) + ' items\n\n' + vp)); + toast.show('inf r', 0, esc(L.fd_busy.format(vps.length + 1, vp)), 'r'); xhr.open('GET', vp + '?delete', true); xhr.onload = xhr.onerror = delete_cb; @@ -2228,27 +2793,27 @@ var fileman = (function () { function delete_cb() { if (this.status !== 200) { var msg = this.responseText; - toast.err(9, 'delete failed:\n' + msg); + toast.err(9, L.fd_err + msg); return; } deleter(); } - modal.confirm('
DANGER
\nDELETE these ' + vps.length + ' items?
    ' + uricom_adec(vps, true).join('') + '
', function () { - modal.confirm('Last chance! Delete?', deleter, null); + modal.confirm('
DANGER
\n' + L.fd_warn1.format(vps.length) + '
    ' + uricom_adec(vps, true).join('') + '
', function () { + modal.confirm(L.fd_warn2, deleter, null); }, null); }; r.cut = function (e) { ev(e); if (clgot(bcut, 'hide')) - return toast.err(3, 'cannot cut:\nyou do not have “move” permission in this folder'); + return toast.err(3, L.fc_eperm); var sel = msel.getsel(), vps = []; if (!sel.length) - toast.err(3, 'select at least 1 item to cut'); + toast.err(3, L.fc_emore); var els = []; for (var a = 0; a < sel.length; a++) { @@ -2273,20 +2838,20 @@ var fileman = (function () { swrite('fman_clip', vps); r.tx(stamp); if (sel.length) - toast.inf(1.5, 'cut ' + sel.length + ' items'); + toast.inf(1.5, L.fc_ok.format(sel.length)); } catch (ex) { - toast.warn(30, 'cut ' + sel.length + ' items\n\nbut: only this browser-tab can paste them\n(since the selection is so absolutely massive)'); + toast.warn(30, L.fc_warn.format(sel.length)); } }; r.paste = function (e) { ev(e); if (clgot(bpst, 'hide')) - return toast.err(3, 'cannot paste:\nyou do not have “write” permission in this folder'); + return toast.err(3, L.fp_eperm); if (!r.clip.length) - return toast.err(5, 'first cut some files/folders to paste\n\nnote: you can cut/paste across different browser tabs'); + return toast.err(5, L.fp_ecut); var req = [], exists = [], @@ -2310,7 +2875,7 @@ var fileman = (function () { } if (exists.length) - toast.warn(30, 'these ' + exists.length + ' items cannot be pasted here (names already exist):
    ' + uricom_adec(exists, true).join('') + '
'); + toast.warn(30, L.fp_ename.format(exists.length) + '
    ' + uricom_adec(exists, true).join('') + '
'); if (!req.length) return; @@ -2320,12 +2885,12 @@ var fileman = (function () { vp = req.shift(); if (!vp) { - toast.ok(2, 'paste OK'); + toast.ok(2, L.fp_ok); treectl.goto(get_evpath()); r.tx(srcdir); return; } - toast.inf(0, esc('pasting ' + (req.length + 1) + ' items\n\n' + uricom_dec(vp)[0])); + toast.show('inf r', 0, esc(L.fp_busy.format(req.length + 1, uricom_dec(vp)[0]))); var dst = get_evpath() + vp.split('/').pop(); @@ -2336,13 +2901,13 @@ var fileman = (function () { function paste_cb() { if (this.status !== 200) { var msg = this.responseText; - toast.err(9, 'paste failed:\n' + msg); + toast.err(9, L.fp_err + msg); return; } paster(); } - modal.confirm('paste these ' + req.length + ' items here?
    ' + uricom_adec(req, true).join('') + '
', function () { + modal.confirm(L.fp_confirm.format(req.length) + '
    ' + uricom_adec(req, true).join('') + '
', function () { paster(); jwrite('fman_clip', [Date.now()]); }, null); @@ -2356,7 +2921,7 @@ var fileman = (function () { var fc = jread('fman_clip', []); if (!fc || !fc.length || fc[0] != n) { if (++tries > 10) - return modal.alert('failed to read clipboard from other browser tab'); + return modal.alert(L.fp_etab); return setTimeout(fun, 100); } @@ -2507,9 +3072,11 @@ var showfile = (function () { if (e.total < 1024 * 256) return; - var m = 'Loading text document:\n\n' + esc(this.fname) + '\n\n' + - f2f(e.loaded / 1024 / 1024, 1) + ' of ' + - f2f(e.total / 1024 / 1024, 1) + ' MiB loaded'; + 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; @@ -2522,7 +3089,7 @@ var showfile = (function () { if (this.toasted) toast.hide(); - if (!xhrchk(this, "could not load textfile:\n\nerror ", "404, file not found")) + if (!xhrchk(this, L.tv_xe1, L.tv_xe2)) return; render([this.url, '', this.responseText], this.no_push); @@ -2660,7 +3227,7 @@ var showfile = (function () { }; r.mktree = function () { - var html = ['
  • list of textfiles in
    ' + linksplit(get_vpath()).join('') + '
  • ']; + var html = ['
  • ' + L.tv_lst + '
    ' + linksplit(get_vpath()).join('') + '
  • ']; for (var a = 0; a < r.files.length; a++) { var file = r.files[a]; html.push('
  • \n' + - '❌ close\n' + - '💾 download\n' + - '⬆ prev\n' + - '⬇ next\n' + - 'sel\n' + + '' + - 'multiselect zoom: ' + + ' ' + - '+ chop: ' + - ' ' + - '+ sort by: ' + - 'name ' + - 'size ' + - 'date ' + - 'type' + + '+ ' + L.gt_chop + ': ' + + ' ' + + '+ ' + L.gt_sort + ': ' + + '' + L.gt_name + ' ' + + '' + L.gt_sz + ' ' + + '' + L.gt_ts + ' ' + + '' + L.gt_ext + '' + '' + '
    ' ); @@ -3007,6 +3574,13 @@ var thegrid = (function () { } ebi('ggrid').innerHTML = html.join('\n'); + 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; @@ -3419,7 +3993,7 @@ document.onkeydown = function (e) { var q = ebi('q_raw').value, vq = ebi('files').getAttribute('q_raw'); - srch_msg(false, (q == vq) ? '' : 'search results below are from a previous query:\n ' + (vq ? vq : '(*)')); + srch_msg(false, (q == vq) ? '' : L.sm_prev + (vq ? vq : '(*)')); } function encode_query() { @@ -3549,8 +4123,7 @@ document.onkeydown = function (e) { var html = mk_files_header(tagord), seen = {}; html.push('
  • '); - html.push(''); + html.push(''); for (var a = 0; a < res.hits.length; a++) { var r = res.hits[a], @@ -3859,7 +4432,7 @@ var treectl = (function () { } function recvtree() { - if (!xhrchk(this, "could not list subfolders:\n\nerror ", "404, folder not found")) + if (!xhrchk(this, L.tl_xe1, L.tl_xe2)) return; var cur = ebi('treeul').getAttribute('ts'); @@ -4042,7 +4615,7 @@ var treectl = (function () { } function recvls() { - if (!xhrchk(this, "could not list files in folder:\n\nerror ", "404, folder not found")) + if (!xhrchk(this, L.fl_xe1, L.fl_xe2)) return; var cur = ebi('files').getAttribute('ts'); @@ -4282,14 +4855,13 @@ function apply_perms(newperms) { perms = newperms || []; var a = QS('#ops a[data-dest="up2k"]'); - var suf = 'multithreaded, and file timestamps are preserved, but it uses more CPU than the basic uploader'; if (have_up2k_idx) { a.removeAttribute('data-perm'); - a.setAttribute('tt', '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, ' + suf); + a.setAttribute('tt', L.ot_u2i); } else { a.setAttribute('data-perm', 'write'); - a.setAttribute('tt', 'up2k: upload files with resume support (close your browser and drop the same files in later)$N$N' + suf); + a.setAttribute('tt', L.ot_u2w); } a.style.display = ''; tt.att(QS('#ops')); @@ -4412,25 +4984,7 @@ function mk_files_header(taglist) { var filecols = (function () { - var hidden = jread('filecols', []), - tts = { - "c": "action buttons", - "dur": "duration", - "q": "quality / bitrate", - "Ac": "audio codec", - "Vc": "video codec", - "Ahash": "audio checksum", - "Vhash": "video checksum", - "Res": "resolution", - "T": "filetype", - "aq": "audio quality / bitrate", - "vq": "video quality / bitrate", - "pixfmt": "subsampling / pixel structure", - "resw": "horizontal resolution", - "resh": "veritcal resolution", - "chs": "audio channels", - "hz": "sample rate" - }; + var hidden = jread('filecols', []); if (JSON.stringify(def_hcols) != sread('hfilecols')) { console.log("applying default hidden-cols"); @@ -4451,7 +5005,7 @@ var filecols = (function () { var ths = QSA('#files th>span'); for (var a = 0, aa = ths.length; a < aa; a++) { var th = ths[a].parentElement, - ttv = tts[ths[a].textContent]; + ttv = L.cols[ths[a].textContent]; th.innerHTML = '' + ths[a].outerHTML; th.getElementsByTagName('a')[0].onclick = ev_row_tgl; @@ -4482,7 +5036,7 @@ var filecols = (function () { hcols = ebi('hcols'); for (var a = 0; a < hidden.length; a++) { - var ttv = tts[hidden[a]], + var ttv = L.cols[hidden[a]], tta = ttv ? ' tt="' + ttv + '">' : '>'; html.push(''); @@ -4719,16 +5273,43 @@ var settheme = (function () { })(); +(function () { + function freshen() { + lang = sread("lang") || lang; + var html = []; + for (var k in Ls) + if (Ls.hasOwnProperty(k)) + html.push('' + k + ''); + + 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); + L = Ls[this.textContent]; + swrite("lang", this.textContent); + freshen(); + modal.confirm(L.lang_set, location.reload.bind(location), null); + }; + + freshen(); +})(); + + var arcfmt = (function () { if (!ebi('arc_fmt')) return { "render": function () { } }; var html = [], fmts = [ - ["tar", "tar", "plain gnutar file"], - ["zip", "zip=utf8", "zip with utf8 filenames (maybe wonky on windows 7 and older)"], - ["zip_dos", "zip", "zip with traditional cp437 filenames, for really old software"], - ["zip_crc", "zip=crc", "cp437 with crc32 computed early,$Nfor MS-DOS PKZIP v2.04g (october 1993)$N(takes longer to process before download can start)"] + ["tar", "tar", L.fz_tar], + ["zip", "zip=utf8", L.fz_zip8], + ["zip_dos", "zip", L.fz_zipd], + ["zip_crc", "zip=crc", L.fz_zipc] ]; for (var a = 0; a < fmts.length; a++) { @@ -4956,7 +5537,7 @@ var msel = (function () { return; } - xhrchk(this, "could not create subfolder:\n\nerror ", "404, parent folder not found"); + xhrchk(this, L.fd_xe1, L.fd_xe2); if (this.status !== 200) { sf.textContent = 'error: ' + this.responseText; @@ -5003,7 +5584,7 @@ var msel = (function () { }; function cb() { - xhrchk(this, "could not send message:\n\nerror ", "404, parent folder not found"); + xhrchk(this, L.fsm_xe1, L.fsm_xe2); if (this.status !== 200) { sf.textContent = 'error: ' + this.responseText; @@ -5021,7 +5602,7 @@ var msel = (function () { function show_md(md, name, div, url, depth) { - var errmsg = 'cannot show ' + name + ':\n\n', + var errmsg = L.md_eshow + name + ':\n\n', now = get_evpath(); url = url || now; @@ -5102,8 +5683,8 @@ function ev_row_tgl(e) { var unpost = (function () { ebi('op_unpost').innerHTML = ( - 'you can delete your recent uploads below – refresh list' + - '

    optional filter:  URL must contain clear filter

    ' + + L.un_m1 + ' – ' + L.un_upd + '' + + '

    ' + L.un_flt + ' ' + L.un_fclr + '

    ' + '
    ' ); @@ -5119,21 +5700,21 @@ var unpost = (function () { html = []; function unpost_load_cb() { - if (!xhrchk(this, "unpost-load failed:\n\nerror ", "404, file not found??")) - return ebi('op_unpost').innerHTML = 'failed to load unpost list from server'; + if (!xhrchk(this, L.fu_xe1, L.fu_xe2)) + return ebi('op_unpost').innerHTML = L.fu_xe1; var res = JSON.parse(this.responseText); if (res.length) { if (res.length == 2000) - html.push("

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

    " + L.un_max); else - html.push("

    " + res.length + " uploads can be deleted"); + html.push("

    " + L.un_avail.format(res.length)); - html.push(" – sorted by upload time – most recent first:

    "); + html.push(" – " + L.un_m2 + "

    "); html.push("
    ' + esc(mkeys[a]) + '
    -[❌] close search results -- showing ' + - res.hits.length + ' hits' + (res.hits.length == cap ? ' -- load more' : '') + '
    -[❌] ' + L.sl_close + ' -- ' + L.sl_hits.format(res.hits.length) + (res.hits.length == cap ? ' -- ' + L.sl_moar + '' : '') + '
    "); } else - html.push("sike! no uploads " + (filt.value ? 'matching that filter' : '') + " are sufficiently recent"); + html.push(filt.value ? L.un_no2 : L.un_no1); var mods = [1000, 100, 10]; for (var a = 0; a < res.length; a++) { @@ -5142,9 +5723,9 @@ var unpost = (function () { html.push( ''); + '" href="#">' + L.un_next.format(Math.min(mods[b], res.length - a)) + ''); html.push( - '' + + '' + '' + '' + ''); @@ -5165,13 +5746,13 @@ var unpost = (function () { xhr.onload = xhr.onerror = unpost_load_cb; xhr.send(); - ct.innerHTML = "

    loading your recent uploads...

    "; + ct.innerHTML = "

    " + L.un_m3 + "

    "; }; function unpost_delete_cb() { if (this.status !== 200) { var msg = this.responseText; - toast.err(9, 'unpost-delete failed:\n' + msg); + toast.err(9, L.un_derr + msg); return; } @@ -5199,7 +5780,7 @@ var unpost = (function () { ev(e); var ame = tgt.getAttribute('me'); if (ame != r.me) - return toast.err(0, 'something broke, please try a refresh'); + return toast.err(0, L.un_f5); var n = parseInt(tgt.className.slice(1)), n2 = parseInt(tgt.getAttribute('n2') || n + 1), @@ -5215,7 +5796,7 @@ var unpost = (function () { links[a].innerHTML = '[busy]'; } - toast.inf(0, "deleting " + req.length + " files..."); + toast.show('inf r', 0, L.un_busy.format(req.length)); var xhr = new XHR(); xhr.n = n; diff --git a/copyparty/web/ui.css b/copyparty/web/ui.css index c025d589..176878d5 100644 --- a/copyparty/web/ui.css +++ b/copyparty/web/ui.css @@ -75,6 +75,9 @@ html { margin-right: -1.2em; padding-right: .7em; } +#toast.r #toastb { + text-align: right; +} #toast pre { margin: 0; } diff --git a/copyparty/web/up2k.js b/copyparty/web/up2k.js index 31851a79..7d6814b9 100644 --- a/copyparty/web/up2k.js +++ b/copyparty/web/up2k.js @@ -592,14 +592,7 @@ function up2k_init(subtle) { ebi('u2notbtn').innerHTML = ''; } - var suggest_up2k = 'this is the basic uploader; up2k is better'; - - var shame = 'your browser disables sha512 unless you use https', - is_https = (window.location + '').indexOf('https:') === 0; - - if (is_https) - // chrome<37 firefox<34 edge<12 opera<24 safari<7 - shame = 'your browser is impressively ancient'; + var suggest_up2k = L.u_su2k; function got_deps() { return subtle || window.asmCrypto || window.hashwasm; @@ -608,15 +601,18 @@ function up2k_init(subtle) { var loading_deps = false; function init_deps() { if (!loading_deps && !got_deps()) { - var fn = 'sha512.' + sha_js + '.js'; - showmodal('

    loading ' + fn + '

    since ' + shame + '

    thanks chrome

    '); + var fn = 'sha512.' + sha_js + '.js', + m = L.u_https1 + ' ' + L.u_https2 + ' ' + L.u_https3; + + showmodal('

    loading ' + fn + '

    '); import_js('/.cpr/deps/' + fn, unmodal); - if (is_https) - ebi('u2foot').innerHTML = shame + ' so this uploader will do like 500 KiB/s at best'; - else - ebi('u2foot').innerHTML = 'seems like ' + shame + ' so do that if you want more performance (expecting 20' : '8a5">(but dont worry too much, expect 100') + ' MiB/s)'; + if (is_https) { + // chrome<37 firefox<34 edge<12 opera<24 safari<7 + m = L.u_ancient; + setmsg(''); + } + ebi('u2foot').innerHTML = '' + m + ''; } loading_deps = true; } @@ -648,16 +644,6 @@ function up2k_init(subtle) { setmsg(suggest_up2k, 'msg'); - if (!String.prototype.format) { - String.prototype.format = function () { - var args = arguments; - return this.replace(/{(\d+)}/g, function (match, number) { - return typeof args[number] != 'undefined' ? - args[number] : match; - }); - }; - } - var parallel_uploads = icfg_get('nthread'), uc = {}, fdom_ctr = 0, @@ -715,7 +701,7 @@ function up2k_init(subtle) { bobslice = File.prototype.slice || File.prototype.mozSlice || File.prototype.webkitSlice; if (!bobslice || !window.FileReader || !window.FileList) - return un2k("this is the basic uploader; up2k needs at least
    chrome 21 // firefox 13 // edge 12 // opera 12 // safari 5.1"); + return un2k(L.u_ever); var flag = false; apply_flag_cfg(); @@ -737,14 +723,14 @@ function up2k_init(subtle) { var mup, up = QS('#up_zd'); var msr, sr = QS('#srch_zd'); if (!has(perms, 'write')) - mup = 'you do not have write-access to this folder'; + mup = L.u_ewrite; if (!has(perms, 'read')) - msr = 'you do not have read-access to this folder'; + msr = L.u_eread; if (!have_up2k_idx) - msr = 'file-search is not enabled in server config'; + msr = L.u_enoi; - up.querySelector('span').textContent = mup || 'drop it here'; - sr.querySelector('span').textContent = msr || 'drop it here'; + up.querySelector('span').textContent = mup || L.udt_drop; + sr.querySelector('span').textContent = msr || L.udt_drop; clmod(up, 'err', mup); clmod(sr, 'err', msr); clmod(up, 'ok', !mup); @@ -970,22 +956,22 @@ function up2k_init(subtle) { function gotallfiles(good_files, nil_files, bad_files) { var ntot = good_files.concat(nil_files, bad_files).length; if (bad_files.length) { - var msg = 'These {0} files (of {1} total) were skipped, possibly due to filesystem permissions:\n'.format(bad_files.length, ntot); + var msg = L.u_badf.format(bad_files.length, ntot); for (var a = 0, aa = Math.min(20, bad_files.length); a < aa; a++) msg += '-- ' + bad_files[a][1] + '\n'; - msg += '\nMaybe it works better if you select just one file'; + msg += L.u_just1; return modal.alert(msg, function () { gotallfiles(good_files, nil_files, []); }); } if (nil_files.length) { - var msg = 'These {0} files (of {1} total) are blank/empty; upload them anyways?\n'.format(nil_files.length, ntot); + var msg = L.u_blankf.format(nil_files.length, ntot); for (var a = 0, aa = Math.min(20, nil_files.length); a < aa; a++) msg += '-- ' + nil_files[a][1] + '\n'; - msg += '\nMaybe it works better if you select just one file'; + msg += L.u_just1; return modal.confirm(msg, function () { gotallfiles(good_files.concat(nil_files), [], []); }, function () { @@ -999,7 +985,7 @@ function up2k_init(subtle) { return a < b ? -1 : a > b ? 1 : 0; }); - var msg = ['{0} these {1} files?
      '.format(uc.fsearch ? 'search' : 'upload', good_files.length)]; + var msg = [(uc.fsearch ? L.u_asks : L.u_asku).format(good_files.length) + '
        ']; for (var a = 0, aa = Math.min(20, good_files.length); a < aa; a++) msg.push('
      • ' + esc(good_files[a][1]) + '
      • '); @@ -1056,7 +1042,7 @@ function up2k_init(subtle) { pvis.addfile([ uc.fsearch ? esc(entry.name) : linksplit( entry.purl + uricom_enc(entry.name)).join(' '), - '📐 hash', + '📐 ' + L.u_hashing, '' ], fobj.size, draw_each); @@ -1104,11 +1090,11 @@ function up2k_init(subtle) { } if (!nhash) - ebi('u2etah').innerHTML = 'Done ({0}, {1} files)'.format(humansize(st.bytes.hashed), pvis.ctr["ok"] + pvis.ctr["ng"]); + ebi('u2etah').innerHTML = L.u_etadone.format(humansize(st.bytes.hashed), pvis.ctr["ok"] + pvis.ctr["ng"]); if (!nsend && !nhash) ebi('u2etau').innerHTML = ebi('u2etat').innerHTML = ( - 'Done ({0}, {1} files)'.format(humansize(st.bytes.uploaded), pvis.ctr["ok"] + pvis.ctr["ng"])); + L.u_etadone.format(humansize(st.bytes.uploaded), pvis.ctr["ok"] + pvis.ctr["ng"])); if (!st.busy.hash.length && !hashing_permitted()) nhash = 0; @@ -1129,7 +1115,7 @@ function up2k_init(subtle) { } if ((nhash || nsend) && !uc.fsearch) { if (!st.bytes.finished) { - ebi('u2etat').innerHTML = '(preparing to upload)'; + ebi('u2etat').innerHTML = L.u_etaprep; } else { st.time.busy += td; @@ -1142,7 +1128,7 @@ function up2k_init(subtle) { eta = Math.floor(rem / bps); if (t[a][1] < 1024 || t[a][3] < 0.1) { - ebi(t[a][0]).innerHTML = '(preparing to upload)'; + ebi(t[a][0]).innerHTML = L.u_etaprep; continue; } @@ -1530,7 +1516,7 @@ function up2k_init(subtle) { t.t_hashed = Date.now(); - pvis.seth(t.n, 2, 'hashing done'); + pvis.seth(t.n, 2, L.u_hashdone); pvis.seth(t.n, 1, '📦 wait'); apop(st.busy.hash, t); st.todo.handshake.push(t); @@ -1734,7 +1720,7 @@ function up2k_init(subtle) { 'npart': t.postlist[a] }); - msg = 'uploading'; + msg = L.u_upping; done = false; if (sort) @@ -1811,8 +1797,8 @@ function up2k_init(subtle) { tasker(); return; } - err = t.t_uploading ? "finalize upload" : t.srch ? "perform search" : "initiate upload"; - xhrchk(xhr, "server rejected the request to " + err + ";\n\nfile: " + t.name + "\n\nerror ", "404, target folder not found"); + err = t.t_uploading ? L.u_ehsfin : t.srch ? L.u_ehssrch : L.u_ehsinit; + xhrchk(xhr, err + ";\n\nfile: " + t.name + "\n\nerror ", "404, target folder not found"); } } xhr.onload = function (e) { @@ -1871,7 +1857,7 @@ function up2k_init(subtle) { console.log("ignoring dupe-segment error", t); } else { - xhrchk(xhr, "server rejected upload (chunk {0} of {1});\n\nfile: {2}\n\nerror ".format(npart, Math.ceil(t.size / chunksize), t.name), "404, target folder not found (???)"); + xhrchk(xhr, L.u_cuerr2.format(npart, Math.ceil(t.size / chunksize), t.name), "404, target folder not found (???)"); chill(t); } @@ -1900,7 +1886,7 @@ function up2k_init(subtle) { return; if (!toast.visible) - toast.warn(9.98, "failed to upload chunk {0} of {1};\nprobably harmless, continuing\n\nfile: {2}".format(npart, Math.ceil(t.size / chunksize), t.name)); + toast.warn(9.98, L.u_cuerr.format(npart, Math.ceil(t.size / chunksize), t.name)); console.log('chunkpit onerror,', ++tries, t); orz2(xhr); @@ -2062,7 +2048,7 @@ function up2k_init(subtle) { try { var ico = uc.fsearch ? '🔎' : '🚀', - desc = uc.fsearch ? 'S E A R C H' : 'U P L O A D'; + desc = uc.fsearch ? L.ul_btns : L.ul_btnu; clmod(ebi('op_up2k'), 'srch', uc.fsearch); ebi('u2bm').innerHTML = ico + '  ' + desc + ''; diff --git a/copyparty/web/util.js b/copyparty/web/util.js index 5d4e616f..a31f1c93 100644 --- a/copyparty/web/util.js +++ b/copyparty/web/util.js @@ -7,6 +7,7 @@ if (!window['console']) var is_touch = 'ontouchstart' in window, + is_https = (window.location + '').indexOf('https:') === 0, IPHONE = is_touch && /iPhone|iPad|iPod/i.test(navigator.userAgent), WINDOWS = navigator.platform ? navigator.platform == 'Win32' : /Windows/.test(navigator.userAgent); @@ -253,6 +254,14 @@ if (!Element.prototype.closest) } while (el !== null && el.nodeType === 1); }; +if (!String.prototype.format) + String.prototype.format = function () { + var args = arguments; + return this.replace(/{(\d+)}/g, function (match, number) { + return typeof args[number] != 'undefined' ? + args[number] : match; + }); + }; // https://stackoverflow.com/a/950146 function import_js(url, cb) { @@ -1397,7 +1406,7 @@ function xhrchk(xhr, prefix, e404) { return true; if (xhr.status == 403) - return toast.err(0, prefix + "403, access denied\n\ntry pressing F5, maybe you got logged out"); + return toast.err(0, prefix + (window.L && L.xhr403 || "403: access denied\n\ntry pressing F5, maybe you got logged out")); if (xhr.status == 404) return toast.err(0, prefix + e404); diff --git a/scripts/make-sfx.sh b/scripts/make-sfx.sh index fc87bba7..465bfbd6 100755 --- a/scripts/make-sfx.sh +++ b/scripts/make-sfx.sh @@ -14,6 +14,9 @@ help() { exec cat <<'EOF' # # `gz` creates a gzip-compressed python sfx instead of bzip2 # +# `lang` limits which languages/translations to include, +# for example `lang eng` or `lang eng|nor` +# # `no-cm` saves ~82k by removing easymde/codemirror # (the fancy markdown editor) # @@ -61,6 +64,7 @@ pybin=$(command -v python3 || command -v python) || { exit 1 } +langs= use_gz= zopf=2560 while [ ! -z "$1" ]; do @@ -73,6 +77,7 @@ while [ ! -z "$1" ]; do no-dd) no_dd=1 ; ;; no-cm) no_cm=1 ; ;; fast) zopf=100 ; ;; + lang) shift;langs="$1"; ;; *) help ; ;; esac shift @@ -262,6 +267,13 @@ rm have tmv "$f" } +[ $langs ] && { + f=copyparty/web/browser.js + gzip -d "$f.gz" || true + awk '/^\}/{l=0} !l; /^var Ls =/{l=1;next} o; /^\t["}]/{o=0} /^\t"'"$langs"'"/{o=1;print}' <$f >t + tmv "$f" +} + [ $repack ] || find | grep -E '\.py$' | grep -vE '__version__' |
    timesizefile
    ' + 'delete the next ' + Math.min(mods[b], res.length - a) + ' files below
    delete
    ' + L.un_del + '' + unix2iso(res[a].at) + '' + res[a].sz + '' + linksplit(res[a].vp).join(' ') + '