Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement autosave #67

Closed
wants to merge 5 commits into from
Closed
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions client/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
"buffer": "^6.0.3",
"luxon": "^3.3.0",
"stream-browserify": "^3.0.0",
"throttle-debounce": "^5.0.0",
"vue": "^3.2.45",
"vue-i18n": "^9.2.2",
"vue-router": "^4.1.5",
Expand All @@ -35,6 +36,7 @@
"@types/bootstrap": "^5.2.6",
"@types/luxon": "^3.2.0",
"@types/node": "^18.15.1",
"@types/throttle-debounce": "^5.0.0",
"@vitejs/plugin-vue": "^4.1.0",
"@vue/eslint-config-prettier": "^7.1.0",
"@vue/eslint-config-typescript": "^11.0.0",
Expand Down
1 change: 1 addition & 0 deletions client/src/types/throttle-debounce.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
declare module "throttle-debounce";
36 changes: 27 additions & 9 deletions client/src/views/PostFormView.vue
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
<div class="col w-50">
<div class="card flex-md-row mb-4 box-shadow h-md-250">
<div class="card-body">
<form @submit="submitForm($event)">
<form @submit.prevent="submitForm" @input="handleAutoSave">
<div class="form-floating mb-3">
<input v-model="form.title" type="text" class="form-control" id="title" placeholder="Titel" required />
<label for="title">{{ t("posts.form.title") }}</label>
Expand Down Expand Up @@ -207,7 +207,6 @@ import ImagePreview from "@client/components/ImagePreview.vue";
import LoadingSpinner from "@client/components/LoadingSpinner.vue";
import MarkDown from "@client/components/MarkDown.vue";
import PostNotAvailable from "@client/components/PostNotAvailable.vue";
import { debounce } from "@client/debounce.js";
import { PostEndpoints } from "@client/util/api-client.js";
import { faUpload } from "@fortawesome/free-solid-svg-icons";
import { t, tc } from "@fumix/fu-blog-client/src/plugins/i18n.js";
Expand All @@ -216,6 +215,7 @@ import { bytesToBase64URL, convertToHumanReadableFileSize } from "@fumix/fu-blog
import { computed, onMounted, reactive, ref } from "vue";
import { useRoute, useRouter } from "vue-router";
import Vue3TagsInput from "vue3-tags-input";
import { debounce } from "throttle-debounce";

const md = ref<string | null>(null);
const loading = ref<boolean>(false);
Expand All @@ -224,6 +224,8 @@ const dropzoneHighlight = ref<boolean>(false);
const router = useRouter();
const markdownArea = ref(null);
const postHasError = ref<boolean>(false);
const autoSavedDraftId = ref<number | undefined>(undefined);
const canBeAutosaved = ref<boolean>(true);

const form = reactive<NewPostRequestDto>({
title: "",
Expand Down Expand Up @@ -251,6 +253,8 @@ onMounted(async () => {
// prefill form with values fom loaded post
if (props.postId) {
try {
// avoid autosaving in edit mode
canBeAutosaved.value = false;
const res = await fetch(`/api/posts/${props.postId}`);
const resJson = (await res.json())?.data as Post;
form.title = resJson.title;
Expand All @@ -265,10 +269,10 @@ onMounted(async () => {
}
}

debounce(() => {
debounce(1000, () => {
loading.value = true;
md.value = form.markdown;
}, 1000);
});
});

const pasteImageFileToMarkdown = (markdown: string) => {
Expand Down Expand Up @@ -354,9 +358,14 @@ const addFile = (file: File) => {
.catch((it) => console.error("Failed to calculate SHA-256 hash!"));
};

const submitForm = (e: Event) => {
e.preventDefault();
send(props.postId);
const handleAutoSave = debounce(1000, () => {
if (canBeAutosaved.value) {
send(props.postId, true);
}
});

const submitForm = () => {
send(props.postId, false);
};

const insertIntoTextarea = (
Expand All @@ -372,10 +381,19 @@ const insertIntoTextarea = (
return before + insertedText + after;
};

const send = async (id: number | undefined) => {
const send = async (id: number | undefined, autosave: boolean) => {
form.draft = autosave;
const successAction = (r: DraftResponseDto) => {
router.push(`/posts/post/${r.postId}`);
if (!autosave) {
router.push(`/posts/post/${r.postId}`);
} else {
// no routing when autosaving
autoSavedDraftId.value = r.postId;
}
};
if (autoSavedDraftId.value) {
id = autoSavedDraftId.value;
}
if (!id) {
await PostEndpoints.createPost(form, Object.values(files))
.then(successAction)
Expand Down
28 changes: 22 additions & 6 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

7 changes: 2 additions & 5 deletions server/src/routes/posts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,13 +24,10 @@ router.get("/page/:page([0-9]+)/count/:count([0-9]+)/search/:search/operator/:op
const skipEntries = page * itemsPerPage - itemsPerPage;
let searchTerm = "";
if (req.params.search) {
const splitSearchParams: string[] = req.params.search.trim().split(" ");
const splitSearchParams: string[] = decodeURI(req.params.search).trim().split(" ");
const operator = req.params.operator === "or" ? " | " : " & ";

searchTerm = splitSearchParams
.map((word) => escape(word))
.filter(Boolean)
.join(operator);
searchTerm = splitSearchParams.filter(Boolean).join(operator);
}

// TODO : add createdBy and tags to search results
Expand Down