You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

352 lines
8.2 KiB
Vue

<template>
<view class="page-container">
<NavBar :title="t('moldOperate.selectMountMold')" />
<!-- 搜索区 -->
<view class="search-bar">
<view class="search-input-wrap">
<text class="search-icon iconfont icon-search"></text>
<input class="search-input" v-model="searchText" :placeholder="t('moldOperate.searchName')" @input="onSearch" placeholder-class="search-placeholder" />
<text v-if="searchText" class="search-clear" @click="clearSearch">✕</text>
</view>
</view>
<!-- 模具列表 -->
<scroll-view scroll-y class="mold-list" v-if="filteredList.length > 0">
<view
v-for="mold in filteredList"
:key="mold.id"
class="mold-card"
:class="{ active: selectedIds.has(String(mold.id)) }"
@click="toggleMold(mold)"
>
<view class="mold-card-header">
<text class="mold-name">{{ textValue(mold.name) }}</text>
<text class="mold-check-icon" v-if="selectedIds.has(String(mold.id))">&#10003;</text>
</view>
<view class="mold-card-body">
<view class="info-row">
<text class="info-label">{{ t('moldOperate.moldCode') }}</text>
<text class="info-value">{{ textValue(mold.code) }}</text>
</view>
<view class="info-row">
<text class="info-label">产品型号</text>
<text class="info-value">{{ textValue(mold.productName) }}</text>
</view>
<view class="info-row">
<text class="info-label">{{ t('moldOperate.status') }}</text>
<view :class="['status-tag', getStatusClass(mold.status)]">{{ getStatusText(mold.status) }}</view>
</view>
</view>
</view>
</scroll-view>
<!-- 空状态 -->
<view v-else class="empty-wrap">
<text v-if="loading" class="empty-text">{{ t('functionCommon.loading') }}</text>
<text v-else class="empty-text">{{ t('moldOperate.noMoldData') }}</text>
</view>
<!-- 底部确认按钮 -->
<view class="bottom-actions">
<view class="bottom-btn confirm-btn" @click="handleConfirm">
{{ t('functionCommon.confirm') }}{{ selectedIds.size > 0 ? `(${selectedIds.size})` : '' }}
</view>
</view>
</view>
</template>
<script setup>
import { ref, computed } from 'vue'
import { onShow } from '@dcloudio/uni-app'
import { useI18n } from 'vue-i18n'
import NavBar from '@/components/common/NavBar.vue'
import { getMoldBrandPage } from '@/api/mes/mold'
const { t } = useI18n()
const moldList = ref([])
const selectedIds = ref(new Set())
const searchText = ref('')
const loading = ref(false)
function textValue(v) {
if (v === 0) return '0'
if (v == null) return '-'
const s = String(v).trim()
return s || '-'
}
const STATUS_MAP = { 0: '在机', 1: '待用', 2: '维修', 3: '报废' }
function getStatusText(s) { return STATUS_MAP[s] || textValue(s) }
function getStatusClass(s) {
if (s === 0) return 'in-use-tag'
if (s === 2) return 'repairing-tag'
if (s === 3) return 'scrapped-tag'
return 'standby-tag'
}
const filteredList = computed(() => {
const keyword = searchText.value.trim().toLowerCase()
if (!keyword) return moldList.value
return moldList.value.filter((m) => {
return (m.name || '').toLowerCase().includes(keyword) ||
(m.code || '').toLowerCase().includes(keyword) ||
(m.productName || '').toLowerCase().includes(keyword)
})
})
function onSearch() {}
function clearSearch() {
searchText.value = ''
}
async function loadMolds() {
loading.value = true
try {
const res = await getMoldBrandPage({ pageNo: 1, pageSize: 100 })
const root = (res && res.data !== undefined) ? res.data : res
// 递归查找任意嵌套的 list/rows/records
function findList(obj, depth = 0) {
if (!obj || typeof obj !== 'object' || depth > 3) return []
if (Array.isArray(obj)) return obj
for (const key of ['list', 'rows', 'records', 'items', 'data']) {
if (Array.isArray(obj[key])) return obj[key]
}
for (const val of Object.values(obj)) {
const found = findList(val, depth + 1)
if (found.length) return found
}
return []
}
moldList.value = findList(root)
// 保留一行关键日志用于排查
if (!moldList.value.length) console.error('moldSelect: 未找到模具数据', root)
} catch (e) {
console.error('loadMolds error', e)
} finally {
loading.value = false
}
}
function toggleMold(mold) {
const id = String(mold.id)
const set = new Set(selectedIds.value)
if (set.has(id)) {
set.delete(id)
} else {
// 上模每次只上一个模具,限制单选
set.clear()
set.add(id)
}
selectedIds.value = set
}
function handleConfirm() {
if (selectedIds.value.size === 0) {
uni.showToast({ title: t('moldOperate.validatorMoldRequired'), icon: 'none' })
return
}
const selected = moldList.value.filter((m) => selectedIds.value.has(String(m.id)))
.map((m) => ({
id: m.id,
name: m.name,
code: m.code,
productName: m.productName || '-',
status: m.status
}))
getApp().globalData._moldSelectResult = selected
uni.navigateBack()
}
onShow(async () => {
// 从 index.vue 传来的已选模具(编辑模式回显)
const preSelected = getApp().globalData._moldSelectPreSelected
if (preSelected && Array.isArray(preSelected)) {
selectedIds.value = new Set(preSelected.map((id) => String(id)))
getApp().globalData._moldSelectPreSelected = null
} else {
selectedIds.value = new Set()
}
await loadMolds()
})
</script>
<style lang="scss" scoped>
.page-container {
min-height: 100vh;
background: #f5f6f8;
padding-bottom: 140rpx;
}
/* 搜索栏 */
.search-bar {
padding: 16rpx 24rpx;
background: #fff;
}
.search-input-wrap {
display: flex;
align-items: center;
height: 72rpx;
padding: 0 16rpx;
background: #f5f6f8;
border-radius: 36rpx;
}
.search-icon {
font-size: 32rpx;
color: #999;
margin-right: 12rpx;
}
.search-input {
flex: 1;
font-size: 28rpx;
color: #333;
}
.search-placeholder {
color: #bbb;
}
.search-clear {
font-size: 28rpx;
color: #999;
padding: 8rpx;
}
/* 模具列表 */
.mold-list {
padding: 16rpx 24rpx;
}
.mold-card {
background: #fff;
border-radius: 14rpx;
padding: 24rpx;
margin-bottom: 16rpx;
border: 2rpx solid transparent;
&.active {
border-color: #2563eb;
background: #f0f5ff;
}
}
.mold-card-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 16rpx;
}
.mold-name {
font-size: 30rpx;
font-weight: 700;
color: #1a1a1a;
}
.mold-check-icon {
font-size: 32rpx;
color: #2563eb;
font-weight: 700;
flex-shrink: 0;
width: 40rpx;
height: 40rpx;
line-height: 36rpx;
text-align: center;
border: 2rpx solid #2563eb;
border-radius: 50%;
}
.mold-card-body {
border-top: 1rpx solid #f0f0f0;
padding-top: 16rpx;
}
.info-row {
display: flex;
align-items: center;
margin-bottom: 10rpx;
&:last-child {
margin-bottom: 0;
}
}
.info-label {
width: 140rpx;
font-size: 24rpx;
color: #999;
flex-shrink: 0;
}
.info-value {
font-size: 26rpx;
color: #333;
}
/* 状态标签 */
.status-tag {
display: inline-flex;
align-items: center;
padding: 4rpx 16rpx;
border-radius: 6rpx;
font-size: 22rpx;
font-weight: 500;
&.standby-tag {
color: #2563eb; background: #eff6ff; border: 1rpx solid #bfdbfe;
}
&.in-use-tag {
color: #059669; background: #ecfdf5; border: 1rpx solid #a7f3d0;
}
&.repairing-tag {
color: #d97706; background: #fffbeb; border: 1rpx solid #fde68a;
}
&.scrapped-tag {
color: #dc2626; background: #fef2f2; border: 1rpx solid #fecaca;
}
}
/* 空状态 */
.empty-wrap {
padding: 120rpx 0;
text-align: center;
}
.empty-text {
font-size: 28rpx;
color: #999;
}
/* 底部确认按钮 */
.bottom-actions {
position: fixed;
left: 0;
right: 0;
bottom: 0;
padding: 18rpx 24rpx calc(18rpx + env(safe-area-inset-bottom));
background: #fff;
box-shadow: 0 -4rpx 20rpx rgba(0, 0, 0, 0.06);
z-index: 99;
}
.bottom-btn {
width: 100%;
height: 84rpx;
line-height: 84rpx;
text-align: center;
border-radius: 16rpx;
font-size: 30rpx;
font-weight: 600;
}
.confirm-btn {
background: #1f4b79;
color: #fff;
}
</style>