李红攀:V2.6.969,客户管理、销售订单
This commit is contained in:
13
src/api/erp/account/index.ts
Normal file
13
src/api/erp/account/index.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import type { PageParam, PageResult } from '@/http/types'
|
||||||
|
import { http } from '@/http/http'
|
||||||
|
|
||||||
|
/** 账户信息 */
|
||||||
|
export interface Account {
|
||||||
|
id?: number
|
||||||
|
name?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取账户精简列表 */
|
||||||
|
export function getAccountSimpleList() {
|
||||||
|
return http.get<Account[]>('/erp/account/simple-list')
|
||||||
|
}
|
||||||
96
src/api/erp/customer/index.ts
Normal file
96
src/api/erp/customer/index.ts
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import type { PageParam, PageResult } from '@/http/types'
|
||||||
|
import { http } from '@/http/http'
|
||||||
|
|
||||||
|
/** 客户信息 */
|
||||||
|
export interface Customer {
|
||||||
|
id?: number
|
||||||
|
name?: string
|
||||||
|
contact?: string
|
||||||
|
mobile?: string
|
||||||
|
telephone?: string
|
||||||
|
email?: string
|
||||||
|
fax?: string
|
||||||
|
remark?: string
|
||||||
|
status?: number
|
||||||
|
sort?: number
|
||||||
|
taxNo?: string
|
||||||
|
taxPercent?: number
|
||||||
|
bankName?: string
|
||||||
|
bankAccount?: string
|
||||||
|
bankAddress?: string
|
||||||
|
tags?: string
|
||||||
|
companyType?: number
|
||||||
|
businessScale?: string
|
||||||
|
annualPurchaseVolume?: number
|
||||||
|
annualPurchaseAmount?: number
|
||||||
|
annualOrderNums?: number
|
||||||
|
annualOrderAmounts?: number
|
||||||
|
annualReceivedAmounts?: number
|
||||||
|
totalUnreceivedAmounts?: number
|
||||||
|
preferredProducts?: string
|
||||||
|
preferredPackaging?: string
|
||||||
|
qualityCertRequirements?: string
|
||||||
|
deliveryCyclePrefer?: string
|
||||||
|
paymentTerms?: string
|
||||||
|
lastDeliveryTime?: number
|
||||||
|
deliveryAddress?: string
|
||||||
|
businessLicenseNo?: string
|
||||||
|
licenseValidity?: number
|
||||||
|
riskLevel?: number
|
||||||
|
logisticsPartner?: string
|
||||||
|
requiresColdChain?: boolean
|
||||||
|
serviceRegion?: string
|
||||||
|
relationshipLevel?: number
|
||||||
|
leadSourceType?: number
|
||||||
|
leadSourceDetail?: string
|
||||||
|
leadScore?: number
|
||||||
|
potentialLevel?: number
|
||||||
|
expectedAnnualVolume?: number
|
||||||
|
expectedAnnualAmount?: number
|
||||||
|
purchaseCycle?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 客户精简信息 */
|
||||||
|
export interface CustomerSimple {
|
||||||
|
id: number
|
||||||
|
name: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取客户分页列表 */
|
||||||
|
export function getCustomerPage(params: PageParam & {
|
||||||
|
name?: string
|
||||||
|
mobile?: string
|
||||||
|
telephone?: string
|
||||||
|
}) {
|
||||||
|
return http.get<PageResult<Customer>>('/erp/customer/page', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取客户详情 */
|
||||||
|
export function getCustomer(id: number) {
|
||||||
|
return http.get<Customer>(`/erp/customer/get?id=${id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 创建客户 */
|
||||||
|
export function createCustomer(data: Customer) {
|
||||||
|
return http.post<number>('/erp/customer/create', data)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 更新客户 */
|
||||||
|
export function updateCustomer(data: Customer) {
|
||||||
|
return http.put<boolean>('/erp/customer/update', data)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除客户 */
|
||||||
|
export function deleteCustomer(id: number) {
|
||||||
|
return http.delete<boolean>(`/erp/customer/delete?id=${id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取客户精简列表 */
|
||||||
|
export function getCustomerSimpleList() {
|
||||||
|
return http.get<CustomerSimple[]>('/erp/customer/simple-list')
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 导出客户 */
|
||||||
|
export function exportCustomer(params: PageParam) {
|
||||||
|
return http.get<string>('/erp/customer/export', params)
|
||||||
|
}
|
||||||
92
src/api/erp/sale-order/index.ts
Normal file
92
src/api/erp/sale-order/index.ts
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
import type { PageParam, PageResult } from '@/http/types'
|
||||||
|
import { http } from '@/http/http'
|
||||||
|
|
||||||
|
/** 销售订单项 */
|
||||||
|
export interface SaleOrderItem {
|
||||||
|
id?: number
|
||||||
|
productId?: number
|
||||||
|
productName?: string
|
||||||
|
productNo?: string
|
||||||
|
count?: number
|
||||||
|
totalCount?: number
|
||||||
|
unitPrice?: number
|
||||||
|
productPrice?: number
|
||||||
|
taxPrice?: number
|
||||||
|
totalPrice?: number
|
||||||
|
taxPercent?: number
|
||||||
|
taxAmount?: number
|
||||||
|
remark?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 销售订单信息 */
|
||||||
|
export interface SaleOrder {
|
||||||
|
id?: number
|
||||||
|
no?: string
|
||||||
|
customerId?: number
|
||||||
|
customerName?: string
|
||||||
|
orderTime?: number | string
|
||||||
|
totalCount?: number
|
||||||
|
totalProductPrice?: number
|
||||||
|
discountPercent?: number
|
||||||
|
discountPrice?: number
|
||||||
|
totalPrice?: number
|
||||||
|
depositPrice?: number
|
||||||
|
accountId?: number
|
||||||
|
status?: number
|
||||||
|
remark?: string
|
||||||
|
fileUrl?: string
|
||||||
|
items?: SaleOrderItem[]
|
||||||
|
saleUserId?: number
|
||||||
|
saleUserName?: string
|
||||||
|
paymentCondition?: number
|
||||||
|
paymentMethod?: number
|
||||||
|
outCount?: number
|
||||||
|
returnCount?: number
|
||||||
|
productNames?: string
|
||||||
|
creatorName?: string
|
||||||
|
hasApprovalRecords?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取销售订单分页列表 */
|
||||||
|
export function getSaleOrderPage(params: PageParam & {
|
||||||
|
no?: string
|
||||||
|
customerId?: number
|
||||||
|
productId?: number
|
||||||
|
orderTime?: string[]
|
||||||
|
status?: number
|
||||||
|
creator?: number
|
||||||
|
outStatus?: string
|
||||||
|
returnStatus?: string
|
||||||
|
}) {
|
||||||
|
return http.get<PageResult<SaleOrder>>('/erp/sale-order/page', params)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取销售订单详情 */
|
||||||
|
export function getSaleOrder(id: number) {
|
||||||
|
return http.get<SaleOrder>(`/erp/sale-order/get?id=${id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 创建销售订单 */
|
||||||
|
export function createSaleOrder(data: SaleOrder) {
|
||||||
|
return http.post<number>('/erp/sale-order/create', data)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 更新销售订单 */
|
||||||
|
export function updateSaleOrder(data: SaleOrder) {
|
||||||
|
return http.put<boolean>('/erp/sale-order/update', data)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 更新销售订单状态 */
|
||||||
|
export function updateSaleOrderStatus(id: number, status: number) {
|
||||||
|
return http.put<boolean>('/erp/sale-order/update-status', undefined, { id, status })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除销售订单 */
|
||||||
|
export function deleteSaleOrder(ids: number[]) {
|
||||||
|
return http.delete<boolean>(`/erp/sale-order/delete?ids=${ids.join(',')}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 导出销售订单 */
|
||||||
|
export function exportSaleOrder(params: PageParam) {
|
||||||
|
return http.get<string>('/erp/sale-order/export-excel', params)
|
||||||
|
}
|
||||||
372
src/pages-erp/customer/form/index.vue
Normal file
372
src/pages-erp/customer/form/index.vue
Normal file
@@ -0,0 +1,372 @@
|
|||||||
|
<template>
|
||||||
|
<view class="yd-page-container">
|
||||||
|
<!-- 顶部导航栏 -->
|
||||||
|
<wd-navbar
|
||||||
|
:title="getTitle"
|
||||||
|
left-arrow placeholder safe-area-inset-top fixed
|
||||||
|
@click-left="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 表单区域 -->
|
||||||
|
<view class="pb-180rpx">
|
||||||
|
<wd-form ref="formRef" :model="formData" :rules="formRules">
|
||||||
|
<!-- 基础信息 -->
|
||||||
|
<wd-cell-group title="基础信息" border>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.name"
|
||||||
|
label="名称"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入客户名称"
|
||||||
|
prop="name"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.contact"
|
||||||
|
label="联系人"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入联系人"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.mobile"
|
||||||
|
label="手机号码"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入手机号码"
|
||||||
|
type="number"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.telephone"
|
||||||
|
label="联系电话"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入联系电话"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.email"
|
||||||
|
label="电子邮箱"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入电子邮箱"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-cell title="企业类型" title-width="180rpx" center>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.companyType"
|
||||||
|
:columns="companyTypeColumns"
|
||||||
|
label=""
|
||||||
|
placeholder="请选择企业类型"
|
||||||
|
@confirm="onCompanyTypeConfirm"
|
||||||
|
/>
|
||||||
|
</wd-cell>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.businessScale"
|
||||||
|
label="企业规模"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入企业规模"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-cell title="客户来源" title-width="180rpx" center>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.leadSourceType"
|
||||||
|
:columns="leadSourceTypeColumns"
|
||||||
|
label=""
|
||||||
|
placeholder="请选择来源类型"
|
||||||
|
@confirm="onLeadSourceTypeConfirm"
|
||||||
|
/>
|
||||||
|
</wd-cell>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<!-- 财务信息 -->
|
||||||
|
<wd-cell-group title="财务信息" border>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.taxNo"
|
||||||
|
label="纳税人识别号"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入纳税人识别号"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.bankName"
|
||||||
|
label="开户行"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入开户行"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.bankAccount"
|
||||||
|
label="开户账号"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入开户账号"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.bankAddress"
|
||||||
|
label="开户地址"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入开户地址"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<!-- 偏好信息 -->
|
||||||
|
<wd-cell-group title="偏好信息" border>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.preferredProducts"
|
||||||
|
label="偏好产品"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入偏好产品或品类"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.preferredPackaging"
|
||||||
|
label="偏好包装"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入偏好包装形式"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.deliveryCyclePrefer"
|
||||||
|
label="偏好交期"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入偏好交期/周期"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<!-- 统计信息 -->
|
||||||
|
<wd-cell-group title="统计信息" border>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatInteger(formData.annualOrderNums)"
|
||||||
|
label="年度订单数"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="系统自动生成"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatAmount(formData.annualOrderAmounts)"
|
||||||
|
label="年度订单额"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="系统自动生成"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatAmount(formData.annualReceivedAmounts)"
|
||||||
|
label="年度已收款"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="系统自动生成"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatAmount(formData.totalUnreceivedAmounts)"
|
||||||
|
label="累计未收款"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="系统自动生成"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<!-- 其他信息 -->
|
||||||
|
<wd-cell-group title="其他信息" border>
|
||||||
|
<wd-cell title="开启状态" title-width="180rpx" center>
|
||||||
|
<wd-switch v-model="statusEnabled" />
|
||||||
|
</wd-cell>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.sort"
|
||||||
|
label="排序"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入排序"
|
||||||
|
type="number"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
<wd-textarea
|
||||||
|
v-model="formData.remark"
|
||||||
|
label="备注"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入备注"
|
||||||
|
:maxlength="200"
|
||||||
|
show-word-limit
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
</wd-form>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 底部保存按钮 -->
|
||||||
|
<view class="yd-detail-footer">
|
||||||
|
<wd-button
|
||||||
|
type="primary"
|
||||||
|
block
|
||||||
|
:loading="formLoading"
|
||||||
|
@click="handleSubmit"
|
||||||
|
>
|
||||||
|
保存
|
||||||
|
</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import type { FormInstance } from 'wot-design-uni/components/wd-form/types'
|
||||||
|
import type { Customer } from '@/api/erp/customer'
|
||||||
|
import { computed, onMounted, ref, watch } from 'vue'
|
||||||
|
import { useToast } from 'wot-design-uni'
|
||||||
|
import { createCustomer, getCustomer, updateCustomer } from '@/api/erp/customer'
|
||||||
|
import { navigateBackPlus } from '@/utils'
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
id?: number | any
|
||||||
|
}>()
|
||||||
|
|
||||||
|
definePage({
|
||||||
|
style: {
|
||||||
|
navigationBarTitleText: '',
|
||||||
|
navigationStyle: 'custom',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const toast = useToast()
|
||||||
|
const getTitle = computed(() => props.id ? '编辑客户' : '新增客户')
|
||||||
|
const formLoading = ref(false)
|
||||||
|
const formData = ref<Customer>({
|
||||||
|
id: undefined,
|
||||||
|
name: undefined,
|
||||||
|
contact: undefined,
|
||||||
|
mobile: undefined,
|
||||||
|
telephone: undefined,
|
||||||
|
email: undefined,
|
||||||
|
remark: undefined,
|
||||||
|
status: 0,
|
||||||
|
sort: 0,
|
||||||
|
taxNo: undefined,
|
||||||
|
taxPercent: undefined,
|
||||||
|
bankName: undefined,
|
||||||
|
bankAccount: undefined,
|
||||||
|
bankAddress: undefined,
|
||||||
|
companyType: undefined,
|
||||||
|
businessScale: undefined,
|
||||||
|
leadSourceType: undefined,
|
||||||
|
preferredProducts: undefined,
|
||||||
|
preferredPackaging: undefined,
|
||||||
|
deliveryCyclePrefer: undefined,
|
||||||
|
annualOrderNums: undefined,
|
||||||
|
annualOrderAmounts: undefined,
|
||||||
|
annualReceivedAmounts: undefined,
|
||||||
|
totalUnreceivedAmounts: undefined,
|
||||||
|
})
|
||||||
|
const formRules = {
|
||||||
|
name: [{ required: true, message: '客户名称不能为空' }],
|
||||||
|
}
|
||||||
|
const formRef = ref<FormInstance>()
|
||||||
|
|
||||||
|
// 状态开关
|
||||||
|
const statusEnabled = ref(true)
|
||||||
|
watch(statusEnabled, (val) => {
|
||||||
|
formData.value.status = val ? 0 : 1
|
||||||
|
})
|
||||||
|
watch(() => formData.value.status, (val) => {
|
||||||
|
statusEnabled.value = val === 0
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 企业类型选项 */
|
||||||
|
const companyTypeOptions = [
|
||||||
|
{ value: 1, label: '餐饮' },
|
||||||
|
{ value: 2, label: '食品批发' },
|
||||||
|
{ value: 3, label: '制造' },
|
||||||
|
{ value: 4, label: '超市' },
|
||||||
|
{ value: 5, label: '其他' },
|
||||||
|
]
|
||||||
|
const companyTypeColumns = computed(() => [
|
||||||
|
companyTypeOptions.map(v => ({ value: v.value, label: v.label })),
|
||||||
|
])
|
||||||
|
function onCompanyTypeConfirm({ value }: any) {
|
||||||
|
formData.value.companyType = value?.[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 客户来源选项 */
|
||||||
|
const leadSourceTypeOptions = [
|
||||||
|
{ value: 1, label: '展会' },
|
||||||
|
{ value: 2, label: '地推' },
|
||||||
|
{ value: 3, label: '转介绍' },
|
||||||
|
{ value: 4, label: '官网' },
|
||||||
|
{ value: 5, label: '平台' },
|
||||||
|
{ value: 6, label: '电销' },
|
||||||
|
{ value: 7, label: '老客户' },
|
||||||
|
{ value: 99, label: '其他' },
|
||||||
|
]
|
||||||
|
const leadSourceTypeColumns = computed(() => [
|
||||||
|
leadSourceTypeOptions.map(v => ({ value: v.value, label: v.label })),
|
||||||
|
])
|
||||||
|
function onLeadSourceTypeConfirm({ value }: any) {
|
||||||
|
formData.value.leadSourceType = value?.[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化整数 */
|
||||||
|
const formatInteger = (value?: string | number) => {
|
||||||
|
if (value === undefined || value === null || value === '') {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
const num = Number(value)
|
||||||
|
if (Number.isNaN(num)) {
|
||||||
|
return String(value)
|
||||||
|
}
|
||||||
|
return String(Math.trunc(num))
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化金额 */
|
||||||
|
const formatAmount = (value?: string | number) => {
|
||||||
|
if (value === undefined || value === null || value === '') {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
const num = Number(value)
|
||||||
|
if (Number.isNaN(num)) {
|
||||||
|
return String(value)
|
||||||
|
}
|
||||||
|
return num.toFixed(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 返回上一页 */
|
||||||
|
function handleBack() {
|
||||||
|
navigateBackPlus('/pages-erp/customer/index')
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 加载详情 */
|
||||||
|
async function getDetail() {
|
||||||
|
if (!props.id) return
|
||||||
|
try {
|
||||||
|
toast.loading('加载中...')
|
||||||
|
formData.value = await getCustomer(props.id)
|
||||||
|
} finally {
|
||||||
|
toast.close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 提交表单 */
|
||||||
|
async function handleSubmit() {
|
||||||
|
const { valid } = await formRef.value!.validate()
|
||||||
|
if (!valid) return
|
||||||
|
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
if (props.id) {
|
||||||
|
await updateCustomer(formData.value)
|
||||||
|
toast.success('修改成功')
|
||||||
|
} else {
|
||||||
|
await createCustomer(formData.value)
|
||||||
|
toast.success('新增成功')
|
||||||
|
}
|
||||||
|
setTimeout(() => {
|
||||||
|
handleBack()
|
||||||
|
}, 500)
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 初始化 */
|
||||||
|
onMounted(() => {
|
||||||
|
getDetail()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
</style>
|
||||||
260
src/pages-erp/customer/index.vue
Normal file
260
src/pages-erp/customer/index.vue
Normal file
@@ -0,0 +1,260 @@
|
|||||||
|
<template>
|
||||||
|
<view class="yd-page-container">
|
||||||
|
<!-- 顶部导航栏 -->
|
||||||
|
<wd-navbar
|
||||||
|
title="客户管理"
|
||||||
|
left-arrow placeholder safe-area-inset-top fixed
|
||||||
|
@click-left="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 搜索组件 -->
|
||||||
|
<view @click="openSearchForm">
|
||||||
|
<wd-search :placeholder="searchPlaceholder" hide-cancel disabled />
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 客户列表 -->
|
||||||
|
<view class="px-24rpx">
|
||||||
|
<view
|
||||||
|
v-for="item in list"
|
||||||
|
:key="item.id"
|
||||||
|
class="mb-20rpx overflow-hidden rounded-12rpx bg-white shadow-sm"
|
||||||
|
@click="handleEdit(item)"
|
||||||
|
>
|
||||||
|
<view class="p-24rpx">
|
||||||
|
<!-- 头部:名称 + 状态 -->
|
||||||
|
<view class="mb-16rpx flex items-center justify-between">
|
||||||
|
<view class="text-30rpx text-[#333] font-semibold">
|
||||||
|
{{ item.name || '-' }}
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
class="rounded-8rpx px-16rpx py-4rpx text-24rpx"
|
||||||
|
:class="item.status === 0 ? 'bg-[#f6ffed] text-[#52c41a]' : 'bg-[#fff1f0] text-[#f5222d]'"
|
||||||
|
>
|
||||||
|
{{ item.status === 0 ? '正常' : '停用' }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 联系人 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">联系人</text>
|
||||||
|
<text>{{ item.contact || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 手机号码 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">手机号码</text>
|
||||||
|
<text>{{ item.mobile || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 联系电话 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">联系电话</text>
|
||||||
|
<text>{{ item.telephone || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 备注 -->
|
||||||
|
<view class="flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">备注</text>
|
||||||
|
<text class="ml-16rpx line-clamp-1 text-right" style="max-width: 400rpx;">{{ item.remark || '无备注' }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 操作按钮 -->
|
||||||
|
<view class="flex flex-wrap gap-12rpx px-24rpx pb-20rpx" @click.stop>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:customer:update']) && item.name !== '-'"
|
||||||
|
size="small" type="primary" plain @click="handleEdit(item)"
|
||||||
|
>
|
||||||
|
编辑
|
||||||
|
</wd-button>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:customer:delete']) && item.name !== '-'"
|
||||||
|
size="small" type="error" plain @click="handleDelete(item.id!)"
|
||||||
|
>
|
||||||
|
删除
|
||||||
|
</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 加载更多 -->
|
||||||
|
<view v-if="loadMoreState !== 'loading' && list.length === 0" class="py-100rpx text-center">
|
||||||
|
<wd-status-tip image="content" tip="暂无客户数据" />
|
||||||
|
</view>
|
||||||
|
<wd-loadmore
|
||||||
|
v-if="list.length > 0"
|
||||||
|
:state="loadMoreState"
|
||||||
|
@reload="loadMore"
|
||||||
|
/>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 新增按钮 -->
|
||||||
|
<wd-fab
|
||||||
|
v-if="hasAccessByCodes(['erp:customer:create'])"
|
||||||
|
position="right-bottom"
|
||||||
|
type="primary"
|
||||||
|
:expandable="false"
|
||||||
|
@click="handleAdd"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 搜索弹窗 -->
|
||||||
|
<wd-popup v-model="searchVisible" position="top" @close="searchVisible = false">
|
||||||
|
<view class="yd-search-form-container">
|
||||||
|
<view class="yd-search-form-item">
|
||||||
|
<view class="yd-search-form-label">客户名称</view>
|
||||||
|
<wd-input v-model="searchForm.name" placeholder="请输入客户名称" clearable />
|
||||||
|
</view>
|
||||||
|
<view class="yd-search-form-item">
|
||||||
|
<view class="yd-search-form-label">手机号码</view>
|
||||||
|
<wd-input v-model="searchForm.mobile" placeholder="请输入手机号码" clearable />
|
||||||
|
</view>
|
||||||
|
<view class="yd-search-form-item">
|
||||||
|
<view class="yd-search-form-label">联系电话</view>
|
||||||
|
<wd-input v-model="searchForm.telephone" placeholder="请输入联系电话" clearable />
|
||||||
|
</view>
|
||||||
|
<view class="yd-search-form-actions">
|
||||||
|
<wd-button class="flex-1" plain @click="handleReset">重置</wd-button>
|
||||||
|
<wd-button class="flex-1" type="primary" @click="handleSearch">搜索</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</wd-popup>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import type { Customer } from '@/api/erp/customer'
|
||||||
|
import type { LoadMoreState } from '@/http/types'
|
||||||
|
import { onReachBottom } from '@dcloudio/uni-app'
|
||||||
|
import { computed, onMounted, reactive, ref } from 'vue'
|
||||||
|
import { useToast } from 'wot-design-uni'
|
||||||
|
import { deleteCustomer, getCustomerPage } from '@/api/erp/customer'
|
||||||
|
import { useAccess } from '@/hooks/useAccess'
|
||||||
|
import { navigateBackPlus } from '@/utils'
|
||||||
|
|
||||||
|
definePage({
|
||||||
|
style: {
|
||||||
|
navigationBarTitleText: '',
|
||||||
|
navigationStyle: 'custom',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const { hasAccessByCodes } = useAccess()
|
||||||
|
const toast = useToast()
|
||||||
|
const total = ref(0)
|
||||||
|
const list = ref<Customer[]>([])
|
||||||
|
const loadMoreState = ref<LoadMoreState>('loading')
|
||||||
|
const queryParams = ref<Record<string, any>>({
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 10,
|
||||||
|
})
|
||||||
|
|
||||||
|
// 搜索相关
|
||||||
|
const searchVisible = ref(false)
|
||||||
|
const searchForm = reactive({
|
||||||
|
name: undefined as string | undefined,
|
||||||
|
mobile: undefined as string | undefined,
|
||||||
|
telephone: undefined as string | undefined,
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 打开搜索表单 */
|
||||||
|
function openSearchForm() {
|
||||||
|
searchVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 搜索条件 placeholder */
|
||||||
|
const searchPlaceholder = computed(() => {
|
||||||
|
const conditions: string[] = []
|
||||||
|
if (searchForm.name) conditions.push(`名称:${searchForm.name}`)
|
||||||
|
if (searchForm.mobile) conditions.push(`手机:${searchForm.mobile}`)
|
||||||
|
if (searchForm.telephone) conditions.push(`电话:${searchForm.telephone}`)
|
||||||
|
return conditions.length > 0 ? conditions.join(' | ') : '搜索客户'
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 返回上一页 */
|
||||||
|
function handleBack() {
|
||||||
|
navigateBackPlus()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 查询客户列表 */
|
||||||
|
async function getList() {
|
||||||
|
loadMoreState.value = 'loading'
|
||||||
|
try {
|
||||||
|
const params = { ...queryParams.value }
|
||||||
|
const data = await getCustomerPage(params)
|
||||||
|
list.value = [...list.value, ...data.list]
|
||||||
|
total.value = data.total
|
||||||
|
loadMoreState.value = list.value.length >= total.value ? 'finished' : 'loading'
|
||||||
|
} catch {
|
||||||
|
queryParams.value.pageNo = queryParams.value.pageNo > 1 ? queryParams.value.pageNo - 1 : 1
|
||||||
|
loadMoreState.value = 'error'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 搜索 */
|
||||||
|
function handleSearch() {
|
||||||
|
searchVisible.value = false
|
||||||
|
queryParams.value = {
|
||||||
|
...searchForm,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: queryParams.value.pageSize,
|
||||||
|
}
|
||||||
|
list.value = []
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 重置 */
|
||||||
|
function handleReset() {
|
||||||
|
searchForm.name = undefined
|
||||||
|
searchForm.mobile = undefined
|
||||||
|
searchForm.telephone = undefined
|
||||||
|
searchVisible.value = false
|
||||||
|
queryParams.value = { pageNo: 1, pageSize: 10 }
|
||||||
|
list.value = []
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 加载更多 */
|
||||||
|
function loadMore() {
|
||||||
|
if (loadMoreState.value === 'finished') return
|
||||||
|
queryParams.value.pageNo++
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 新增客户 */
|
||||||
|
function handleAdd() {
|
||||||
|
uni.navigateTo({ url: '/pages-erp/customer/form/index' })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 编辑 */
|
||||||
|
function handleEdit(item: Customer) {
|
||||||
|
if (item.name === '-') return
|
||||||
|
uni.navigateTo({ url: `/pages-erp/customer/form/index?id=${item.id}` })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除 */
|
||||||
|
function handleDelete(id: number) {
|
||||||
|
uni.showModal({
|
||||||
|
title: '提示',
|
||||||
|
content: '确定要删除该客户吗?',
|
||||||
|
success: async (res) => {
|
||||||
|
if (!res.confirm) return
|
||||||
|
try {
|
||||||
|
await deleteCustomer(id)
|
||||||
|
toast.success('删除成功')
|
||||||
|
list.value = []
|
||||||
|
queryParams.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
} catch {
|
||||||
|
// error handled by http
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 触底加载更多 */
|
||||||
|
onReachBottom(() => {
|
||||||
|
loadMore()
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 初始化 */
|
||||||
|
onMounted(() => {
|
||||||
|
getList()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
</style>
|
||||||
311
src/pages-erp/sale-order/detail/index.vue
Normal file
311
src/pages-erp/sale-order/detail/index.vue
Normal file
@@ -0,0 +1,311 @@
|
|||||||
|
<template>
|
||||||
|
<view class="yd-page-container">
|
||||||
|
<!-- 顶部导航栏 -->
|
||||||
|
<wd-navbar
|
||||||
|
title="订单详情"
|
||||||
|
left-arrow placeholder safe-area-inset-top fixed
|
||||||
|
@click-left="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<view class="p-24rpx" :class="{ 'opacity-60': loading }">
|
||||||
|
<!-- 基本信息 -->
|
||||||
|
<view class="bg-white rounded-12rpx p-24rpx mb-24rpx">
|
||||||
|
<view class="text-32rpx font-semibold mb-24rpx text-[#333]">基本信息</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">订单单号</text>
|
||||||
|
<text class="info-value">{{ detail.no || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">客户</text>
|
||||||
|
<text class="info-value">{{ detail.customerName || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">订单时间</text>
|
||||||
|
<text class="info-value">{{ formatDate(detail.orderTime) || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">状态</text>
|
||||||
|
<view :class="getStatusClass(detail.status)">
|
||||||
|
{{ getStatusText(detail.status) }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 付款信息 -->
|
||||||
|
<view class="bg-white rounded-12rpx p-24rpx mb-24rpx" v-if="detail.depositPrice || detail.paymentCondition">
|
||||||
|
<view class="text-32rpx font-semibold mb-24rpx text-[#333]">付款信息</view>
|
||||||
|
<view class="info-row" v-if="detail.paymentCondition">
|
||||||
|
<text class="info-label">付款条件</text>
|
||||||
|
<text class="info-value">{{ getPaymentConditionText(detail.paymentCondition) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row" v-if="detail.paymentMethod">
|
||||||
|
<text class="info-label">付款方式</text>
|
||||||
|
<text class="info-value">{{ getPaymentMethodText(detail.paymentMethod) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row" v-if="detail.depositPrice">
|
||||||
|
<text class="info-label">收取订金</text>
|
||||||
|
<text class="info-value text-[#409eff]">¥{{ formatPrice(detail.depositPrice) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 产品清单 -->
|
||||||
|
<view class="bg-white rounded-12rpx p-24rpx mb-24rpx">
|
||||||
|
<view class="text-32rpx font-semibold mb-24rpx text-[#333]">产品清单</view>
|
||||||
|
<view
|
||||||
|
v-for="(item, index) in detail.items"
|
||||||
|
:key="index"
|
||||||
|
class="product-item"
|
||||||
|
>
|
||||||
|
<view class="flex justify-between items-center mb-8rpx">
|
||||||
|
<text class="text-28rpx font-medium">{{ item.productName }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="flex justify-between text-24rpx text-[#999]">
|
||||||
|
<text>单价: ¥{{ formatPrice(item.unitPrice) }}</text>
|
||||||
|
<text>数量: {{ item.count }}</text>
|
||||||
|
<text class="text-[#409eff]">小计: ¥{{ formatPrice(item.totalPrice) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view v-if="!detail.items || detail.items.length === 0" class="text-center text-26rpx text-[#999] py-20rpx">
|
||||||
|
暂无产品
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 金额汇总 -->
|
||||||
|
<view class="bg-white rounded-12rpx p-24rpx mb-24rpx">
|
||||||
|
<view class="text-32rpx font-semibold mb-24rpx text-[#333]">金额汇总</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">总数量</text>
|
||||||
|
<text class="info-value">{{ formatCount(detail.totalCount) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row">
|
||||||
|
<text class="info-label">产品金额</text>
|
||||||
|
<text class="info-value">¥{{ formatPrice(detail.totalProductPrice) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row" v-if="detail.discountPercent">
|
||||||
|
<text class="info-label">优惠率</text>
|
||||||
|
<text class="info-value">{{ detail.discountPercent }}%</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row" v-if="detail.discountPrice">
|
||||||
|
<text class="info-label">优惠金额</text>
|
||||||
|
<text class="info-value">-¥{{ formatPrice(detail.discountPrice) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="info-row total-row">
|
||||||
|
<text class="info-label">含税金额</text>
|
||||||
|
<text class="info-value text-[#409eff] text-36rpx">¥{{ formatPrice(detail.totalPrice) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 其他信息 -->
|
||||||
|
<view class="bg-white rounded-12rpx p-24rpx mb-24rpx" v-if="detail.remark">
|
||||||
|
<view class="text-32rpx font-semibold mb-24rpx text-[#333]">备注</view>
|
||||||
|
<view class="text-26rpx text-[#666]">{{ detail.remark }}</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 底部操作按钮 -->
|
||||||
|
<view class="yd-detail-footer" v-if="detail.status === 10 && !detail.hasApprovalRecords">
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:update'])"
|
||||||
|
type="primary" plain
|
||||||
|
@click="handleEdit"
|
||||||
|
>
|
||||||
|
编辑
|
||||||
|
</wd-button>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:update-status'])"
|
||||||
|
type="success"
|
||||||
|
@click="handleApprove"
|
||||||
|
>
|
||||||
|
审批
|
||||||
|
</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import type { SaleOrder } from '@/api/erp/sale-order'
|
||||||
|
import { onMounted, ref } from 'vue'
|
||||||
|
import { useToast } from 'wot-design-uni'
|
||||||
|
import { getSaleOrder, updateSaleOrderStatus } from '@/api/erp/sale-order'
|
||||||
|
import { useAccess } from '@/hooks/useAccess'
|
||||||
|
import { navigateBackPlus } from '@/utils'
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
id?: number | any
|
||||||
|
}>()
|
||||||
|
|
||||||
|
definePage({
|
||||||
|
style: {
|
||||||
|
navigationBarTitleText: '',
|
||||||
|
navigationStyle: 'custom',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const { hasAccessByCodes } = useAccess()
|
||||||
|
const toast = useToast()
|
||||||
|
const loading = ref(false)
|
||||||
|
const detail = ref<Partial<SaleOrder>>({})
|
||||||
|
|
||||||
|
/** 返回上一页 */
|
||||||
|
function handleBack() {
|
||||||
|
navigateBackPlus('/pages-erp/sale-order/index')
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化日期 */
|
||||||
|
function formatDate(date: any) {
|
||||||
|
if (!date) return ''
|
||||||
|
const d = new Date(date)
|
||||||
|
const year = d.getFullYear()
|
||||||
|
const month = String(d.getMonth() + 1).padStart(2, '0')
|
||||||
|
const day = String(d.getDate()).padStart(2, '0')
|
||||||
|
return `${year}-${month}-${day}`
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化数量 */
|
||||||
|
function formatCount(count?: number) {
|
||||||
|
if (count === undefined || count === null) return '0'
|
||||||
|
return String(Math.trunc(count))
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化价格 */
|
||||||
|
function formatPrice(price?: number) {
|
||||||
|
if (price === undefined || price === null) return '0.00'
|
||||||
|
return Number(price).toFixed(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取状态样式 */
|
||||||
|
function getStatusClass(status?: number) {
|
||||||
|
switch (status) {
|
||||||
|
case 10:
|
||||||
|
return 'status-tag bg-[#fffbe6] text-[#faad14]'
|
||||||
|
case 20:
|
||||||
|
return 'status-tag bg-[#f6ffed] text-[#52c41a]'
|
||||||
|
default:
|
||||||
|
return 'status-tag bg-[#f5f5f5] text-[#999]'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取状态文本 */
|
||||||
|
function getStatusText(status?: number) {
|
||||||
|
switch (status) {
|
||||||
|
case 10:
|
||||||
|
return '未审核'
|
||||||
|
case 20:
|
||||||
|
return '已审核'
|
||||||
|
default:
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取付款条件文本 */
|
||||||
|
function getPaymentConditionText(condition?: number) {
|
||||||
|
const map: Record<number, string> = {
|
||||||
|
1: '款到发货',
|
||||||
|
2: '货到付款',
|
||||||
|
3: '月结',
|
||||||
|
}
|
||||||
|
return map[condition || 0] || '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取付款方式文本 */
|
||||||
|
function getPaymentMethodText(method?: number) {
|
||||||
|
const map: Record<number, string> = {
|
||||||
|
1: '现金',
|
||||||
|
2: '银行转账',
|
||||||
|
3: '微信支付',
|
||||||
|
4: '支付宝',
|
||||||
|
}
|
||||||
|
return map[method || 0] || '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 编辑 */
|
||||||
|
function handleEdit() {
|
||||||
|
uni.navigateTo({ url: `/pages-erp/sale-order/form/index?id=${props.id}` })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 审批 */
|
||||||
|
function handleApprove() {
|
||||||
|
uni.showModal({
|
||||||
|
title: '提示',
|
||||||
|
content: '确定审批该订单吗?',
|
||||||
|
success: async (res) => {
|
||||||
|
if (!res.confirm) return
|
||||||
|
try {
|
||||||
|
await updateSaleOrderStatus(props.id, 20)
|
||||||
|
toast.success('审批成功')
|
||||||
|
getDetail()
|
||||||
|
} catch {
|
||||||
|
// error handled by http
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 加载详情 */
|
||||||
|
async function getDetail() {
|
||||||
|
if (!props.id) return
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
detail.value = await getSaleOrder(props.id)
|
||||||
|
} catch (e) {
|
||||||
|
toast.error('加载失败')
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 初始化 */
|
||||||
|
onMounted(() => {
|
||||||
|
getDetail()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.info-row {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 16rpx 0;
|
||||||
|
border-bottom: 1rpx solid #f5f5f5;
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-label {
|
||||||
|
color: #999;
|
||||||
|
font-size: 28rpx;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-value {
|
||||||
|
color: #333;
|
||||||
|
font-size: 28rpx;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.total-row {
|
||||||
|
margin-top: 16rpx;
|
||||||
|
padding-top: 16rpx;
|
||||||
|
border-top: 2rpx solid #eee;
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-tag {
|
||||||
|
padding: 4rpx 16rpx;
|
||||||
|
border-radius: 8rpx;
|
||||||
|
font-size: 24rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-item {
|
||||||
|
padding: 16rpx;
|
||||||
|
background: #fafafa;
|
||||||
|
border-radius: 8rpx;
|
||||||
|
margin-bottom: 12rpx;
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
423
src/pages-erp/sale-order/form/index.vue
Normal file
423
src/pages-erp/sale-order/form/index.vue
Normal file
@@ -0,0 +1,423 @@
|
|||||||
|
<template>
|
||||||
|
<view class="yd-page-container">
|
||||||
|
<wd-navbar
|
||||||
|
:title="getTitle"
|
||||||
|
left-arrow
|
||||||
|
placeholder
|
||||||
|
safe-area-inset-top
|
||||||
|
fixed
|
||||||
|
@click-left="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<view class="pb-180rpx">
|
||||||
|
<wd-form ref="formRef" :model="formData" :rules="formRules">
|
||||||
|
<wd-cell-group title="基本信息" border>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.customerId"
|
||||||
|
label="客户"
|
||||||
|
label-width="180rpx"
|
||||||
|
prop="customerId"
|
||||||
|
:columns="customerColumns"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
<wd-datetime-picker
|
||||||
|
v-model="formData.orderTime"
|
||||||
|
type="date"
|
||||||
|
label="订单时间"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.saleUserId"
|
||||||
|
label="销售人员"
|
||||||
|
label-width="180rpx"
|
||||||
|
:columns="userColumns"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<wd-cell-group title="付款信息" border>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.paymentCondition"
|
||||||
|
label="付款条件"
|
||||||
|
label-width="180rpx"
|
||||||
|
:columns="paymentConditionColumns"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.paymentMethod"
|
||||||
|
label="付款方式"
|
||||||
|
label-width="180rpx"
|
||||||
|
:columns="paymentMethodColumns"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
<wd-picker
|
||||||
|
v-model="formData.accountId"
|
||||||
|
label="结算账户"
|
||||||
|
label-width="180rpx"
|
||||||
|
:columns="accountColumns"
|
||||||
|
placeholder="请选择"
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.depositPrice"
|
||||||
|
label="收取订金"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入订金"
|
||||||
|
type="number"
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<wd-cell-group title="订单产品" border>
|
||||||
|
<view class="p-24rpx">
|
||||||
|
<view
|
||||||
|
v-for="(item, index) in formData.items"
|
||||||
|
:key="index"
|
||||||
|
class="mb-16rpx rounded-8rpx bg-gray-50 p-24rpx"
|
||||||
|
>
|
||||||
|
<view class="mb-12rpx flex items-center justify-between">
|
||||||
|
<text class="text-28rpx font-semibold">{{ item.productName || `产品${index + 1}` }}</text>
|
||||||
|
<wd-button size="small" type="error" plain @click="removeItem(index)">删除</wd-button>
|
||||||
|
</view>
|
||||||
|
<view class="mb-12rpx flex items-center">
|
||||||
|
<text class="w-140rpx text-24rpx text-gray-500">数量</text>
|
||||||
|
<wd-input
|
||||||
|
v-model="item.count"
|
||||||
|
type="number"
|
||||||
|
placeholder="数量"
|
||||||
|
clearable
|
||||||
|
@change="calculateTotal"
|
||||||
|
/>
|
||||||
|
</view>
|
||||||
|
<view class="mb-12rpx flex items-center">
|
||||||
|
<text class="w-140rpx text-24rpx text-gray-500">单价</text>
|
||||||
|
<wd-input
|
||||||
|
v-model="item.unitPrice"
|
||||||
|
type="number"
|
||||||
|
placeholder="单价"
|
||||||
|
clearable
|
||||||
|
@change="calculateTotal"
|
||||||
|
/>
|
||||||
|
</view>
|
||||||
|
<view class="flex items-center">
|
||||||
|
<text class="w-140rpx text-24rpx text-gray-500">小计</text>
|
||||||
|
<text class="text-28rpx text-[#409eff]">{{ formatPrice(calculateItemTotal(item)) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<wd-button type="primary" plain block @click="showProductPicker = true">添加产品</wd-button>
|
||||||
|
</view>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<wd-cell-group title="优惠信息" border>
|
||||||
|
<wd-input
|
||||||
|
v-model="formData.discountPercent"
|
||||||
|
label="优惠率(%)"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入优惠率"
|
||||||
|
type="number"
|
||||||
|
clearable
|
||||||
|
@change="calculateTotal"
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatPrice(formData.discountPrice || 0)"
|
||||||
|
label="收款优惠"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="自动计算"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<wd-input
|
||||||
|
:model-value="formatPrice(formData.totalPrice || 0)"
|
||||||
|
label="优惠后金额"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="自动计算"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
|
||||||
|
<wd-cell-group title="其他信息" border>
|
||||||
|
<wd-textarea
|
||||||
|
v-model="formData.remark"
|
||||||
|
label="备注"
|
||||||
|
label-width="180rpx"
|
||||||
|
placeholder="请输入备注"
|
||||||
|
:maxlength="200"
|
||||||
|
show-word-limit
|
||||||
|
clearable
|
||||||
|
/>
|
||||||
|
</wd-cell-group>
|
||||||
|
</wd-form>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view class="yd-detail-footer">
|
||||||
|
<wd-button type="primary" block :loading="formLoading" @click="handleSubmit">
|
||||||
|
保存
|
||||||
|
</wd-button>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<wd-popup v-model="showProductPicker" position="bottom" custom-style="height: 60%">
|
||||||
|
<view class="p-24rpx">
|
||||||
|
<view class="mb-24rpx flex items-center justify-between">
|
||||||
|
<text class="text-32rpx font-semibold">选择产品</text>
|
||||||
|
<wd-button size="small" @click="showProductPicker = false">关闭</wd-button>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<scroll-view scroll-y class="product-picker-list">
|
||||||
|
<view
|
||||||
|
v-for="product in productList"
|
||||||
|
:key="product.id"
|
||||||
|
class="product-picker-item"
|
||||||
|
:class="{ 'is-active': selectedProductId === product.id }"
|
||||||
|
@click="selectedProductId = product.id"
|
||||||
|
>
|
||||||
|
<view class="product-picker-item__main">
|
||||||
|
<text class="product-picker-item__name">{{ product.name }}</text>
|
||||||
|
<text class="product-picker-item__price">¥{{ formatPrice(product.salePrice) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</scroll-view>
|
||||||
|
|
||||||
|
<wd-button type="primary" block class="mt-24rpx" @click="addProduct">确定添加</wd-button>
|
||||||
|
</view>
|
||||||
|
</wd-popup>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import type { FormInstance } from 'wot-design-uni/components/wd-form/types'
|
||||||
|
import type { SaleOrder, SaleOrderItem } from '@/api/erp/sale-order'
|
||||||
|
import { computed, onMounted, ref } from 'vue'
|
||||||
|
import { useToast } from 'wot-design-uni'
|
||||||
|
import { createSaleOrder, getSaleOrder, updateSaleOrder } from '@/api/erp/sale-order'
|
||||||
|
import { getProductSimpleList } from '@/api/erp/product'
|
||||||
|
import { getCustomerSimpleList } from '@/api/erp/customer'
|
||||||
|
import { getSimpleUserList } from '@/api/system/user'
|
||||||
|
import { getAccountSimpleList } from '@/api/erp/account'
|
||||||
|
import { navigateBackPlus } from '@/utils'
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
id?: number | any
|
||||||
|
}>()
|
||||||
|
|
||||||
|
definePage({
|
||||||
|
style: {
|
||||||
|
navigationBarTitleText: '',
|
||||||
|
navigationStyle: 'custom',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const toast = useToast()
|
||||||
|
const getTitle = computed(() => (props.id ? '编辑订单' : '新增订单'))
|
||||||
|
const formLoading = ref(false)
|
||||||
|
const formRef = ref<FormInstance>()
|
||||||
|
|
||||||
|
const customerList = ref<any[]>([])
|
||||||
|
const userList = ref<any[]>([])
|
||||||
|
const productList = ref<any[]>([])
|
||||||
|
const accountList = ref<any[]>([])
|
||||||
|
|
||||||
|
const showProductPicker = ref(false)
|
||||||
|
const selectedProductId = ref<number>()
|
||||||
|
|
||||||
|
const formData = ref<SaleOrder>({
|
||||||
|
id: undefined,
|
||||||
|
customerId: undefined,
|
||||||
|
orderTime: Date.now(),
|
||||||
|
saleUserId: undefined,
|
||||||
|
paymentCondition: undefined,
|
||||||
|
paymentMethod: undefined,
|
||||||
|
accountId: undefined,
|
||||||
|
depositPrice: undefined,
|
||||||
|
discountPercent: undefined,
|
||||||
|
discountPrice: 0,
|
||||||
|
totalPrice: 0,
|
||||||
|
remark: undefined,
|
||||||
|
items: [],
|
||||||
|
})
|
||||||
|
|
||||||
|
const formRules = {
|
||||||
|
customerId: [{ required: true, message: '请选择客户' }],
|
||||||
|
}
|
||||||
|
|
||||||
|
const customerColumns = computed(() => customerList.value.map(v => ({ label: v.name, value: v.id })))
|
||||||
|
const userColumns = computed(() => userList.value.map(v => ({ label: v.nickname, value: v.id })))
|
||||||
|
const accountColumns = computed(() => accountList.value.map(v => ({ label: v.name, value: v.id })))
|
||||||
|
|
||||||
|
const paymentConditionOptions = [
|
||||||
|
{ label: '款到发货', value: 1 },
|
||||||
|
{ label: '货到付款', value: 2 },
|
||||||
|
{ label: '月结', value: 3 },
|
||||||
|
]
|
||||||
|
const paymentConditionColumns = computed(() => paymentConditionOptions.map(v => ({ label: v.label, value: v.value })))
|
||||||
|
|
||||||
|
const paymentMethodOptions = [
|
||||||
|
{ label: '现金', value: 1 },
|
||||||
|
{ label: '银行转账', value: 2 },
|
||||||
|
{ label: '微信支付', value: 3 },
|
||||||
|
{ label: '支付宝', value: 4 },
|
||||||
|
]
|
||||||
|
const paymentMethodColumns = computed(() => paymentMethodOptions.map(v => ({ label: v.label, value: v.value })))
|
||||||
|
|
||||||
|
function formatPrice(price?: number) {
|
||||||
|
if (price === undefined || price === null) return '0.00'
|
||||||
|
return Number(price).toFixed(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
function calculateItemTotal(item: SaleOrderItem) {
|
||||||
|
const count = Number(item.count) || 0
|
||||||
|
const unitPrice = Number(item.unitPrice) || 0
|
||||||
|
return count * unitPrice
|
||||||
|
}
|
||||||
|
|
||||||
|
function calculateTotal() {
|
||||||
|
if (!formData.value.items || formData.value.items.length === 0) {
|
||||||
|
formData.value.totalPrice = 0
|
||||||
|
formData.value.discountPrice = 0
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const total = formData.value.items.reduce((sum, item) => sum + calculateItemTotal(item), 0)
|
||||||
|
const discountPercent = Number(formData.value.discountPercent) || 0
|
||||||
|
formData.value.discountPrice = total * (discountPercent / 100)
|
||||||
|
formData.value.totalPrice = total - formData.value.discountPrice
|
||||||
|
}
|
||||||
|
|
||||||
|
function addProduct() {
|
||||||
|
const product = productList.value.find(p => p.id === selectedProductId.value)
|
||||||
|
if (!product) {
|
||||||
|
toast.warning('请选择产品')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const newItem: SaleOrderItem = {
|
||||||
|
productId: product.id,
|
||||||
|
productName: product.name,
|
||||||
|
count: 1,
|
||||||
|
unitPrice: product.salePrice || 0,
|
||||||
|
totalPrice: product.salePrice || 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
formData.value.items = [...(formData.value.items || []), newItem]
|
||||||
|
selectedProductId.value = undefined
|
||||||
|
showProductPicker.value = false
|
||||||
|
calculateTotal()
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeItem(index: number) {
|
||||||
|
formData.value.items?.splice(index, 1)
|
||||||
|
calculateTotal()
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleBack() {
|
||||||
|
navigateBackPlus('/pages-erp/sale-order/index')
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDetail() {
|
||||||
|
if (!props.id) return
|
||||||
|
try {
|
||||||
|
toast.loading('加载中...')
|
||||||
|
formData.value = await getSaleOrder(props.id)
|
||||||
|
calculateTotal()
|
||||||
|
} finally {
|
||||||
|
toast.close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadOptions() {
|
||||||
|
const [customersResult, usersResult, productsResult, accountsResult] = await Promise.allSettled([
|
||||||
|
getCustomerSimpleList(),
|
||||||
|
getSimpleUserList(),
|
||||||
|
getProductSimpleList(),
|
||||||
|
getAccountSimpleList(),
|
||||||
|
])
|
||||||
|
|
||||||
|
customerList.value = customersResult.status === 'fulfilled' ? (customersResult.value || []) : []
|
||||||
|
userList.value = usersResult.status === 'fulfilled' ? (usersResult.value || []) : []
|
||||||
|
productList.value = productsResult.status === 'fulfilled' ? (productsResult.value || []) : []
|
||||||
|
accountList.value = accountsResult.status === 'fulfilled' ? (accountsResult.value || []) : []
|
||||||
|
|
||||||
|
if ([customersResult, usersResult, productsResult, accountsResult].some(result => result.status === 'rejected')) {
|
||||||
|
console.error('加载选项失败', {
|
||||||
|
customersResult,
|
||||||
|
usersResult,
|
||||||
|
productsResult,
|
||||||
|
accountsResult,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleSubmit() {
|
||||||
|
const { valid } = await formRef.value!.validate()
|
||||||
|
if (!valid) return
|
||||||
|
|
||||||
|
if (!formData.value.items || formData.value.items.length === 0) {
|
||||||
|
toast.warning('请添加订单产品')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
const submitData = { ...formData.value }
|
||||||
|
if (submitData.orderTime) {
|
||||||
|
submitData.orderTime = new Date(submitData.orderTime).getTime()
|
||||||
|
}
|
||||||
|
|
||||||
|
submitData.items = (submitData.items || []).map(item => ({
|
||||||
|
...item,
|
||||||
|
totalPrice: calculateItemTotal(item),
|
||||||
|
}))
|
||||||
|
|
||||||
|
if (props.id) {
|
||||||
|
await updateSaleOrder(submitData)
|
||||||
|
toast.success('修改成功')
|
||||||
|
} else {
|
||||||
|
await createSaleOrder(submitData)
|
||||||
|
toast.success('新增成功')
|
||||||
|
}
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
handleBack()
|
||||||
|
}, 500)
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
await loadOptions()
|
||||||
|
await getDetail()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.product-picker-list {
|
||||||
|
max-height: 720rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-picker-item {
|
||||||
|
padding: 24rpx;
|
||||||
|
border-bottom: 1rpx solid #f0f0f0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-picker-item.is-active {
|
||||||
|
background: #f0f7ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-picker-item__main {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 24rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-picker-item__name {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.product-picker-item__price {
|
||||||
|
flex-shrink: 0;
|
||||||
|
font-size: 24rpx;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
403
src/pages-erp/sale-order/index.vue
Normal file
403
src/pages-erp/sale-order/index.vue
Normal file
@@ -0,0 +1,403 @@
|
|||||||
|
<template>
|
||||||
|
<view class="yd-page-container">
|
||||||
|
<!-- 顶部导航栏 -->
|
||||||
|
<wd-navbar
|
||||||
|
title="销售订单"
|
||||||
|
left-arrow placeholder safe-area-inset-top fixed
|
||||||
|
@click-left="handleBack"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 搜索组件 -->
|
||||||
|
<view @click="openSearchForm">
|
||||||
|
<wd-search :placeholder="searchPlaceholder" hide-cancel disabled />
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 分类标签 -->
|
||||||
|
<view class="status-tabs">
|
||||||
|
<view
|
||||||
|
:class="['status-tabs__item', { 'is-active': queryParams.status === undefined }]"
|
||||||
|
@click="onStatusTabChange(undefined)"
|
||||||
|
>
|
||||||
|
全部
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
:class="['status-tabs__item', { 'is-active': queryParams.status === 10 }]"
|
||||||
|
@click="onStatusTabChange(10)"
|
||||||
|
>
|
||||||
|
未审核
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
:class="['status-tabs__item', { 'is-active': queryParams.status === 20 }]"
|
||||||
|
@click="onStatusTabChange(20)"
|
||||||
|
>
|
||||||
|
已审核
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 订单列表 -->
|
||||||
|
<view class="px-24rpx">
|
||||||
|
<view
|
||||||
|
v-for="item in list"
|
||||||
|
:key="item.id"
|
||||||
|
class="mb-20rpx overflow-hidden rounded-12rpx bg-white shadow-sm"
|
||||||
|
>
|
||||||
|
<view class="p-24rpx" @click="handleDetail(item)">
|
||||||
|
<!-- 头部:单号 + 状态 -->
|
||||||
|
<view class="mb-16rpx flex items-center justify-between">
|
||||||
|
<view class="text-28rpx text-[#333] font-semibold">
|
||||||
|
{{ item.no || '-' }}
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
class="rounded-8rpx px-16rpx py-4rpx text-24rpx"
|
||||||
|
:class="getStatusClass(item.status)"
|
||||||
|
>
|
||||||
|
{{ getStatusText(item.status) }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 客户 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">客户</text>
|
||||||
|
<text>{{ item.customerName || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 产品 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">产品</text>
|
||||||
|
<text class="ml-16rpx line-clamp-1 text-right" style="max-width: 400rpx;">{{ item.productNames || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 订单时间 -->
|
||||||
|
<view class="mb-8rpx flex items-center justify-between text-26rpx text-[#666]">
|
||||||
|
<text class="text-[#999]">订单时间</text>
|
||||||
|
<text>{{ formatDate(item.orderTime) || '-' }}</text>
|
||||||
|
</view>
|
||||||
|
<!-- 数量统计 -->
|
||||||
|
<view class="flex justify-around pt-16rpx border-t border-[#f0f0f0]">
|
||||||
|
<view class="text-center">
|
||||||
|
<view class="text-28rpx font-semibold text-[#333]">{{ formatCount(item.totalCount) }}</view>
|
||||||
|
<view class="text-22rpx text-[#999]">总数量</view>
|
||||||
|
</view>
|
||||||
|
<view class="text-center">
|
||||||
|
<view class="text-28rpx font-semibold text-[#333]">{{ formatCount(item.outCount) }}</view>
|
||||||
|
<view class="text-22rpx text-[#999]">出库数</view>
|
||||||
|
</view>
|
||||||
|
<view class="text-center">
|
||||||
|
<view class="text-28rpx font-semibold text-[#409eff]">¥{{ formatPrice(item.totalPrice) }}</view>
|
||||||
|
<view class="text-22rpx text-[#999]">含税金额</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 操作按钮 -->
|
||||||
|
<view class="flex flex-wrap gap-12rpx px-24rpx pb-20rpx" @click.stop>
|
||||||
|
<wd-button
|
||||||
|
size="small" plain @click="handleDetail(item)"
|
||||||
|
>
|
||||||
|
详情
|
||||||
|
</wd-button>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:update']) && item.status === 10 && !item.hasApprovalRecords"
|
||||||
|
size="small" type="primary" plain @click="handleEdit(item)"
|
||||||
|
>
|
||||||
|
编辑
|
||||||
|
</wd-button>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:update-status']) && item.status === 10"
|
||||||
|
size="small" type="success" plain @click="handleApprove(item.id)"
|
||||||
|
>
|
||||||
|
审批
|
||||||
|
</wd-button>
|
||||||
|
<wd-button
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:delete']) && !item.hasApprovalRecords"
|
||||||
|
size="small" type="error" plain @click="handleDelete(item.id)"
|
||||||
|
>
|
||||||
|
删除
|
||||||
|
</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 加载更多 -->
|
||||||
|
<view v-if="loadMoreState !== 'loading' && list.length === 0" class="py-100rpx text-center">
|
||||||
|
<wd-status-tip image="content" tip="暂无订单数据" />
|
||||||
|
</view>
|
||||||
|
<wd-loadmore
|
||||||
|
v-if="list.length > 0"
|
||||||
|
:state="loadMoreState"
|
||||||
|
@reload="loadMore"
|
||||||
|
/>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 新增按钮 -->
|
||||||
|
<wd-fab
|
||||||
|
v-if="hasAccessByCodes(['erp:sale-order:create'])"
|
||||||
|
position="right-bottom"
|
||||||
|
type="primary"
|
||||||
|
:expandable="false"
|
||||||
|
@click="handleAdd"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 搜索弹窗 -->
|
||||||
|
<wd-popup v-model="searchVisible" position="top" @close="searchVisible = false">
|
||||||
|
<view class="yd-search-form-container">
|
||||||
|
<view class="yd-search-form-item">
|
||||||
|
<view class="yd-search-form-label">订单单号</view>
|
||||||
|
<wd-input v-model="searchForm.no" placeholder="请输入订单单号" clearable />
|
||||||
|
</view>
|
||||||
|
<view class="yd-search-form-actions">
|
||||||
|
<wd-button class="flex-1" plain @click="handleReset">重置</wd-button>
|
||||||
|
<wd-button class="flex-1" type="primary" @click="handleSearch">搜索</wd-button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</wd-popup>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import type { SaleOrder } from '@/api/erp/sale-order'
|
||||||
|
import type { LoadMoreState } from '@/http/types'
|
||||||
|
import { onReachBottom } from '@dcloudio/uni-app'
|
||||||
|
import { computed, onMounted, reactive, ref } from 'vue'
|
||||||
|
import { useToast } from 'wot-design-uni'
|
||||||
|
import { deleteSaleOrder, getSaleOrderPage, updateSaleOrderStatus } from '@/api/erp/sale-order'
|
||||||
|
import { useAccess } from '@/hooks/useAccess'
|
||||||
|
import { navigateBackPlus } from '@/utils'
|
||||||
|
|
||||||
|
definePage({
|
||||||
|
style: {
|
||||||
|
navigationBarTitleText: '',
|
||||||
|
navigationStyle: 'custom',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 打开搜索表单 */
|
||||||
|
function openSearchForm() {
|
||||||
|
searchVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const { hasAccessByCodes } = useAccess()
|
||||||
|
const toast = useToast()
|
||||||
|
const total = ref(0)
|
||||||
|
const list = ref<SaleOrder[]>([])
|
||||||
|
const loadMoreState = ref<LoadMoreState>('loading')
|
||||||
|
const queryParams = ref<Record<string, any>>({
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 10,
|
||||||
|
status: undefined as number | undefined,
|
||||||
|
})
|
||||||
|
|
||||||
|
// 搜索相关
|
||||||
|
const searchVisible = ref(false)
|
||||||
|
const searchForm = reactive({
|
||||||
|
no: undefined as string | undefined,
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 搜索条件 placeholder */
|
||||||
|
const searchPlaceholder = computed(() => {
|
||||||
|
const conditions: string[] = []
|
||||||
|
if (searchForm.no) conditions.push(`单号:${searchForm.no}`)
|
||||||
|
return conditions.length > 0 ? conditions.join(' | ') : '搜索订单'
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 返回上一页 */
|
||||||
|
function handleBack() {
|
||||||
|
navigateBackPlus()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 状态切换 */
|
||||||
|
function onStatusTabChange(status: number | undefined) {
|
||||||
|
queryParams.value.status = status
|
||||||
|
queryParams.value.pageNo = 1
|
||||||
|
list.value = []
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化日期 */
|
||||||
|
function formatDate(date?: string) {
|
||||||
|
if (!date) return '-'
|
||||||
|
return new Date(date).toLocaleDateString('zh-CN')
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化数量 */
|
||||||
|
function formatCount(count?: number) {
|
||||||
|
if (count === undefined || count === null) return '0'
|
||||||
|
return String(Math.trunc(count))
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 格式化价格 */
|
||||||
|
function formatPrice(price?: number) {
|
||||||
|
if (price === undefined || price === null) return '0.00'
|
||||||
|
return Number(price).toFixed(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取状态样式 */
|
||||||
|
function getStatusClass(status?: number) {
|
||||||
|
switch (status) {
|
||||||
|
case 10:
|
||||||
|
return 'bg-[#fffbe6] text-[#faad14]'
|
||||||
|
case 20:
|
||||||
|
return 'bg-[#f6ffed] text-[#52c41a]'
|
||||||
|
default:
|
||||||
|
return 'bg-[#f5f5f5] text-[#999]'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 获取状态文本 */
|
||||||
|
function getStatusText(status?: number) {
|
||||||
|
switch (status) {
|
||||||
|
case 10:
|
||||||
|
return '未审核'
|
||||||
|
case 20:
|
||||||
|
return '已审核'
|
||||||
|
default:
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 查询订单列表 */
|
||||||
|
async function getList() {
|
||||||
|
loadMoreState.value = 'loading'
|
||||||
|
try {
|
||||||
|
const params = { ...queryParams.value }
|
||||||
|
const data = await getSaleOrderPage(params)
|
||||||
|
list.value = [...list.value, ...data.list]
|
||||||
|
total.value = data.total
|
||||||
|
loadMoreState.value = list.value.length >= total.value ? 'finished' : 'loading'
|
||||||
|
} catch {
|
||||||
|
queryParams.value.pageNo = queryParams.value.pageNo > 1 ? queryParams.value.pageNo - 1 : 1
|
||||||
|
loadMoreState.value = 'error'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 搜索 */
|
||||||
|
function handleSearch() {
|
||||||
|
searchVisible.value = false
|
||||||
|
queryParams.value = {
|
||||||
|
...queryParams.value,
|
||||||
|
...searchForm,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: queryParams.value.pageSize,
|
||||||
|
}
|
||||||
|
list.value = []
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 重置 */
|
||||||
|
function handleReset() {
|
||||||
|
searchForm.no = undefined
|
||||||
|
searchVisible.value = false
|
||||||
|
queryParams.value = { pageNo: 1, pageSize: 10, status: queryParams.value.status }
|
||||||
|
list.value = []
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 加载更多 */
|
||||||
|
function loadMore() {
|
||||||
|
if (loadMoreState.value === 'finished') return
|
||||||
|
queryParams.value.pageNo++
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 新增订单 */
|
||||||
|
function handleAdd() {
|
||||||
|
uni.navigateTo({ url: '/pages-erp/sale-order/form/index' })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 编辑订单 */
|
||||||
|
function handleEdit(item: SaleOrder) {
|
||||||
|
uni.navigateTo({ url: `/pages-erp/sale-order/form/index?id=${item.id}` })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 查看详情 */
|
||||||
|
function handleDetail(item: SaleOrder) {
|
||||||
|
uni.navigateTo({ url: `/pages-erp/sale-order/detail/index?id=${item.id}` })
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 审批 */
|
||||||
|
function handleApprove(id: number) {
|
||||||
|
uni.showModal({
|
||||||
|
title: '提示',
|
||||||
|
content: '确定审批该订单吗?',
|
||||||
|
success: async (res) => {
|
||||||
|
if (!res.confirm) return
|
||||||
|
try {
|
||||||
|
await updateSaleOrderStatus(id, 20)
|
||||||
|
toast.success('审批成功')
|
||||||
|
list.value = []
|
||||||
|
queryParams.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
} catch {
|
||||||
|
// error handled by http
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除 */
|
||||||
|
function handleDelete(id: number) {
|
||||||
|
uni.showModal({
|
||||||
|
title: '提示',
|
||||||
|
content: '确定要删除该订单吗?',
|
||||||
|
success: async (res) => {
|
||||||
|
if (!res.confirm) return
|
||||||
|
try {
|
||||||
|
await deleteSaleOrder([id])
|
||||||
|
toast.success('删除成功')
|
||||||
|
list.value = []
|
||||||
|
queryParams.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
} catch {
|
||||||
|
// error handled by http
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 触底加载更多 */
|
||||||
|
onReachBottom(() => {
|
||||||
|
loadMore()
|
||||||
|
})
|
||||||
|
|
||||||
|
/** 初始化 */
|
||||||
|
onMounted(() => {
|
||||||
|
getList()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.status-tabs {
|
||||||
|
display: flex;
|
||||||
|
margin: 0 24rpx 16rpx;
|
||||||
|
background: #fff;
|
||||||
|
border-radius: 8rpx;
|
||||||
|
overflow: hidden;
|
||||||
|
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.04);
|
||||||
|
|
||||||
|
&__item {
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
padding: 20rpx 0;
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #606266;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
&.is-active {
|
||||||
|
color: #018d71;
|
||||||
|
font-weight: 600;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 25%;
|
||||||
|
width: 50%;
|
||||||
|
height: 4rpx;
|
||||||
|
background: #018d71;
|
||||||
|
border-radius: 2rpx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&:not(:last-child) {
|
||||||
|
border-right: 1rpx solid #f0f0f0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -88,6 +88,28 @@ const menuGroupsData: MenuGroup[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: 'sale',
|
||||||
|
name: '销售管理',
|
||||||
|
menus: [
|
||||||
|
{
|
||||||
|
key: 'customer',
|
||||||
|
name: '客户管理',
|
||||||
|
icon: 'user',
|
||||||
|
url: '/pages-erp/customer/index',
|
||||||
|
iconColor: '#1890ff',
|
||||||
|
permission: 'erp:customer:query',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'saleOrder',
|
||||||
|
name: '销售订单',
|
||||||
|
icon: 'order',
|
||||||
|
url: '/pages-erp/sale-order/index',
|
||||||
|
iconColor: '#1890ff',
|
||||||
|
permission: 'erp:sale-order:query',
|
||||||
|
}
|
||||||
|
],
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: 'agri',
|
key: 'agri',
|
||||||
name: '农业溯源',
|
name: '农业溯源',
|
||||||
|
|||||||
Reference in New Issue
Block a user