d174ceea32
- Add new interactive components for frontend routing, browser rendering pipeline, and database transactions - Improve existing demos with better visuals, explanations, and examples - Update documentation structure and content for better clarity - Add new utility scripts and update package.json with new commands - Fix formatting and alignment in documentation tables
326 lines
6.5 KiB
Vue
326 lines
6.5 KiB
Vue
<template>
|
||
<div class="mobx-reactivity-demo">
|
||
<div class="demo-header">
|
||
<span class="icon">⚡</span>
|
||
<span class="title">MobX 响应式原理</span>
|
||
<span class="subtitle">自动追踪依赖的魔法</span>
|
||
</div>
|
||
|
||
<div class="intro-text">
|
||
想象你在<span class="highlight">魔术表演</span>现场:魔术师(Observable)改变物品,所有盯着看的观众(Reaction)都会自动注意到变化,不需要一个个去通知他们。
|
||
</div>
|
||
|
||
<div class="demo-content">
|
||
<div class="state-display">
|
||
<div class="state-header">
|
||
<span class="state-icon">📦</span>
|
||
<span class="state-title">Observable 状态</span>
|
||
</div>
|
||
<div class="todo-list">
|
||
<div
|
||
v-for="todo in todos"
|
||
:key="todo.id"
|
||
class="todo-item"
|
||
:class="{ completed: todo.completed, changed: recentlyChanged === todo.id }"
|
||
@click="toggleTodo(todo.id)"
|
||
>
|
||
<span class="todo-status">{{ todo.completed ? '✓' : '○' }}</span>
|
||
<span class="todo-text">{{ todo.text }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="reaction-display">
|
||
<div class="reaction-header">
|
||
<span class="reaction-icon">🔄</span>
|
||
<span class="reaction-title">自动响应</span>
|
||
</div>
|
||
<div class="reaction-stats">
|
||
<div class="stat-item">
|
||
<span class="stat-label">总计:</span>
|
||
<span class="stat-value">{{ todos.length }} 项</span>
|
||
</div>
|
||
<div class="stat-item">
|
||
<span class="stat-label">已完成:</span>
|
||
<span class="stat-value completed">{{ completedCount }} 项</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="interaction-area">
|
||
<input v-model="newTodoText" placeholder="输入待办事项..." @keyup.enter="addTodo" class="todo-input" />
|
||
<button @click="addTodo" class="add-btn">➕ 添加</button>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="info-box">
|
||
<span class="icon">💡</span>
|
||
<strong>核心思想:</strong>MobX 自动追踪状态和响应的关系,状态变化时自动触发相关更新。就像魔术,你只管改变数据,UI 会自动更新。
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed, watch } from 'vue'
|
||
|
||
const todos = ref([
|
||
{ id: 1, text: '学习 MobX', completed: false },
|
||
{ id: 2, text: '理解响应式原理', completed: true }
|
||
])
|
||
|
||
const newTodoText = ref('')
|
||
const recentlyChanged = ref(null)
|
||
|
||
const completedCount = computed(() => {
|
||
return todos.value.filter(t => t.completed).length
|
||
})
|
||
|
||
const addTodo = () => {
|
||
if (!newTodoText.value.trim()) return
|
||
|
||
const newTodo = {
|
||
id: Date.now(),
|
||
text: newTodoText.value,
|
||
completed: false
|
||
}
|
||
|
||
todos.value.push(newTodo)
|
||
recentlyChanged.value = newTodo.id
|
||
newTodoText.value = ''
|
||
|
||
setTimeout(() => {
|
||
recentlyChanged.value = null
|
||
}, 500)
|
||
}
|
||
|
||
const toggleTodo = (id) => {
|
||
const todo = todos.value.find(t => t.id === id)
|
||
if (todo) {
|
||
todo.completed = !todo.completed
|
||
recentlyChanged.value = id
|
||
setTimeout(() => {
|
||
recentlyChanged.value = null
|
||
}, 500)
|
||
}
|
||
}
|
||
</script>
|
||
|
||
<style scoped>
|
||
.mobx-reactivity-demo {
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 8px;
|
||
background: var(--vp-c-bg-soft);
|
||
padding: 1rem;
|
||
margin: 1rem 0;
|
||
max-height: 600px;
|
||
overflow-y: auto;
|
||
}
|
||
|
||
.demo-header {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.5rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
.demo-header .icon {
|
||
font-size: 1.25rem;
|
||
}
|
||
|
||
.demo-header .title {
|
||
font-weight: bold;
|
||
font-size: 1rem;
|
||
}
|
||
|
||
.demo-header .subtitle {
|
||
color: var(--vp-c-text-2);
|
||
font-size: 0.85rem;
|
||
margin-left: 0.5rem;
|
||
}
|
||
|
||
.intro-text {
|
||
font-size: 0.9rem;
|
||
color: var(--vp-c-text-2);
|
||
line-height: 1.6;
|
||
margin-bottom: 1rem;
|
||
padding: 0.75rem;
|
||
background: var(--vp-c-bg);
|
||
border-radius: 6px;
|
||
}
|
||
|
||
.intro-text .highlight {
|
||
color: var(--vp-c-brand-1);
|
||
font-weight: 500;
|
||
}
|
||
|
||
.demo-content {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 1rem;
|
||
margin-bottom: 1rem;
|
||
}
|
||
|
||
.state-display,
|
||
.reaction-display {
|
||
background: var(--vp-c-bg);
|
||
border: 2px solid var(--vp-c-divider);
|
||
border-radius: 8px;
|
||
padding: 1rem;
|
||
}
|
||
|
||
.state-header,
|
||
.reaction-header {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.5rem;
|
||
margin-bottom: 0.75rem;
|
||
padding-bottom: 0.5rem;
|
||
border-bottom: 1px solid var(--vp-c-divider);
|
||
}
|
||
|
||
.state-icon,
|
||
.reaction-icon {
|
||
font-size: 1.25rem;
|
||
}
|
||
|
||
.state-title,
|
||
.reaction-title {
|
||
font-weight: 600;
|
||
font-size: 0.9rem;
|
||
}
|
||
|
||
.todo-list {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 0.5rem;
|
||
}
|
||
|
||
.todo-item {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.75rem;
|
||
padding: 0.6rem 0.75rem;
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 6px;
|
||
cursor: pointer;
|
||
transition: all 0.2s ease;
|
||
}
|
||
|
||
.todo-item:hover {
|
||
background: var(--vp-c-bg);
|
||
transform: translateX(4px);
|
||
}
|
||
|
||
.todo-item.completed {
|
||
background: #f0fdf4;
|
||
}
|
||
|
||
.todo-item.completed .todo-text {
|
||
text-decoration: line-through;
|
||
color: var(--vp-c-text-3);
|
||
}
|
||
|
||
.todo-item.changed {
|
||
animation: highlight 0.5s ease;
|
||
}
|
||
|
||
@keyframes highlight {
|
||
0%, 100% { background: var(--vp-c-bg-soft); }
|
||
50% { background: #fef3c7; }
|
||
}
|
||
|
||
.todo-status {
|
||
font-size: 1.25rem;
|
||
color: var(--vp-c-brand);
|
||
}
|
||
|
||
.todo-text {
|
||
font-size: 0.9rem;
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.reaction-stats {
|
||
display: flex;
|
||
gap: 1.5rem;
|
||
}
|
||
|
||
.stat-item {
|
||
display: flex;
|
||
gap: 0.5rem;
|
||
font-size: 0.85rem;
|
||
}
|
||
|
||
.stat-label {
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.stat-value {
|
||
font-weight: 600;
|
||
color: var(--vp-c-brand);
|
||
}
|
||
|
||
.stat-value.completed {
|
||
color: #22c55e;
|
||
}
|
||
|
||
.interaction-area {
|
||
display: flex;
|
||
gap: 0.75rem;
|
||
}
|
||
|
||
.todo-input {
|
||
flex: 1;
|
||
padding: 0.6rem 1rem;
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
font-size: 0.9rem;
|
||
background: var(--vp-c-bg);
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.todo-input:focus {
|
||
outline: none;
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.add-btn {
|
||
padding: 0.6rem 1.5rem;
|
||
background: var(--vp-c-brand);
|
||
color: white;
|
||
border: none;
|
||
border-radius: 6px;
|
||
font-size: 0.9rem;
|
||
font-weight: 500;
|
||
cursor: pointer;
|
||
transition: all 0.2s ease;
|
||
}
|
||
|
||
.add-btn:hover {
|
||
opacity: 0.9;
|
||
transform: translateY(-1px);
|
||
}
|
||
|
||
.info-box {
|
||
background: var(--vp-c-bg-alt);
|
||
padding: 0.75rem;
|
||
border-radius: 6px;
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-2);
|
||
}
|
||
|
||
.info-box .icon {
|
||
margin-right: 0.25rem;
|
||
}
|
||
|
||
@media (max-width: 768px) {
|
||
.interaction-area {
|
||
flex-direction: column;
|
||
}
|
||
|
||
.reaction-stats {
|
||
flex-direction: column;
|
||
gap: 0.5rem;
|
||
}
|
||
}
|
||
</style>
|