2026-02-11 09:45:45 +08:00
|
|
|
<template>
|
|
|
|
|
<div class="pricing-calculator">
|
|
|
|
|
<div class="config-section">
|
|
|
|
|
<div class="config-row">
|
|
|
|
|
<span class="label">实例规格</span>
|
|
|
|
|
<select v-model="config.spec">
|
2026-02-18 17:38:10 +08:00
|
|
|
<option value="small">
|
|
|
|
|
1核2G (入门)
|
|
|
|
|
</option>
|
|
|
|
|
<option value="medium">
|
|
|
|
|
2核4G (标准)
|
|
|
|
|
</option>
|
|
|
|
|
<option value="large">
|
|
|
|
|
4核8G (高性能)
|
|
|
|
|
</option>
|
2026-02-11 09:45:45 +08:00
|
|
|
</select>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="config-row">
|
|
|
|
|
<span class="label">运行时长</span>
|
2026-02-18 17:38:10 +08:00
|
|
|
<input
|
|
|
|
|
v-model.number="config.hours"
|
|
|
|
|
type="range"
|
|
|
|
|
min="1"
|
|
|
|
|
max="24"
|
|
|
|
|
>
|
2026-02-11 09:45:45 +08:00
|
|
|
<span class="value">{{ config.hours }} 小时/天</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="config-row">
|
|
|
|
|
<span class="label">运行天数</span>
|
2026-02-18 17:38:10 +08:00
|
|
|
<input
|
|
|
|
|
v-model.number="config.days"
|
|
|
|
|
type="range"
|
|
|
|
|
min="1"
|
|
|
|
|
max="31"
|
|
|
|
|
>
|
2026-02-11 09:45:45 +08:00
|
|
|
<span class="value">{{ config.days }} 天/月</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="result-section">
|
2026-02-18 17:38:10 +08:00
|
|
|
<div class="result-header">
|
|
|
|
|
月度成本对比
|
|
|
|
|
</div>
|
2026-02-11 09:45:45 +08:00
|
|
|
<div class="result-cards">
|
|
|
|
|
<div class="result-card">
|
2026-02-18 17:38:10 +08:00
|
|
|
<div class="model">
|
|
|
|
|
按需付费
|
|
|
|
|
</div>
|
|
|
|
|
<div class="price">
|
|
|
|
|
${{ costs.ondemand }}/月
|
|
|
|
|
</div>
|
2026-02-11 09:45:45 +08:00
|
|
|
</div>
|
|
|
|
|
<div class="result-card recommended">
|
2026-02-18 17:38:10 +08:00
|
|
|
<div class="model">
|
|
|
|
|
预留实例
|
|
|
|
|
</div>
|
|
|
|
|
<div class="price">
|
|
|
|
|
${{ costs.reserved }}/月
|
|
|
|
|
</div>
|
|
|
|
|
<div class="saving">
|
|
|
|
|
省 {{ savings }}%
|
|
|
|
|
</div>
|
2026-02-11 09:45:45 +08:00
|
|
|
</div>
|
|
|
|
|
<div class="result-card">
|
2026-02-18 17:38:10 +08:00
|
|
|
<div class="model">
|
|
|
|
|
抢占式
|
|
|
|
|
</div>
|
|
|
|
|
<div class="price">
|
|
|
|
|
${{ costs.spot }}/月
|
|
|
|
|
</div>
|
2026-02-11 09:45:45 +08:00
|
|
|
</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);
|
2026-02-14 20:23:34 +08:00
|
|
|
border-radius: 6px;
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
margin: 0.5rem 0;
|
2026-02-11 09:45:45 +08:00
|
|
|
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>
|