Files
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

343 lines
8.2 KiB
Vue
Raw Permalink 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="component-tree-demo">
<div class="demo-header">
<span class="title">组件化拆分</span>
<span class="subtitle">一个页面如何拆成多个独立组件</span>
</div>
<div class="demo-body">
<div class="tree-panel">
<div class="tree-title">组件树结构</div>
<div class="tree-list">
<div
v-for="comp in components"
:key="comp.id"
:class="['tree-item', { active: selected === comp.id }]"
:style="{ paddingLeft: comp.depth * 1 + 'rem' }"
@click="selected = comp.id"
>
<span class="tree-icon">{{ comp.icon }}</span>
<span class="tree-name">{{ comp.name }}</span>
<span v-if="comp.reused" class="reuse-badge">×{{ comp.reused }}</span>
</div>
</div>
</div>
<div class="preview-panel">
<div class="tree-title">页面预览</div>
<div class="page-mock">
<div
:class="['mock-navbar', { highlighted: selected === 'navbar' }]"
@click="selected = 'navbar'"
>
<span>🏠 电商网站</span>
<span
:class="['mock-search', { highlighted: selected === 'search' }]"
@click.stop="selected = 'search'"
>🔍 搜索框</span>
<span
:class="['mock-cart-icon', { highlighted: selected === 'cart' }]"
@click.stop="selected = 'cart'"
>🛒 购物车(3)</span>
</div>
<div class="mock-content">
<div
v-for="i in 3"
:key="i"
:class="['mock-product-card', { highlighted: selected === 'product' }]"
@click="selected = 'product'"
>
<div class="mock-img">📦</div>
<div class="mock-info">
<div class="mock-product-name">商品 {{ i }}</div>
<div class="mock-price">¥{{ i * 99 + 100 }}</div>
</div>
</div>
</div>
<div
:class="['mock-footer', { highlighted: selected === 'footer' }]"
@click="selected = 'footer'"
>
© 2025 电商网站
</div>
</div>
</div>
</div>
<div v-if="selectedComp" class="detail-card">
<div class="detail-name">{{ selectedComp.icon }} {{ selectedComp.name }}</div>
<div class="detail-desc">{{ selectedComp.desc }}</div>
<div class="detail-tags">
<span class="detail-tag">数据独立</span>
<span class="detail-tag">样式隔离</span>
<span v-if="selectedComp.reused" class="detail-tag reuse">
复用 {{ selectedComp.reused }}
</span>
</div>
</div>
<div class="info-box">
<strong>核心思想</strong>
<span>组件化就是把一个大页面拆成多个独立的小块每个组件管理自己的数据界面和样式互不干扰同一个组件可以在不同地方复用多次传入不同的数据就会显示不同的内容</span>
</div>
</div>
</template>
<script setup>
import { ref, computed } from 'vue'
const selected = ref('product')
const components = [
{ id: 'app', name: 'App(根组件)', icon: '📱', depth: 0, desc: '整个应用的根组件,包含所有其他组件。' },
{ id: 'navbar', name: 'NavBar(导航栏)', icon: '🧭', depth: 1, desc: '页面顶部的导航栏,包含 Logo、搜索框和购物车入口。' },
{ id: 'search', name: 'SearchBox(搜索框)', icon: '🔍', depth: 2, desc: '独立的搜索框组件,管理搜索关键词和搜索结果。' },
{ id: 'cart', name: 'CartIcon(购物车图标)', icon: '🛒', depth: 2, desc: '显示购物车数量的小图标,数据来自全局购物车状态。' },
{ id: 'product', name: 'ProductCard(商品卡片)', icon: '📦', depth: 1, reused: 3, desc: '单个商品的展示卡片。写一次代码,传入不同的商品数据就能复用多次,每次显示不同的商品信息。' },
{ id: 'footer', name: 'Footer(页脚)', icon: '📄', depth: 1, desc: '页面底部信息,一般包含版权声明等。' }
]
const selectedComp = computed(() => components.find(c => c.id === selected.value))
</script>
<style scoped>
.component-tree-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;
}
.demo-header {
display: flex;
flex-wrap: wrap;
align-items: baseline;
gap: 0.5rem;
margin-bottom: 0.75rem;
}
.demo-header .title {
font-size: 1rem;
font-weight: 600;
}
.demo-header .subtitle {
font-size: 0.85rem;
color: var(--vp-c-text-2);
}
.demo-body {
display: grid;
grid-template-columns: 1fr 1.4fr;
gap: 0.75rem;
margin-bottom: 0.75rem;
}
.tree-panel,
.preview-panel {
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 6px;
padding: 0.75rem;
}
.tree-title {
font-size: 0.82rem;
font-weight: 600;
color: var(--vp-c-text-2);
margin-bottom: 0.5rem;
}
.tree-list {
display: flex;
flex-direction: column;
gap: 0.2rem;
}
.tree-item {
display: flex;
align-items: center;
gap: 0.4rem;
padding: 0.3rem 0.5rem;
border-radius: 4px;
cursor: pointer;
font-size: 0.8rem;
transition: all 0.2s;
border: 1px solid transparent;
}
.tree-item:hover {
background: var(--vp-c-bg-alt);
}
.tree-item.active {
background: rgba(59, 130, 246, 0.08);
border-color: var(--vp-c-brand);
}
.tree-icon {
font-size: 0.85rem;
}
.tree-name {
font-family: var(--vp-font-family-mono);
font-size: 0.78rem;
}
.reuse-badge {
margin-left: auto;
background: rgba(16, 185, 129, 0.1);
color: var(--vp-c-green-1);
font-size: 0.65rem;
padding: 1px 5px;
border-radius: 4px;
font-weight: 600;
}
.page-mock {
border: 1px solid var(--vp-c-divider);
border-radius: 6px;
overflow: hidden;
font-size: 0.75rem;
}
.mock-navbar {
display: flex;
align-items: center;
justify-content: space-between;
gap: 0.5rem;
padding: 0.4rem 0.6rem;
background: var(--vp-c-bg-alt);
border-bottom: 1px solid var(--vp-c-divider);
cursor: pointer;
transition: all 0.2s;
}
.mock-search,
.mock-cart-icon {
cursor: pointer;
padding: 0.15rem 0.35rem;
border-radius: 3px;
transition: all 0.2s;
}
.mock-content {
display: flex;
gap: 0.5rem;
padding: 0.5rem;
flex-wrap: wrap;
}
.mock-product-card {
flex: 1;
min-width: 60px;
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
padding: 0.4rem;
text-align: center;
cursor: pointer;
transition: all 0.2s;
}
.mock-img {
font-size: 1.2rem;
margin-bottom: 0.2rem;
}
.mock-product-name {
font-size: 0.7rem;
font-weight: 600;
}
.mock-price {
font-size: 0.65rem;
color: var(--vp-c-danger-1);
}
.mock-footer {
padding: 0.3rem 0.6rem;
background: var(--vp-c-bg-alt);
border-top: 1px solid var(--vp-c-divider);
text-align: center;
color: var(--vp-c-text-2);
font-size: 0.65rem;
cursor: pointer;
transition: all 0.2s;
}
.highlighted {
outline: 2px solid var(--vp-c-brand);
outline-offset: -1px;
background: rgba(59, 130, 246, 0.06) !important;
}
.detail-card {
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 6px;
padding: 0.75rem;
margin-bottom: 0.75rem;
}
.detail-name {
font-weight: 600;
font-size: 0.9rem;
margin-bottom: 0.3rem;
}
.detail-desc {
font-size: 0.82rem;
color: var(--vp-c-text-2);
margin-bottom: 0.5rem;
line-height: 1.5;
}
.detail-tags {
display: flex;
gap: 0.35rem;
flex-wrap: wrap;
}
.detail-tag {
font-size: 0.68rem;
padding: 0.15rem 0.4rem;
border-radius: 4px;
background: var(--vp-c-bg-alt);
color: var(--vp-c-text-2);
border: 1px solid var(--vp-c-divider);
}
.detail-tag.reuse {
background: rgba(16, 185, 129, 0.1);
color: var(--vp-c-green-1);
border-color: var(--vp-c-green-1);
}
.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;
}
.info-box strong {
white-space: nowrap;
flex-shrink: 0;
color: var(--vp-c-text-1);
}
@media (max-width: 720px) {
.demo-body {
grid-template-columns: 1fr;
}
.mock-content {
flex-direction: column;
}
}
</style>