Merge branch 'master' into bugfix/refresh_disk_usage
						commit
						c22a724b11
					
				|  | @ -2,13 +2,18 @@ FROM ghcr.io/linuxserver/baseimage-alpine:3.20 | |||
| 
 | ||||
| RUN apk --update add ca-certificates \ | ||||
|                      mailcap \ | ||||
|                      curl | ||||
|                      curl \ | ||||
|                      jq | ||||
| 
 | ||||
| COPY healthcheck.sh /healthcheck.sh | ||||
| RUN chmod +x /healthcheck.sh  # Make the script executable | ||||
| 
 | ||||
| HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \ | ||||
|   CMD curl -f http://localhost/health || exit 1 | ||||
|     CMD /healthcheck.sh || exit 1 | ||||
| 
 | ||||
| # copy local files | ||||
| COPY docker/root/ / | ||||
| RUN ln -s /config/settings.json /.filebrowser.json | ||||
| COPY filebrowser /usr/bin/filebrowser | ||||
| 
 | ||||
| # ports and volumes | ||||
|  |  | |||
|  | @ -2,13 +2,18 @@ FROM ghcr.io/linuxserver/baseimage-alpine:arm64v8-3.20 | |||
| 
 | ||||
| RUN apk --update add ca-certificates \ | ||||
|                      mailcap \ | ||||
|                      curl | ||||
|                      curl \ | ||||
|                      jq | ||||
| 
 | ||||
| COPY healthcheck.sh /healthcheck.sh | ||||
| RUN chmod +x /healthcheck.sh  # Make the script executable | ||||
| 
 | ||||
| HEALTHCHECK --start-period=2s --interval=5s --timeout=3s \ | ||||
|   CMD curl -f http://localhost/health || exit 1 | ||||
|     CMD /healthcheck.sh || exit 1 | ||||
| 
 | ||||
| # copy local files | ||||
| COPY docker/root/ / | ||||
| RUN ln -s /config/settings.json /.filebrowser.json | ||||
| COPY filebrowser /usr/bin/filebrowser | ||||
| 
 | ||||
| # ports and volumes | ||||
|  |  | |||
|  | @ -1,2 +1,3 @@ | |||
| # Ignore artifacts: | ||||
| dist | ||||
| pnpm-lock.yaml | ||||
|  | @ -19,14 +19,13 @@ export default [ | |||
| 
 | ||||
|   { | ||||
|     rules: { | ||||
|       "no-var": "warn", | ||||
|       "prefer-const": "warn", | ||||
|       // Note: you must disable the base rule as it can report incorrect errors
 | ||||
|       "no-unused-expressions": "off", | ||||
|       "@typescript-eslint/no-unused-expressions": "warn", | ||||
|       "@typescript-eslint/no-explicit-any": "warn", | ||||
|       "@typescript-eslint/ban-ts-comment": "warn", | ||||
|       "vue/block-lang": "warn", | ||||
|       "@typescript-eslint/no-unused-expressions": "off", | ||||
|       // TODO: theres too many of these from before ts
 | ||||
|       "@typescript-eslint/no-explicit-any": "off", | ||||
|       // TODO: finish the ts conversion
 | ||||
|       "vue/block-lang": "off", | ||||
|       "vue/multi-word-component-names": "off", | ||||
|       "vue/no-mutating-props": [ | ||||
|         "error", | ||||
|  |  | |||
|  | @ -19,64 +19,59 @@ | |||
|   }, | ||||
|   "dependencies": { | ||||
|     "@chenfengyuan/vue-number-input": "^2.0.1", | ||||
|     "@vueuse/core": "^12.0.0", | ||||
|     "@vueuse/integrations": "^12.0.0", | ||||
|     "ace-builds": "^1.32.9", | ||||
|     "core-js": "^3.36.1", | ||||
|     "@vueuse/core": "^12.5.0", | ||||
|     "@vueuse/integrations": "^12.5.0", | ||||
|     "ace-builds": "^1.37.5", | ||||
|     "core-js": "^3.40.0", | ||||
|     "dayjs": "^1.11.10", | ||||
|     "epubjs": "^0.3.93", | ||||
|     "filesize": "^10.1.1", | ||||
|     "js-base64": "^3.7.7", | ||||
|     "jwt-decode": "^4.0.0", | ||||
|     "lodash-es": "^4.17.21", | ||||
|     "marked": "^15.0.3", | ||||
|     "material-icons": "^1.13.12", | ||||
|     "marked": "^15.0.6", | ||||
|     "material-icons": "^1.13.13", | ||||
|     "normalize.css": "^8.0.1", | ||||
|     "pinia": "^2.1.7", | ||||
|     "pinia": "^2.3.1", | ||||
|     "pretty-bytes": "^6.1.1", | ||||
|     "qrcode.vue": "^3.4.1", | ||||
|     "tus-js-client": "^4.1.0", | ||||
|     "tus-js-client": "^4.3.1", | ||||
|     "utif": "^3.1.0", | ||||
|     "video.js": "^8.10.0", | ||||
|     "video.js": "^8.21.0", | ||||
|     "videojs-hotkeys": "^0.2.28", | ||||
|     "videojs-mobile-ui": "^1.1.1", | ||||
|     "vue": "^3.4.21", | ||||
|     "vue-final-modal": "^4.5.4", | ||||
|     "vue-i18n": "^10.0.5", | ||||
|     "vue-i18n": "^11.0.1", | ||||
|     "vue-lazyload": "^3.0.0", | ||||
|     "vue-reader": "^1.2.14", | ||||
|     "vue-reader": "^1.2.17", | ||||
|     "vue-router": "^4.3.0", | ||||
|     "vue-toastification": "^2.0.0-rc.5" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@intlify/unplugin-vue-i18n": "^6.0.0", | ||||
|     "@playwright/test": "^1.42.1", | ||||
|     "@intlify/unplugin-vue-i18n": "^6.0.3", | ||||
|     "@playwright/test": "^1.50.0", | ||||
|     "@tsconfig/node22": "^22.0.0", | ||||
|     "@types/lodash-es": "^4.17.12", | ||||
|     "@types/node": "^22.10.1", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.17.0", | ||||
|     "@types/node": "^22.10.10", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.21.0", | ||||
|     "@vitejs/plugin-legacy": "^6.0.0", | ||||
|     "@vitejs/plugin-vue": "^5.0.4", | ||||
|     "@vue/eslint-config-prettier": "^10.1.0", | ||||
|     "@vue/eslint-config-typescript": "^14.1.4", | ||||
|     "@vue/eslint-config-prettier": "^10.2.0", | ||||
|     "@vue/eslint-config-typescript": "^14.3.0", | ||||
|     "@vue/tsconfig": "^0.7.0", | ||||
|     "autoprefixer": "^10.4.19", | ||||
|     "concurrently": "^9.1.0", | ||||
|     "eslint": "^9.16.0", | ||||
|     "eslint-plugin-prettier": "^5.1.3", | ||||
|     "concurrently": "^9.1.2", | ||||
|     "eslint": "^9.19.0", | ||||
|     "eslint-plugin-prettier": "^5.2.3", | ||||
|     "eslint-plugin-vue": "^9.24.0", | ||||
|     "jsdom": "^25.0.1", | ||||
|     "postcss": "^8.4.38", | ||||
|     "prettier": "^3.2.5", | ||||
|     "terser": "^5.30.0", | ||||
|     "vite": "^6.0.2", | ||||
|     "jsdom": "^26.0.0", | ||||
|     "postcss": "^8.5.1", | ||||
|     "prettier": "^3.4.2", | ||||
|     "terser": "^5.37.0", | ||||
|     "vite": "^6.0.11", | ||||
|     "vite-plugin-compression2": "^1.0.0", | ||||
|     "vue-tsc": "^2.0.7" | ||||
|     "vue-tsc": "^2.2.0" | ||||
|   }, | ||||
|   "pnpm": { | ||||
|     "overrides": { | ||||
|       "typescript": "~5.6.3" | ||||
|     } | ||||
|   }, | ||||
|   "packageManager": "pnpm@9.14.4+sha512.c8180b3fbe4e4bca02c94234717896b5529740a6cbadf19fa78254270403ea2f27d4e1d46a08a0f56c89b63dc8ebfd3ee53326da720273794e6200fcf0d184ab" | ||||
|   "packageManager": "pnpm@9.15.4+sha512.b2dc20e2fc72b3e18848459b37359a32064663e5627a51e4c74b2c29dd8e8e0491483c3abb40789cfd578bf362fb6ba8261b05f0387d76792ed6e23ea3b1b6a0" | ||||
| } | ||||
|  |  | |||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -25,7 +25,7 @@ export async function create(user: IUser) { | |||
|   throw new StatusError(await res.text(), res.status); | ||||
| } | ||||
| 
 | ||||
| export async function update(user: IUser, which = ["all"]) { | ||||
| export async function update(user: Partial<IUser>, which = ["all"]) { | ||||
|   await fetchURL(`/api/users/${user.id}`, { | ||||
|     method: "PUT", | ||||
|     body: JSON.stringify({ | ||||
|  |  | |||
|  | @ -34,7 +34,7 @@ const props = defineProps<{ | |||
| 
 | ||||
| const items = computed(() => { | ||||
|   const relativePath = route.path.replace(props.base, ""); | ||||
|   let parts = relativePath.split("/"); | ||||
|   const parts = relativePath.split("/"); | ||||
| 
 | ||||
|   if (parts[0] === "") { | ||||
|     parts.shift(); | ||||
|  | @ -44,7 +44,7 @@ const items = computed(() => { | |||
|     parts.pop(); | ||||
|   } | ||||
| 
 | ||||
|   let breadcrumbs: BreadCrumb[] = []; | ||||
|   const breadcrumbs: BreadCrumb[] = []; | ||||
| 
 | ||||
|   for (let i = 0; i < parts.length; i++) { | ||||
|     if (i === 0) { | ||||
|  |  | |||
|  | @ -46,7 +46,7 @@ https://raw.githubusercontent.com/dzwillia/vue-simple-progress/master/src/compon | |||
| 
 | ||||
| <script> | ||||
| // We're leaving this untouched as you can read in the beginning | ||||
| var isNumber = function (n) { | ||||
| const isNumber = function (n) { | ||||
|   return !isNaN(parseFloat(n)) && isFinite(n); | ||||
| }; | ||||
| 
 | ||||
|  | @ -107,7 +107,7 @@ export default { | |||
|   }, | ||||
|   computed: { | ||||
|     pct() { | ||||
|       var pct = (this.val / this.max) * 100; | ||||
|       let pct = (this.val / this.max) * 100; | ||||
|       pct = pct.toFixed(2); | ||||
|       return Math.min(pct, this.max); | ||||
|     }, | ||||
|  | @ -160,7 +160,7 @@ export default { | |||
|       return isNumber(this.fontSize) ? this.fontSize : 13; | ||||
|     }, | ||||
|     progress_style() { | ||||
|       var style = { | ||||
|       const style = { | ||||
|         background: this.bgColor, | ||||
|       }; | ||||
| 
 | ||||
|  | @ -177,7 +177,7 @@ export default { | |||
|       return style; | ||||
|     }, | ||||
|     bar_style() { | ||||
|       var style = { | ||||
|       const style = { | ||||
|         background: this.barColor, | ||||
|         width: this.pct + "%", | ||||
|         height: this.size_px + "px", | ||||
|  | @ -198,7 +198,7 @@ export default { | |||
|       return style; | ||||
|     }, | ||||
|     text_style() { | ||||
|       var style = { | ||||
|       const style = { | ||||
|         color: this.textFgColor, | ||||
|         "font-size": this.text_font_size + "px", | ||||
|         "text-align": this.textAlign, | ||||
|  |  | |||
|  | @ -163,7 +163,7 @@ export default { | |||
|       this.canInput = false; | ||||
|       event.target.innerHTML = ""; | ||||
| 
 | ||||
|       let results = { | ||||
|       const results = { | ||||
|         text: `${cmd}\n\n`, | ||||
|       }; | ||||
| 
 | ||||
|  | @ -180,7 +180,7 @@ export default { | |||
|         }, | ||||
|         () => { | ||||
|           results.text = results.text | ||||
|             // eslint-disable-next-line no-control-regex | ||||
| 
 | ||||
|             .replace(/\u001b\[[0-9;]+m/g, "") // Filter ANSI color for now | ||||
|             .trimEnd(); | ||||
|           this.canInput = true; | ||||
|  |  | |||
|  | @ -158,7 +158,7 @@ export default { | |||
|   methods: { | ||||
|     ...mapActions(useLayoutStore, ["closeHovers", "showHover"]), | ||||
|     async fetchUsage() { | ||||
|       let path = this.$route.path.endsWith("/") | ||||
|       const path = this.$route.path.endsWith("/") | ||||
|         ? this.$route.path | ||||
|         : this.$route.path + "/"; | ||||
|       let usageStats = USAGE_DEFAULT; | ||||
|  | @ -166,7 +166,7 @@ export default { | |||
|         return Object.assign(this.usage, usageStats); | ||||
|       } | ||||
|       try { | ||||
|         let usage = await api.usage(path); | ||||
|         const usage = await api.usage(path); | ||||
|         usageStats = { | ||||
|           used: prettyBytes(usage.used, { binary: true }), | ||||
|           total: prettyBytes(usage.total, { binary: true }), | ||||
|  |  | |||
|  | @ -102,10 +102,11 @@ const decodeUTIF = () => { | |||
|   if (document?.location?.pathname === undefined) { | ||||
|     return; | ||||
|   } | ||||
|   let suff = document.location.pathname.split(".")?.pop()?.toLowerCase() ?? ""; | ||||
|   const suff = | ||||
|     document.location.pathname.split(".")?.pop()?.toLowerCase() ?? ""; | ||||
| 
 | ||||
|   if (sufs.indexOf(suff) == -1) return false; | ||||
|   let xhr = new XMLHttpRequest(); | ||||
|   const xhr = new XMLHttpRequest(); | ||||
|   UTIF._xhrs.push(xhr); | ||||
|   UTIF._imgs.push(imgex.value); | ||||
|   xhr.open("GET", props.src); | ||||
|  | @ -230,7 +231,7 @@ const touchMove = (event: TouchEvent) => { | |||
|   if (imgex.value === null) { | ||||
|     return; | ||||
|   } | ||||
|   let step = imgex.value.width / 5; | ||||
|   const step = imgex.value.width / 5; | ||||
|   if (event.targetTouches.length === 2) { | ||||
|     moveDisabled.value = true; | ||||
|     if (disabledTimer.value) clearTimeout(disabledTimer.value); | ||||
|  | @ -239,9 +240,9 @@ const touchMove = (event: TouchEvent) => { | |||
|       props.moveDisabledTime | ||||
|     ); | ||||
| 
 | ||||
|     let p1 = event.targetTouches[0]; | ||||
|     let p2 = event.targetTouches[1]; | ||||
|     let touchDistance = Math.sqrt( | ||||
|     const p1 = event.targetTouches[0]; | ||||
|     const p2 = event.targetTouches[1]; | ||||
|     const touchDistance = Math.sqrt( | ||||
|       Math.pow(p2.pageX - p1.pageX, 2) + Math.pow(p2.pageY - p1.pageY, 2) | ||||
|     ); | ||||
|     if (!lastTouchDistance.value) { | ||||
|  | @ -253,8 +254,8 @@ const touchMove = (event: TouchEvent) => { | |||
|     setZoom(); | ||||
|   } else if (event.targetTouches.length === 1) { | ||||
|     if (moveDisabled.value) return; | ||||
|     let x = event.targetTouches[0].pageX - (lastX.value ?? 0); | ||||
|     let y = event.targetTouches[0].pageY - (lastY.value ?? 0); | ||||
|     const x = event.targetTouches[0].pageX - (lastX.value ?? 0); | ||||
|     const y = event.targetTouches[0].pageY - (lastY.value ?? 0); | ||||
|     if (Math.abs(x) >= step && Math.abs(y) >= step) return; | ||||
|     lastX.value = event.targetTouches[0].pageX; | ||||
|     lastY.value = event.targetTouches[0].pageY; | ||||
|  | @ -268,8 +269,8 @@ const doMove = (x: number, y: number) => { | |||
|   } | ||||
|   const style = imgex.value.style; | ||||
| 
 | ||||
|   let posX = pxStringToNumber(style.left) + x; | ||||
|   let posY = pxStringToNumber(style.top) + y; | ||||
|   const posX = pxStringToNumber(style.left) + x; | ||||
|   const posY = pxStringToNumber(style.top) + y; | ||||
| 
 | ||||
|   style.left = posX + "px"; | ||||
|   style.top = posY + "px"; | ||||
|  |  | |||
|  | @ -82,7 +82,7 @@ const isDraggable = computed( | |||
| const canDrop = computed(() => { | ||||
|   if (!props.isDir || props.readOnly) return false; | ||||
| 
 | ||||
|   for (let i of fileStore.selected) { | ||||
|   for (const i of fileStore.selected) { | ||||
|     if (fileStore.req?.items[i].url === props.url) { | ||||
|       return false; | ||||
|     } | ||||
|  | @ -156,9 +156,9 @@ const drop = async (event: Event) => { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let items: any[] = []; | ||||
|   const items: any[] = []; | ||||
| 
 | ||||
|   for (let i of fileStore.selected) { | ||||
|   for (const i of fileStore.selected) { | ||||
|     if (fileStore.req) { | ||||
|       items.push({ | ||||
|         from: fileStore.req?.items[i].url, | ||||
|  | @ -172,10 +172,10 @@ const drop = async (event: Event) => { | |||
|   if (el === null) { | ||||
|     return; | ||||
|   } | ||||
|   let path = el.__vue__.url; | ||||
|   let baseItems = (await api.fetch(path)).items; | ||||
|   const path = el.__vue__.url; | ||||
|   const baseItems = (await api.fetch(path)).items; | ||||
| 
 | ||||
|   let action = (overwrite: boolean, rename: boolean) => { | ||||
|   const action = (overwrite: boolean, rename: boolean) => { | ||||
|     api | ||||
|       .move(items, overwrite, rename) | ||||
|       .then(() => { | ||||
|  | @ -184,7 +184,7 @@ const drop = async (event: Event) => { | |||
|       .catch($showError); | ||||
|   }; | ||||
| 
 | ||||
|   let conflict = upload.checkConflict(items, baseItems); | ||||
|   const conflict = upload.checkConflict(items, baseItems); | ||||
| 
 | ||||
|   let overwrite = false; | ||||
|   let rename = false; | ||||
|  |  | |||
|  | @ -73,11 +73,16 @@ const initVideoPlayer = async () => { | |||
|     const langOpt = { language: "videoPlayerLocal" }; | ||||
|     // support for playback at different speeds. | ||||
|     const playbackRatesOpt = { playbackRates: [0.5, 1, 1.5, 2, 2.5, 3] }; | ||||
|     let options = getOptions(props.options, langOpt, srcOpt, playbackRatesOpt); | ||||
|     const options = getOptions( | ||||
|       props.options, | ||||
|       langOpt, | ||||
|       srcOpt, | ||||
|       playbackRatesOpt | ||||
|     ); | ||||
|     player.value = videojs(videoPlayer.value!, options, () => {}); | ||||
| 
 | ||||
|     // TODO: need to test on mobile | ||||
|     // @ts-ignore | ||||
|     // @ts-expect-error no ts definition for mobileUi | ||||
|     player.value!.mobileUi(); | ||||
|   } catch (error) { | ||||
|     console.error("Error initializing video player:", error); | ||||
|  |  | |||
|  | @ -82,10 +82,10 @@ export default { | |||
|     ...mapActions(useLayoutStore, ["showHover", "closeHovers"]), | ||||
|     copy: async function (event) { | ||||
|       event.preventDefault(); | ||||
|       let items = []; | ||||
|       const items = []; | ||||
| 
 | ||||
|       // Create a new promise for each file. | ||||
|       for (let item of this.selected) { | ||||
|       for (const item of this.selected) { | ||||
|         items.push({ | ||||
|           from: this.req.items[item].url, | ||||
|           to: this.dest + encodeURIComponent(this.req.items[item].name), | ||||
|  | @ -93,7 +93,7 @@ export default { | |||
|         }); | ||||
|       } | ||||
| 
 | ||||
|       let action = async (overwrite, rename) => { | ||||
|       const action = async (overwrite, rename) => { | ||||
|         buttons.loading("copy"); | ||||
| 
 | ||||
|         await api | ||||
|  | @ -122,8 +122,8 @@ export default { | |||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       let dstItems = (await api.fetch(this.dest)).items; | ||||
|       let conflict = upload.checkConflict(items, dstItems); | ||||
|       const dstItems = (await api.fetch(this.dest)).items; | ||||
|       const conflict = upload.checkConflict(items, dstItems); | ||||
| 
 | ||||
|       let overwrite = false; | ||||
|       let rename = false; | ||||
|  |  | |||
|  | @ -74,8 +74,8 @@ export default { | |||
|           return; | ||||
|         } | ||||
| 
 | ||||
|         let promises = []; | ||||
|         for (let index of this.selected) { | ||||
|         const promises = []; | ||||
|         for (const index of this.selected) { | ||||
|           promises.push(api.remove(this.req.items[index].url)); | ||||
|         } | ||||
| 
 | ||||
|  |  | |||
|  | @ -43,7 +43,7 @@ export default { | |||
|     submit: async function () { | ||||
|       this.updateRequest(null); | ||||
| 
 | ||||
|       let uri = url.removeLastDir(this.$route.path) + "/"; | ||||
|       const uri = url.removeLastDir(this.$route.path) + "/"; | ||||
|       this.$router.push({ path: uri }); | ||||
|     }, | ||||
|   }, | ||||
|  |  | |||
|  | @ -80,7 +80,7 @@ export default { | |||
| 
 | ||||
|       // Otherwise we add every directory to the | ||||
|       // move options. | ||||
|       for (let item of req.items) { | ||||
|       for (const item of req.items) { | ||||
|         if (!item.isDir) continue; | ||||
| 
 | ||||
|         this.items.push({ | ||||
|  | @ -93,12 +93,12 @@ export default { | |||
|       // Retrieves the URL of the directory the user | ||||
|       // just clicked in and fill the options with its | ||||
|       // content. | ||||
|       let uri = event.currentTarget.dataset.url; | ||||
|       const uri = event.currentTarget.dataset.url; | ||||
| 
 | ||||
|       files.fetch(uri).then(this.fillOptions).catch(this.$showError); | ||||
|     }, | ||||
|     touchstart(event) { | ||||
|       let url = event.currentTarget.dataset.url; | ||||
|       const url = event.currentTarget.dataset.url; | ||||
| 
 | ||||
|       // In 300 milliseconds, we shall reset the count. | ||||
|       setTimeout(() => { | ||||
|  |  | |||
|  | @ -124,7 +124,7 @@ export default { | |||
| 
 | ||||
|       let sum = 0; | ||||
| 
 | ||||
|       for (let selected of this.selected) { | ||||
|       for (const selected of this.selected) { | ||||
|         sum += this.req.items[selected].size; | ||||
|       } | ||||
| 
 | ||||
|  |  | |||
|  | @ -81,9 +81,9 @@ export default { | |||
|     ...mapActions(useLayoutStore, ["showHover", "closeHovers"]), | ||||
|     move: async function (event) { | ||||
|       event.preventDefault(); | ||||
|       let items = []; | ||||
|       const items = []; | ||||
| 
 | ||||
|       for (let item of this.selected) { | ||||
|       for (const item of this.selected) { | ||||
|         items.push({ | ||||
|           from: this.req.items[item].url, | ||||
|           to: this.dest + encodeURIComponent(this.req.items[item].name), | ||||
|  | @ -91,7 +91,7 @@ export default { | |||
|         }); | ||||
|       } | ||||
| 
 | ||||
|       let action = async (overwrite, rename) => { | ||||
|       const action = async (overwrite, rename) => { | ||||
|         buttons.loading("move"); | ||||
| 
 | ||||
|         await api | ||||
|  | @ -106,8 +106,8 @@ export default { | |||
|           }); | ||||
|       }; | ||||
| 
 | ||||
|       let dstItems = (await api.fetch(this.dest)).items; | ||||
|       let conflict = upload.checkConflict(items, dstItems); | ||||
|       const dstItems = (await api.fetch(this.dest)).items; | ||||
|       const conflict = upload.checkConflict(items, dstItems); | ||||
| 
 | ||||
|       let overwrite = false; | ||||
|       let rename = false; | ||||
|  |  | |||
|  | @ -196,13 +196,23 @@ export default { | |||
|   methods: { | ||||
|     ...mapActions(useLayoutStore, ["closeHovers"]), | ||||
|     copyToClipboard: function (text) { | ||||
|       copy(text).then( | ||||
|       copy({ text }).then( | ||||
|         () => { | ||||
|           // clipboard successfully set | ||||
|           this.$showSuccess(this.$t("success.linkCopied")); | ||||
|         }, | ||||
|         () => { | ||||
|           // clipboard write failed | ||||
|           copy({ text }, { permission: true }).then( | ||||
|             () => { | ||||
|               // clipboard successfully set | ||||
|               this.$showSuccess(this.$t("success.linkCopied")); | ||||
|             }, | ||||
|             (e) => { | ||||
|               // clipboard write failed | ||||
|               this.$showError(e); | ||||
|             } | ||||
|           ); | ||||
|         } | ||||
|       ); | ||||
|     }, | ||||
|  |  | |||
|  | @ -48,10 +48,10 @@ const layoutStore = useLayoutStore(); | |||
| 
 | ||||
| // TODO: this is a copy of the same function in FileListing.vue | ||||
| const uploadInput = (event: Event) => { | ||||
|   let files = (event.currentTarget as HTMLInputElement)?.files; | ||||
|   const files = (event.currentTarget as HTMLInputElement)?.files; | ||||
|   if (files === null) return; | ||||
| 
 | ||||
|   let folder_upload = !!files[0].webkitRelativePath; | ||||
|   const folder_upload = !!files[0].webkitRelativePath; | ||||
| 
 | ||||
|   const uploadFiles: UploadList = []; | ||||
|   for (let i = 0; i < files.length; i++) { | ||||
|  | @ -66,8 +66,8 @@ const uploadInput = (event: Event) => { | |||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   let path = route.path.endsWith("/") ? route.path : route.path + "/"; | ||||
|   let conflict = upload.checkConflict(uploadFiles, fileStore.req!.items); | ||||
|   const path = route.path.endsWith("/") ? route.path : route.path + "/"; | ||||
|   const conflict = upload.checkConflict(uploadFiles, fileStore.req!.items); | ||||
| 
 | ||||
|   if (conflict) { | ||||
|     layoutStore.showHover({ | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ export default { | |||
|   name: "languages", | ||||
|   props: ["locale"], | ||||
|   data() { | ||||
|     let dataObj = {}; | ||||
|     const dataObj = {}; | ||||
|     const locales = { | ||||
|       he: "עברית", | ||||
|       hu: "Magyar", | ||||
|  |  | |||
|  | @ -39,7 +39,7 @@ export default { | |||
|   methods: { | ||||
|     remove(event, index) { | ||||
|       event.preventDefault(); | ||||
|       let rules = [...this.rules]; | ||||
|       const rules = [...this.rules]; | ||||
|       rules.splice(index, 1); | ||||
|       this.$emit("update:rules", [...rules]); | ||||
|     }, | ||||
|  |  | |||
|  | @ -17,7 +17,6 @@ defineProps<{ | |||
| }>(); | ||||
| 
 | ||||
| const emit = defineEmits<{ | ||||
|   // eslint-disable-next-line @typescript-eslint/no-unused-vars | ||||
|   (e: "update:theme", val: string | null): void; | ||||
| }>(); | ||||
| 
 | ||||
|  |  | |||
|  | @ -116,7 +116,7 @@ watch(createUserDirData, () => { | |||
|   if (props.user?.scope) { | ||||
|     props.user.scope = createUserDirData.value | ||||
|       ? "" | ||||
|       : originalUserScope.value ?? ""; | ||||
|       : (originalUserScope.value ?? ""); | ||||
|   } | ||||
| }); | ||||
| </script> | ||||
|  |  | |||
|  | @ -63,8 +63,8 @@ | |||
|     local("Roboto"), | ||||
|     local("Roboto-Regular"), | ||||
|     url(../assets/fonts/roboto/normal-latin-ext.woff2) format("woff2"); | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, | ||||
|     U+A720-A7FF; | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, | ||||
|     U+2C60-2C7F, U+A720-A7FF; | ||||
| } | ||||
| 
 | ||||
| @font-face { | ||||
|  | @ -142,8 +142,8 @@ | |||
|     local("Roboto Medium"), | ||||
|     local("Roboto-Medium"), | ||||
|     url(../assets/fonts/roboto/medium-latin-ext.woff2) format("woff2"); | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, | ||||
|     U+A720-A7FF; | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, | ||||
|     U+2C60-2C7F, U+A720-A7FF; | ||||
| } | ||||
| 
 | ||||
| @font-face { | ||||
|  | @ -221,8 +221,8 @@ | |||
|     local("Roboto Bold"), | ||||
|     local("Roboto-Bold"), | ||||
|     url(../assets/fonts/roboto/bold-latin-ext.woff2) format("woff2"); | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, U+2C60-2C7F, | ||||
|     U+A720-A7FF; | ||||
|   unicode-range: U+0100-024F, U+1E00-1EFF, U+20A0-20AB, U+20AD-20CF, | ||||
|     U+2C60-2C7F, U+A720-A7FF; | ||||
| } | ||||
| 
 | ||||
| @font-face { | ||||
|  |  | |||
|  | @ -173,7 +173,7 @@ | |||
|     "executeOnShellDescription": "Por defecto, FileBrowser ejecuta los comandos llamando directamente a sus binarios. Si quieres ejecutarlos en un shell en su lugar (como Bash o PowerShell), puedes definirlo aquí con los argumentos y banderas (flags) necesarios. Si se define, el comando que se ejecuta se añadirá como argumento. Esto se aplica tanto a los comandos de usuario como a los ganchos de eventos.", | ||||
|     "globalRules": "Se trata de un conjunto global de reglas de permiso y rechazo. Se aplican a todos los usuarios. Puedes definir reglas específicas en la configuración de cada usuario para anular estas.", | ||||
|     "globalSettings": "Ajustes globales", | ||||
|     "hideDotfiles": "", | ||||
|     "hideDotfiles": "Ocultar archivos empezados por punto", | ||||
|     "insertPath": "Introduce la ruta", | ||||
|     "insertRegex": "Introducir expresión regular", | ||||
|     "instanceName": "Nombre de la instancia", | ||||
|  |  | |||
|  | @ -142,7 +142,7 @@ export const i18n = createI18n({ | |||
| 
 | ||||
| export const isRtl = (locale?: string) => { | ||||
|   // see below
 | ||||
|   // @ts-ignore
 | ||||
|   // @ts-expect-error incorrect type when legacy
 | ||||
|   return rtlLanguages.includes(locale || i18n.global.locale.value); | ||||
| }; | ||||
| 
 | ||||
|  | @ -150,7 +150,7 @@ export function setLocale(locale: string) { | |||
|   dayjs.locale(locale); | ||||
|   // according to doc u only need .value if legacy: false but they lied
 | ||||
|   // https://vue-i18n.intlify.dev/guide/essentials/scope.html#local-scope-1
 | ||||
|   //@ts-ignore
 | ||||
|   // @ts-expect-error incorrect type when legacy
 | ||||
|   i18n.global.locale.value = locale; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -9,7 +9,7 @@ | |||
|     "create": "생성", | ||||
|     "delete": "삭제", | ||||
|     "download": "다운로드", | ||||
|     "hideDotfiles": "", | ||||
|     "hideDotfiles": "숨김파일(dotfile)을 표시 안함", | ||||
|     "info": "정보", | ||||
|     "more": "더보기", | ||||
|     "move": "이동", | ||||
|  | @ -38,7 +38,7 @@ | |||
|   "download": { | ||||
|     "downloadFile": "파일 다운로드", | ||||
|     "downloadFolder": "폴더 다운로드", | ||||
|     "downloadSelected": "" | ||||
|     "downloadSelected": "선택 항목 다운로드" | ||||
|   }, | ||||
|   "errors": { | ||||
|     "forbidden": "접근 권한이 없습니다.", | ||||
|  | @ -120,8 +120,8 @@ | |||
|     "scheduleMessage": "이 글을 공개할 시간을 알려주세요.", | ||||
|     "show": "보기", | ||||
|     "size": "크기", | ||||
|     "upload": "", | ||||
|     "uploadMessage": "" | ||||
|     "upload": "업로드", | ||||
|     "uploadMessage": "업로드 옵션을 선택하세요." | ||||
|   }, | ||||
|   "search": { | ||||
|     "images": "이미지", | ||||
|  | @ -160,7 +160,7 @@ | |||
|     "executeOnShellDescription": "기본적으로 File Browser 는 바이너리를 명령어로 호출하여 실행합니다. 쉘을 통해 실행하기를 원한다면, Bash 또는 PowerShell 에 필요한 인수와 플래그를 설정하세요. 사용자 명령어와 이벤트 훅에 모두 적용됩니다.", | ||||
|     "globalRules": "규칙에 대한 전역설정으로 모든 사용자에게 적용됩니다. 지정된 규칙은 사용자 설정을 덮어쓰기 합니다.", | ||||
|     "globalSettings": "전역 설정", | ||||
|     "hideDotfiles": "", | ||||
|     "hideDotfiles": "숨김파일(dotfile)을 표시하지 않습니다.", | ||||
|     "insertPath": "경로 입력", | ||||
|     "insertRegex": "정규식 입력", | ||||
|     "instanceName": "인스턴스 이름", | ||||
|  | @ -171,7 +171,7 @@ | |||
|     "newUser": "새로운 사용자", | ||||
|     "password": "비밀번호", | ||||
|     "passwordUpdated": "비밀번호 수정 완료!", | ||||
|     "path": "", | ||||
|     "path": "경로", | ||||
|     "perm": { | ||||
|       "create": "파일이나 디렉토리 생성하기", | ||||
|       "delete": "화일이나 디렉토리 삭제하기", | ||||
|  | @ -190,13 +190,13 @@ | |||
|     "rulesHelp": "사용자별로 규칙을 허용/방지를 지정할 수 있습니다. 방지된 파일은 보이지 않고 사용자들은 접근할 수 없습니다. 사용자의 접근 허용 범위와 관련해 정규표현식(regex)과 경로를 지원합니다.\n", | ||||
|     "scope": "범위", | ||||
|     "settingsUpdated": "설정 수정됨!", | ||||
|     "shareDuration": "", | ||||
|     "shareManagement": "", | ||||
|     "singleClick": "", | ||||
|     "shareDuration": "공유 기간", | ||||
|     "shareManagement": "공유 내역 관리", | ||||
|     "singleClick": "한번 클릭으로 파일과 폴더를 열도록 합니다.", | ||||
|     "themes": { | ||||
|       "dark": "", | ||||
|       "light": "", | ||||
|       "title": "" | ||||
|       "dark": "다크테마", | ||||
|       "light": "라이트테마", | ||||
|       "title": "테마" | ||||
|     }, | ||||
|     "user": "사용자", | ||||
|     "userCommands": "명령어", | ||||
|  |  | |||
|  | @ -24,7 +24,7 @@ export async function validateLogin() { | |||
|       await renew(<string>localStorage.getItem("jwt")); | ||||
|     } | ||||
|   } catch (error) { | ||||
|     console.warn("Invalid JWT token in storage"); // eslint-disable-line
 | ||||
|     console.warn("Invalid JWT token in storage"); | ||||
|     throw error; | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -4,7 +4,7 @@ function loading(button: string) { | |||
|   ); | ||||
| 
 | ||||
|   if (el === undefined || el === null) { | ||||
|     console.log("Error getting button " + button); // eslint-disable-line
 | ||||
|     console.log("Error getting button " + button); | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|  | @ -30,7 +30,7 @@ function done(button: string) { | |||
|   ); | ||||
| 
 | ||||
|   if (el === undefined || el === null) { | ||||
|     console.log("Error getting button " + button); // eslint-disable-line
 | ||||
|     console.log("Error getting button " + button); | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|  | @ -51,7 +51,7 @@ function success(button: string) { | |||
|   ); | ||||
| 
 | ||||
|   if (el === undefined || el === null) { | ||||
|     console.log("Error getting button " + button); // eslint-disable-line
 | ||||
|     console.log("Error getting button " + button); | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,39 +1,36 @@ | |||
| // Based on code by the following links:
 | ||||
| // https://stackoverflow.com/a/74528564
 | ||||
| // https://web.dev/articles/async-clipboard
 | ||||
| export function copy(text: string) { | ||||
| 
 | ||||
| interface ClipboardArgs { | ||||
|   text?: string; | ||||
|   data?: ClipboardItems; | ||||
| } | ||||
| 
 | ||||
| interface ClipboardOpts { | ||||
|   permission?: boolean; | ||||
| } | ||||
| 
 | ||||
| export function copy(data: ClipboardArgs, opts?: ClipboardOpts) { | ||||
|   return new Promise<void>((resolve, reject) => { | ||||
|     if ( | ||||
|       // Clipboard API requires secure context
 | ||||
|       window.isSecureContext && | ||||
|       typeof navigator.clipboard !== "undefined" && | ||||
|       // @ts-ignore
 | ||||
|       navigator.permissions !== "undefined" | ||||
|       typeof navigator.clipboard !== "undefined" | ||||
|     ) { | ||||
|       navigator.permissions | ||||
|         // @ts-ignore
 | ||||
|         .query({ name: "clipboard-write" }) | ||||
|         .then((permission) => { | ||||
|           if (permission.state === "granted" || permission.state === "prompt") { | ||||
|             // simple writeText should work for all modern browsers
 | ||||
|             navigator.clipboard.writeText(text).then(resolve).catch(reject); | ||||
|       if (opts?.permission) { | ||||
|         getPermission("clipboard-write") | ||||
|           .then(() => writeToClipboard(data).then(resolve).catch(reject)) | ||||
|           .catch(reject); | ||||
|       } else { | ||||
|             reject(new Error("Permission not granted!")); | ||||
|         writeToClipboard(data).then(resolve).catch(reject); | ||||
|       } | ||||
|         }) | ||||
|         .catch((e) => { | ||||
|           // Firefox doesn't support clipboard-write permission
 | ||||
|           if (navigator.userAgent.indexOf("Firefox") != -1) { | ||||
|             navigator.clipboard.writeText(text).then(resolve).catch(reject); | ||||
|           } else { | ||||
|             reject(e); | ||||
|           } | ||||
|         }); | ||||
|     } else if ( | ||||
|       document.queryCommandSupported && | ||||
|       document.queryCommandSupported("copy") | ||||
|       document.queryCommandSupported("copy") && | ||||
|       data.text // old method only supports text
 | ||||
|     ) { | ||||
|       const textarea = createTemporaryTextarea(text); | ||||
|       const textarea = createTemporaryTextarea(data.text); | ||||
|       const body = document.activeElement || document.body; | ||||
|       try { | ||||
|         body.appendChild(textarea); | ||||
|  | @ -54,6 +51,35 @@ export function copy(text: string) { | |||
|   }); | ||||
| } | ||||
| 
 | ||||
| function getPermission(name: string) { | ||||
|   return new Promise<void>((resolve, reject) => { | ||||
|     typeof navigator.permissions !== "undefined" && | ||||
|       navigator.permissions | ||||
|         // @ts-expect-error chrome specific api
 | ||||
|         .query({ name }) | ||||
|         .then((permission) => { | ||||
|           if (permission.state === "granted" || permission.state === "prompt") { | ||||
|             resolve(); | ||||
|           } else { | ||||
|             reject(new Error("Permission denied!")); | ||||
|           } | ||||
|         }); | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| function writeToClipboard(data: ClipboardArgs) { | ||||
|   if (data.text) { | ||||
|     return navigator.clipboard.writeText(data.text); | ||||
|   } | ||||
|   if (data.data) { | ||||
|     return navigator.clipboard.write(data.data); | ||||
|   } | ||||
| 
 | ||||
|   return new Promise<void>((resolve, reject) => { | ||||
|     reject(new Error("No data was supplied!")); | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| const styles = { | ||||
|   fontSize: "12pt", | ||||
|   position: "fixed", | ||||
|  | @ -69,10 +95,10 @@ const styles = { | |||
|   background: "transparent", | ||||
| }; | ||||
| 
 | ||||
| const createTemporaryTextarea = (text: string) => { | ||||
| function createTemporaryTextarea(text: string) { | ||||
|   const textarea = document.createElement("textarea"); | ||||
|   textarea.value = text; | ||||
|   textarea.setAttribute("readonly", ""); | ||||
|   Object.assign(textarea.style, styles); | ||||
|   return textarea; | ||||
| }; | ||||
| } | ||||
|  |  | |||
|  | @ -6,13 +6,16 @@ export default function getRule(rules: string[]) { | |||
|   let result = null; | ||||
|   const find = Array.prototype.find; | ||||
| 
 | ||||
|   find.call(document.styleSheets, (styleSheet) => { | ||||
|     result = find.call(styleSheet.cssRules, (cssRule) => { | ||||
|   find.call(document.styleSheets, (styleSheet: CSSStyleSheet) => { | ||||
|     result = find.call(styleSheet.cssRules, (cssRule: CSSRule) => { | ||||
|       let found = false; | ||||
| 
 | ||||
|       if (cssRule instanceof window.CSSStyleRule) { | ||||
|       // faster than checking instanceof for every element
 | ||||
|       if (cssRule.constructor.name === "CSSStyleRule") { | ||||
|         for (let i = 0; i < rules.length; i++) { | ||||
|           if (cssRule.selectorText.toLowerCase() === rules[i]) { | ||||
|           if ( | ||||
|             (cssRule as CSSStyleRule).selectorText.toLowerCase() === rules[i] | ||||
|           ) { | ||||
|             found = true; | ||||
|           } | ||||
|         } | ||||
|  | @ -24,5 +27,5 @@ export default function getRule(rules: string[]) { | |||
|     return result != null; | ||||
|   }); | ||||
| 
 | ||||
|   return result; | ||||
|   return result as CSSStyleRule | null; | ||||
| } | ||||
|  |  | |||
|  | @ -325,6 +325,7 @@ const token = ref<string>(""); | |||
| const audio = ref<HTMLAudioElement>(); | ||||
| const tag = ref<boolean>(false); | ||||
| 
 | ||||
| const $showError = inject<IToastError>("$showError")!; | ||||
| const $showSuccess = inject<IToastSuccess>("$showSuccess")!; | ||||
| 
 | ||||
| const { t } = useI18n({}); | ||||
|  | @ -463,9 +464,9 @@ const download = () => { | |||
|       if (req.value === null) return false; | ||||
|       layoutStore.closeHovers(); | ||||
| 
 | ||||
|       let files: string[] = []; | ||||
|       const files: string[] = []; | ||||
| 
 | ||||
|       for (let i of fileStore.selected) { | ||||
|       for (const i of fileStore.selected) { | ||||
|         files.push(req.value.items[i].path); | ||||
|       } | ||||
| 
 | ||||
|  | @ -488,13 +489,23 @@ const linkSelected = () => { | |||
| }; | ||||
| 
 | ||||
| const copyToClipboard = (text: string) => { | ||||
|   copy(text).then( | ||||
|   copy({ text }).then( | ||||
|     () => { | ||||
|       // clipboard successfully set | ||||
|       $showSuccess(t("success.linkCopied")); | ||||
|     }, | ||||
|     () => { | ||||
|       // clipboard write failed | ||||
|       copy({ text }, { permission: true }).then( | ||||
|         () => { | ||||
|           // clipboard successfully set | ||||
|           $showSuccess(t("success.linkCopied")); | ||||
|         }, | ||||
|         (e) => { | ||||
|           // clipboard write failed | ||||
|           $showError(e); | ||||
|         } | ||||
|       ); | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
|  |  | |||
|  | @ -108,7 +108,7 @@ onMounted(() => { | |||
|     showPrintMargin: false, | ||||
|     readOnly: fileStore.req?.type === "textImmutable", | ||||
|     theme: "ace/theme/chrome", | ||||
|     mode: modelist.getModeForPath(fileStore.req?.name).mode, | ||||
|     mode: modelist.getModeForPath(fileStore.req!.name).mode, | ||||
|     wrap: true, | ||||
|     enableBasicAutocompletion: true, | ||||
|     enableLiveAutocompletion: true, | ||||
|  | @ -173,7 +173,7 @@ const close = () => { | |||
| 
 | ||||
|   fileStore.updateRequest(null); | ||||
| 
 | ||||
|   let uri = url.removeLastDir(route.path) + "/"; | ||||
|   const uri = url.removeLastDir(route.path) + "/"; | ||||
|   router.push({ path: uri }); | ||||
| }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -523,12 +523,12 @@ const keyEvent = (event: KeyboardEvent) => { | |||
|       break; | ||||
|     case "a": | ||||
|       event.preventDefault(); | ||||
|       for (let file of items.value.files) { | ||||
|       for (const file of items.value.files) { | ||||
|         if (fileStore.selected.indexOf(file.index) === -1) { | ||||
|           fileStore.selected.push(file.index); | ||||
|         } | ||||
|       } | ||||
|       for (let dir of items.value.dirs) { | ||||
|       for (const dir of items.value.dirs) { | ||||
|         if (fileStore.selected.indexOf(dir.index) === -1) { | ||||
|           fileStore.selected.push(dir.index); | ||||
|         } | ||||
|  | @ -551,9 +551,9 @@ const copyCut = (event: Event | KeyboardEvent): void => { | |||
| 
 | ||||
|   if (fileStore.req === null) return; | ||||
| 
 | ||||
|   let items = []; | ||||
|   const items = []; | ||||
| 
 | ||||
|   for (let i of fileStore.selected) { | ||||
|   for (const i of fileStore.selected) { | ||||
|     items.push({ | ||||
|       from: fileStore.req.items[i].url, | ||||
|       name: fileStore.req.items[i].name, | ||||
|  | @ -575,9 +575,9 @@ const paste = (event: Event) => { | |||
|   if ((event.target as HTMLElement).tagName?.toLowerCase() === "input") return; | ||||
| 
 | ||||
|   // TODO router location should it be | ||||
|   let items: any[] = []; | ||||
|   const items: any[] = []; | ||||
| 
 | ||||
|   for (let item of clipboardStore.items) { | ||||
|   for (const item of clipboardStore.items) { | ||||
|     const from = item.from.endsWith("/") ? item.from.slice(0, -1) : item.from; | ||||
|     const to = route.path + encodeURIComponent(item.name); | ||||
|     items.push({ from, to, name: item.name }); | ||||
|  | @ -614,7 +614,7 @@ const paste = (event: Event) => { | |||
|     return; | ||||
|   } | ||||
| 
 | ||||
|   let conflict = upload.checkConflict(items, fileStore.req!.items); | ||||
|   const conflict = upload.checkConflict(items, fileStore.req!.items); | ||||
| 
 | ||||
|   let overwrite = false; | ||||
|   let rename = false; | ||||
|  | @ -640,14 +640,13 @@ const paste = (event: Event) => { | |||
| 
 | ||||
| const colunmsResize = () => { | ||||
|   // Update the columns size based on the window width. | ||||
|   let items_ = css(["#listing.mosaic .item", ".mosaic#listing .item"]); | ||||
|   const items_ = css(["#listing.mosaic .item", ".mosaic#listing .item"]); | ||||
|   if (items_ === null) return; | ||||
| 
 | ||||
|   let columns = Math.floor( | ||||
|     (document.querySelector("main")?.offsetWidth ?? 0) / columnWidth.value | ||||
|   ); | ||||
|   if (columns === 0) columns = 1; | ||||
|   // @ts-ignore never type error | ||||
|   items_.style.width = `calc(${100 / columns}% - 1em)`; | ||||
| }; | ||||
| 
 | ||||
|  | @ -677,11 +676,10 @@ const dragEnter = () => { | |||
| 
 | ||||
|   // When the user starts dragging an item, put every | ||||
|   // file on the listing with 50% opacity. | ||||
|   let items = document.getElementsByClassName("item"); | ||||
|   const items = document.getElementsByClassName("item"); | ||||
| 
 | ||||
|   // @ts-ignore | ||||
|   Array.from(items).forEach((file: HTMLElement) => { | ||||
|     file.style.opacity = "0.5"; | ||||
|   Array.from(items).forEach((file: Element) => { | ||||
|     (file as HTMLElement).style.opacity = "0.5"; | ||||
|   }); | ||||
| }; | ||||
| 
 | ||||
|  | @ -698,7 +696,7 @@ const drop = async (event: DragEvent) => { | |||
|   dragCounter.value = 0; | ||||
|   resetOpacity(); | ||||
| 
 | ||||
|   let dt = event.dataTransfer; | ||||
|   const dt = event.dataTransfer; | ||||
|   let el: HTMLElement | null = event.target as HTMLElement; | ||||
| 
 | ||||
|   if (fileStore.req === null || dt === null || dt.files.length <= 0) return; | ||||
|  | @ -709,7 +707,7 @@ const drop = async (event: DragEvent) => { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let files: UploadList = (await upload.scanFiles(dt)) as UploadList; | ||||
|   const files: UploadList = (await upload.scanFiles(dt)) as UploadList; | ||||
|   let items = fileStore.req.items; | ||||
|   let path = route.path.endsWith("/") ? route.path : route.path + "/"; | ||||
| 
 | ||||
|  | @ -729,7 +727,7 @@ const drop = async (event: DragEvent) => { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   let conflict = upload.checkConflict(files, items); | ||||
|   const conflict = upload.checkConflict(files, items); | ||||
| 
 | ||||
|   if (conflict) { | ||||
|     layoutStore.showHover({ | ||||
|  | @ -753,10 +751,10 @@ const drop = async (event: DragEvent) => { | |||
| }; | ||||
| 
 | ||||
| const uploadInput = (event: Event) => { | ||||
|   let files = (event.currentTarget as HTMLInputElement)?.files; | ||||
|   const files = (event.currentTarget as HTMLInputElement)?.files; | ||||
|   if (files === null) return; | ||||
| 
 | ||||
|   let folder_upload = !!files[0].webkitRelativePath; | ||||
|   const folder_upload = !!files[0].webkitRelativePath; | ||||
| 
 | ||||
|   const uploadFiles: UploadList = []; | ||||
|   for (let i = 0; i < files.length; i++) { | ||||
|  | @ -771,8 +769,8 @@ const uploadInput = (event: Event) => { | |||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   let path = route.path.endsWith("/") ? route.path : route.path + "/"; | ||||
|   let conflict = upload.checkConflict(uploadFiles, fileStore.req!.items); | ||||
|   const path = route.path.endsWith("/") ? route.path : route.path + "/"; | ||||
|   const conflict = upload.checkConflict(uploadFiles, fileStore.req!.items); | ||||
| 
 | ||||
|   if (conflict) { | ||||
|     layoutStore.showHover({ | ||||
|  | @ -796,7 +794,7 @@ const uploadInput = (event: Event) => { | |||
| }; | ||||
| 
 | ||||
| const resetOpacity = () => { | ||||
|   let items = document.getElementsByClassName("item"); | ||||
|   const items = document.getElementsByClassName("item"); | ||||
| 
 | ||||
|   Array.from(items).forEach((file: Element) => { | ||||
|     (file as HTMLElement).style.opacity = "1"; | ||||
|  | @ -822,7 +820,6 @@ const sort = async (by: string) => { | |||
| 
 | ||||
|   try { | ||||
|     if (authStore.user?.id) { | ||||
|       // @ts-ignore | ||||
|       await users.update({ id: authStore.user?.id, sorting: { by, asc } }, [ | ||||
|         "sorting", | ||||
|       ]); | ||||
|  | @ -873,10 +870,10 @@ const download = () => { | |||
|     confirm: (format: any) => { | ||||
|       layoutStore.closeHovers(); | ||||
| 
 | ||||
|       let files = []; | ||||
|       const files = []; | ||||
| 
 | ||||
|       if (fileStore.selectedCount > 0 && fileStore.req !== null) { | ||||
|         for (let i of fileStore.selected) { | ||||
|         for (const i of fileStore.selected) { | ||||
|           files.push(fileStore.req.items[i].url); | ||||
|         } | ||||
|       } else { | ||||
|  | @ -899,13 +896,12 @@ const switchView = async () => { | |||
| 
 | ||||
|   const data = { | ||||
|     id: authStore.user?.id, | ||||
|     viewMode: modes[authStore.user?.viewMode ?? "list"] || "list", | ||||
|     viewMode: (modes[authStore.user?.viewMode ?? "list"] || | ||||
|       "list") as ViewModeType, | ||||
|   }; | ||||
| 
 | ||||
|   // @ts-ignore | ||||
|   users.update(data, ["viewMode"]).catch($showError); | ||||
| 
 | ||||
|   // @ts-ignore | ||||
|   authStore.updateUser(data); | ||||
| 
 | ||||
|   setItemWeight(); | ||||
|  |  | |||
|  | @ -353,7 +353,7 @@ const updatePreview = async () => { | |||
|     autoPlay.value = false; | ||||
|   } | ||||
| 
 | ||||
|   let dirs = route.fullPath.split("/"); | ||||
|   const dirs = route.fullPath.split("/"); | ||||
|   name.value = decodeURIComponent(dirs[dirs.length - 1]); | ||||
| 
 | ||||
|   if (!listing.value) { | ||||
|  | @ -422,7 +422,7 @@ const toggleNavigation = throttle(function () { | |||
| const close = () => { | ||||
|   fileStore.updateRequest(null); | ||||
| 
 | ||||
|   let uri = url.removeLastDir(route.path) + "/"; | ||||
|   const uri = url.removeLastDir(route.path) + "/"; | ||||
|   router.push({ path: uri }); | ||||
| }; | ||||
| 
 | ||||
|  |  | |||
|  | @ -282,7 +282,7 @@ const formattedChunkSize = computed({ | |||
| // Define funcs | ||||
| const capitalize = (name: string, where: string | RegExp = "_") => { | ||||
|   if (where === "caps") where = /(?=[A-Z])/; | ||||
|   let split = name.split(where); | ||||
|   const split = name.split(where); | ||||
|   name = ""; | ||||
| 
 | ||||
|   for (let i = 0; i < split.length; i++) { | ||||
|  | @ -294,7 +294,7 @@ const capitalize = (name: string, where: string | RegExp = "_") => { | |||
| 
 | ||||
| const save = async () => { | ||||
|   if (settings.value === null) return false; | ||||
|   let newSettings: ISettings = { | ||||
|   const newSettings: ISettings = { | ||||
|     ...settings.value, | ||||
|     shell: | ||||
|       settings.value?.shell | ||||
|  | @ -376,7 +376,7 @@ onMounted(async () => { | |||
|   try { | ||||
|     layoutStore.loading = true; | ||||
|     const original: ISettings = await api.get(); | ||||
|     let newSettings: ISettings = { ...original, commands: {} }; | ||||
|     const newSettings: ISettings = { ...original, commands: {} }; | ||||
| 
 | ||||
|     const keys = Object.keys(original.commands) as Array<keyof SettingsCommand>; | ||||
|     for (const key of keys) { | ||||
|  |  | |||
|  | @ -87,12 +87,12 @@ onMounted(async () => { | |||
|   layoutStore.loading = true; | ||||
| 
 | ||||
|   try { | ||||
|     let newLinks = await api.list(); | ||||
|     const newLinks = await api.list(); | ||||
|     if (authStore.user?.perm.admin) { | ||||
|       let userMap = new Map<number, string>(); | ||||
|       for (let user of await users.getAll()) | ||||
|       const userMap = new Map<number, string>(); | ||||
|       for (const user of await users.getAll()) | ||||
|         userMap.set(user.id, user.username); | ||||
|       for (let link of newLinks) { | ||||
|       for (const link of newLinks) { | ||||
|         if (link.userID && userMap.has(link.userID)) | ||||
|           link.username = userMap.get(link.userID); | ||||
|       } | ||||
|  | @ -108,13 +108,23 @@ onMounted(async () => { | |||
| }); | ||||
| 
 | ||||
| const copyToClipboard = (text: string) => { | ||||
|   copy(text).then( | ||||
|   copy({ text }).then( | ||||
|     () => { | ||||
|       // clipboard successfully set | ||||
|       $showSuccess(t("success.linkCopied")); | ||||
|     }, | ||||
|     () => { | ||||
|       // clipboard write failed | ||||
|       copy({ text }, { permission: true }).then( | ||||
|         () => { | ||||
|           // clipboard successfully set | ||||
|           $showSuccess(t("success.linkCopied")); | ||||
|         }, | ||||
|         (e) => { | ||||
|           // clipboard write failed | ||||
|           $showError(e); | ||||
|         } | ||||
|       ); | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
|  |  | |||
|  | @ -90,7 +90,7 @@ const fetchData = async () => { | |||
| 
 | ||||
|   try { | ||||
|     if (isNew.value) { | ||||
|       let { defaults, createUserDir: _createUserDir } = await settings.get(); | ||||
|       const { defaults, createUserDir: _createUserDir } = await settings.get(); | ||||
|       createUserDir.value = _createUserDir; | ||||
|       user.value = { | ||||
|         ...defaults, | ||||
|  |  | |||
							
								
								
									
										2
									
								
								go.mod
								
								
								
								
							
							
						
						
									
										2
									
								
								go.mod
								
								
								
								
							|  | @ -64,7 +64,7 @@ require ( | |||
| 	github.com/yusufpapurcu/wmi v1.2.4 // indirect | ||||
| 	go.uber.org/multierr v1.11.0 // indirect | ||||
| 	golang.org/x/exp v0.0.0-20240325151524-a685a6edb6d8 // indirect | ||||
| 	golang.org/x/net v0.23.0 // indirect | ||||
| 	golang.org/x/net v0.33.0 // indirect | ||||
| 	golang.org/x/sys v0.28.0 // indirect | ||||
| 	gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect | ||||
| 	gopkg.in/ini.v1 v1.67.0 // indirect | ||||
|  |  | |||
							
								
								
									
										4
									
								
								go.sum
								
								
								
								
							
							
						
						
									
										4
									
								
								go.sum
								
								
								
								
							|  | @ -190,8 +190,8 @@ golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/ | |||
| golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200904194848-62affa334b73/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20221002022538-bcab6841153b/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= | ||||
| golang.org/x/net v0.23.0 h1:7EYJ93RZ9vYSZAIb2x3lnuvqO5zneoD6IvWjuhfxjTs= | ||||
| golang.org/x/net v0.23.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg= | ||||
| golang.org/x/net v0.33.0 h1:74SYHlV8BIgHIFC/LrYkOGIwL19eTYXQ5wc6TBuO36I= | ||||
| golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= | ||||
| golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= | ||||
| golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	 Oleg Lobanov
						Oleg Lobanov