Files
test-repo/docs/.vitepress/theme/components/appendix/cloud-services/PricingCalculator.vue
T
sanbuphy 99d608d2a0 feat(docs): add interactive cloud services demo components
Add 10 new Vue components for cloud services documentation with interactive demos including:
- Cloud services overview and provider comparison
- Pricing calculator and region latency visualization
- Compute instance configurator and storage type explorer
- API call workflow and deployment process steps
- IAM structure and policy editor demos
2026-02-11 09:45:45 +08:00

224 lines
4.8 KiB
Vue

<template>
<div class="pricing-calculator">
<div class="config-section">
<div class="config-row">
<span class="label">实例规格</span>
<select v-model="config.spec">
<option value="small">1核2G (入门)</option>
<option value="medium">2核4G (标准)</option>
<option value="large">4核8G (高性能)</option>
</select>
</div>
<div class="config-row">
<span class="label">运行时长</span>
<input type="range" v-model.number="config.hours" min="1" max="24" />
<span class="value">{{ config.hours }} 小时/</span>
</div>
<div class="config-row">
<span class="label">运行天数</span>
<input type="range" v-model.number="config.days" min="1" max="31" />
<span class="value">{{ config.days }} /</span>
</div>
</div>
<div class="result-section">
<div class="result-header">月度成本对比</div>
<div class="result-cards">
<div class="result-card">
<div class="model">按需付费</div>
<div class="price">${{ costs.ondemand }}/</div>
</div>
<div class="result-card recommended">
<div class="model">预留实例</div>
<div class="price">${{ costs.reserved }}/</div>
<div class="saving"> {{ savings }}%</div>
</div>
<div class="result-card">
<div class="model">抢占式</div>
<div class="price">${{ costs.spot }}/</div>
</div>
</div>
</div>
<div class="tip-box">
<span class="tip-icon">💡</span>
<span class="tip-text">{{ recommendation }}</span>
</div>
</div>
</template>
<script setup>
import { ref, computed } from 'vue'
const config = ref({
spec: 'medium',
hours: 12,
days: 22
})
const specPrices = {
small: { ondemand: 0.08, reserved: 45, spot: 0.024 },
medium: { ondemand: 0.16, reserved: 89, spot: 0.048 },
large: { ondemand: 0.32, reserved: 179, spot: 0.096 }
}
const costs = computed(() => {
const price = specPrices[config.value.spec]
const monthlyHours = config.value.hours * config.value.days
return {
ondemand: Math.round(price.ondemand * monthlyHours),
reserved: price.reserved,
spot: Math.round(price.spot * monthlyHours)
}
})
const savings = computed(() => {
const save = costs.value.ondemand - costs.value.reserved
return Math.round((save / costs.value.ondemand) * 100)
})
const recommendation = computed(() => {
if (config.value.days < 15) {
return '当前使用频率较低,建议选择按需付费'
} else if (savings.value > 30) {
return `当前使用负载稳定,切换预留实例可省 ${savings.value}%`
} else {
return '根据当前配置,预留实例更具成本优势'
}
})
</script>
<style scoped>
.pricing-calculator {
border: 1px solid var(--vp-c-divider);
border-radius: 8px;
padding: 1rem;
margin: 1rem 0;
background: var(--vp-c-bg-soft);
}
.config-section {
margin-bottom: 1rem;
padding-bottom: 1rem;
border-bottom: 1px solid var(--vp-c-divider);
}
.config-row {
display: flex;
align-items: center;
gap: 0.75rem;
margin-bottom: 0.5rem;
font-size: 0.85rem;
}
.config-row:last-child {
margin-bottom: 0;
}
.config-row .label {
width: 70px;
color: var(--vp-c-text-2);
flex-shrink: 0;
}
.config-row select {
padding: 0.25rem 0.5rem;
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
background: var(--vp-c-bg);
font-size: 0.85rem;
}
.config-row input[type="range"] {
flex: 1;
min-width: 80px;
}
.config-row .value {
width: 85px;
text-align: right;
font-size: 0.8rem;
color: var(--vp-c-text-2);
}
.result-header {
font-weight: 600;
margin-bottom: 0.75rem;
font-size: 0.9rem;
}
.result-cards {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.75rem;
margin-bottom: 1rem;
}
.result-card {
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 6px;
padding: 0.75rem;
text-align: center;
}
.result-card.recommended {
border-color: var(--vp-c-brand);
background: var(--vp-c-brand-soft);
}
.result-card .model {
font-size: 0.8rem;
color: var(--vp-c-text-2);
margin-bottom: 0.25rem;
}
.result-card .price {
font-size: 1.1rem;
font-weight: 600;
}
.result-card .saving {
font-size: 0.75rem;
color: var(--vp-c-brand);
margin-top: 0.25rem;
}
.tip-box {
display: flex;
align-items: center;
gap: 0.5rem;
padding: 0.75rem;
background: var(--vp-c-bg);
border-radius: 6px;
font-size: 0.85rem;
}
.tip-icon {
font-size: 1rem;
}
.tip-text {
color: var(--vp-c-text-2);
}
@media (max-width: 640px) {
.result-cards {
grid-template-columns: 1fr;
}
.config-row {
flex-wrap: wrap;
}
.config-row .label {
width: 100%;
}
.config-row .value {
width: auto;
}
}
</style>