From 1b8463aca302e7805740db835326eef7e1ee0892 Mon Sep 17 00:00:00 2001 From: Hangzhi Yu <hangzhi@protonmail.com> Date: Wed, 15 May 2024 02:34:44 +0200 Subject: [PATCH] Drop iterator mixin --- .../components/SubstitutionOverview.vue | 6 +- .../mixins/dateSortedIteratorMixin.js | 206 ------------------ 2 files changed, 2 insertions(+), 210 deletions(-) delete mode 100644 aleksis/apps/chronos/frontend/mixins/dateSortedIteratorMixin.js diff --git a/aleksis/apps/chronos/frontend/components/SubstitutionOverview.vue b/aleksis/apps/chronos/frontend/components/SubstitutionOverview.vue index 84607ef5..e3206e12 100644 --- a/aleksis/apps/chronos/frontend/components/SubstitutionOverview.vue +++ b/aleksis/apps/chronos/frontend/components/SubstitutionOverview.vue @@ -8,8 +8,6 @@ import { createOrUpdateSubstitutions, groupsByOwner, } from "./amendLesson.graphql"; - -import dateSortedIteratorMixin from "../mixins/dateSortedIteratorMixin.js"; </script> <template> @@ -26,6 +24,7 @@ import dateSortedIteratorMixin from "../mixins/dateSortedIteratorMixin.js"; :enable-edit="true" :elevated="false" :force-model-item-update="true" + ref="iterator" > <template #additionalActions="{ attrs, on }"> <v-autocomplete @@ -104,7 +103,6 @@ import dateSortedIteratorMixin from "../mixins/dateSortedIteratorMixin.js"; <script> export default { name: "SubstitutionOverview", - mixins: [dateSortedIteratorMixin], props: { objId: { type: [Number, String], @@ -131,7 +129,7 @@ export default { }, hash: this.$route.hash, }); - this.resetDate(); + this.$refs.iterator.resetDate(); }, }, computed: { diff --git a/aleksis/apps/chronos/frontend/mixins/dateSortedIteratorMixin.js b/aleksis/apps/chronos/frontend/mixins/dateSortedIteratorMixin.js deleted file mode 100644 index 399351aa..00000000 --- a/aleksis/apps/chronos/frontend/mixins/dateSortedIteratorMixin.js +++ /dev/null @@ -1,206 +0,0 @@ -import { DateTime } from "luxon"; - -export default { - props: { - /** - * Number of consecutive days to load at once - * This number of days is initially loaded and loaded - * incrementally while scrolling. - */ - dayIncrement: { - type: Number, - required: false, - default: 7, - }, - /** - * Margin from list to top of viewport in pixels - */ - topMargin: { - type: Number, - required: false, - default: 165, - }, - }, - data() { - return { - lastQuery: null, - dateStart: "", - dateEnd: "", - ready: false, - initDate: false, - currentDate: "", - hashUpdater: false, - }; - }, - methods: { - resetDate(toDate) { - // Assure current date - console.log("Resetting date range", this.$route.hash); - this.currentDate = toDate || this.$route.hash?.substring(1); - if (!this.currentDate) { - console.log("Set default date"); - this.setDate(DateTime.now().toISODate()); - } - - const date = DateTime.fromISO(this.currentDate); - this.initDate = date; - this.dateStart = date.minus({ days: this.dayIncrement }).toISODate(); - this.dateEnd = date.plus({ days: this.dayIncrement }).toISODate(); - }, - transition() { - this.initDate = false; - this.ready = true; - }, - groupItemsByDay(items) { - // => {dt: {date: dt, itemsOfDay: doc ...} ...} - const itemsByDay = items.reduce((byDay, item) => { - // This works with dummy. Does actual doc have dateStart instead? - const day = DateTime.fromISO(item.datetimeStart).startOf("day"); - byDay[day] ??= { date: day, itemsOfDay: [] }; - byDay[day].itemsOfDay.push(item); - return byDay; - }, {}); - // => [{date: dt, itemsOfDay: doc ..., idx: idx, lastIdx: last-idx} ...] - // sorting is necessary since backend can send items unordered - return Object.keys(itemsByDay) - .sort() - .map((key, idx, { length }) => { - const day = itemsByDay[key]; - day.first = idx === 0; - const lastIdx = length - 1; - day.last = idx === lastIdx; - return day; - }); - }, - fetchMore(from, to, then) { - console.log("fetching", from, to); - this.lastQuery.fetchMore({ - variables: { - dateStart: from, - dateEnd: to, - }, - // Transform the previous result with new data - updateQuery: (previousResult, { fetchMoreResult }) => { - console.log("Received more"); - then(); - return { items: previousResult.items.concat(fetchMoreResult.items) }; - }, - }); - }, - setDate(date) { - this.currentDate = date; - if (!this.hashUpdater) { - this.hashUpdater = window.requestIdleCallback(() => { - if (!(this.$route.hash.substring(1) === this.currentDate)) { - this.$router.replace({ hash: this.currentDate }); - } - this.hashUpdater = false; - }); - } - }, - fixScrollPos(height, top) { - this.$nextTick(() => { - if (height < document.documentElement.scrollHeight) { - document.documentElement.scrollTop = - document.documentElement.scrollHeight - height + top; - this.ready = true; - } else { - // Update top, could have changed in the meantime. - this.fixScrollPos(height, document.documentElement.scrollTop); - } - }); - }, - intersectHandler(date, first, last) { - let once = true; - return (entries) => { - const entry = entries[0]; - if (entry.isIntersecting) { - if (entry.boundingClientRect.top <= this.topMargin || first) { - console.log("@ ", date.toISODate()); - this.setDate(date.toISODate()); - } - - if (once && this.ready && first) { - console.log("load up", date.toISODate()); - this.ready = false; - this.fetchMore( - date.minus({ days: this.dayIncrement }).toISODate(), - date.minus({ days: 1 }).toISODate(), - () => { - this.fixScrollPos( - document.documentElement.scrollHeight, - document.documentElement.scrollTop, - ); - }, - ); - once = false; - } else if (once && this.ready && last) { - console.log("load down", date.toISODate()); - this.ready = false; - this.fetchMore( - date.plus({ days: 1 }).toISODate(), - date.plus({ days: this.dayIncrement }).toISODate(), - () => { - this.ready = true; - }, - ); - once = false; - } - } - }; - }, - // Improve me? - // The navigation logic could be a bit simpler if the current days - // where known as a sorted array (= result of groupItemsByDay) But - // then the list would need its own component and this gets rather - // complicated. Then the calendar could also show the present days - // / gray out the missing. - // - // Next two: arg date is ts object - findPrev(date) { - return this.$refs.days - .map((day) => day.date) - .sort() - .reverse() - .find((date2) => date2 < date); - }, - findNext(date) { - return this.$refs.days - .map((day) => day.date) - .sort() - .find((date2) => date2 > date); - }, - gotoDate(date) { - const present = this.$refs.days.find( - (day) => day.date.toISODate() === date, - ); - - if (present) { - // React immediatly -> smoother navigation - // Also intersect handler does not always react to scrollIntoView - this.setDate(date); - present.focus("smooth"); - } else if ( - !this.findPrev(DateTime.fromISO(date)) || - !this.findNext(DateTime.fromISO(date)) - ) { - this.resetDate(date); - } - }, - gotoPrev() { - const prev = this.findPrev(DateTime.fromISO(this.currentDate)); - if (prev) { - this.gotoDate(prev.toISODate()); - } - }, - gotoNext() { - const next = this.findNext(DateTime.fromISO(this.currentDate)); - if (next) { - this.gotoDate(next.toISODate()); - } - }, - }, - created() { - this.resetDate(); - }, -}; -- GitLab