合并CRM

This commit is contained in:
xuli3099
2025-08-25 16:41:33 +08:00
parent fd5c5739e4
commit 0a1d4de168
78 changed files with 19363 additions and 159 deletions

View File

@@ -0,0 +1,238 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'通用表单添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="通用表单类型" name="generalFormType" class="f-c-right">
<picker @change="onGeneralFormTypeChange" :value="generalFormTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[generalFormTypeIndex]?.name || '请选择通用表单类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="信息内容" name="informationContent" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.informationContent"
placeholder="请输入信息内容" class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd
} from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
generalFormType:"专项市场调研信息",//通用表单类型
informationContent: "", //信息内容
picture: "", //、图片
photos: null,
description: null,
informationType:"通用表单" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
generalFormType: {
rules: [{
required: true,
errorMessage: '请选择通用表单类型'
}]
},
informationContent: {
rules: [{
required: true,
errorMessage: '请输入信息内容'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '专项市场调研信息'
},
{
id: 1,
name: '供货改进需求信息'
},
{
id: 2,
name: '客户赞扬、抱怨信息'
},
{
id: 3,
name: '领导承诺'
},
{
id: 4,
name: '待办事项'
},
{
id: 5,
name: '备忘录'
}
])
const generalFormTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
console.log('表单数据:', formData.value)
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onGeneralFormTypeChange = (e) => {
generalFormTypeIndex.value = e.detail.value
console.log('generalFormTypeIndex:', array.value[e.detail.value]?.name)
formData.value.generalFormType = array.value[e.detail.value]?.name || ''
}
// 如果你原来在 onShow 中做了类似这样:
// let res = currPage.data.cusData; 判断是否传入了客户信息
// 那么在 Vue3 中通常是通过路由参数或者 Vuex/Pinia 等状态管理获取
// 暂时不做,如你后续需要可继续补充
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,272 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'通用表单添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="通用表单类型" name="generalFormType" class="f-c-right">
<picker @change="onGeneralFormTypeChange" :value="generalFormTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[generalFormTypeIndex]?.name || '请选择通用表单类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="信息内容" name="informationContent" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.informationContent"
placeholder="请输入信息内容" class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete
} from '@/api/crm/api_ys.js'
import { onLoad } from '@dcloudio/uni-app'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
generalFormType:"专项市场调研信息",//通用表单类型
informationContent: "", //信息内容
picture: "", //、图片
photos: null,
description: null,
informationType:"通用表单" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
generalFormType: {
rules: [{
required: true,
errorMessage: '请选择通用表单类型'
}]
},
informationContent: {
rules: [{
required: true,
errorMessage: '请输入信息内容'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '专项市场调研信息'
},
{
id: 1,
name: '供货改进需求信息'
},
{
id: 2,
name: '客户赞扬、抱怨信息'
},
{
id: 3,
name: '领导承诺'
},
{
id: 4,
name: '待办事项'
},
{
id: 5,
name: '备忘录'
}
])
const generalFormTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onGeneralFormTypeChange = (e) => {
generalFormTypeIndex.value = e.detail.value
console.log('generalFormTypeIndex:', array.value[e.detail.value]?.name)
formData.value.generalFormType = array.value[e.detail.value]?.name || ''
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete(formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,461 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'竞争对手信息添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<!-- <uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item> -->
<uni-forms-item label="竞争单位" name="competitiveUnits" class="f-c-right">
<picker @change="onCompetitiveUnitsChange" :value="competitiveUnitsIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[competitiveUnitsIndex]?.name || '请选择竞争单位' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="信息类型" name="competitorLevelOne" class="f-c-right">
<picker @change="onCompetitorLevelOneChange" :value="competitorLevelOne" :range="array1"
:range-key="'name'">
<view class="picker">
{{ array1[competitorLevelOneIndex]?.name || '请选择信息类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="具体分类" name="competitorLevelTwo" class="f-c-right">
<picker
@change="onCompetitorLevelTwoChange"
:value="competitorLevelTwoIndex"
:range="currentCompetitorLevelTwoArray"
:range-key="'name'"
>
<view class="picker">
{{ currentCompetitorLevelTwoArray[competitorLevelTwoIndex]?.name || '请选择具体分类' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="具体信息" name="specificMatters" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.specificMatters" :placeholder="dynamicPlaceholder"
class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd
} from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
competitiveUnits: "振华云科", //竞争对手
competitorLevelOne: "基本信息", //竞争对手一级标签
competitorLevelTwo: "", //竞争对手二级标签
specificMatters: "", //具体事情
remark: "", //、备注
picture: "", //、图片
informationType: "竞争对手信息" //信息类型
})
// 表单验证规则
const rules = {
competitiveUnits: {
rules: [{
required: true,
errorMessage: '请选择竞争单位'
}]
},
competitorLevelOne: {
rules: [{
required: true,
errorMessage: '请选择信息类型'
}]
},
competitorLevelTwo: {
rules: [{
required: true,
errorMessage: '请选择具体分类'
}]
},
specificMatters: {
rules: [{
required: true,
errorMessage: '请输入具体信息'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '振华云科'
},
{
id: 1,
name: '四川永星'
},
{
id: 2,
name: '风华邦科'
},
{
id: 3,
name: '株洲宏达'
},
{
id: 4,
name: '福建火炬'
},
{
id: 5,
name: '福建毫米'
},
{
id: 6,
name: '广州创天'
},
{
id: 7,
name: '贝迪斯'
},
{
id: 8,
name: '禹龙通电子'
},
{
id: 9,
name: '成都昊天'
},
{
id: 10,
name: '风华特种'
},
{
id: 11,
name: '济宁正和'
},
{
id: 12,
name: '盛世'
},
{
id: 13,
name: '广东意杰'
},
{
id: 14,
name: '西安威特'
},
{
id: 15,
name: '盛雷城'
},
{
id: 16,
name: '广东福德'
},
{
id: 17,
name: '咸阳智联'
},
{
id: 18,
name: '安徽来福'
},
{
id: 19,
name: '国巨'
},
{
id: 20,
name: '开步电子'
},
{
id: 21,
name: '成都宏鸣'
},
{
id: 22,
name: '其他'
},
])
const array1 = ref([{
id: 0,
name: '基本信息'
},
{
id: 1,
name: '业务信息'
},
])
const array2 = ref([{
id: 0,
name: '资质情况'
},
{
id: 1,
name: '人员情况'
},
{
id: 2,
name: '产品种类(新研制产品、主推产品)'
},
{
id: 3,
name: '近三年销售情况'
},
{
id: 4,
name: '市场组织架构'
},
{
id: 5,
name: '市场策略'
},
{
id: 6,
name: '竞争对手变化产生的市场机会'
},
{
id: 7,
name: '其他'
},
])
const array3 = ref([{
id: 0,
name: '质量问题'
},
{
id: 1,
name: '重大市场活动'
},
{
id: 2,
name: '市场占有率'
},
{
id: 3,
name: '与客户人员的关系'
},
{
id: 4,
name: '竞争动向'
},
{
id: 5,
name: '其他'
},
])
const dynamicPlaceholder = computed(() => {
const placeholderMap = {
'资质情况': '2020年通过IATF16949 汽车质量管理体系认证IATF证书编号0368475CASC证书编号2020A069/四川永星目前有二筛条件已经拿到CANS认证可以对电阻器进行二筛。',
'人员情况': '893厂现有员工900余人在全国设立6个办事处/云科成都办事处有五人,负责成都、绵阳、重庆,对接凯天的是罗晓波',
'产品种类(新研制产品、主推产品)': '云科已申请RN5161系列进入SAST-G目录目前尚未完成进目录工作。/云科2022年薄膜电路产品上市。',
'近三年销售情况': '2021年云科销售浆料8亿元/1、云科一年在隆盛的电阻销售额在200万左右保险丝的销售额也在200万左右。2、云科一季度在607所的销售额大概在150万左右',
'市场组织架构': '1、云科市场部新上任一位领导与九院13所领导副所长关系密切2、西安办事处新换办事处主任杨文建是振华集团领导的亲属3、云科北京片区今年更换领导是从振华富调过去的并且把原来负责九院的业务员吴明金撤了更换了新的业务员1、振华云科的河南地区新增一位业务员胡心圣。2、佰威达的老板是施广勤同时也是博威的老板。',
'市场策略': '1、外围进攻通过熔断器、射频等产品抢占常规电阻器份额例如中航工业金城南京机电液压工程研究中心云科的片阻已有少量供货。2、盯住高价值产品抢占我们高价值产品合金箔、熔断器、射频的最主要客户的市场份额例如九院16所、兵器212所但目前仍未得手。低价策略风华和火炬今年想以低价策略进驻抢占份额。目前普军级F精度报价0.5J精度报价0.2。',
'竞争对手变化产生的市场机会': '893将一部分股份卖给北海银河银河快倒闭时改行做医药行业不景气',
'其他': '请输入其他补充信息',
'质量问题': '在核九院五所云科产品出现批次性硫化问题云科及时处理用抗硫化产品做了替代。893在梅岭说我司08年质量问题基础采购员都知道了这个情况说不敢选择我司火工品。目前常规片式产品普军等级云科与我司均为8毛893报6毛未选择他们。',
'重大市场活动': '云科在航天五院举办党建活动',
'市场占有率': '云科在航天领域的市场占有率为30%信息来源是XXX)/振华云科在航天航空的份额占比大概在60%,大都基于领导层的关系基础,几乎不会来司拜访用户。',
'与客户人员的关系': '振华集团高层跟118厂高层关系很好有时会来上海组织联谊活动/青岛整流器制造有限公司有云科内线,了解我公司价格。',
'竞争动向': '中电科技集团重庆声光电有限公司的106项目893中标/云科最近一次竞标常规国军标0.55元/只单一规格用量60万只)D精度厚膜0.9元/只(2个规格共20万只),通过这次竞标得到三个单一规格电阻的采购。'
}
return placeholderMap[formData.value.competitorLevelTwo] || '请输入具体信息'
})
const competitiveUnitsIndex = ref(0)
// 当前选中的信息类型
const competitorLevelOneIndex = ref(0)
const competitorLevelOne = ref('基本信息')
const competitorLevelTwo = ref('资质情况')
// 当前选中的具体分类索引
const competitorLevelTwoIndex = ref(0)
// 当前显示的具体分类选项数组(动态切换 array2 / array3
const currentCompetitorLevelTwoArray = ref(array2.value)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
console.log('表单数据:', formData.value)
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onCompetitiveUnitsChange = (e) => {
competitiveUnitsIndex.value = e.detail.value
console.log('competitiveUnitsIndex:', array.value[e.detail.value]?.name)
formData.value.competitiveUnits = array.value[e.detail.value]?.name || ''
}
const onCompetitorLevelOneChange = (e) => {
if (e.detail.value >= 0 && e.detail.value < array1.value.length) {
competitorLevelOneIndex.value = e.detail.value
} else {
console.error('选择了无效的索引,重置为 0')
competitorLevelOneIndex.value = 0
}
const selectedType = array1.value[e.detail.value]?.name
console.log('选中的信息类型:', selectedType)
formData.value.competitorLevelOne = selectedType
if (selectedType === '基本信息') {
currentCompetitorLevelTwoArray.value = array2.value
competitorLevelTwoIndex.value = 0
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[0]?.name || ''
} else if (selectedType === '业务信息') {
currentCompetitorLevelTwoArray.value = array3.value
competitorLevelTwoIndex.value = 0
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[0]?.name || ''
} else {
currentCompetitorLevelTwoArray.value = []
formData.value.competitorLevelTwo = ''
}
}
const onCompetitorLevelTwoChange = (e) => {
competitorLevelTwoIndex.value = e.detail.value
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[e.detail.value]?.name || ''
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,504 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'竞争对手信息添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<!-- <uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item> -->
<uni-forms-item label="竞争单位" name="competitiveUnits" class="f-c-right">
<picker @change="onCompetitiveUnitsChange" :value="competitiveUnitsIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[competitiveUnitsIndex]?.name || '请选择竞争单位' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="信息类型" name="competitorLevelOne" class="f-c-right">
<picker @change="onCompetitorLevelOneChange" :value="competitorLevelOne" :range="array1"
:range-key="'name'">
<view class="picker">
{{ array1[competitorLevelOneIndex]?.name || '请选择信息类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="具体分类" name="competitorLevelTwo" class="f-c-right">
<picker
@change="onCompetitorLevelTwoChange"
:value="competitorLevelTwoIndex"
:range="currentCompetitorLevelTwoArray"
:range-key="'name'"
>
<view class="picker">
{{ currentCompetitorLevelTwoArray[competitorLevelTwoIndex]?.name || '请选择具体分类' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="具体信息" name="specificMatters" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.specificMatters" :placeholder="dynamicPlaceholder"
class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete
} from '@/api/crm/api_ys.js'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
import { onLoad } from '@dcloudio/uni-app'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
competitiveUnits: "振华云科", //竞争对手
competitorLevelOne: "基本信息", //竞争对手一级标签
competitorLevelTwo: "", //竞争对手二级标签
specificMatters: "", //具体事情
remark: "", //、备注
picture: "", //、图片
informationType: "竞争对手信息" //信息类型
})
// 表单验证规则
const rules = {
competitiveUnits: {
rules: [{
required: true,
errorMessage: '请选择竞争单位'
}]
},
competitorLevelOne: {
rules: [{
required: true,
errorMessage: '请选择信息类型'
}]
},
competitorLevelTwo: {
rules: [{
required: true,
errorMessage: '请选择具体分类'
}]
},
specificMatters: {
rules: [{
required: true,
errorMessage: '请输入具体信息'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '振华云科'
},
{
id: 1,
name: '四川永星'
},
{
id: 2,
name: '风华邦科'
},
{
id: 3,
name: '株洲宏达'
},
{
id: 4,
name: '福建火炬'
},
{
id: 5,
name: '福建毫米'
},
{
id: 6,
name: '广州创天'
},
{
id: 7,
name: '贝迪斯'
},
{
id: 8,
name: '禹龙通电子'
},
{
id: 9,
name: '成都昊天'
},
{
id: 10,
name: '风华特种'
},
{
id: 11,
name: '济宁正和'
},
{
id: 12,
name: '盛世'
},
{
id: 13,
name: '广东意杰'
},
{
id: 14,
name: '西安威特'
},
{
id: 15,
name: '盛雷城'
},
{
id: 16,
name: '广东福德'
},
{
id: 17,
name: '咸阳智联'
},
{
id: 18,
name: '安徽来福'
},
{
id: 19,
name: '国巨'
},
{
id: 20,
name: '开步电子'
},
{
id: 21,
name: '成都宏鸣'
},
{
id: 22,
name: '其他'
},
])
const array1 = ref([{
id: 0,
name: '基本信息'
},
{
id: 1,
name: '业务信息'
},
])
const array2 = ref([{
id: 0,
name: '资质情况'
},
{
id: 1,
name: '人员情况'
},
{
id: 2,
name: '产品种类(新研制产品、主推产品)'
},
{
id: 3,
name: '近三年销售情况'
},
{
id: 4,
name: '市场组织架构'
},
{
id: 5,
name: '市场策略'
},
{
id: 6,
name: '竞争对手变化产生的市场机会'
},
{
id: 7,
name: '其他'
},
])
const array3 = ref([{
id: 0,
name: '质量问题'
},
{
id: 1,
name: '重大市场活动'
},
{
id: 2,
name: '市场占有率'
},
{
id: 3,
name: '与客户人员的关系'
},
{
id: 4,
name: '竞争动向'
},
{
id: 5,
name: '其他'
},
])
const dynamicPlaceholder = computed(() => {
const placeholderMap = {
'资质情况': '2020年通过IATF16949 汽车质量管理体系认证IATF证书编号0368475CASC证书编号2020A069/四川永星目前有二筛条件已经拿到CANS认证可以对电阻器进行二筛。',
'人员情况': '893厂现有员工900余人在全国设立6个办事处/云科成都办事处有五人,负责成都、绵阳、重庆,对接凯天的是罗晓波',
'产品种类(新研制产品、主推产品)': '云科已申请RN5161系列进入SAST-G目录目前尚未完成进目录工作。/云科2022年薄膜电路产品上市。',
'近三年销售情况': '2021年云科销售浆料8亿元/1、云科一年在隆盛的电阻销售额在200万左右保险丝的销售额也在200万左右。2、云科一季度在607所的销售额大概在150万左右',
'市场组织架构': '1、云科市场部新上任一位领导与九院13所领导副所长关系密切2、西安办事处新换办事处主任杨文建是振华集团领导的亲属3、云科北京片区今年更换领导是从振华富调过去的并且把原来负责九院的业务员吴明金撤了更换了新的业务员1、振华云科的河南地区新增一位业务员胡心圣。2、佰威达的老板是施广勤同时也是博威的老板。',
'市场策略': '1、外围进攻通过熔断器、射频等产品抢占常规电阻器份额例如中航工业金城南京机电液压工程研究中心云科的片阻已有少量供货。2、盯住高价值产品抢占我们高价值产品合金箔、熔断器、射频的最主要客户的市场份额例如九院16所、兵器212所但目前仍未得手。低价策略风华和火炬今年想以低价策略进驻抢占份额。目前普军级F精度报价0.5J精度报价0.2。',
'竞争对手变化产生的市场机会': '893将一部分股份卖给北海银河银河快倒闭时改行做医药行业不景气',
'其他': '请输入其他补充信息',
'质量问题': '在核九院五所云科产品出现批次性硫化问题云科及时处理用抗硫化产品做了替代。893在梅岭说我司08年质量问题基础采购员都知道了这个情况说不敢选择我司火工品。目前常规片式产品普军等级云科与我司均为8毛893报6毛未选择他们。',
'重大市场活动': '云科在航天五院举办党建活动',
'市场占有率': '云科在航天领域的市场占有率为30%信息来源是XXX)/振华云科在航天航空的份额占比大概在60%,大都基于领导层的关系基础,几乎不会来司拜访用户。',
'与客户人员的关系': '振华集团高层跟118厂高层关系很好有时会来上海组织联谊活动/青岛整流器制造有限公司有云科内线,了解我公司价格。',
'竞争动向': '中电科技集团重庆声光电有限公司的106项目893中标/云科最近一次竞标常规国军标0.55元/只单一规格用量60万只)D精度厚膜0.9元/只(2个规格共20万只),通过这次竞标得到三个单一规格电阻的采购。'
}
return placeholderMap[formData.value.competitorLevelTwo] || '请输入具体信息'
})
const competitiveUnitsIndex = ref(0)
// 当前选中的信息类型
const competitorLevelOneIndex = ref(0)
const competitorLevelOne = ref('基本信息')
// 当前选中的具体分类索引
const competitorLevelTwoIndex = ref(0)
// 当前显示的具体分类选项数组(动态切换 array2 / array3
const currentCompetitorLevelTwoArray = ref(array2.value)
// 当前选中的具体分类名称
const competitorLevelTwo = ref('')
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onCompetitiveUnitsChange = (e) => {
competitiveUnitsIndex.value = e.detail.value
console.log('competitiveUnitsIndex:', array.value[e.detail.value]?.name)
formData.value.competitiveUnits = array.value[e.detail.value]?.name || ''
}
const onCompetitorLevelOneChange = (e) => {
if (e.detail.value >= 0 && e.detail.value < array1.value.length) {
competitorLevelOneIndex.value = e.detail.value
} else {
console.error('选择了无效的索引,重置为 0')
competitorLevelOneIndex.value = 0
}
const selectedType = array1.value[e.detail.value]?.name
console.log('选中的信息类型:', selectedType)
formData.value.competitorLevelOne = selectedType
if (selectedType === '基本信息') {
currentCompetitorLevelTwoArray.value = array2.value
competitorLevelTwoIndex.value = 0
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[0]?.name || ''
} else if (selectedType === '业务信息') {
currentCompetitorLevelTwoArray.value = array3.value
competitorLevelTwoIndex.value = 0
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[0]?.name || ''
} else {
currentCompetitorLevelTwoArray.value = []
formData.value.competitorLevelTwo = ''
}
}
const onCompetitorLevelTwoChange = (e) => {
competitorLevelTwoIndex.value = e.detail.value
formData.value.competitorLevelTwo = currentCompetitorLevelTwoArray.value[e.detail.value]?.name || ''
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete(formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,411 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场信息查看'" :leftFlag="true" :rightFlag="true">
<template #right>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height" :style="{ paddingTop: navBarPaddingTop + 'px' }"></view>
<!-- 正文内容 -->
<view class="all-body">
<!-- 搜索 @blur="blur" @focus="focus" @input="input" @cancel="cancel" @clear="clear"-->
<!-- <view class="search">
<uni-search-bar class="custom-search" radius="28" placeholder="请输入客户名称" clearButton="auto"
cancelButton="none" bgColor="#6FA2F8" textColor="#ffffff" v-model="searchValue" />
<button type="default" @click="handleSearch" size="mini" class="btn-search">查询</button>
</view> -->
<view class="search_center">
<view class="category">
<view class="flex_row_center_center" @click="showCate">
<text>{{selCategory.val||'全部'}}</text>
<image src="@/static/images/icon-notice@2x.png" mode=""></image>
</view>
<view class="pop_arrow" v-if="maskShow"></view>
<view class="pop" v-if="maskShow">
<text v-for="(item,index) in categories" :key="index"
@click="toList(item.categoryId,item.categoryName)">{{item.categoryName}}</text>
</view>
</view>
<image class="search_icon" src="@/static/images/icon-notice@2x.png"></image>
<input class='sea_input' :focus="inputFocus" type='text' :value="inputval" placeholder="请输入搜索条件"
@input="inputChange" @confirm='search' maxlength="50" placeholder-class="placeClass"></input>
<image class='clear_content' v-show="inputval" @click="clearInputVal"
src="@/static/images/icon-notice@2x.png" />
</view>
<text class='sea_btn' @click="btnSearch(1)">{{'搜索'}}</text>
<!-- 分页部分 -->
<mescroll-uni ref="mescrollRef" @init="mescrollInit" @down="downCallback" @up="upCallback"
:up="upOption" :down="downOption" :fixed="false" textColor="#ffffff" bgColor="#ffffff"
class="scroll-h" :class="{ 'loading-scroll': cssFlag }">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="showDetail(item)">
<view>
<view class="report-list">
<view class="title">信息类型:{{ item.informationType }}</view>
<view class="r-list">
<view class="r-name"v-if="item.cusName!=null">公司名称:{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="item.myselfBrowsing==0?'btn-blue':'btn-green'"
size="mini">{{item.myselfBrowsing==0?'未读':'已读'}}</view>
</view>
<view class="border-bottom"></view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">报告人</view>
<view class="r-right">{{ item.createName }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">报告日期</view>
<view class="r-right">{{ item.createTime }}</view>
</view>
</view>
</view>
</view>
</mescroll-uni>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
watch
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import MescrollUni from 'mescroll-uni/mescroll-uni.vue';
import {
getNavBarPaddingTop
} from '@/utils/system.js'
import {
visitorReportList
} from '@/api/business.js'
import {
viewingMarketInfForAllMembers
} from '@/api/crm/api_ys.js';
import {
onLoad,
onShow,
onUnload
} from '@dcloudio/uni-app'
// 获取导航栏高度用于内容区域padding
const navBarPaddingTop = ref(0);
onMounted(() => {
navBarPaddingTop.value = getNavBarPaddingTop() * 2;
uni.$on('updateStatus', markVisited)
})
// 查询列表
let list = ref([]);
let searchValue = ref(null)
//监视查询的内容的变化
watch(searchValue, (newValue, oldValue) => {
//变化了之后,重新查询内容
var data = {
pageNum: 1,
pageSize: 10,
searchContent: searchValue.value
};
viewingMarketInfForAllMembers(data).then(res => {
if (res.code == 200) {
//设置列表数据
list.value = res.rows;
}
})
})
const index = ref(0)
const categories = ref([{
categoryId: 1,
categoryName: '市场机会'
}, {
categoryId: 2,
categoryName: '重大事项信息'
}, {
categoryId: 3,
categoryName: '竞争对手信息'
}, {
categoryId: 4,
categoryName: '人员变化信息'
}, {
categoryId: 5,
categoryName: '重点型号任务信息'
}, {
categoryId: 6,
categoryName: '通用信息'
}])
const selCategory = ref ({
id: 1,
val: ''
})
// 查询搜索跳转
let handleSearch = () => {
var data = {
pageNum: 1,
pageSize: 10,
searchContent: searchValue.value
};
viewingMarketInfForAllMembers(data).then(res => {
if (res.code == 200) {
//设置列表数据
list.value = res.rows;
} else {
uni.showToast({
icon: 'none',
title: "请求失败",
});
list.value = null;
}
})
console.log(searchValue.value)
}
const mescrollRef = ref(null);
const upOption = ref({
page: {
num: 0,
size: 10
},
noMoreSize: 5,
empty: {
tip: '~ 空空如也 ~'
},
textLoading: '加载中...',
textNoMore: '已经到底了'
});
const downOption = ref({
auto: true,
textInOffset: '下拉刷新',
textOutOffset: '释放更新',
textLoading: '刷新中...'
});
let cssFlag = ref(false); //控制样式
const mescrollInit = (mescroll) => {
cssFlag.value = true;
mescrollRef.value = mescroll;
};
// 下拉刷新
const downCallback = async (mescroll) => {
try {
setTimeout(async () => {
const res = await getViewingMarketInfForAllMembers(1, upOption.value.page.size);
cssFlag.value = false;
list.value = res.list;
mescroll.resetUpScroll();
}, 500);
} catch (error) {
mescroll.endErr();
} finally {
setTimeout(async () => {
mescroll.endSuccess();
}, 500);
}
}
// 上拉加载更多
const upCallback = async (mescroll) => {
try {
setTimeout(async () => {
const res = await getViewingMarketInfForAllMembers(mescroll.num, mescroll.size);
if (mescroll.num === 1) {
list.value = res.list;
} else {
list.value.push(...res.list);
}
mescroll.endBySize(res.list.length, res.total);
}, 500);
} catch (error) {
mescroll.endErr();
}
}
// 获取数据列表
const getViewingMarketInfForAllMembers = (pageNum, pageSize) => {
return new Promise(async (resolve) => {
let param = {
pageNum,
pageSize
}
let res = await viewingMarketInfForAllMembers(param);
resolve({
list: res.rows,
total: res.total
});
});
}
let Loop = ref(0)
let now
const visistId = ref();
const cusName = ref();
const cusId = ref();
const status = ref()
function showDetail(item) {
uni.navigateTo({
url: "/pages/business/CRM/marketInformation/informationDetail?informationId=" + item.informationId + '&selValue=' + item.selValue+ '&inputval=' + item.inputval
})
}
onUnload(() => {
uni.$off('updateStatus')
})
const markVisited = (informationId) => {
const newList = [...list.value].map(item => {
if (item.informationId == informationId) {
return { ...item, myselfBrowsing: 1 };
}
return item;
});
list.value = newList;
};
</script>
<style scoped>
.all-body {
/* #ifdef APP-PLUS */
top: 150rpx;
height: calc(100vh - 75px);
/* #endif */
/* #ifndef APP-PLUS */
top: 120rpx;
height: calc(100vh);
/* #endif */
}
.search {
display: flex;
}
.search .btn-search {
border: none;
background: none;
line-height: normal;
color: #fff;
line-height: 56rpx !important;
padding: 10rpx 0 0;
text-align: left;
cursor: pointer;
}
.search .btn-search::after {
display: none;
}
.search .custom-search {
width: 80%;
}
.search .custom-search.uni-searchbar {
padding-right: 0 !important;
}
.scroll-h {
/* #ifdef APP-PLUS */
height: calc(100vh - 120px);
/* #endif */
/* #ifndef APP-PLUS */
height: calc(100vh - 110px);
/* #endif */
}
.white-bg {
padding-bottom: 10rpx;
}
.search_center {
display: flex;
align-items: center;
border: none;
flex: 1;
height: 70rpx;
margin-left: 20rpx;
padding-left: 20rpx;
border-radius: 32.5rpx;
background-color: #f5f5f5;
.search_icon {
width: 30rpx;
height: 30rpx;
margin-right: 22rpx;
}
.category {
display: flex;
align-items: center;
margin-right: 16rpx;
padding-right: 16rpx;
border-right: 2rpx solid #D4D4D4;
position: relative;
text {
font-size: 13px;
margin-left: 10rpx;
margin-right: 10rpx;
}
image {
margin-top: 4rpx;
width: 20rpx;
height: 11rpx;
}
.pop {
position: absolute;
top: 50rpx;
background: #FFFFFF;
box-shadow: 0px 0px 40rpx 0px rgba(59, 59, 59, 0.2);
border-radius: 12rpx;
display: flex;
flex-direction: column;
align-items: center;
padding: 20rpx;
z-index: 100;
max-height: 566rpx;
overflow-y: auto;
text {
font-size: 28rpx;
font-family: PingFang SC;
font-weight: 500;
color: #333333;
white-space: nowrap;
margin: 10rpx 0;
line-height: 56rpx;
}
}
.pop_arrow {
position: absolute;
top: 48rpx;
z-index: 999;
&::after {
position: absolute;
content: "";
left: 68rpx;
top: -26rpx;
border: 14rpx solid #fff;
border-color: transparent transparent #fff transparent;
}
}
}
}
</style>

View File

@@ -0,0 +1,288 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场信息详情'" :leftFlag="true" :rightFlag="false"></customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<!-- 详情内容 -->
<view class="white-bg">
<view class="report-list">
<view class="title">{{ detailList.informationType }}</view>
<view class="r-list">
<view class="r-name">{{ detailList.cusName }}</view>
</view>
<view class="border-bottom b-width"></view>
<view class="r-list">
<view class="r-left">报告人</view>
<view class="r-right">{{ detailList.createName }}</view>
</view>
<view class="border-bottom b-width"></view>
<view class="r-list">
<view class="r-left">创建日期</view>
<view class="r-right">{{ detailList.createTime }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.createTime != null"></view>
<view class="r-list" v-if="detailList.createTime != null">
<view class="r-left">创建时间</view>
<view class="r-right">{{ detailList.createTime }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityType != null"></view>
<view class="r-list" v-if="detailList.opportunityType != null">
<view class="r-left">机会类型</view>
<view class="r-right">{{ detailList.opportunityType }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.understandTheWay != null"></view>
<view class="r-list" v-if="detailList.understandTheWay != null">
<view class="r-left">了解途径</view>
<view class="r-right">{{ detailList.understandTheWay }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityDescription != null"></view>
<view class="r-list" v-if="detailList.opportunityDescription != null">
<view class="r-left">机会描述</view>
<view class="r-right">{{ detailList.opportunityDescription }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityStatus != null"></view>
<view class="r-list" v-if="detailList.opportunityStatus != null">
<view class="r-left">机会所处状态</view>
<view class="r-right">{{ detailList.opportunityStatus }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.predictedAmount != null"></view>
<view class="r-list" v-if="detailList.predictedAmount != null">
<view class="r-left">预测金额或情况</view>
<view class="r-right">{{ detailList.predictedAmount }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitionSituation != null"></view>
<view class="r-list" v-if="detailList.competitionSituation != null">
<view class="r-left">竞争情况</view>
<view class="r-right">{{ detailList.competitionSituation }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.majorTypesOfMatters != null"></view>
<view class="r-list" v-if="detailList.majorTypesOfMatters != null">
<view class="r-left">重大事项类型</view>
<view class="r-right">{{ detailList.majorTypesOfMatters }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.theImpactGenerated != null"></view>
<view class="r-list" v-if="detailList.theImpactGenerated != null">
<view class="r-left">产生的影响</view>
<view class="r-right">{{ detailList.theImpactGenerated }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.companyResponseStrategy != null"></view>
<view class="r-list" v-if="detailList.companyResponseStrategy != null">
<view class="r-left">公司应对策略</view>
<view class="r-right">{{ detailList.companyResponseStrategy }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitiveUnits != null"></view>
<view class="r-list" v-if="detailList.competitiveUnits != null">
<view class="r-left">竞争单位</view>
<view class="r-right">{{ detailList.competitiveUnits }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitorLevelOne != null"></view>
<view class="r-list" v-if="detailList.competitorLevelOne != null">
<view class="r-left">信息类型</view>
<view class="r-right">{{ detailList.competitorLevelOne }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitorLevelTwo != null"></view>
<view class="r-list" v-if="detailList.competitorLevelTwo != null">
<view class="r-left">具体分类</view>
<view class="r-right">{{ detailList.competitorLevelTwo }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.specificMatters != null"></view>
<view class="r-list" v-if="detailList.specificMatters != null">
<view class="r-left">具体事情</view>
<view class="r-right">{{ detailList.specificMatters }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.originalPosition != null"></view>
<view class="r-list" v-if="detailList.originalPosition != null">
<view class="r-left">原职务</view>
<view class="r-right">{{ detailList.originalPosition }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.currentPosition != null"></view>
<view class="r-list" v-if="detailList.currentPosition != null">
<view class="r-left">现职务</view>
<view class="r-right">{{ detailList.currentPosition }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.positionOfOurCompany != null"></view>
<view class="r-list" v-if="detailList.positionOfOurCompany != null">
<view class="r-left">现职务是否与我公司业务相关</view>
<view class="r-right">{{ detailList.positionOfOurCompany }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.developmentEfforts != null"></view>
<view class="r-list" v-if="detailList.developmentEfforts != null">
<view class="r-left">攻关力度</view>
<view class="r-right">{{ detailList.developmentEfforts }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.successor != null"></view>
<view class="r-list" v-if="detailList.successor != null">
<view class="r-left">继任者</view>
<view class="r-right">{{ detailList.successor }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.keyModels != null"></view>
<view class="r-list" v-if="detailList.keyModels != null">
<view class="r-left">重点型号</view>
<view class="r-right">{{ detailList.keyModels }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.currentStatus != null"></view>
<view class="r-list" v-if="detailList.currentStatus != null">
<view class="r-left">目前状态</view>
<view class="r-right">{{ detailList.currentStatus }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.batchProductionPlan != null"></view>
<view class="r-list" v-if="detailList.batchProductionPlan != null">
<view class="r-left">批产计划</view>
<view class="r-right">{{ detailList.batchProductionPlan }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.situation != null"></view>
<view class="r-list" v-if="detailList.situation != null">
<view class="r-left">外协外包上级单位情况</view>
<view class="r-right">{{ detailList.situation }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.subordinateSupportingUnits != null"></view>
<view class="r-list" v-if="detailList.subordinateSupportingUnits != null">
<view class="r-left">下级配套单位</view>
<view class="r-right">{{ detailList.subordinateSupportingUnits }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.generalFormType != null"></view>
<view class="r-list" v-if="detailList.generalFormType != null">
<view class="r-left">通用表单类型</view>
<view class="r-right">{{ detailList.generalFormType }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.informationContent != null"></view>
<view class="r-list" v-if="detailList.informationContent != null">
<view class="r-left">信息内容</view>
<view class="r-right">{{ detailList.informationContent }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.remark != null"></view>
<view class="r-list" v-if="detailList.remark != null">
<view class="r-left">备注</view>
<view class="r-right">{{ detailList.remark }}</view>
</view>
<view class="border-bottom b-width"></view>
</view>
</view>
<!-- 底部加高度来避免tabbar遮挡 -->
<view class="bottom-height"></view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted
} from 'vue'
import {
onLoad,
onShow
} from '@dcloudio/uni-app'
import customHeader from '@/components/customHeader.vue'
import customTabs from '@/components/customTabs.vue';
import {
viewingMarketInfgetDetail
} from '@/api/crm/api_ys.js';
let informationId = ref(0)
// 加载后调用
let id = ref(null)
onLoad((options) => {
informationId.value = options.informationId
getVisitorReportDetail();
})
// 查询详情
let item = ref({});
//明细List
let detailList = ref({})
const getVisitorReportDetail = async () => {
let param = {
informationId: informationId.value
}
let detailRes = await viewingMarketInfgetDetail(param);
detailList.value = detailRes.rows[0];
uni.$emit('updateStatus',informationId.value);
}
</script>
<style scoped>
.white-bg {
width: 690rpx;
margin: 0;
border-radius: 8px 8px 0 0;
}
.white-bg.white-bg-2 {
border-radius: 0;
margin-bottom: 20rpx;
}
:deep(.tabs-header) {
/* background: none !important; */
border-bottom: none !important;
margin: 0 auto;
gap: 5px;
}
:deep(.tab-item) {
color: #919191;
font-size: 32rpx;
font-weight: bold;
/* flex:none; */
/* margin: 0 -50rpx; */
/* padding: 0 50rpx; */
}
:deep(.tab-item:first-child) {
text-align: right;
margin-left: 230rpx;
}
:deep(.tab-item:last-child) {
text-align: left;
margin-right: 230rpx;
}
:deep(.tab-item.active) {
color: #3384DF;
font-weight: bold;
}
:deep(.tab-item.active::after) {
width: 100rpx;
height: 8rpx;
}
</style>

View File

@@ -0,0 +1,232 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'重点型号任务信息添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="重点型号" name="keyModels" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.keyModels"
placeholder="请输入重点型号" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="目前状态" name="currentStatus" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.currentStatus"
placeholder="请输入目前状态" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="批产计划" name="batchProductionPlan" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.batchProductionPlan"
placeholder="请输入批产计划" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="外协、外包、上级单位情况" name="situationOfSuperiorUnits" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.situationOfSuperiorUnits"
placeholder="请输入外协、外包、上级单位情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="下级配套单位" name="lowerLevelSupportingUnits" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.lowerLevelSupportingUnits"
placeholder="请输入下级配套单位" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd
} from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
keyModels: null,//重点型号
currentStatus: null,//目前状态
batchProductionPlan: null,//批产计划
situationOfSuperiorUnits: null,//外协、外包、上级单位情况
lowerLevelSupportingUnits: null,//下级配套单位
remark: "", //、备注
informationType:"重点型号任务信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
keyModels: {
rules: [{
required: true,
errorMessage: '请输入重点型号'
}]
},
currentStatus: {
rules: [{
required: true,
errorMessage: '请输入目前状态'
}]
},
batchProductionPlan: {
rules: [{
required: true,
errorMessage: '请输入批产计划'
}]
},
situationOfSuperiorUnits: {
rules: [{
required: true,
errorMessage: '请输入外协、外包、上级单位情况'
}]
},
lowerLevelSupportingUnits: {
rules: [{
required: true,
errorMessage: '请输入下级配套单位'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
console.log('表单数据:', formData.value)
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,271 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'重点型号任务信息修改'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="重点型号" name="keyModels" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.keyModels"
placeholder="请输入重点型号" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="目前状态" name="currentStatus" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.currentStatus"
placeholder="请输入目前状态" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="批产计划" name="batchProductionPlan" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.batchProductionPlan"
placeholder="请输入批产计划" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="外协、外包、上级单位情况" name="situationOfSuperiorUnits" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.situationOfSuperiorUnits"
placeholder="请输入外协、外包、上级单位情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="下级配套单位" name="lowerLevelSupportingUnits" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.lowerLevelSupportingUnits"
placeholder="请输入下级配套单位" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete
} from '@/api/crm/api_ys.js'
import { onLoad } from '@dcloudio/uni-app'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
keyModels: null,//重点型号
currentStatus: null,//目前状态
batchProductionPlan: null,//批产计划
situationOfSuperiorUnits: null,//外协、外包、上级单位情况
lowerLevelSupportingUnits: null,//下级配套单位
remark: "", //、备注
informationType:"重点型号任务信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
keyModels: {
rules: [{
required: true,
errorMessage: '请输入重点型号'
}]
},
currentStatus: {
rules: [{
required: true,
errorMessage: '请输入目前状态'
}]
},
batchProductionPlan: {
rules: [{
required: true,
errorMessage: '请输入批产计划'
}]
},
situationOfSuperiorUnits: {
rules: [{
required: true,
errorMessage: '请输入外协、外包、上级单位情况'
}]
},
lowerLevelSupportingUnits: {
rules: [{
required: true,
errorMessage: '请输入下级配套单位'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete(formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,267 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'重大事项添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="重大事项类型" name="opportunityType" class="f-c-right">
<picker @change="onOpportunityTypeChange" :value="opportunityTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[opportunityTypeIndex]?.name || '请选择重大事项类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="了解途径" name="understandTheWay" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.understandTheWay"
placeholder="请输入了解途径" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="产生的影响" name="theImpactGenerated" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.theImpactGenerated"
placeholder="请输入产生的影响" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="公司应对策略" name="companyResponseStrategy" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.companyResponseStrategy"
placeholder="请输入公司应对策略" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd
} from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
majorTypesOfMatters: "", //重大事项类型
understandTheWay: "", //了解途径
theImpactGenerated: "", //、产生的影响
companyResponseStrategy: "", //、公司应对策略
remark: "", //、备注
picture: "", //、图片
photos: null,
informationType: "重大事项信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
majorTypesOfMatters: {
rules: [{
required: true,
errorMessage: '请选择重大事项类型'
}]
},
understandTheWay: {
rules: [{
required: true,
errorMessage: '请输入了解途径'
}]
},
theImpactGenerated: {
rules: [{
required: true,
errorMessage: '请输入产生的影响'
}]
},
companyResponseStrategy: {
rules: [{
required: true,
errorMessage: '请输入公司应对策略'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '质量问题'
},
{
id: 1,
name: '客户重大变更'
},
{
id: 2,
name: '行业重大事项'
}
])
const form = ref({
cusId: null,
cusName: null,
opportunityType: "", // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
const opportunityTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
console.log('表单数据:', formData.value)
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
// 如果你原来在 onShow 中做了类似这样:
// let res = currPage.data.cusData; 判断是否传入了客户信息
// 那么在 Vue3 中通常是通过路由参数或者 Vuex/Pinia 等状态管理获取
// 暂时不做,如你后续需要可继续补充
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,304 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'重大事项修改'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="重大事项类型" name="opportunityType" class="f-c-right">
<picker @change="onOpportunityTypeChange" :value="opportunityTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[opportunityTypeIndex]?.name || '请选择重大事项类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="了解途径" name="understandTheWay" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.understandTheWay"
placeholder="请输入了解途径" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="产生的影响" name="theImpactGenerated" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.theImpactGenerated"
placeholder="请输入产生的影响" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="公司应对策略" name="companyResponseStrategy" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.companyResponseStrategy"
placeholder="请输入公司应对策略" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import { onLoad } from '@dcloudio/uni-app'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete
} from '@/api/crm/api_ys.js'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
majorTypesOfMatters: "", //重大事项类型
understandTheWay: "", //了解途径
theImpactGenerated: "", //、产生的影响
companyResponseStrategy: "", //、公司应对策略
remark: "", //、备注
picture: "", //、图片
photos: null,
informationType: "重大事项信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
majorTypesOfMatters: {
rules: [{
required: true,
errorMessage: '请选择重大事项类型'
}]
},
understandTheWay: {
rules: [{
required: true,
errorMessage: '请输入了解途径'
}]
},
theImpactGenerated: {
rules: [{
required: true,
errorMessage: '请输入产生的影响'
}]
},
companyResponseStrategy: {
rules: [{
required: true,
errorMessage: '请输入公司应对策略'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '质量问题'
},
{
id: 1,
name: '客户重大变更'
},
{
id: 2,
name: '行业重大事项'
}
])
const form = ref({
cusId: null,
cusName: null,
opportunityType: "", // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
const opportunityTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete (formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,497 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场信息管理'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="handleAdd">
<uni-icons type="plus" size="24" color="#B7D2FF"></uni-icons>新增
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height" :style="{ paddingTop: navBarPaddingTop + 'px' }"></view>
<!-- 正文内容 -->
<view class="all-body market-con">
<view class="nav-list">
<view class="nav-item" :class="{active:index==activeTab}" v-for="(item,index) in tabList"
:key="index" @click="handleNav(index)">{{ item }}</view>
</view>
<!-- 分页部分 -->
<mescroll-uni ref="mescrollRef" @init="mescrollInit" @down="downCallback" @up="upCallback"
:up="upOption" :down="downOption" :fixed="false" textColor="#ffffff" bgColor="#ffffff"
class="scroll-h" :class="{'loading-scroll':cssFlag}">
<!-- 市场机会信息 -->
<block v-if="activeTab === 0">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index"
@click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">机会类型:{{item.opportunityType}}
</view>
<view class="font-bold" style="padding-bottom:10rpx">
机会所处状态:{{item.opportunityStatus}}</view>
<!-- <view class="font-gray">{{ item.desc }}</view>灰色样式 -->
</view>
</view>
</view>
</block>
<!-- 重大事项信息 -->
<block v-if="activeTab === 1">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">
产生的影响:{{item.theImpactGenerated}}</view>
<view class="font-bold" style="padding-bottom:10rpx">
重大事项类型:{{item.opportunityType}}</view>
<!-- <view class="font-gray">{{ item.desc }}</view> -->
</view>
</view>
</view>
</block>
<!-- 竞争对手信息 -->
<block v-if="activeTab === 2">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.competitiveUnits }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">
竞争对手一级标签:{{item.competitorLevelOne}}</view>
<view class="font-bold" style="padding-bottom:10rpx">
竞争对手二级标签:{{item.competitorLevelTwo}}</view>
<!-- <view class="font-gray">{{ item.desc }}</view> -->
</view>
</view>
</view>
</block>
<!-- 人员变化信息 -->
<block v-if="activeTab === 3">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">客户人员名称:{{item.cusUserName}}
</view>
<view class="font-bold" style="padding-bottom:10rpx">原职务:{{item.originalPosition}}
</view>
<view class="font-bold" style="padding-bottom:10rpx">现职务:{{item.currentPosition}}
</view>
<!-- <view class="font-gray">{{ item.desc }}</view> -->
</view>
</view>
</view>
</block>
<!-- 重点型号任务信息 -->
<block v-if="activeTab === 4">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">重点型号:{{item.keyModels}}</view>
<view class="font-bold" style="padding-bottom:10rpx">目前状态:{{item.currentStatus}}
</view>
<view class="font-gray">{{ item.desc }}</view>
</view>
</view>
</view>
</block>
<!-- 通用表单 -->
<block v-if="activeTab === 5">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="itemClick(item)">
<view class="report-list">
<view class="title r-list">
<view class="r-left" style="font-size:38rpx;">{{ item.cusName }}</view>
<view class="r-right btn-gray flex-auto" :class="{'btn-blue':item.status==2}"
size="mini" @click.stop="itemView(item)">{{ item.state }}</view>
</view>
<view style="padding:0rpx 0 10rpx">
<view class="font-bold" style="padding-bottom:10rpx">标签:{{item.generalFormType}}
</view>
<view class="font-gray">内容:{{ item.informationContent }}</view>
</view>
</view>
</view>
</block>
</mescroll-uni>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
onUnmounted
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import MescrollUni from 'mescroll-uni/mescroll-uni.vue';
import {
getNavBarPaddingTop
} from '@/utils/system.js'
import {
mattersList
} from '@/api/business.js'
import {
CrmMarketInformationList,
handleCrmMarketInfo
} from '@/api/crm/api_ys.js'
// 获取导航栏高度用于内容区域padding
const navBarPaddingTop = ref(0);
onMounted(() => {
navBarPaddingTop.value = getNavBarPaddingTop() * 2;
uni.$on('refreshMarketList', handleRefreshList);
})
onUnmounted(() => {
// 移除事件监听
uni.$off('refreshMarketList', handleRefreshList);
});
const mescrollInstance = ref(null);
const activeTab = ref(0); // 默认市场机会
const tabList = ['市场机会', '重大事项信息', '竞争对手信息', '人员变化信息', '重点型号任务信息', '通用表单'];
let handleNav = (index) => {
console.log(index + " 切换列表");
activeTab.value = index;
// 先重置列表(可选,避免新旧数据混在一起)
list.value = [];
// 手动触发 mescroll 下拉刷新,重新加载当前 tab 的数据
if (mescrollInstance.value) {
mescrollInstance.value.triggerDownScroll(); // 关键!触发下拉刷新,会调用 downCallback
}
}
// 新增
let handleAdd = () => {
if (activeTab.value == 0) {
console.log("市场机会")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/marketOpportunities'
})
} else if (activeTab.value == 1) {
console.log("重大机会事项")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/majorMatter'
})
} else if (activeTab.value == 2) {
console.log("竞争对手信息")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/competitor'
})
} else if (activeTab.value == 3) {
console.log("人员变化信息")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/personnelChanges'
})
} else if (activeTab.value == 4) {
console.log("重点型号任务信息")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/keyModelTasks'
})
} else if (activeTab.value == 5) {
console.log("通用表单")
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/beCurrent'
})
} else {
console.log("其他?")
}
// uni.navigateTo({ url: '/pages/business/CRM/visitorReportAdd' })
}
// 查询列表
let list = ref([]);
const mescrollRef = ref(null);
const upOption = ref({
page: {
num: 0,
size: 10
},
noMoreSize: 5,
empty: {
tip: '~ 空空如也 ~'
},
textLoading: '加载中...',
textNoMore: '已经到底了'
});
const downOption = ref({
auto: true,
textInOffset: '下拉刷新',
textOutOffset: '释放更新',
textLoading: '刷新中...'
});
let cssFlag = ref(false); //控制样式
const mescrollInit = (mescroll) => {
cssFlag.value = true;
mescrollRef.value = mescroll;
mescrollInstance.value = mescroll;
};
// 下拉刷新
const downCallback = async (mescroll) => {
try {
const pageNo = 1;
const pageSize = upOption.value.size;
const type = activeTab.value + 1; // 当前tab索引
const description = tabList[activeTab.value]; // 当前tab文字
const res = await getCrmMarketInformationList(pageNo, pageSize, type, description);
list.value = res.list;
mescroll.resetUpScroll();
} catch (error) {
mescroll.endErr();
} finally {
mescroll.endSuccess();
}
}
// 上拉加载更多
const upCallback = async (mescroll) => {
try {
const pageNo = mescroll.num;
const pageSize = mescroll.size;
const type = activeTab.value; // 当前tab索引
const description = tabList[activeTab.value]; // 当前tab文字
const res = await getCrmMarketInformationList(pageNo, pageSize, type, description);
if (mescroll.num === 1) {
list.value = res.list;
} else {
list.value.push(...res.list);
}
mescroll.endBySize(res.list.length, res.total);
} catch (error) {
mescroll.endErr();
}
}
// 获取数据列表
const getCrmMarketInformationList = async (pageNo, pageSize, type, description) => {
const param = {
pageNo,
pageSize,
type,
description
};
const res = await CrmMarketInformationList(param);
return {
list: res.rows,
total: res.total
};
}
const itemView = async (item) => {
const state = (item.state || "").trim().toLowerCase()
if (state === "待提交" || state === "驳回") {
uni.showModal({
title: '操作确认',
content: '是否为商业航天行业板块信息?',
confirmText: '是',
cancelText: '否',
success: async (res) => {
if (res.confirm) {
item.commercialAerospace = "是"
try {
const res = await handleCrmMarketInfo(item)
if (res.code == 200) {
uni.showToast({
title: '提交成功',
duration: 2000
})
handleRefreshList();
} else {
uni.showToast({
title: res.msg || '操作失败',
icon: 'none'
})
}
} catch {
uni.showToast({
title: '提交失败,请重试',
icon: 'none'
})
}
} else if (res.cancel) {
item.commercialAerospace = "否"
try {
const res = await handleCrmMarketInfo(item)
if (res.code == 200) {
uni.showToast({
title: '提交成功',
duration: 2000
})
handleRefreshList();
} else {
uni.showToast({
title: res.msg || '操作失败',
icon: 'none'
})
}
} catch {
uni.showToast({
title: '提交失败,请重试',
icon: 'none'
})
}
}
}
})
} else {
uni.showToast({
title: "当前状态不可操作",
icon: "none",
duration: 2000
})
}
}
const itemClick = async (item) => {
if(item.informationType=='市场机会'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/marketOpportunitiesDetail?informationId='+ item.informationId
})
}else if(item.informationType=='重大事项信息'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/majorMatterDetail?informationId='+ item.informationId
})
}else if(item.informationType=='竞争对手信息'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/competitorDetail?informationId='+ item.informationId
})
}else if(item.informationType=='人员变化信息'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/personnelChangesDetail?informationId='+ item.informationId
})
}else if(item.informationType=='重点型号任务信息'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/keyModelTasksDetail?informationId='+ item.informationId
})
}else if(item.informationType=='通用表单'){
uni.navigateTo({
url: '/pages/business/CRM/marketInformation/beCurrentDetail?informationId='+ item.informationId
})
}
else{
console.log("其他")
}
// console.log(item)
// uni.navigateTo({ url: '/pages/business/CRM/visitorReportAdd' })
}
const handleRefreshList = () => {
if (mescrollInstance.value) {
mescrollInstance.value.triggerDownScroll();
}
};
</script>
<style scoped>
.all-body {
/* #ifdef APP-PLUS */
top: 88rpx;
height: calc(100vh - 44px);
/* #endif */
/* #ifndef APP-PLUS */
top: 100rpx;
height: calc(100vh - 48px);
/* #endif */
}
/*.market-con{
} */
.nav-list {
display: flex;
flex-wrap: wrap;
/* #ifdef APP-PLUS */
padding: 80rpx 30rpx 0;
/* #endif */
/* #ifndef APP-PLUS */
padding: 20rpx 30rpx 0;
/* #endif */
}
.nav-list .nav-item {
background-color: #05A3F4;
border-radius: 10rpx;
color: #FFFFFF;
font-size: 28rpx;
text-align: center;
padding: 10rpx 25rpx;
margin-right: 15rpx;
margin-bottom: 20rpx;
}
.nav-list .nav-item:nth-child(3n) {
margin-right: 0;
}
.nav-list .nav-item.active {
background-color: #fff;
color: #3384DF;
font-weight: bold;
}
.report-list .r-list {
align-items: flex-start;
}
.scroll-h {
/* #ifdef APP-PLUS */
height: calc(100vh - 155px);
/* #endif */
/* #ifndef APP-PLUS */
height: calc(100vh - 135px);
/* #endif */
}
.white-bg {
padding-top: 10rpx;
padding-bottom: 10rpx;
}
.report-list .title {
display: flex;
}
</style>

View File

@@ -0,0 +1,249 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场信息审核'" :leftFlag="true" :rightFlag="true">
<template #right>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height" :style="{ paddingTop: navBarPaddingTop + 'px' }"></view>
<!-- 正文内容 -->
<view class="all-body">
<!-- 搜索 @blur="blur" @focus="focus" @input="input" @cancel="cancel" @clear="clear"-->
<!-- <view class="search">
<uni-search-bar class="custom-search" radius="28" placeholder="请输入客户名称" clearButton="auto"
cancelButton="none" bgColor="#6FA2F8" textColor="#ffffff" v-model="searchValue" />
<button type="default" @click="handleSearch" size="mini" class="btn-search">查询</button>
</view> -->
<!-- 分页部分 -->
<mescroll-uni ref="mescrollRef" @init="mescrollInit" @down="downCallback" @up="upCallback"
:up="upOption" :down="downOption" :fixed="false" textColor="#ffffff" bgColor="#ffffff"
class="scroll-h" :class="{ 'loading-scroll': cssFlag }">
<view class="white-bg margin-bottom20" v-for="(item, index) in list" :key="index" @click="touchstart(item)">
<view>
<view class="report-list">
<view class="title">信息类型{{ item.informationType }}</view>
<view class="r-list">
<view class="r-name">公司名称{{ item.cusName }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">创建人</view>
<view class="r-right">{{ item.createName }}</view>
</view>
<view class="r-list">
<view class="r-left">创建时间</view>
<view class="r-right">{{ item.createTime }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">是否为商业航天类型</view>
<view class="r-name">{{ item.commercialAerospace }}</view>
</view>
</view>
</view>
</view>
</mescroll-uni>
</view>
</view>
</view>
</template>
<script setup>
import { ref, onMounted, watch,onUnmounted } from 'vue'
import customHeader from '@/components/customHeader.vue'
import MescrollUni from 'mescroll-uni/mescroll-uni.vue';
import { getNavBarPaddingTop } from '@/utils/system.js'
import { visitorReportList } from '@/api/business.js'
import { approvalMarketInfget } from '@/api/crm/api_ys.js';
import { onShow } from '@dcloudio/uni-app'
// 获取导航栏高度用于内容区域padding
const navBarPaddingTop = ref(0);
onMounted(() => {
navBarPaddingTop.value = getNavBarPaddingTop() * 2;
})
// 查询列表
let list = ref([]);
let searchValue = ref(null)
//监视查询的内容的变化
watch(searchValue, (newValue, oldValue) => {
//变化了之后,重新查询内容
var data = {
pageNum: 1,
pageSize: 10,
searchContent: searchValue.value
};
getYsVisistList(data).then(res => {
if (res.code == 200) {
//设置列表数据
list.value = res.rows;
}
})
})
// 新增
let handleAdd = () => {
uni.navigateTo({url: '/pages/business/CRM/paymentCollection/addPaymentCollection'})
}
const mescrollRef = ref(null);
const upOption = ref({
page: {num: 0, size: 10},
noMoreSize: 5,
empty: {tip: '~ 空空如也 ~'},
textLoading: '加载中...',
textNoMore: '已经到底了'
});
const downOption = ref({
auto: true,
textInOffset: '下拉刷新',
textOutOffset: '释放更新',
textLoading: '刷新中...'
});
let cssFlag = ref(false);//控制样式
const mescrollInit = (mescroll) => {
cssFlag.value = true;
mescrollRef.value = mescroll;
};
// 下拉刷新
const downCallback = async (mescroll) => {
try {
setTimeout(async () => {
const res = await ApprovalMarketInfget(1, upOption.value.page.size);
cssFlag.value = false;
list.value = res.list;
mescroll.resetUpScroll();
}, 500);
} catch (error) {
mescroll.endErr();
} finally {
setTimeout(async () => {
mescroll.endSuccess();
}, 500);
}
}
// 上拉加载更多
const upCallback = async (mescroll) => {
try {
setTimeout(async () => {
const res = await ApprovalMarketInfget(mescroll.num, mescroll.size);
if (mescroll.num === 1) {
list.value = res.list;
} else {
list.value.push(...res.list);
}
mescroll.endBySize(res.list.length, res.total);
}, 500);
} catch (error) {
mescroll.endErr();
}
}
// 获取数据列表
const ApprovalMarketInfget = (pageNum, pageSize) => {
return new Promise(async (resolve) => {
let param = {
pageNum,
pageSize
}
let res = await approvalMarketInfget(param);
resolve({
list: res.rows,
total: res.total
});
});
}
let Loop = ref(0)
function touchstart(item) {
uni.navigateTo({
url: "/pages/business/CRM/marketInformation/marketInformationReviewDetail?informationId=" + item.informationId
})
}
// 生命周期钩子
onShow(() => {
// 监听刷新事件
uni.$on('refreshList', ApprovalMarketInfget(1,10))
// 页面显示时也加载一次数据
ApprovalMarketInfget(1,10)
})
onUnmounted(() => {
// 移除监听(避免重复触发)
uni.$off('refreshList')
})
onMounted(() => {
ApprovalMarketInfget()
})
</script>
<style scoped>
.all-body {
/* #ifdef APP-PLUS */
top: 150rpx;
height: calc(100vh - 75px);
/* #endif */
/* #ifndef APP-PLUS */
top: 120rpx;
height: calc(100vh);
/* #endif */
}
.search {
display: flex;
}
.search .btn-search {
border: none;
background: none;
line-height: normal;
color: #fff;
line-height: 56rpx !important;
padding: 10rpx 0 0;
text-align: left;
cursor: pointer;
}
.search .btn-search::after {
display: none;
}
.search .custom-search {
width: 80%;
}
.search .custom-search.uni-searchbar {
padding-right: 0 !important;
}
.scroll-h {
/* #ifdef APP-PLUS */
height: calc(100vh - 120px);
/* #endif */
/* #ifndef APP-PLUS */
height: calc(100vh - 110px);
/* #endif */
}
.white-bg {
padding-bottom: 10rpx;
}
</style>

View File

@@ -0,0 +1,385 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场信息详情'" :leftFlag="true" :rightFlag="false"></customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<!-- 详情内容 -->
<view class="white-bg">
<view class="report-list">
<view class="title">{{ detailList.informationType }}</view>
<view class="r-list">
<view class="r-name">{{ detailList.cusName }}</view>
</view>
<view class="border-bottom b-width"></view>
<view class="r-list">
<view class="r-left">报告人</view>
<view class="r-right">{{ detailList.createName }}</view>
</view>
<view class="border-bottom b-width"></view>
<view class="r-list">
<view class="r-left">创建日期</view>
<view class="r-right">{{ detailList.createTime }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityType != null"></view>
<view class="r-list" v-if="detailList.opportunityType != null">
<view class="r-left">机会类型</view>
<view class="r-right">{{ detailList.opportunityType }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.understandTheWay != null"></view>
<view class="r-list" v-if="detailList.understandTheWay != null">
<view class="r-left">了解途径</view>
<view class="r-right">{{ detailList.understandTheWay }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityDescription != null"></view>
<view class="r-list" v-if="detailList.opportunityDescription != null">
<view class="r-left">机会描述</view>
<view class="r-right">{{ detailList.opportunityDescription }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.opportunityStatus != null"></view>
<view class="r-list" v-if="detailList.opportunityStatus != null">
<view class="r-left">机会所处状态</view>
<view class="r-right">{{ detailList.opportunityStatus }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.predictedAmount != null"></view>
<view class="r-list" v-if="detailList.predictedAmount != null">
<view class="r-left">预测金额或情况</view>
<view class="r-right">{{ detailList.predictedAmount }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitionSituation != null"></view>
<view class="r-list" v-if="detailList.competitionSituation != null">
<view class="r-left">竞争情况</view>
<view class="r-right">{{ detailList.competitionSituation }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.majorTypesOfMatters != null"></view>
<view class="r-list" v-if="detailList.majorTypesOfMatters != null">
<view class="r-left">重大事项类型</view>
<view class="r-right">{{ detailList.majorTypesOfMatters }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.theImpactGenerated != null"></view>
<view class="r-list" v-if="detailList.theImpactGenerated != null">
<view class="r-left">产生的影响</view>
<view class="r-right">{{ detailList.theImpactGenerated }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.companyResponseStrategy != null"></view>
<view class="r-list" v-if="detailList.companyResponseStrategy != null">
<view class="r-left">公司应对策略</view>
<view class="r-right">{{ detailList.companyResponseStrategy }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitiveUnits != null"></view>
<view class="r-list" v-if="detailList.competitiveUnits != null">
<view class="r-left">竞争单位</view>
<view class="r-right">{{ detailList.competitiveUnits }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitorLevelOne != null"></view>
<view class="r-list" v-if="detailList.competitorLevelOne != null">
<view class="r-left">信息类型</view>
<view class="r-right">{{ detailList.competitorLevelOne }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.competitorLevelTwo != null"></view>
<view class="r-list" v-if="detailList.competitorLevelTwo != null">
<view class="r-left">具体分类</view>
<view class="r-right">{{ detailList.competitorLevelTwo }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.specificMatters != null"></view>
<view class="r-list" v-if="detailList.specificMatters != null">
<view class="r-left">具体事情</view>
<view class="r-right">{{ detailList.specificMatters }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.originalPosition != null"></view>
<view class="r-list" v-if="detailList.originalPosition != null">
<view class="r-left">原职务</view>
<view class="r-right">{{ detailList.originalPosition }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.currentPosition != null"></view>
<view class="r-list" v-if="detailList.currentPosition != null">
<view class="r-left">现职务</view>
<view class="r-right">{{ detailList.currentPosition }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.positionOfOurCompany != null"></view>
<view class="r-list" v-if="detailList.positionOfOurCompany != null">
<view class="r-left">现职务是否与我公司业务相关</view>
<view class="r-right">{{ detailList.positionOfOurCompany }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.developmentEfforts != null"></view>
<view class="r-list" v-if="detailList.developmentEfforts != null">
<view class="r-left">攻关力度</view>
<view class="r-right">{{ detailList.developmentEfforts }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.successor != null"></view>
<view class="r-list" v-if="detailList.successor != null">
<view class="r-left">继任者</view>
<view class="r-right">{{ detailList.successor }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.keyModels != null"></view>
<view class="r-list" v-if="detailList.keyModels != null">
<view class="r-left">重点型号</view>
<view class="r-right">{{ detailList.keyModels }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.currentStatus != null"></view>
<view class="r-list" v-if="detailList.currentStatus != null">
<view class="r-left">目前状态</view>
<view class="r-right">{{ detailList.currentStatus }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.batchProductionPlan != null"></view>
<view class="r-list" v-if="detailList.batchProductionPlan != null">
<view class="r-left">批产计划</view>
<view class="r-right">{{ detailList.batchProductionPlan }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.situation != null"></view>
<view class="r-list" v-if="detailList.situation != null">
<view class="r-left">外协外包上级单位情况</view>
<view class="r-right">{{ detailList.situation }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.subordinateSupportingUnits != null"></view>
<view class="r-list" v-if="detailList.subordinateSupportingUnits != null">
<view class="r-left">下级配套单位</view>
<view class="r-right">{{ detailList.subordinateSupportingUnits }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.generalFormType != null"></view>
<view class="r-list" v-if="detailList.generalFormType != null">
<view class="r-left">通用表单类型</view>
<view class="r-right">{{ detailList.generalFormType }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.informationContent != null"></view>
<view class="r-list" v-if="detailList.informationContent != null">
<view class="r-left">信息内容</view>
<view class="r-right">{{ detailList.informationContent }}</view>
</view>
<view class="border-bottom b-width" v-if="detailList.remark != null"></view>
<view class="r-list" v-if="detailList.remark != null">
<view class="r-left">备注</view>
<view class="r-right">{{ detailList.remark }}</view>
</view>
<view class="border-bottom b-width"></view>
</view>
</view>
<!-- 底部加高度来避免tabbar遮挡 -->
<!-- <view class="bottom-height"></view> -->
<view class="tipsPopBtn">
<view class="btnCancal" @click="refuse">驳回</view>
<!-- <view class="btnComfire" @click="refusePass">通过不得分</view> -->
<view class="btnComfire" @click="adopt">通过</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted
} from 'vue'
import {
onLoad,
onShow
} from '@dcloudio/uni-app'
import customHeader from '@/components/customHeader.vue'
import customTabs from '@/components/customTabs.vue';
import {
viewingMarketInfgetDetail,
crmMarketInformationApprovalSuccess,
crmMarketInformationApprovalUnSuccess
} from '@/api/crm/api_ys.js';
let informationId = ref(0)
// 加载后调用
let id = ref(null)
onLoad((options) => {
informationId.value = options.informationId
getVisitorReportDetail();
})
// 查询详情
let item = ref({});
//明细List
let detailList = ref({})
const getVisitorReportDetail = async () => {
let param = {
informationId: informationId.value
}
let detailRes = await viewingMarketInfgetDetail(param);
detailList.value = detailRes.rows[0];
}
//通过审批
const adopt = async () => {
try {
const param = {
informationId: informationId.value
}
const res = await crmMarketInformationApprovalSuccess(param);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
});
// 操作完成后返回
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('提交失败:', err);
uni.showToast({
title: '提交失败',
icon: 'error'
});
}
}
// 修正后的驳回方法
const refuse = async () => {
uni.showModal({
title: '驳回原因',
content: '',
editable: true,
success: async function (modalRes) { // 使用modalRes避免重名
if (modalRes.confirm) {
try {
const param = {
informationId: informationId.value,
opinion: modalRes.content // 用户输入的驳回原因
};
const apiRes = await crmMarketInformationApprovalUnSuccess(param); // 避免与modalRes重名
console.log('驳回成功:', apiRes);
uni.showToast({
title: '驳回成功',
icon: 'success'
});
// 操作完成后返回
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('驳回失败:', err);
uni.showToast({
title: '驳回失败',
icon: 'error'
});
}
}
}
});
}
</script>
<style scoped>
.white-bg {
width: 690rpx;
margin: 0;
border-radius: 8px 8px 0 0;
}
.white-bg.white-bg-2 {
border-radius: 0;
margin-bottom: 20rpx;
}
.tipsPopBtn {
width: 300px;
display: flex;
justify-content: space-between;
align-items: center;
margin: 50px auto 56px auto;
bottom: 0;
left: 0;
}
.btnCancal {
width: 100px;
height: 60px;
border: 2px solid #29abe2;
border-radius: 40px;
display: flex;
justify-content: center;
align-items: center;
color: #29abe2;
font-size: 15px;
}
.btnComfire {
width: 100px;
height: 60px;
background-color: #29abe2;
border-radius: 40px;
display: flex;
justify-content: center;
align-items: center;
color: #FFF;
font-size: 15px;
}
:deep(.tabs-header) {
/* background: none !important; */
border-bottom: none !important;
margin: 0 auto;
gap: 5px;
}
:deep(.tab-item) {
color: #919191;
font-size: 32rpx;
font-weight: bold;
/* flex:none; */
/* margin: 0 -50rpx; */
/* padding: 0 50rpx; */
}
:deep(.tab-item:first-child) {
text-align: right;
margin-left: 230rpx;
}
:deep(.tab-item:last-child) {
text-align: left;
margin-right: 230rpx;
}
:deep(.tab-item.active) {
color: #3384DF;
font-weight: bold;
}
:deep(.tab-item.active::after) {
width: 100rpx;
height: 8rpx;
}
</style>

View File

@@ -0,0 +1,340 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场机会添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view
@click="chooseCustomer"
class="form-item-container"
>
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="机会类型" name="opportunityType" class="f-c-right">
<picker @change="onOpportunityTypeChange" :value="opportunityTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[opportunityTypeIndex]?.name || '请选择机会类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="了解途径" name="understandTheWay" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.understandTheWay"
placeholder="请输入了解途径" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="机会描述" name="opportunityDescription" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.opportunityDescription"
placeholder="请输入机会描述" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="机会所处状态" name="opportunityStatus" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.opportunityStatus"
placeholder="请输入机会所处状态" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="预测金额情况" name="predictedAmount" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.predictedAmount"
placeholder="请输入预测金额或情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="竞争情况" name="competitionSituation" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.competitionSituation"
placeholder="请输入竞争情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark"
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref, onMounted, reactive, onUnmounted, computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import { isEmpty } from '@/utils/validate.js'
import {crmMarketInformationAdd } from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
opportunityType: "", // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
opportunityType: {
rules: [{
required: true,
errorMessage: '请选择机会类型'
}]
},
understandTheWay: {
rules: [{
required: true,
errorMessage: '请输入了解途径'
}]
},
opportunityDescription: {
rules: [{
required: true,
errorMessage: '请输入机会描述'
}]
},
opportunityStatus: {
rules: [{
required: true,
errorMessage: '请输入机会所处状态'
}]
},
predictedAmount: {
rules: [{
required: true,
errorMessage: '请输入预测金额或情况'
}]
},
competitionSituation: {
rules: [{
required: true,
errorMessage: '请输入竞争情况'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '新产品需求'
},
{
id: 1,
name: '新客户开发'
},
{
id: 2,
name: '批产任务'
},
{
id: 3,
name: '新研任务'
},
{
id: 4,
name: '二筛服务'
},
{
id: 5,
name: '对手失利'
},
{
id: 6,
name: '其它'
}
])
const form = ref({
cusId: null,
cusName: null,
opportunityType: array.value[0].name, // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
// picker 选项
const opportunityTypeOptions = ref([{
id: 0,
name: '新产品需求'
},
{
id: 1,
name: '新客户开发'
},
{
id: 2,
name: '批产任务'
},
{
id: 3,
name: '新研任务'
},
{
id: 4,
name: '二筛服务'
},
{
id: 5,
name: '对手失利'
},
{
id: 6,
name: '其它'
}
])
const opportunityTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer(){
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
// 设置机会类型的默认值
formData.value.opportunityType = array.value[0].name;
opportunityTypeIndex.value = 0;
// 原有的监听客户选择事件
uni.$on('onCustomerSelected', handleCustomerSelected);
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
console.log("收到客户数据的值:", customerUser)
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
console.log('表单数据:', formData.value)
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
// 如果你原来在 onShow 中做了类似这样:
// let res = currPage.data.cusData; 判断是否传入了客户信息
// 那么在 Vue3 中通常是通过路由参数或者 Vuex/Pinia 等状态管理获取
// 暂时不做,如你后续需要可继续补充
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,369 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'市场机会修改'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view
@click="chooseCustomer"
class="form-item-container"
>
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="机会类型" name="opportunityType" class="f-c-right">
<picker @change="onOpportunityTypeChange" :value="opportunityTypeIndex" :range="array"
:range-key="'name'">
<view class="picker">
{{ array[opportunityTypeIndex]?.name || '请选择机会类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="了解途径" name="understandTheWay" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.understandTheWay"
placeholder="请输入了解途径" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="机会描述" name="opportunityDescription" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.opportunityDescription"
placeholder="请输入机会描述" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="机会所处状态" name="opportunityStatus" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.opportunityStatus"
placeholder="请输入机会所处状态" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="预测金额情况" name="predictedAmount" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.predictedAmount"
placeholder="请输入预测金额或情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="竞争情况" name="competitionSituation" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.competitionSituation"
placeholder="请输入竞争情况" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark"
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref, onMounted, reactive, onUnmounted, computed ,watch
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import {
getGuestList
} from '@/api/business.js'
import { isEmpty } from '@/utils/validate.js'
import { onLoad } from '@dcloudio/uni-app'
import {crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete } from '@/api/crm/api_ys.js'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
const message = useMessage();
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
opportunityType: "", // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
opportunityType: {
rules: [{
required: true,
errorMessage: '请选择机会类型'
}]
},
understandTheWay: {
rules: [{
required: true,
errorMessage: '请输入了解途径'
}]
},
opportunityDescription: {
rules: [{
required: true,
errorMessage: '请输入机会描述'
}]
},
opportunityStatus: {
rules: [{
required: true,
errorMessage: '请输入机会所处状态'
}]
},
predictedAmount: {
rules: [{
required: true,
errorMessage: '请输入预测金额或情况'
}]
},
competitionSituation: {
rules: [{
required: true,
errorMessage: '请输入竞争情况'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '新产品需求'
},
{
id: 1,
name: '新客户开发'
},
{
id: 2,
name: '批产任务'
},
{
id: 3,
name: '新研任务'
},
{
id: 4,
name: '二筛服务'
},
{
id: 5,
name: '对手失利'
},
{
id: 6,
name: '其它'
}
])
const form = ref({
cusId: null,
cusName: null,
opportunityType: "", // 机会类型
understandTheWay: "", // 了解途径
opportunityDescription: "", // 机会描述
opportunityStatus: "", // 机会所处状态
predictedAmount: "", // 预测金额或情况
competitionSituation: "", // 竞争情况
remark: "", // 备注
picture: "", // 图片
informationType: "市场机会" // 信息类型
})
// picker 选项
const opportunityTypeOptions = ref([{
id: 0,
name: '新产品需求'
},
{
id: 1,
name: '新客户开发'
},
{
id: 2,
name: '批产任务'
},
{
id: 3,
name: '新研任务'
},
{
id: 4,
name: '二筛服务'
},
{
id: 5,
name: '对手失利'
},
{
id: 6,
name: '其它'
}
])
const opportunityTypeIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer(){
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
console.log("收到客户数据的值:", customerUser)
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete (formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,337 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'人员变化添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="客户人员" name="cusUserName" class="f-c-right">
<uni-easyinput v-model="formData.cusUserName" placeholder="请选择客户人员" name="input"
@focus="chooseClientUser"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="原职务" name="originalPosition" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.originalPosition"
placeholder="请输入原职务" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="现职务" name="currentPosition" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.currentPosition"
placeholder="请输入现职务" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="现职务是否与我司业务相关" name="positionOfOurCompany" class="f-c-right">
<picker @change="onpoSitionOfOurCompanyChange" :value="positionOfOurCompanyIndex"
:range="array" :range-key="'name'">
<view class="picker">
{{ array[positionOfOurCompanyIndex]?.name || '请选择机会类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="公关力度" name="developmentEfforts" class="f-c-right">
<picker @change="onDevelopmentEffortsChange" :value="developmentEffortsIndex"
:range="array1" :range-key="'name'">
<view class="picker">
{{ array1[developmentEffortsIndex]?.name || '请选择公关力度' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="继任者" name="successor" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.successor" placeholder="请输入继任者"
class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import cache from '@/utils/cache'
import {
onShow,
onUnload
} from '@dcloudio/uni-app';
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd
} from '@/api/crm/api_ys.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
cusUserName: null, //客户人员名称
originalPosition: null, //原职务
currentPosition: null, //现职务
positionOfOurCompany: '是', //现职务是否与我公司业务相关
developmentEfforts: '加强', //攻关力度
successor: null, //继任者
remark: "", //、备注
picture: "", //、图片
photos: null,
cusPersonnel: '',
informationType: "人员变化信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
cusUserName: {
rules: [{
required: true,
errorMessage: '请选择客户人员'
}]
},
originalPosition: {
rules: [{
required: true,
errorMessage: '请输入原职务'
}]
},
currentPosition: {
rules: [{
required: true,
errorMessage: '请输入现职务'
}]
},
positionOfOurCompany: {
rules: [{
required: true,
errorMessage: '请选择现职务是否与我公司业务相关'
}]
},
developmentEfforts: {
rules: [{
required: true,
errorMessage: '请选择公关力度'
}]
},
successor: {
rules: [{
required: true,
errorMessage: '请输入继任者'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '是'
},
{
id: 1,
name: '否'
}
])
const array1 = ref([{
id: 0,
name: '加强'
},
{
id: 1,
name: '维持'
},
{
id: 2,
name: '减弱'
}
])
const opportunityTypeIndex = ref(0)
const positionOfOurCompanyIndex = ref(0)
const developmentEffortsIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
console.log("收到客户数据的值:", customerUser)
}
//选择客户人员
function chooseClientUser() {
console.log('点击了输入框', formData.value.cusName);
uni.navigateTo({
url: '/pages/business/CRM/customerUserList?cusName=' + formData.value.cusName
})
}
//页面渲染完成后查询catch的get
onShow(() => {
if (cache.get('checkedRCClientList') != null && cache.get('checkedRCClientList') != []) {
formData.value.cusUserName = cache.get('checkedRCClientList')
}
})
//页面卸载之后,删除缓存信息
onUnload(() => {
handleDeleteLocal()
})
//删除缓存
let handleDeleteLocal = () => {
cache.remove('checkedRCClientList');
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const cusUserName1 = formData.value.cusUserName;
const stringOfNames = cusUserName1.join(",")
formData.value.cusUserName = stringOfNames
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
const onpoSitionOfOurCompanyChange = (e) => {
positionOfOurCompanyIndex.value = e.detail.value
console.log('positionOfOurCompanyIndex:', array.value[e.detail.value]?.name)
formData.value.positionOfOurCompany = array.value[e.detail.value]?.name || ''
}
const onDevelopmentEffortsChange = (e) => {
developmentEffortsIndex.value = e.detail.value
console.log('developmentEffortsIndex:', array1.value[e.detail.value]?.name)
formData.value.opportunityType = array1.value[e.detail.value]?.name || ''
}
// 如果你原来在 onShow 中做了类似这样:
// let res = currPage.data.cusData; 判断是否传入了客户信息
// 那么在 Vue3 中通常是通过路由参数或者 Vuex/Pinia 等状态管理获取
// 暂时不做,如你后续需要可继续补充
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>

View File

@@ -0,0 +1,373 @@
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'人员变化添加'" :leftFlag="true" :rightFlag="true">
<template #right>
<view class="head-right" @click="submitForm">
<uni-icons custom-prefix="iconfont" type="icon-phonebaocun" size="22"
color="#B7D2FF"></uni-icons>保存
</view>
</template>
</customHeader>
<!-- 高度来避免头部遮挡 -->
<view class="top-height"></view>
<!-- 正文内容 -->
<view class="white-bg">
<view class="form-con">
<uni-forms ref="formRef" :model="formData" :rules="rules" label-width="100px">
<uni-forms-item label="客户名称" name="cusName" class="f-c-right">
<view @click="chooseCustomer" class="form-item-container">
<text class="name">{{ formData.cusName || '点击选择客户' }}</text>
</view>
</uni-forms-item>
<uni-forms-item label="客户人员" name="cusUserName" class="f-c-right">
<uni-easyinput v-model="formData.cusUserName" placeholder="请选择客户人员" name="input"
@focus="chooseClientUser"></uni-easyinput>
</uni-forms-item>
<uni-forms-item label="原职务" name="originalPosition" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.originalPosition"
placeholder="请输入原职务" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="现职务" name="currentPosition" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.currentPosition"
placeholder="请输入现职务" class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="现职务是否与我司业务相关" name="positionOfOurCompany" class="f-c-right">
<picker @change="onpoSitionOfOurCompanyChange" :value="positionOfOurCompanyIndex"
:range="array" :range-key="'name'">
<view class="picker">
{{ array[positionOfOurCompanyIndex]?.name || '请选择机会类型' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="公关力度" name="developmentEfforts" class="f-c-right">
<picker @change="onDevelopmentEffortsChange" :value="developmentEffortsIndex"
:range="array1" :range-key="'name'">
<view class="picker">
{{ array1[developmentEffortsIndex]?.name || '请选择公关力度' }}
<uni-icons type="right" size="20" color="#A0A0A0"></uni-icons>
</view>
</picker>
</uni-forms-item>
<uni-forms-item label="继任者" name="successor" required
class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.successor" placeholder="请输入继任者"
class="form-texarea" />
</uni-forms-item>
<uni-forms-item label="备注" name="remark" class="uni-forms-item is-direction-top is-top">
<uni-easyinput type="textarea" autoHeight v-model="formData.remark" placeholder="请输入备注"
class="form-texarea" />
</uni-forms-item>
</uni-forms>
<view class="footer-con">
<button class="btn-default" type="default" @click="handleDelete" size="mini"> </button>
<button class="btn-primary" type="primary" @click="submitForm" size="mini">保存/修改</button>
</view>
</view>
</view>
</view>
</view>
</template>
<script setup>
import {
ref,
onMounted,
reactive,
onUnmounted,
computed
} from 'vue'
import customHeader from '@/components/customHeader.vue'
import { onShow, onUnload,onLoad } from '@dcloudio/uni-app';
import {
getGuestList
} from '@/api/business.js'
import {
isEmpty
} from '@/utils/validate.js'
import {
crmMarketInformationAdd,crminformationItemForDetail,crmMarketInformationDelete
} from '@/api/crm/api_ys.js'
import { useMessage } from '@/utils/message.js'
import cache from '@/utils/cache.js'
let customerUser = reactive({})
// 客户相关
const guestList = ref([])
const guestArr = ref([])
const guestIndex = ref(0)
// 表单数据
const formData = ref({
cusId: null,
cusName: null,
cusUserName: null, //客户人员名称
originalPosition: null, //原职务
currentPosition: null, //现职务
positionOfOurCompany: '是', //现职务是否与我公司业务相关
developmentEfforts: '加强', //攻关力度
successor: null, //继任者
remark: "", //、备注
picture: "", //、图片
photos: null,
cusPersonnel: '',
informationType: "人员变化信息" //信息类型
})
// 表单验证规则
const rules = {
cusName: {
rules: [{
required: true,
errorMessage: '请选择客户'
}]
},
cusUserName: {
rules: [{
required: true,
errorMessage: '请选择客户人员'
}]
},
originalPosition: {
rules: [{
required: true,
errorMessage: '请输入原职务'
}]
},
currentPosition: {
rules: [{
required: true,
errorMessage: '请输入现职务'
}]
},
positionOfOurCompany: {
rules: [{
required: true,
errorMessage: '请选择现职务是否与我公司业务相关'
}]
},
developmentEfforts: {
rules: [{
required: true,
errorMessage: '请选择公关力度'
}]
},
successor: {
rules: [{
required: true,
errorMessage: '请输入继任者'
}]
},
}
const imgList = ref([])
// picker 相关
const index = ref(0)
const array = ref([{
id: 0,
name: '是'
},
{
id: 1,
name: '否'
}
])
const array1 = ref([{
id: 0,
name: '加强'
},
{
id: 1,
name: '维持'
},
{
id: 2,
name: '减弱'
}
])
const opportunityTypeIndex = ref(0)
const positionOfOurCompanyIndex = ref(0)
const developmentEffortsIndex = ref(0)
// 表单引用 & 客户选择器引用
const formRef = ref(null)
const customHeaderRef = ref(null)
// 选择客户页面跳转
function chooseCustomer() {
uni.navigateTo({
url: '/pages/business/CRM/chooseCus'
})
}
//定义数据接收的值
let selectedCustomer = reactive(null)
//监听时间
onMounted(() => {
uni.$on('onCustomerSelected', handleCustomerSelected)
})
//取消监听
onUnmounted(() => {
uni.$off('onCustomerSelected', handleCustomerSelected)
})
//处理 接收数据
const handleCustomerSelected = (data) => {
formData.value.cusName = data.cusName
formData.value.cusId = data.cusId
console.log("收到客户数据的值:", customerUser)
}
//选择客户人员
function chooseClientUser() {
console.log('点击了输入框', formData.value.cusName);
uni.navigateTo({
url: '/pages/business/CRM/customerUserList?cusName=' + formData.value.cusName
})
}
//页面渲染完成后查询catch的get
onShow(() => {
if (cache.get('checkedRCClientList') != null && cache.get('checkedRCClientList') != []) {
formData.value.cusUserName = cache.get('checkedRCClientList')
}
})
//页面卸载之后,删除缓存信息
onUnload(() => {
handleDeleteLocal()
})
//删除缓存
let handleDeleteLocal = () => {
cache.remove('checkedRCClientList');
}
// 提交表单
const submitForm = async () => {
try {
// 表单校验
await formRef.value.validate()
const cusUserName1 = formData.value.cusUserName;
const stringOfNames = cusUserName1.join(",")
formData.value.cusUserName = stringOfNames
const res = await crmMarketInformationAdd(formData.value);
console.log(res)
uni.showToast({
title: '提交成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
// TODO: 在这里添加提交到后端的逻辑,比如调用 api.CrmMarketInformationAdd(formData.value)
// 暂时只做校验提示
} catch (err) {
console.log('表单验证失败:', err)
}
}
const onOpportunityTypeChange = (e) => {
opportunityTypeIndex.value = e.detail.value
console.log('opportunityTypeIndex:', array.value[e.detail.value]?.name)
formData.value.opportunityType = array.value[e.detail.value]?.name || ''
}
const onpoSitionOfOurCompanyChange = (e) => {
positionOfOurCompanyIndex.value = e.detail.value
console.log('positionOfOurCompanyIndex:', array.value[e.detail.value]?.name)
formData.value.positionOfOurCompany = array.value[e.detail.value]?.name || ''
}
const onDevelopmentEffortsChange = (e) => {
developmentEffortsIndex.value = e.detail.value
console.log('developmentEffortsIndex:', array1.value[e.detail.value]?.name)
formData.value.opportunityType = array1.value[e.detail.value]?.name || ''
}
//定义查询参数
const queryParams = ref({
informationId: 0
})
onMounted(() => {
crminformationItemForDetail1()
})
onLoad(options => {
queryParams.value.informationId = options.informationId
})
const dataList = ref([])
function crminformationItemForDetail1() {
crminformationItemForDetail(queryParams.value).then(res => {
formData.value = res.rows[0]
})
}
// 根据ID删除表单
const handleDelete = async () => {
try {
const res = await crmMarketInformationDelete(formData.value);
uni.showToast({
title: '删除成功',
icon: 'success'
})
uni.$emit('refreshMarketList');
setTimeout(() => {
uni.navigateBack(1);
}, 1500);
} catch (err) {
console.log('表单验证失败:', err)
}
}
</script>
<style scoped>
.white-bg {
width: 750rpx;
padding: 30rpx 0 0;
margin-bottom: 0;
border-radius: 8px 8px 0 0;
}
.form-con {
/* #ifdef APP-PLUS */
height: calc(120vh - 100px)
/* #endif */
/* #ifndef APP-PLUS */
height:calc(120vh - 80px)
/* #endif */
}
:deep(.uni-date-x) {
display: block;
}
:deep(.uni-date-x .icon-calendar) {
float: right;
margin-top: 15rpx;
margin-right: 20rpx;
background: url('../../../static/images/business/icon-date.png') no-repeat;
background-size: 32rpx 35rpx;
width: 32rpx;
height: 35rpx;
}
:deep(.uni-date-x .icon-calendar::before) {
display: none;
}
:deep(.uni-date-x .uni-date__x-input) {
padding-left: 20rpx;
color: #919191;
}
</style>