Browse Source

1

yuhongzhe
于宏哲PHP 11 months ago
parent
commit
275211cb83
  1. 4
      admin/components.d.ts
  2. 30
      admin/src/app/api/customer_resources.ts
  3. 27
      admin/src/app/api/six_speed.ts
  4. 12
      admin/src/app/api/sys.ts
  5. 72
      admin/src/app/lang/zh-cn/customer_resources.customer_resources.json
  6. 60
      admin/src/app/lang/zh-cn/six_speed.six_speed.json
  7. 717
      admin/src/app/views/customer_resources/components/customer-resources-edit.vue
  8. 248
      admin/src/app/views/customer_resources/customer_resources.vue
  9. 331
      admin/src/app/views/six_speed/components/six-speed-edit.vue
  10. 324
      admin/src/app/views/six_speed/six_speed.vue
  11. 44
      niucloud/app/adminapi/controller/customer_resources/CustomerResources.php
  12. 34
      niucloud/app/adminapi/controller/six_speed/SixSpeed.php
  13. 4
      niucloud/app/adminapi/route/customer_resources.php
  14. 5
      niucloud/app/adminapi/route/six_speed.php
  15. 28
      niucloud/app/common.php
  16. 6
      niucloud/app/model/customer_resources/CustomerResources.php
  17. 172
      niucloud/app/model/six_speed/SixSpeed.php
  18. 149
      niucloud/app/service/admin/customer_resources/CustomerResourcesService.php
  19. 16
      niucloud/app/service/admin/six_speed/SixSpeedService.php
  20. 8
      niucloud/app/validate/six_speed/SixSpeed.php

4
admin/components.d.ts

@ -10,7 +10,6 @@ declare module '@vue/runtime-core' {
Attachment: typeof import('./src/components/upload-attachment/attachment.vue')['default']
DiyLink: typeof import('./src/components/diy-link/index.vue')['default']
Editor: typeof import('./src/components/editor/index.vue')['default']
ElAlert: typeof import('element-plus/es')['ElAlert']
ElAside: typeof import('element-plus/es')['ElAside']
ElAvatar: typeof import('element-plus/es')['ElAvatar']
ElBreadcrumb: typeof import('element-plus/es')['ElBreadcrumb']
@ -45,7 +44,6 @@ declare module '@vue/runtime-core' {
ElPopover: typeof import('element-plus/es')['ElPopover']
ElRadio: typeof import('element-plus/es')['ElRadio']
ElRadioGroup: typeof import('element-plus/es')['ElRadioGroup']
ElResult: typeof import('element-plus/es')['ElResult']
ElRow: typeof import('element-plus/es')['ElRow']
ElScrollbar: typeof import('element-plus/es')['ElScrollbar']
ElSelect: typeof import('element-plus/es')['ElSelect']
@ -58,8 +56,6 @@ declare module '@vue/runtime-core' {
ElTabPane: typeof import('element-plus/es')['ElTabPane']
ElTabs: typeof import('element-plus/es')['ElTabs']
ElTag: typeof import('element-plus/es')['ElTag']
ElTimeline: typeof import('element-plus/es')['ElTimeline']
ElTimelineItem: typeof import('element-plus/es')['ElTimelineItem']
ElTooltip: typeof import('element-plus/es')['ElTooltip']
ElTree: typeof import('element-plus/es')['ElTree']
ElTreeSelect: typeof import('element-plus/es')['ElTreeSelect']

30
admin/src/app/api/customer_resources.ts

@ -1,5 +1,9 @@
import request from '@/utils/request'
// USER_CODE_BEGIN -- customer_resources
/**
*
@ -7,7 +11,7 @@ import request from '@/utils/request'
* @returns
*/
export function getCustomerResourcesList(params: Record<string, any>) {
return request.get(`customer_resources/customer_resources`, { params })
return request.get(`customer_resources/customer_resources`, {params})
}
/**
@ -16,7 +20,7 @@ export function getCustomerResourcesList(params: Record<string, any>) {
* @returns
*/
export function getCustomerResourcesInfo(id: number) {
return request.get(`customer_resources/customer_resources/${id}`)
return request.get(`customer_resources/customer_resources/${id}`);
}
/**
@ -25,10 +29,7 @@ export function getCustomerResourcesInfo(id: number) {
* @returns
*/
export function addCustomerResources(params: Record<string, any>) {
return request.post('customer_resources/customer_resources', params, {
showErrorMessage: true,
showSuccessMessage: true,
})
return request.post('customer_resources/customer_resources', params, { showErrorMessage: true, showSuccessMessage: true })
}
/**
@ -38,11 +39,7 @@ export function addCustomerResources(params: Record<string, any>) {
* @returns
*/
export function editCustomerResources(params: Record<string, any>) {
return request.put(
`customer_resources/customer_resources/${params.id}`,
params,
{ showErrorMessage: true, showSuccessMessage: true }
)
return request.put(`customer_resources/customer_resources/${params.id}`, params, { showErrorMessage: true, showSuccessMessage: true })
}
/**
@ -51,14 +48,13 @@ export function editCustomerResources(params: Record<string, any>) {
* @returns
*/
export function deleteCustomerResources(id: number) {
return request.delete(`customer_resources/customer_resources/${id}`, {
showErrorMessage: true,
showSuccessMessage: true,
})
return request.delete(`customer_resources/customer_resources/${id}`, { showErrorMessage: true, showSuccessMessage: true })
}
export function getWithCampusList(params: Record<string, any>) {
return request.get('customer_resources/campus_all', { params })
export function getWithPersonnelList(params: Record<string,any>){
return request.get('customer_resources/personnel_all', {params})
}export function getWithCampusList(params: Record<string,any>){
return request.get('customer_resources/campus_all', {params})
}
// USER_CODE_END -- customer_resources

27
admin/src/app/api/six_speed.ts

@ -1,5 +1,7 @@
import request from '@/utils/request'
// USER_CODE_BEGIN -- six_speed
/**
*
@ -7,7 +9,7 @@ import request from '@/utils/request'
* @returns
*/
export function getSixSpeedList(params: Record<string, any>) {
return request.get(`six_speed/six_speed`, { params })
return request.get(`six_speed/six_speed`, {params})
}
/**
@ -16,7 +18,7 @@ export function getSixSpeedList(params: Record<string, any>) {
* @returns
*/
export function getSixSpeedInfo(id: number) {
return request.get(`six_speed/six_speed/${id}`)
return request.get(`six_speed/six_speed/${id}`);
}
/**
@ -25,10 +27,7 @@ export function getSixSpeedInfo(id: number) {
* @returns
*/
export function addSixSpeed(params: Record<string, any>) {
return request.post('six_speed/six_speed', params, {
showErrorMessage: true,
showSuccessMessage: true,
})
return request.post('six_speed/six_speed', params, { showErrorMessage: true, showSuccessMessage: true })
}
/**
@ -38,10 +37,7 @@ export function addSixSpeed(params: Record<string, any>) {
* @returns
*/
export function editSixSpeed(params: Record<string, any>) {
return request.put(`six_speed/six_speed/${params.id}`, params, {
showErrorMessage: true,
showSuccessMessage: true,
})
return request.put(`six_speed/six_speed/${params.id}`, params, { showErrorMessage: true, showSuccessMessage: true })
}
/**
@ -50,10 +46,13 @@ export function editSixSpeed(params: Record<string, any>) {
* @returns
*/
export function deleteSixSpeed(id: number) {
return request.delete(`six_speed/six_speed/${id}`, {
showErrorMessage: true,
showSuccessMessage: true,
})
return request.delete(`six_speed/six_speed/${id}`, { showErrorMessage: true, showSuccessMessage: true })
}
export function getWithPersonnelList(params: Record<string,any>){
return request.get('six_speed/personnel_all', {params})
}export function getWithCustomerResourcesList(params: Record<string,any>){
return request.get('six_speed/customer_resources_all', {params})
}
// USER_CODE_END -- six_speed

12
admin/src/app/api/sys.ts

@ -720,3 +720,15 @@ export function deleteExport(id: number) {
export function getInstallConfig() {
return request.get('sys/install/config')
}
//业绩配置
export function getYjpzConfig() {
return request.get('sys/get_yjpz_config')
}
export function yjpzConfig(params: Record<string, any>) {
return request.post(`sys/yjpz_config`, params)
}

72
admin/src/app/lang/zh-cn/customer_resources.customer_resources.json

@ -1,38 +1,38 @@
{
"source": "来源",
"sourcePlaceholder": "请输入来源",
"sourceChannel": "来源渠道",
"sourceChannelPlaceholder": "请输入来源渠道",
"consultant": "顾问",
"name": "姓名",
"namePlaceholder": "请输入姓名",
"age": "年龄",
"agePlaceholder": "请输入年龄",
"gender": "性别",
"genderPlaceholder": "请输入性别",
"phoneNumber": "联系电话",
"phoneNumberPlaceholder": "请输入联系电话",
"demand": "需求",
"demandPlaceholder": "请输入需求",
"purchasingPower": "购买力",
"purchasingPowerPlaceholder": "请输入购买力",
"cognitiveIdea": "认知理念",
"cognitiveIdeaPlaceholder": "请输入认知理念",
"optionalClassTime": "可选上课时间",
"optionalClassTimePlaceholder": "请输入可选上课时间",
"distance": "距离",
"distancePlaceholder": "请输入距离",
"decisionMaker": "决策人",
"decisionMakerPlaceholder": "请输入决策人",
"initialIntent": "客户初步意向度",
"initialIntentPlaceholder": "请输入客户初步意向度",
"campus": "所属校区",
"campusPlaceholder": "请输入所属校区",
"status": "客户状态",
"statusPlaceholder": "请输入客户状态",
"addCustomerResources": "添加客户资源",
"updateCustomerResources": "编辑客户资源",
"customerResourcesDeleteTips": "确定要删除该数据吗?",
"startDate": "请选择开始时间",
"endDate": "请选择结束时间"
"source":"来源",
"sourcePlaceholder":"请输入来源",
"sourceChannel":"来源渠道",
"sourceChannelPlaceholder":"请输入来源渠道",
"consultant":"顾问",
"name":"姓名",
"namePlaceholder":"请输入姓名",
"age":"年龄",
"agePlaceholder":"请输入年龄",
"gender":"性别",
"genderPlaceholder":"请输入性别",
"phoneNumber":"联系电话",
"phoneNumberPlaceholder":"请输入联系电话",
"demand":"需求",
"demandPlaceholder":"请输入需求",
"purchasingPower":"购买力",
"purchasingPowerPlaceholder":"请输入购买力",
"cognitiveIdea":"认知理念",
"cognitiveIdeaPlaceholder":"请输入认知理念",
"optionalClassTime":"可选上课时间",
"optionalClassTimePlaceholder":"请输入可选上课时间",
"distance":"距离",
"distancePlaceholder":"请输入距离",
"decisionMaker":"决策人",
"decisionMakerPlaceholder":"请输入决策人",
"initialIntent":"客户初步意向度",
"initialIntentPlaceholder":"请输入客户初步意向度",
"campus":"所属校区",
"campusPlaceholder":"请输入所属校区",
"status":"客户状态",
"statusPlaceholder":"请输入客户状态",
"addCustomerResources":"添加客户资源",
"updateCustomerResources":"编辑客户资源",
"customerResourcesDeleteTips":"确定要删除该数据吗?",
"startDate":"请选择开始时间",
"endDate":"请选择结束时间"
}

60
admin/src/app/lang/zh-cn/six_speed.six_speed.json

@ -1,35 +1,29 @@
{
"id": "编号",
"idPlaceholder": "请输入编号",
"purchasePower": "需求购买力",
"purchasePowerPlaceholder": "请输入需求购买力",
"conceptAwareness": "认知理念",
"conceptAwarenessPlaceholder": "请输入认知理念",
"preferredClassTime": "可选上课时间",
"preferredClassTimePlaceholder": "请输入可选上课时间",
"distance": "距离",
"distancePlaceholder": "请输入距离",
"communication": "沟通备注",
"communicationPlaceholder": "请输入沟通备注",
"promisedVisitTime": "承诺到访时间",
"promisedVisitTimePlaceholder": "请输入承诺到访时间",
"actualVisitTime": "实际到访时间",
"actualVisitTimePlaceholder": "请输入实际到访时间",
"callIntent": "电话后的意向程度: low-低, medium-中, high-高",
"callIntentPlaceholder": "请输入电话后的意向程度: low-低, medium-中, high-高",
"firstVisitStatus": "一访情况",
"firstVisitStatusPlaceholder": "请输入一访情况",
"secondVisitStatus": "二访情况",
"secondVisitStatusPlaceholder": "请输入二访情况",
"isClosed": "是否关单: 1-是, 0-否",
"isClosedPlaceholder": "请输入是否关单: 1-是, 0-否",
"staffId": "人员ID",
"staffIdPlaceholder": "请输入人员ID",
"resourceId": "资源ID",
"resourceIdPlaceholder": "请输入资源ID",
"addSixSpeed": "添加六一速",
"updateSixSpeed": "编辑六一速",
"sixSpeedDeleteTips": "确定要删除该数据吗?",
"startDate": "请选择开始时间",
"endDate": "请选择结束时间"
"purchasePower":"需求购买力",
"purchasePowerPlaceholder":"请输入需求购买力",
"conceptAwareness":"认知理念",
"conceptAwarenessPlaceholder":"请输入认知理念",
"preferredClassTime":"可选上课时间",
"preferredClassTimePlaceholder":"请输入可选上课时间",
"distance":"距离",
"distancePlaceholder":"请输入距离",
"communication":"沟通备注",
"communicationPlaceholder":"请输入沟通备注",
"promisedVisitTime":"承诺到访时间",
"promisedVisitTimePlaceholder":"请输入承诺到访时间",
"actualVisitTime":"实际到访时间",
"actualVisitTimePlaceholder":"请输入实际到访时间",
"callIntent":"电话后的意向程度",
"callIntentPlaceholder":"请输入电话后的意向程度",
"firstVisitStatus":"一访情况",
"firstVisitStatusPlaceholder":"请输入一访情况",
"secondVisitStatus":"二访情况",
"secondVisitStatusPlaceholder":"请输入二访情况",
"isClosed":"是否关单",
"isClosedPlaceholder":"请输入是否关单",
"addSixSpeed":"添加六一速",
"updateSixSpeed":"编辑六一速",
"sixSpeedDeleteTips":"确定要删除该数据吗?",
"startDate":"请选择开始时间",
"endDate":"请选择结束时间"
}

717
admin/src/app/views/customer_resources/components/customer-resources-edit.vue

@ -1,121 +1,134 @@
<template>
<el-dialog
v-model="showDialog"
:title="
formData.id ? t('updateCustomerResources') : t('addCustomerResources')
"
width="50%"
class="diy-dialog-wrap"
:destroy-on-close="true"
>
<el-form
:model="formData"
label-width="120px"
ref="formRef"
:rules="formRules"
class="page-form"
v-loading="loading"
>
<el-dialog v-model="showDialog" :title="formData.id ? t('updateCustomerResources') : t('addCustomerResources')"
width="50%" class="diy-dialog-wrap" :destroy-on-close="true">
<div class="tab-container">
<el-tabs v-model="activeTab" class="tab-pane-half">
<!-- Tab 1: 基本信息 -->
<el-tab-pane label="客户信息" name="base">
<el-form :model="formData" label-width="120px" ref="formRef" :rules="formRules" class="page-form"
v-loading="loading">
<el-form-item :label="t('source')" prop="source">
<el-select
class="input-width"
v-model="formData.source"
clearable
:placeholder="t('sourcePlaceholder')"
>
<el-select class="input-width" v-model="formData.source" clearable
:placeholder="t('sourcePlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in sourceList"
:key="index"
:label="item.name"
:value="item.value"
/>
<el-option v-for="(item, index) in sourceList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('sourceChannel')" v-if="formData.source == 1">
<el-select
class="input-width"
v-model="formData.source_channel"
clearable
:placeholder="t('sourceChannelPlaceholder')"
>
<el-select class="input-width" v-model="formData.source_channel" clearable
:placeholder="t('sourceChannelPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in source_channelList"
:key="index"
:label="item.name"
:value="item.value"
/>
<el-option v-for="(item, index) in source_channelList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('name')" prop="name">
<el-input
v-model="formData.name"
clearable
:placeholder="t('namePlaceholder')"
class="input-width"
/>
<el-input v-model="formData.name" clearable :placeholder="t('namePlaceholder')"
class="input-width" />
</el-form-item>
<el-form-item :label="t('age')" prop="age">
<el-input-number
v-model="formData.age"
clearable
:placeholder="t('agePlaceholder')"
class="input-width"
:min="3"
:max="80"
/>
<el-input-number v-model="formData.age" clearable :placeholder="t('agePlaceholder')"
class="input-width" :min="3" max="80" />
</el-form-item>
<el-form-item :label="t('gender')" prop="gender">
<el-select
class="input-width"
v-model="formData.gender"
clearable
:placeholder="t('genderPlaceholder')"
>
<el-select class="input-width" v-model="formData.gender" clearable
:placeholder="t('genderPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in genderList"
:key="index"
:label="item.name"
:value="item.value"
/>
<el-option v-for="(item, index) in genderList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('phoneNumber')" prop="phone_number">
<el-input
v-model="formData.phone_number"
clearable
:placeholder="t('phoneNumberPlaceholder')"
class="input-width"
/>
<el-input v-model="formData.phone_number" clearable
:placeholder="t('phoneNumberPlaceholder')" class="input-width" />
</el-form-item>
<el-form-item :label="t('demand')" prop="demand">
<el-input
v-model="formData.demand"
type="textarea"
rows="4"
clearable
:placeholder="t('demandPlaceholder')"
class="input-width"
/>
<el-input v-model="formData.demand" type="textarea" rows="4" clearable
:placeholder="t('demandPlaceholder')" class="input-width" />
</el-form-item>
<el-form-item :label="t('purchasingPower')" prop="purchasing_power">
<el-select
class="input-width"
v-model="formData.purchasing_power"
clearable
:placeholder="t('purchasingPowerPlaceholder')"
>
<el-select class="input-width" v-model="formData.purchasing_power" clearable
:placeholder="t('purchasingPowerPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option v-for="(item, index) in purchasing_powerList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('cognitiveIdea')" prop="cognitive_idea">
<el-select class="input-width" v-model="formData.cognitive_idea" clearable
:placeholder="t('cognitiveIdeaPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option v-for="(item, index) in cognitive_ideaList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('optionalClassTime')" prop="optional_class_time" class="input-width">
<el-date-picker class="flex-1 !flex" v-model="formData.optional_class_time" clearable
type="datetime" value-format="YYYY-MM-DD HH:mm:ss"
:placeholder="t('optionalClassTimePlaceholder')">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('distance')" prop="distance">
<el-input v-model="formData.distance" clearable :placeholder="t('distancePlaceholder')"
class="input-width" />
</el-form-item>
<el-form-item :label="t('decisionMaker')" prop="decision_maker">
<el-input v-model="formData.decision_maker" clearable
:placeholder="t('decisionMakerPlaceholder')" class="input-width" />
</el-form-item>
<el-form-item :label="t('initialIntent')" prop="initial_intent">
<el-select class="input-width" v-model="formData.initial_intent" clearable
:placeholder="t('initialIntentPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option v-for="(item, index) in initial_intentList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
<el-form-item :label="t('campus')" prop="campus">
<el-select class="input-width" v-model="formData.campus" clearable
:placeholder="t('campusPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option v-for="(item, index) in campusList" :key="index" :label="item['campus_name']"
:value="item['id']" />
</el-select>
</el-form-item>
<el-form-item :label="t('status')" prop="status">
<el-select class="input-width" v-model="formData.status" clearable
:placeholder="t('statusPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option v-for="(item, index) in statusList" :key="index" :label="item.name"
:value="item.value" />
</el-select>
</el-form-item>
</el-form>
</el-tab-pane>
<el-tab-pane label="客户六要素" name="sixElements">
<!-- 客户六要素表单内容 -->
<el-form :model="formData" label-width="120px">
<el-form-item label="需求购买力" prop="purchase_power">
<el-select class="input-width" v-model="formData.purchase_power" clearable placeholder="请选择需求购买力">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in purchasing_powerList"
v-for="(item, index) in purchase_powerList"
:key="index"
:label="item.name"
:value="item.value"
@ -123,16 +136,12 @@
</el-select>
</el-form-item>
<el-form-item :label="t('cognitiveIdea')" prop="cognitive_idea">
<el-select
class="input-width"
v-model="formData.cognitive_idea"
clearable
:placeholder="t('cognitiveIdeaPlaceholder')"
>
<el-form-item label="认知理念" prop="concept_awareness">
<el-select class="input-width" v-model="formData.concept_awareness" clearable placeholder="请选择认知理念">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in cognitive_ideaList"
v-for="(item, index) in concept_awarenessList"
:key="index"
:label="item.name"
:value="item.value"
@ -140,49 +149,55 @@
</el-select>
</el-form-item>
<el-form-item
:label="t('optionalClassTime')"
prop="optional_class_time"
class="input-width"
>
<el-form-item label="可选上课时间" prop="preferred_class_time" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.optional_class_time"
v-model="formData.preferred_class_time"
clearable
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
:placeholder="t('optionalClassTimePlaceholder')"
>
placeholder="可选上课时间">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('distance')" prop="distance">
<el-input
v-model="formData.distance"
clearable
:placeholder="t('distancePlaceholder')"
class="input-width"
/>
<el-form-item label="距离" prop="distance_tow">
<el-input v-model="formData.distance_tow" clearable placeholder="请填写距离" class="input-width" />
</el-form-item>
<el-form-item :label="t('decisionMaker')" prop="decision_maker">
<el-input
v-model="formData.decision_maker"
<el-form-item label="沟通备注" prop="communication">
<el-input v-model="formData.communication" type="textarea" rows="4" clearable placeholder="请填写沟通备注" class="input-width"/>
</el-form-item>
<el-form-item label="承诺到访时间" prop="promised_visit_time" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.promised_visit_time"
clearable
:placeholder="t('decisionMakerPlaceholder')"
class="input-width"
/>
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
placeholder="请选择承诺到访时间">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('initialIntent')" prop="initial_intent">
<el-select
class="input-width"
v-model="formData.initial_intent"
<el-form-item label="实际到访时间" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.actual_visit_time"
clearable
:placeholder="t('initialIntentPlaceholder')"
>
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
placeholder="请填写实际到访时间">
</el-date-picker>
</el-form-item>
<el-form-item label="电话后的意向程度" prop="call_intent">
<el-select class="input-width" v-model="formData.call_intent" clearable placeholder="请选择电话后的意向程度">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in initial_intentList"
v-for="(item, index) in call_intentList"
:key="index"
:label="item.name"
:value="item.value"
@ -190,74 +205,61 @@
</el-select>
</el-form-item>
<el-form-item :label="t('campus')" prop="campus">
<el-select
class="input-width"
v-model="formData.campus"
clearable
:placeholder="t('campusPlaceholder')"
>
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in campusList"
:key="index"
:label="item['campus_name']"
:value="item['campus_name']"
/>
</el-select>
<el-form-item label="一访情况" >
<el-input v-model="formData.first_visit_status" type="textarea" rows="4" clearable placeholder="请填写一访情况" class="input-width"/>
</el-form-item>
<el-form-item :label="t('status')" prop="status">
<el-select
class="input-width"
v-model="formData.status"
clearable
:placeholder="t('statusPlaceholder')"
>
<el-form-item label="二访情况" >
<el-input v-model="formData.second_visit_status" type="textarea" rows="4" clearable placeholder="请填写二访情况" class="input-width"/>
</el-form-item>
<el-form-item label="是否关单" prop="is_closed">
<el-select class="input-width" v-model="formData.is_closed" clearable placeholder="请选择是否关单">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in statusList"
v-for="(item, index) in is_closedList"
:key="index"
:label="item.name"
:value="item.value"
:value="Number(item.value)"
/>
</el-select>
</el-form-item>
</el-form>
</el-tab-pane>
</el-tabs>
</div>
<!-- Footer -->
<template #footer>
<span class="dialog-footer">
<el-button @click="showDialog = false">{{ t('cancel') }}</el-button>
<el-button
type="primary"
:loading="loading"
@click="confirm(formRef)"
>{{ t('confirm') }}</el-button
>
<el-button type="primary" :loading="loading" @click="confirm(formRef)">
{{ t('confirm') }}
</el-button>
</span>
</template>
</el-dialog>
</template>
<script lang="ts" setup>
import { ref, reactive, computed, watch } from 'vue'
import { useDictionary } from '@/app/api/dict'
import { t } from '@/lang'
import type { FormInstance } from 'element-plus'
import {
addCustomerResources,
editCustomerResources,
getCustomerResourcesInfo,
getWithCampusList,
} from '@/app/api/customer_resources'
let showDialog = ref(false)
const loading = ref(false)
/**
import { ref, reactive, computed, watch } from 'vue'
import { useDictionary } from '@/app/api/dict'
import { t } from '@/lang'
import type { FormInstance } from 'element-plus'
import { addCustomerResources, editCustomerResources, getCustomerResourcesInfo, getWithPersonnelList, getWithCampusList } from '@/app/api/customer_resources'
import { addSixSpeed, editSixSpeed, getSixSpeedInfo, getWithCustomerResourcesList } from '@/app/api/six_speed'
const activeTab = ref('base')
let showDialog = ref(false)
const loading = ref(false)
/**
* 表单数据
*/
const initialFormData = {
const initialFormData = {
id: '',
source: '',
source_channel: '',
@ -274,103 +276,124 @@ const initialFormData = {
initial_intent: '',
campus: '',
status: '',
}
const formData: Record<string, any> = reactive({ ...initialFormData })
const formRef = ref<FormInstance>()
purchase_power: '',
concept_awareness: '',
preferred_class_time: '',
distance_tow: '',
communication: '',
promised_visit_time: '',
actual_visit_time: '',
call_intent: '',
first_visit_status: '',
second_visit_status: '',
is_closed: '',
}
const formData : Record<string, any> = reactive({ ...initialFormData })
const formRef = ref<FormInstance>()
//
const formRules = computed(() => {
//
const formRules = computed(() => {
return {
source: [
{ required: true, message: t('sourcePlaceholder'), trigger: 'blur' },
],
]
,
source_channel: [
{
required: true,
message: t('sourceChannelPlaceholder'),
trigger: 'blur',
},
],
name: [{ required: true, message: t('namePlaceholder'), trigger: 'blur' }],
{ required: true, message: t('sourceChannelPlaceholder'), trigger: 'blur' },
]
,
name: [
{ required: true, message: t('namePlaceholder'), trigger: 'blur' },
]
,
age: [
{ required: true, message: t('agePlaceholder'), trigger: 'blur' },
{
validator: (rule: any, value: number, callback: any) => {
validator: (rule : any, value : number, callback : any) => {
if (value === undefined || value === null || value === '') {
callback()
callback();
} else if (value < 3 || value > 80) {
callback(new Error(t('generateBetween')))
callback(new Error(t('generateBetween')));
} else {
callback()
callback();
}
},
trigger: 'blur',
},
],
trigger: 'blur'
}
]
,
gender: [
{ required: true, message: t('genderPlaceholder'), trigger: 'blur' },
],
]
,
phone_number: [
{ required: true, message: t('phoneNumberPlaceholder'), trigger: 'blur' },
],
]
,
demand: [
{ required: true, message: t('demandPlaceholder'), trigger: 'blur' },
],
]
,
purchasing_power: [
{
required: true,
message: t('purchasingPowerPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('purchasingPowerPlaceholder'), trigger: 'blur' },
]
,
cognitive_idea: [
{
required: true,
message: t('cognitiveIdeaPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('cognitiveIdeaPlaceholder'), trigger: 'blur' },
]
,
optional_class_time: [
{
required: true,
message: t('optionalClassTimePlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('optionalClassTimePlaceholder'), trigger: 'blur' },
]
,
distance: [
{ required: true, message: t('distancePlaceholder'), trigger: 'blur' },
],
]
,
decision_maker: [
{
required: true,
message: t('decisionMakerPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('decisionMakerPlaceholder'), trigger: 'blur' },
]
,
initial_intent: [
{
required: true,
message: t('initialIntentPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('initialIntentPlaceholder'), trigger: 'blur' },
]
,
campus: [
{ required: true, message: t('campusPlaceholder'), trigger: 'blur' },
],
]
,
status: [
{ required: true, message: t('statusPlaceholder'), trigger: 'blur' },
],
]
,
}
})
})
const emit = defineEmits(['complete'])
const emit = defineEmits(['complete'])
/**
/**
* 确认
* @param formEl
*/
const confirm = async (formEl: FormInstance | undefined) => {
const confirm = async (formEl : FormInstance | undefined) => {
console.log(123123)
if (loading.value || !formEl) return
let save = formData.id ? editCustomerResources : addCustomerResources
@ -380,174 +403,172 @@ const confirm = async (formEl: FormInstance | undefined) => {
let data = formData
save(data)
.then((res) => {
save(data).then(res => {
loading.value = false
showDialog.value = false
emit('complete')
})
.catch((err) => {
}).catch(err => {
loading.value = false
})
}
})
}
}
//
let sourceList = ref([])
const sourceDictList = async () => {
//
let sourceList = ref([])
const sourceDictList = async () => {
sourceList.value = await (await useDictionary('source')).data.dictionary
}
sourceDictList()
watch(
() => sourceList.value,
() => {
formData.source = sourceList.value[0].value
}
)
let source_channelList = ref([])
const source_channelDictList = async () => {
source_channelList.value = await (
await useDictionary('SourceChannel')
).data.dictionary
}
source_channelDictList()
watch(
() => source_channelList.value,
() => {
formData.source_channel = source_channelList.value[0].value
}
)
let genderList = ref([])
const genderDictList = async () => {
}
sourceDictList();
watch(() => sourceList.value, () => { formData.source = sourceList.value[0].value })
let source_channelList = ref([])
const source_channelDictList = async () => {
source_channelList.value = await (await useDictionary('SourceChannel')).data.dictionary
}
source_channelDictList();
watch(() => source_channelList.value, () => { formData.source_channel = source_channelList.value[0].value })
let genderList = ref([])
const genderDictList = async () => {
genderList.value = await (await useDictionary('zy_sex')).data.dictionary
}
genderDictList()
watch(
() => genderList.value,
() => {
formData.gender = genderList.value[0].value
}
)
let purchasing_powerList = ref([])
const purchasing_powerDictList = async () => {
purchasing_powerList.value = await (
await useDictionary('customer_purchasing_power')
).data.dictionary
}
purchasing_powerDictList()
watch(
() => purchasing_powerList.value,
() => {
formData.purchasing_power = purchasing_powerList.value[0].value
}
)
let cognitive_ideaList = ref([])
const cognitive_ideaDictList = async () => {
cognitive_ideaList.value = await (
await useDictionary('cognitive_concept')
).data.dictionary
}
cognitive_ideaDictList()
watch(
() => cognitive_ideaList.value,
() => {
formData.cognitive_idea = cognitive_ideaList.value[0].value
}
)
let initial_intentList = ref([])
const initial_intentDictList = async () => {
initial_intentList.value = await (
await useDictionary('preliminarycustomerintention')
).data.dictionary
}
initial_intentDictList()
watch(
() => initial_intentList.value,
() => {
formData.initial_intent = initial_intentList.value[0].value
}
)
let statusList = ref([])
const statusDictList = async () => {
}
genderDictList();
watch(() => genderList.value, () => { formData.gender = genderList.value[0].value })
let purchasing_powerList = ref([])
const purchasing_powerDictList = async () => {
purchasing_powerList.value = await (await useDictionary('customer_purchasing_power')).data.dictionary
}
purchasing_powerDictList();
watch(() => purchasing_powerList.value, () => { formData.purchasing_power = purchasing_powerList.value[0].value })
let cognitive_ideaList = ref([])
const cognitive_ideaDictList = async () => {
cognitive_ideaList.value = await (await useDictionary('cognitive_concept')).data.dictionary
}
cognitive_ideaDictList();
watch(() => cognitive_ideaList.value, () => { formData.cognitive_idea = cognitive_ideaList.value[0].value })
let initial_intentList = ref([])
const initial_intentDictList = async () => {
initial_intentList.value = await (await useDictionary('preliminarycustomerintention')).data.dictionary
}
initial_intentDictList();
watch(() => initial_intentList.value, () => { formData.initial_intent = initial_intentList.value[0].value })
let statusList = ref([])
const statusDictList = async () => {
statusList.value = await (await useDictionary('kh_status')).data.dictionary
}
statusDictList()
watch(
() => statusList.value,
() => {
formData.status = statusList.value[0].value
}
)
statusDictList();
watch(() => statusList.value, () => { formData.status = statusList.value[0].value })
const campusList = ref([] as any[])
const setCampusList = async () => {
const consultantList = ref([] as any[])
const setConsultantList = async () => {
consultantList.value = await (await getWithPersonnelList({})).data
}
setConsultantList()
const campusList = ref([] as any[])
const setCampusList = async () => {
campusList.value = await (await getWithCampusList({})).data
}
setCampusList()
const setFormData = async (row: any = null) => {
}
setCampusList()
//
let purchase_powerList = ref([])
const purchase_powerDictList = async () => {
purchase_powerList.value = await (await useDictionary('customer_purchasing_power')).data.dictionary
}
purchase_powerDictList();
watch(() => purchase_powerList.value, () => { formData.purchase_power = purchase_powerList.value[0].value })
let concept_awarenessList = ref([])
const concept_awarenessDictList = async () => {
concept_awarenessList.value = await (await useDictionary('cognitive_concept')).data.dictionary
}
concept_awarenessDictList();
watch(() => concept_awarenessList.value, () => { formData.concept_awareness = concept_awarenessList.value[0].value })
let call_intentList = ref([])
const call_intentDictList = async () => {
call_intentList.value = await (await useDictionary('preliminarycustomerintention')).data.dictionary
}
call_intentDictList();
watch(() => call_intentList.value, () => { formData.call_intent = call_intentList.value[0].value })
let is_closedList = ref([])
const is_closedDictList = async () => {
is_closedList.value = await (await useDictionary('global_true_or_false')).data.dictionary
}
is_closedDictList();
watch(() => is_closedList.value, () => { formData.is_closed = is_closedList.value[0].value })
const staffIdList = ref([] as any[])
const setStaffIdList = async () => {
staffIdList.value = await (await getWithPersonnelList({})).data
}
setStaffIdList()
const resourceIdList = ref([] as any[])
const setResourceIdList = async () => {
resourceIdList.value = await (await getWithCustomerResourcesList({})).data
}
setResourceIdList()
const setFormData = async (row : any = null) => {
Object.assign(formData, initialFormData)
loading.value = true
if (row) {
const data = await (await getCustomerResourcesInfo(row.id)).data
if (data)
Object.keys(formData).forEach((key: string) => {
if (data) Object.keys(formData).forEach((key : string) => {
if (data[key] != undefined) formData[key] = data[key]
})
}
loading.value = false
}
}
//
const mobileVerify = (rule: any, value: any, callback: any) => {
//
const mobileVerify = (rule : any, value : any, callback : any) => {
if (value && !/^1[3-9]\d{9}$/.test(value)) {
callback(new Error(t('generateMobile')))
} else {
callback()
}
}
}
//
const idCardVerify = (rule: any, value: any, callback: any) => {
if (
value &&
!/^[1-9]\d{5}[1-9]\d{3}((0\d)|(1[0-2]))(([0|1|2]\d)|3[0-1])\d{3}([0-9]|X)$/.test(
value
)
) {
//
const idCardVerify = (rule : any, value : any, callback : any) => {
if (value && !/^[1-9]\d{5}[1-9]\d{3}((0\d)|(1[0-2]))(([0|1|2]\d)|3[0-1])\d{3}([0-9]|X)$/.test(value)) {
callback(new Error(t('generateIdCard')))
} else {
callback()
}
}
}
//
const emailVerify = (rule: any, value: any, callback: any) => {
//
const emailVerify = (rule : any, value : any, callback : any) => {
if (value && !/\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*/.test(value)) {
callback(new Error(t('generateEmail')))
} else {
callback()
}
}
}
//
const numberVerify = (rule: any, value: any, callback: any) => {
//
const numberVerify = (rule : any, value : any, callback : any) => {
if (!Number.isInteger(value)) {
callback(new Error(t('generateNumber')))
} else {
callback()
}
}
}
defineExpose({
defineExpose({
showDialog,
setFormData,
})
setFormData
})
</script>
<style lang="scss" scoped></style>
<style lang="scss">
.diy-dialog-wrap .el-form-item__label {
.diy-dialog-wrap .el-form-item__label {
height: auto !important;
}
}
</style>

248
admin/src/app/views/customer_resources/customer_resources.vue

@ -1,63 +1,35 @@
<template>
<div class="main-container">
<el-card class="box-card !border-none" shadow="never">
<div class="flex justify-between items-center">
<span class="text-lg">{{ pageName }}</span>
<span class="text-lg">{{pageName}}</span>
<el-button type="primary" @click="addEvent">
{{ t('addCustomerResources') }}
</el-button>
</div>
<el-card
class="box-card !border-none my-[10px] table-search-wrap"
shadow="never"
>
<el-form
:inline="true"
:model="customerResourcesTable.searchParam"
ref="searchFormRef"
>
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="customerResourcesTable.searchParam" ref="searchFormRef">
<el-form-item :label="t('name')" prop="name">
<el-input
v-model="customerResourcesTable.searchParam.name"
:placeholder="t('namePlaceholder')"
/>
<el-input v-model="customerResourcesTable.searchParam.name" :placeholder="t('namePlaceholder')" />
</el-form-item>
<el-form-item :label="t('phoneNumber')" prop="phone_number">
<el-input
v-model="customerResourcesTable.searchParam.phone_number"
:placeholder="t('phoneNumberPlaceholder')"
/>
<el-input v-model="customerResourcesTable.searchParam.phone_number" :placeholder="t('phoneNumberPlaceholder')" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="loadCustomerResourcesList()">{{
t('search')
}}</el-button>
<el-button @click="resetForm(searchFormRef)">{{
t('reset')
}}</el-button>
<el-button type="primary" @click="loadCustomerResourcesList()">{{ t('search') }}</el-button>
<el-button @click="resetForm(searchFormRef)">{{ t('reset') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<div class="mt-[10px]">
<el-table
:data="customerResourcesTable.data"
size="large"
v-loading="customerResourcesTable.loading"
>
<el-table :data="customerResourcesTable.data" size="large" v-loading="customerResourcesTable.loading">
<template #empty>
<span>{{
!customerResourcesTable.loading ? t('emptyData') : ''
}}</span>
<span>{{ !customerResourcesTable.loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column
:label="t('source')"
min-width="180"
align="center"
:show-overflow-tooltip="true"
>
<el-table-column :label="t('source')" min-width="180" align="center" :show-overflow-tooltip="true">
<template #default="{ row }">
<div v-for="(item, index) in sourceList">
<div v-if="item.value == row.source">{{ item.name }}</div>
@ -65,33 +37,13 @@
</template>
</el-table-column>
<el-table-column
prop="consultant"
:label="t('consultant')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="consultant_name" :label="t('consultant')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="name"
:label="t('name')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="name" :label="t('name')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="age"
:label="t('age')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="age" :label="t('age')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
:label="t('gender')"
min-width="180"
align="center"
:show-overflow-tooltip="true"
>
<el-table-column :label="t('gender')" min-width="180" align="center" :show-overflow-tooltip="true">
<template #default="{ row }">
<div v-for="(item, index) in genderList">
<div v-if="item.value == row.gender">{{ item.name }}</div>
@ -99,51 +51,26 @@
</template>
</el-table-column>
<el-table-column
prop="phone_number"
:label="t('phoneNumber')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="phone_number" :label="t('phoneNumber')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="decision_maker"
:label="t('decisionMaker')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="decision_maker" :label="t('decisionMaker')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
:label="t('operation')"
fixed="right"
min-width="120"
>
<el-table-column :label="t('operation')" fixed="right" min-width="120">
<template #default="{ row }">
<el-button type="primary" link @click="editEvent(row)">{{
t('edit')
}}</el-button>
<el-button type="primary" link @click="deleteEvent(row.id)">{{
t('delete')
}}</el-button>
<el-button type="primary" link @click="editEvent(row)">{{ t('edit') }}</el-button>
<el-button type="primary" link @click="deleteEvent(row.id)">{{ t('delete') }}</el-button>
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination
v-model:current-page="customerResourcesTable.page"
v-model:page-size="customerResourcesTable.limit"
layout="total, sizes, prev, pager, next, jumper"
:total="customerResourcesTable.total"
@size-change="loadCustomerResourcesList()"
@current-change="loadCustomerResourcesList"
/>
<el-pagination v-model:current-page="customerResourcesTable.page" v-model:page-size="customerResourcesTable.limit"
layout="total, sizes, prev, pager, next, jumper" :total="customerResourcesTable.total"
@size-change="loadCustomerResourcesList()" @current-change="loadCustomerResourcesList" />
</div>
</div>
<edit
ref="editCustomerResourcesDialog"
@complete="loadCustomerResourcesList"
/>
<edit ref="editCustomerResourcesDialog" @complete="loadCustomerResourcesList" />
</el-card>
</div>
</template>
@ -152,17 +79,13 @@
import { reactive, ref, watch } from 'vue'
import { t } from '@/lang'
import { useDictionary } from '@/app/api/dict'
import {
getCustomerResourcesList,
deleteCustomerResources,
getWithCampusList,
} from '@/app/api/customer_resources'
import { getCustomerResourcesList, deleteCustomerResources, getWithPersonnelList, getWithCampusList } from '@/app/api/customer_resources'
import { img } from '@/utils/common'
import { ElMessageBox, FormInstance } from 'element-plus'
import { ElMessageBox,FormInstance } from 'element-plus'
import Edit from '@/app/views/customer_resources/components/customer-resources-edit.vue'
import { useRoute } from 'vue-router'
const route = useRoute()
const pageName = route.meta.title
const pageName = route.meta.title;
let customerResourcesTable = reactive({
page: 1,
@ -170,10 +93,10 @@ let customerResourcesTable = reactive({
total: 0,
loading: true,
data: [],
searchParam: {
name: '',
phone_number: '',
},
searchParam:{
"name":"",
"phone_number":""
}
})
const searchFormRef = ref<FormInstance>()
@ -182,49 +105,41 @@ const searchFormRef = ref<FormInstance>()
const selectData = ref<any[]>([])
//
const sourceList = ref([] as any[])
const sourceDictList = async () => {
const sourceList = ref([] as any[])
const sourceDictList = async () => {
sourceList.value = await (await useDictionary('source')).data.dictionary
}
sourceDictList()
const source_channelList = ref([] as any[])
const source_channelDictList = async () => {
source_channelList.value = await (
await useDictionary('SourceChannel')
).data.dictionary
}
source_channelDictList()
const genderList = ref([] as any[])
const genderDictList = async () => {
}
sourceDictList();
const source_channelList = ref([] as any[])
const source_channelDictList = async () => {
source_channelList.value = await (await useDictionary('SourceChannel')).data.dictionary
}
source_channelDictList();
const genderList = ref([] as any[])
const genderDictList = async () => {
genderList.value = await (await useDictionary('zy_sex')).data.dictionary
}
genderDictList()
const purchasing_powerList = ref([] as any[])
const purchasing_powerDictList = async () => {
purchasing_powerList.value = await (
await useDictionary('customer_purchasing_power')
).data.dictionary
}
purchasing_powerDictList()
const cognitive_ideaList = ref([] as any[])
const cognitive_ideaDictList = async () => {
cognitive_ideaList.value = await (
await useDictionary('cognitive_concept')
).data.dictionary
}
cognitive_ideaDictList()
const initial_intentList = ref([] as any[])
const initial_intentDictList = async () => {
initial_intentList.value = await (
await useDictionary('preliminarycustomerintention')
).data.dictionary
}
initial_intentDictList()
const statusList = ref([] as any[])
const statusDictList = async () => {
}
genderDictList();
const purchasing_powerList = ref([] as any[])
const purchasing_powerDictList = async () => {
purchasing_powerList.value = await (await useDictionary('customer_purchasing_power')).data.dictionary
}
purchasing_powerDictList();
const cognitive_ideaList = ref([] as any[])
const cognitive_ideaDictList = async () => {
cognitive_ideaList.value = await (await useDictionary('cognitive_concept')).data.dictionary
}
cognitive_ideaDictList();
const initial_intentList = ref([] as any[])
const initial_intentDictList = async () => {
initial_intentList.value = await (await useDictionary('preliminarycustomerintention')).data.dictionary
}
initial_intentDictList();
const statusList = ref([] as any[])
const statusDictList = async () => {
statusList.value = await (await useDictionary('kh_status')).data.dictionary
}
statusDictList()
}
statusDictList();
/**
* 获取客户资源列表
@ -236,14 +151,12 @@ const loadCustomerResourcesList = (page: number = 1) => {
getCustomerResourcesList({
page: customerResourcesTable.page,
limit: customerResourcesTable.limit,
...customerResourcesTable.searchParam,
})
.then((res) => {
...customerResourcesTable.searchParam
}).then(res => {
customerResourcesTable.loading = false
customerResourcesTable.data = res.data.data
customerResourcesTable.total = res.data.total
})
.catch(() => {
}).catch(() => {
customerResourcesTable.loading = false
})
}
@ -272,24 +185,31 @@ const editEvent = (data: any) => {
* 删除客户资源
*/
const deleteEvent = (id: number) => {
ElMessageBox.confirm(t('customerResourcesDeleteTips'), t('warning'), {
ElMessageBox.confirm(t('customerResourcesDeleteTips'), t('warning'),
{
confirmButtonText: t('confirm'),
cancelButtonText: t('cancel'),
type: 'warning',
}).then(() => {
deleteCustomerResources(id)
.then(() => {
}
).then(() => {
deleteCustomerResources(id).then(() => {
loadCustomerResourcesList()
}).catch(() => {
})
.catch(() => {})
})
}
const campusList = ref([])
const setCampusList = async () => {
const consultantList = ref([])
const setConsultantList = async () => {
consultantList.value = await (await getWithPersonnelList({})).data
}
setConsultantList()
const campusList = ref([])
const setCampusList = async () => {
campusList.value = await (await getWithCampusList({})).data
}
setCampusList()
}
setCampusList()
const resetForm = (formEl: FormInstance | undefined) => {
if (!formEl) return
@ -307,5 +227,5 @@ const resetForm = (formEl: FormInstance | undefined) => {
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
}
}
</style>

331
admin/src/app/views/six_speed/components/six-speed-edit.vue

@ -1,149 +1,105 @@
<template>
<el-dialog
v-model="showDialog"
:title="formData.id ? t('updateSixSpeed') : t('addSixSpeed')"
width="50%"
class="diy-dialog-wrap"
:destroy-on-close="true"
>
<el-form
:model="formData"
label-width="120px"
ref="formRef"
:rules="formRules"
class="page-form"
v-loading="loading"
>
<el-dialog v-model="showDialog" :title="formData.id ? t('updateSixSpeed') : t('addSixSpeed')" width="50%" class="diy-dialog-wrap" :destroy-on-close="true">
<el-form :model="formData" label-width="120px" ref="formRef" :rules="formRules" class="page-form" v-loading="loading">
<el-form-item :label="t('purchasePower')" prop="purchase_power">
<el-input
v-model="formData.purchase_power"
clearable
:placeholder="t('purchasePowerPlaceholder')"
class="input-width"
<el-select class="input-width" v-model="formData.purchase_power" clearable :placeholder="t('purchasePowerPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in purchase_powerList"
:key="index"
:label="item.name"
:value="item.value"
/>
</el-select>
</el-form-item>
<el-form-item :label="t('conceptAwareness')" prop="concept_awareness">
<el-input
v-model="formData.concept_awareness"
clearable
:placeholder="t('conceptAwarenessPlaceholder')"
class="input-width"
<el-select class="input-width" v-model="formData.concept_awareness" clearable :placeholder="t('conceptAwarenessPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in concept_awarenessList"
:key="index"
:label="item.name"
:value="item.value"
/>
</el-select>
</el-form-item>
<el-form-item
:label="t('preferredClassTime')"
prop="preferred_class_time"
>
<el-input
<el-form-item :label="t('preferredClassTime')" prop="preferred_class_time" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.preferred_class_time"
clearable
:placeholder="t('preferredClassTimePlaceholder')"
class="input-width"
/>
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
:placeholder="t('preferredClassTimePlaceholder')">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('distance')" prop="distance">
<el-input
v-model="formData.distance"
clearable
:placeholder="t('distancePlaceholder')"
class="input-width"
/>
<el-input v-model="formData.distance" clearable :placeholder="t('distancePlaceholder')" class="input-width" />
</el-form-item>
<el-form-item :label="t('communication')" prop="communication">
<el-input
v-model="formData.communication"
clearable
:placeholder="t('communicationPlaceholder')"
class="input-width"
/>
<el-input v-model="formData.communication" type="textarea" rows="4" clearable :placeholder="t('communicationPlaceholder')" class="input-width"/>
</el-form-item>
<el-form-item :label="t('promisedVisitTime')" prop="promised_visit_time">
<el-input
<el-form-item :label="t('promisedVisitTime')" prop="promised_visit_time" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.promised_visit_time"
clearable
:placeholder="t('promisedVisitTimePlaceholder')"
class="input-width"
/>
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
:placeholder="t('promisedVisitTimePlaceholder')">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('actualVisitTime')">
<el-input
<el-form-item :label="t('actualVisitTime')" class="input-width">
<el-date-picker
class="flex-1 !flex"
v-model="formData.actual_visit_time"
clearable
:placeholder="t('actualVisitTimePlaceholder')"
class="input-width"
/>
type="datetime"
value-format="YYYY-MM-DD HH:mm:ss"
:placeholder="t('actualVisitTimePlaceholder')">
</el-date-picker>
</el-form-item>
<el-form-item :label="t('callIntent')" prop="call_intent">
<el-input
v-model="formData.call_intent"
clearable
:placeholder="t('callIntentPlaceholder')"
class="input-width"
<el-select class="input-width" v-model="formData.call_intent" clearable :placeholder="t('callIntentPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in call_intentList"
:key="index"
:label="item.name"
:value="item.value"
/>
</el-select>
</el-form-item>
<el-form-item :label="t('firstVisitStatus')">
<el-input
v-model="formData.first_visit_status"
clearable
:placeholder="t('firstVisitStatusPlaceholder')"
class="input-width"
/>
<el-form-item :label="t('firstVisitStatus')" >
<el-input v-model="formData.first_visit_status" type="textarea" rows="4" clearable :placeholder="t('firstVisitStatusPlaceholder')" class="input-width"/>
</el-form-item>
<el-form-item :label="t('secondVisitStatus')">
<el-input
v-model="formData.second_visit_status"
clearable
:placeholder="t('secondVisitStatusPlaceholder')"
class="input-width"
/>
<el-form-item :label="t('secondVisitStatus')" >
<el-input v-model="formData.second_visit_status" type="textarea" rows="4" clearable :placeholder="t('secondVisitStatusPlaceholder')" class="input-width"/>
</el-form-item>
<el-form-item :label="t('isClosed')" prop="is_closed">
<el-input
v-model="formData.is_closed"
clearable
:placeholder="t('isClosedPlaceholder')"
class="input-width"
<el-select class="input-width" v-model="formData.is_closed" clearable :placeholder="t('isClosedPlaceholder')">
<el-option label="请选择" value=""></el-option>
<el-option
v-for="(item, index) in is_closedList"
:key="index"
:label="item.name"
:value="item.value"
/>
</el-select>
</el-form-item>
<el-form-item :label="t('staffId')" prop="staff_id">
<el-input
v-model="formData.staff_id"
clearable
:placeholder="t('staffIdPlaceholder')"
class="input-width"
/>
</el-form-item>
<el-form-item :label="t('resourceId')" prop="resource_id">
<el-input
v-model="formData.resource_id"
clearable
:placeholder="t('resourceIdPlaceholder')"
class="input-width"
/>
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="showDialog = false">{{ t('cancel') }}</el-button>
<el-button
type="primary"
:loading="loading"
@click="confirm(formRef)"
>{{ t('confirm') }}</el-button
>
<el-button type="primary" :loading="loading" @click="confirm(formRef)">{{
t('confirm')
}}</el-button>
</span>
</template>
</el-dialog>
@ -154,7 +110,7 @@ import { ref, reactive, computed, watch } from 'vue'
import { useDictionary } from '@/app/api/dict'
import { t } from '@/lang'
import type { FormInstance } from 'element-plus'
import { addSixSpeed, editSixSpeed, getSixSpeedInfo } from '@/app/api/six_speed'
import { addSixSpeed, editSixSpeed, getSixSpeedInfo, getWithPersonnelList, getWithCustomerResourcesList } from '@/app/api/six_speed'
let showDialog = ref(false)
const loading = ref(false)
@ -175,8 +131,6 @@ const initialFormData = {
first_visit_status: '',
second_visit_status: '',
is_closed: '',
staff_id: '',
resource_id: '',
}
const formData: Record<string, any> = reactive({ ...initialFormData })
@ -186,76 +140,60 @@ const formRef = ref<FormInstance>()
const formRules = computed(() => {
return {
purchase_power: [
{
required: true,
message: t('purchasePowerPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('purchasePowerPlaceholder'), trigger: 'blur' },
]
,
concept_awareness: [
{
required: true,
message: t('conceptAwarenessPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('conceptAwarenessPlaceholder'), trigger: 'blur' },
]
,
preferred_class_time: [
{
required: true,
message: t('preferredClassTimePlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('preferredClassTimePlaceholder'), trigger: 'blur' },
]
,
distance: [
{ required: true, message: t('distancePlaceholder'), trigger: 'blur' },
],
]
,
communication: [
{
required: true,
message: t('communicationPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('communicationPlaceholder'), trigger: 'blur' },
]
,
promised_visit_time: [
{
required: true,
message: t('promisedVisitTimePlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('promisedVisitTimePlaceholder'), trigger: 'blur' },
]
,
actual_visit_time: [
{
required: true,
message: t('actualVisitTimePlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('actualVisitTimePlaceholder'), trigger: 'blur' },
]
,
call_intent: [
{ required: true, message: t('callIntentPlaceholder'), trigger: 'blur' },
],
]
,
first_visit_status: [
{
required: true,
message: t('firstVisitStatusPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('firstVisitStatusPlaceholder'), trigger: 'blur' },
]
,
second_visit_status: [
{
required: true,
message: t('secondVisitStatusPlaceholder'),
trigger: 'blur',
},
],
{ required: true, message: t('secondVisitStatusPlaceholder'), trigger: 'blur' },
]
,
is_closed: [
{ required: true, message: t('isClosedPlaceholder'), trigger: 'blur' },
],
staff_id: [
{ required: true, message: t('staffIdPlaceholder'), trigger: 'blur' },
],
resource_id: [
{ required: true, message: t('resourceIdPlaceholder'), trigger: 'blur' },
],
]
,
}
})
@ -275,13 +213,11 @@ const confirm = async (formEl: FormInstance | undefined) => {
let data = formData
save(data)
.then((res) => {
save(data).then(res => {
loading.value = false
showDialog.value = false
emit('complete')
})
.catch((err) => {
}).catch(err => {
loading.value = false
})
}
@ -289,14 +225,48 @@ const confirm = async (formEl: FormInstance | undefined) => {
}
//
let purchase_powerList = ref([])
const purchase_powerDictList = async () => {
purchase_powerList.value = await (await useDictionary('customer_purchasing_power')).data.dictionary
}
purchase_powerDictList();
watch(() => purchase_powerList.value, () => { formData.purchase_power = purchase_powerList.value[0].value })
let concept_awarenessList = ref([])
const concept_awarenessDictList = async () => {
concept_awarenessList.value = await (await useDictionary('cognitive_concept')).data.dictionary
}
concept_awarenessDictList();
watch(() => concept_awarenessList.value, () => { formData.concept_awareness = concept_awarenessList.value[0].value })
let call_intentList = ref([])
const call_intentDictList = async () => {
call_intentList.value = await (await useDictionary('preliminarycustomerintention')).data.dictionary
}
call_intentDictList();
watch(() => call_intentList.value, () => { formData.call_intent = call_intentList.value[0].value })
let is_closedList = ref([])
const is_closedDictList = async () => {
is_closedList.value = await (await useDictionary('global_true_or_false')).data.dictionary
}
is_closedDictList();
watch(() => is_closedList.value, () => { formData.is_closed = is_closedList.value[0].value })
const staffIdList = ref([] as any[])
const setStaffIdList = async () => {
staffIdList.value = await (await getWithPersonnelList({})).data
}
setStaffIdList()
const resourceIdList = ref([] as any[])
const setResourceIdList = async () => {
resourceIdList.value = await (await getWithCustomerResourcesList({})).data
}
setResourceIdList()
const setFormData = async (row: any = null) => {
Object.assign(formData, initialFormData)
loading.value = true
if (row) {
if(row){
const data = await (await getSixSpeedInfo(row.id)).data
if (data)
Object.keys(formData).forEach((key: string) => {
if (data) Object.keys(formData).forEach((key: string) => {
if (data[key] != undefined) formData[key] = data[key]
})
}
@ -314,12 +284,7 @@ const mobileVerify = (rule: any, value: any, callback: any) => {
//
const idCardVerify = (rule: any, value: any, callback: any) => {
if (
value &&
!/^[1-9]\d{5}[1-9]\d{3}((0\d)|(1[0-2]))(([0|1|2]\d)|3[0-1])\d{3}([0-9]|X)$/.test(
value
)
) {
if (value && !/^[1-9]\d{5}[1-9]\d{3}((0\d)|(1[0-2]))(([0|1|2]\d)|3[0-1])\d{3}([0-9]|X)$/.test(value)) {
callback(new Error(t('generateIdCard')))
} else {
callback()
@ -346,13 +311,13 @@ const numberVerify = (rule: any, value: any, callback: any) => {
defineExpose({
showDialog,
setFormData,
setFormData
})
</script>
<style lang="scss" scoped></style>
<style lang="scss">
.diy-dialog-wrap .el-form-item__label {
.diy-dialog-wrap .el-form-item__label{
height: auto !important;
}
</style>

324
admin/src/app/views/six_speed/six_speed.vue

@ -1,248 +1,71 @@
<template>
<div class="main-container">
<el-card class="box-card !border-none" shadow="never">
<div class="flex justify-between items-center">
<span class="text-lg">{{ pageName }}</span>
<span class="text-lg">{{pageName}}</span>
<el-button type="primary" @click="addEvent">
{{ t('addSixSpeed') }}
</el-button>
</div>
<el-card
class="box-card !border-none my-[10px] table-search-wrap"
shadow="never"
>
<el-form
:inline="true"
:model="sixSpeedTable.searchParam"
ref="searchFormRef"
>
<el-form-item :label="t('purchasePower')" prop="purchase_power">
<el-input
v-model="sixSpeedTable.searchParam.purchase_power"
:placeholder="t('purchasePowerPlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('conceptAwareness')" prop="concept_awareness">
<el-input
v-model="sixSpeedTable.searchParam.concept_awareness"
:placeholder="t('conceptAwarenessPlaceholder')"
/>
</el-form-item>
<el-form-item
:label="t('preferredClassTime')"
prop="preferred_class_time"
>
<el-input
v-model="sixSpeedTable.searchParam.preferred_class_time"
:placeholder="t('preferredClassTimePlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('distance')" prop="distance">
<el-input
v-model="sixSpeedTable.searchParam.distance"
:placeholder="t('distancePlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('communication')" prop="communication">
<el-input
v-model="sixSpeedTable.searchParam.communication"
:placeholder="t('communicationPlaceholder')"
/>
</el-form-item>
<el-form-item
:label="t('promisedVisitTime')"
prop="promised_visit_time"
>
<el-input
v-model="sixSpeedTable.searchParam.promised_visit_time"
:placeholder="t('promisedVisitTimePlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('actualVisitTime')" prop="actual_visit_time">
<el-input
v-model="sixSpeedTable.searchParam.actual_visit_time"
:placeholder="t('actualVisitTimePlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('callIntent')" prop="call_intent">
<el-input
v-model="sixSpeedTable.searchParam.call_intent"
:placeholder="t('callIntentPlaceholder')"
/>
</el-form-item>
<el-form-item
:label="t('firstVisitStatus')"
prop="first_visit_status"
>
<el-input
v-model="sixSpeedTable.searchParam.first_visit_status"
:placeholder="t('firstVisitStatusPlaceholder')"
/>
</el-form-item>
<el-form-item
:label="t('secondVisitStatus')"
prop="second_visit_status"
>
<el-input
v-model="sixSpeedTable.searchParam.second_visit_status"
:placeholder="t('secondVisitStatusPlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('isClosed')" prop="is_closed">
<el-input
v-model="sixSpeedTable.searchParam.is_closed"
:placeholder="t('isClosedPlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('staffId')" prop="staff_id">
<el-input
v-model="sixSpeedTable.searchParam.staff_id"
:placeholder="t('staffIdPlaceholder')"
/>
</el-form-item>
<el-form-item :label="t('resourceId')" prop="resource_id">
<el-input
v-model="sixSpeedTable.searchParam.resource_id"
:placeholder="t('resourceIdPlaceholder')"
/>
</el-form-item>
<el-card class="box-card !border-none my-[10px] table-search-wrap" shadow="never">
<el-form :inline="true" :model="sixSpeedTable.searchParam" ref="searchFormRef">
<el-form-item>
<el-button type="primary" @click="loadSixSpeedList()">{{
t('search')
}}</el-button>
<el-button @click="resetForm(searchFormRef)">{{
t('reset')
}}</el-button>
<el-button type="primary" @click="loadSixSpeedList()">{{ t('search') }}</el-button>
<el-button @click="resetForm(searchFormRef)">{{ t('reset') }}</el-button>
</el-form-item>
</el-form>
</el-card>
<div class="mt-[10px]">
<el-table
:data="sixSpeedTable.data"
size="large"
v-loading="sixSpeedTable.loading"
>
<el-table :data="sixSpeedTable.data" size="large" v-loading="sixSpeedTable.loading">
<template #empty>
<span>{{ !sixSpeedTable.loading ? t('emptyData') : '' }}</span>
</template>
<el-table-column
prop="purchase_power"
:label="t('purchasePower')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column
prop="concept_awareness"
:label="t('conceptAwareness')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column
prop="preferred_class_time"
:label="t('preferredClassTime')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column
prop="distance"
:label="t('distance')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="preferred_class_time" :label="t('preferredClassTime')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="communication"
:label="t('communication')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="distance" :label="t('distance')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="promised_visit_time"
:label="t('promisedVisitTime')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="communication" :label="t('communication')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="actual_visit_time"
:label="t('actualVisitTime')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="promised_visit_time" :label="t('promisedVisitTime')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="call_intent"
:label="t('callIntent')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="actual_visit_time" :label="t('actualVisitTime')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="first_visit_status"
:label="t('firstVisitStatus')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column
prop="second_visit_status"
:label="t('secondVisitStatus')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column :label="t('callIntent')" min-width="180" align="center" :show-overflow-tooltip="true">
<template #default="{ row }">
<div v-for="(item, index) in call_intentList">
<div v-if="item.value == row.call_intent">{{ item.name }}</div>
</div>
</template>
</el-table-column>
<el-table-column
prop="is_closed"
:label="t('isClosed')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="first_visit_status" :label="t('firstVisitStatus')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="staff_id"
:label="t('staffId')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column prop="second_visit_status" :label="t('secondVisitStatus')" min-width="120" :show-overflow-tooltip="true"/>
<el-table-column
prop="resource_id"
:label="t('resourceId')"
min-width="120"
:show-overflow-tooltip="true"
/>
<el-table-column :label="t('isClosed')" min-width="180" align="center" :show-overflow-tooltip="true">
<template #default="{ row }">
<div v-for="(item, index) in is_closedList">
<div v-if="item.value == row.is_closed">{{ item.name }}</div>
</div>
</template>
</el-table-column>
<el-table-column
:label="t('operation')"
fixed="right"
min-width="120"
>
<el-table-column :label="t('operation')" fixed="right" min-width="120">
<template #default="{ row }">
<el-button type="primary" link @click="editEvent(row)">{{
t('edit')
}}</el-button>
<el-button type="primary" link @click="deleteEvent(row.id)">{{
t('delete')
}}</el-button>
<el-button type="primary" link @click="editEvent(row)">{{ t('edit') }}</el-button>
<el-button type="primary" link @click="deleteEvent(row.id)">{{ t('delete') }}</el-button>
</template>
</el-table-column>
</el-table>
<div class="mt-[16px] flex justify-end">
<el-pagination
v-model:current-page="sixSpeedTable.page"
v-model:page-size="sixSpeedTable.limit"
layout="total, sizes, prev, pager, next, jumper"
:total="sixSpeedTable.total"
@size-change="loadSixSpeedList()"
@current-change="loadSixSpeedList"
/>
<el-pagination v-model:current-page="sixSpeedTable.page" v-model:page-size="sixSpeedTable.limit"
layout="total, sizes, prev, pager, next, jumper" :total="sixSpeedTable.total"
@size-change="loadSixSpeedList()" @current-change="loadSixSpeedList" />
</div>
</div>
@ -255,13 +78,13 @@
import { reactive, ref, watch } from 'vue'
import { t } from '@/lang'
import { useDictionary } from '@/app/api/dict'
import { getSixSpeedList, deleteSixSpeed } from '@/app/api/six_speed'
import { getSixSpeedList, deleteSixSpeed, getWithPersonnelList, getWithCustomerResourcesList } from '@/app/api/six_speed'
import { img } from '@/utils/common'
import { ElMessageBox, FormInstance } from 'element-plus'
import { ElMessageBox,FormInstance } from 'element-plus'
import Edit from '@/app/views/six_speed/components/six-speed-edit.vue'
import { useRoute } from 'vue-router'
const route = useRoute()
const pageName = route.meta.title
const pageName = route.meta.title;
let sixSpeedTable = reactive({
page: 1,
@ -269,21 +92,9 @@ let sixSpeedTable = reactive({
total: 0,
loading: true,
data: [],
searchParam: {
purchase_power: '',
concept_awareness: '',
preferred_class_time: '',
distance: '',
communication: '',
promised_visit_time: '',
actual_visit_time: '',
call_intent: '',
first_visit_status: '',
second_visit_status: '',
is_closed: '',
staff_id: '',
resource_id: '',
},
searchParam:{
}
})
const searchFormRef = ref<FormInstance>()
@ -292,6 +103,26 @@ const searchFormRef = ref<FormInstance>()
const selectData = ref<any[]>([])
//
const purchase_powerList = ref([] as any[])
const purchase_powerDictList = async () => {
purchase_powerList.value = await (await useDictionary('customer_purchasing_power')).data.dictionary
}
purchase_powerDictList();
const concept_awarenessList = ref([] as any[])
const concept_awarenessDictList = async () => {
concept_awarenessList.value = await (await useDictionary('cognitive_concept')).data.dictionary
}
concept_awarenessDictList();
const call_intentList = ref([] as any[])
const call_intentDictList = async () => {
call_intentList.value = await (await useDictionary('preliminarycustomerintention')).data.dictionary
}
call_intentDictList();
const is_closedList = ref([] as any[])
const is_closedDictList = async () => {
is_closedList.value = await (await useDictionary('global_true_or_false')).data.dictionary
}
is_closedDictList();
/**
* 获取六一速列表
@ -303,14 +134,12 @@ const loadSixSpeedList = (page: number = 1) => {
getSixSpeedList({
page: sixSpeedTable.page,
limit: sixSpeedTable.limit,
...sixSpeedTable.searchParam,
})
.then((res) => {
...sixSpeedTable.searchParam
}).then(res => {
sixSpeedTable.loading = false
sixSpeedTable.data = res.data.data
sixSpeedTable.total = res.data.total
})
.catch(() => {
}).catch(() => {
sixSpeedTable.loading = false
})
}
@ -339,19 +168,32 @@ const editEvent = (data: any) => {
* 删除六一速
*/
const deleteEvent = (id: number) => {
ElMessageBox.confirm(t('sixSpeedDeleteTips'), t('warning'), {
ElMessageBox.confirm(t('sixSpeedDeleteTips'), t('warning'),
{
confirmButtonText: t('confirm'),
cancelButtonText: t('cancel'),
type: 'warning',
}).then(() => {
deleteSixSpeed(id)
.then(() => {
}
).then(() => {
deleteSixSpeed(id).then(() => {
loadSixSpeedList()
}).catch(() => {
})
.catch(() => {})
})
}
const staffIdList = ref([])
const setStaffIdList = async () => {
staffIdList.value = await (await getWithPersonnelList({})).data
}
setStaffIdList()
const resourceIdList = ref([])
const setResourceIdList = async () => {
resourceIdList.value = await (await getWithCustomerResourcesList({})).data
}
setResourceIdList()
const resetForm = (formEl: FormInstance | undefined) => {
if (!formEl) return
formEl.resetFields()
@ -368,5 +210,5 @@ const resetForm = (formEl: FormInstance | undefined) => {
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
}
}
</style>

44
niucloud/app/adminapi/controller/customer_resources/CustomerResources.php

@ -65,12 +65,22 @@ class CustomerResources extends BaseAdminController
["campus",""],
["status",""],
["create_year_month",date("Y-m")],
["create_date",date("Y-m-d")]
]);
["create_date",date("Y-m-d")],
["purchase_power",""],
["concept_awareness",""],
["preferred_class_time",""],
["distance_tow",""],
["communication",""],
["promised_visit_time",""],
["actual_visit_time",""],
["call_intent",""],
["first_visit_status",""],
["second_visit_status",""],
["is_closed",""]
]);
$this->validate($data, 'app\validate\customer_resources\CustomerResources.add');
$id = (new CustomerResourcesService())->add($data);
return success('ADD_SUCCESS', ['id' => $id]);
return (new CustomerResourcesService())->add($data);
}
/**
@ -94,11 +104,27 @@ class CustomerResources extends BaseAdminController
["decision_maker",""],
["initial_intent",""],
["campus",""],
["status",""]
["status",""],
["create_year_month",date("Y-m")],
["create_date",date("Y-m-d")],
["purchase_power",""],
["concept_awareness",""],
["preferred_class_time",""],
["distance_tow",""],
["communication",""],
["promised_visit_time",""],
["actual_visit_time",""],
["call_intent",""],
["first_visit_status",""],
["second_visit_status",""],
["is_closed",""]
]);
$this->validate($data, 'app\validate\customer_resources\CustomerResources.edit');
(new CustomerResourcesService())->edit($id, $data);
return success('EDIT_SUCCESS');
return (new CustomerResourcesService())->edit($id, $data);
}
/**
@ -112,6 +138,10 @@ class CustomerResources extends BaseAdminController
}
public function getPersonnelAll(){
return success(( new CustomerResourcesService())->getPersonnelAll());
}
public function getCampusAll(){
return success(( new CustomerResourcesService())->getCampusAll());
}

34
niucloud/app/adminapi/controller/six_speed/SixSpeed.php

@ -28,19 +28,7 @@ class SixSpeed extends BaseAdminController
*/
public function lists(){
$data = $this->request->params([
["purchase_power",""],
["concept_awareness",""],
["preferred_class_time",""],
["distance",""],
["communication",""],
["promised_visit_time",""],
["actual_visit_time",""],
["call_intent",""],
["first_visit_status",""],
["second_visit_status",""],
["is_closed",""],
["staff_id",""],
["resource_id",""]
]);
return success((new SixSpeedService())->getPage($data));
}
@ -65,14 +53,12 @@ class SixSpeed extends BaseAdminController
["preferred_class_time",""],
["distance",""],
["communication",""],
["promised_visit_time","2025-05-16 17:57:14"],
["actual_visit_time","2025-05-16 17:57:14"],
["promised_visit_time","2025-05-18 16:21:59"],
["actual_visit_time","2025-05-18 16:21:59"],
["call_intent",""],
["first_visit_status",""],
["second_visit_status",""],
["is_closed",0],
["staff_id",0],
["resource_id",0],
]);
$this->validate($data, 'app\validate\six_speed\SixSpeed.add');
@ -92,14 +78,12 @@ class SixSpeed extends BaseAdminController
["preferred_class_time",""],
["distance",""],
["communication",""],
["promised_visit_time","2025-05-16 17:57:14"],
["actual_visit_time","2025-05-16 17:57:14"],
["promised_visit_time","2025-05-18 16:21:59"],
["actual_visit_time","2025-05-18 16:21:59"],
["call_intent",""],
["first_visit_status",""],
["second_visit_status",""],
["is_closed",0],
["staff_id",0],
["resource_id",0],
]);
$this->validate($data, 'app\validate\six_speed\SixSpeed.edit');
@ -118,4 +102,12 @@ class SixSpeed extends BaseAdminController
}
public function getPersonnelAll(){
return success(( new SixSpeedService())->getPersonnelAll());
}
public function getCustomerResourcesAll(){
return success(( new SixSpeedService())->getCustomerResourcesAll());
}
}

4
niucloud/app/adminapi/route/customer_resources.php

@ -18,6 +18,8 @@ use app\adminapi\middleware\AdminLog;
// USER_CODE_BEGIN -- customer_resources
Route::group('customer_resources', function () {
@ -33,6 +35,8 @@ Route::group('customer_resources', function () {
//删除客户资源
Route::delete('customer_resources/:id', 'customer_resources.CustomerResources/del');
Route::get('personnel_all','customer_resources.CustomerResources/getPersonnelAll');
Route::get('campus_all','customer_resources.CustomerResources/getCampusAll');
})->middleware([

5
niucloud/app/adminapi/route/six_speed.php

@ -14,6 +14,7 @@ use think\facade\Route;
use app\adminapi\middleware\AdminCheckRole;
use app\adminapi\middleware\AdminCheckToken;
use app\adminapi\middleware\AdminLog;
// USER_CODE_BEGIN -- six_speed
Route::group('six_speed', function () {
@ -29,6 +30,10 @@ Route::group('six_speed', function () {
//删除六一速
Route::delete('six_speed/:id', 'six_speed.SixSpeed/del');
Route::get('personnel_all','six_speed.SixSpeed/getPersonnelAll');
Route::get('customer_resources_all','six_speed.SixSpeed/getCustomerResourcesAll');
})->middleware([
AdminCheckToken::class,
AdminCheckRole::class,

28
niucloud/app/common.php

@ -1046,3 +1046,31 @@ function get_campus_where($user_id){
return $where;
}
function getModifiedFields(array $oldData, array $newData): array
{
$modifiedFields = [];
$oldValues = [];
$newValues = [];
foreach ($newData as $key => $newValue) {
if (!array_key_exists($key, $oldData)) {
continue;
}
if ($oldData[$key] != $newValue) {
$modifiedFields[] = $key;
$oldValues[$key] = $oldData[$key];
$newValues[$key] = $newValue;
}
}
return [
'is_save' => !empty($modifiedFields),
'modified_fields' => json_encode($modifiedFields, JSON_UNESCAPED_UNICODE),
'old_values' => json_encode($oldValues, JSON_UNESCAPED_UNICODE),
'new_values' => json_encode($newValues, JSON_UNESCAPED_UNICODE),
];
}

6
niucloud/app/model/customer_resources/CustomerResources.php

@ -16,6 +16,8 @@ use think\model\concern\SoftDelete;
use think\model\relation\HasMany;
use think\model\relation\HasOne;
use app\model\personnel\Personnel;
use app\model\campus\Campus;
/**
@ -81,6 +83,10 @@ class CustomerResources extends BaseModel
public function personnel(){
return $this->hasOne(Personnel::class, 'id', 'consultant')->joinType('left')->withField('name,id')->bind(['consultant_name'=>'name']);
}
public function campus(){
return $this->hasOne(Campus::class, 'id', 'campus')->joinType('left')->withField('campus_name,id')->bind(['campus_name'=>'campus_name']);
}

172
niucloud/app/model/six_speed/SixSpeed.php

@ -16,6 +16,10 @@ use think\model\concern\SoftDelete;
use think\model\relation\HasMany;
use think\model\relation\HasOne;
use app\model\personnel\Personnel;
use app\model\customer_resources\CustomerResources;
/**
* 六一速模型
* Class SixSpeed
@ -50,177 +54,17 @@ class SixSpeed extends BaseModel
*/
protected $defaultSoftDelete = 0;
/**
* 搜索器:六一速编号
* @param $value
* @param $data
*/
public function searchIdAttr($query, $value, $data)
{
if ($value) {
$query->where("id", $value);
}
}
/**
* 搜索器:六一速需求购买力
* @param $value
* @param $data
*/
public function searchPurchasePowerAttr($query, $value, $data)
{
if ($value) {
$query->where("purchase_power", $value);
}
}
/**
* 搜索器:六一速认知理念
* @param $value
* @param $data
*/
public function searchConceptAwarenessAttr($query, $value, $data)
{
if ($value) {
$query->where("concept_awareness", $value);
}
}
/**
* 搜索器:六一速可选上课时间
* @param $value
* @param $data
*/
public function searchPreferredClassTimeAttr($query, $value, $data)
{
if ($value) {
$query->where("preferred_class_time", $value);
}
}
/**
* 搜索器:六一速距离
* @param $value
* @param $data
*/
public function searchDistanceAttr($query, $value, $data)
{
if ($value) {
$query->where("distance", $value);
}
}
/**
* 搜索器:六一速沟通备注
* @param $value
* @param $data
*/
public function searchCommunicationAttr($query, $value, $data)
{
if ($value) {
$query->where("communication", $value);
}
}
/**
* 搜索器:六一速承诺到访时间
* @param $value
* @param $data
*/
public function searchPromisedVisitTimeAttr($query, $value, $data)
{
if ($value) {
$query->where("promised_visit_time", $value);
}
}
/**
* 搜索器:六一速实际到访时间
* @param $value
* @param $data
*/
public function searchActualVisitTimeAttr($query, $value, $data)
{
if ($value) {
$query->where("actual_visit_time", $value);
}
}
/**
* 搜索器:六一速电话后的意向程度: low-低, medium-中, high-高
* @param $value
* @param $data
*/
public function searchCallIntentAttr($query, $value, $data)
{
if ($value) {
$query->where("call_intent", $value);
}
}
/**
* 搜索器:六一速一访情况
* @param $value
* @param $data
*/
public function searchFirstVisitStatusAttr($query, $value, $data)
{
if ($value) {
$query->where("first_visit_status", $value);
}
}
/**
* 搜索器:六一速二访情况
* @param $value
* @param $data
*/
public function searchSecondVisitStatusAttr($query, $value, $data)
{
if ($value) {
$query->where("second_visit_status", $value);
}
}
/**
* 搜索器:六一速是否关单: 1-是, 0-否
* @param $value
* @param $data
*/
public function searchIsClosedAttr($query, $value, $data)
{
if ($value) {
$query->where("is_closed", $value);
}
}
/**
* 搜索器:六一速人员ID
* @param $value
* @param $data
*/
public function searchStaffIdAttr($query, $value, $data)
{
if ($value) {
$query->where("staff_id", $value);
}
public function personnel(){
return $this->hasOne(Personnel::class, 'id', 'staff_id')->joinType('left')->withField('name,id')->bind(['staff_id_name'=>'name']);
}
/**
* 搜索器:六一速资源ID
* @param $value
* @param $data
*/
public function searchResourceIdAttr($query, $value, $data)
{
if ($value) {
$query->where("resource_id", $value);
public function customerResources(){
return $this->hasOne(CustomerResources::class, 'id', 'resource_id')->joinType('left')->withField('name,id')->bind(['resource_id_name'=>'name']);
}
}
}

149
niucloud/app/service/admin/customer_resources/CustomerResourcesService.php

@ -11,9 +11,13 @@
namespace app\service\admin\customer_resources;
use app\model\customer_resource_changes\CustomerResourceChanges;
use app\model\customer_resources\CustomerResources;
use app\model\personnel\Personnel;
use app\model\campus\Campus;
use app\model\six_speed\SixSpeed;
use app\model\six_speed_modification_log\SixSpeedModificationLog;
use core\base\BaseAdminService;
@ -40,7 +44,7 @@ class CustomerResourcesService extends BaseAdminService
$field = 'id,create_year_month,create_date,source,source_channel,consultant,name,age,gender,phone_number,demand,purchasing_power,cognitive_idea,optional_class_time,distance,decision_maker,initial_intent,campus,created_at,updated_at,deleted_at,status';
$order = 'id desc';
$search_model = $this->model->where(get_campus_where($this->uid))->withSearch(["name","phone_number"], $where)->with(['campus'])->field($field)->order($order);
$search_model = $this->model->withSearch(["name","phone_number"], $where)->with(['personnel'])->field($field)->order($order);
$list = $this->pageQuery($search_model);
return $list;
}
@ -54,7 +58,12 @@ class CustomerResourcesService extends BaseAdminService
{
$field = 'id,create_year_month,create_date,source,source_channel,consultant,name,age,gender,phone_number,demand,purchasing_power,cognitive_idea,optional_class_time,distance,decision_maker,initial_intent,campus,created_at,updated_at,deleted_at,status';
$info = $this->model->field($field)->where([['id', "=", $id]])->findOrEmpty()->toArray();
$info = $this->model->field($field)->where([['id', "=", $id]])->with(['personnel'])->findOrEmpty()->toArray();
$sixSpeed = new SixSpeed();
$data = $sixSpeed->where(['resource_id' => $id])->field("*,distance as distance_tow")->findOrEmpty()->toArray();
$info = $info+$data;
return $info;
}
@ -65,9 +74,41 @@ class CustomerResourcesService extends BaseAdminService
*/
public function add(array $data)
{
$data['consultant'] = $this->username;
$personnel = new Personnel();
$data['consultant'] = $personnel->where(['sys_user_id' => $this->uid])->value("id");
if(!$data['consultant']){
return fail("操作失败");
}
$sixSpeed = new SixSpeed();
$res = $this->model->create($data);
return $res->id;
if($data['purchase_power']){
$six_id = $sixSpeed->where(['resource_id' => $res->id])->value("id");
$data['staff_id'] = $data['consultant'];
$field = [
'purchase_power' => $data['purchase_power'],
'concept_awareness' => $data['concept_awareness'],
'preferred_class_time' => $data['preferred_class_time'],
'distance' => $data['distance_tow'],
'communication' => $data['communication'],
'promised_visit_time' => $data['promised_visit_time'],
'actual_visit_time' => $data['actual_visit_time'],
'call_intent' => $data['call_intent'],
'first_visit_status' => $data['first_visit_status'],
'second_visit_status' => $data['second_visit_status'],
'is_closed' => $data['is_closed'],
'staff_id' => $data['staff_id'],
'resource_id' => $res->id
];
if($six_id){
$sixSpeed->where(['resource_id' => $res->id])->update($field);
}else{
$sixSpeed->insert($field);
}
}
return success("操作成功");
}
@ -75,14 +116,101 @@ class CustomerResourcesService extends BaseAdminService
* 客户资源编辑
* @param int $id
* @param array $data
* @return bool
*/
public function edit(int $id, array $data)
{
$personnel = new Personnel();
// $data['consultant'] = $personnel->where(['sys_user_id' => $this->uid])->value("id");
// if(!$data['consultant']){
// return fail("操作失败");
// }
$data['consultant'] = 1;
$res = $this->model->where([['id', '=', $id]])->findOrEmpty()->toArray();
$this->model->where([['id', '=', $id]])->update([
'source' => $data['source'],
'source_channel' => $data['source_channel'],
'consultant' => $data['consultant'],
'name' => $data['name'],
'age' => $data['age'],
'gender' => $data['gender'],
'phone_number' => $data['phone_number'],
'demand' => $data['demand'],
'purchasing_power' => $data['purchasing_power'],
'cognitive_idea' => $data['cognitive_idea'],
'optional_class_time' => $data['optional_class_time'],
'distance' => $data['distance'],
'decision_maker' => $data['decision_maker'],
'initial_intent' => $data['initial_intent'],
'campus' => $data['campus'],
'status' => $data['status'],
'create_year_month' => $data['create_year_month'],
'create_date' => $data['create_date']
]);
$resources_save = getModifiedFields($res,$data);
$customerResourceChanges = new CustomerResourceChanges();
if($resources_save['is_save']){
$customerResourceChanges->insert([
'customer_resource_id' => $id,
'operator_id' => $data['consultant'],
'campus_id' => $data['campus'],
'modified_fields' => $resources_save['modified_fields'],
'old_values' => $resources_save['old_values'],
'new_values' => $resources_save['new_values']
]);
}
$data['consultant'] = $this->username;
$this->model->where([['id', '=', $id]])->update($data);
return true;
$sixSpeed = new SixSpeed();
if($data['purchase_power']){
$sixSpeedModificationLog = new SixSpeedModificationLog();
$six_id = $sixSpeed->where(['resource_id' => $id])->value("id");
$data['staff_id'] = $data['consultant'];
$field = [
'purchase_power' => $data['purchase_power'],
'concept_awareness' => $data['concept_awareness'],
'preferred_class_time' => $data['preferred_class_time'],
'distance' => $data['distance_tow'],
'communication' => $data['communication'],
'promised_visit_time' => $data['promised_visit_time'],
'actual_visit_time' => $data['actual_visit_time'],
'call_intent' => $data['call_intent'],
'first_visit_status' => $data['first_visit_status'],
'second_visit_status' => $data['second_visit_status'],
'is_closed' => $data['is_closed'],
'staff_id' => $data['staff_id'],
'resource_id' => $id
];
if($six_id){
$six_log = $sixSpeed->where(['resource_id' => $id])->findOrEmpty()->toArray();
$six_save = getModifiedFields($six_log,$field);
if($six_save['is_save']){
$sixSpeedModificationLog->insert([
'customer_resource_id' => $id,
'operator_id' => $data['consultant'],
'campus_id' => $data['campus'],
'modified_field' => $six_save['modified_fields'],
'old_value' => $six_save['old_values'],
'new_value' => $six_save['new_values']
]);
}
$sixSpeed->where(['resource_id' => $id])->update($field);
}else{
$sixSpeed->insert($field);
}
}
return success("操作成功");
}
/**
@ -98,6 +226,11 @@ class CustomerResourcesService extends BaseAdminService
}
public function getPersonnelAll(){
$personnelModel = new Personnel();
return $personnelModel->select()->toArray();
}
public function getCampusAll(){
$campusModel = new Campus();
return $campusModel->select()->toArray();

16
niucloud/app/service/admin/six_speed/SixSpeedService.php

@ -12,6 +12,8 @@
namespace app\service\admin\six_speed;
use app\model\six_speed\SixSpeed;
use app\model\personnel\Personnel;
use app\model\customer_resources\CustomerResources;
use core\base\BaseAdminService;
@ -39,7 +41,7 @@ class SixSpeedService extends BaseAdminService
$field = 'id,purchase_power,concept_awareness,preferred_class_time,distance,communication,promised_visit_time,actual_visit_time,call_intent,first_visit_status,second_visit_status,is_closed,staff_id,resource_id,created_at,updated_at,deleted_at';
$order = 'id desc';
$search_model = $this->model->withSearch(["id","purchase_power","concept_awareness","preferred_class_time","distance","communication","promised_visit_time","actual_visit_time","call_intent","first_visit_status","second_visit_status","is_closed","staff_id","resource_id"], $where)->field($field)->order($order);
$search_model = $this->model->withSearch([], $where)->with(['personnel','customerResources'])->field($field)->order($order);
$list = $this->pageQuery($search_model);
return $list;
}
@ -53,7 +55,7 @@ class SixSpeedService extends BaseAdminService
{
$field = 'id,purchase_power,concept_awareness,preferred_class_time,distance,communication,promised_visit_time,actual_visit_time,call_intent,first_visit_status,second_visit_status,is_closed,staff_id,resource_id,created_at,updated_at,deleted_at';
$info = $this->model->field($field)->where([['id', "=", $id]])->findOrEmpty()->toArray();
$info = $this->model->field($field)->where([['id', "=", $id]])->with(['personnel','customerResources'])->findOrEmpty()->toArray();
return $info;
}
@ -95,5 +97,15 @@ class SixSpeedService extends BaseAdminService
}
public function getPersonnelAll(){
$personnelModel = new Personnel();
return $personnelModel->select()->toArray();
}
public function getCustomerResourcesAll(){
$customerResourcesModel = new CustomerResources();
return $customerResourcesModel->select()->toArray();
}
}

8
niucloud/app/validate/six_speed/SixSpeed.php

@ -28,8 +28,6 @@ class SixSpeed extends BaseValidate
'promised_visit_time' => 'require',
'call_intent' => 'require',
'is_closed' => 'require',
'staff_id' => 'require',
'resource_id' => 'require',
];
protected $message = [
@ -41,13 +39,11 @@ class SixSpeed extends BaseValidate
'promised_visit_time.require' => ['common_validate.require', ['promised_visit_time']],
'call_intent.require' => ['common_validate.require', ['call_intent']],
'is_closed.require' => ['common_validate.require', ['is_closed']],
'staff_id.require' => ['common_validate.require', ['staff_id']],
'resource_id.require' => ['common_validate.require', ['resource_id']],
];
protected $scene = [
"add" => ['purchase_power', 'concept_awareness', 'preferred_class_time', 'distance', 'communication', 'promised_visit_time', 'actual_visit_time', 'call_intent', 'first_visit_status', 'second_visit_status', 'is_closed', 'staff_id', 'resource_id'],
"edit" => ['purchase_power', 'concept_awareness', 'preferred_class_time', 'distance', 'communication', 'promised_visit_time', 'actual_visit_time', 'call_intent', 'first_visit_status', 'second_visit_status', 'is_closed', 'staff_id', 'resource_id']
"add" => ['purchase_power', 'concept_awareness', 'preferred_class_time', 'distance', 'communication', 'promised_visit_time', 'actual_visit_time', 'call_intent', 'first_visit_status', 'second_visit_status', 'is_closed'],
"edit" => ['purchase_power', 'concept_awareness', 'preferred_class_time', 'distance', 'communication', 'promised_visit_time', 'actual_visit_time', 'call_intent', 'first_visit_status', 'second_visit_status', 'is_closed']
];
}

Loading…
Cancel
Save