Revert all the way back before "turn <img> in VideoScroll into its own component"
This commit is contained in:
parent
91d6af38a0
commit
15b377d4a2
2 changed files with 51 additions and 120 deletions
|
@ -1,6 +1,7 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="video-scroll" ref="root">
|
<div class="video-scroll" ref="root">
|
||||||
<VideoScrollImage :definition="{fullres: definition.files[0]}"
|
<img class="visible loaded"
|
||||||
|
:src="definition.files[0]"
|
||||||
:style="{
|
:style="{
|
||||||
top: `${Math.round(definition.top)}px`,
|
top: `${Math.round(definition.top)}px`,
|
||||||
left: `${Math.round(definition.left)}px`,
|
left: `${Math.round(definition.left)}px`,
|
||||||
|
@ -8,14 +9,11 @@
|
||||||
height: isVertical ? `${Math.round(definition.height)}px` : 'auto',
|
height: isVertical ? `${Math.round(definition.height)}px` : 'auto',
|
||||||
rotate: `${definition.angle}deg`
|
rotate: `${definition.angle}deg`
|
||||||
}"
|
}"
|
||||||
:visible="true"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<!--suppress RequiredAttributes -->
|
<!--suppress RequiredAttributes -->
|
||||||
<VideoScrollImage v-for="file in dynamicFiles"
|
<img v-for="file in dynamicFiles"
|
||||||
:key="file.idx"
|
:data-src="file.src"
|
||||||
:data-idx="file.idx"
|
|
||||||
:definition="file.image"
|
|
||||||
:style="{
|
:style="{
|
||||||
top: `${Math.round(file.top)}px`,
|
top: `${Math.round(file.top)}px`,
|
||||||
left: `${Math.round(file.left)}px`,
|
left: `${Math.round(file.left)}px`,
|
||||||
|
@ -23,7 +21,6 @@
|
||||||
height: `${Math.round(definition.height)}px`,
|
height: `${Math.round(definition.height)}px`,
|
||||||
rotate: `${definition.angle}deg`
|
rotate: `${definition.angle}deg`
|
||||||
}"
|
}"
|
||||||
:visible="imageVisibility[file.idx] || false"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -31,16 +28,9 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import {defineComponent, PropType} from "vue";
|
import {defineComponent, PropType} from "vue";
|
||||||
import {rotate} from "@/utils";
|
import {rotate} from "@/utils";
|
||||||
import VideoScrollImage, {VideoScrollImageDef} from "@/components/VideoScrollImage.vue";
|
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: "VideoScroll",
|
name: "VideoScroll",
|
||||||
components: {VideoScrollImage},
|
|
||||||
data: () => {
|
|
||||||
return {
|
|
||||||
imageVisibility: [] as Boolean[]
|
|
||||||
};
|
|
||||||
},
|
|
||||||
props: {
|
props: {
|
||||||
definition: {
|
definition: {
|
||||||
type: Object as PropType<VideoScrollDef>,
|
type: Object as PropType<VideoScrollDef>,
|
||||||
|
@ -48,14 +38,14 @@ export default defineComponent({
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
dynamicFiles(): { idx: number, image: VideoScrollImageDef, top: number, left: number }[] {
|
dynamicFiles(): { top: number, left: number, src: string }[] {
|
||||||
return this.definition.files.slice(1).map((src: string, idx: number) => {
|
return this.definition.files.slice(1).map((src: string, idx: number) => {
|
||||||
const cy = this.definition.top +
|
const cy = this.definition.top +
|
||||||
(this.isVertical ? (this.definition.height * (idx + 1) * this.verticalDirection) : 0);
|
(this.isVertical ? (this.definition.height * (idx + 1) * this.verticalDirection) : 0);
|
||||||
const cx = this.definition.left +
|
const cx = this.definition.left +
|
||||||
(this.isHorizontal ? (this.definition.width * (idx + 1) * this.horizontalDirection) : 0);
|
(this.isHorizontal ? (this.definition.width * (idx + 1) * this.horizontalDirection) : 0);
|
||||||
const [left, top] = rotate(cx, cy, this.definition.left, this.definition.top, this.definition.angle);
|
const [left, top] = rotate(cx, cy, this.definition.left, this.definition.top, this.definition.angle);
|
||||||
return {idx: idx + 1, top, left, image: {fullres: src}};
|
return {top, left, src};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
isHorizontal(): boolean {
|
isHorizontal(): boolean {
|
||||||
|
@ -79,7 +69,26 @@ export default defineComponent({
|
||||||
const observer = new IntersectionObserver((entries, _) => {
|
const observer = new IntersectionObserver((entries, _) => {
|
||||||
entries.forEach((entry) => {
|
entries.forEach((entry) => {
|
||||||
const element = entry.target as HTMLImageElement;
|
const element = entry.target as HTMLImageElement;
|
||||||
this.imageVisibility[parseInt(element.dataset.idx as string)] = entry.isIntersecting;
|
if (entry.isIntersecting) {
|
||||||
|
if (!element.src) {
|
||||||
|
console.debug(`[VIDEOSCROLL] Intersected, loading ${element.dataset.src}`);
|
||||||
|
element.src = element.dataset.src!;
|
||||||
|
const grayTimeout = setTimeout(() => {
|
||||||
|
element.classList.add("visible");
|
||||||
|
element.style.background = "grey";
|
||||||
|
}, 3000);
|
||||||
|
element.onload = () => {
|
||||||
|
clearTimeout(grayTimeout);
|
||||||
|
element.classList.add("visible");
|
||||||
|
element.style.background = "none";
|
||||||
|
if (this.isHorizontal) {
|
||||||
|
element.style.height = "auto";
|
||||||
|
} else {
|
||||||
|
element.style.width = "auto";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
Array.from((this.$refs.root as Element).children).forEach((el) => {
|
Array.from((this.$refs.root as Element).children).forEach((el) => {
|
||||||
|
@ -109,4 +118,15 @@ export interface VideoScrollDef {
|
||||||
|
|
||||||
<!-- Add "scoped" attribute to limit CSS to this component only -->
|
<!-- Add "scoped" attribute to limit CSS to this component only -->
|
||||||
<style scoped>
|
<style scoped>
|
||||||
|
.video-scroll img {
|
||||||
|
position: absolute;
|
||||||
|
image-rendering: optimizeSpeed;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity .5s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.visible {
|
||||||
|
opacity: 1 !important;
|
||||||
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -1,89 +0,0 @@
|
||||||
<template>
|
|
||||||
<img ref="img" :class="['video-scroll-image', {visible, displayed, loaded}]" :src="currentSrc"/>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
import {defineComponent, PropType, ref, Ref} from "vue";
|
|
||||||
|
|
||||||
export default defineComponent({
|
|
||||||
name: "VideoScrollImage",
|
|
||||||
props: {
|
|
||||||
definition: {
|
|
||||||
type: Object as PropType<VideoScrollImageDef>,
|
|
||||||
required: true
|
|
||||||
},
|
|
||||||
visible: {
|
|
||||||
type: Boolean,
|
|
||||||
required: true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
data() {
|
|
||||||
return {
|
|
||||||
currentSrc: undefined as undefined | string,
|
|
||||||
displayed: false,
|
|
||||||
loaded: false,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
watch: {
|
|
||||||
visible() {
|
|
||||||
if (!this.img) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (this.visible && !this.loaded) {
|
|
||||||
console.debug(`[VIDEOSCROLLIMAGE] Intersected, loading ${this.definition.fullres}`);
|
|
||||||
this.currentSrc = this.definition.fullres;
|
|
||||||
setTimeout(() => {
|
|
||||||
this.displayed = true;
|
|
||||||
}, 3000);
|
|
||||||
this.img.onload = () => {
|
|
||||||
this.displayed = true;
|
|
||||||
this.loaded = true;
|
|
||||||
// if (this.isHorizontal) {
|
|
||||||
// this.img!.style.height = "auto";
|
|
||||||
// } else {
|
|
||||||
// this.img!.style.width = "auto";
|
|
||||||
// }
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
setup(props) {
|
|
||||||
const img = ref<HTMLImageElement | null>(null);
|
|
||||||
const currentSrc: Ref<string | undefined> = ref(undefined);
|
|
||||||
|
|
||||||
return {
|
|
||||||
img,
|
|
||||||
currentSrc
|
|
||||||
};
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
export interface VideoScrollImageDef {
|
|
||||||
fullres: string
|
|
||||||
}
|
|
||||||
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<!-- Add "scoped" attribute to limit CSS to this component only -->
|
|
||||||
<style>
|
|
||||||
.video-scroll-image {
|
|
||||||
position: absolute;
|
|
||||||
image-rendering: optimizeSpeed;
|
|
||||||
visibility: hidden;
|
|
||||||
opacity: 0;
|
|
||||||
transition: opacity .5s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.video-scroll-image.visible {
|
|
||||||
visibility: visible !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.video-scroll-image.loaded {
|
|
||||||
background: transparent !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.video-scroll-image.displayed {
|
|
||||||
background: gray;
|
|
||||||
opacity: 1 !important;
|
|
||||||
}
|
|
||||||
</style>
|
|
Loading…
Reference in a new issue