memories/src/components/ScrollerManager.vue

465 lines
14 KiB
Vue
Raw Normal View History

2022-10-14 23:29:20 +00:00
<template>
<div class="scroller"
v-bind:class="{
'scrolling-recycler': scrollingRecycler,
'scrolling': scrolling,
}"
@mousemove="mousemove"
@touchmove="touchmove"
@mouseleave="mouseleave"
@mousedown="mousedown">
<span class="cursor st" ref="cursorSt"
:style="{ transform: `translateY(${cursorY}px)` }">
</span>
<span class="cursor hv"
:style="{ transform: `translateY(${hoverCursorY}px)` }">
<div class="text"> {{ hoverCursorText }} </div>
<div class="icon"> <ScrollIcon :size="20" /> </div>
</span>
<div v-for="tick of visibleTicks" :key="tick.dayId"
class="tick"
:class="{ 'dash': !tick.text }"
2022-10-15 17:41:49 +00:00
:style="{ top: tick.top + 'px' }">
2022-10-14 23:29:20 +00:00
<span v-if="tick.text">{{ tick.text }}</span>
</div>
</div>
</template>
<script lang="ts">
import { Component, Mixins, Prop } from 'vue-property-decorator';
2022-10-15 18:18:32 +00:00
import { IRow, IRowType, ITick } from '../types';
2022-10-14 23:29:20 +00:00
import GlobalMixin from '../mixins/GlobalMixin';
import ScrollIcon from 'vue-material-design-icons/UnfoldMoreHorizontal.vue';
import * as utils from "../services/Utils";
@Component({
components: {
ScrollIcon,
},
})
export default class ScrollerManager extends Mixins(GlobalMixin) {
2022-10-15 17:41:49 +00:00
/** Rows from Timeline */
@Prop() rows!: IRow[];
2022-10-14 23:29:20 +00:00
/** Total height */
@Prop() height!: number;
/** Actual recycler component */
@Prop() recycler!: any;
/** Recycler before slot component */
@Prop() recyclerBefore!: any;
/** Height of the entire photo view */
private recyclerHeight!: number;
/** Computed ticks */
private ticks: ITick[] = [];
/** Computed cursor top */
private cursorY = 0;
/** Hover cursor top */
private hoverCursorY = -5;
/** Hover cursor text */
private hoverCursorText = "";
/** Scrolling currently */
private scrolling = false;
/** Scrolling timer */
private scrollingTimer = null as number | null;
/** Scrolling recycler currently */
private scrollingRecycler = false;
/** Scrolling recycler timer */
private scrollingRecyclerTimer = null as number | null;
/** View size reflow timer */
private reflowRequest = false;
2022-10-16 02:55:53 +00:00
/** Tick adjust timer */
private adjustTimer = null as number | null;
2022-10-14 23:29:20 +00:00
/** Get the visible ticks */
get visibleTicks() {
return this.ticks.filter(tick => tick.s);
}
/** Reset state */
public reset() {
this.ticks = [];
this.cursorY = 0;
this.hoverCursorY = -5;
this.hoverCursorText = "";
this.scrolling = false;
this.scrollingTimer = null;
this.reflowRequest = false;
}
/** Recycler scroll event, must be called by timeline */
public recyclerScrolled(event?: any) {
2022-10-16 18:46:26 +00:00
// Ignore if not initialized
if (!this.ticks.length) return;
// Move hover cursor to px position
2022-10-16 02:55:53 +00:00
this.cursorY = utils.roundHalf(event ? event.target.scrollTop * this.height / this.recyclerHeight : 0);
2022-10-14 23:29:20 +00:00
this.moveHoverCursor(this.cursorY);
2022-10-16 18:46:26 +00:00
// Show the scroller for some time
2022-10-14 23:29:20 +00:00
if (this.scrollingRecyclerTimer) window.clearTimeout(this.scrollingRecyclerTimer);
this.scrollingRecycler = true;
this.scrollingRecyclerTimer = window.setTimeout(() => {
this.scrollingRecycler = false;
this.scrollingRecyclerTimer = null;
}, 1500);
}
/** Re-create tick data in the next frame */
2022-10-15 17:41:49 +00:00
public async reflow() {
2022-10-14 23:29:20 +00:00
if (this.reflowRequest) {
return;
}
this.reflowRequest = true;
await this.$nextTick();
2022-10-15 17:41:49 +00:00
this.reflowNow();
2022-10-14 23:29:20 +00:00
this.reflowRequest = false;
}
2022-10-16 02:55:53 +00:00
private setTickTop(tick: ITick) {
const extraY = this.recyclerBefore?.clientHeight || 0;
tick.topF = (extraY + tick.y) * (this.height / this.recyclerHeight);
tick.top = utils.roundHalf(tick.topF);
}
2022-10-14 23:29:20 +00:00
/** Re-create tick data */
2022-10-15 17:41:49 +00:00
private reflowNow() {
2022-10-16 18:46:26 +00:00
// Ignore if not initialized
if (!this.recycler?.$refs.wrapper) return;
2022-10-16 02:55:53 +00:00
// Refresh height of recycler
this.recyclerHeight = this.recycler.$refs.wrapper.clientHeight;
2022-10-15 17:41:49 +00:00
// Recreate ticks data
this.recreate();
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
// Recompute which ticks are visible
this.computeVisibleTicks();
}
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
/** Recreate from scratch */
private recreate() {
// Clear
this.ticks = [];
// Ticks
let y = 0;
let prevYear = 9999;
let prevMonth = 0;
const thisYear = new Date().getFullYear();
// Get a new tick
const getTick = (dayId: number, text?: string | number): ITick => {
const tick = {
dayId,
y: y,
text,
topF: 0,
top: 0,
s: false,
};
this.setTickTop(tick);
return tick;
}
// Itearte over rows
for (const row of this.rows) {
if (row.type === IRowType.HEAD) {
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 if month changed
const dtYear = dateTaken.getUTCFullYear();
const dtMonth = dateTaken.getUTCMonth()
if (Number.isInteger(row.dayId) && (dtMonth !== prevMonth || dtYear !== prevYear)) {
const text = (dtYear === prevYear || dtYear === thisYear) ? undefined : dtYear;
this.ticks.push(getTick(row.dayId, text));
}
prevMonth = dtMonth;
prevYear = dtYear;
}
}
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
y += row.size;
2022-10-14 23:29:20 +00:00
}
2022-10-16 02:55:53 +00:00
}
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
/** Mark ticks as visible or invisible */
private computeVisibleTicks() {
2022-10-14 23:29:20 +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);
const minGap = fontSizePx + (window.innerWidth <= 768 ? 5 : 2);
let prevShow = -9999;
for (const [idx, tick] of this.ticks.entries()) {
// You can't see these anyway, why bother?
2022-10-15 17:41:49 +00:00
if (tick.top < minGap || tick.top > this.height - minGap) {
2022-10-14 23:29:20 +00:00
tick.s = false;
continue;
}
// Will overlap with the previous tick. Skip anyway.
2022-10-15 17:41:49 +00:00
if (tick.top - prevShow < minGap) {
2022-10-14 23:29:20 +00:00
tick.s = false;
continue;
}
// This is a labelled tick then show it anyway for the sake of best effort
if (tick.text) {
tick.s = true;
2022-10-15 17:41:49 +00:00
prevShow = tick.top;
2022-10-14 23:29:20 +00:00
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;
}
i++;
}
if (i < this.ticks.length) {
// A labelled tick was found
const nextLabelledTick = this.ticks[i];
2022-10-15 17:41:49 +00:00
if (tick.top + minGap > nextLabelledTick.top &&
nextLabelledTick.top < this.height - minGap) { // make sure this will be shown
2022-10-14 23:29:20 +00:00
tick.s = false;
continue;
}
}
// Show this tick
tick.s = true;
2022-10-15 17:41:49 +00:00
prevShow = tick.top;
2022-10-14 23:29:20 +00:00
}
}
2022-10-16 02:55:53 +00:00
/**
* Update tick positions without truncating the list
* This is much cheaper than reflowing the whole thing
*/
public adjust() {
if (this.adjustTimer) return;
this.adjustTimer = window.setTimeout(() => {
this.adjustTimer = null;
this.adjustNow();
}, 300);
}
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
/** Do adjustment synchrnously */
private adjustNow() {
// Refresh height of recycler
this.recyclerHeight = this.recycler.$refs.wrapper.clientHeight;
2022-10-14 23:29:20 +00:00
2022-10-16 02:55:53 +00:00
// Start with the first tick. Walk over all rows counting the
// y position. When you hit a row with the tick, update y and
2022-10-16 05:48:25 +00:00
// top values and move to the next tick.
let tickId = 0;
2022-10-16 02:55:53 +00:00
let y = 0;
2022-10-14 23:29:20 +00:00
2022-10-15 17:41:49 +00:00
for (const row of this.rows) {
2022-10-16 02:55:53 +00:00
// Check if tick is valid
if (tickId >= this.ticks.length) {
return;
}
2022-10-15 17:41:49 +00:00
2022-10-16 02:55:53 +00:00
// Check if we hit the next tick
const tick = this.ticks[tickId];
if (tick.dayId === row.dayId) {
tick.y = y;
this.setTickTop(tick);
tickId++;
2022-10-14 23:29:20 +00:00
}
2022-10-15 17:41:49 +00:00
y += row.size;
2022-10-14 23:29:20 +00:00
}
}
/** Change actual position of the hover cursor */
private moveHoverCursor(y: number) {
2022-10-16 02:55:53 +00:00
this.hoverCursorY = utils.roundHalf(y);
2022-10-14 23:29:20 +00:00
// Get index of previous tick
2022-10-16 02:55:53 +00:00
let idx = utils.binarySearch(this.ticks, y, 'topF');
2022-10-15 17:41:49 +00:00
if (idx === 0) {
// use this tick
} else if (idx >= 1 && idx <= this.ticks.length) {
2022-10-14 23:29:20 +00:00
idx = idx - 1;
} else {
return;
}
// DayId of current hover
const dayId = this.ticks[idx]?.dayId
2022-10-14 23:29:20 +00:00
// Special days
if (dayId === undefined || this.TagDayIDValueSet.has(dayId)) {
2022-10-14 23:29:20 +00:00
this.hoverCursorText = "";
return;
}
const date = utils.dayIdToDate(dayId);
this.hoverCursorText = utils.getShortDateStr(date);
}
/** Handle mouse hover */
private mousemove(event: MouseEvent) {
if (event.buttons) {
this.mousedown(event);
}
this.moveHoverCursor(event.offsetY);
}
/** Handle mouse leave */
private mouseleave() {
this.moveHoverCursor(this.cursorY);
}
/** Handle mouse click */
private mousedown(event: MouseEvent) {
this.recycler.scrollToPosition(this.getRecyclerY(event.offsetY));
this.handleScroll();
}
/** Handle touch */
private touchmove(event: any) {
const rect = event.target.getBoundingClientRect();
const y = event.targetTouches[0].pageY - rect.top;
this.recycler.scrollToPosition(this.getRecyclerY(y));
event.preventDefault();
event.stopPropagation();
this.handleScroll();
}
/** Update this is being used to scroll recycler */
private handleScroll() {
if (this.scrollingTimer) window.clearTimeout(this.scrollingTimer);
this.scrolling = true;
this.scrollingTimer = window.setTimeout(() => {
this.scrolling = false;
this.scrollingTimer = null;
}, 1500);
}
/** Get recycler equivalent position from event */
private getRecyclerY(y: number) {
const tH = this.recyclerHeight;
const maxH = this.height;
return y * tH / maxH;
}
}
</script>
<style lang="scss" scoped>
@mixin phone {
@media (max-width: 768px) { @content; }
}
.scroller {
overflow-y: clip;
position: absolute;
height: 100%;
width: 36px;
top: 0; right: 0;
cursor: ns-resize;
opacity: 0;
transition: opacity .2s ease-in-out;
// Show on hover or scroll of main window
&:hover, &.scrolling-recycler {
opacity: 1;
}
// Hide ticks on mobile unless hovering
@include phone {
&:not(.scrolling) {
.cursor.hv {
left: 12px;
border: none;
box-shadow: 0 0 5px -3px #000;
height: 30px;
border-radius: 15px;
> .text { display: none; }
> .icon { display: block; }
}
> .tick { opacity: 0; }
}
.cursor.st { display: none; }
}
> .tick {
pointer-events: none;
position: absolute;
font-size: 0.75em;
line-height: 0.75em;
font-weight: 600;
opacity: 0.95;
right: 9px;
transform: translateY(-50%);
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; }
}
@include phone {
background-color: var(--color-main-background);
padding: 4px;
border-radius: 4px;
}
}
> .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;
}
&.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(-4px, 2px);
}
}
}
&:hover > .cursor.st {
opacity: 1;
}
}
</style>