memories/src/components/ScrollerManager.vue

649 lines
16 KiB
Vue
Raw Normal View History

2022-10-14 23:29:20 +00:00
<template>
2022-10-28 19:08:34 +00:00
<div
class="scroller"
ref="scroller"
v-bind:class="{
2022-10-29 23:47:37 +00:00
'scrolling-recycler-now': scrollingRecyclerNowTimer,
2022-10-30 00:11:12 +00:00
'scrolling-recycler': scrollingRecyclerTimer,
'scrolling-now': scrollingNowTimer,
scrolling: scrollingTimer,
2022-10-28 19:08:34 +00:00
}"
2022-10-30 00:17:34 +00:00
@mousemove.passive="mousemove"
@mouseleave.passive="mouseleave"
@mousedown.passive="mousedown"
2022-12-06 20:32:52 +00:00
@mouseup.passive="interactend"
@touchmove.prevent="touchmove"
@touchstart.passive="interactstart"
@touchend.passive="interactend"
@touchcancel.passive="interactend"
2022-10-28 19:08:34 +00:00
>
<span
class="cursor st"
ref="cursorSt"
:style="{ transform: `translateY(${cursorY}px)` }"
>
</span>
<span
class="cursor hv"
:style="{ transform: `translateY(${hoverCursorY}px)` }"
2022-12-06 20:28:02 +00:00
@touchmove.prevent="touchmove"
2022-12-06 20:32:52 +00:00
@touchstart.passive="interactstart"
@touchend.passive="interactend"
@touchcancel.passive="interactend"
2022-10-28 19:08:34 +00:00
>
<div class="text">{{ hoverCursorText }}</div>
<div class="icon"><ScrollIcon :size="22" /></div>
</span>
<div
v-for="tick of visibleTicks"
:key="tick.key"
class="tick"
:class="{ dash: !tick.text }"
:style="{ transform: `translateY(calc(${tick.top}px - 50%))` }"
>
<span v-if="tick.text">{{ tick.text }}</span>
2022-10-14 23:29:20 +00:00
</div>
2022-10-28 19:08:34 +00:00
</div>
2022-10-14 23:29:20 +00:00
</template>
<script lang="ts">
2022-12-06 20:32:52 +00:00
import { Component, Mixins, Prop } from "vue-property-decorator";
2022-10-28 19:08:34 +00:00
import { IRow, IRowType, ITick } from "../types";
import GlobalMixin from "../mixins/GlobalMixin";
import ScrollIcon from "vue-material-design-icons/UnfoldMoreHorizontal.vue";
2022-10-14 23:29:20 +00:00
import * as utils from "../services/Utils";
2022-12-06 20:40:56 +00:00
// Pixels to snap at
const SNAP_OFFSET = -35;
2022-10-14 23:29:20 +00:00
@Component({
2022-10-28 19:08:34 +00:00
components: {
ScrollIcon,
},
2022-10-14 23:29:20 +00:00
})
export default class ScrollerManager extends Mixins(GlobalMixin) {
2022-10-28 19:08:34 +00:00
/** Rows from Timeline */
@Prop() rows!: IRow[];
/** Total height */
@Prop() height!: number;
/** Actual recycler component */
@Prop() recycler!: any;
/** Recycler before slot component */
@Prop() recyclerBefore!: any;
/** Last known height at adjustment */
private lastAdjustHeight = 0;
/** Height of the entire photo view */
private recyclerHeight: number = 100;
2022-10-30 00:11:12 +00:00
/** Rect of scroller */
private scrollerRect: DOMRect = null;
2022-10-28 19:08:34 +00:00
/** Computed ticks */
private ticks: ITick[] = [];
/** Computed cursor top */
private cursorY = 0;
/** Hover cursor top */
private hoverCursorY = -5;
/** Hover cursor text */
private hoverCursorText = "";
2022-10-30 00:11:12 +00:00
/** Scrolling using the scroller */
private scrollingTimer = 0;
/** Scrolling now using the scroller */
private scrollingNowTimer = 0;
/** Scrolling recycler */
2022-10-29 23:47:37 +00:00
private scrollingRecyclerTimer = 0;
2022-10-30 00:11:12 +00:00
/** Scrolling recycler now */
2022-10-29 23:47:37 +00:00
private scrollingRecyclerNowTimer = 0;
/** Recycler scrolling throttle */
private scrollingRecyclerUpdateTimer = 0;
2022-10-28 19:08:34 +00:00
/** View size reflow timer */
private reflowRequest = false;
2022-10-28 19:08:34 +00:00
/** Tick adjust timer */
private adjustRequest = false;
2022-12-06 20:32:52 +00:00
/** Scroller is being moved with interaction */
private interacting = false;
/** Track the last requested y position when interacting */
private lastRequestedRecyclerY = 0;
2022-10-28 19:08:34 +00:00
/** Get the visible ticks */
get visibleTicks() {
2022-11-23 10:10:00 +00:00
let key = 9999999900;
2022-10-28 19:08:34 +00:00
return this.ticks
.filter((tick) => tick.s)
.map((tick) => {
if (tick.text) {
tick.key = key = tick.dayId * 100;
2022-10-22 17:15:28 +00:00
} else {
2022-10-28 19:08:34 +00:00
tick.key = ++key; // days are sorted descending
2022-10-22 17:15:28 +00:00
}
2022-10-28 19:08:34 +00:00
return tick;
});
}
/** Reset state */
public reset() {
this.ticks = [];
this.cursorY = 0;
this.hoverCursorY = -5;
this.hoverCursorText = "";
this.reflowRequest = false;
2022-10-30 00:11:12 +00:00
// Clear all timers
clearTimeout(this.scrollingTimer);
clearTimeout(this.scrollingNowTimer);
clearTimeout(this.scrollingRecyclerTimer);
clearTimeout(this.scrollingRecyclerNowTimer);
clearTimeout(this.scrollingRecyclerUpdateTimer);
this.scrollingTimer = 0;
this.scrollingNowTimer = 0;
this.scrollingRecyclerTimer = 0;
this.scrollingRecyclerNowTimer = 0;
this.scrollingRecyclerUpdateTimer = 0;
2022-10-28 19:08:34 +00:00
}
/** Recycler scroll event, must be called by timeline */
public recyclerScrolled() {
2022-10-30 00:11:12 +00:00
// This isn't a renewing timer, it's a scheduled task
2022-10-29 23:47:37 +00:00
if (this.scrollingRecyclerUpdateTimer) return;
this.scrollingRecyclerUpdateTimer = window.setTimeout(() => {
this.scrollingRecyclerUpdateTimer = 0;
this.updateFromRecyclerScroll();
2022-10-29 23:47:37 +00:00
}, 100);
2022-10-30 00:11:12 +00:00
// Update that we're scrolling with the recycler
utils.setRenewingTimeout(this, "scrollingRecyclerNowTimer", null, 200);
utils.setRenewingTimeout(this, "scrollingRecyclerTimer", null, 1500);
2022-10-29 23:15:18 +00:00
}
/** Update cursor position from recycler scroll position */
public updateFromRecyclerScroll() {
2022-12-06 20:32:52 +00:00
// Ignore if not initialized or moving
if (!this.ticks.length || this.interacting) return;
2022-10-28 19:08:34 +00:00
// Get the scroll position
const scroll = this.recycler?.$el?.scrollTop || 0;
// Get cursor px position
const { top1, top2, y1, y2 } = this.getCoords(scroll, "y");
const topfrac = (scroll - y1) / (y2 - y1);
2022-11-03 23:03:50 +00:00
const rtop = top1 + (top2 - top1) * (topfrac || 0);
2022-10-28 19:08:34 +00:00
// Always move static cursor to right position
this.cursorY = rtop;
// Move hover cursor to same position unless hovering
// Regardless, we need this call because the internal mapping might have changed
if ((<HTMLElement>this.$refs.scroller).matches(":hover")) {
this.moveHoverCursor(this.hoverCursorY);
} else {
this.moveHoverCursor(rtop);
2022-10-16 02:55:53 +00:00
}
2022-10-28 19:08:34 +00:00
}
/** Re-create tick data in the next frame */
public async reflow() {
if (this.reflowRequest) return;
this.reflowRequest = true;
await this.$nextTick();
this.reflowNow();
this.reflowRequest = false;
}
/** Re-create tick data */
private reflowNow() {
// Ignore if not initialized
if (!this.recycler?.$refs.wrapper) return;
// Refresh height of recycler
this.recyclerHeight = this.recycler.$refs.wrapper.clientHeight;
// Recreate ticks data
this.recreate();
// Adjust top
this.adjustNow();
}
/** Recreate from scratch */
private recreate() {
// Clear and override any adjust timer
this.ticks = [];
// Ticks
let prevYear = 9999;
let prevMonth = 0;
// Get a new tick
const getTick = (
dayId: number,
isMonth = false,
text?: string | number
): ITick => {
return {
dayId,
isMonth,
text,
y: 0,
count: 0,
topF: 0,
top: 0,
s: false,
};
};
// Iterate over rows
for (const row of this.rows) {
if (row.type === IRowType.HEAD) {
// Create tick
if (this.TagDayIDValueSet.has(row.dayId)) {
// Blank tick
this.ticks.push(getTick(row.dayId));
} else {
// Make date string
const dateTaken = utils.dayIdToDate(row.dayId);
// Create tick
const dtYear = dateTaken.getUTCFullYear();
const dtMonth = dateTaken.getUTCMonth();
const isMonth = dtMonth !== prevMonth || dtYear !== prevYear;
2022-11-03 23:03:50 +00:00
const text = dtYear === prevYear ? undefined : dtYear;
2022-10-28 19:08:34 +00:00
this.ticks.push(getTick(row.dayId, isMonth, text));
prevMonth = dtMonth;
prevYear = dtYear;
2022-10-21 06:48:28 +00:00
}
2022-10-28 19:08:34 +00:00
}
2022-10-21 06:48:28 +00:00
}
2022-10-28 19:08:34 +00:00
}
/**
* Update tick positions without truncating the list
* This is much cheaper than reflowing the whole thing
*/
public async adjust() {
if (this.adjustRequest) return;
this.adjustRequest = true;
await this.$nextTick();
this.adjustNow();
this.adjustRequest = false;
}
/** Do adjustment synchronously */
private adjustNow() {
// Refresh height of recycler
this.recyclerHeight = this.recycler.$refs.wrapper.clientHeight;
const extraY = this.recyclerBefore?.clientHeight || 0;
// Start with the first tick. Walk over all rows counting the
// y position. When you hit a row with the tick, update y and
// top values and move to the next tick.
let tickId = 0;
let y = extraY;
let count = 0;
// We only need to recompute top and visible ticks if count
// of some tick has changed.
let needRecomputeTop = false;
// Check if height changed
if (this.lastAdjustHeight !== this.height) {
needRecomputeTop = true;
this.lastAdjustHeight = this.height;
2022-10-21 06:48:28 +00:00
}
2022-10-28 19:08:34 +00:00
for (const row of this.rows) {
// Check if tick is valid
if (tickId >= this.ticks.length) break;
2022-10-21 06:48:28 +00:00
2022-10-28 19:08:34 +00:00
// Check if we hit the next tick
const tick = this.ticks[tickId];
if (tick.dayId === row.dayId) {
tick.y = y;
2022-10-21 06:48:28 +00:00
2022-10-28 19:08:34 +00:00
// Check if count has changed
needRecomputeTop ||= tick.count !== count;
tick.count = count;
2022-10-14 23:29:20 +00:00
2022-10-28 19:08:34 +00:00
// Move to next tick
count += row.day.count;
tickId++;
}
2022-10-21 06:48:28 +00:00
2022-10-28 19:08:34 +00:00
y += row.size;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
// Compute visible ticks
if (needRecomputeTop) {
this.setTicksTop(count);
this.computeVisibleTicks();
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
}
/** Mark ticks as visible or invisible */
private computeVisibleTicks() {
2022-10-30 00:11:12 +00:00
// Kind of unrelated here, but refresh rect
this.scrollerRect = (
this.$refs.scroller as HTMLElement
).getBoundingClientRect();
2022-10-28 19:08:34 +00:00
// Do another pass to figure out which points are visible
// This is not as bad as it looks, it's actually 12*O(n)
// because there are only 12 months in a year
const fontSizePx = parseFloat(
getComputedStyle(this.$refs.cursorSt as any).fontSize
);
2022-11-23 11:16:45 +00:00
const minGap = fontSizePx + (globalThis.windowInnerWidth <= 768 ? 5 : 2);
2022-10-28 19:08:34 +00:00
let prevShow = -9999;
for (const [idx, tick] of this.ticks.entries()) {
// Conservative
tick.s = false;
// These aren't for showing
if (!tick.isMonth) continue;
// You can't see these anyway, why bother?
if (tick.top < minGap || tick.top > this.height - minGap) continue;
// Will overlap with the previous tick. Skip anyway.
if (tick.top - prevShow < minGap) continue;
// This is a labelled tick then show it anyway for the sake of best effort
if (tick.text) {
prevShow = tick.top;
tick.s = true;
continue;
}
// Lookahead for next labelled tick
// If showing this tick would overlap the next one, don't show this one
let i = idx + 1;
while (i < this.ticks.length) {
if (this.ticks[i].text) {
break;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
i++;
}
if (i < this.ticks.length) {
// A labelled tick was found
const nextLabelledTick = this.ticks[i];
if (
tick.top + minGap > nextLabelledTick.top &&
nextLabelledTick.top < this.height - minGap
) {
// make sure this will be shown
continue;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
}
2022-10-14 23:29:20 +00:00
2022-10-28 19:08:34 +00:00
// Show this tick
tick.s = true;
prevShow = tick.top;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
}
2022-10-14 23:29:20 +00:00
2022-10-28 19:08:34 +00:00
private setTicksTop(total: number) {
for (const tick of this.ticks) {
tick.topF = this.height * (tick.count / total);
tick.top = utils.roundHalf(tick.topF);
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
}
/** Change actual position of the hover cursor */
private moveHoverCursor(y: number) {
this.hoverCursorY = y;
// Get index of previous tick
let idx = utils.binarySearch(this.ticks, y, "topF");
if (idx === 0) {
// use this tick
} else if (idx >= 1 && idx <= this.ticks.length) {
idx = idx - 1;
} else {
return;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
// DayId of current hover
const dayId = this.ticks[idx]?.dayId;
2022-10-21 06:48:28 +00:00
2022-10-28 19:08:34 +00:00
// Special days
if (dayId === undefined || this.TagDayIDValueSet.has(dayId)) {
this.hoverCursorText = "";
return;
2022-10-21 06:48:28 +00:00
}
2022-10-28 19:08:34 +00:00
const date = utils.dayIdToDate(dayId);
this.hoverCursorText = utils.getShortDateStr(date);
}
2022-10-21 06:48:28 +00:00
2022-10-28 19:08:34 +00:00
/** Handle mouse hover */
private mousemove(event: MouseEvent) {
if (event.buttons) {
this.mousedown(event);
2022-10-21 05:24:00 +00:00
}
2022-10-28 19:08:34 +00:00
this.moveHoverCursor(event.offsetY);
}
/** Handle mouse leave */
private mouseleave() {
this.interactend();
2022-10-28 19:08:34 +00:00
this.moveHoverCursor(this.cursorY);
}
/** Binary search and get coords surrounding position */
private getCoords(y: number, field: "topF" | "y") {
// Top of first and second ticks
let top1 = 0,
top2 = 0,
y1 = 0,
y2 = 0;
// Get index of previous tick
let idx = utils.binarySearch(this.ticks, y, field);
if (idx <= 0) {
top1 = 0;
top2 = this.ticks[0].topF;
y1 = 0;
y2 = this.ticks[0].y;
} else if (idx >= this.ticks.length) {
const t = this.ticks[this.ticks.length - 1];
top1 = t.topF;
top2 = this.height;
y1 = t.y;
y2 = this.recyclerHeight;
} else {
const t1 = this.ticks[idx - 1];
const t2 = this.ticks[idx];
top1 = t1.topF;
top2 = t2.topF;
y1 = t1.y;
y2 = t2.y;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
return { top1, top2, y1, y2 };
}
/** Move to given scroller Y */
2022-12-06 20:32:52 +00:00
private moveto(y: number, snap: boolean) {
2022-10-30 00:11:12 +00:00
// Move cursor immediately to prevent jank
this.cursorY = y;
this.hoverCursorY = y;
2022-10-28 19:08:34 +00:00
const { top1, top2, y1, y2 } = this.getCoords(y, "topF");
const yfrac = (y - top1) / (top2 - top1);
2022-11-03 23:03:50 +00:00
const ry = y1 + (y2 - y1) * (yfrac || 0);
2022-12-06 20:40:56 +00:00
const targetY = snap ? y1 + SNAP_OFFSET : ry;
if (this.lastRequestedRecyclerY !== targetY) {
this.lastRequestedRecyclerY = targetY;
this.recycler.scrollToPosition(targetY);
}
2022-10-28 19:08:34 +00:00
this.handleScroll();
}
/** Handle mouse click */
private mousedown(event: MouseEvent) {
2022-12-06 20:32:52 +00:00
this.interactstart(); // end called on mouseup
this.moveto(event.offsetY, false);
2022-10-28 19:08:34 +00:00
}
/** Handle touch */
private touchmove(event: any) {
2022-10-30 00:11:12 +00:00
const y = event.targetTouches[0].pageY - this.scrollerRect.top;
2022-12-06 20:32:52 +00:00
this.moveto(y, true);
}
private interactstart() {
this.interacting = true;
}
private interactend() {
this.interacting = false;
this.recyclerScrolled(); // make sure final position is correct
2022-10-28 19:08:34 +00:00
}
2022-10-30 00:11:12 +00:00
/** Update scroller is being used to scroll recycler */
2022-10-28 19:08:34 +00:00
private handleScroll() {
2022-10-30 00:11:12 +00:00
utils.setRenewingTimeout(this, "scrollingNowTimer", null, 200);
utils.setRenewingTimeout(this, "scrollingTimer", null, 1500);
2022-10-28 19:08:34 +00:00
}
2022-10-14 23:29:20 +00:00
}
</script>
<style lang="scss" scoped>
@mixin phone {
2022-10-28 19:08:34 +00:00
@media (max-width: 768px) {
@content;
}
2022-10-14 23:29:20 +00:00
}
.scroller {
2022-12-05 03:49:16 +00:00
contain: layout style;
2022-10-28 19:08:34 +00:00
overflow-y: clip;
position: absolute;
height: 100%;
width: 36px;
top: 0;
right: 0;
cursor: ns-resize;
opacity: 0;
transition: opacity 0.2s ease-in-out;
// Show on hover or scroll of main window
&:hover,
&.scrolling-recycler {
opacity: 1;
}
> .tick {
pointer-events: none;
position: absolute;
font-size: 0.75em;
line-height: 0.75em;
font-weight: 600;
opacity: 0.95;
right: 9px;
top: 0;
transition: transform 0.2s linear;
z-index: 1;
&.dash {
height: 4px;
width: 4px;
border-radius: 50%;
background-color: var(--color-main-text);
opacity: 0.15;
display: block;
@include phone {
display: none;
}
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
@include phone {
background-color: var(--color-main-background);
padding: 4px;
border-radius: 4px;
}
}
2022-10-14 23:29:20 +00:00
2022-10-28 19:08:34 +00:00
> .cursor {
position: absolute;
pointer-events: none;
right: 0;
background-color: var(--color-primary);
min-width: 100%;
min-height: 1.5px;
will-change: transform;
&.st {
font-size: 0.75em;
opacity: 0;
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
&.hv {
background-color: var(--color-main-background);
padding: 2px 5px;
border-top: 2px solid var(--color-primary);
border-radius: 2px;
width: auto;
white-space: nowrap;
z-index: 100;
font-size: 0.95em;
font-weight: 600;
> .icon {
display: none;
transform: translate(-16px, 6px);
}
2022-10-14 23:29:20 +00:00
}
2022-10-28 19:08:34 +00:00
}
2022-10-30 00:11:12 +00:00
&.scrolling-recycler-now:not(.scrolling-now) > .cursor {
2022-10-29 23:47:37 +00:00
transition: transform 0.1s linear;
}
2022-10-29 23:15:18 +00:00
&:hover > .cursor {
2022-10-30 00:11:12 +00:00
transition: none !important;
2022-10-29 23:15:18 +00:00
&.st {
opacity: 1;
}
2022-10-28 19:08:34 +00:00
}
2022-12-06 20:06:02 +00:00
// Hide ticks on mobile unless hovering
@include phone {
// Shift pointer events to hover cursor
pointer-events: none;
.cursor.hv {
pointer-events: all;
}
> .tick {
right: 40px;
}
&:not(.scrolling) {
> .tick {
display: none;
}
}
.cursor.hv {
left: 5px;
border: none;
box-shadow: 0 0 5px -3px #000;
height: 40px;
width: 70px;
border-radius: 20px;
> .text {
display: none;
}
> .icon {
display: block;
}
}
.cursor.st {
display: none;
}
}
2022-10-14 23:29:20 +00:00
}
</style>