import React, { useCallback, useRef, useState } from 'react'; import { UploadOutlined } from '@ant-design/icons'; import { Button, Tooltip, Upload } from 'antd'; import type { UploadFile, UploadProps } from 'antd/es/upload/interface'; import update from 'immutability-helper'; import { DndProvider, useDrag, useDrop } from 'react-dnd'; import { HTML5Backend } from 'react-dnd-html5-backend'; const type = 'DragableUploadList'; interface DragableUploadListItemProps { originNode: React.ReactElement>; file: UploadFile; fileList: UploadFile[]; moveRow: (dragIndex: any, hoverIndex: any) => void; } const DragableUploadListItem = ({ originNode, moveRow, file, fileList, }: DragableUploadListItemProps) => { const ref = useRef(null); const index = fileList.indexOf(file); const [{ isOver, dropClassName }, drop] = useDrop({ accept: type, collect: monitor => { const { index: dragIndex } = monitor.getItem() || {}; if (dragIndex === index) { return {}; } return { isOver: monitor.isOver(), dropClassName: dragIndex < index ? ' drop-over-downward' : ' drop-over-upward', }; }, drop: (item: any) => { moveRow(item.index, index); }, }); const [, drag] = useDrag({ type, item: { index }, collect: monitor => ({ isDragging: monitor.isDragging(), }), }); drop(drag(ref)); const errorNode = {originNode.props.children}; return (
{file.status === 'error' ? errorNode : originNode}
); }; const App: React.FC = () => { const [fileList, setFileList] = useState([ { uid: '-1', name: 'image1.png', status: 'done', url: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png', }, { uid: '-2', name: 'image2.png', status: 'done', url: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png', }, { uid: '-3', name: 'image3.png', status: 'done', url: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png', }, { uid: '-4', name: 'image4.png', status: 'done', url: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png', }, { uid: '-5', name: 'image.png', status: 'error', }, ]); const moveRow = useCallback( (dragIndex: number, hoverIndex: number) => { const dragRow = fileList[dragIndex]; setFileList( update(fileList, { $splice: [ [dragIndex, 1], [hoverIndex, 0, dragRow], ], }), ); }, [fileList], ); const onChange: UploadProps['onChange'] = ({ fileList: newFileList }) => { setFileList(newFileList); }; return ( ( )} > ); }; export default App;