Browse Source

(*)回款和业绩明细 导入功能

master
wuqun 2 years ago
parent
commit
d187905a46
  1. 72
      api/app/lib/controllers/report/achievement.js
  2. 50
      api/app/lib/models/performance_detail.js
  3. 28
      api/app/lib/models/receivable_detail.js
  4. 12
      api/app/lib/routes/report/index.js
  5. 35
      web/client/src/sections/business/actions/achievement-report.js
  6. 48
      web/client/src/sections/business/constants/index.js
  7. 175
      web/client/src/sections/business/containers/performanceReport/achievementDetails.jsx
  8. 139
      web/client/src/sections/business/containers/performanceReport/backMoneyDetails.jsx
  9. 220
      web/client/src/sections/business/containers/performanceReport/importAchieveModal.jsx
  10. 256
      web/client/src/sections/business/containers/performanceReport/importBackModal.jsx
  11. 4
      web/client/src/utils/webapi.js

72
api/app/lib/controllers/report/achievement.js

@ -256,7 +256,79 @@ async function exportAchievementDetail(ctx, dataList) {
}
}
}
async function getReceivedNumbers(ctx) {
try {
const models = ctx.fs.dc.models;
let list = await models.ReceivableDetail.findAll({//查编号
attributes: ['number']
});
ctx.status = 200
ctx.body = list;
} catch (error) {
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
ctx.status = 400;
ctx.body = {
message: typeof error == 'string' ? error : undefined
}
}
}
async function importBackDetails(ctx) {
let errorMsg = { message: '导入回款明细失败' };
const transaction = await ctx.fs.dc.orm.transaction();
try {
const models = ctx.fs.dc.models;
const data = ctx.request.body;
let addArr = [];
let dataList = await models.ReceivableDetail.findAll({//查编号
attributes: ['number']
});
data.map(d => {
let exist = dataList.find(m => m.dataValues.number == d.number);
if (!exist) {
addArr.push(d);
}
})
//只处理新增的
if (addArr.length) {
await models.ReceivableDetail.bulkCreate(addArr);
}
await transaction.commit();
ctx.status = 204;
} catch (error) {
await transaction.rollback();
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
ctx.status = 400;
ctx.body = errorMsg;
}
}
async function importAchieveDetails(ctx) {
let errorMsg = { message: '导入业绩明细失败' };
const transaction = await ctx.fs.dc.orm.transaction();
try {
const models = ctx.fs.dc.models;
const data = ctx.request.body;
//业绩明细 没有唯一标识,前端校验后,导啥存啥
if (data.length) {
await models.PerformanceDetail.bulkCreate(data);
}
await transaction.commit();
ctx.status = 204;
} catch (error) {
await transaction.rollback();
ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
ctx.status = 400;
ctx.body = errorMsg;
}
}
module.exports = {
getReceivedDetail,//回款
getAchievementDetail,//业绩
getReceivedNumbers,//查询回款明细表 已有的所有编号
importBackDetails,//导入回款明细
importAchieveDetails,//导入业绩明细
}

50
api/app/lib/models/performance_detail.js

@ -13,11 +13,11 @@ module.exports = dc => {
comment: null,
primaryKey: true,
field: "id",
autoIncrement: false
autoIncrement: true
},
recConDate: {
type: DataTypes.DATEONLY,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "收到合同日期",
primaryKey: false,
@ -26,7 +26,7 @@ module.exports = dc => {
},
month: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "月份",
primaryKey: false,
@ -35,7 +35,7 @@ module.exports = dc => {
},
department: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "部门:申请部门",
primaryKey: false,
@ -44,7 +44,7 @@ module.exports = dc => {
},
sale: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "销售人员",
primaryKey: false,
@ -53,7 +53,7 @@ module.exports = dc => {
},
customer: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "客户名称:【甲方名称】",
primaryKey: false,
@ -62,7 +62,7 @@ module.exports = dc => {
},
item: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "项目名称",
primaryKey: false,
@ -71,7 +71,7 @@ module.exports = dc => {
},
amount: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "合同金额:【合同金额\n(元)】",
primaryKey: false,
@ -80,7 +80,7 @@ module.exports = dc => {
},
realPerformance: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "实际业绩;(合同金额-预\n支提成及委外费用)*特批折算比例\n(G-L)*K",
primaryKey: false,
@ -89,7 +89,7 @@ module.exports = dc => {
},
assessmentPerformance: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "合同金额*省外业务(1.1)*复购业务\n(1.05)*可复制的业务路径(1.1)",
primaryKey: false,
@ -98,7 +98,7 @@ module.exports = dc => {
},
isApproval: {
type: DataTypes.BOOLEAN,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "价格是否特批:根据折算比例-《合同明细表》推\n算(100%——“是”;其他为否)",
primaryKey: false,
@ -107,7 +107,7 @@ module.exports = dc => {
},
approvalProp: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "特批折算比例:【业绩折\n算比例】",
primaryKey: false,
@ -116,7 +116,7 @@ module.exports = dc => {
},
cost: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "预支提成及委外费用",
primaryKey: false,
@ -125,7 +125,7 @@ module.exports = dc => {
},
serviceLine: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "业务线",
primaryKey: false,
@ -134,7 +134,7 @@ module.exports = dc => {
},
cusType: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "客户类型",
primaryKey: false,
@ -143,7 +143,7 @@ module.exports = dc => {
},
industry: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "行业",
primaryKey: false,
@ -152,7 +152,7 @@ module.exports = dc => {
},
source: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "信息来源",
primaryKey: false,
@ -161,7 +161,7 @@ module.exports = dc => {
},
itemType: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "项目类型",
primaryKey: false,
@ -170,7 +170,7 @@ module.exports = dc => {
},
cusProvince: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "客户省份",
primaryKey: false,
@ -179,7 +179,7 @@ module.exports = dc => {
},
cusAttribute: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "客户属性:G:政府、事业单位\nB:央企、国企、平台商\nC:资源方",
primaryKey: false,
@ -188,7 +188,7 @@ module.exports = dc => {
},
repurchaseCount: {
type: DataTypes.INTEGER,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "复购次数:同一客户储备成单-成单项目数量\n(PM)",
primaryKey: false,
@ -197,7 +197,7 @@ module.exports = dc => {
},
reproducible: {
type: DataTypes.BOOLEAN,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "是否可复制的业务路径:《合同评审》表单获取",
primaryKey: false,
@ -206,7 +206,7 @@ module.exports = dc => {
},
outProvince: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "省外业务1.1:当【客户省份】=“江西”时,省外业务=1;当【客户省份】<>“江西”时,省外业务=1.1",
primaryKey: false,
@ -215,7 +215,7 @@ module.exports = dc => {
},
repurchase: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "复购业务1.05:当【复购次数】\n<2时,复购业务=1;当【复购次数】\n>=2时,复购业务=1.05;",
primaryKey: false,
@ -224,7 +224,7 @@ module.exports = dc => {
},
isreproduce: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "可复制的业务路径1.1:当【是否可复\n制的业务路径】=“是”,可复制的业务路径=1.1;当【是否可复制的业务路径】=“否”,可复制的业务路径=1;",
primaryKey: false,

28
api/app/lib/models/receivable_detail.js

@ -13,7 +13,7 @@ module.exports = dc => {
comment: null,
primaryKey: true,
field: "id",
autoIncrement: false
autoIncrement: true
},
year: {
type: DataTypes.STRING,
@ -44,7 +44,7 @@ module.exports = dc => {
},
department: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "部门:申请部门",
primaryKey: false,
@ -53,7 +53,7 @@ module.exports = dc => {
},
sale: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "销售人员:申请人",
primaryKey: false,
@ -62,7 +62,7 @@ module.exports = dc => {
},
contractNo: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "合同编号",
primaryKey: false,
@ -71,7 +71,7 @@ module.exports = dc => {
},
customer: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "客户名称:【甲方名称】",
primaryKey: false,
@ -80,7 +80,7 @@ module.exports = dc => {
},
item: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "项目名称",
primaryKey: false,
@ -89,7 +89,7 @@ module.exports = dc => {
},
amount: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "合同金额:【合同金额\n(元)】",
primaryKey: false,
@ -98,7 +98,7 @@ module.exports = dc => {
},
changeAmount: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "变更后合同金额",
primaryKey: false,
@ -107,7 +107,7 @@ module.exports = dc => {
},
receivableYear: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "回款年份",
primaryKey: false,
@ -116,7 +116,7 @@ module.exports = dc => {
},
receivableDate: {
type: DataTypes.DATEONLY,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "回款日期",
primaryKey: false,
@ -125,7 +125,7 @@ module.exports = dc => {
},
receivableAmount: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "回款金额",
primaryKey: false,
@ -134,7 +134,7 @@ module.exports = dc => {
},
invoicedBack: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "开票-回款",
primaryKey: false,
@ -143,7 +143,7 @@ module.exports = dc => {
},
remainConAmount: {
type: DataTypes.STRING,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "剩余合同金额",
primaryKey: false,
@ -152,7 +152,7 @@ module.exports = dc => {
},
incomeConfirmdate: {
type: DataTypes.DATEONLY,
allowNull: false,
allowNull: true,
defaultValue: null,
comment: "收入确认时间",
primaryKey: false,

12
api/app/lib/routes/report/index.js

@ -15,4 +15,16 @@ module.exports = function (app, router, opts) {
app.fs.api.logAttr['GET/detail/achievement'] = { content: '查询业绩明细表', visible: false };
router.get('/detail/achievement', achieve.getAchievementDetail);
app.fs.api.logAttr['GET/detail/received/numbers'] = { content: '查询业绩明细已有的编号集合', visible: false };
router.get('/detail/received/numbers', achieve.getReceivedNumbers);
app.fs.api.logAttr['POST/add/received/back/bulk'] = { content: '导入回款明细', visible: true };
router.post('/add/received/back/bulk', achieve.importBackDetails);
app.fs.api.logAttr['POST/add/achievement/bulk'] = { content: '导入业绩明细', visible: true };
router.post('/add/achievement/bulk', achieve.importAchieveDetails);
};

35
web/client/src/sections/business/actions/achievement-report.js

@ -26,4 +26,39 @@ export function getAchievementDetail(query) {
msg: { option: "查询业绩明细表" },
reducer: { name: "AchievementDetail", params: { noClear: true } },
});
}
//回款明细 编号集合
export function getReceivedNumbers() {
return (dispatch) => basicAction({
type: "get",
dispatch: dispatch,
actionType: "GET_RECEIVED_DETAIL_NUMBERS",
url: `${ApiTable.getReceivedNumbers}`,
msg: { option: "查询回款明细编号集合" },
reducer: { name: "ReceivedDetailNumbers", params: { noClear: true } },
});
}
//导入回款明细
export function importBackDetails(values) {
return dispatch => basicAction({
type: 'post',
dispatch: dispatch,
actionType: 'RECEIVED_DETAIL_BULK_ADD',
url: ApiTable.importBackDetails,
data: values,
msg: { option: '导入回款明细' },
});
}
//导入业绩明细
export function importAchieveDetails(values) {
return dispatch => basicAction({
type: 'post',
dispatch: dispatch,
actionType: 'ACHIEVEMENT_DETAIL_BULK_ADD',
url: ApiTable.importAchieveDetails,
data: values,
msg: { option: '导入业绩明细' },
});
}

48
web/client/src/sections/business/constants/index.js

@ -71,4 +71,52 @@ export const invoicingDetailsColumnKeys = {
taxRate13: '税率13%',
taxRate9: '税率9%',
taxRate6: '税率6%'
}
export const backMoneyColumnKeys = {
year: '年度',
serialNo: '序号',
number: '编号',
department: '部门',
sale: '销售人员',
contractNo: '合同编号',
customer: '客户名称',
item: '项目名称',
amount: '合同金额',
changeAmount: '变更后合同金额',
receivableYear: '回款年份',
receivableDate: '回款日期',
receivableAmount: '回款金额',
invoicedBack: '开票-回款',
remainConAmount: '剩余合同金额',
incomeConfirmdate: '收入确认时间',
thirdPayment: '第三方付款单位',
remark: '备注',
}
export const achievementColumnKeys = {
recConDate: '收到合同日期',
month: '月份',
department: '部门',
sale: '销售人员',
customer: '客户名称',
item: '项目名称',
amount: '合同金额',
realPerformance: '实际业绩',
assessmentPerformance: '考核业绩',
isApproval: '价格是否特批',
approvalProp: '特批折算比例',
cost: '预支提成及委外费用',
serviceLine: '业务线',
cusType: '客户类型',
industry: '行业',
source: '信息来源',
itemType: '项目类型',
cusProvince: '客户省份',
cusAttribute: '客户属性',
repurchaseCount: '复购次数',
reproducible: '是否可复制的业务路径',
outProvince: '省外业务1.1',
repurchase: '复购业务1.05',
isreproduce: '可复制的业务路径1.1',
}

175
web/client/src/sections/business/containers/performanceReport/achievementDetails.jsx

@ -4,6 +4,8 @@ import moment from 'moment'
import { Select, Input, Button, Toast, Radio, Tooltip, Table, Pagination, Skeleton } from '@douyinfe/semi-ui';
import { IconSearch } from '@douyinfe/semi-icons';
import { SkeletonScreen } from "$components";
import { achievementColumnKeys } from '../../constants/index';
import ImportAchieveModal from './importAchieveModal'
import '../../style.less';
const AchievementDetails = (props) => {
@ -14,7 +16,8 @@ const AchievementDetails = (props) => {
const [limits, setLimits] = useState()//
const [query, setQuery] = useState({ limit: 10, page: 0 }); //
const [tableData, setTableData] = useState([]);
const [exportUrl, setExportUrl] = useState('')
const [importModalV, setImportModalV] = useState(false);
const [exportUrl, setExportUrl] = useState('');
const page = useRef(query.page);
function seachValueChange(value) {
setKeyword(value)
@ -37,151 +40,22 @@ const AchievementDetails = (props) => {
})
}
const columns = [{
title: '收到合同日期',
dataIndex: 'recConDate',
key: 'recConDate',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '月份',
dataIndex: 'month',
key: 'month',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '部门',
dataIndex: 'department',
key: 'department',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '销售人员',
dataIndex: 'sale',
key: 'sale',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '客户名称',
dataIndex: 'customer',
key: 'customer',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '项目名称',
dataIndex: 'item',
key: 'item',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '合同金额',
dataIndex: 'amount',
key: 'amount',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '实际业绩',
dataIndex: 'realPerformance',
key: 'realPerformance',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '考核业绩',
dataIndex: 'assessmentPerformance',
key: 'assessmentPerformance',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '价格是否特批',
dataIndex: 'isApproval',
key: 'isApproval',
width: 120,
render: (text, record) => <span>{text ? '是' : '否'}</span>
}, {
title: '特批折算比例',
dataIndex: 'approvalProp',
key: 'approvalProp',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '预支提成及委外费用',
dataIndex: 'cost',
key: 'cost',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '业务线',
dataIndex: 'serviceLine',
key: 'serviceLine',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '客户类型',
dataIndex: 'cusType',
key: 'cusType',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '行业',
dataIndex: 'industry',
key: 'industry',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '信息来源',
dataIndex: 'source',
key: 'source',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '项目类型',
dataIndex: 'itemType',
key: 'itemType',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '客户省份',
dataIndex: 'cusProvince',
key: 'cusProvince',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '客户属性',
dataIndex: 'cusAttribute',
key: 'cusAttribute',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '复购次数',
dataIndex: 'repurchaseCount',
key: 'repurchaseCount',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '是否可复制的业务路径',
dataIndex: 'reproducible',
key: 'reproducible',
width: 120,
render: (text, record) => <span>{text ? '是' : '否'}</span>
}, {
title: '省外业务1.1',
dataIndex: 'outProvince',
key: 'outProvince',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '复购业务1.05',
dataIndex: 'repurchase',
key: 'repurchase',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '可复制的业务路径1.1',
dataIndex: 'isreproduce',
key: 'isreproduce',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}]
const renderColumns = (columnKeys) => {
let columns = [];
Object.keys(columnKeys).map(key => {
switch (key) {
default:
columns.push({
title: columnKeys[key], dataIndex: key, key: key, width: 130,
render: (text, record) => ['isApproval', 'reproducible'].indexOf(key) != -1 ? JSON.stringify(text) === 'null' ? '-' : text ? '是' : '否'
: text === 0 ? text : text ? text : '-'
});
break;
}
})
return columns;
}
function handleRow(record, index) {//
if (index % 2 === 0) {
return {
@ -273,7 +147,7 @@ const AchievementDetails = (props) => {
placeholder={SkeletonScreen()}
>
<Table
columns={columns}
columns={renderColumns(achievementColumnKeys)}
dataSource={tableData}
bordered={false}
empty="暂无数据"
@ -307,6 +181,13 @@ const AchievementDetails = (props) => {
</div>
</div>
</div>
{
importModalV ? <ImportAchieveModal
onCancel={() => {
setImportModalV(false);
getAchievementDetails();
}} /> : ''
}
{
exportUrl ? <iframe src={`/_api/${exportUrl}`} style={{ display: 'none' }} /> : ''
}

139
web/client/src/sections/business/containers/performanceReport/backMoneyDetails.jsx

@ -4,8 +4,9 @@ import moment from 'moment'
import { Select, Input, Button, Toast, Radio, Tooltip, Table, Pagination, Skeleton } from '@douyinfe/semi-ui';
import { IconSearch } from '@douyinfe/semi-icons';
import { SkeletonScreen } from "$components";
import ImportBackModal from './importBackModal';
import '../../style.less';
import { backMoneyColumnKeys } from '../../constants/index';
const BackMoneyDetails = (props) => {
const { dispatch, actions, user } = props
const { businessManagement } = actions
@ -14,7 +15,8 @@ const BackMoneyDetails = (props) => {
const [limits, setLimits] = useState()//
const [query, setQuery] = useState({ limit: 10, page: 0 }); //
const [tableData, setTableData] = useState([]);
const [exportUrl, setExportUrl] = useState('')
const [importModalV, setImportModalV] = useState(false);
const [exportUrl, setExportUrl] = useState('');
const page = useRef(query.page);
function seachValueChange(value) {
setKeyword(value)
@ -37,115 +39,21 @@ const BackMoneyDetails = (props) => {
})
}
const columns = [{
title: '年度',
dataIndex: 'year',
key: 'year',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '序号',
dataIndex: 'serialNo',
key: 'serialNo',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '编号',
dataIndex: 'number',
key: 'number',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '部门',
dataIndex: 'department',
key: 'department',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '销售人员',
dataIndex: 'sale',
key: 'sale',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '合同编号',
dataIndex: 'contractNo',
key: 'contractNo',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '客户名称',
dataIndex: 'customer',
key: 'customer',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '项目名称',
dataIndex: 'item',
key: 'item',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '合同金额',
dataIndex: 'amount',
key: 'amount',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '变更后合同金额',
dataIndex: 'changeAmount',
key: 'changeAmount',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '回款年份',
dataIndex: 'receivableYear',
key: 'receivableYear',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '回款日期',
dataIndex: 'receivableDate',
key: 'receivableDate',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '回款金额',
dataIndex: 'receivableAmount',
key: 'receivableAmount',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '开票-回款',
dataIndex: 'invoicedBack',
key: 'invoicedBack',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '剩余合同金额',
dataIndex: 'remainConAmount',
key: 'remainConAmount',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '收入确认时间',
dataIndex: 'incomeConfirmdate',
key: 'incomeConfirmdate',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '第三方付款单位',
dataIndex: 'thirdPayment',
key: 'thirdPayment',
width: 140,
render: (text, record) => <span>{text || '-'}</span>
}, {
title: '备注',
dataIndex: 'remark',
key: 'remark',
width: 120,
render: (text, record) => <span>{text || '-'}</span>
}]
const renderColumns = (columnKeys) => {
let columns = [];
Object.keys(columnKeys).map(key => {
switch (key) {
default:
columns.push({
title: columnKeys[key], dataIndex: key, key: key, width: 130,
render: (text, record) => text === 0 ? text : text ? text : '-'
});
break;
}
})
return columns;
}
function handleRow(record, index) {//
if (index % 2 === 0) {
return {
@ -237,7 +145,7 @@ const BackMoneyDetails = (props) => {
placeholder={SkeletonScreen()}
>
<Table
columns={columns}
columns={renderColumns(backMoneyColumnKeys)}
dataSource={tableData}
bordered={false}
empty="暂无数据"
@ -271,6 +179,13 @@ const BackMoneyDetails = (props) => {
</div>
</div>
</div>
{
importModalV ? <ImportBackModal
onCancel={() => {
setImportModalV(false);
getBackMoneyDetails();
}} /> : ''
}
{
exportUrl ? <iframe src={`/_api/${exportUrl}`} style={{ display: 'none' }} /> : ''
}

220
web/client/src/sections/business/containers/performanceReport/importAchieveModal.jsx

@ -0,0 +1,220 @@
'use strict';
import React, { useState, useEffect } from 'react';
import { connect } from 'react-redux';
import moment from 'moment';
import { Modal, Form, Button, Notification } from '@douyinfe/semi-ui';
import { IconUpload } from '@douyinfe/semi-icons';
import XLSX from 'xlsx';
import { achievementColumnKeys } from '../../constants/index';
//
const ImportAchieveModal = props => {
const { dispatch, actions, onCancel } = props;
const { businessManagement } = actions
const [msg, setMsg] = useState('');
const [loading, setLoading] = useState('');
const [postData, setPostData] = useState([]);
//
useEffect(() => {
}, []);
const confirm = () => {
if (postData.length) {
setLoading(true)
dispatch(businessManagement.importAchieveDetails(postData)).then(res => {
if (res.success) {
onCancel()
}
setLoading(false)
})
} else {
Notification.warning({ content: '没有数据可以提交,请上传数据文件', duration: 2 })
}
}
const dldCsvMb = () => {
//
let head = [];
Object.keys(achievementColumnKeys).map(key => {
head.push(achievementColumnKeys[key]);
})
head = head.join(',') + "\n";
//
//let data = 1 + ',' + 2 + ',' + 3 + ',' + 4 + ',' + 5
let templateCsv = "data:text/csv;charset=utf-8,\ufeff" + head;
//a
let link = document.createElement("a");
//a
link.setAttribute("href", templateCsv);
link.setAttribute("download", "业绩明细表导入模板.csv");
//a
link.click();
}
const download = () => {
dldCsvMb();
}
const fileLimit = '.csv';
const getFileBlob = (url) => {
return new Promise((resolve, reject) => {
let request = new XMLHttpRequest()
request.open("GET", url, true)
request.responseType = "blob"
request.onreadystatechange = e => {
if (request.readyState == 4) {
if (request.status == 200) {
if (window.FileReader) {
let reader = new FileReader();
reader.readAsBinaryString(request.response);
reader.onload = event => {
try {
const { result } = event.target;
// excel
const workbook = XLSX.read(result, {
type: "binary",
cellDates: true,//true
codepage: 936//
});
let data = []; //
//
for (const sheet in workbook.Sheets) {
if (workbook.Sheets.hasOwnProperty(sheet)) {
data = data.concat(XLSX.utils.sheet_to_json(workbook.Sheets[sheet]));
}
}
resolve(data);//
} catch (e) {
reject("失败");
}
}
}
}
}
}
request.send();
})
}
const judgeNull = (value) => {
return value ? String(value).trim().replace(/\s*/g, "") : null;
}
const judgeNullTime = (v) => {
//xlsxexcel
//2020/8/1xlsx Fri Jul 31 2020 23:59:17 GMT+0800 43
let a = new Date(v);
a.setTime(a.getTime() + 43 * 1000);
return v ? a : null;
}
const judgeTimeValid = (time) => {
let valid = true;
if (!time) {
return valid;//
}
const ymd = /^((19|20)[0-9]{2})[\/\-]((0[1-9])|(1[0-2]))[\/\-]((0[1-9])|((1|2)[0-9])|(3[0-1]))$/;//
if (time instanceof Date) {
let timeStr = moment(time).format('YYYY/MM/DD');
if (!ymd.test(timeStr)) {
valid = false;
}
} else {
valid = false;
}
return valid;
}
return (
<Modal
title="导入" visible={true}
onOk={confirm} width={620}
confirmLoading={loading}
onCancel={() => {
setMsg('')
setLoading(false)
setPostData([])
onCancel()
}}
>
<div style={{ borderBottom: '1px solid #DCDEE0', margin: '0px -24px' }}></div>
<Form>
<Form.Upload
label={'业绩明细表'} labelPosition='left'
action={'/'} accept={fileLimit}
maxSize={200} limit={1}
onRemove={(currentFile, fileList, fileItem) => {
setMsg('');
setPostData([]);
}}
customRequest={(data) => {
const { file, onSuccess, onError } = data
getFileBlob(file.url).then(res => {
const error = (msg) => {
setMsg(msg)
onError({ message: msg })
}
if (res.length > 1000) {
error('一次性上传数据行数应小于1000行,请分批上传')
return
}
if (!res.length) {
error('请填写至少一行数据')
return
}
let postData = [];
for (let i = 0; i < res.length; i++) {
let d = res[i];
let obj = {};
Object.keys(achievementColumnKeys).map(key => {
if (key === 'recConDate') {//
obj[key] = judgeNullTime(d[achievementColumnKeys[key]]);
} else {
obj[key] = d[achievementColumnKeys[key]] || null;
}
})
let tValid = judgeTimeValid(obj.recConDate);
if (!tValid) {
error(`${i + 2}行【收到合同日期】错误,请填写yyyy/mm/dd格式`)
return
}
if (obj.isApproval && ['是', '否'].indexOf(obj.isApproval) == -1) {
error(`${i + 2}行【价格是否特批】错误,请填写是或否`)
return
}
if (obj.reproducible && ['是', '否'].indexOf(obj.reproducible) == -1) {
error(`${i + 2}行【是否可复制的业务路径】错误,请填写是或否`)
return
}
postData.push(obj);
}
setPostData(postData)
let msg = '文件解析完成,点击确定按钮上传保存!'
setMsg(msg)
onSuccess({ message: msg })
})
}}>
<Button icon={<IconUpload />} theme="light">
请选择文件
</Button>
</Form.Upload>
<span>{msg}</span>
<div style={{ color: '#ccc' }}>最大不超过200M导入文件需与
<span onClick={() => download()} style={{ cursor: 'pointer', color: '#0066FF' }}>导入模板</span>
一致</div>
</Form>
</Modal >
)
}
function mapStateToProps(state) {
const { auth, global } = state;
return {
user: auth.user,
actions: global.actions,
}
}
export default connect(mapStateToProps)(ImportAchieveModal);

256
web/client/src/sections/business/containers/performanceReport/importBackModal.jsx

@ -0,0 +1,256 @@
'use strict';
import React, { useState, useEffect } from 'react';
import { connect } from 'react-redux';
import moment from 'moment';
import { Modal, Form, Button, Notification } from '@douyinfe/semi-ui';
import { IconUpload } from '@douyinfe/semi-icons';
import XLSX from 'xlsx';
import { backMoneyColumnKeys } from '../../constants/index';
//
const ImportBackModal = props => {
const { dispatch, actions, onCancel } = props;
const { businessManagement } = actions
const [msg, setMsg] = useState('');
const [loading, setLoading] = useState('');
const [postData, setPostData] = useState([]);
const [uploadAble, setUploadAble] = useState(false);
const [allNumbers, setAllNumbers] = useState([]);
//
useEffect(() => {
dispatch(businessManagement.getReceivedNumbers()).then(r => {
if (r.success) {
setUploadAble(true);
setAllNumbers(r.payload.data);
}
})
}, []);
const confirm = () => {
if (postData.length) {
setLoading(true)
dispatch(businessManagement.importBackDetails(postData)).then(res => {
if (res.success) {
onCancel()
}
setLoading(false)
})
} else {
Notification.warning({ content: '没有数据可以提交,请上传数据文件', duration: 2 })
}
}
const dldCsvMb = () => {
//
let head = [];
Object.keys(backMoneyColumnKeys).map(key => {
head.push(backMoneyColumnKeys[key]);
})
head = head.join(',') + "\n";
//
//let data = 1 + ',' + 2 + ',' + 3 + ',' + 4 + ',' + 5
let templateCsv = "data:text/csv;charset=utf-8,\ufeff" + head;
//a
let link = document.createElement("a");
//a
link.setAttribute("href", templateCsv);
link.setAttribute("download", "回款明细表导入模板.csv");
//a
link.click();
}
const download = () => {
dldCsvMb();
}
const fileLimit = '.csv';
const getFileBlob = (url) => {
return new Promise((resolve, reject) => {
let request = new XMLHttpRequest()
request.open("GET", url, true)
request.responseType = "blob"
request.onreadystatechange = e => {
if (request.readyState == 4) {
if (request.status == 200) {
if (window.FileReader) {
let reader = new FileReader();
reader.readAsBinaryString(request.response);
reader.onload = event => {
try {
const { result } = event.target;
// excel
const workbook = XLSX.read(result, {
type: "binary",
cellDates: true,//true
codepage: 936//
});
let data = []; //
//
for (const sheet in workbook.Sheets) {
if (workbook.Sheets.hasOwnProperty(sheet)) {
data = data.concat(XLSX.utils.sheet_to_json(workbook.Sheets[sheet]));
}
}
resolve(data);//
} catch (e) {
reject("失败");
}
}
}
}
}
}
request.send();
})
}
const judgeNull = (value) => {
return value ? String(value).trim().replace(/\s*/g, "") : null;
}
const judgeNullTime = (v) => {
//xlsxexcel
//2020/8/1xlsx Fri Jul 31 2020 23:59:17 GMT+0800 43
let a = new Date(v);
a.setTime(a.getTime() + 43 * 1000);
return v ? a : null;
}
const judgeTimeValid = (time) => {
let valid = true;
if (!time) {
return valid;//
}
const ymd = /^((19|20)[0-9]{2})[\/\-]((0[1-9])|(1[0-2]))[\/\-]((0[1-9])|((1|2)[0-9])|(3[0-1]))$/;//
if (time instanceof Date) {
let timeStr = moment(time).format('YYYY/MM/DD');
if (!ymd.test(timeStr)) {
valid = false;
}
} else {
valid = false;
}
return valid;
}
return (
<Modal
title="导入" visible={true}
onOk={confirm} width={620}
confirmLoading={loading}
onCancel={() => {
setMsg('')
setLoading(false)
setPostData([])
onCancel()
}}
>
<div style={{ borderBottom: '1px solid #DCDEE0', margin: '0px -24px' }}></div>
<Form>
<Form.Upload
label={'回款明细表'} labelPosition='left'
action={'/'} accept={fileLimit}
maxSize={200} limit={1}
onRemove={(currentFile, fileList, fileItem) => {
setMsg('');
setPostData([]);
}}
customRequest={(data) => {
const { file, onSuccess, onError } = data
getFileBlob(file.url).then(res => {
const error = (msg) => {
setMsg(msg)
onError({ message: msg })
}
if (res.length > 1000) {
error('一次性上传数据行数应小于1000行,请分批上传')
return
}
if (!res.length) {
error('请填写至少一行数据')
return
}
let postData = [];
const numPattern = /^\d+(\.\d+)?$/;//
for (let i = 0; i < res.length; i++) {
let d = res[i];
let obj = {};
Object.keys(backMoneyColumnKeys).map(key => {
if (['receivableDate', 'incomeConfirmdate'].indexOf(key) != -1) {//
obj[key] = judgeNullTime(d[backMoneyColumnKeys[key]]);
} else {
obj[key] = d[backMoneyColumnKeys[key]] || null;
}
})
//
if (!obj.year || !obj.serialNo || !obj.number) {
error(`${i + 2}行【年度】、【序号】、【编号】存在空值,请填写`)
return
}
let exist = allNumbers.find(m => m.number == obj.number);//
if (exist) {
error(`${i + 2}行的【编号】在系统中已存在`)
return
}
if (postData.some(p => p.number == obj.number)) {//
error(`${i + 2}行【编号】重复,请更改后重新上传`)
return
}
//
if (obj.amount && !numPattern.test(obj.amount)) {
error(`${i + 2}行【合同金额】填写错误,需要为数字`)
return
}
if (obj.changeAmount && !numPattern.test(obj.changeAmount)) {
error(`${i + 2}行【变更后合同金额】填写错误,需要为数字`)
return
}
if (obj.receivableAmount && !numPattern.test(obj.receivableAmount)) {
error(`${i + 2}行【回款金额】填写错误,需要为数字`)
return
}
if (obj.remainConAmount && !numPattern.test(obj.remainConAmount)) {
error(`${i + 2}行【剩余合同金额】填写错误,需要为数字`)
return
}
//
let tValid = judgeTimeValid(obj.receivableDate);
if (!tValid) {
error(`${i + 2}行回款日期错误,请填写yyyy/mm/dd格式`)
return
}
let cValid = judgeTimeValid(obj.incomeConfirmdate);
if (!cValid) {
error(`${i + 2}行收入确认时间错误,请填写yyyy/mm/dd格式`)
return
}
postData.push(obj);
}
setPostData(postData)
let msg = '文件解析完成,点击确定按钮上传保存!'
setMsg(msg)
onSuccess({ message: msg })
})
}}>
<Button icon={<IconUpload />} theme="light" disabled={!uploadAble}>
请选择文件
</Button>
</Form.Upload>
<span>{msg}</span>
<div style={{ color: '#ccc' }}>最大不超过200M导入文件需与
<span onClick={() => download()} style={{ cursor: 'pointer', color: '#0066FF' }}>导入模板</span>
一致</div>
</Form>
</Modal >
)
}
function mapStateToProps(state) {
const { auth, global } = state;
return {
user: auth.user,
actions: global.actions,
}
}
export default connect(mapStateToProps)(ImportBackModal);

4
web/client/src/utils/webapi.js

@ -23,6 +23,10 @@ export const ApiTable = {
//业绩报表
getReceivedDetail: 'detail/received/back',
getAchievementDetail: 'detail/achievement',
getReceivedNumbers: 'detail/received/numbers',
importBackDetails: 'add/received/back/bulk',
importAchieveDetails: 'add/achievement/bulk'
};
export const RouteTable = {
apiRoot: "/api/root",

Loading…
Cancel
Save