Files
test-repo/docs/.vitepress/theme/components/appendix/web-basics/TcpHandshakeDemo.vue
T

229 lines
4.6 KiB
Vue
Raw Normal View History

2026-01-15 20:10:19 +08:00
<template>
<div class="tcp-handshake-demo">
<div class="diagram">
<!-- Client Column -->
<div class="column client">
<div class="actor-icon">💻 Client</div>
<div class="state-label">{{ clientState }}</div>
2026-01-15 20:10:19 +08:00
</div>
<!-- Interaction Area -->
<div class="interaction-zone">
<!-- Step 1: SYN -->
<div class="packet-row" :class="{ active: step === 1, done: step > 1 }">
<button
@click="sendSyn"
:disabled="step !== 0"
class="packet-btn syn"
2026-01-15 20:10:19 +08:00
>
SYN (SEQ=x)
</button>
2026-01-15 20:10:19 +08:00
</div>
<!-- Step 2: SYN-ACK -->
<div
class="packet-row reverse"
:class="{ active: step === 2, done: step > 2 }"
>
<button
@click="sendSynAck"
:disabled="step !== 1"
class="packet-btn syn-ack"
>
SYN-ACK (ACK=x+1, SEQ=y)
</button>
</div>
2026-01-15 20:10:19 +08:00
<!-- Step 3: ACK -->
<div class="packet-row" :class="{ active: step === 3, done: step > 3 }">
<button
@click="sendAck"
:disabled="step !== 2"
class="packet-btn ack"
>
ACK (ACK=y+1)
</button>
</div>
2026-01-15 20:10:19 +08:00
</div>
<!-- Server Column -->
<div class="column server">
<div class="actor-icon">🖥 Server</div>
<div class="state-label">{{ serverState }}</div>
2026-01-15 20:10:19 +08:00
</div>
</div>
<div class="status-message">
<p v-if="step === 0">点击 <strong>SYN</strong> 开始连接</p>
<p v-if="step === 1">
服务器收到了请求现在需要回复 <strong>SYN-ACK</strong>
</p>
<p v-if="step === 2">
客户端收到了确认最后发送 <strong>ACK</strong> 完成握手
</p>
<p v-if="step === 3" class="success">🎉 连接已建立 (ESTABLISHED)!</p>
2026-01-15 20:10:19 +08:00
</div>
<button v-if="step === 3" @click="reset" class="reset-btn">Reset</button>
2026-01-15 20:10:19 +08:00
</div>
</template>
<script setup>
import { ref } from 'vue'
const step = ref(0)
const clientState = ref('CLOSED')
const serverState = ref('LISTEN')
2026-01-15 20:10:19 +08:00
const sendSyn = () => {
step.value = 1
clientState.value = 'SYN_SENT'
}
2026-01-15 20:10:19 +08:00
const sendSynAck = () => {
step.value = 2
serverState.value = 'SYN_RCVD'
}
2026-01-15 20:10:19 +08:00
const sendAck = () => {
step.value = 3
clientState.value = 'ESTABLISHED'
serverState.value = 'ESTABLISHED'
2026-01-15 20:10:19 +08:00
}
const reset = () => {
step.value = 0
clientState.value = 'CLOSED'
serverState.value = 'LISTEN'
2026-01-15 20:10:19 +08:00
}
</script>
<style scoped>
.tcp-handshake-demo {
border: 1px solid var(--vp-c-divider);
border-radius: 8px;
background-color: var(--vp-c-bg-soft);
padding: 1.5rem;
margin: 1rem 0;
font-family: var(--vp-font-family-mono);
text-align: center;
2026-01-15 20:10:19 +08:00
}
.diagram {
2026-01-15 20:10:19 +08:00
display: flex;
justify-content: space-between;
margin-bottom: 2rem;
2026-01-15 20:10:19 +08:00
}
.column {
width: 120px;
display: flex;
flex-direction: column;
gap: 1rem;
2026-01-15 20:10:19 +08:00
}
.actor-icon {
font-size: 1.2rem;
2026-01-15 20:10:19 +08:00
font-weight: bold;
}
.state-label {
padding: 0.5rem;
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
2026-01-15 20:10:19 +08:00
font-size: 0.8rem;
color: var(--vp-c-text-2);
2026-01-15 20:10:19 +08:00
}
.interaction-zone {
2026-01-15 20:10:19 +08:00
flex: 1;
display: flex;
flex-direction: column;
justify-content: center;
gap: 1.5rem;
padding: 0 2rem;
2026-01-15 20:10:19 +08:00
}
.packet-row {
display: flex;
justify-content: flex-start;
2026-01-15 20:10:19 +08:00
opacity: 0.3;
transition: all 0.3s;
}
.packet-row.reverse {
justify-content: flex-end;
2026-01-15 20:10:19 +08:00
}
.packet-row.active {
opacity: 1;
transform: scale(1.05);
2026-01-15 20:10:19 +08:00
}
.packet-row.done {
opacity: 1;
2026-01-15 20:10:19 +08:00
}
.packet-btn {
padding: 0.5rem 1rem;
border-radius: 20px;
2026-01-15 20:10:19 +08:00
border: none;
cursor: pointer;
font-family: monospace;
font-size: 0.85rem;
2026-01-15 20:10:19 +08:00
transition: all 0.2s;
background: var(--vp-c-bg-alt);
border: 1px solid var(--vp-c-divider);
2026-01-15 20:10:19 +08:00
}
.packet-btn:not(:disabled):hover {
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
2026-01-15 20:10:19 +08:00
}
.packet-btn.syn {
background: #3b82f6;
color: white;
2026-01-15 20:10:19 +08:00
}
.packet-btn.syn-ack {
background: #f59e0b;
color: white;
2026-01-15 20:10:19 +08:00
}
.packet-btn.ack {
background: #10b981;
color: white;
2026-01-15 20:10:19 +08:00
}
.packet-btn:disabled {
background: var(--vp-c-bg-alt);
color: var(--vp-c-text-3);
cursor: not-allowed;
border-color: transparent;
2026-01-15 20:10:19 +08:00
}
.status-message {
height: 2rem;
margin-bottom: 1rem;
2026-01-15 20:10:19 +08:00
font-size: 0.9rem;
color: var(--vp-c-text-2);
}
.status-message .success {
color: #10b981;
2026-01-15 20:10:19 +08:00
font-weight: bold;
}
.reset-btn {
padding: 0.5rem 1.5rem;
2026-01-15 20:10:19 +08:00
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
cursor: pointer;
2026-01-15 20:10:19 +08:00
}
.reset-btn:hover {
background: var(--vp-c-bg-alt);
2026-01-15 20:10:19 +08:00
}
</style>