index.tsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. import {Button, Drawer, Input, Modal, Popover, Space, Switch, Table} from "antd";
  2. import {ColumnsType} from "antd/es/table";
  3. import {
  4. AdvanceInputConfigs,
  5. AutoForm,
  6. AutoFormInstance,
  7. AutoTypeInputProps,
  8. isNull,
  9. Message,
  10. uniqueKey
  11. } from "planning-tools";
  12. import {useEffect, useRef, useState} from "react";
  13. import { useTranslation } from 'react-i18next'
  14. import {
  15. ArrowDownOutlined,
  16. ArrowUpOutlined,
  17. CopyOutlined,
  18. DeleteOutlined,
  19. EditOutlined,
  20. PlusOutlined
  21. } from "@ant-design/icons";
  22. import {INginxLocation} from "../../../../models/nginx.ts";
  23. import {cloneDeep} from "lodash";
  24. import FormConfig from './config.json'
  25. import {SiteInput} from "../site";
  26. import {renderLocation} from "./utils.ts";
  27. import './index.less'
  28. /**
  29. * 部分的重要信息
  30. * @param data
  31. * @param onChange
  32. * @constructor
  33. */
  34. const LocationInfo = ({data, onChange}:{ data: INginxLocation, onChange?: (data: INginxLocation) => void})=>{
  35. const rootDir = ()=>{
  36. if (data.alias){
  37. return `alias: ${data.alias}`
  38. }
  39. return `root: ${data.root || '--'}`
  40. }
  41. return (<div>
  42. {
  43. data.proxy_type === 'proxy' ? <div>{`proxy: ${data.proxy_pass}`}</div> : null
  44. }
  45. {
  46. data.proxy_type === 'static' ? <div>{rootDir()}<SiteInput onChange={onChange} location={data} /></div>:null
  47. }
  48. <div>
  49. {
  50. data.rewrite?.regex && data.rewrite?.replacement ? `${data.rewrite.regex} ${data.rewrite.replacement}` : ''
  51. }
  52. </div>
  53. </div>)
  54. }
  55. /**
  56. * 路由,站点,规则编辑
  57. * @param value
  58. * @param onChange
  59. * @param column
  60. * @constructor
  61. */
  62. export const LocationInput = ({value, onChange }: AutoTypeInputProps) => {
  63. const [locations, setLocations] = useState<INginxLocation[]>([])
  64. const [editData, setEditData] = useState<INginxLocation>()
  65. const isAddRef = useRef(false)
  66. const [modal,contextHolder] = Modal.useModal()
  67. const {t} = useTranslation()
  68. const formRef = useRef<AutoFormInstance>()
  69. useEffect(() => {
  70. if (Array.isArray(value)) {
  71. setLocations(value.map((item: INginxLocation) => {
  72. if (!item.id) {
  73. item.id = uniqueKey(20)
  74. }
  75. if (!item.lines){
  76. renderLocation(item)
  77. }
  78. return item
  79. }))
  80. }
  81. }, [value])
  82. const onEditRow = (data: INginxLocation) => {
  83. isAddRef.current = false
  84. setEditData(cloneDeep(data))
  85. }
  86. const onAddData = (data?: INginxLocation, index?: number)=>{
  87. isAddRef.current = true
  88. setEditData({ ...data,id: uniqueKey(20),__index__: index} as never)
  89. }
  90. const moveUp = (data: INginxLocation, index: number) => {
  91. if (index == 0){
  92. Message.warning(t('location.is_first'))
  93. return
  94. }
  95. const current = locations[index-1]
  96. locations[index-1] = data
  97. locations[index] = current
  98. setLocations([...locations])
  99. onChange?.(locations)
  100. }
  101. const moveDown = (data: INginxLocation,index: number)=>{
  102. if (index == locations.length-1){
  103. Message.warning(t('location.is_last'))
  104. return
  105. }
  106. const current = locations[index+1]
  107. locations[index+1] = data
  108. locations[index] = current
  109. setLocations([...locations])
  110. onChange?.(locations)
  111. }
  112. const onRemoveData = (data: INginxLocation)=>{
  113. const onOk = ()=>{
  114. const list = locations.filter(item=>item.id !== data.id);
  115. onChange?.(list)
  116. }
  117. modal.confirm({
  118. title: '提示',
  119. type: 'warning',
  120. content: '您确定要删除该代理/站点吗?删除操作不可恢复,请谨慎操作!',
  121. okType: 'danger',
  122. okText: '仍要删除',
  123. cancelText: '先不了',
  124. onOk,
  125. })
  126. }
  127. const onQuickChangeStatus = (data: INginxLocation, enable: boolean) => {
  128. const list = locations.map(item=>{
  129. if (item.id === data.id){
  130. return { ...item, enable }
  131. }
  132. return item
  133. })
  134. onChange?.(list)
  135. }
  136. const onSubmitData = async () => {
  137. const values = await formRef.current?.onSyncSubmit(true);
  138. const newData = {...editData, ...values} as INginxLocation;
  139. console.log('newLocation', newData);
  140. if (!editData) {
  141. console.warn('editData is null ,skip ');
  142. return
  143. }
  144. renderLocation(newData)
  145. let list: INginxLocation[]
  146. if (isAddRef.current){
  147. const index = newData.__index__ || locations.length;
  148. delete newData.__index__;
  149. let exist = locations.find(item=>item.name == newData.name);
  150. if (exist){
  151. Message.warning('名称不能相同,请修改后再保存!');
  152. return
  153. }
  154. exist = locations.find(item=> {
  155. if (item.match && newData.match){
  156. return item.match.regex === newData.match.regex && item.match.path === newData.match.path
  157. }
  158. return false
  159. });
  160. if (exist){
  161. Message.warning('匹配规则不能完全一样,请修改后重新添加!');
  162. return
  163. }
  164. renderLocation(newData);
  165. if (isNull(index) || index < 0 || index >= locations.length-1){
  166. list = locations.concat([newData])
  167. }else {
  168. list = []
  169. locations.forEach((item,idx)=>{
  170. if (idx === index){
  171. list.push(item)
  172. list.push(newData)
  173. }else {
  174. list.push(item)
  175. }
  176. })
  177. }
  178. }else {
  179. renderLocation(newData);
  180. list = locations.map(item => {
  181. if (item.id === newData.id) {
  182. return {...item, ...newData}
  183. }
  184. return item
  185. })
  186. }
  187. onChange?.(list)
  188. setEditData(undefined)
  189. }
  190. /**
  191. * 部署数据变化,不重新渲染
  192. * @param data
  193. */
  194. const onDeployDataChange = (data: INginxLocation) => {
  195. const newList = locations.map(item=>{
  196. if (item.id === data.id){
  197. return { ...item, ...data}
  198. }
  199. return item;
  200. });
  201. onChange?.(newList)
  202. }
  203. const renderPreview = (data: INginxLocation)=>{
  204. let content ='';
  205. let rows = 0;
  206. if (data.http?.length){
  207. content = data.http.join('\n') + '\n';
  208. rows = data.http.length;
  209. }
  210. if (data.lines){
  211. content = content+ data.lines.join('\n')
  212. rows +=data.lines.length
  213. }
  214. return (<div className="location-conf-preview">
  215. <Input.TextArea rows={Math.max(Math.min(10,rows),5)} disabled value={content} />
  216. </div>)
  217. }
  218. const renderOps = (_: never, data: INginxLocation, index: number) => {
  219. return (
  220. <div className="location-btns">
  221. <ArrowUpOutlined className='move-btn' onClick={()=>moveUp(data, index)}/>
  222. <ArrowDownOutlined className='move-btn' onClick={()=>moveDown(data,index)}/>
  223. <Button onClick={() => onRemoveData(data)} type="text" danger icon={<DeleteOutlined/>}/>
  224. <Button onClick={() => onEditRow(data)} type="link" icon={<EditOutlined/>}/>
  225. <Button onClick={()=>onAddData(data, index)} type="link" icon={<CopyOutlined/>}/>
  226. <Popover trigger="click" destroyTooltipOnHide
  227. placement="top"
  228. content={()=>renderPreview(data as never)} >
  229. <Button type="link">预览</Button>
  230. </Popover>
  231. </div>
  232. )
  233. }
  234. const columns: ColumnsType = [
  235. {
  236. dataIndex: 'name',
  237. title: '路由名称',
  238. width: 120
  239. },
  240. {
  241. dataIndex: 'match',
  242. title: "规则",
  243. render: (value) => <span>{`${value.regex || ''} ${value.path}`}</span>
  244. },
  245. {
  246. dataIndex: 'enable',
  247. title: '状态',
  248. render: (value,record) => <Switch onChange={c=>onQuickChangeStatus(record as never,c)} checked={value}/>
  249. },
  250. {
  251. dataIndex: 'proxy_pass',
  252. title: '代理或路径',
  253. render: (_,record: any)=>{
  254. return (<LocationInfo onChange={onDeployDataChange} data={record} />)
  255. }
  256. },
  257. {
  258. dataIndex: 'remark',
  259. title:"备注",
  260. },
  261. {
  262. title: '操作',
  263. render: renderOps as never,
  264. width: 200,
  265. fixed: 'right'
  266. }
  267. ]
  268. return (
  269. <>
  270. {
  271. locations.length ? (<Table pagination={false}
  272. style={{marginRight: 5}}
  273. rowKey="id"
  274. columns={columns as never}
  275. className="location-table"
  276. dataSource={locations}>
  277. <div>Empty</div>
  278. </Table>) : (
  279. <>
  280. <Button onClick={()=>onAddData()} className="add-btn" type="link" icon={<PlusOutlined/>}/>
  281. </>
  282. )
  283. }
  284. <Drawer title={isAddRef.current? '新增' : '编辑'}
  285. placement="right"
  286. open={!!editData}
  287. onClose={() => setEditData(undefined)}
  288. destroyOnClose
  289. width={900}
  290. className="location-input"
  291. extra={<Space>
  292. <Button onClick={onSubmitData} ghost type="primary">保存</Button>
  293. </Space>}
  294. >
  295. <AutoForm
  296. columns={FormConfig.form}
  297. ref={formRef as never}
  298. data={editData}/>
  299. </Drawer>
  300. {contextHolder}
  301. </>
  302. )
  303. }
  304. AdvanceInputConfigs['locations'] = LocationInput