Files
ys-app/src/pages/business/CRM/customer/customerUserBelong.vue
2025-08-25 16:47:32 +08:00

347 lines
9.0 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!--
* @author wangzhuo
* @date 2025年8月18日
* @description 客户人员所属
-->
<template>
<view class="con-body">
<view class="con-bg">
<!-- 头部 -->
<customHeader ref="customHeaderRef" :title="'客户人员查看'" :leftFlag="true" :rightFlag="false">
</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"
@clear="clearSearchValue"
v-model="searchValue"
/>
</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.stop="handleDetail(item)">
<view class="report-list">
<view class="w-b-title title" @touchstart="handleTouchStart(item)"
@touchend="handleTouchEnd">
{{ item.cusName }}
<view v-if="item.nodeCode" class="r-right btn-edit" :class="statusColorMap[item.nodeCode]">
{{ item.nodeCode }}
</view><!---->
</view>
<view class="r-list">
<view class="r-left">客户人员名称</view>
<view class="r-right">{{ item.userName }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">客户等级</view>
<view class="r-right">{{ item.cusEstate }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">职能</view>
<view class="r-right">{{ item.function }}</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 class="border-bottom"></view>
<view class="r-list">
<view class="r-left">业务员认定等级</view>
<view class="r-right">{{ item.salesmanThinkLevel }}</view>
</view>
<view class="border-bottom"></view>
<view class="r-list">
<view class="r-left">系统认定等级</view>
<view class="r-right">{{ item.systemThinkLevel }}</view>
</view>
<view v-if="item.opinion" class="border-bottom"></view>
<view v-if="item.opinion" class="r-list">
<view class="r-left">驳回原因</view>
<view class="r-right">{{ item.opinion }}</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 {SearchForAllCustomersSalesperson} from "@/api/crm/customer/getCustomer"
import {statusColorMap} from "./dataMap";
// 获取导航栏高度用于内容区域padding
const navBarPaddingTop = ref(0);
onMounted(() => {
navBarPaddingTop.value = getNavBarPaddingTop() * 2;
})
// 搜索内容
let searchValue = ref(null);
// 分页
const mescrollRef = ref(null);
// 防抖计时
let timerId = null;
// 查询搜索跳转
let handleSearch = () => {
console.log(searchValue.value)
if (timerId) clearTimeout(timerId);
timerId = setTimeout(async () => {
// let res = await getList(1, upOption.value.page.size)
await downCallback(mescrollRef.value.mescroll);
clearTimeout(timerId);
timerId = null;
}, 500)
}
watch(searchValue, (newValue, oldValue) => {
handleSearch()
})
let clearSearchValue = () => {
searchValue.value = '';
}
// 查询列表
let list = ref([]);
const upOption = ref({
page: {num: 0, size: 10},
noMoreSize: 5,
empty: {
tip: '~ 空空如也 ~',
icon: "../../static/images/mescroll-empty.png"
},
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 {
cssFlag.value = true;
uni.showLoading();
setTimeout(async () => {
// 重置页码为第一页
const res = await getList(1, mescroll.size || upOption.page.size);
cssFlag.value = false;
list.value = res.list;
// 正确传递 total 参数
mescroll.endSuccess(res.list.length, res.total > (mescroll.size || upOption.page.size));
uni.hideLoading();
}, 500);
} catch (error) {
console.log(error)
mescroll.endErr();
}
}
// 上拉加载更多
const upCallback = async (mescroll) => {
try {
setTimeout(async () => {
// 使用 mescroll 提供的页码和大小参数
const res = await getList(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)
// mescroll.endSuccess(res.list.length, res.total > mescroll.num * mescroll.size);
}, 500);
} catch (error) {
console.log(error)
mescroll.endErr();
}
}
// 获取数据列表
const getList = (pageIndex, pageSize) => {
return new Promise(async (resolve) => {
let param = {
pageNum: pageIndex,
pageSize,
searchContent: searchValue.value
}
let res = await SearchForAllCustomersSalesperson(param);
resolve({
list: res.rows,
total: res.total
});
});
}
// 跳转到详情
let handleDetail = (item) => {
if (item.nodeCode !== '完成') {
uni.navigateTo({
url: "/pages/business/CRM/customer/components/customerUserEdit",
events: {
refreshCusUserList() {
handleSearch();
}
},
success(res) {
res.eventChannel.emit('editCusData', {param: item, isAdd: false})
}
})
}else{
uni.showToast({
title: '已完成审核,不可修改',
icon: 'none'
})
}
}
// 长按定时器ID
let touchTimerId = null;
// 开始触摸
let handleTouchStart = (item) => {
touchTimerId = setTimeout(() => {
// console.log(item, "长按客户人员项")
uni.showModal({
title: "提示",
content: "是否确认删除本条消息?",
success(res) {
if (res.confirm) {
handleDelete(item);
} else if (res.cancel) {
}
},
})
clearTimeout(touchTimerId);
}, 1500)
}
// 结束触摸
let handleTouchEnd = () => {
if (touchTimerId !== null) {
clearTimeout(touchTimerId);
}
}
// 删除消息
let handleDelete = async (item) => {
// console.log(item)
const visitId = item.visitId;
// await removeVisit({visitId})
// uni.hideLoading();
uni.showToast({
title: visitId ? visitId : "没有visitId字段无法删除",
icon: 'none'
});
}
</script>
<style lang="scss" scoped>
.all-body {
/* #ifdef APP-PLUS */
top: 150rpx;
height: calc(100vh - 75px);
/* #endif */
/* #ifndef APP-PLUS */
top: 120rpx;
height: calc(100vh);
/* #endif */
}
.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;
}
.scroll-h {
/* #ifdef APP-PLUS */
height: calc(100vh - 120px);
/* #endif */
/* #ifndef APP-PLUS */
height: calc(100vh - 110px);
/* #endif */
}
.white-bg {
padding-bottom: 10rpx;
.w-b-title {
align-items: flex-start;
}
}
.btn-edit {
white-space: nowrap;
&.wancheng-deal {
border: solid 1rpx #00aa7f;
background-color: #00aa7f;
}
&.daishenhe-un-deal {
border: solid 1rpx #ffaa7f;
background-color: #ffaa7f;
}
&.chaoqi-deal {
border: 1rpx solid #ff0000;
background-color: #ff0000;
}
&.tijiao-deal {
border: 1rpx solid #00aaff;
background-color: #00aaff;
}
}
</style>