335 lines
8.0 KiB
Vue
335 lines
8.0 KiB
Vue
<template>
|
||
<div class="what-is-dom-demo">
|
||
<div class="demo-header">
|
||
<span class="title">HTML → DOM 树</span>
|
||
<span class="subtitle">浏览器如何理解你写的 HTML</span>
|
||
</div>
|
||
|
||
<div class="demo-body">
|
||
<div class="html-panel">
|
||
<div class="panel-title">你写的 HTML 代码</div>
|
||
<div class="code-display">
|
||
<div
|
||
v-for="(line, i) in htmlLines"
|
||
:key="i"
|
||
:class="['code-line', { highlighted: highlightedTag === line.tag }]"
|
||
@mouseenter="highlightedTag = line.tag"
|
||
@mouseleave="highlightedTag = ''"
|
||
>
|
||
<span class="line-num">{{ i + 1 }}</span>
|
||
<span class="line-code" :style="{ paddingLeft: line.indent * 12 + 'px' }">{{ line.text }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="arrow-col">
|
||
<div class="arrow-label">浏览器解析</div>
|
||
<div class="arrow-icon">→</div>
|
||
</div>
|
||
|
||
<div class="tree-panel">
|
||
<div class="panel-title">浏览器生成的 DOM 树</div>
|
||
<div class="tree-display">
|
||
<div
|
||
v-for="node in treeNodes"
|
||
:key="node.id"
|
||
:class="['tree-node', { highlighted: highlightedTag === node.tag }]"
|
||
:style="{ marginLeft: node.depth * 20 + 'px' }"
|
||
@mouseenter="highlightedTag = node.tag"
|
||
@mouseleave="highlightedTag = ''"
|
||
>
|
||
<span v-if="node.depth > 0" class="connector">└─</span>
|
||
<span class="node-tag">{{ node.label }}</span>
|
||
<span v-if="node.text" class="node-text">"{{ node.text }}"</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="dom-explain">
|
||
<div class="explain-item">
|
||
<span class="explain-icon">📄</span>
|
||
<div class="explain-content">
|
||
<strong>节点(Node)</strong>
|
||
<span>DOM 树上的每一个方块就是一个节点。每个 HTML 标签(如 <code><h1></code>、<code><p></code>)都对应一个节点。</span>
|
||
</div>
|
||
</div>
|
||
<div class="explain-item">
|
||
<span class="explain-icon">🌳</span>
|
||
<div class="explain-content">
|
||
<strong>父子关系</strong>
|
||
<span>标签嵌套在另一个标签里面,在 DOM 树上就是父节点和子节点的关系。<code><body></code> 里包含 <code><h1></code>,所以 body 是 h1 的父节点。</span>
|
||
</div>
|
||
</div>
|
||
<div class="explain-item">
|
||
<span class="explain-icon">✏️</span>
|
||
<div class="explain-content">
|
||
<strong>DOM 操作</strong>
|
||
<span>JavaScript 可以增加、删除、修改 DOM 树上的节点。修改节点后,浏览器会重新计算布局并重新绘制页面,这就是"DOM 操作"。</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="info-box">
|
||
<strong>关键概念:</strong>
|
||
<span>DOM 是浏览器在内存中维护的一棵树,它和你写的 HTML 一一对应。JavaScript 无法直接修改 HTML 文件,它修改的是这棵 DOM 树——浏览器再根据 DOM 树的变化更新屏幕上的显示。</span>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref } from 'vue'
|
||
|
||
const highlightedTag = ref('')
|
||
|
||
const htmlLines = [
|
||
{ text: '<html>', indent: 0, tag: 'html' },
|
||
{ text: '<body>', indent: 1, tag: 'body' },
|
||
{ text: '<h1>我的购物车</h1>', indent: 2, tag: 'h1' },
|
||
{ text: '<p>共 3 件商品</p>', indent: 2, tag: 'p' },
|
||
{ text: '<ul>', indent: 2, tag: 'ul' },
|
||
{ text: '<li>耳机</li>', indent: 3, tag: 'li1' },
|
||
{ text: '<li>键盘</li>', indent: 3, tag: 'li2' },
|
||
{ text: '<li>鼠标</li>', indent: 3, tag: 'li3' },
|
||
{ text: '</ul>', indent: 2, tag: 'ul' },
|
||
{ text: '<button>结算</button>', indent: 2, tag: 'btn' },
|
||
{ text: '</body>', indent: 1, tag: 'body' },
|
||
{ text: '</html>', indent: 0, tag: 'html' }
|
||
]
|
||
|
||
const treeNodes = [
|
||
{ id: 1, label: 'html', depth: 0, tag: 'html' },
|
||
{ id: 2, label: 'body', depth: 1, tag: 'body' },
|
||
{ id: 3, label: 'h1', depth: 2, tag: 'h1', text: '我的购物车' },
|
||
{ id: 4, label: 'p', depth: 2, tag: 'p', text: '共 3 件商品' },
|
||
{ id: 5, label: 'ul', depth: 2, tag: 'ul' },
|
||
{ id: 6, label: 'li', depth: 3, tag: 'li1', text: '耳机' },
|
||
{ id: 7, label: 'li', depth: 3, tag: 'li2', text: '键盘' },
|
||
{ id: 8, label: 'li', depth: 3, tag: 'li3', text: '鼠标' },
|
||
{ id: 9, label: 'button', depth: 2, tag: 'btn', text: '结算' }
|
||
]
|
||
</script>
|
||
|
||
<style scoped>
|
||
.what-is-dom-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 auto 1fr;
|
||
gap: 0.5rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
.html-panel,
|
||
.tree-panel {
|
||
background: var(--vp-c-bg);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
padding: 0.6rem;
|
||
}
|
||
|
||
.panel-title {
|
||
font-size: 0.78rem;
|
||
font-weight: 600;
|
||
color: var(--vp-c-text-2);
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
.code-display {
|
||
font-family: var(--vp-font-family-mono);
|
||
font-size: 0.75rem;
|
||
line-height: 1.6;
|
||
}
|
||
|
||
.code-line {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.4rem;
|
||
padding: 0.1rem 0.3rem;
|
||
border-radius: 3px;
|
||
cursor: default;
|
||
transition: background 0.15s;
|
||
}
|
||
|
||
.code-line.highlighted {
|
||
background: rgba(59, 130, 246, 0.1);
|
||
}
|
||
|
||
.line-num {
|
||
color: var(--vp-c-text-3);
|
||
font-size: 0.65rem;
|
||
min-width: 1rem;
|
||
text-align: right;
|
||
flex-shrink: 0;
|
||
user-select: none;
|
||
}
|
||
|
||
.line-code {
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.arrow-col {
|
||
display: flex;
|
||
flex-direction: column;
|
||
align-items: center;
|
||
justify-content: center;
|
||
gap: 0.3rem;
|
||
padding-top: 1.5rem;
|
||
}
|
||
|
||
.arrow-label {
|
||
font-size: 0.68rem;
|
||
color: var(--vp-c-text-2);
|
||
writing-mode: vertical-rl;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.arrow-icon {
|
||
font-size: 1.2rem;
|
||
color: var(--vp-c-brand);
|
||
font-weight: bold;
|
||
}
|
||
|
||
.tree-display {
|
||
font-family: var(--vp-font-family-mono);
|
||
font-size: 0.75rem;
|
||
line-height: 1.7;
|
||
}
|
||
|
||
.tree-node {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.3rem;
|
||
padding: 0.1rem 0.3rem;
|
||
border-radius: 3px;
|
||
cursor: default;
|
||
transition: background 0.15s;
|
||
}
|
||
|
||
.tree-node.highlighted {
|
||
background: rgba(59, 130, 246, 0.1);
|
||
}
|
||
|
||
.connector {
|
||
color: var(--vp-c-text-3);
|
||
font-size: 0.7rem;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.node-tag {
|
||
background: var(--vp-c-bg-alt);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 3px;
|
||
padding: 0 0.3rem;
|
||
font-weight: 600;
|
||
font-size: 0.72rem;
|
||
color: var(--vp-c-brand);
|
||
}
|
||
|
||
.tree-node.highlighted .node-tag {
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.node-text {
|
||
color: var(--vp-c-text-2);
|
||
font-size: 0.7rem;
|
||
}
|
||
|
||
.dom-explain {
|
||
display: grid;
|
||
grid-template-columns: repeat(3, 1fr);
|
||
gap: 0.5rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
.explain-item {
|
||
background: var(--vp-c-bg);
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
padding: 0.6rem;
|
||
display: flex;
|
||
gap: 0.4rem;
|
||
}
|
||
|
||
.explain-icon {
|
||
font-size: 1rem;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.explain-content {
|
||
font-size: 0.78rem;
|
||
color: var(--vp-c-text-2);
|
||
line-height: 1.4;
|
||
}
|
||
|
||
.explain-content strong {
|
||
display: block;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 0.15rem;
|
||
font-size: 0.8rem;
|
||
}
|
||
|
||
.explain-content code {
|
||
background: var(--vp-c-bg-alt);
|
||
padding: 0 0.2rem;
|
||
border-radius: 2px;
|
||
font-size: 0.72rem;
|
||
}
|
||
|
||
.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;
|
||
}
|
||
.arrow-col {
|
||
flex-direction: row;
|
||
padding-top: 0;
|
||
}
|
||
.arrow-label {
|
||
writing-mode: horizontal-tb;
|
||
}
|
||
.dom-explain {
|
||
grid-template-columns: 1fr;
|
||
}
|
||
}
|
||
</style>
|