TaskBuilderChapter.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. import {
  2. Button,
  3. Divider,
  4. Input,
  5. message,
  6. Modal,
  7. Space,
  8. Steps,
  9. Typography,
  10. } from "antd";
  11. import { useState } from "react";
  12. import Workflow from "./Workflow";
  13. import {
  14. IProjectTreeData,
  15. IProjectTreeInsertRequest,
  16. IProjectTreeResponse,
  17. ITaskData,
  18. ITaskGroupInsertData,
  19. ITaskGroupInsertRequest,
  20. ITaskGroupResponse,
  21. } from "../api/task";
  22. import ChapterToc from "../article/ChapterToc";
  23. import { IChapterToc } from "../api/Corpus";
  24. import { post } from "../../request";
  25. import TaskBuilderProp, { IParam, IProp } from "./TaskBuilderProp";
  26. import {
  27. IPayload,
  28. ITokenCreate,
  29. ITokenCreateResponse,
  30. ITokenData,
  31. TPower,
  32. } from "../api/token";
  33. import ProjectWithTasks from "./ProjectWithTasks";
  34. import { useIntl } from "react-intl";
  35. const { Text, Paragraph } = Typography;
  36. interface IModal {
  37. studioName?: string;
  38. channels?: string[];
  39. book?: number;
  40. para?: number;
  41. open?: boolean;
  42. onClose?: () => void;
  43. }
  44. export const TaskBuilderChapterModal = ({
  45. studioName,
  46. channels,
  47. book,
  48. para,
  49. open = false,
  50. onClose,
  51. }: IModal) => {
  52. return (
  53. <>
  54. <Modal
  55. destroyOnClose={true}
  56. width={1400}
  57. style={{ top: 10 }}
  58. title={""}
  59. footer={false}
  60. open={open}
  61. onOk={onClose}
  62. onCancel={onClose}
  63. >
  64. <TaskBuilderChapter
  65. style={{ marginTop: 20 }}
  66. studioName={studioName}
  67. channels={channels}
  68. book={book}
  69. para={para}
  70. />
  71. </Modal>
  72. </>
  73. );
  74. };
  75. interface IWidget {
  76. studioName?: string;
  77. channels?: string[];
  78. book?: number;
  79. para?: number;
  80. style?: React.CSSProperties;
  81. }
  82. const TaskBuilderChapter = ({
  83. studioName,
  84. book,
  85. para,
  86. style,
  87. channels,
  88. }: IWidget) => {
  89. const intl = useIntl();
  90. const [current, setCurrent] = useState(0);
  91. const [workflow, setWorkflow] = useState<ITaskData[]>();
  92. const [chapter, setChapter] = useState<IChapterToc[]>();
  93. const [tokens, setTokens] = useState<ITokenData[]>();
  94. const [messages, setMessages] = useState<string[]>([]);
  95. const [prop, setProp] = useState<IProp[]>();
  96. const [title, setTitle] = useState<string>();
  97. const [loading, setLoading] = useState(false);
  98. const [projects, setProjects] = useState<IProjectTreeData[]>();
  99. const [done, setDone] = useState(false);
  100. const steps = [
  101. {
  102. title: "选择章节",
  103. content: (
  104. <div style={{ padding: 8 }}>
  105. <Space key={1}>
  106. <Text type="secondary">{"任务组标题"}</Text>
  107. <Input
  108. value={title}
  109. onChange={(e) => {
  110. setTitle(e.target.value);
  111. }}
  112. />
  113. </Space>
  114. <ChapterToc
  115. key={2}
  116. book={book}
  117. para={para}
  118. maxLevel={7}
  119. onData={(data: IChapterToc[]) => {
  120. setChapter(data);
  121. if (data.length > 0) {
  122. if (!title && data[0].text) {
  123. setTitle(data[0].text);
  124. }
  125. }
  126. }}
  127. />
  128. </div>
  129. ),
  130. },
  131. {
  132. title: "选择工作流",
  133. content: (
  134. <Workflow
  135. studioName={studioName}
  136. onSelect={(data) => {
  137. if (typeof data === "undefined") {
  138. setWorkflow(undefined);
  139. }
  140. }}
  141. onData={(data) => {
  142. console.debug("workflow", data);
  143. setWorkflow(data);
  144. }}
  145. />
  146. ),
  147. },
  148. {
  149. title: "参数设置",
  150. content: (
  151. <div>
  152. <TaskBuilderProp
  153. book={book}
  154. para={para}
  155. workflow={workflow}
  156. channelsId={channels}
  157. onChange={(data: IProp[] | undefined) => {
  158. console.info("prop value", data);
  159. setProp(data);
  160. let channels = new Map<string, number>();
  161. data?.forEach((value) => {
  162. value.param?.forEach((param) => {
  163. if (param.type.includes("channel")) {
  164. channels.set(param.value, 1);
  165. }
  166. });
  167. });
  168. //获取channel token
  169. let payload: IPayload[] = [];
  170. if (chapter) {
  171. channels.forEach((value, key) => {
  172. const [channelId, power] = key.split("@");
  173. payload = payload.concat(
  174. chapter.map((item) => {
  175. return {
  176. res_id: channelId,
  177. res_type: "channel",
  178. book: item.book,
  179. para_start: item.paragraph,
  180. para_end: item.paragraph + item.chapter_len,
  181. power: power as TPower,
  182. };
  183. })
  184. );
  185. });
  186. const url = "/v2/access-token";
  187. const values = { payload: payload };
  188. console.info("api request", url, values);
  189. post<ITokenCreate, ITokenCreateResponse>(url, values).then(
  190. (json) => {
  191. console.info("api response", json);
  192. setTokens(json.data.rows);
  193. }
  194. );
  195. }
  196. }}
  197. />
  198. </div>
  199. ),
  200. },
  201. {
  202. title: "生成任务",
  203. content: (
  204. <div style={{ padding: 8 }}>
  205. <div>
  206. <Space>
  207. <Text type="secondary">title</Text>
  208. <Text>{title}</Text>
  209. </Space>
  210. </div>
  211. <div>
  212. <Space>
  213. <Text type="secondary">新增任务组</Text>
  214. <Text>{chapter?.length}</Text>
  215. </Space>
  216. </div>
  217. <div>
  218. <Space>
  219. <Text type="secondary">每个任务组任务数量</Text>
  220. <Text>{workflow?.length}</Text>
  221. </Space>
  222. </div>
  223. <div>
  224. <Paragraph>点击生成按钮生成</Paragraph>
  225. </div>
  226. <div>
  227. {messages?.map((item, id) => {
  228. return <div key={id}>{item}</div>;
  229. })}
  230. </div>
  231. </div>
  232. ),
  233. },
  234. {
  235. title: "完成",
  236. content: projects ? (
  237. <ProjectWithTasks projectId={projects[0].id} />
  238. ) : (
  239. <></>
  240. ),
  241. },
  242. ];
  243. const next = () => {
  244. setCurrent(current + 1);
  245. };
  246. const prev = () => {
  247. setCurrent(current - 1);
  248. };
  249. const items = steps.map((item) => ({ key: item.title, title: item.title }));
  250. const DoButton = () => (
  251. <Button
  252. loading={loading}
  253. disabled={loading}
  254. type="primary"
  255. onClick={async () => {
  256. if (!studioName || !chapter) {
  257. console.error("缺少参数", studioName, chapter);
  258. return;
  259. }
  260. setLoading(true);
  261. //生成projects
  262. setMessages((origin) => [...origin, "正在生成任务组……"]);
  263. const url = "/v2/project-tree";
  264. const values: IProjectTreeInsertRequest = {
  265. studio_name: studioName,
  266. data: chapter.map((item, id) => {
  267. return {
  268. id: item.paragraph.toString(),
  269. title: id === 0 && title ? title : item.text ?? "",
  270. type: "instance",
  271. weight: item.chapter_strlen,
  272. parent_id: item.parent.toString(),
  273. res_id: `${item.book}-${item.paragraph}`,
  274. };
  275. }),
  276. };
  277. console.info("api request", url, values);
  278. const res = await post<IProjectTreeInsertRequest, IProjectTreeResponse>(
  279. url,
  280. values
  281. );
  282. console.info("api response", res);
  283. if (!res.ok) {
  284. setMessages((origin) => [...origin, "正在生成任务组失败"]);
  285. return;
  286. } else {
  287. setProjects(res.data.rows);
  288. setMessages((origin) => [...origin, "生成任务组成功"]);
  289. }
  290. //生成tasks
  291. setMessages((origin) => [...origin, "正在生成任务……"]);
  292. const taskUrl = "/v2/task-group";
  293. if (!workflow) {
  294. return;
  295. }
  296. let taskData: ITaskGroupInsertData[] = res.data.rows
  297. .filter((value) => value.isLeaf)
  298. .map((project, pId) => {
  299. return {
  300. project_id: project.id,
  301. tasks: workflow.map((task, tId) => {
  302. let newContent = task.description;
  303. prop
  304. ?.find((pValue) => pValue.taskId === task.id)
  305. ?.param?.forEach((value: IParam) => {
  306. //替换数字参数
  307. if (value.type === "number") {
  308. const searchValue = `${value.key}=${value.value}`;
  309. const replaceValue =
  310. `${value.key}=` +
  311. (value.initValue + value.step * pId).toString();
  312. newContent = newContent?.replace(
  313. searchValue,
  314. replaceValue
  315. );
  316. } else {
  317. //替换book
  318. if (project.resId) {
  319. const [book, paragraph] = project.resId.split("-");
  320. newContent = newContent?.replace(
  321. "book=#",
  322. `book=${book}`
  323. );
  324. newContent = newContent?.replace(
  325. "paragraphs=#",
  326. `paragraphs=${paragraph}`
  327. );
  328. //替换channel
  329. //查找toke
  330. const [channel, power] = value.value.split("@");
  331. const mToken = tokens?.find(
  332. (token) =>
  333. token.payload.book?.toString() === book &&
  334. token.payload.para_start?.toString() ===
  335. paragraph &&
  336. token.payload.res_id === channel &&
  337. (power && power.length > 0
  338. ? token.payload.power === power
  339. : true)
  340. );
  341. newContent = newContent?.replace(
  342. value.key,
  343. channel + (mToken ? "@" + mToken?.token : "")
  344. );
  345. }
  346. }
  347. });
  348. console.debug("description", newContent);
  349. return {
  350. ...task,
  351. type: "instance",
  352. description: newContent,
  353. };
  354. }),
  355. };
  356. });
  357. console.info("api request", taskUrl, taskData);
  358. const taskRes = await post<ITaskGroupInsertRequest, ITaskGroupResponse>(
  359. taskUrl,
  360. { data: taskData }
  361. );
  362. if (taskRes.ok) {
  363. message.success("ok");
  364. setMessages((origin) => [...origin, "生成任务成功"]);
  365. setMessages((origin) => [
  366. ...origin,
  367. "生成任务" + taskRes.data.taskCount,
  368. ]);
  369. setMessages((origin) => [
  370. ...origin,
  371. "生成任务关联" + taskRes.data.taskRelationCount,
  372. ]);
  373. setMessages((origin) => [
  374. ...origin,
  375. "打开译经楼-我的任务查看已经生成的任务",
  376. ]);
  377. setDone(true);
  378. }
  379. setLoading(false);
  380. }}
  381. >
  382. Done
  383. </Button>
  384. );
  385. return (
  386. <div style={style}>
  387. <Steps current={current} items={items} />
  388. <div className="steps-content" style={{ minHeight: 400 }}>
  389. {steps[current].content}
  390. </div>
  391. <Divider></Divider>
  392. <div style={{ display: "flex", justifyContent: "space-between" }}>
  393. {current < steps.length - 1 ? (
  394. <Button
  395. style={{ margin: "0 8px" }}
  396. disabled={current === 0}
  397. onClick={() => prev()}
  398. >
  399. {intl.formatMessage({ id: "buttons.previous" })}
  400. </Button>
  401. ) : (
  402. <></>
  403. )}
  404. {current < steps.length - 2 && (
  405. <Button
  406. type="primary"
  407. disabled={current === 1 && typeof workflow === "undefined"}
  408. onClick={() => next()}
  409. >
  410. {intl.formatMessage({ id: "buttons.next" })}
  411. </Button>
  412. )}
  413. {current === steps.length - 2 && (
  414. <>
  415. {done ? (
  416. <Button type="primary" onClick={() => next()}>
  417. 完成
  418. </Button>
  419. ) : (
  420. <DoButton />
  421. )}
  422. </>
  423. )}
  424. </div>
  425. </div>
  426. );
  427. };
  428. export default TaskBuilderChapter;