TaskReader.tsx 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. import { useEffect, useState } from "react";
  2. import { Divider, Space, Tag, Typography, message } from "antd";
  3. import { CodeSandboxOutlined } from "@ant-design/icons";
  4. import { ITaskData, ITaskResponse, ITaskUpdateRequest } from "../api/task";
  5. import { get, patch } from "../../request";
  6. import User from "../auth/User";
  7. import TimeShow from "../general/TimeShow";
  8. import TaskEditButton, { TRelation } from "./TaskEditButton";
  9. import PreTask from "./PreTask";
  10. import Like from "../like/Like";
  11. import Assignees from "./Assignees";
  12. import PlanDate from "./PlanDate";
  13. import TaskTitle from "./TaskTitle";
  14. import TaskStatus from "./TaskStatus";
  15. import Description from "./Description";
  16. import Category from "./Category";
  17. const { Text } = Typography;
  18. export const Milestone = ({ task }: { task?: ITaskData }) => {
  19. return task?.is_milestone ? (
  20. <Tag icon={<CodeSandboxOutlined />} color="error">
  21. 里程碑
  22. </Tag>
  23. ) : null;
  24. };
  25. interface IWidget {
  26. taskId?: string;
  27. onChange?: (data: ITaskData[]) => void;
  28. onEdit?: () => void;
  29. }
  30. const TaskReader = ({ taskId, onChange, onEdit }: IWidget) => {
  31. const [openPreTask, setOpenPreTask] = useState(false);
  32. const [openNextTask, setOpenNextTask] = useState(false);
  33. const [task, setTask] = useState<ITaskData>();
  34. useEffect(() => {
  35. const url = `/v2/task/${taskId}`;
  36. console.info("task api request", url);
  37. get<ITaskResponse>(url).then((json) => {
  38. console.info("task api response", json);
  39. if (json.ok) {
  40. setTask(json.data);
  41. }
  42. });
  43. }, [taskId]);
  44. const updatePreTask = (type: TRelation, data?: ITaskData | null) => {
  45. if (!taskId || !data) {
  46. return;
  47. }
  48. let setting: ITaskUpdateRequest = {
  49. id: taskId,
  50. studio_name: "",
  51. };
  52. if (type === "pre") {
  53. const hasPre = task?.pre_task?.find((value) => value.id === data.id);
  54. if (hasPre) {
  55. setting.pre_task_id = task?.pre_task
  56. ?.filter((value) => value.id !== data.id)
  57. .map((item) => item.id)
  58. .join();
  59. } else {
  60. const newRelation = task?.pre_task
  61. ? [...task.pre_task.map((item) => item.id), data.id]
  62. : [data.id];
  63. setting.pre_task_id = newRelation.join();
  64. }
  65. } else if (type === "next") {
  66. const hasPre = task?.next_task?.find((value) => value.id === data.id);
  67. if (hasPre) {
  68. setting.next_task_id = task?.next_task
  69. ?.filter((value) => value.id !== data.id)
  70. .map((item) => item.id)
  71. .join();
  72. } else {
  73. const newRelation = task?.next_task
  74. ? [...task.next_task.map((item) => item.id), data.id]
  75. : [data.id];
  76. setting.next_task_id = newRelation.join();
  77. }
  78. }
  79. const url = `/v2/task/${setting.id}`;
  80. console.info("api request", url, setting);
  81. patch<ITaskUpdateRequest, ITaskResponse>(url, setting).then((json) => {
  82. console.info("api response", json);
  83. if (json.ok) {
  84. message.success("Success");
  85. setTask(json.data);
  86. onChange && onChange([json.data]);
  87. } else {
  88. message.error(json.message);
  89. }
  90. });
  91. };
  92. return (
  93. <div>
  94. <div style={{ display: "flex", justifyContent: "space-between" }}>
  95. <Space>
  96. <TaskStatus task={task} />
  97. <Milestone task={task} />
  98. <PreTask
  99. task={task}
  100. open={openPreTask}
  101. type="pre"
  102. onClick={(data) => {
  103. updatePreTask("pre", data);
  104. setOpenPreTask(false);
  105. }}
  106. onTagClick={() => setOpenPreTask(true)}
  107. onClose={() => setOpenPreTask(false)}
  108. />
  109. <PreTask
  110. task={task}
  111. open={openNextTask}
  112. type="next"
  113. onClick={(data) => {
  114. updatePreTask("next", data);
  115. setOpenNextTask(false);
  116. }}
  117. onClose={() => setOpenNextTask(false)}
  118. onTagClick={() => setOpenNextTask(true)}
  119. />
  120. </Space>
  121. <div>
  122. <TaskEditButton
  123. task={task}
  124. onChange={(tasks: ITaskData[]) => {
  125. setTask(tasks.find((value) => value.id === taskId));
  126. onChange && onChange(tasks);
  127. }}
  128. onEdit={onEdit}
  129. onPreTask={(type: TRelation) => {
  130. if (type === "pre") {
  131. setOpenPreTask(true);
  132. } else if (type === "next") {
  133. setOpenNextTask(true);
  134. }
  135. }}
  136. />
  137. </div>
  138. </div>
  139. <TaskTitle
  140. task={task}
  141. onChange={(data) => {
  142. setTask(data[0]);
  143. onChange && onChange(data);
  144. }}
  145. />
  146. <div style={{ display: "flex", flexDirection: "column" }}>
  147. <Space>
  148. <User {...task?.editor} />
  149. <TimeShow updatedAt={task?.updated_at} />
  150. <Like resId={task?.id} resType="task" />
  151. </Space>
  152. <Space>
  153. <Text type="secondary" key={"2"}>
  154. 执行人
  155. </Text>
  156. <User key={"executor"} {...task?.executor} />
  157. </Space>
  158. <Space>
  159. <Text type="secondary" key={"1"}>
  160. 指派给
  161. </Text>
  162. <Assignees
  163. key={"assignees"}
  164. task={task}
  165. onChange={(data) => {
  166. setTask(data[0]);
  167. onChange && onChange(data);
  168. }}
  169. />
  170. </Space>
  171. <Space>
  172. <Text type="secondary">起止日期</Text>
  173. <div style={{ width: 400 }}>
  174. <PlanDate />
  175. </div>
  176. </Space>
  177. <Space>
  178. <Text type="secondary">类别</Text>
  179. <Category
  180. task={task}
  181. onChange={(data) => {
  182. setTask(data[0]);
  183. onChange && onChange(data);
  184. }}
  185. />
  186. </Space>
  187. </div>
  188. <Divider />
  189. <Description
  190. task={task}
  191. onChange={(data) => {
  192. setTask(data[0]);
  193. onChange && onChange(data);
  194. }}
  195. />
  196. </div>
  197. );
  198. };
  199. export default TaskReader;