feat(docs): enhance JavaScript runtime and browser-as-os content

refactor(demos): improve variable box, scope, and type annotation demos
style(demos): update visual styles and animations for better UX
docs(browser-as-os): restructure content with tables and practical examples
feat(demos): add new TypeScript and runtime environment demos
This commit is contained in:
sanbuphy
2026-02-17 01:39:59 +08:00
parent 9ee3312569
commit 47377646df
20 changed files with 8097 additions and 1310 deletions
@@ -0,0 +1,652 @@
<script setup>
import { ref } from 'vue'
const isAnimating = ref(false)
const currentStep = ref(0)
const syncCode = ref([
{ id: 1, code: 'console.log("1")', type: 'sync', output: '1' },
{ id: 2, code: 'setTimeout(() => console.log("2"), 0)', type: 'macro', output: '2' },
{ id: 3, code: 'Promise.resolve().then(() => console.log("3"))', type: 'micro', output: '3' },
{ id: 4, code: 'console.log("4")', type: 'sync', output: '4' },
{ id: 5, code: 'setTimeout(() => console.log("5"), 0)', type: 'macro', output: '5' }
])
const microTaskQueue = ref([])
const macroTaskQueue = ref([])
const outputLog = ref([])
const executionSteps = [
{ description: '执行 console.log("1")', action: 'execute', output: '1', source: '同步' },
{ description: '遇到 setTimeout,将回调加入宏任务队列', action: 'add-macro', task: 'console.log("2")' },
{ description: '遇到 Promise.then,将回调加入微任务队列', action: 'add-micro', task: 'console.log("3")' },
{ description: '执行 console.log("4")', action: 'execute', output: '4', source: '同步' },
{ description: '遇到 setTimeout,将回调加入宏任务队列', action: 'add-macro', task: 'console.log("5")' },
{ description: '同步代码执行完毕,检查微任务队列', action: 'check-micro' },
{ description: '执行微任务: console.log("3")', action: 'execute-micro', output: '3', source: '微任务' },
{ description: '微任务队列为空,检查宏任务队列', action: 'check-macro' },
{ description: '执行宏任务: console.log("2")', action: 'execute-macro', output: '2', source: '宏任务' },
{ description: '检查微任务队列(空)', action: 'check-micro' },
{ description: '执行宏任务: console.log("5")', action: 'execute-macro', output: '5', source: '宏任务' },
{ description: '所有任务执行完毕', action: 'done' }
]
const reset = () => {
currentStep.value = 0
microTaskQueue.value = []
macroTaskQueue.value = []
outputLog.value = []
isAnimating.value = false
}
const nextStep = () => {
if (currentStep.value >= executionSteps.length) return
const step = executionSteps[currentStep.value]
switch (step.action) {
case 'execute':
outputLog.value.push({ output: step.output, source: step.source })
break
case 'add-macro':
macroTaskQueue.value.push({ code: step.task, status: 'pending' })
break
case 'add-micro':
microTaskQueue.value.push({ code: step.task, status: 'pending' })
break
case 'check-micro':
if (microTaskQueue.value.length > 0) {
microTaskQueue.value[0].status = 'ready'
}
break
case 'execute-micro':
if (microTaskQueue.value.length > 0) {
outputLog.value.push({ output: step.output, source: step.source })
microTaskQueue.value.shift()
}
break
case 'check-macro':
if (macroTaskQueue.value.length > 0) {
macroTaskQueue.value[0].status = 'ready'
}
break
case 'execute-macro':
if (macroTaskQueue.value.length > 0) {
outputLog.value.push({ output: step.output, source: step.source })
macroTaskQueue.value.shift()
}
break
}
currentStep.value++
}
const play = async () => {
if (isAnimating.value) return
isAnimating.value = true
reset()
while (currentStep.value < executionSteps.length && isAnimating.value) {
nextStep()
await new Promise(resolve => setTimeout(resolve, 1000))
}
isAnimating.value = false
}
const stop = () => {
isAnimating.value = false
}
</script>
<template>
<div class="task-queue-demo">
<h3>任务队列:宏任务 vs 微任务</h3>
<!-- 代码展示 -->
<div class="code-section">
<h4>代码示例</h4>
<div class="code-display">
<div
v-for="(item, index) in syncCode"
:key="item.id"
class="code-item"
:class="{
'current': currentStep === index,
'executed': currentStep > index && index < 4
}"
>
<span class="item-number">{{ item.id }}</span>
<span class="item-code" :class="`type-${item.type}`">{{ item.code }}</span>
<span v-if="item.type === 'sync'" class="item-tag">同步</span>
<span v-else-if="item.type === 'micro'" class="item-tag micro">微任务</span>
<span v-else-if="item.type === 'macro'" class="item-tag macro">宏任务</span>
</div>
</div>
</div>
<!-- 执行过程可视化 -->
<div class="visualization">
<!-- 调用栈 -->
<div class="stack-panel">
<h4>调用栈 (正在执行)</h4>
<div class="stack-content">
<div v-if="currentStep < executionSteps.length" class="current-action">
{{ executionSteps[currentStep]?.description }}
</div>
<div v-else class="current-action done">
执行完成
</div>
</div>
</div>
<!-- 微任务队列 -->
<div class="queue-panel micro">
<h4>
微任务队列
<span class="badge">Microtask</span>
</h4>
<div class="queue-content">
<transition-group name="task-item">
<div
v-for="(task, index) in microTaskQueue"
:key="`micro-${index}`"
class="task-item micro"
:class="{ 'ready': task.status === 'ready' }"
>
<div class="task-code">{{ task.code }}</div>
<div v-if="task.status === 'ready'" class="task-status"> 就绪</div>
<div v-else class="task-status"> 等待</div>
</div>
</transition-group>
<div v-if="microTaskQueue.length === 0" class="empty-queue">
队列为空
</div>
</div>
</div>
<!-- 宏任务队列 -->
<div class="queue-panel macro">
<h4>
宏任务队列
<span class="badge">Macrotask</span>
</h4>
<div class="queue-content">
<transition-group name="task-item">
<div
v-for="(task, index) in macroTaskQueue"
:key="`macro-${index}`"
class="task-item macro"
:class="{ 'ready': task.status === 'ready' }"
>
<div class="task-code">{{ task.code }}</div>
<div v-if="task.status === 'ready'" class="task-status"> 就绪</div>
<div v-else class="task-status"> 等待</div>
</div>
</transition-group>
<div v-if="macroTaskQueue.length === 0" class="empty-queue">
队列为空
</div>
</div>
</div>
</div>
<!-- 输出日志 -->
<div class="output-section">
<h4>输出日志 (执行顺序)</h4>
<div class="output-log">
<div v-if="outputLog.length === 0" class="empty-log">
等待输出...
</div>
<transition-group name="output">
<div
v-for="(log, index) in outputLog"
:key="`log-${index}`"
class="log-entry"
>
<span class="log-output">{{ log.output }}</span>
<span class="log-source">({{ log.source }})</span>
</div>
</transition-group>
</div>
</div>
<!-- 控制按钮 -->
<div class="controls">
<button @click="play" :disabled="isAnimating" class="btn-play">
{{ isAnimating ? '执行中...' : '▶ 自动演示' }}
</button>
<button @click="nextStep" :disabled="isAnimating || currentStep >= executionSteps.length" class="btn-step">
单步执行
</button>
<button @click="stop" :disabled="!isAnimating" class="btn-stop">
停止
</button>
<button @click="reset" :disabled="isAnimating" class="btn-reset">
🔄 重置
</button>
</div>
<!-- 执行规则 -->
<div class="rules-box">
<h4>执行顺序规则</h4>
<div class="rule-list">
<div class="rule-item">
<span class="rule-number">1</span>
<span class="rule-text">执行所有同步代码</span>
</div>
<div class="rule-item">
<span class="rule-number">2</span>
<span class="rule-text">执行微任务队列中的所有任务</span>
</div>
<div class="rule-item">
<span class="rule-number">3</span>
<span class="rule-text">执行一个宏任务</span>
</div>
<div class="rule-item">
<span class="rule-number">4</span>
<span class="rule-text">重复步骤 2-3</span>
</div>
</div>
<p class="highlight">
<strong>核心要点:</strong> 微任务优先级高于宏任务每次执行完一个宏任务后,都会检查并执行所有微任务,然后再执行下一个宏任务
</p>
</div>
</div>
</template>
<style scoped>
.task-queue-demo {
border: 1px solid var(--vp-c-border);
border-radius: 12px;
padding: 24px;
margin: 24px 0;
background: var(--vp-c-bg);
}
h3 {
margin: 0 0 20px 0;
font-size: 18px;
font-weight: 600;
color: var(--vp-c-text-1);
}
h4 {
margin: 0 0 12px 0;
font-size: 14px;
font-weight: 600;
color: var(--vp-c-text-2);
}
.code-section {
margin-bottom: 20px;
}
.code-display {
background: #1e1e1e;
border-radius: 8px;
padding: 16px;
font-family: 'Courier New', monospace;
}
.code-item {
display: flex;
align-items: center;
gap: 12px;
padding: 8px;
border-radius: 4px;
transition: all 0.3s ease;
}
.code-item.current {
background: rgba(62, 175, 124, 0.2);
border-left: 3px solid var(--vp-c-brand-1);
}
.code-item.executed {
opacity: 0.5;
}
.item-number {
color: #858585;
font-size: 12px;
min-width: 20px;
}
.item-code {
flex: 1;
color: #d4d4d4;
font-size: 13px;
}
.item-code.type-micro {
color: #68d391;
}
.item-code.type-macro {
color: #f687b3;
}
.item-tag {
padding: 4px 8px;
border-radius: 4px;
font-size: 11px;
font-weight: 600;
background: var(--vp-c-bg-soft);
color: var(--vp-c-text-2);
}
.item-tag.micro {
background: rgba(104, 217, 145, 0.2);
color: #68d391;
}
.item-tag.macro {
background: rgba(246, 135, 179, 0.2);
color: #f687b3;
}
.visualization {
display: grid;
grid-template-columns: 1fr 1fr 1fr;
gap: 16px;
margin-bottom: 20px;
}
@media (max-width: 1024px) {
.visualization {
grid-template-columns: 1fr;
}
}
.stack-panel,
.queue-panel {
border: 2px solid var(--vp-c-border);
border-radius: 8px;
padding: 16px;
background: var(--vp-c-bg-soft);
min-height: 250px;
}
.queue-panel.micro {
border-color: #68d391;
}
.queue-panel.macro {
border-color: #f687b3;
}
.badge {
margin-left: 8px;
padding: 4px 8px;
border-radius: 4px;
font-size: 11px;
font-weight: 600;
background: var(--vp-c-bg);
color: var(--vp-c-text-2);
}
.stack-content,
.queue-content {
min-height: 200px;
}
.current-action {
padding: 16px;
background: var(--vp-c-bg);
border-radius: 8px;
border-left: 4px solid var(--vp-c-brand-1);
font-size: 14px;
color: var(--vp-c-text-1);
line-height: 1.6;
}
.current-action.done {
border-color: #48bb78;
text-align: center;
font-weight: 600;
}
.task-item {
padding: 12px;
margin-bottom: 8px;
background: var(--vp-c-bg);
border: 2px solid var(--vp-c-border);
border-radius: 6px;
transition: all 0.3s ease;
}
.task-item.micro {
border-color: #68d391;
}
.task-item.macro {
border-color: #f687b3;
}
.task-item.ready {
animation: pulse 1s ease infinite;
}
@keyframes pulse {
0%, 100% { box-shadow: 0 0 0 0 rgba(104, 217, 145, 0.4); }
50% { box-shadow: 0 0 0 6px rgba(104, 217, 145, 0); }
}
.task-code {
font-family: 'Courier New', monospace;
font-size: 12px;
color: var(--vp-c-text-1);
margin-bottom: 4px;
}
.task-status {
font-size: 11px;
font-weight: 600;
color: var(--vp-c-text-2);
}
.task-item-enter-active,
.task-item-leave-active {
transition: all 0.3s ease;
}
.task-item-enter-from {
opacity: 0;
transform: translateX(-20px);
}
.task-item-leave-to {
opacity: 0;
transform: translateX(20px);
}
.empty-queue {
display: flex;
align-items: center;
justify-content: center;
height: 100%;
color: var(--vp-c-text-3);
font-size: 13px;
}
.output-section {
margin-bottom: 20px;
}
.output-log {
min-height: 60px;
padding: 12px;
border: 2px solid var(--vp-c-border);
border-radius: 8px;
background: var(--vp-c-bg);
display: flex;
flex-wrap: wrap;
gap: 8px;
}
.empty-log {
color: var(--vp-c-text-3);
font-size: 14px;
}
.log-entry {
padding: 8px 12px;
background: var(--vp-c-brand-1);
color: white;
border-radius: 6px;
font-size: 14px;
font-weight: 600;
animation: slideIn 0.3s ease;
}
@keyframes slideIn {
from {
opacity: 0;
transform: translateY(-10px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
.log-output {
font-family: 'Courier New', monospace;
}
.log-source {
margin-left: 8px;
font-size: 12px;
opacity: 0.8;
}
.output-enter-active,
.output-leave-active {
transition: all 0.3s ease;
}
.output-enter-from {
opacity: 0;
transform: translateY(-10px);
}
.output-leave-to {
opacity: 0;
transform: translateY(10px);
}
.controls {
display: flex;
gap: 12px;
justify-content: center;
margin-bottom: 20px;
flex-wrap: wrap;
}
button {
padding: 10px 20px;
border: none;
border-radius: 6px;
font-size: 14px;
font-weight: 600;
cursor: pointer;
transition: all 0.2s ease;
}
button:active {
transform: scale(0.95);
}
.btn-play {
background: var(--vp-c-brand-1);
color: white;
}
.btn-play:hover:not(:disabled) {
background: var(--vp-c-brand-2);
}
.btn-step {
background: var(--vp-c-bg-soft);
color: var(--vp-c-text-1);
}
.btn-step:hover:not(:disabled) {
background: var(--vp-c-bg-soft-hover);
}
.btn-stop {
background: #ed8936;
color: white;
}
.btn-stop:hover:not(:disabled) {
background: #dd6b20;
}
.btn-reset {
background: #f56565;
color: white;
}
.btn-reset:hover:not(:disabled) {
background: #e53e3e;
}
button:disabled {
opacity: 0.5;
cursor: not-allowed;
}
.rules-box {
background: var(--vp-c-bg-soft);
border-left: 4px solid var(--vp-c-brand-1);
border-radius: 8px;
padding: 16px;
}
.rule-list {
display: flex;
flex-direction: column;
gap: 12px;
margin-bottom: 16px;
}
.rule-item {
display: flex;
align-items: center;
gap: 12px;
}
.rule-number {
display: flex;
align-items: center;
justify-content: center;
width: 28px;
height: 28px;
border-radius: 50%;
background: var(--vp-c-brand-1);
color: white;
font-size: 14px;
font-weight: 600;
flex-shrink: 0;
}
.rule-text {
font-size: 14px;
color: var(--vp-c-text-1);
}
.highlight {
padding: 12px;
background: rgba(62, 175, 124, 0.1);
border-radius: 6px;
font-size: 14px;
line-height: 1.6;
color: var(--vp-c-text-1);
margin: 0;
}
.highlight strong {
color: var(--vp-c-brand-1);
}
</style>