task_view.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. const { div, h2, p, section, button, form, input, select, option, a, br, textarea, label, span } = require("../server/node_modules/hyperaxe");
  2. const moment = require("../server/node_modules/moment");
  3. const { template, i18n } = require("./main_views");
  4. const { config } = require("../server/SSB_server.js");
  5. const { renderUrl } = require("../backend/renderUrl");
  6. const userId = config.keys.id;
  7. const opt = (value, isSelected, text) =>
  8. option(Object.assign({ value }, isSelected ? { selected: "selected" } : {}), text);
  9. const safeArray = (v) => Array.isArray(v) ? v : [];
  10. const toValueChildren = (v) => {
  11. if (v === undefined || v === null) return [];
  12. if (Array.isArray(v)) return v;
  13. if (typeof v === "string") return renderUrl(v);
  14. if (typeof v === "number" || typeof v === "boolean") return renderUrl(String(v));
  15. return [v];
  16. };
  17. const renderCardField = (labelText, valueNode) =>
  18. div(
  19. { class: "card-field" },
  20. span({ class: "card-label" }, labelText),
  21. span({ class: "card-value" }, ...toValueChildren(valueNode))
  22. );
  23. const normalizeStatus = (v) => {
  24. const up = String(v || "").toUpperCase();
  25. if (up === "OPEN" || up === "IN-PROGRESS" || up === "CLOSED") return up;
  26. return "OPEN";
  27. };
  28. const statusLabel = (s) => {
  29. const up = normalizeStatus(s);
  30. if (up === "OPEN") return i18n.taskStatusOpen;
  31. if (up === "IN-PROGRESS") return i18n.taskStatusInProgress;
  32. return i18n.taskStatusClosed;
  33. };
  34. const visibilityLabel = (v) => {
  35. const vv = String(v || "").toUpperCase();
  36. if (vv === "PRIVATE") return i18n.taskPrivate;
  37. return i18n.taskPublic;
  38. };
  39. const renderTaskOwnerActions = (task, returnTo) => {
  40. const st = normalizeStatus(task.status || "OPEN");
  41. const setStatusLabel = i18n.taskSetStatus;
  42. return [
  43. form(
  44. { method: "GET", action: `/tasks/edit/${encodeURIComponent(task.id)}` },
  45. input({ type: "hidden", name: "returnTo", value: returnTo }),
  46. button({ type: "submit", class: "update-btn" }, i18n.taskUpdateButton)
  47. ),
  48. form(
  49. { method: "POST", action: `/tasks/delete/${encodeURIComponent(task.id)}` },
  50. input({ type: "hidden", name: "returnTo", value: returnTo }),
  51. button({ type: "submit", class: "delete-btn" }, i18n.taskDeleteButton)
  52. ),
  53. form(
  54. { method: "POST", action: `/tasks/status/${encodeURIComponent(task.id)}`, class: "project-control-form project-control-form--status" },
  55. input({ type: "hidden", name: "returnTo", value: returnTo }),
  56. select(
  57. { name: "status", class: "project-control-select" },
  58. option({ value: "OPEN", selected: st === "OPEN" }, i18n.taskStatusOpen),
  59. option({ value: "IN-PROGRESS", selected: st === "IN-PROGRESS" }, i18n.taskStatusInProgress),
  60. option({ value: "CLOSED", selected: st === "CLOSED" }, i18n.taskStatusClosed)
  61. ),
  62. button({ class: "status-btn project-control-btn", type: "submit" }, setStatusLabel)
  63. )
  64. ];
  65. };
  66. const renderTaskAssignAction = (task, isAssignedToMe, returnTo) => {
  67. const st = normalizeStatus(task.status || "OPEN");
  68. if (st === "CLOSED") return null;
  69. return form(
  70. { method: "POST", action: `/tasks/assign/${encodeURIComponent(task.id)}` },
  71. input({ type: "hidden", name: "returnTo", value: returnTo }),
  72. button({ type: "submit", class: "filter-btn" }, isAssignedToMe ? i18n.taskUnassignButton : i18n.taskAssignButton)
  73. );
  74. };
  75. const renderTaskTopbar = (task, filter, opts = {}) => {
  76. const currentFilter = filter || "all";
  77. const isSingle = !!opts.single;
  78. const returnToList = `/tasks?filter=${encodeURIComponent(currentFilter)}`;
  79. const returnToSelf = `/tasks/${encodeURIComponent(task.id)}?filter=${encodeURIComponent(currentFilter)}`;
  80. const rt = isSingle ? returnToSelf : returnToList;
  81. const assignees = safeArray(task.assignees);
  82. const isAssignedToMe = assignees.includes(userId);
  83. const leftActions = [];
  84. if (!isSingle) {
  85. leftActions.push(
  86. form(
  87. { method: "GET", action: `/tasks/${encodeURIComponent(task.id)}` },
  88. input({ type: "hidden", name: "filter", value: currentFilter }),
  89. button({ type: "submit", class: "filter-btn" }, i18n.viewDetails)
  90. )
  91. );
  92. }
  93. if (task.author && task.author !== userId) {
  94. leftActions.push(
  95. form(
  96. { method: "GET", action: "/pm" },
  97. input({ type: "hidden", name: "recipients", value: task.author }),
  98. button({ type: "submit", class: "filter-btn" }, i18n.privateMessage)
  99. )
  100. );
  101. }
  102. const ownerActions = task.author === userId ? renderTaskOwnerActions(task, rt) : [];
  103. const assignNode = renderTaskAssignAction(task, isAssignedToMe, rt);
  104. const rightActions = [];
  105. if (assignNode) rightActions.push(assignNode);
  106. if (ownerActions.length) rightActions.push(...ownerActions);
  107. const leftNode = leftActions.length ? div({ class: "bookmark-topbar-left task-topbar-left" }, ...leftActions) : null;
  108. const rightNode = rightActions.length ? div({ class: "bookmark-actions task-actions" }, ...rightActions) : null;
  109. const nodes = [];
  110. if (leftNode) nodes.push(leftNode);
  111. if (rightNode) nodes.push(rightNode);
  112. return nodes.length ? div({ class: isSingle ? "bookmark-topbar task-topbar-single" : "bookmark-topbar" }, ...nodes) : null;
  113. };
  114. const renderTaskCommentsSection = (taskId, comments = [], currentFilter = "all") => {
  115. const commentsCount = Array.isArray(comments) ? comments.length : 0;
  116. const returnTo = `/tasks/${encodeURIComponent(taskId)}?filter=${encodeURIComponent(currentFilter || "all")}`;
  117. return div(
  118. { class: "vote-comments-section" },
  119. div(
  120. { class: "comments-count" },
  121. span({ class: "card-label" }, i18n.voteCommentsLabel + ": "),
  122. span({ class: "card-value" }, String(commentsCount))
  123. ),
  124. div(
  125. { class: "comment-form-wrapper" },
  126. h2({ class: "comment-form-title" }, i18n.voteNewCommentLabel),
  127. form(
  128. { method: "POST", action: `/tasks/${encodeURIComponent(taskId)}/comments`, class: "comment-form", enctype: "multipart/form-data" },
  129. input({ type: "hidden", name: "returnTo", value: returnTo }),
  130. textarea({
  131. id: "comment-text",
  132. name: "text",
  133. rows: 4,
  134. class: "comment-textarea",
  135. placeholder: i18n.voteNewCommentPlaceholder
  136. }),
  137. div({ class: "comment-file-upload" }, label(i18n.uploadMedia), input({ type: "file", name: "blob" })),
  138. br(),
  139. button({ type: "submit", class: "comment-submit-btn" }, i18n.voteNewCommentButton)
  140. )
  141. ),
  142. comments && comments.length
  143. ? div(
  144. { class: "comments-list" },
  145. comments.map((c) => {
  146. const author = c.value && c.value.author ? c.value.author : "";
  147. const ts = c.value && c.value.timestamp ? c.value.timestamp : c.timestamp;
  148. const absDate = ts ? moment(ts).format("YYYY/MM/DD HH:mm:ss") : "";
  149. const relDate = ts ? moment(ts).fromNow() : "";
  150. const userName = author && author.includes("@") ? author.split("@")[1] : author;
  151. const content = c.value && c.value.content ? c.value.content : {};
  152. const root = content.fork || content.root || "";
  153. const text = content.text || "";
  154. return div(
  155. { class: "votations-comment-card" },
  156. span(
  157. { class: "created-at" },
  158. span(i18n.createdBy),
  159. author ? a({ href: `/author/${encodeURIComponent(author)}` }, `@${userName}`) : span("(unknown)"),
  160. absDate ? span(" | ") : "",
  161. absDate ? span({ class: "votations-comment-date" }, absDate) : "",
  162. relDate ? span({ class: "votations-comment-date" }, " | ", i18n.sendTime) : "",
  163. relDate && root ? a({ href: `/thread/${encodeURIComponent(root)}#${encodeURIComponent(c.key)}` }, relDate) : ""
  164. ),
  165. p({ class: "votations-comment-text" }, ...renderUrl(String(text)))
  166. );
  167. })
  168. )
  169. : p({ class: "votations-no-comments" }, i18n.voteNoCommentsYet)
  170. );
  171. };
  172. const renderTaskItem = (task, filter) => {
  173. const currentFilter = filter || "all";
  174. const assignees = safeArray(task.assignees);
  175. const commentCount = typeof task.commentCount === "number" ? task.commentCount : 0;
  176. const topbar = renderTaskTopbar(task, currentFilter, { single: false });
  177. return div(
  178. { class: "card card-section task" },
  179. topbar ? topbar : null,
  180. renderCardField(i18n.taskTitleLabel + ":", task.title),
  181. renderCardField(i18n.taskDescriptionLabel + ":", ""),
  182. p(...renderUrl(task.description)),
  183. task.location && String(task.location).trim() ? renderCardField(i18n.taskLocationLabel + ":", task.location) : null,
  184. renderCardField(i18n.taskStatus + ":", statusLabel(task.status)),
  185. renderCardField(i18n.taskPriorityLabel + ":", task.priority),
  186. renderCardField(i18n.taskVisibilityLabel + ":", visibilityLabel(task.isPublic)),
  187. renderCardField(i18n.taskStartTimeLabel + ":", task.startTime ? moment(task.startTime).format("YYYY/MM/DD HH:mm:ss") : ""),
  188. renderCardField(i18n.taskEndTimeLabel + ":", task.endTime ? moment(task.endTime).format("YYYY/MM/DD HH:mm:ss") : ""),
  189. br(),
  190. div(
  191. { class: "card-field" },
  192. span({ class: "card-label" }, i18n.taskAssignedTo + ":"),
  193. span(
  194. { class: "card-value" },
  195. assignees.length
  196. ? assignees.map((id, i) => [i > 0 ? ", " : "", a({ class: "user-link", href: `/author/${encodeURIComponent(id)}` }, id)]).flat()
  197. : i18n.noAssignees
  198. )
  199. ),
  200. br(),
  201. Array.isArray(task.tags) && task.tags.length
  202. ? div(
  203. { class: "card-tags" },
  204. task.tags.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
  205. )
  206. : null,
  207. div(
  208. { class: "card-comments-summary" },
  209. span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
  210. span({ class: "card-value" }, String(commentCount)),
  211. br(),
  212. br(),
  213. form(
  214. { method: "GET", action: `/tasks/${encodeURIComponent(task.id)}` },
  215. input({ type: "hidden", name: "filter", value: currentFilter }),
  216. button({ type: "submit", class: "filter-btn" }, i18n.voteCommentsForumButton)
  217. )
  218. ),
  219. br(),
  220. p(
  221. { class: "card-footer" },
  222. span({ class: "date-link" }, `${moment(task.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
  223. a({ href: `/author/${encodeURIComponent(task.author)}`, class: "user-link" }, `${task.author}`)
  224. )
  225. );
  226. };
  227. exports.taskView = async (tasks, filter, taskId, returnTo) => {
  228. const list = Array.isArray(tasks) ? tasks : [tasks];
  229. const currentFilter = filter || "all";
  230. const title =
  231. currentFilter === "mine" ? i18n.taskMineSectionTitle :
  232. currentFilter === "create" ? i18n.taskCreateSectionTitle :
  233. currentFilter === "edit" ? i18n.taskUpdateSectionTitle :
  234. currentFilter === "open" ? i18n.taskOpenTitle :
  235. currentFilter === "in-progress" ? i18n.taskInProgressTitle :
  236. currentFilter === "closed" ? i18n.taskClosedTitle :
  237. currentFilter === "assigned" ? i18n.taskAssignedTitle :
  238. currentFilter === "priority-urgent" ? i18n.taskFilterUrgent :
  239. currentFilter === "priority-high" ? i18n.taskFilterHigh :
  240. currentFilter === "priority-medium" ? i18n.taskFilterMedium :
  241. currentFilter === "priority-low" ? i18n.taskFilterLow :
  242. i18n.taskAllSectionTitle;
  243. const canSee = (t) => {
  244. const vis = String(t.isPublic || "").toUpperCase();
  245. if (vis === "PUBLIC") return true;
  246. if (t.author === userId) return true;
  247. return safeArray(t.assignees).includes(userId);
  248. };
  249. const visible = list.filter(canSee);
  250. let filtered;
  251. if (currentFilter === "mine") filtered = visible.filter((t) => t.author === userId);
  252. else if (currentFilter === "assigned") filtered = visible.filter((t) => safeArray(t.assignees).includes(userId));
  253. else if (currentFilter === "open") filtered = visible.filter((t) => normalizeStatus(t.status) === "OPEN");
  254. else if (currentFilter === "in-progress") filtered = visible.filter((t) => normalizeStatus(t.status) === "IN-PROGRESS");
  255. else if (currentFilter === "closed") filtered = visible.filter((t) => normalizeStatus(t.status) === "CLOSED");
  256. else if (currentFilter === "priority-urgent") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "URGENT");
  257. else if (currentFilter === "priority-high") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "HIGH");
  258. else if (currentFilter === "priority-medium") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "MEDIUM");
  259. else if (currentFilter === "priority-low") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "LOW");
  260. else filtered = visible;
  261. filtered = filtered.sort((a, b) => new Date(b.createdAt) - new Date(a.createdAt));
  262. const editTask = list.find((t) => t.id === taskId) || {};
  263. const editTags = Array.isArray(editTask.tags) ? editTask.tags : [];
  264. const minCreate = moment().add(1, "minute").format("YYYY-MM-DDTHH:mm");
  265. const ret = typeof returnTo === "string" && returnTo.startsWith("/tasks")
  266. ? returnTo
  267. : "/tasks?filter=mine";
  268. return template(
  269. title,
  270. section(
  271. div(
  272. { class: "tags-header" },
  273. h2(i18n.tasksTitle),
  274. p(i18n.tasksDescription)
  275. ),
  276. div(
  277. { class: "filters" },
  278. form(
  279. { method: "GET", action: "/tasks" },
  280. button({ type: "submit", name: "filter", value: "all", class: currentFilter === "all" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAll),
  281. button({ type: "submit", name: "filter", value: "mine", class: currentFilter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMine),
  282. button({ type: "submit", name: "filter", value: "assigned", class: currentFilter === "assigned" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAssigned),
  283. button({ type: "submit", name: "filter", value: "open", class: currentFilter === "open" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterOpen),
  284. button({ type: "submit", name: "filter", value: "in-progress", class: currentFilter === "in-progress" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterInProgress),
  285. button({ type: "submit", name: "filter", value: "closed", class: currentFilter === "closed" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterClosed),
  286. button({ type: "submit", name: "filter", value: "priority-low", class: currentFilter === "priority-low" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterLow),
  287. button({ type: "submit", name: "filter", value: "priority-medium", class: currentFilter === "priority-medium" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMedium),
  288. button({ type: "submit", name: "filter", value: "priority-high", class: currentFilter === "priority-high" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterHigh),
  289. button({ type: "submit", name: "filter", value: "priority-urgent", class: currentFilter === "priority-urgent" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterUrgent),
  290. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.taskCreateButton)
  291. )
  292. )
  293. ),
  294. section(
  295. currentFilter === "edit" || currentFilter === "create"
  296. ? div(
  297. { class: "task-form" },
  298. form(
  299. { action: currentFilter === "edit" ? `/tasks/update/${encodeURIComponent(taskId)}` : "/tasks/create", method: "POST" },
  300. input({ type: "hidden", name: "returnTo", value: ret }),
  301. label(i18n.taskTitleLabel), br(),
  302. input({ type: "text", name: "title", required: true, value: currentFilter === "edit" ? (editTask.title || "") : "" }), br(), br(),
  303. label(i18n.taskDescriptionLabel), br(),
  304. textarea({ name: "description", required: true, placeholder: i18n.taskDescriptionPlaceholder, rows: "4" }, currentFilter === "edit" ? (editTask.description || "") : ""), br(), br(),
  305. label(i18n.taskStartTimeLabel), br(),
  306. input({
  307. type: "datetime-local",
  308. name: "startTime",
  309. required: true,
  310. min: currentFilter === "create" ? minCreate : undefined,
  311. value: currentFilter === "edit" && editTask.startTime ? moment(editTask.startTime).format("YYYY-MM-DDTHH:mm") : ""
  312. }), br(), br(),
  313. label(i18n.taskEndTimeLabel), br(),
  314. input({
  315. type: "datetime-local",
  316. name: "endTime",
  317. required: true,
  318. min: currentFilter === "create" ? minCreate : undefined,
  319. value: currentFilter === "edit" && editTask.endTime ? moment(editTask.endTime).format("YYYY-MM-DDTHH:mm") : ""
  320. }), br(), br(),
  321. label(i18n.taskPriorityLabel), br(),
  322. select(
  323. { name: "priority", required: true },
  324. opt("URGENT", String(editTask.priority || "").toUpperCase() === "URGENT", i18n.taskPriorityUrgent),
  325. opt("HIGH", String(editTask.priority || "").toUpperCase() === "HIGH", i18n.taskPriorityHigh),
  326. opt("MEDIUM", String(editTask.priority || "").toUpperCase() === "MEDIUM", i18n.taskPriorityMedium),
  327. opt("LOW", !editTask.priority || String(editTask.priority || "").toUpperCase() === "LOW", i18n.taskPriorityLow)
  328. ), br(), br(),
  329. label(i18n.taskLocationLabel), br(),
  330. input({ type: "text", name: "location", value: editTask.location || "" }), br(), br(),
  331. label(i18n.taskTagsLabel), br(),
  332. input({ type: "text", name: "tags", value: editTags.join(", ") }), br(), br(),
  333. label(i18n.taskVisibilityLabel), br(),
  334. select(
  335. { name: "isPublic", id: "isPublic" },
  336. opt("PUBLIC", String(editTask.isPublic || "PUBLIC").toUpperCase() === "PUBLIC", i18n.taskPublic),
  337. opt("PRIVATE", String(editTask.isPublic || "").toUpperCase() === "PRIVATE", i18n.taskPrivate)
  338. ), br(), br(),
  339. button({ type: "submit" }, currentFilter === "edit" ? i18n.taskUpdateButton : i18n.taskCreateButton)
  340. )
  341. )
  342. : div(
  343. { class: "task-list" },
  344. filtered.length > 0
  345. ? filtered.map((t) => renderTaskItem(t, currentFilter))
  346. : p(i18n.notasks)
  347. )
  348. )
  349. );
  350. };
  351. exports.singleTaskView = async (task, filter, comments = []) => {
  352. const currentFilter = filter || "all";
  353. const assignees = safeArray(task.assignees);
  354. const commentCount = typeof task.commentCount === "number" ? task.commentCount : 0;
  355. const topbar = renderTaskTopbar(task, currentFilter, { single: true });
  356. return template(
  357. task.title,
  358. section(
  359. div(
  360. { class: "filters" },
  361. form(
  362. { method: "GET", action: "/tasks" },
  363. button({ type: "submit", name: "filter", value: "all", class: currentFilter === "all" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAll),
  364. button({ type: "submit", name: "filter", value: "mine", class: currentFilter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMine),
  365. button({ type: "submit", name: "filter", value: "assigned", class: currentFilter === "assigned" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAssigned),
  366. button({ type: "submit", name: "filter", value: "open", class: currentFilter === "open" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterOpen),
  367. button({ type: "submit", name: "filter", value: "in-progress", class: currentFilter === "in-progress" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterInProgress),
  368. button({ type: "submit", name: "filter", value: "closed", class: currentFilter === "closed" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterClosed),
  369. button({ type: "submit", name: "filter", value: "priority-low", class: currentFilter === "priority-low" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterLow),
  370. button({ type: "submit", name: "filter", value: "priority-medium", class: currentFilter === "priority-medium" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMedium),
  371. button({ type: "submit", name: "filter", value: "priority-high", class: currentFilter === "priority-high" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterHigh),
  372. button({ type: "submit", name: "filter", value: "priority-urgent", class: currentFilter === "priority-urgent" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterUrgent),
  373. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.taskCreateButton)
  374. )
  375. ),
  376. div(
  377. { class: "card card-section task" },
  378. topbar ? topbar : null,
  379. renderCardField(i18n.taskTitleLabel + ":", task.title),
  380. renderCardField(i18n.taskDescriptionLabel + ":", ""),
  381. p(...renderUrl(task.description)),
  382. renderCardField(i18n.taskStartTimeLabel + ":", task.startTime ? moment(task.startTime).format("YYYY/MM/DD HH:mm:ss") : ""),
  383. renderCardField(i18n.taskEndTimeLabel + ":", task.endTime ? moment(task.endTime).format("YYYY/MM/DD HH:mm:ss") : ""),
  384. renderCardField(i18n.taskPriorityLabel + ":", task.priority),
  385. task.location && String(task.location).trim() ? renderCardField(i18n.taskLocationLabel + ":", task.location) : null,
  386. renderCardField(i18n.taskCreatedAt + ":", task.createdAt ? moment(task.createdAt).format(i18n.dateFormat) : ""),
  387. renderCardField(i18n.taskBy + ":", a({ href: `/author/${encodeURIComponent(task.author)}`, class: "user-link" }, task.author)),
  388. renderCardField(i18n.taskStatus + ":", statusLabel(task.status)),
  389. renderCardField(i18n.taskVisibilityLabel + ":", visibilityLabel(task.isPublic)),
  390. div(
  391. { class: "card-field" },
  392. span({ class: "card-label" }, i18n.taskAssignedTo + ":"),
  393. span(
  394. { class: "card-value" },
  395. assignees.length
  396. ? assignees.map((id, i) => [i > 0 ? ", " : "", a({ class: "user-link", href: `/author/${encodeURIComponent(id)}` }, id)]).flat()
  397. : i18n.noAssignees
  398. )
  399. ),
  400. Array.isArray(task.tags) && task.tags.length
  401. ? div(
  402. { class: "card-tags" },
  403. task.tags.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
  404. )
  405. : null,
  406. div(
  407. { class: "card-comments-summary" },
  408. span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
  409. span({ class: "card-value" }, String(commentCount))
  410. )
  411. ),
  412. renderTaskCommentsSection(task.id, comments, currentFilter)
  413. )
  414. );
  415. };