feat: select item on file list after navigating back (#5329)

This commit is contained in:
Ramires Viana 2025-07-27 08:03:00 -03:00 committed by GitHub
parent 25e47c3ce8
commit cbeec6d225
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
13 changed files with 143 additions and 79 deletions

View File

@ -62,6 +62,7 @@ import FileList from "./FileList.vue";
import { files as api } from "@/api"; import { files as api } from "@/api";
import buttons from "@/utils/buttons"; import buttons from "@/utils/buttons";
import * as upload from "@/utils/upload"; import * as upload from "@/utils/upload";
import { removePrefix } from "@/api/utils";
export default { export default {
name: "copy", name: "copy",
@ -76,7 +77,7 @@ export default {
computed: { computed: {
...mapState(useFileStore, ["req", "selected"]), ...mapState(useFileStore, ["req", "selected"]),
...mapState(useAuthStore, ["user"]), ...mapState(useAuthStore, ["user"]),
...mapWritableState(useFileStore, ["reload"]), ...mapWritableState(useFileStore, ["reload", "preselect"]),
}, },
methods: { methods: {
...mapActions(useLayoutStore, ["showHover", "closeHovers"]), ...mapActions(useLayoutStore, ["showHover", "closeHovers"]),
@ -100,6 +101,7 @@ export default {
.copy(items, overwrite, rename) .copy(items, overwrite, rename)
.then(() => { .then(() => {
buttons.success("copy"); buttons.success("copy");
this.preselect = removePrefix(items[0].to);
if (this.$route.path === this.dest) { if (this.$route.path === this.dest) {
this.reload = true; this.reload = true;

View File

@ -48,16 +48,15 @@ export default {
"selectedCount", "selectedCount",
"req", "req",
"selected", "selected",
"currentPrompt",
]), ]),
...mapWritableState(useFileStore, ["reload"]), ...mapState(useLayoutStore, ["currentPrompt"]),
...mapWritableState(useFileStore, ["reload", "preselect"]),
}, },
methods: { methods: {
...mapActions(useLayoutStore, ["closeHovers"]), ...mapActions(useLayoutStore, ["closeHovers"]),
submit: async function () { submit: async function () {
buttons.loading("delete"); buttons.loading("delete");
window.sessionStorage.setItem("modified", "true");
try { try {
if (!this.isListing) { if (!this.isListing) {
await api.remove(this.$route.path); await api.remove(this.$route.path);
@ -81,6 +80,12 @@ export default {
await Promise.all(promises); await Promise.all(promises);
buttons.success("delete"); buttons.success("delete");
const nearbyItem =
this.req.items[Math.max(0, Math.min(this.selected) - 1)];
this.preselect = nearbyItem?.path;
this.reload = true; this.reload = true;
} catch (e) { } catch (e) {
buttons.done("delete"); buttons.done("delete");

View File

@ -17,7 +17,7 @@
</button> </button>
<button <button
id="focus-prompt" id="focus-prompt"
@click="submit" @click="currentPrompt.confirm"
class="button button--flat button--red" class="button button--flat button--red"
:aria-label="$t('buttons.discardChanges')" :aria-label="$t('buttons.discardChanges')"
:title="$t('buttons.discardChanges')" :title="$t('buttons.discardChanges')"
@ -30,22 +30,16 @@
</template> </template>
<script> <script>
import { mapActions } from "pinia"; import { mapState, mapActions } from "pinia";
import url from "@/utils/url";
import { useLayoutStore } from "@/stores/layout"; import { useLayoutStore } from "@/stores/layout";
import { useFileStore } from "@/stores/file";
export default { export default {
name: "discardEditorChanges", name: "discardEditorChanges",
computed: {
...mapState(useLayoutStore, ["currentPrompt"]),
},
methods: { methods: {
...mapActions(useLayoutStore, ["closeHovers"]), ...mapActions(useLayoutStore, ["closeHovers"]),
...mapActions(useFileStore, ["updateRequest"]),
submit: async function () {
this.updateRequest(null);
const uri = url.removeLastDir(this.$route.path) + "/";
this.$router.push({ path: uri });
},
}, },
}; };
</script> </script>

View File

@ -55,7 +55,7 @@
</template> </template>
<script> <script>
import { mapActions, mapState } from "pinia"; import { mapActions, mapState, mapWritableState } from "pinia";
import { useFileStore } from "@/stores/file"; import { useFileStore } from "@/stores/file";
import { useLayoutStore } from "@/stores/layout"; import { useLayoutStore } from "@/stores/layout";
import { useAuthStore } from "@/stores/auth"; import { useAuthStore } from "@/stores/auth";
@ -63,6 +63,7 @@ import FileList from "./FileList.vue";
import { files as api } from "@/api"; import { files as api } from "@/api";
import buttons from "@/utils/buttons"; import buttons from "@/utils/buttons";
import * as upload from "@/utils/upload"; import * as upload from "@/utils/upload";
import { removePrefix } from "@/api/utils";
export default { export default {
name: "move", name: "move",
@ -77,6 +78,7 @@ export default {
computed: { computed: {
...mapState(useFileStore, ["req", "selected"]), ...mapState(useFileStore, ["req", "selected"]),
...mapState(useAuthStore, ["user"]), ...mapState(useAuthStore, ["user"]),
...mapWritableState(useFileStore, ["preselect"]),
excludedFolders() { excludedFolders() {
return this.selected return this.selected
.filter((idx) => this.req.items[idx].isDir) .filter((idx) => this.req.items[idx].isDir)
@ -104,6 +106,7 @@ export default {
.move(items, overwrite, rename) .move(items, overwrite, rename)
.then(() => { .then(() => {
buttons.success("move"); buttons.success("move");
this.preselect = removePrefix(items[0].to);
this.$router.push({ path: this.dest }); this.$router.push({ path: this.dest });
}) })
.catch((e) => { .catch((e) => {

View File

@ -46,6 +46,7 @@ import { useFileStore } from "@/stores/file";
import { useLayoutStore } from "@/stores/layout"; import { useLayoutStore } from "@/stores/layout";
import url from "@/utils/url"; import url from "@/utils/url";
import { files as api } from "@/api"; import { files as api } from "@/api";
import { removePrefix } from "@/api/utils";
export default { export default {
name: "rename", name: "rename",
@ -65,7 +66,7 @@ export default {
"selectedCount", "selectedCount",
"isListing", "isListing",
]), ]),
...mapWritableState(useFileStore, ["reload"]), ...mapWritableState(useFileStore, ["reload", "preselect"]),
}, },
methods: { methods: {
...mapActions(useLayoutStore, ["closeHovers"]), ...mapActions(useLayoutStore, ["closeHovers"]),
@ -97,7 +98,6 @@ export default {
newLink = newLink =
url.removeLastDir(oldLink) + "/" + encodeURIComponent(this.name); url.removeLastDir(oldLink) + "/" + encodeURIComponent(this.name);
window.sessionStorage.setItem("modified", "true");
try { try {
await api.move([{ from: oldLink, to: newLink }]); await api.move([{ from: oldLink, to: newLink }]);
if (!this.isListing) { if (!this.isListing) {
@ -105,6 +105,8 @@ export default {
return; return;
} }
this.preselect = removePrefix(newLink);
this.reload = true; this.reload = true;
} catch (e) { } catch (e) {
this.$showError(e); this.$showError(e);

View File

@ -96,6 +96,9 @@ main {
height: 3em; height: 3em;
background: var(--background); background: var(--background);
border-bottom: 1px solid var(--divider); border-bottom: 1px solid var(--divider);
position: sticky;
z-index: 1000;
top: 4em;
} }
.breadcrumbs span, .breadcrumbs span,

View File

@ -329,6 +329,7 @@ main .spinner .bounce2 {
#editor-container { #editor-container {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
justify-content: center;
background-color: var(--background); background-color: var(--background);
position: fixed; position: fixed;
padding-top: 4em; padding-top: 4em;
@ -351,6 +352,8 @@ main .spinner .bounce2 {
#editor-container .breadcrumbs { #editor-container .breadcrumbs {
height: 2.3em; height: 2.3em;
padding: 0 1em; padding: 0 1em;
position: relative;
top: 0;
} }
/*** RTL - flip and position arrow of path ***/ /*** RTL - flip and position arrow of path ***/

View File

@ -9,6 +9,7 @@ export const useFileStore = defineStore("file", {
selected: number[]; selected: number[];
multiple: boolean; multiple: boolean;
isFiles: boolean; isFiles: boolean;
preselect: string | null;
} => ({ } => ({
req: null, req: null,
oldReq: null, oldReq: null,
@ -16,6 +17,7 @@ export const useFileStore = defineStore("file", {
selected: [], selected: [],
multiple: false, multiple: false,
isFiles: false, isFiles: false,
preselect: null,
}), }),
getters: { getters: {
selectedCount: (state) => state.selected.length, selectedCount: (state) => state.selected.length,

View File

@ -1,7 +1,7 @@
<template> <template>
<div> <div>
<header-bar <header-bar
v-if="error || fileStore.req?.type === null" v-if="error || fileStore.req?.type === undefined"
showMenu showMenu
showLogo showLogo
/> />
@ -9,7 +9,7 @@
<breadcrumbs base="/files" /> <breadcrumbs base="/files" />
<errors v-if="error" :errorCode="error.status" /> <errors v-if="error" :errorCode="error.status" />
<component v-else-if="currentView" :is="currentView"></component> <component v-else-if="currentView" :is="currentView"></component>
<div v-else-if="currentView !== null"> <div v-else>
<h2 class="message delayed"> <h2 class="message delayed">
<div class="spinner"> <div class="spinner">
<div class="bounce1"></div> <div class="bounce1"></div>
@ -102,15 +102,7 @@ onUnmounted(() => {
fetchDataController.abort(); fetchDataController.abort();
}); });
watch(route, (to, from) => { watch(route, () => {
if (from.path.endsWith("/")) {
window.sessionStorage.setItem(
"listFrozen",
(!to.path.endsWith("/")).toString()
);
} else if (to.path.endsWith("/")) {
fileStore.updateRequest(null);
}
fetchData(); fetchData();
}); });
watch(reload, (newValue) => { watch(reload, (newValue) => {
@ -122,6 +114,32 @@ watch(uploadError, (newValue) => {
// Define functions // Define functions
const applyPreSelection = () => {
const preselect = fileStore.preselect;
fileStore.preselect = null;
if (!fileStore.req?.isDir || fileStore.oldReq === null) return;
let index = -1;
if (preselect) {
// Find item with the specified path
index = fileStore.req.items.findIndex((item) => item.path === preselect);
} else if (fileStore.oldReq.path.startsWith(fileStore.req.path)) {
// Get immediate child folder of the previous path
const name = fileStore.oldReq.path
.substring(fileStore.req.path.length)
.split("/")
.shift();
index = fileStore.req.items.findIndex(
(val) => val.path == fileStore.req!.path + name
);
}
if (index === -1) return;
fileStore.selected.push(index);
};
const fetchData = async () => { const fetchData = async () => {
// Reset view information. // Reset view information.
fileStore.reload = false; fileStore.reload = false;
@ -130,12 +148,7 @@ const fetchData = async () => {
layoutStore.closeHovers(); layoutStore.closeHovers();
// Set loading to true and reset the error. // Set loading to true and reset the error.
if ( layoutStore.loading = true;
window.sessionStorage.getItem("listFrozen") !== "true" &&
window.sessionStorage.getItem("modified") !== "true"
) {
layoutStore.loading = true;
}
error.value = null; error.value = null;
let url = route.path; let url = route.path;
@ -149,6 +162,9 @@ const fetchData = async () => {
fileStore.updateRequest(res); fileStore.updateRequest(res);
document.title = `${res.name || t("sidebar.myFiles")} - ${t("files.files")} - ${name}`; document.title = `${res.name || t("sidebar.myFiles")} - ${t("files.files")} - ${name}`;
layoutStore.loading = false; layoutStore.loading = false;
// Selects the post-reload target item or the previously visited child folder
applyPreSelection();
} catch (err) { } catch (err) {
if (err instanceof StatusError && err.is_canceled) { if (err instanceof StatusError && err.is_canceled) {
return; return;

View File

@ -32,17 +32,25 @@
/> />
</header-bar> </header-bar>
<Breadcrumbs base="/files" noLink />
<!-- preview container --> <!-- preview container -->
<div <div class="loading delayed" v-if="layoutStore.loading">
v-show="isPreview && isMarkdownFile" <div class="spinner">
id="preview-container" <div class="bounce1"></div>
class="md_preview" <div class="bounce2"></div>
v-html="previewContent" <div class="bounce3"></div>
></div> </div>
</div>
<template v-else>
<Breadcrumbs base="/files" noLink />
<form v-show="!isPreview || !isMarkdownFile" id="editor"></form> <div
v-show="isPreview && isMarkdownFile"
id="preview-container"
class="md_preview"
v-html="previewContent"
></div>
<form v-show="!isPreview || !isMarkdownFile" id="editor"></form>
</template>
</div> </div>
</template> </template>
@ -146,12 +154,19 @@ onBeforeUnmount(() => {
}); });
onBeforeRouteUpdate((to, from, next) => { onBeforeRouteUpdate((to, from, next) => {
if (!editor.value?.session.getUndoManager().isClean()) { if (editor.value?.session.getUndoManager().isClean()) {
layoutStore.showHover("discardEditorChanges");
next(false);
} else {
next(); next();
return;
} }
layoutStore.showHover({
prompt: "discardEditorChanges",
confirm: (event: Event) => {
event.preventDefault();
next();
},
});
}); });
const keyEvent = (event: KeyboardEvent) => { const keyEvent = (event: KeyboardEvent) => {
@ -216,13 +231,6 @@ const decreaseFontSize = () => {
}; };
const close = () => { const close = () => {
if (!editor.value?.session.getUndoManager().isClean()) {
layoutStore.showHover("discardEditorChanges");
return;
}
fileStore.updateRequest(null);
const uri = url.removeLastDir(route.path) + "/"; const uri = url.removeLastDir(route.path) + "/";
router.push({ path: uri }); router.push({ path: uri });
}; };

View File

@ -303,6 +303,7 @@ import {
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
import { useI18n } from "vue-i18n"; import { useI18n } from "vue-i18n";
import { storeToRefs } from "pinia"; import { storeToRefs } from "pinia";
import { removePrefix } from "@/api/utils";
const showLimit = ref<number>(50); const showLimit = ref<number>(50);
const columnWidth = ref<number>(280); const columnWidth = ref<number>(280);
@ -420,25 +421,19 @@ const isMobile = computed(() => {
watch(req, () => { watch(req, () => {
// Reset the show value // Reset the show value
if ( showLimit.value = 50;
window.sessionStorage.getItem("listFrozen") !== "true" &&
window.sessionStorage.getItem("modified") !== "true"
) {
showLimit.value = 50;
nextTick(() => { nextTick(() => {
// Ensures that the listing is displayed // Ensures that the listing is displayed
// How much every listing item affects the window height // How much every listing item affects the window height
setItemWeight(); setItemWeight();
// Scroll to the item opened previously
if (!revealPreviousItem()) {
// Fill and fit the window with listing items // Fill and fit the window with listing items
fillWindow(true); fillWindow(true);
}); }
} });
if (req.value?.isDir) {
window.sessionStorage.setItem("listFrozen", "false");
window.sessionStorage.setItem("modified", "false");
}
}); });
onMounted(() => { onMounted(() => {
@ -448,8 +443,11 @@ onMounted(() => {
// How much every listing item affects the window height // How much every listing item affects the window height
setItemWeight(); setItemWeight();
// Fill and fit the window with listing items // Scroll to the item opened previously
fillWindow(true); if (!revealPreviousItem()) {
// Fill and fit the window with listing items
fillWindow(true);
}
// Add the needed event listeners to the window and document. // Add the needed event listeners to the window and document.
window.addEventListener("keydown", keyEvent); window.addEventListener("keydown", keyEvent);
@ -589,10 +587,13 @@ const paste = (event: Event) => {
return; return;
} }
const preselect = removePrefix(route.path) + items[0].name;
let action = (overwrite: boolean, rename: boolean) => { let action = (overwrite: boolean, rename: boolean) => {
api api
.copy(items, overwrite, rename) .copy(items, overwrite, rename)
.then(() => { .then(() => {
fileStore.preselect = preselect;
fileStore.reload = true; fileStore.reload = true;
}) })
.catch($showError); .catch($showError);
@ -604,6 +605,7 @@ const paste = (event: Event) => {
.move(items, overwrite, rename) .move(items, overwrite, rename)
.then(() => { .then(() => {
clipboardStore.resetClipboard(); clipboardStore.resetClipboard();
fileStore.preselect = preselect;
fileStore.reload = true; fileStore.reload = true;
}) })
.catch($showError); .catch($showError);
@ -731,6 +733,8 @@ const drop = async (event: DragEvent) => {
const conflict = upload.checkConflict(files, items); const conflict = upload.checkConflict(files, items);
const preselect = removePrefix(path) + (files[0].fullPath || files[0].name);
if (conflict) { if (conflict) {
layoutStore.showHover({ layoutStore.showHover({
prompt: "replace", prompt: "replace",
@ -738,11 +742,13 @@ const drop = async (event: DragEvent) => {
event.preventDefault(); event.preventDefault();
layoutStore.closeHovers(); layoutStore.closeHovers();
upload.handleFiles(files, path, false); upload.handleFiles(files, path, false);
fileStore.preselect = preselect;
}, },
confirm: (event: Event) => { confirm: (event: Event) => {
event.preventDefault(); event.preventDefault();
layoutStore.closeHovers(); layoutStore.closeHovers();
upload.handleFiles(files, path, true); upload.handleFiles(files, path, true);
fileStore.preselect = preselect;
}, },
}); });
@ -750,6 +756,7 @@ const drop = async (event: DragEvent) => {
} }
upload.handleFiles(files, path); upload.handleFiles(files, path);
fileStore.preselect = preselect;
}; };
const uploadInput = (event: Event) => { const uploadInput = (event: Event) => {
@ -953,4 +960,21 @@ const fillWindow = (fit = false) => {
// Set the number of displayed items // Set the number of displayed items
showLimit.value = showQuantity > totalItems ? totalItems : showQuantity; showLimit.value = showQuantity > totalItems ? totalItems : showQuantity;
}; };
const revealPreviousItem = () => {
if (!fileStore.req || !fileStore.oldReq) return;
const index = fileStore.selected[0];
if (index === undefined) return;
showLimit.value =
index + Math.ceil((window.innerHeight * 2) / itemWeight.value);
nextTick(() => {
const items = document.querySelectorAll("#listing .item");
items[index].scrollIntoView({ block: "center" });
});
return true;
};
</script> </script>

View File

@ -301,10 +301,8 @@ watch(route, () => {
// Specify hooks // Specify hooks
onMounted(async () => { onMounted(async () => {
window.addEventListener("keydown", key); window.addEventListener("keydown", key);
if (fileStore.oldReq) { listing.value = fileStore.oldReq?.items ?? null;
listing.value = fileStore.oldReq.items; updatePreview();
updatePreview();
}
}); });
onBeforeUnmount(() => window.removeEventListener("keydown", key)); onBeforeUnmount(() => window.removeEventListener("keydown", key));
@ -317,11 +315,16 @@ const deleteFile = () => {
if (listing.value === null) { if (listing.value === null) {
return; return;
} }
listing.value = listing.value.filter((item) => item.name !== name.value);
const index = listing.value.findIndex((item) => item.name == name.value);
listing.value.splice(index, 1);
if (hasNext.value) { if (hasNext.value) {
next(); next();
} else if (!hasPrevious.value && !hasNext.value) { } else if (!hasPrevious.value && !hasNext.value) {
const nearbyItem = listing.value[Math.max(0, index - 1)];
fileStore.preselect = nearbyItem?.path;
close(); close();
} else { } else {
prev(); prev();
@ -427,8 +430,6 @@ const toggleNavigation = throttle(function () {
}, 500); }, 500);
const close = () => { const close = () => {
fileStore.updateRequest(null);
const uri = url.removeLastDir(route.path) + "/"; const uri = url.removeLastDir(route.path) + "/";
router.push({ path: uri }); router.push({ path: uri });
}; };

View File

@ -5,6 +5,7 @@
"exclude": [ "exclude": [
"src/components/Shell.vue", "src/components/Shell.vue",
"src/components/prompts/Copy.vue", "src/components/prompts/Copy.vue",
"src/components/prompts/Move.vue",
"src/components/prompts/Delete.vue", "src/components/prompts/Delete.vue",
"src/components/prompts/FileList.vue", "src/components/prompts/FileList.vue",
"src/components/prompts/Rename.vue", "src/components/prompts/Rename.vue",