Files
EmailBill/Web/src/components/Budget/BudgetEditPopup.vue

222 lines
5.6 KiB
Vue
Raw Normal View History

<template>
2026-01-16 11:15:44 +08:00
<PopupContainer
v-model="visible"
:title="
isEdit
? `编辑${getCategoryName(form.category)}预算`
: `新增${getCategoryName(form.category)}预算`
"
height="75%"
>
<div class="add-budget-form">
<van-form>
<van-cell-group inset>
<van-field
v-model="form.name"
name="name"
label="预算名称"
placeholder="例如:每月餐饮、年度奖金"
:rules="[{ required: true, message: '请填写预算名称' }]"
/>
2026-01-15 10:53:05 +08:00
<!-- 新增不记额预算复选框 -->
<van-field label="不记额预算">
<template #input>
2026-01-16 11:15:44 +08:00
<van-checkbox v-model="form.noLimit" @update:model-value="onNoLimitChange">
不记额预算仅限年度
</van-checkbox>
2026-01-15 10:53:05 +08:00
</template>
</van-field>
<van-field name="type" label="统计周期">
<template #input>
2026-01-16 11:15:44 +08:00
<van-radio-group
v-model="form.type"
direction="horizontal"
2026-01-15 10:53:05 +08:00
:disabled="isEdit || form.noLimit"
>
2026-01-16 11:15:44 +08:00
<van-radio :name="BudgetPeriodType.Month"> </van-radio>
<van-radio :name="BudgetPeriodType.Year"> </van-radio>
</van-radio-group>
</template>
</van-field>
2026-01-15 10:53:05 +08:00
<!-- 仅当未选中"不记额预算"时显示预算金额 -->
<van-field
2026-01-15 10:53:05 +08:00
v-if="!form.noLimit"
v-model="form.limit"
type="number"
name="limit"
label="预算金额"
placeholder="0.00"
:rules="[{ required: true, message: '请填写预算金额' }]"
>
<template #extra>
<span></span>
</template>
</van-field>
<van-field label="相关分类">
<template #input>
2026-01-16 11:15:44 +08:00
<div
v-if="form.selectedCategories.length === 0"
style="color: var(--van-text-color-3)"
>
可多选分类
</div>
<div v-else class="selected-categories">
<span class="ellipsis-text">
{{ form.selectedCategories.join('、') }}
</span>
</div>
</template>
</van-field>
2026-01-08 14:41:50 +08:00
<ClassifySelector
v-model="form.selectedCategories"
:type="budgetType"
multiple
:show-add="false"
:show-clear="false"
/>
</van-cell-group>
</van-form>
</div>
<template #footer>
2026-01-16 11:15:44 +08:00
<van-button block round type="primary" @click="onSubmit"> 保存预算 </van-button>
</template>
</PopupContainer>
</template>
<script setup>
2026-01-08 14:41:50 +08:00
import { ref, reactive, computed } from 'vue'
import { showToast } from 'vant'
import { createBudget, updateBudget } from '@/api/budget'
import { BudgetPeriodType, BudgetCategory } from '@/constants/enums'
import PopupContainer from '@/components/PopupContainer.vue'
2026-01-08 14:41:50 +08:00
import ClassifySelector from '@/components/ClassifySelector.vue'
const emit = defineEmits(['success'])
const visible = ref(false)
const isEdit = ref(false)
const form = reactive({
id: undefined,
name: '',
type: BudgetPeriodType.Month,
category: BudgetCategory.Expense,
limit: '',
2026-01-15 10:53:05 +08:00
selectedCategories: [],
2026-01-16 11:15:44 +08:00
noLimit: false // 新增字段
})
2026-01-16 11:15:44 +08:00
const open = ({ data, isEditFlag, category }) => {
if (category === undefined) {
showToast('缺少必要参数category')
return
}
isEdit.value = isEditFlag
if (data) {
Object.assign(form, {
id: data.id,
name: data.name,
type: data.type,
category: category,
limit: data.limit,
2026-01-15 10:53:05 +08:00
selectedCategories: data.selectedCategories ? [...data.selectedCategories] : [],
2026-01-16 11:15:44 +08:00
noLimit: data.noLimit || false // 新增
})
} else {
Object.assign(form, {
id: undefined,
name: '',
type: BudgetPeriodType.Month,
category: category,
limit: '',
2026-01-15 10:53:05 +08:00
selectedCategories: [],
2026-01-16 11:15:44 +08:00
noLimit: false // 新增
})
}
visible.value = true
}
defineExpose({
open
})
2026-01-08 14:41:50 +08:00
const budgetType = computed(() => {
2026-01-16 11:15:44 +08:00
return form.category === BudgetCategory.Expense
? 0
: form.category === BudgetCategory.Income
? 1
: 2
})
const onSubmit = async () => {
try {
const data = {
...form,
2026-01-16 11:15:44 +08:00
limit: form.noLimit ? 0 : parseFloat(form.limit), // 不记额时金额为0
2026-01-15 10:53:05 +08:00
selectedCategories: form.selectedCategories,
2026-01-16 11:15:44 +08:00
noLimit: form.noLimit // 新增
}
2026-01-16 11:15:44 +08:00
const res = form.id ? await updateBudget(data) : await createBudget(data)
if (res.success) {
showToast('保存成功')
visible.value = false
emit('success')
}
} catch (err) {
showToast(err.message || '保存失败')
console.error('保存预算失败', err)
}
}
const getCategoryName = (category) => {
2026-01-16 11:15:44 +08:00
switch (category) {
case BudgetCategory.Expense:
return '支出'
case BudgetCategory.Income:
return '收入'
case BudgetCategory.Savings:
return '存款'
default:
return ''
}
}
2026-01-15 10:53:05 +08:00
const onNoLimitChange = (value) => {
if (value) {
// 选中不记额时,自动设为年度预算
form.type = BudgetPeriodType.Year
}
}
</script>
<style scoped>
.add-budget-form {
padding: 20px 0;
}
.selected-categories {
display: flex;
align-items: center;
padding: 4px 0;
width: 100%;
overflow: hidden;
}
.ellipsis-text {
font-size: 14px;
2026-01-13 17:00:44 +08:00
color: var(--van-text-color);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
width: 100%;
}
.no-data {
font-size: 13px;
2026-01-13 17:00:44 +08:00
color: var(--van-text-color-2);
2026-01-08 14:41:50 +08:00
padding: 8px 16px;
}
</style>