index.tsx 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. /*
  2. * @Author: code4eat awesomedema@gmail.com
  3. * @Date: 2023-03-03 11:30:33
  4. * @LastEditors: code4eat awesomedema@gmail.com
  5. * @LastEditTime: 2023-05-30 11:21:57
  6. * @FilePath: /KC-MiddlePlatform/src/pages/platform/setting/pubDicTypeMana/index.tsx
  7. * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
  8. */
  9. import BMSPagecontainer from '@/components/BMSPageContainer';
  10. import { BMSTable, BMSTablePageDefaultConfig } from '@/components/BMSTable';
  11. import { deepEqual } from '@/utils/tooljs';
  12. import { createFromIconfontCN } from '@ant-design/icons';
  13. import { ActionType } from '@ant-design/pro-components';
  14. import { ModalForm, ProFormDigit, ProFormRadio, ProFormSelect, ProFormText, ProFormTextArea } from '@ant-design/pro-form';
  15. import { ProColumns } from '@ant-design/pro-table';
  16. import { Input, message, Modal, Popconfirm, Table } from 'antd';
  17. import { ColumnsType, TableRowSelection } from 'antd/es/table/interface';
  18. import Transfer, { TransferItem, TransferProps } from 'antd/es/transfer';
  19. import React from 'react';
  20. import { useEffect, useImperativeHandle, useRef, useState } from 'react';
  21. import { getAllUnit } from '../indicGroupWeightSet/service';
  22. import difference from 'lodash/difference';
  23. import { copyGroupUnit, delData, editUnitIndicTargetTbaleData, getUnitGroupList, getUnitIndicTargetTableData, getUnitUnderGroupById } from './service';
  24. import './style.less';
  25. import '../../../../utils/zhongtaiB'
  26. const IconFont = createFromIconfontCN({
  27. scriptUrl: '',
  28. });
  29. const UnitIndicTargetSet = () => {
  30. const [tableDataFilterParams, set_tableDataFilterParams] = useState<any | undefined>();
  31. const [tableDataSearchKeywords, set_tableDataSearchKeywords] = useState<string>('');
  32. const [tableColumn, set_tableColumn] = useState<ProColumns[] | any[]>([]);
  33. const [typeList, set_typeList] = useState<any[]>([]);
  34. const [showTypeListArr, set_showTypeListArr] = useState<any[]>([]);
  35. const [currentSelectedType, set_currentSelectedType] = useState<any | undefined>(undefined);
  36. const tableRef = useRef<ActionType>();
  37. const getTableData = async (params: any) => {
  38. const { groupId } = params;
  39. console.log({ params });
  40. if (groupId) {
  41. const resp = await getUnitIndicTargetTableData(params);
  42. if (resp) {
  43. const { title, unitWeightData } = resp;
  44. const columns = title.map((item: any) => {
  45. return {
  46. title: item.name,
  47. dataIndex: `${item.code}`,
  48. key: `${item.code}`,
  49. // render:(_:number,record:any)=>{
  50. // return `${Number(_ * 100).toFixed(2)}%`
  51. // }
  52. // width: 140,
  53. }
  54. });
  55. set_tableColumn([{
  56. title: '核算单元',
  57. dataIndex: 'unitName',
  58. }, ...columns, {
  59. title: '操作',
  60. key: 'option',
  61. width: 120,
  62. valueType: 'option',
  63. render: (_: any, record: any) => {
  64. return [
  65. <UpDataActBtn key={'act'} record={record} type='EDIT' />,
  66. <a key={'copy'} onClick={()=>copyHandle(record)}>复制</a>
  67. ]
  68. },
  69. }]);
  70. const data = unitWeightData.map((item: any) => {
  71. let rowData: { [key: string]: any } = {};
  72. for (let index = 0; index < item.value.length; index++) {
  73. for (const key in item.value[index]) {
  74. if (key == 'code') {
  75. rowData[`${item.value[index][`code`]}`] = item.value[index][`target`];
  76. rowData[`targetValue${item.value[index][`code`]}`] = item.value[index][`targetValue`]
  77. }
  78. }
  79. }
  80. return { ...item, ...rowData, columns }
  81. });
  82. console.log({data});
  83. return {
  84. data,
  85. success: true
  86. }
  87. }
  88. return []
  89. }
  90. return []
  91. }
  92. //获取左侧单元
  93. const getTypeList = async () => {
  94. const resp = await getUnitGroupList();
  95. if (resp) {
  96. set_typeList(resp);
  97. set_showTypeListArr(resp);
  98. }
  99. }
  100. const updateTable = async (formVal: any, type: 'EDIT' | "ADD") => {
  101. const temp = formVal.columns.map((a:any)=>{
  102. return {
  103. code:a.key,
  104. targetValue:formVal[`targetValue${a.key}`],
  105. }
  106. })
  107. if (type == 'EDIT') {
  108. const result = {
  109. unitCode:formVal.unitCode,
  110. unitName:formVal.unitName,
  111. value:temp
  112. }
  113. const resp = await editUnitIndicTargetTbaleData(result);
  114. if (resp) {
  115. tableRef.current?.reload();
  116. }
  117. }
  118. return true;
  119. }
  120. const UpDataActBtn = ({ record, type }: { record: any, type: 'EDIT' | 'ADD' }) => {
  121. return (
  122. <ModalForm
  123. title={`设定指标目标值(${record.unitName})`}
  124. width={352}
  125. initialValues={type == 'EDIT' ? { ...record } : { }}
  126. trigger={
  127. type == 'EDIT' ? <a key="edit" >编辑</a> : <span className='add'>新增</span>
  128. }
  129. onFinish={(val) => {
  130. return updateTable(type == 'EDIT' ? { ...record, ...val } : val, type);
  131. }}
  132. >
  133. {
  134. record.value.map((item:any, index:number) => {
  135. const label = record.columns.filter((a:ProColumns)=>a.key == item.code);
  136. if(label.length>0){
  137. return (
  138. <ProFormDigit
  139. key={index}
  140. name={`targetValue${item.code}`}
  141. label={`${label[0].title}`}
  142. placeholder="请输入"
  143. rules={[{ required: true, message:`请填写${label[0].title}` }]}
  144. />
  145. )
  146. }
  147. })
  148. }
  149. </ModalForm>
  150. )
  151. }
  152. interface DataType {
  153. key: string;
  154. title: string;
  155. description: string;
  156. disabled: boolean;
  157. tag: string;
  158. }
  159. interface TableTransferProps extends TransferProps<TransferItem> {
  160. dataSource: DataType[];
  161. leftColumns: ColumnsType<DataType>;
  162. rightColumns: ColumnsType<DataType>;
  163. record: any
  164. }
  165. const transferTableColumn:any[] = [
  166. {
  167. title: '核算单元',
  168. dataIndex: 'unitName',
  169. key: 'unitName',
  170. },
  171. {
  172. title: '职类',
  173. dataIndex: 'unitType',
  174. key: 'unitType',
  175. render: (_: any, record: any) => {
  176. switch (parseInt(record.unitType)) {
  177. case 1:
  178. return '临床'
  179. break;
  180. case 2:
  181. return '医技'
  182. break;
  183. case 3:
  184. return '护理'
  185. break;
  186. case 6:
  187. return '行政后勤'
  188. break;
  189. default:
  190. break;
  191. }
  192. }
  193. },
  194. ];
  195. const copyHandle = (record: any) => {
  196. const ref = React.createRef<{ save: any; }>();
  197. Modal.confirm({
  198. title: `复制单元考核目标设定信息(${record.unitName})`,
  199. icon: <></>,
  200. width: 672,
  201. content: <TableTransfer
  202. ref={ref}
  203. record={record}
  204. leftColumns={transferTableColumn}
  205. rightColumns={transferTableColumn} dataSource={[]}
  206. ></TableTransfer>,
  207. onOk: () => {
  208. return ref.current && ref.current.save();
  209. }
  210. })
  211. }
  212. const TableTransfer = React.forwardRef(({ leftColumns, rightColumns, record, ...restProps }: TableTransferProps, ref) => {
  213. const [_data, _set_data] = useState<any>();
  214. const [targetKeys, setTargetKeys] = useState<string[]>([]);
  215. const [datasource, set_datasource] = useState<any[]>([]);
  216. const [selectedKeys, setSelectedKeys] = useState<string[]>([]);
  217. //获取单元
  218. const getFuncList = async () => {
  219. const resp = await getAllUnit();
  220. if (resp) {
  221. set_datasource(resp);
  222. if (record && record.unitInfoVos) {
  223. const defaultSelctedkeys = record.unitInfoVos.map((item: any) => item.unitCode);
  224. setTargetKeys(defaultSelctedkeys);
  225. }
  226. }
  227. }
  228. const onChange = (nextTargetKeys: string[]) => {
  229. setTargetKeys(nextTargetKeys);
  230. };
  231. const onSelectChange = (sourceSelectedKeys: string[], targetSelectedKeys: string[]) => {
  232. //console.log('sourceSelectedKeys:', sourceSelectedKeys,'targetSelectedKeys:',targetSelectedKeys);
  233. setSelectedKeys([...sourceSelectedKeys, ...targetSelectedKeys]);
  234. };
  235. useImperativeHandle(ref, () => ({
  236. save: async () => {
  237. const needData = datasource.filter(item => targetKeys.includes(item.code));
  238. const result = needData.map(a=>({unitCode:a.code,unitName:a.unitName}));
  239. // console.log({result,record});
  240. const resp = await copyGroupUnit({
  241. unitInfos:result,
  242. value:record.value
  243. });
  244. if (resp) {
  245. message.success('操作成功!');
  246. tableRef.current?.reload();
  247. }
  248. }
  249. }));
  250. useEffect(() => {
  251. getFuncList();
  252. }, [])
  253. return (
  254. <Transfer className='TableTransfer' showSearch
  255. titles={['待选项', '已选项']}
  256. locale={{
  257. itemUnit: '项',
  258. itemsUnit: '项',
  259. searchPlaceholder: '请输入'
  260. }}
  261. onChange={onChange}
  262. onSelectChange={onSelectChange}
  263. dataSource={datasource}
  264. rowKey={record => record.code}
  265. targetKeys={targetKeys}
  266. selectedKeys={selectedKeys}
  267. filterOption={(inputValue, item) => {
  268. return item.unitName!.indexOf(inputValue) !== -1
  269. }}
  270. >
  271. {({
  272. direction,
  273. filteredItems,
  274. onItemSelectAll,
  275. onItemSelect,
  276. selectedKeys: listSelectedKeys,
  277. disabled: listDisabled,
  278. }) => {
  279. // console.log({ filteredItems, listSelectedKeys,direction });
  280. const columns = direction === 'left' ? leftColumns : rightColumns;
  281. const rowSelection: TableRowSelection<TransferItem> = {
  282. getCheckboxProps: (item) => ({ disabled: listDisabled || item.disabled }),
  283. onSelectAll(selected, selectedRows) {
  284. const treeSelectedKeys = selectedRows.map(({ code }) => code);
  285. const diffKeys = selected
  286. ? difference(treeSelectedKeys, listSelectedKeys)
  287. : difference(listSelectedKeys, treeSelectedKeys);
  288. onItemSelectAll(diffKeys as string[], selected);
  289. },
  290. onSelect({ code }, selected) {
  291. onItemSelect(code as string, selected);
  292. },
  293. selectedRowKeys: listSelectedKeys,
  294. };
  295. return (
  296. <Table
  297. rowSelection={rowSelection}
  298. columns={columns as TransferItem[]}
  299. dataSource={filteredItems}
  300. size="small"
  301. rowKey={'code'}
  302. style={{ pointerEvents: listDisabled ? 'none' : undefined }}
  303. onRow={({ code, disabled: itemDisabled }) => ({
  304. onClick: () => {
  305. if (itemDisabled || listDisabled) return;
  306. onItemSelect(code as string, !listSelectedKeys.includes(code as string));
  307. },
  308. })}
  309. />
  310. );
  311. }}
  312. </Transfer>
  313. )
  314. });
  315. const tableDataSearchHandle = (paramName: string) => {
  316. set_tableDataFilterParams({
  317. ...tableDataFilterParams,
  318. [`${paramName}`]: tableDataSearchKeywords
  319. })
  320. }
  321. useEffect(() => {
  322. if (currentSelectedType) {
  323. set_tableDataFilterParams({ ...tableDataFilterParams, groupId: currentSelectedType.id })
  324. }
  325. }, [currentSelectedType])
  326. useEffect(() => {
  327. if (showTypeListArr.length > 0) {
  328. set_currentSelectedType(showTypeListArr[0]);
  329. set_tableDataFilterParams({ ...tableDataFilterParams, groupId: showTypeListArr[0].id });
  330. }
  331. }, [showTypeListArr])
  332. useEffect(() => {
  333. getTypeList();
  334. }, [])
  335. return (
  336. <BMSPagecontainer title={false} className='UnitIndicTargetSet'>
  337. <div className='left'>
  338. <Input placeholder={'请输入'} allowClear
  339. suffix={
  340. <IconFont type="iconsousuo" />
  341. }
  342. onChange={(e) => {
  343. const result = typeList.filter(item => item.groupName.indexOf(e.target.value) != -1);
  344. set_showTypeListArr(result);
  345. }}
  346. />
  347. <div className='wrap'>
  348. {
  349. showTypeListArr.map((item, index) => {
  350. return (
  351. <div className={currentSelectedType ? currentSelectedType.id == item.id ? 'type on' : 'type' : 'type'}
  352. key={index}
  353. onClick={() => set_currentSelectedType(item)}
  354. >{item.groupName}</div>
  355. )
  356. })
  357. }
  358. </div>
  359. </div>
  360. <div className='right'>
  361. <div className='toolBar'>
  362. <div className='filter'>
  363. <div className='filterItem'>
  364. <span className='label' style={{ whiteSpace: 'nowrap' }}> 检索:</span>
  365. <Input placeholder={'请输入单元'} allowClear
  366. suffix={
  367. <IconFont type="iconsousuo" onClick={() => tableDataSearchHandle('unitCode')} />
  368. }
  369. onChange={(e) => {
  370. set_tableDataSearchKeywords(e.target.value);
  371. if (e.target.value.length == 0) {
  372. set_tableDataFilterParams({
  373. ...tableDataFilterParams,
  374. unitCode: ''
  375. });
  376. }
  377. }}
  378. onPressEnter={(e) => {
  379. set_tableDataFilterParams({
  380. ...tableDataFilterParams,
  381. unitCode: (e.target as HTMLInputElement).value
  382. });
  383. }}
  384. />
  385. </div>
  386. </div>
  387. <div className='btnGroup'>
  388. {/* <UpDataActBtn record type='ADD' /> */}
  389. </div>
  390. </div>
  391. <div style={{ marginTop: 16 }}>
  392. {currentSelectedType && <BMSTable actionRef={tableRef} columns={tableColumn} rowKey='unitCode' params={tableDataFilterParams} request={(params) => getTableData(params)} />}
  393. </div>
  394. </div>
  395. </BMSPagecontainer>
  396. )
  397. }
  398. export default UnitIndicTargetSet;