47377646df
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
653 lines
15 KiB
Vue
653 lines
15 KiB
Vue
<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>
|