index.tsx 9.2 KB

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