6098908eee
- Add Vue components for interactive demos (SSH auth, regex, env vars, ports) - Complete markdown content for SSH, regex, environment variables, and ports - Remove placeholder "待实现" sections and replace with detailed guides - Add visual explanations for key concepts like ports and localhost - Include practical examples and troubleshooting tips - Add component for showing evolution from transistors to CPU - Improve documentation structure and navigation - Add security best practices for API keys and environment variables
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>
|