Files
test-repo/docs/.vitepress/theme/components/appendix/api-intro/ApiMethodDemo.vue
T
2026-01-19 23:45:08 +08:00

439 lines
8.7 KiB
Vue
Raw 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.
<!--
ApiMethodDemo.vue
参考 ide-intro 虚拟环境演示风格
目标 GET/POST/DELETE 讲成//三个按钮并用可视化列表展示效果
注意这是选读内容但组件本身要足够好玩足够直观
-->
<template>
<div class="wrap">
<div class="head">
<div class="title">三个按钮GET/ POST/ DELETE</div>
<div class="sub">你不用记英文先玩点按钮看看列表怎么变</div>
</div>
<div class="board">
<div class="left">
<div class="panelTitle">小列表服务器里有的东西</div>
<div class="list">
<div v-if="items.length === 0" class="empty">空的</div>
<div v-for="it in items" :key="it.id" class="row">
<div class="pillId">#{{ it.id }}</div>
<div class="name">{{ it.name }}</div>
</div>
</div>
<div class="mini">
<span class="miniK">提示</span>
<span class="miniV">你可以一直点GET列表不会变</span>
</div>
</div>
<div class="right">
<div class="panelTitle">按按钮模拟 API</div>
<div class="btnRow">
<button class="btn get" :disabled="busy" @click="getList">
GET
</button>
<button class="btn post" :disabled="busy" @click="addOne">
POST
</button>
<button class="btn del" :disabled="busy" @click="removeOne">
DELETE
</button>
</div>
<div class="inputs">
<label class="field">
<span class="k">要加什么</span>
<input v-model="newName" class="input" placeholder="随便写个名字" />
</label>
</div>
<div class="result">
<div class="resultTitle">返回结果</div>
<div v-if="!last" class="muted">点一个按钮试试</div>
<div v-else class="resBox" :class="{ ok: last.ok, bad: !last.ok }">
<div class="badge">{{ last.ok ? '成功' : '失败' }}</div>
<div class="text">{{ last.text }}</div>
</div>
</div>
<div class="foot">
<div class="stat">
<span class="statK">你点了</span>
<span class="statV">{{ clicks }}</span>
<span class="statK"></span>
</div>
<button class="ghost" :disabled="busy" @click="reset">重置</button>
</div>
</div>
</div>
<div class="one">
<div class="oneTitle">一句话总结</div>
<div class="oneText">
GET 通常只是拿数据POST/DELETE 改数据所以网络抖动时重试 POST
要更小心
</div>
</div>
</div>
</template>
<script setup>
import { ref } from 'vue'
const busy = ref(false)
const clicks = ref(0)
const seq = ref(3)
const items = ref([
{ id: 1, name: 'Apple' },
{ id: 2, name: 'Banana' },
{ id: 3, name: 'Cookie' }
])
const newName = ref('Donut')
const last = ref(null)
function sleep(ms) {
return new Promise((r) => setTimeout(r, ms))
}
async function getList() {
clicks.value += 1
busy.value = true
await sleep(220)
last.value = {
ok: true,
text: `拿到了 ${items.value.length} 条数据(列表不变)`
}
busy.value = false
}
async function addOne() {
clicks.value += 1
busy.value = true
await sleep(260)
const name = String(newName.value || '').trim()
if (!name) {
last.value = { ok: false, text: '你还没写“要加什么”' }
busy.value = false
return
}
seq.value += 1
items.value = [...items.value, { id: seq.value, name }]
last.value = { ok: true, text: `已添加:#${seq.value} ${name}` }
busy.value = false
}
async function removeOne() {
clicks.value += 1
busy.value = true
await sleep(240)
if (items.value.length === 0) {
last.value = { ok: false, text: '已经空了,删不了' }
busy.value = false
return
}
const removed = items.value[items.value.length - 1]
items.value = items.value.slice(0, -1)
last.value = { ok: true, text: `已删除:#${removed.id} ${removed.name}` }
busy.value = false
}
function reset() {
seq.value = 3
items.value = [
{ id: 1, name: 'Apple' },
{ id: 2, name: 'Banana' },
{ id: 3, name: 'Cookie' }
]
newName.value = 'Donut'
last.value = null
clicks.value = 0
}
</script>
<style scoped>
.wrap {
border: 1px solid var(--vp-c-divider);
border-radius: 14px;
background: var(--vp-c-bg-soft);
padding: 16px;
}
.head {
display: flex;
flex-direction: column;
gap: 6px;
}
.title {
font-weight: 900;
font-size: 16px;
color: var(--vp-c-text-1);
}
.sub {
font-size: 13px;
color: var(--vp-c-text-2);
}
.board {
margin-top: 12px;
display: grid;
grid-template-columns: 1fr 1fr;
gap: 12px;
}
.left,
.right {
border: 1px solid var(--vp-c-divider);
border-radius: 12px;
background: var(--vp-c-bg);
padding: 12px;
}
.panelTitle {
font-weight: 900;
font-size: 13px;
color: var(--vp-c-text-1);
}
.list {
margin-top: 10px;
border: 1px solid var(--vp-c-divider);
border-radius: 12px;
background: var(--vp-c-bg-soft);
padding: 10px;
display: grid;
gap: 8px;
min-height: 160px;
}
.empty {
font-size: 12px;
color: var(--vp-c-text-3);
}
.row {
display: flex;
gap: 10px;
align-items: center;
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg);
border-radius: 12px;
padding: 8px 10px;
}
.pillId {
font-size: 12px;
font-weight: 900;
color: var(--vp-c-text-2);
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg-soft);
border-radius: 999px;
padding: 2px 8px;
}
.name {
font-size: 13px;
font-weight: 900;
color: var(--vp-c-text-1);
}
.mini {
margin-top: 10px;
font-size: 12px;
color: var(--vp-c-text-2);
}
.miniK {
font-weight: 900;
}
.btnRow {
margin-top: 10px;
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 10px;
}
.btn {
border-radius: 12px;
padding: 10px 12px;
font-weight: 900;
cursor: pointer;
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg-soft);
color: var(--vp-c-text-1);
}
.btn.get {
border-color: color-mix(in srgb, #60a5fa 45%, var(--vp-c-divider));
}
.btn.post {
border-color: color-mix(in srgb, #22c55e 45%, var(--vp-c-divider));
}
.btn.del {
border-color: color-mix(in srgb, #ef4444 45%, var(--vp-c-divider));
}
.btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.inputs {
margin-top: 12px;
}
.field {
display: grid;
grid-template-columns: 72px 1fr;
gap: 10px;
align-items: center;
}
.k {
font-size: 12px;
color: var(--vp-c-text-2);
font-weight: 900;
}
.input {
width: 100%;
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg-soft);
border-radius: 10px;
padding: 8px 10px;
color: var(--vp-c-text-1);
font-size: 13px;
}
.result {
margin-top: 12px;
border: 1px solid var(--vp-c-divider);
border-radius: 12px;
background: var(--vp-c-bg-soft);
padding: 10px 12px;
}
.resultTitle {
font-weight: 900;
font-size: 13px;
color: var(--vp-c-text-1);
}
.muted {
margin-top: 8px;
font-size: 12px;
color: var(--vp-c-text-2);
}
.resBox {
margin-top: 8px;
border: 1px solid var(--vp-c-divider);
border-radius: 12px;
background: var(--vp-c-bg);
padding: 10px 12px;
}
.resBox.ok {
border-color: color-mix(in srgb, #22c55e 45%, var(--vp-c-divider));
}
.resBox.bad {
border-color: color-mix(in srgb, #ef4444 45%, var(--vp-c-divider));
}
.badge {
display: inline-block;
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg-soft);
border-radius: 999px;
padding: 2px 10px;
font-size: 12px;
font-weight: 900;
color: var(--vp-c-text-1);
}
.text {
margin-top: 8px;
font-size: 13px;
font-weight: 900;
color: var(--vp-c-text-1);
}
.foot {
margin-top: 12px;
display: flex;
justify-content: space-between;
gap: 10px;
align-items: center;
}
.stat {
font-size: 12px;
color: var(--vp-c-text-2);
}
.statV {
font-weight: 900;
color: var(--vp-c-text-1);
margin: 0 6px;
}
.ghost {
border: 1px solid var(--vp-c-divider);
background: var(--vp-c-bg);
color: var(--vp-c-text-1);
border-radius: 10px;
padding: 8px 12px;
font-weight: 900;
cursor: pointer;
}
.ghost:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.one {
margin-top: 12px;
border: 1px dashed var(--vp-c-divider);
border-radius: 12px;
background: var(--vp-c-bg);
padding: 10px 12px;
}
.oneTitle {
font-weight: 900;
font-size: 13px;
color: var(--vp-c-text-1);
}
.oneText {
margin-top: 8px;
font-size: 12px;
color: var(--vp-c-text-2);
line-height: 1.6;
}
@media (max-width: 720px) {
.board {
grid-template-columns: 1fr;
}
.btnRow {
grid-template-columns: 1fr;
}
}
</style>