smart-delivery/src/pages/consumables/temp-catalog.vue

589 lines
12 KiB
Vue

<template>
<view class="page">
<!-- 顶部标题栏 -->
<view class="header">
<text class="title">耗材种类管理</text>
</view>
<!-- 搜索栏 -->
<view class="search-bar">
<view class="search-box">
<text class="search-icon">🔍</text>
<input class="search-input" v-model="searchKeyword" placeholder="搜索物料名称或规格" @confirm="loadList(true)" />
</view>
<button class="btn-add" @tap="showAddModal">+ 新增</button>
</view>
<!-- 列表 -->
<scroll-view scroll-y class="list-container" @scrolltolower="loadMore">
<view v-if="list.length === 0 && !loading" class="empty">
<text class="empty-text">暂无耗材种类</text>
</view>
<view v-for="item in list" :key="item.id" class="catalog-card">
<view class="card-main">
<view class="catalog-info">
<text class="catalog-name">{{ item.name }}</text>
<view class="catalog-meta">
<text v-if="item.spec" class="meta-item">规格: {{ item.spec }}</text>
<text v-if="item.unit" class="meta-item">单位: {{ item.unit }}</text>
</view>
</view>
<view class="catalog-status" :class="item.isActive ? 'active' : 'inactive'">
{{ item.isActive ? '启用' : '停用' }}
</view>
</view>
<view class="card-footer">
<text class="time">{{ formatTime(item.updatedAt) }}</text>
<view class="actions">
<button class="btn-edit" @tap="handleEdit(item)">编辑</button>
<button v-if="item.isActive" class="btn-disable" @tap="handleDisable(item.id)">停用</button>
<button v-else class="btn-enable" @tap="handleEnable(item.id)">启用</button>
</view>
</view>
</view>
<view v-if="loading" class="loading">
<text>加载中...</text>
</view>
<view v-if="!hasMore && list.length > 0" class="no-more">
<text>没有更多了</text>
</view>
</scroll-view>
<!-- 编辑/新增弹窗 -->
<view v-if="showModal" class="modal-mask" @tap="closeModal">
<view class="modal-content" @tap.stop>
<view class="modal-header">
<text class="modal-title">{{ isEdit ? '编辑种类' : '新增种类' }}</text>
<text class="modal-close" @tap="closeModal">✕</text>
</view>
<view class="modal-body">
<view class="form-item">
<text class="form-label required">物料名称</text>
<input class="form-input" v-model="formData.name" placeholder="请输入物料名称" />
</view>
<view class="form-item">
<text class="form-label">规格</text>
<input class="form-input" v-model="formData.spec" placeholder="请输入规格(可选)" />
</view>
<view class="form-item">
<text class="form-label">单位</text>
<input class="form-input" v-model="formData.unit" placeholder="请输入单位(可选)" />
</view>
<view class="form-item">
<text class="form-label">状态</text>
<view class="switch-box">
<switch :checked="formData.isActive" @change="onSwitchChange" color="#667eea" />
<text class="switch-label">{{ formData.isActive ? '启用' : '停用' }}</text>
</view>
</view>
</view>
<view class="modal-footer">
<button class="btn-cancel" @tap="closeModal">取消</button>
<button class="btn-confirm" @tap="handleSubmit" :disabled="submitting">
{{ submitting ? '提交中...' : '确认' }}
</button>
</view>
</view>
</view>
</view>
</template>
<script setup lang="ts">
import { ref, onMounted } from 'vue'
import consumableTempAPI, { type CatalogItem } from '@/pages/api/consumable-temp'
// 数据状态
const list = ref<CatalogItem[]>([])
const loading = ref(false)
const searchKeyword = ref('')
const page = ref(1)
const pageSize = ref(20)
const hasMore = ref(true)
// 弹窗相关
const showModal = ref(false)
const isEdit = ref(false)
const formData = ref<CatalogItem>({
name: '',
spec: '',
unit: '',
isActive: true
})
const submitting = ref(false)
// 加载列表
async function loadList(reset = false) {
if (loading.value) return
if (reset) {
page.value = 1
hasMore.value = true
list.value = []
}
if (!hasMore.value) return
loading.value = true
try {
const res = await consumableTempAPI.getCatalogList({
page: page.value,
pageSize: pageSize.value,
q: searchKeyword.value || undefined
})
if (res.statusCode === 200 && (res.data as any)?.data) {
const newData = (res.data as any).data
list.value = reset ? newData : [...list.value, ...newData]
hasMore.value = newData.length >= pageSize.value
} else {
uni.showToast({ title: '加载失败', icon: 'none' })
}
} catch (err) {
console.error('加载列表失败:', err)
uni.showToast({ title: '加载失败', icon: 'none' })
} finally {
loading.value = false
}
}
// 加载更多
function loadMore() {
if (!loading.value && hasMore.value) {
page.value++
loadList()
}
}
// 格式化时间
function formatTime(time?: string) {
if (!time) return '-'
const date = new Date(time)
return `${date.getFullYear()}-${String(date.getMonth() + 1).padStart(2, '0')}-${String(date.getDate()).padStart(2, '0')}`
}
// 显示新增弹窗
function showAddModal() {
isEdit.value = false
formData.value = {
name: '',
spec: '',
unit: '',
isActive: true
}
showModal.value = true
}
// 编辑
function handleEdit(item: CatalogItem) {
isEdit.value = true
formData.value = { ...item }
showModal.value = true
}
// 关闭弹窗
function closeModal() {
showModal.value = false
}
// 开关改变
function onSwitchChange(e: any) {
formData.value.isActive = e.detail.value
}
// 提交表单
async function handleSubmit() {
if (!formData.value.name) {
uni.showToast({ title: '请输入物料名称', icon: 'none' })
return
}
if (submitting.value) return
submitting.value = true
try {
let res
if (isEdit.value && formData.value.id) {
res = await consumableTempAPI.updateCatalog(formData.value.id, formData.value)
} else {
res = await consumableTempAPI.createCatalog(formData.value)
}
if (res.statusCode === 200 || res.statusCode === 201) {
uni.showToast({
title: isEdit.value ? '更新成功' : '创建成功',
icon: 'success'
})
closeModal()
loadList(true)
} else {
uni.showToast({ title: '操作失败', icon: 'none' })
}
} catch (err) {
console.error('提交失败:', err)
uni.showToast({ title: '操作失败', icon: 'none' })
} finally {
submitting.value = false
}
}
// 停用
function handleDisable(id?: string) {
if (!id) return
uni.showModal({
title: '确认停用',
content: '停用后该种类将不会出现在申请选择列表中',
success: async (res) => {
if (res.confirm) {
try {
const result = await consumableTempAPI.deleteCatalog(id)
if (result.statusCode === 204) {
uni.showToast({ title: '已停用', icon: 'success' })
loadList(true)
} else {
uni.showToast({ title: '操作失败', icon: 'none' })
}
} catch (err) {
console.error('停用失败:', err)
uni.showToast({ title: '操作失败', icon: 'none' })
}
}
}
})
}
// 启用
async function handleEnable(id?: string) {
if (!id) return
try {
const res = await consumableTempAPI.updateCatalog(id, { isActive: true })
if (res.statusCode === 200) {
uni.showToast({ title: '已启用', icon: 'success' })
loadList(true)
} else {
uni.showToast({ title: '操作失败', icon: 'none' })
}
} catch (err) {
console.error('启用失败:', err)
uni.showToast({ title: '操作失败', icon: 'none' })
}
}
onMounted(() => {
loadList(true)
})
</script>
<style scoped>
.page {
min-height: 100vh;
background: #f5f7fb;
display: flex;
flex-direction: column;
box-sizing: border-box;
}
.header {
background: #fff;
padding: 32rpx;
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.06);
}
.title {
font-size: 36rpx;
font-weight: 600;
color: #111;
}
.search-bar {
background: #fff;
padding: 24rpx 32rpx;
margin-bottom: 12rpx;
display: flex;
gap: 12rpx;
}
.search-box {
flex: 1;
display: flex;
align-items: center;
padding: 0 20rpx;
background: #f5f7fb;
border-radius: 12rpx;
height: 72rpx;
}
.search-icon {
font-size: 28rpx;
margin-right: 12rpx;
}
.search-input {
flex: 1;
height: 72rpx;
line-height: 72rpx;
font-size: 28rpx;
}
.btn-add {
padding: 0 24rpx;
height: 72rpx;
line-height: 72rpx;
background: #3a5ddd;
color: #fff;
border-radius: 12rpx;
font-size: 28rpx;
white-space: nowrap;
}
.list-container {
flex: 1;
padding: 0 32rpx 24rpx;
}
.empty {
padding: 120rpx 0;
text-align: center;
}
.empty-text {
font-size: 28rpx;
color: #999;
}
.catalog-card {
background: #fff;
border-radius: 20rpx;
padding: 24rpx;
margin-bottom: 16rpx;
box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.06);
}
.card-main {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 16rpx;
}
.catalog-info {
flex: 1;
display: flex;
flex-direction: column;
gap: 8rpx;
}
.catalog-name {
font-size: 30rpx;
font-weight: 600;
color: #111;
}
.catalog-meta {
display: flex;
gap: 16rpx;
flex-wrap: wrap;
}
.meta-item {
font-size: 24rpx;
color: #666;
padding: 4rpx 12rpx;
background: #f5f7fb;
border-radius: 6rpx;
}
.catalog-status {
padding: 8rpx 16rpx;
border-radius: 8rpx;
font-size: 24rpx;
font-weight: 500;
}
.catalog-status.active {
background: #e8f5e9;
color: #2e7d32;
}
.catalog-status.inactive {
background: #f5f5f5;
color: #757575;
}
.card-footer {
display: flex;
justify-content: space-between;
align-items: center;
padding-top: 16rpx;
border-top: 1rpx solid #f0f0f0;
}
.time {
font-size: 24rpx;
color: #999;
}
.actions {
display: flex;
gap: 8rpx;
}
.btn-edit,
.btn-disable,
.btn-enable {
padding: 8rpx 20rpx;
border-radius: 8rpx;
font-size: 24rpx;
}
.btn-edit {
background: #e3f2fd;
color: #1976d2;
}
.btn-disable {
background: #fff3e0;
color: #f57c00;
}
.btn-enable {
background: #e8f5e9;
color: #2e7d32;
}
.loading,
.no-more {
padding: 32rpx;
text-align: center;
font-size: 26rpx;
color: #999;
}
/* 弹窗样式 */
.modal-mask {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.5);
display: flex;
align-items: center;
justify-content: center;
z-index: 1000;
}
.modal-content {
width: 600rpx;
background: #fff;
border-radius: 24rpx;
}
.modal-header {
padding: 32rpx;
display: flex;
justify-content: space-between;
align-items: center;
border-bottom: 1rpx solid #f0f0f0;
}
.modal-title {
font-size: 32rpx;
font-weight: 600;
color: #111;
}
.modal-close {
font-size: 40rpx;
color: #999;
width: 60rpx;
height: 60rpx;
display: flex;
align-items: center;
justify-content: center;
}
.modal-body {
padding: 32rpx;
}
.form-item {
margin-bottom: 24rpx;
}
.form-item:last-child {
margin-bottom: 0;
}
.form-label {
display: block;
font-size: 28rpx;
color: #333;
margin-bottom: 12rpx;
}
.form-label.required::before {
content: '*';
color: #ff4d4f;
margin-right: 4rpx;
}
.form-input {
width: 100%;
height: 72rpx;
line-height: 72rpx;
padding: 0 20rpx;
background: #f5f7fb;
border-radius: 12rpx;
font-size: 28rpx;
}
.switch-box {
display: flex;
align-items: center;
gap: 16rpx;
}
.switch-label {
font-size: 28rpx;
color: #333;
}
.modal-footer {
padding: 24rpx 32rpx;
border-top: 1rpx solid #f0f0f0;
display: flex;
gap: 16rpx;
}
.btn-cancel,
.btn-confirm {
flex: 1;
height: 80rpx;
line-height: 80rpx;
border-radius: 12rpx;
font-size: 30rpx;
font-weight: 500;
}
.btn-cancel {
background: #f5f5f5;
color: #666;
}
.btn-confirm {
background: #3a5ddd;
color: #fff;
}
.btn-confirm[disabled] {
opacity: 0.6;
}
</style>