328 lines
6.7 KiB
Vue
328 lines
6.7 KiB
Vue
|
|
<template>
|
|||
|
|
<div
|
|||
|
|
class="reactivity-mechanism-demo"
|
|||
|
|
:style="{ '--tab-accent': currentTab?.color ?? 'var(--vp-c-brand)' }"
|
|||
|
|
>
|
|||
|
|
<div class="toggle-bar">
|
|||
|
|
<button
|
|||
|
|
v-for="tab in tabs"
|
|||
|
|
:key="tab.id"
|
|||
|
|
:class="['toggle-btn', { active: activeTab === tab.id }]"
|
|||
|
|
:style="activeTab === tab.id ? { borderColor: tab.color, background: tab.color } : {}"
|
|||
|
|
@click="switchTab(tab.id)"
|
|||
|
|
>
|
|||
|
|
{{ tab.label }}
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<div class="visualization-area">
|
|||
|
|
<div class="counter-row">
|
|||
|
|
<span class="counter-label">count:</span>
|
|||
|
|
<span class="counter-value">{{ count }}</span>
|
|||
|
|
</div>
|
|||
|
|
<button
|
|||
|
|
class="modify-btn"
|
|||
|
|
:disabled="isAnimating"
|
|||
|
|
@click="modifyData"
|
|||
|
|
>
|
|||
|
|
修改数据
|
|||
|
|
</button>
|
|||
|
|
|
|||
|
|
<div class="steps-title">引擎盖下</div>
|
|||
|
|
<div class="steps-list">
|
|||
|
|
<div
|
|||
|
|
v-for="(step, idx) in currentSteps"
|
|||
|
|
:key="idx"
|
|||
|
|
:class="['step-item', stepState(idx)]"
|
|||
|
|
:style="stepStyle(idx)"
|
|||
|
|
>
|
|||
|
|
<span class="step-badge">{{ idx + 1 }}</span>
|
|||
|
|
<span class="step-text">{{ step }}</span>
|
|||
|
|
<span v-if="stepStatus(idx) === 'done'" class="step-check">✓</span>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<div class="info-box">
|
|||
|
|
<strong>核心思想:</strong>
|
|||
|
|
{{ infoMessage }}
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</template>
|
|||
|
|
|
|||
|
|
<script setup>
|
|||
|
|
import { ref, computed } from 'vue'
|
|||
|
|
|
|||
|
|
const TABS = {
|
|||
|
|
vue: {
|
|||
|
|
id: 'vue',
|
|||
|
|
label: 'Vue (Proxy)',
|
|||
|
|
color: 'var(--vp-c-green-1)',
|
|||
|
|
steps: [
|
|||
|
|
'count = 1 → Proxy 的 set 陷阱被触发',
|
|||
|
|
'通知依赖收集器:"count 变了"',
|
|||
|
|
'找到所有依赖 count 的组件',
|
|||
|
|
'自动更新 DOM'
|
|||
|
|
],
|
|||
|
|
info: 'Vue 通过 Proxy 自动拦截数据读写,开发者无需额外操作——写法最自然。'
|
|||
|
|
},
|
|||
|
|
react: {
|
|||
|
|
id: 'react',
|
|||
|
|
label: 'React (setState)',
|
|||
|
|
color: 'var(--vp-c-brand)',
|
|||
|
|
steps: [
|
|||
|
|
'调用 setCount(count + 1)',
|
|||
|
|
'React 将更新加入队列',
|
|||
|
|
'批量处理队列,触发 re-render',
|
|||
|
|
'虚拟 DOM Diff → 更新真实 DOM'
|
|||
|
|
],
|
|||
|
|
info: 'React 要求显式调用 setState,虽然多一步,但数据流更可预测。'
|
|||
|
|
},
|
|||
|
|
svelte: {
|
|||
|
|
id: 'svelte',
|
|||
|
|
label: 'Svelte (编译器)',
|
|||
|
|
color: 'var(--vp-c-warning-1)',
|
|||
|
|
steps: [
|
|||
|
|
'count += 1 被编译器识别为赋值',
|
|||
|
|
'编译时已生成 $$invalidate(count)',
|
|||
|
|
'直接更新对应的 DOM 节点(无 Diff)',
|
|||
|
|
'零运行时开销'
|
|||
|
|
],
|
|||
|
|
info: 'Svelte 在编译时完成分析,运行时零开销——但依赖编译器魔法。'
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const activeTab = ref('vue')
|
|||
|
|
const count = ref(0)
|
|||
|
|
const currentStepIndex = ref(-1)
|
|||
|
|
const isAnimating = ref(false)
|
|||
|
|
|
|||
|
|
const tabs = computed(() => Object.values(TABS))
|
|||
|
|
|
|||
|
|
const currentTab = computed(() => TABS[activeTab.value])
|
|||
|
|
|
|||
|
|
const currentSteps = computed(() => currentTab.value?.steps ?? [])
|
|||
|
|
|
|||
|
|
const infoMessage = computed(() => currentTab.value?.info ?? '')
|
|||
|
|
|
|||
|
|
function stepState(idx) {
|
|||
|
|
if (currentStepIndex.value < idx) return 'pending'
|
|||
|
|
if (currentStepIndex.value === idx) return 'active'
|
|||
|
|
return 'done'
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
function stepStatus(idx) {
|
|||
|
|
if (currentStepIndex.value < idx) return 'pending'
|
|||
|
|
if (currentStepIndex.value === idx) return 'active'
|
|||
|
|
return 'done'
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
function stepStyle(idx) {
|
|||
|
|
if (currentStepIndex.value !== idx) return {}
|
|||
|
|
const color = currentTab.value?.color ?? 'var(--vp-c-brand)'
|
|||
|
|
return {
|
|||
|
|
borderColor: color,
|
|||
|
|
boxShadow: `0 0 8px color-mix(in srgb, ${color} 40%, transparent)`
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
function switchTab(id) {
|
|||
|
|
if (isAnimating.value) return
|
|||
|
|
activeTab.value = id
|
|||
|
|
currentStepIndex.value = -1
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
async function modifyData() {
|
|||
|
|
if (isAnimating.value) return
|
|||
|
|
isAnimating.value = true
|
|||
|
|
count.value += 1
|
|||
|
|
currentStepIndex.value = -1
|
|||
|
|
|
|||
|
|
for (let i = 0; i < 4; i++) {
|
|||
|
|
currentStepIndex.value = i
|
|||
|
|
await new Promise((r) => setTimeout(r, 300))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
currentStepIndex.value = 4
|
|||
|
|
isAnimating.value = false
|
|||
|
|
}
|
|||
|
|
</script>
|
|||
|
|
|
|||
|
|
<style scoped>
|
|||
|
|
.reactivity-mechanism-demo {
|
|||
|
|
border: 1px solid var(--vp-c-divider);
|
|||
|
|
border-radius: 6px;
|
|||
|
|
background-color: var(--vp-c-bg-soft);
|
|||
|
|
padding: 0.75rem;
|
|||
|
|
margin: 0.5rem 0;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.toggle-bar {
|
|||
|
|
display: flex;
|
|||
|
|
gap: 0.5rem;
|
|||
|
|
flex-wrap: wrap;
|
|||
|
|
margin-bottom: 1rem;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.toggle-btn {
|
|||
|
|
padding: 0.4rem 0.8rem;
|
|||
|
|
border: 1px solid var(--vp-c-divider);
|
|||
|
|
border-radius: 6px;
|
|||
|
|
background: var(--vp-c-bg);
|
|||
|
|
cursor: pointer;
|
|||
|
|
font-size: 0.85rem;
|
|||
|
|
color: var(--vp-c-text-1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.toggle-btn:hover {
|
|||
|
|
border-color: var(--vp-c-brand);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.toggle-btn.active {
|
|||
|
|
color: white;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.visualization-area {
|
|||
|
|
background: var(--vp-c-bg);
|
|||
|
|
border: 1px solid var(--vp-c-divider);
|
|||
|
|
border-radius: 6px;
|
|||
|
|
padding: 1rem;
|
|||
|
|
margin-bottom: 0.75rem;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.counter-row {
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
justify-content: center;
|
|||
|
|
gap: 0.5rem;
|
|||
|
|
margin-bottom: 1rem;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.counter-label {
|
|||
|
|
font-size: 0.9rem;
|
|||
|
|
color: var(--vp-c-text-2);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.counter-value {
|
|||
|
|
font-size: 1.5rem;
|
|||
|
|
font-weight: 700;
|
|||
|
|
color: var(--vp-c-brand);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.modify-btn {
|
|||
|
|
display: block;
|
|||
|
|
margin: 0 auto 1rem;
|
|||
|
|
padding: 0.4rem 1rem;
|
|||
|
|
border: 1px solid var(--vp-c-divider);
|
|||
|
|
border-radius: 6px;
|
|||
|
|
background: var(--vp-c-bg-soft);
|
|||
|
|
cursor: pointer;
|
|||
|
|
font-size: 0.85rem;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.modify-btn:hover:not(:disabled) {
|
|||
|
|
border-color: var(--vp-c-brand);
|
|||
|
|
color: var(--vp-c-brand);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.modify-btn:disabled {
|
|||
|
|
opacity: 0.7;
|
|||
|
|
cursor: not-allowed;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.steps-title {
|
|||
|
|
font-size: 0.9rem;
|
|||
|
|
font-weight: 600;
|
|||
|
|
margin-bottom: 0.5rem;
|
|||
|
|
color: var(--vp-c-text-2);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.steps-list {
|
|||
|
|
display: flex;
|
|||
|
|
flex-direction: column;
|
|||
|
|
gap: 0.5rem;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-item {
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
gap: 0.5rem;
|
|||
|
|
background: var(--vp-c-bg);
|
|||
|
|
border: 1px solid var(--vp-c-divider);
|
|||
|
|
border-radius: 6px;
|
|||
|
|
padding: 0.5rem 0.75rem;
|
|||
|
|
font-size: 0.82rem;
|
|||
|
|
transition: border-color 0.2s, box-shadow 0.2s;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-item.pending {
|
|||
|
|
opacity: 0.6;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-badge {
|
|||
|
|
flex-shrink: 0;
|
|||
|
|
width: 1.25rem;
|
|||
|
|
height: 1.25rem;
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
justify-content: center;
|
|||
|
|
border-radius: 4px;
|
|||
|
|
background: var(--vp-c-bg-alt);
|
|||
|
|
font-size: 0.75rem;
|
|||
|
|
font-weight: 600;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-item.active .step-badge {
|
|||
|
|
background: var(--tab-accent);
|
|||
|
|
color: white;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-item.done .step-badge {
|
|||
|
|
background: var(--vp-c-green-1);
|
|||
|
|
color: white;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-text {
|
|||
|
|
flex: 1;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-check {
|
|||
|
|
color: var(--vp-c-green-1);
|
|||
|
|
font-weight: 700;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.step-item.done {
|
|||
|
|
border-color: var(--vp-c-green-1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.info-box {
|
|||
|
|
display: flex;
|
|||
|
|
gap: 0.25rem;
|
|||
|
|
padding: 0.75rem;
|
|||
|
|
border-radius: 6px;
|
|||
|
|
background: var(--vp-c-bg-alt);
|
|||
|
|
font-size: 0.85rem;
|
|||
|
|
color: var(--vp-c-text-2);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.info-box strong {
|
|||
|
|
white-space: nowrap;
|
|||
|
|
flex-shrink: 0;
|
|||
|
|
color: var(--vp-c-text-1);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
@media (max-width: 720px) {
|
|||
|
|
.toggle-bar {
|
|||
|
|
flex-direction: column;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.toggle-btn {
|
|||
|
|
width: 100%;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.steps-list {
|
|||
|
|
flex-direction: column;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
</style>
|