0eba9e87e9
- Disable formatting rules (handled by Prettier) - Relaxed strict Vue/JS rules for demo code compatibility - Fix syntax errors in ApiPlayground and VoiceCloningDemo - Fix duplicate else-if condition in ApiPlayground - Fix Promise executor async pattern in AutoregressiveAudioDemo - Add TypeScript file support to ESLint config Warnings reduced from 295 to 251 problems. Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
476 lines
9.6 KiB
Vue
476 lines
9.6 KiB
Vue
<!--
|
||
LazyLoadingDemo.vue
|
||
懒加载演示
|
||
-->
|
||
<template>
|
||
<div class="lazy-loading-demo">
|
||
<div class="header">
|
||
<div class="title">
|
||
图片懒加载:节省带宽,提升性能
|
||
</div>
|
||
<div class="subtitle">
|
||
对比懒加载和立即加载的区别
|
||
</div>
|
||
</div>
|
||
|
||
<div class="demo-container">
|
||
<div class="mode-selector">
|
||
<button
|
||
:class="['mode-btn', { active: mode === 'eager' }]"
|
||
@click="mode = 'eager'"
|
||
>
|
||
📦 立即加载
|
||
</button>
|
||
<button
|
||
:class="['mode-btn', { active: mode === 'lazy' }]"
|
||
@click="mode = 'lazy'"
|
||
>
|
||
⏳ 懒加载
|
||
</button>
|
||
</div>
|
||
|
||
<div class="stats-bar">
|
||
<div class="stat">
|
||
<span class="stat-label">已加载图片</span>
|
||
<span class="stat-value">{{ loadedImages }} / {{ totalImages }}</span>
|
||
</div>
|
||
<div class="stat">
|
||
<span class="stat-label">节省流量</span>
|
||
<span
|
||
class="stat-value"
|
||
:class="{ positive: savedBandwidth > 0 }"
|
||
>
|
||
{{ savedBandwidth > 0 ? '-' : '' }}{{ savedBandwidth }} KB
|
||
</span>
|
||
</div>
|
||
<div class="stat">
|
||
<span class="stat-label">加载时间</span>
|
||
<span class="stat-value">{{ loadTime }} ms</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div
|
||
ref="scrollContainer"
|
||
class="scroll-container"
|
||
@scroll="handleScroll"
|
||
>
|
||
<div class="content-area">
|
||
<div class="placeholder">
|
||
向下滚动查看更多内容
|
||
</div>
|
||
|
||
<div
|
||
v-for="(image, index) in images"
|
||
:key="index"
|
||
:ref="(el) => setImageRef(el, index)"
|
||
class="image-item"
|
||
>
|
||
<div
|
||
class="image-wrapper"
|
||
:class="{ loading: image.loading, loaded: image.loaded }"
|
||
>
|
||
<div
|
||
v-if="!image.loaded && mode === 'lazy'"
|
||
class="placeholder-box"
|
||
>
|
||
<div class="spinner" />
|
||
<div class="placeholder-text">
|
||
加载中...
|
||
</div>
|
||
</div>
|
||
<div
|
||
v-else-if="image.loaded"
|
||
class="image-box"
|
||
>
|
||
<div class="image-icon">
|
||
🖼️
|
||
</div>
|
||
<div class="image-info">
|
||
<div class="image-size">
|
||
{{ image.size }}
|
||
</div>
|
||
<div class="image-dim">
|
||
{{ image.dimensions }}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="image-caption">
|
||
图片 {{ index + 1 }}
|
||
</div>
|
||
</div>
|
||
|
||
<div class="placeholder">
|
||
已经到底了
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="explanation">
|
||
<div class="explanation-item">
|
||
<h4>💡 懒加载原理</h4>
|
||
<p>
|
||
只有当图片进入视口(用户可见区域)时才开始加载。使用 Intersection
|
||
Observer API 可以高效实现。
|
||
</p>
|
||
</div>
|
||
|
||
<div class="explanation-item">
|
||
<h4>📊 性能收益</h4>
|
||
<p>
|
||
懒加载可以节省 30-60%
|
||
的带宽,大幅提升首屏加载速度,特别是在移动端效果显著。
|
||
</p>
|
||
</div>
|
||
|
||
<div class="explanation-item">
|
||
<h4>🔧 实现方式</h4>
|
||
<p>
|
||
<code>loading="lazy"</code>
|
||
属性是最简单的方式,现代浏览器都支持。需要更多控制时使用
|
||
Intersection Observer。
|
||
</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed, onMounted, watch } from 'vue'
|
||
|
||
const mode = ref('eager')
|
||
const scrollContainer = ref(null)
|
||
const totalImages = 12
|
||
const imageRefs = ref([])
|
||
|
||
const images = ref([])
|
||
|
||
const loadedImages = computed(() => {
|
||
return images.value.filter((img) => img.loaded).length
|
||
})
|
||
|
||
const savedBandwidth = computed(() => {
|
||
if (mode.value === 'eager') return 0
|
||
const notLoaded = images.value.filter(
|
||
(img) =>
|
||
!img.loaded && !imageRefs.value[images.value.indexOf(img)]?.isVisible
|
||
).length
|
||
return notLoaded * 150 // 假设每张图片 150KB
|
||
})
|
||
|
||
const loadTime = computed(() => {
|
||
if (mode.value === 'eager') return 2400
|
||
return loadedImages.value * 150
|
||
})
|
||
|
||
function initializeImages() {
|
||
images.value = Array.from({ length: totalImages }, (_, i) => ({
|
||
loaded: mode.value === 'eager',
|
||
loading: false,
|
||
size: '150 KB',
|
||
dimensions: '800×600'
|
||
}))
|
||
}
|
||
|
||
function setImageRef(el, index) {
|
||
if (el) {
|
||
imageRefs.value[index] = el
|
||
}
|
||
}
|
||
|
||
function handleScroll() {
|
||
if (mode.value === 'lazy') {
|
||
checkVisibility()
|
||
}
|
||
}
|
||
|
||
function checkVisibility() {
|
||
const container = scrollContainer.value
|
||
if (!container) return
|
||
|
||
const containerRect = container.getBoundingClientRect()
|
||
const threshold = 100 // 提前 100px 开始加载
|
||
|
||
images.value.forEach((image, index) => {
|
||
if (image.loaded || image.loading) return
|
||
|
||
const ref = imageRefs.value[index]
|
||
if (!ref) return
|
||
|
||
const rect = ref.getBoundingClientRect()
|
||
const isVisible = rect.top < containerRect.bottom + threshold
|
||
|
||
if (isVisible) {
|
||
loadImage(index)
|
||
}
|
||
})
|
||
}
|
||
|
||
function loadImage(index) {
|
||
const image = images.value[index]
|
||
if (!image || image.loaded || image.loading) return
|
||
|
||
image.loading = true
|
||
|
||
// 模拟加载延迟
|
||
setTimeout(
|
||
() => {
|
||
image.loaded = true
|
||
image.loading = false
|
||
},
|
||
300 + Math.random() * 500
|
||
)
|
||
}
|
||
|
||
watch(mode, () => {
|
||
initializeImages()
|
||
if (mode.value === 'lazy') {
|
||
setTimeout(() => checkVisibility(), 100)
|
||
}
|
||
})
|
||
|
||
onMounted(() => {
|
||
initializeImages()
|
||
if (mode.value === 'lazy') {
|
||
setTimeout(() => checkVisibility(), 100)
|
||
}
|
||
})
|
||
</script>
|
||
|
||
<style scoped>
|
||
.lazy-loading-demo {
|
||
border: 1px solid var(--vp-c-divider);
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 12px;
|
||
padding: 1.5rem;
|
||
margin: 1.5rem 0;
|
||
font-family: var(--vp-font-family-base);
|
||
}
|
||
|
||
.header {
|
||
margin-bottom: 1.5rem;
|
||
}
|
||
|
||
.title {
|
||
font-weight: 700;
|
||
font-size: 1.05rem;
|
||
}
|
||
|
||
.subtitle {
|
||
color: var(--vp-c-text-2);
|
||
font-size: 0.9rem;
|
||
margin-top: 0.3rem;
|
||
}
|
||
|
||
.demo-container {
|
||
background: var(--vp-c-bg);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 10px;
|
||
padding: 1.5rem;
|
||
}
|
||
|
||
.mode-selector {
|
||
display: flex;
|
||
gap: 0.5rem;
|
||
margin-bottom: 1.5rem;
|
||
justify-content: center;
|
||
}
|
||
|
||
.mode-btn {
|
||
padding: 0.6rem 1.2rem;
|
||
border: 2px solid var(--vp-c-divider);
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 6px;
|
||
cursor: pointer;
|
||
font-size: 0.9rem;
|
||
font-weight: 500;
|
||
transition: all 0.3s;
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.mode-btn:hover {
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.mode-btn.active {
|
||
background: var(--vp-c-brand);
|
||
border-color: var(--vp-c-brand);
|
||
color: #fff;
|
||
}
|
||
|
||
.stats-bar {
|
||
display: flex;
|
||
gap: 1rem;
|
||
margin-bottom: 1.5rem;
|
||
flex-wrap: wrap;
|
||
}
|
||
|
||
.stat {
|
||
flex: 1;
|
||
min-width: 120px;
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 6px;
|
||
padding: 0.8rem;
|
||
text-align: center;
|
||
}
|
||
|
||
.stat-label {
|
||
display: block;
|
||
font-size: 0.75rem;
|
||
color: var(--vp-c-text-2);
|
||
margin-bottom: 0.3rem;
|
||
}
|
||
|
||
.stat-value {
|
||
display: block;
|
||
font-size: 1.1rem;
|
||
font-weight: 700;
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.stat-value.positive {
|
||
color: #22c55e;
|
||
}
|
||
|
||
.scroll-container {
|
||
height: 400px;
|
||
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
margin-bottom: 1.5rem;
|
||
background: var(--vp-c-bg-soft);
|
||
}
|
||
|
||
.content-area {
|
||
padding: 0.75rem;
|
||
}
|
||
|
||
.placeholder {
|
||
text-align: center;
|
||
padding: 1.5rem;
|
||
color: var(--vp-c-text-2);
|
||
font-size: 0.9rem;
|
||
}
|
||
|
||
.image-item {
|
||
margin-bottom: 1rem;
|
||
}
|
||
|
||
.image-wrapper {
|
||
background: var(--vp-c-bg);
|
||
border: 2px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
overflow: hidden;
|
||
transition: all 0.3s;
|
||
}
|
||
|
||
.image-wrapper.loading {
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.image-wrapper.loaded {
|
||
border-color: #22c55e;
|
||
}
|
||
|
||
.placeholder-box {
|
||
height: 150px;
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
justify-content: center;
|
||
background: var(--vp-c-bg-soft);
|
||
}
|
||
|
||
.spinner {
|
||
width: 30px;
|
||
height: 30px;
|
||
border: 3px solid var(--vp-c-divider);
|
||
border-top-color: var(--vp-c-brand);
|
||
border-radius: 50%;
|
||
animation: spin 0.8s linear infinite;
|
||
}
|
||
|
||
@keyframes spin {
|
||
to {
|
||
transform: rotate(360deg);
|
||
}
|
||
}
|
||
|
||
.placeholder-text {
|
||
margin-top: 0.5rem;
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.image-box {
|
||
height: 150px;
|
||
display: flex;
|
||
align-items: center;
|
||
padding: 0.75rem;
|
||
gap: 1rem;
|
||
}
|
||
|
||
.image-icon {
|
||
font-size: 3rem;
|
||
}
|
||
|
||
.image-info {
|
||
flex: 1;
|
||
}
|
||
|
||
.image-size {
|
||
font-size: 0.9rem;
|
||
font-weight: 600;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 0.3rem;
|
||
}
|
||
|
||
.image-dim {
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.image-caption {
|
||
text-align: center;
|
||
font-size: 0.8rem;
|
||
color: var(--vp-c-text-2);
|
||
margin-top: 0.5rem;
|
||
}
|
||
|
||
.explanation {
|
||
display: grid;
|
||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||
gap: 1rem;
|
||
}
|
||
|
||
.explanation-item {
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 6px;
|
||
padding: 0.75rem;
|
||
}
|
||
|
||
.explanation-item h4 {
|
||
font-size: 0.9rem;
|
||
font-weight: 600;
|
||
margin-bottom: 0.5rem;
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.explanation-item p {
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-2);
|
||
line-height: 1.6;
|
||
margin: 0;
|
||
}
|
||
|
||
.explanation-item code {
|
||
background: var(--vp-c-bg);
|
||
padding: 0.2rem 0.4rem;
|
||
border-radius: 4px;
|
||
font-family: 'Monaco', 'Menlo', monospace;
|
||
font-size: 0.8rem;
|
||
color: var(--vp-c-brand);
|
||
}
|
||
</style>
|