Files
test-repo/docs/.vitepress/theme/components/appendix/framework-nature/ManualVsAutoSyncDemo.vue
T
sanbuphy 6098908eee feat(docs): add interactive demos and complete content for development tools
- 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
2026-02-21 10:04:47 +08:00

434 lines
9.4 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<template>
<div class="sync-demo">
<div class="comparison-container">
<div class="side manual-side">
<div class="side-header">
<span class="badge manual">手动同步 / jQuery 风格</span>
</div>
<div class="cart-control">
<button class="action-btn" @click="addManual">添加商品</button>
<button class="action-btn outline" @click="resetManual">重置</button>
</div>
<div class="sync-areas">
<div
v-for="area in manualAreas"
:key="area.id"
class="sync-area"
:class="{ synced: area.synced, unsynced: !area.synced }"
>
<div class="area-header">
<span class="area-icon">{{ area.icon }}</span>
<span class="area-name">{{ area.name }}</span>
<span class="sync-badge" :class="{ synced: area.synced }">
{{ area.synced ? '已同步' : '未同步' }}
</span>
</div>
<div class="area-value">{{ area.synced ? area.actual : area.stale }}</div>
<button
v-if="!area.synced"
class="sync-btn"
@click="syncArea(area)"
>
手动同步
</button>
</div>
</div>
<div class="miss-counter">
<span class="miss-label">遗漏次数</span>
<span class="miss-value" :class="{ danger: missCount > 0 }">{{ missCount }}</span>
</div>
</div>
<div class="vs-divider">
<div class="vs-badge">VS</div>
</div>
<div class="side auto-side">
<div class="side-header">
<span class="badge auto">自动同步 / 框架风格</span>
</div>
<div class="cart-control">
<button class="action-btn" @click="addAuto">添加商品</button>
<button class="action-btn outline" @click="resetAuto">重置</button>
</div>
<div class="sync-areas">
<div
v-for="area in autoAreas"
:key="area.id"
class="sync-area synced"
>
<div class="area-header">
<span class="area-icon">{{ area.icon }}</span>
<span class="area-name">{{ area.name }}</span>
<span class="sync-badge synced">已同步</span>
</div>
<div class="area-value">{{ area.value }}</div>
</div>
</div>
<div class="miss-counter">
<span class="miss-label">遗漏次数</span>
<span class="miss-value">0</span>
</div>
</div>
</div>
<div class="info-box">
<strong>核心思想</strong>
<span>前端框架的本质价值在于"自动同步"你只需修改数据框架保证所有依赖该数据的 UI 自动更新不会遗漏</span>
</div>
</div>
</template>
<script setup>
import { ref, reactive, computed } from 'vue'
const products = ['耳机 ¥99', '键盘 ¥199', '鼠标 ¥59', '显示器 ¥1299', '摄像头 ¥149', '音箱 ¥79']
let productIndex = ref(0)
const manualCount = ref(0)
const manualItems = ref([])
const missCount = ref(0)
let pendingManualCount = 0
const manualAreas = reactive([
{
id: 'count',
icon: '🔴',
name: '购物车数量',
synced: true,
stale: '0 件',
actual: '0 件'
},
{
id: 'list',
icon: '📋',
name: '商品列表',
synced: true,
stale: '(空)',
actual: '(空)'
},
{
id: 'total',
icon: '💰',
name: '总价',
synced: true,
stale: '¥0',
actual: '¥0'
},
{
id: 'status',
icon: '⚠️',
name: '状态提示',
synced: true,
stale: '正常',
actual: '正常'
}
])
function addManual() {
const name = products[productIndex.value % products.length]
productIndex.value++
manualCount.value++
manualItems.value.push(name)
pendingManualCount = manualCount.value
const price = parseInt(name.match(/¥(\d+)/)[1])
const totalPrice = manualItems.value.reduce((sum, item) => {
return sum + parseInt(item.match(/¥(\d+)/)[1])
}, 0)
manualAreas[0].actual = `${manualCount.value}`
manualAreas[0].synced = false
manualAreas[1].actual = manualItems.value.join('、')
manualAreas[1].synced = false
manualAreas[2].actual = `¥${totalPrice}`
manualAreas[2].synced = false
manualAreas[3].actual = manualCount.value > 5 ? '⚠️ 商品过多!' : '正常'
manualAreas[3].synced = false
const unsyncedBefore = manualAreas.filter(a => !a.synced).length
if (unsyncedBefore > 0 && manualCount.value > 1) {
missCount.value++
}
}
function syncArea(area) {
area.synced = true
area.stale = area.actual
}
function resetManual() {
manualCount.value = 0
manualItems.value = []
missCount.value = 0
pendingManualCount = 0
manualAreas.forEach(a => {
a.synced = true
a.stale = a.id === 'count' ? '0 件' : a.id === 'list' ? '(空)' : a.id === 'total' ? '¥0' : '正常'
a.actual = a.stale
})
}
const autoCount = ref(0)
const autoItems = ref([])
const autoAreas = computed(() => {
const totalPrice = autoItems.value.reduce((sum, item) => {
return sum + parseInt(item.match(/¥(\d+)/)[1])
}, 0)
return [
{ id: 'count', icon: '🔴', name: '购物车数量', value: `${autoCount.value}` },
{ id: 'list', icon: '📋', name: '商品列表', value: autoItems.value.length ? autoItems.value.join('、') : '(空)' },
{ id: 'total', icon: '💰', name: '总价', value: `¥${totalPrice}` },
{ id: 'status', icon: '⚠️', name: '状态提示', value: autoCount.value > 5 ? '⚠️ 商品过多!' : '正常' }
]
})
function addAuto() {
const name = products[productIndex.value % products.length]
productIndex.value++
autoCount.value++
autoItems.value.push(name)
}
function resetAuto() {
autoCount.value = 0
autoItems.value = []
}
</script>
<style scoped>
.sync-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;
}
.comparison-container {
display: grid;
grid-template-columns: 1fr auto 1fr;
gap: 1rem;
align-items: start;
}
.side {
display: flex;
flex-direction: column;
gap: 0.75rem;
}
.side-header {
text-align: center;
}
.badge {
display: inline-block;
padding: 0.25rem 0.75rem;
border-radius: 9999px;
font-size: 0.75rem;
font-weight: 600;
}
.badge.manual {
background: rgba(239, 68, 68, 0.1);
color: var(--vp-c-danger-1);
}
.badge.auto {
background: rgba(16, 185, 129, 0.1);
color: var(--vp-c-green-1);
}
.cart-control {
display: flex;
gap: 0.5rem;
justify-content: center;
}
.action-btn {
padding: 0.35rem 0.75rem;
background-color: var(--vp-c-brand);
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 0.82rem;
transition: opacity 0.2s;
}
.action-btn:hover {
opacity: 0.85;
}
.action-btn.outline {
background: transparent;
border: 1px solid var(--vp-c-divider);
color: var(--vp-c-text-1);
}
.action-btn.outline:hover {
border-color: var(--vp-c-brand);
color: var(--vp-c-brand);
}
.sync-areas {
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.sync-area {
background: var(--vp-c-bg);
border: 2px solid var(--vp-c-divider);
border-radius: 6px;
padding: 0.5rem 0.75rem;
transition: all 0.3s ease;
}
.sync-area.synced {
border-color: var(--vp-c-green-1);
}
.sync-area.unsynced {
border-color: var(--vp-c-danger-1);
background: rgba(239, 68, 68, 0.05);
}
.area-header {
display: flex;
align-items: center;
gap: 0.4rem;
margin-bottom: 0.25rem;
}
.area-icon {
font-size: 0.85rem;
}
.area-name {
font-size: 0.82rem;
font-weight: 600;
}
.sync-badge {
margin-left: auto;
font-size: 0.65rem;
padding: 1px 6px;
border-radius: 4px;
font-weight: 600;
background: var(--vp-c-danger-1);
color: white;
}
.sync-badge.synced {
background: var(--vp-c-green-1);
}
.area-value {
font-size: 0.8rem;
color: var(--vp-c-text-2);
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.sync-btn {
margin-top: 0.35rem;
padding: 0.2rem 0.5rem;
font-size: 0.72rem;
border: 1px solid var(--vp-c-danger-1);
color: var(--vp-c-danger-1);
background: transparent;
border-radius: 4px;
cursor: pointer;
transition: all 0.2s;
}
.sync-btn:hover {
background: var(--vp-c-danger-1);
color: white;
}
.miss-counter {
text-align: center;
font-size: 0.82rem;
padding: 0.5rem;
background: var(--vp-c-bg);
border-radius: 6px;
border: 1px solid var(--vp-c-divider);
}
.miss-label {
color: var(--vp-c-text-2);
}
.miss-value {
font-weight: bold;
color: var(--vp-c-green-1);
}
.miss-value.danger {
color: var(--vp-c-danger-1);
}
.vs-divider {
display: flex;
align-items: center;
justify-content: center;
padding-top: 4rem;
}
.vs-badge {
width: 40px;
height: 40px;
border-radius: 50%;
background: var(--vp-c-brand);
color: white;
display: flex;
align-items: center;
justify-content: center;
font-weight: bold;
font-size: 0.875rem;
flex-shrink: 0;
}
.info-box {
background: var(--vp-c-bg-alt);
padding: 0.75rem;
border-radius: 6px;
font-size: 0.85rem;
color: var(--vp-c-text-2);
display: flex;
gap: 0.25rem;
margin-top: 0.75rem;
}
.info-box strong {
white-space: nowrap;
flex-shrink: 0;
color: var(--vp-c-text-1);
}
@media (max-width: 768px) {
.comparison-container {
grid-template-columns: 1fr;
gap: 0.75rem;
}
.vs-divider {
padding-top: 0;
}
}
</style>