客户详细
This commit is contained in:
parent
6bc8515f48
commit
40685b41a5
|
|
@ -210,10 +210,42 @@ export const getCustomerList = (params = {}) => {
|
|||
* @returns {Promise} 返回客户详情
|
||||
*/
|
||||
export const getCustomerDetail = (id) => {
|
||||
return uni.$uv.http.get(`customer/${id}`, {
|
||||
return uni.$uv.http.get(`bst/customer/${id}`, {
|
||||
custom: {
|
||||
auth: true // 启用 token 认证
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 获取客户跟进动态列表
|
||||
* @param {string} customerId 客户ID
|
||||
* @returns {Promise} 返回跟进动态列表
|
||||
*/
|
||||
export const getCustomerFollowupList = (customerId) => {
|
||||
return uni.$uv.http.get(`bst/customer/followup/list`, {
|
||||
params: {
|
||||
customerId: customerId
|
||||
},
|
||||
custom: {
|
||||
auth: true
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 获取客户项目列表
|
||||
* @param {string} customerId 客户ID
|
||||
* @returns {Promise} 返回项目列表
|
||||
*/
|
||||
export const getCustomerProjects = (customerId) => {
|
||||
return uni.$uv.http.get(`bst/customer/projects`, {
|
||||
params: {
|
||||
customerId: customerId
|
||||
},
|
||||
custom: {
|
||||
auth: true
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
|
|
|
|||
|
|
@ -235,10 +235,10 @@ const loadCustomerList = async () => {
|
|||
// 处理客户点击
|
||||
const handleCustomerClick = (customer) => {
|
||||
console.log('点击客户:', customer);
|
||||
// 可以跳转到客户详情页
|
||||
// uni.navigateTo({
|
||||
// url: `/pages/customer-detail/index?id=${customer.id}`
|
||||
// });
|
||||
// 跳转到客户详情页
|
||||
uni.navigateTo({
|
||||
url: `/pages/customer-detail/index?id=${customer.id}`
|
||||
});
|
||||
};
|
||||
|
||||
// 处理跟进
|
||||
|
|
|
|||
|
|
@ -68,6 +68,13 @@
|
|||
"navigationBarTitleText": "申请详情",
|
||||
"navigationStyle": "custom"
|
||||
}
|
||||
},
|
||||
{
|
||||
"path": "pages/customer-detail/index",
|
||||
"style": {
|
||||
"navigationBarTitleText": "客户详情",
|
||||
"navigationStyle": "custom"
|
||||
}
|
||||
}
|
||||
],
|
||||
|
||||
|
|
|
|||
960
pages/customer-detail/index.vue
Normal file
960
pages/customer-detail/index.vue
Normal file
|
|
@ -0,0 +1,960 @@
|
|||
<template>
|
||||
<view class="customer-detail-page">
|
||||
<!-- 自定义导航栏 -->
|
||||
<view class="custom-navbar">
|
||||
<view class="navbar-content">
|
||||
<text class="nav-btn" @click="handleBack">‹</text>
|
||||
<text class="nav-title">{{ customerDetail.name || '客户详情' }}</text>
|
||||
<text class="nav-btn" style="opacity: 0;">占位</text>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 客户摘要卡片 -->
|
||||
<view class="customer-summary-card">
|
||||
<view class="summary-row">
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">客户状态</text>
|
||||
<view class="status-badge" :class="getStatusClass(customerDetail.status)">
|
||||
<text>{{ getStatusText(customerDetail.status) }}</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">客户星级</text>
|
||||
<view class="stars">
|
||||
<text
|
||||
class="star"
|
||||
v-for="i in 5"
|
||||
:key="i"
|
||||
:class="{ 'filled': i <= getRatingFromIntentLevel(customerDetail.intentLevel) }"
|
||||
>★</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="summary-row">
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">客户归属</text>
|
||||
<text class="summary-value">{{ customerDetail.followName || '未分配' }}</text>
|
||||
</view>
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">客户类型</text>
|
||||
<text class="summary-value">{{ customerDetail.customerType || '--' }}</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="summary-row">
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">联系人</text>
|
||||
<text class="summary-value">{{ customerDetail.contactName || customerDetail.name || '--' }}</text>
|
||||
</view>
|
||||
<view class="summary-item">
|
||||
<text class="summary-label">最近跟进</text>
|
||||
<text class="summary-value">{{ formatDateTime(customerDetail.lastFollowTime) }}</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 标签页导航 -->
|
||||
<view class="tab-navigation">
|
||||
<view
|
||||
class="tab-item"
|
||||
:class="{ active: activeTab === 'followup' }"
|
||||
@click="switchTab('followup')"
|
||||
>
|
||||
<text>跟进动态</text>
|
||||
</view>
|
||||
<view
|
||||
class="tab-item"
|
||||
:class="{ active: activeTab === 'projects' }"
|
||||
@click="switchTab('projects')"
|
||||
>
|
||||
<text>项目列表</text>
|
||||
</view>
|
||||
<view
|
||||
class="tab-item"
|
||||
:class="{ active: activeTab === 'info' }"
|
||||
@click="switchTab('info')"
|
||||
>
|
||||
<text>客户信息</text>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 内容区域 -->
|
||||
<scroll-view class="content-scroll" scroll-y>
|
||||
<!-- 跟进动态标签页 -->
|
||||
<view class="tab-content" v-if="activeTab === 'followup'">
|
||||
<view class="followup-timeline">
|
||||
<view
|
||||
class="followup-item"
|
||||
v-for="(item, index) in followupList"
|
||||
:key="index"
|
||||
@click="handleFollowupClick(item)"
|
||||
>
|
||||
<view class="timeline-dot">
|
||||
<text class="dot-date">{{ formatDate(item.createTime) }}</text>
|
||||
</view>
|
||||
<view class="followup-content">
|
||||
<view class="followup-header">
|
||||
<image
|
||||
class="followup-avatar"
|
||||
:src="item.userAvatar || '/static/default-avatar.png'"
|
||||
mode="aspectFill"
|
||||
/>
|
||||
<view class="followup-user-info">
|
||||
<text class="followup-user-name">{{ item.userName }}</text>
|
||||
<text class="followup-user-role">{{ item.userRole || '销售经理' }}</text>
|
||||
</view>
|
||||
<text class="followup-arrow">›</text>
|
||||
</view>
|
||||
<text class="followup-text">{{ item.content }}</text>
|
||||
<text class="followup-time">{{ formatDateTime(item.createTime) }}</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="empty-state" v-if="followupList.length === 0">
|
||||
<text>暂无跟进记录</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 项目列表标签页 -->
|
||||
<view class="tab-content" v-if="activeTab === 'projects'">
|
||||
<view class="project-list">
|
||||
<view
|
||||
class="project-card"
|
||||
v-for="(project, index) in projectList"
|
||||
:key="index"
|
||||
>
|
||||
<view class="project-header">
|
||||
<view class="project-status-tag" :class="getProjectStatusClass(project.status)">
|
||||
<text>{{ getProjectStatusText(project.status) }}</text>
|
||||
</view>
|
||||
<text class="project-more" @click.stop="handleProjectMore(project)">⋮</text>
|
||||
</view>
|
||||
<text class="project-name">{{ project.name }}</text>
|
||||
<view class="project-progress">
|
||||
<view class="progress-bar">
|
||||
<view
|
||||
class="progress-fill"
|
||||
:style="{ width: project.progress + '%' }"
|
||||
></view>
|
||||
</view>
|
||||
<text class="progress-text">{{ project.progress }}%</text>
|
||||
</view>
|
||||
<view class="project-participants">
|
||||
<text class="participants-text">{{ formatParticipants(project.participants) }}</text>
|
||||
</view>
|
||||
<view class="project-deadline" :class="{ 'overdue': project.isOverdue }">
|
||||
<text class="deadline-icon">🕐</text>
|
||||
<text class="deadline-text">{{ formatDeadline(project) }}</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="empty-state" v-if="projectList.length === 0">
|
||||
<text>暂无项目</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<!-- 客户信息标签页 -->
|
||||
<view class="tab-content" v-if="activeTab === 'info'">
|
||||
<view class="info-section">
|
||||
<view class="section-title">
|
||||
<view class="title-line"></view>
|
||||
<text>基础信息</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">客户名称</text>
|
||||
<text class="info-value">{{ customerDetail.name || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">联系电话</text>
|
||||
<text class="info-value">{{ customerDetail.mobile || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">微信号</text>
|
||||
<text class="info-value">{{ customerDetail.wechat || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">客户来源</text>
|
||||
<text class="info-value">{{ customerDetail.source || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">客户意向</text>
|
||||
<text class="info-value">{{ customerDetail.intent || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">意向强度</text>
|
||||
<text class="info-value">{{ getIntentStrengthText(customerDetail.intentLevel) }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">客户地区</text>
|
||||
<text class="info-value">{{ formatRegion(customerDetail.region, customerDetail.city) }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">工作微信</text>
|
||||
<text class="info-value">{{ customerDetail.workWechat || '--' }}</text>
|
||||
</view>
|
||||
</view>
|
||||
|
||||
<view class="info-section">
|
||||
<view class="section-title">
|
||||
<view class="title-line"></view>
|
||||
<text>其他信息</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">客户星级</text>
|
||||
<view class="info-value-stars">
|
||||
<text
|
||||
class="star"
|
||||
v-for="i in 5"
|
||||
:key="i"
|
||||
:class="{ 'filled': i <= getRatingFromIntentLevel(customerDetail.intentLevel) }"
|
||||
>★</text>
|
||||
</view>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">备注</text>
|
||||
<text class="info-value">{{ customerDetail.remark || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">顾虑点</text>
|
||||
<text class="info-value">{{ customerDetail.concerns || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">痛点</text>
|
||||
<text class="info-value">{{ customerDetail.painPoints || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">关注点</text>
|
||||
<text class="info-value">{{ customerDetail.focusPoints || '--' }}</text>
|
||||
</view>
|
||||
<view class="info-item">
|
||||
<text class="info-label">需求点</text>
|
||||
<text class="info-value">{{ customerDetail.requirements || '--' }}</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</scroll-view>
|
||||
|
||||
<!-- 底部操作栏 -->
|
||||
<view class="bottom-actions">
|
||||
<view class="action-btn" @click="handleNewFollowup">
|
||||
<text class="action-icon">✎</text>
|
||||
<text class="action-text">写新跟进</text>
|
||||
</view>
|
||||
<view class="action-btn" @click="handleNewTask">
|
||||
<text class="action-icon">☑</text>
|
||||
<text class="action-text">新建任务</text>
|
||||
</view>
|
||||
<view class="action-btn" @click="handleCall">
|
||||
<text class="action-icon">☎</text>
|
||||
<text class="action-text">拨打电话</text>
|
||||
</view>
|
||||
<view class="action-btn" @click="handleMore">
|
||||
<text class="action-icon">⋯</text>
|
||||
<text class="action-text">更多操作</text>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, onMounted } from 'vue';
|
||||
import { onLoad } from '@dcloudio/uni-app';
|
||||
import { getCustomerDetail, getCustomerFollowupList, getCustomerProjects } from '@/common/api';
|
||||
|
||||
// 页面参数
|
||||
const customerId = ref('');
|
||||
const customerDetail = ref({});
|
||||
const activeTab = ref('followup');
|
||||
const followupList = ref([]);
|
||||
const projectList = ref([]);
|
||||
const loading = ref(false);
|
||||
|
||||
// 获取页面参数
|
||||
onLoad((options) => {
|
||||
if (options && options.id) {
|
||||
customerId.value = options.id;
|
||||
loadCustomerDetail();
|
||||
loadFollowupList();
|
||||
}
|
||||
});
|
||||
|
||||
// 切换标签页
|
||||
const switchTab = (tab) => {
|
||||
activeTab.value = tab;
|
||||
if (tab === 'followup' && followupList.value.length === 0) {
|
||||
loadFollowupList();
|
||||
} else if (tab === 'projects' && projectList.value.length === 0) {
|
||||
loadProjectList();
|
||||
}
|
||||
};
|
||||
|
||||
// 加载客户详情
|
||||
const loadCustomerDetail = async () => {
|
||||
if (!customerId.value) return;
|
||||
|
||||
loading.value = true;
|
||||
try {
|
||||
const res = await getCustomerDetail(customerId.value);
|
||||
if (res) {
|
||||
customerDetail.value = res;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('加载客户详情失败:', error);
|
||||
uni.$uv.toast('加载客户详情失败');
|
||||
} finally {
|
||||
loading.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// 加载跟进动态列表
|
||||
const loadFollowupList = async () => {
|
||||
if (!customerId.value) return;
|
||||
|
||||
try {
|
||||
const res = await getCustomerFollowupList(customerId.value);
|
||||
if (res && Array.isArray(res)) {
|
||||
followupList.value = res;
|
||||
} else if (res && res.rows && Array.isArray(res.rows)) {
|
||||
followupList.value = res.rows;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('加载跟进动态失败:', error);
|
||||
}
|
||||
};
|
||||
|
||||
// 加载项目列表
|
||||
const loadProjectList = async () => {
|
||||
if (!customerId.value) return;
|
||||
|
||||
try {
|
||||
const res = await getCustomerProjects(customerId.value);
|
||||
if (res && Array.isArray(res)) {
|
||||
projectList.value = res;
|
||||
} else if (res && res.rows && Array.isArray(res.rows)) {
|
||||
projectList.value = res.rows;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('加载项目列表失败:', error);
|
||||
}
|
||||
};
|
||||
|
||||
// 获取状态样式类
|
||||
const getStatusClass = (status) => {
|
||||
return {
|
||||
'status-following': status === '1',
|
||||
'status-pending': status === '2'
|
||||
};
|
||||
};
|
||||
|
||||
// 获取状态文本
|
||||
const getStatusText = (status) => {
|
||||
const statusMap = {
|
||||
'1': '正在跟进',
|
||||
'2': '待跟进',
|
||||
'3': '其他',
|
||||
'4': '无效客户'
|
||||
};
|
||||
return statusMap[status] || '未知';
|
||||
};
|
||||
|
||||
// 根据意向等级获取星级
|
||||
const getRatingFromIntentLevel = (intentLevel) => {
|
||||
const levelMap = {
|
||||
'1': 5,
|
||||
'2': 3
|
||||
};
|
||||
return levelMap[intentLevel] || 0;
|
||||
};
|
||||
|
||||
// 格式化日期时间
|
||||
const formatDateTime = (dateTime) => {
|
||||
if (!dateTime) return '暂无';
|
||||
try {
|
||||
const date = new Date(dateTime);
|
||||
const year = date.getFullYear();
|
||||
const month = String(date.getMonth() + 1).padStart(2, '0');
|
||||
const day = String(date.getDate()).padStart(2, '0');
|
||||
const hours = String(date.getHours()).padStart(2, '0');
|
||||
const minutes = String(date.getMinutes()).padStart(2, '0');
|
||||
return `${year}-${month}-${day} ${hours}:${minutes}`;
|
||||
} catch (e) {
|
||||
return dateTime;
|
||||
}
|
||||
};
|
||||
|
||||
// 格式化日期(仅显示月-日)
|
||||
const formatDate = (dateTime) => {
|
||||
if (!dateTime) return '';
|
||||
try {
|
||||
const date = new Date(dateTime);
|
||||
const month = String(date.getMonth() + 1).padStart(2, '0');
|
||||
const day = String(date.getDate()).padStart(2, '0');
|
||||
return `${month}-${day}`;
|
||||
} catch (e) {
|
||||
return '';
|
||||
}
|
||||
};
|
||||
|
||||
// 获取项目状态样式类
|
||||
const getProjectStatusClass = (status) => {
|
||||
return {
|
||||
'status-developing': status === 'developing' || status === '1',
|
||||
'status-expiring': status === 'expiring' || status === '2'
|
||||
};
|
||||
};
|
||||
|
||||
// 获取项目状态文本
|
||||
const getProjectStatusText = (status) => {
|
||||
const statusMap = {
|
||||
'developing': '开发中',
|
||||
'expiring': '即将到期',
|
||||
'1': '开发中',
|
||||
'2': '即将到期'
|
||||
};
|
||||
return statusMap[status] || '未知';
|
||||
};
|
||||
|
||||
// 格式化参与人
|
||||
const formatParticipants = (participants) => {
|
||||
if (!participants || !Array.isArray(participants)) return '--';
|
||||
if (participants.length <= 4) {
|
||||
return participants.join('、');
|
||||
}
|
||||
const firstFour = participants.slice(0, 4).join('、');
|
||||
return `${firstFour}等${participants.length}人`;
|
||||
};
|
||||
|
||||
// 格式化截止日期
|
||||
const formatDeadline = (project) => {
|
||||
if (!project.deadline) return '--';
|
||||
const deadline = new Date(project.deadline);
|
||||
const now = new Date();
|
||||
const diffTime = deadline - now;
|
||||
const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
|
||||
|
||||
if (diffDays < 0) {
|
||||
return `逾期${Math.abs(diffDays)}天 ${formatDateTime(project.deadline)}`;
|
||||
} else {
|
||||
return `剩余${diffDays}天 ${formatDateTime(project.deadline)}`;
|
||||
}
|
||||
};
|
||||
|
||||
// 格式化地区
|
||||
const formatRegion = (region, city) => {
|
||||
if (!region && !city) return '--';
|
||||
return [region, city].filter(Boolean).join('/');
|
||||
};
|
||||
|
||||
// 获取意向强度文本
|
||||
const getIntentStrengthText = (intentLevel) => {
|
||||
const levelMap = {
|
||||
'1': '高',
|
||||
'2': '中',
|
||||
'3': '低'
|
||||
};
|
||||
return levelMap[intentLevel] || '--';
|
||||
};
|
||||
|
||||
// 返回
|
||||
const handleBack = () => {
|
||||
uni.navigateBack();
|
||||
};
|
||||
|
||||
// 跟进项点击
|
||||
const handleFollowupClick = (item) => {
|
||||
// 可以跳转到跟进详情
|
||||
console.log('点击跟进:', item);
|
||||
};
|
||||
|
||||
// 项目更多操作
|
||||
const handleProjectMore = (project) => {
|
||||
uni.showActionSheet({
|
||||
itemList: ['查看详情', '编辑项目', '删除项目'],
|
||||
success: (res) => {
|
||||
console.log('选择了第' + (res.tapIndex + 1) + '个选项');
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
// 写新跟进
|
||||
const handleNewFollowup = () => {
|
||||
uni.navigateTo({
|
||||
url: `/pages/customer-follow/index?customerId=${customerId.value}&customerName=${customerDetail.value.name}`
|
||||
});
|
||||
};
|
||||
|
||||
// 新建任务
|
||||
const handleNewTask = () => {
|
||||
uni.navigateTo({
|
||||
url: `/pages/customer-tasks/index?customerId=${customerId.value}&customerName=${customerDetail.value.name}`
|
||||
});
|
||||
};
|
||||
|
||||
// 拨打电话
|
||||
const handleCall = () => {
|
||||
if (customerDetail.value.mobile) {
|
||||
uni.makePhoneCall({
|
||||
phoneNumber: customerDetail.value.mobile,
|
||||
fail: (err) => {
|
||||
console.error('拨打电话失败:', err);
|
||||
uni.$uv.toast('拨打电话失败');
|
||||
}
|
||||
});
|
||||
} else {
|
||||
uni.$uv.toast('客户未设置电话号码');
|
||||
}
|
||||
};
|
||||
|
||||
// 更多操作
|
||||
const handleMore = () => {
|
||||
uni.showActionSheet({
|
||||
itemList: ['编辑客户', '删除客户', '分享客户'],
|
||||
success: (res) => {
|
||||
console.log('选择了第' + (res.tapIndex + 1) + '个选项');
|
||||
if (res.tapIndex === 0) {
|
||||
// 编辑客户
|
||||
uni.navigateTo({
|
||||
url: `/pages/edit-customer/index?id=${customerId.value}`
|
||||
});
|
||||
}
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
// 组件挂载时的初始化(如果需要)
|
||||
onMounted(() => {
|
||||
// 数据已在 onLoad 中加载
|
||||
});
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.customer-detail-page {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
height: 100vh;
|
||||
background-color: #f5f5f5;
|
||||
}
|
||||
|
||||
.custom-navbar {
|
||||
background-color: #fff;
|
||||
padding-top: var(--status-bar-height, 0);
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
|
||||
position: relative;
|
||||
z-index: 100;
|
||||
}
|
||||
|
||||
.navbar-content {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
height: 44px;
|
||||
padding: 0 16px;
|
||||
}
|
||||
|
||||
.nav-btn {
|
||||
font-size: 24px;
|
||||
color: #333;
|
||||
font-weight: bold;
|
||||
min-width: 44px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.nav-title {
|
||||
font-size: 18px;
|
||||
font-weight: 600;
|
||||
color: #333;
|
||||
flex: 1;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.customer-summary-card {
|
||||
background-color: #fff;
|
||||
padding: 16px;
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.summary-row {
|
||||
display: flex;
|
||||
margin-bottom: 12px;
|
||||
|
||||
&:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.summary-item {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
|
||||
&:first-child {
|
||||
margin-right: 16px;
|
||||
}
|
||||
}
|
||||
|
||||
.summary-label {
|
||||
font-size: 12px;
|
||||
color: #999;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
.summary-value {
|
||||
font-size: 14px;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.status-badge {
|
||||
display: inline-block;
|
||||
padding: 2px 8px;
|
||||
border-radius: 4px;
|
||||
font-size: 12px;
|
||||
|
||||
&.status-following {
|
||||
background-color: #e3f2fd;
|
||||
color: #1976d2;
|
||||
}
|
||||
|
||||
&.status-pending {
|
||||
background-color: #f5f5f5;
|
||||
color: #666;
|
||||
}
|
||||
}
|
||||
|
||||
.stars {
|
||||
display: flex;
|
||||
gap: 2px;
|
||||
}
|
||||
|
||||
.star {
|
||||
font-size: 14px;
|
||||
color: #ddd;
|
||||
|
||||
&.filled {
|
||||
color: #ffc107;
|
||||
}
|
||||
}
|
||||
|
||||
.tab-navigation {
|
||||
display: flex;
|
||||
background-color: #fff;
|
||||
border-bottom: 1px solid #eee;
|
||||
padding: 0 16px;
|
||||
}
|
||||
|
||||
.tab-item {
|
||||
flex: 1;
|
||||
padding: 12px 0;
|
||||
text-align: center;
|
||||
position: relative;
|
||||
|
||||
text {
|
||||
font-size: 14px;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
&.active {
|
||||
text {
|
||||
color: #1976d2;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
&::after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
height: 2px;
|
||||
background-color: #1976d2;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.content-scroll {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.tab-content {
|
||||
padding: 16px;
|
||||
}
|
||||
|
||||
// 跟进动态样式
|
||||
.followup-timeline {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.followup-item {
|
||||
display: flex;
|
||||
margin-bottom: 16px;
|
||||
background-color: #fff;
|
||||
border-radius: 8px;
|
||||
padding: 12px;
|
||||
box-shadow: 0 1px 4px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.timeline-dot {
|
||||
width: 50px;
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
justify-content: center;
|
||||
padding-top: 4px;
|
||||
}
|
||||
|
||||
.dot-date {
|
||||
font-size: 12px;
|
||||
color: #1976d2;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.followup-content {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.followup-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.followup-avatar {
|
||||
width: 32px;
|
||||
height: 32px;
|
||||
border-radius: 16px;
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
.followup-user-info {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.followup-user-name {
|
||||
font-size: 14px;
|
||||
color: #333;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.followup-user-role {
|
||||
font-size: 12px;
|
||||
color: #999;
|
||||
}
|
||||
|
||||
.followup-arrow {
|
||||
font-size: 18px;
|
||||
color: #ccc;
|
||||
}
|
||||
|
||||
.followup-text {
|
||||
font-size: 14px;
|
||||
color: #666;
|
||||
line-height: 1.5;
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.followup-time {
|
||||
font-size: 12px;
|
||||
color: #999;
|
||||
}
|
||||
|
||||
// 项目列表样式
|
||||
.project-card {
|
||||
background-color: #fff;
|
||||
border-radius: 8px;
|
||||
padding: 16px;
|
||||
margin-bottom: 12px;
|
||||
box-shadow: 0 1px 4px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.project-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.project-status-tag {
|
||||
padding: 4px 12px;
|
||||
border-radius: 4px;
|
||||
font-size: 12px;
|
||||
|
||||
&.status-developing {
|
||||
background-color: #e3f2fd;
|
||||
color: #1976d2;
|
||||
}
|
||||
|
||||
&.status-expiring {
|
||||
background-color: #fff3e0;
|
||||
color: #f57c00;
|
||||
}
|
||||
}
|
||||
|
||||
.project-more {
|
||||
font-size: 18px;
|
||||
color: #999;
|
||||
transform: rotate(90deg);
|
||||
}
|
||||
|
||||
.project-name {
|
||||
font-size: 16px;
|
||||
font-weight: 600;
|
||||
color: #333;
|
||||
margin-bottom: 12px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.project-progress {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.progress-bar {
|
||||
flex: 1;
|
||||
height: 6px;
|
||||
background-color: #f0f0f0;
|
||||
border-radius: 3px;
|
||||
overflow: hidden;
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
.progress-fill {
|
||||
height: 100%;
|
||||
background-color: #1976d2;
|
||||
border-radius: 3px;
|
||||
transition: width 0.3s;
|
||||
}
|
||||
|
||||
.progress-text {
|
||||
font-size: 12px;
|
||||
color: #1976d2;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.project-participants {
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.participants-text {
|
||||
font-size: 12px;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.project-deadline {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
|
||||
&.overdue {
|
||||
.deadline-text {
|
||||
color: #f44336;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.deadline-icon {
|
||||
font-size: 14px;
|
||||
margin-right: 4px;
|
||||
}
|
||||
|
||||
.deadline-text {
|
||||
font-size: 12px;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
// 客户信息样式
|
||||
.info-section {
|
||||
background-color: #fff;
|
||||
border-radius: 8px;
|
||||
padding: 16px;
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.section-title {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.title-line {
|
||||
width: 3px;
|
||||
height: 16px;
|
||||
background-color: #1976d2;
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
.section-title text {
|
||||
font-size: 16px;
|
||||
font-weight: 600;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
display: flex;
|
||||
padding: 12px 0;
|
||||
border-bottom: 1px solid #f5f5f5;
|
||||
|
||||
&:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
}
|
||||
|
||||
.info-label {
|
||||
width: 100px;
|
||||
font-size: 14px;
|
||||
color: #666;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.info-value {
|
||||
flex: 1;
|
||||
font-size: 14px;
|
||||
color: #333;
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
.info-value-stars {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
gap: 2px;
|
||||
}
|
||||
|
||||
.empty-state {
|
||||
text-align: center;
|
||||
padding: 40px 0;
|
||||
color: #999;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
// 底部操作栏
|
||||
.bottom-actions {
|
||||
display: flex;
|
||||
background-color: #fff;
|
||||
border-top: 1px solid #eee;
|
||||
padding: 8px 0;
|
||||
padding-bottom: calc(8px + env(safe-area-inset-bottom));
|
||||
box-shadow: 0 -2px 8px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.action-btn {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 8px;
|
||||
}
|
||||
|
||||
.action-icon {
|
||||
font-size: 20px;
|
||||
color: #1976d2;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
|
||||
.action-text {
|
||||
font-size: 12px;
|
||||
color: #666;
|
||||
}
|
||||
</style>
|
||||
Loading…
Reference in New Issue
Block a user