| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457 |
- const { div, h2, p, section, button, form, input, select, option, a, br, textarea, label, span } = require("../server/node_modules/hyperaxe");
- const moment = require("../server/node_modules/moment");
- const { template, i18n } = require("./main_views");
- const { config } = require("../server/SSB_server.js");
- const { renderUrl } = require("../backend/renderUrl");
- const userId = config.keys.id;
- const opt = (value, isSelected, text) =>
- option(Object.assign({ value }, isSelected ? { selected: "selected" } : {}), text);
- const safeArray = (v) => Array.isArray(v) ? v : [];
- const toValueChildren = (v) => {
- if (v === undefined || v === null) return [];
- if (Array.isArray(v)) return v;
- if (typeof v === "string") return renderUrl(v);
- if (typeof v === "number" || typeof v === "boolean") return renderUrl(String(v));
- return [v];
- };
- const renderCardField = (labelText, valueNode) =>
- div(
- { class: "card-field" },
- span({ class: "card-label" }, labelText),
- span({ class: "card-value" }, ...toValueChildren(valueNode))
- );
- const normalizeStatus = (v) => {
- const up = String(v || "").toUpperCase();
- if (up === "OPEN" || up === "IN-PROGRESS" || up === "CLOSED") return up;
- return "OPEN";
- };
- const statusLabel = (s) => {
- const up = normalizeStatus(s);
- if (up === "OPEN") return i18n.taskStatusOpen;
- if (up === "IN-PROGRESS") return i18n.taskStatusInProgress;
- return i18n.taskStatusClosed;
- };
- const visibilityLabel = (v) => {
- const vv = String(v || "").toUpperCase();
- if (vv === "PRIVATE") return i18n.taskPrivate;
- return i18n.taskPublic;
- };
- const renderTaskOwnerActions = (task, returnTo) => {
- const st = normalizeStatus(task.status || "OPEN");
- const setStatusLabel = i18n.taskSetStatus;
- return [
- form(
- { method: "GET", action: `/tasks/edit/${encodeURIComponent(task.id)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button({ type: "submit", class: "update-btn" }, i18n.taskUpdateButton)
- ),
- form(
- { method: "POST", action: `/tasks/delete/${encodeURIComponent(task.id)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button({ type: "submit", class: "delete-btn" }, i18n.taskDeleteButton)
- ),
- form(
- { method: "POST", action: `/tasks/status/${encodeURIComponent(task.id)}`, class: "project-control-form project-control-form--status" },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- select(
- { name: "status", class: "project-control-select" },
- option({ value: "OPEN", selected: st === "OPEN" }, i18n.taskStatusOpen),
- option({ value: "IN-PROGRESS", selected: st === "IN-PROGRESS" }, i18n.taskStatusInProgress),
- option({ value: "CLOSED", selected: st === "CLOSED" }, i18n.taskStatusClosed)
- ),
- button({ class: "status-btn project-control-btn", type: "submit" }, setStatusLabel)
- )
- ];
- };
- const renderTaskAssignAction = (task, isAssignedToMe, returnTo) => {
- const st = normalizeStatus(task.status || "OPEN");
- if (st === "CLOSED") return null;
- return form(
- { method: "POST", action: `/tasks/assign/${encodeURIComponent(task.id)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button({ type: "submit", class: "filter-btn" }, isAssignedToMe ? i18n.taskUnassignButton : i18n.taskAssignButton)
- );
- };
- const renderTaskTopbar = (task, filter, opts = {}) => {
- const currentFilter = filter || "all";
- const isSingle = !!opts.single;
- const returnToList = `/tasks?filter=${encodeURIComponent(currentFilter)}`;
- const returnToSelf = `/tasks/${encodeURIComponent(task.id)}?filter=${encodeURIComponent(currentFilter)}`;
- const rt = isSingle ? returnToSelf : returnToList;
- const assignees = safeArray(task.assignees);
- const isAssignedToMe = assignees.includes(userId);
- const leftActions = [];
- if (!isSingle) {
- leftActions.push(
- form(
- { method: "GET", action: `/tasks/${encodeURIComponent(task.id)}` },
- input({ type: "hidden", name: "filter", value: currentFilter }),
- button({ type: "submit", class: "filter-btn" }, i18n.viewDetails)
- )
- );
- }
- if (task.author && task.author !== userId) {
- leftActions.push(
- form(
- { method: "GET", action: "/pm" },
- input({ type: "hidden", name: "recipients", value: task.author }),
- button({ type: "submit", class: "filter-btn" }, i18n.privateMessage)
- )
- );
- }
- const ownerActions = task.author === userId ? renderTaskOwnerActions(task, rt) : [];
- const assignNode = renderTaskAssignAction(task, isAssignedToMe, rt);
- const rightActions = [];
- if (assignNode) rightActions.push(assignNode);
- if (ownerActions.length) rightActions.push(...ownerActions);
- const leftNode = leftActions.length ? div({ class: "bookmark-topbar-left task-topbar-left" }, ...leftActions) : null;
- const rightNode = rightActions.length ? div({ class: "bookmark-actions task-actions" }, ...rightActions) : null;
- const nodes = [];
- if (leftNode) nodes.push(leftNode);
- if (rightNode) nodes.push(rightNode);
- return nodes.length ? div({ class: isSingle ? "bookmark-topbar task-topbar-single" : "bookmark-topbar" }, ...nodes) : null;
- };
- const renderTaskCommentsSection = (taskId, comments = [], currentFilter = "all") => {
- const commentsCount = Array.isArray(comments) ? comments.length : 0;
- const returnTo = `/tasks/${encodeURIComponent(taskId)}?filter=${encodeURIComponent(currentFilter || "all")}`;
- return div(
- { class: "vote-comments-section" },
- div(
- { class: "comments-count" },
- span({ class: "card-label" }, i18n.voteCommentsLabel + ": "),
- span({ class: "card-value" }, String(commentsCount))
- ),
- div(
- { class: "comment-form-wrapper" },
- h2({ class: "comment-form-title" }, i18n.voteNewCommentLabel),
- form(
- { method: "POST", action: `/tasks/${encodeURIComponent(taskId)}/comments`, class: "comment-form", enctype: "multipart/form-data" },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- textarea({
- id: "comment-text",
- name: "text",
- rows: 4,
- class: "comment-textarea",
- placeholder: i18n.voteNewCommentPlaceholder
- }),
- div({ class: "comment-file-upload" }, label(i18n.uploadMedia), input({ type: "file", name: "blob" })),
- br(),
- button({ type: "submit", class: "comment-submit-btn" }, i18n.voteNewCommentButton)
- )
- ),
- comments && comments.length
- ? div(
- { class: "comments-list" },
- comments.map((c) => {
- const author = c.value && c.value.author ? c.value.author : "";
- const ts = c.value && c.value.timestamp ? c.value.timestamp : c.timestamp;
- const absDate = ts ? moment(ts).format("YYYY/MM/DD HH:mm:ss") : "";
- const relDate = ts ? moment(ts).fromNow() : "";
- const userName = author && author.includes("@") ? author.split("@")[1] : author;
- const content = c.value && c.value.content ? c.value.content : {};
- const root = content.fork || content.root || "";
- const text = content.text || "";
- return div(
- { class: "votations-comment-card" },
- span(
- { class: "created-at" },
- span(i18n.createdBy),
- author ? a({ href: `/author/${encodeURIComponent(author)}` }, `@${userName}`) : span("(unknown)"),
- absDate ? span(" | ") : "",
- absDate ? span({ class: "votations-comment-date" }, absDate) : "",
- relDate ? span({ class: "votations-comment-date" }, " | ", i18n.sendTime) : "",
- relDate && root ? a({ href: `/thread/${encodeURIComponent(root)}#${encodeURIComponent(c.key)}` }, relDate) : ""
- ),
- p({ class: "votations-comment-text" }, ...renderUrl(String(text)))
- );
- })
- )
- : p({ class: "votations-no-comments" }, i18n.voteNoCommentsYet)
- );
- };
- const renderTaskItem = (task, filter) => {
- const currentFilter = filter || "all";
- const assignees = safeArray(task.assignees);
- const commentCount = typeof task.commentCount === "number" ? task.commentCount : 0;
- const topbar = renderTaskTopbar(task, currentFilter, { single: false });
- return div(
- { class: "card card-section task" },
- topbar ? topbar : null,
- renderCardField(i18n.taskTitleLabel + ":", task.title),
- renderCardField(i18n.taskDescriptionLabel + ":", ""),
- p(...renderUrl(task.description)),
- task.location && String(task.location).trim() ? renderCardField(i18n.taskLocationLabel + ":", task.location) : null,
- renderCardField(i18n.taskStatus + ":", statusLabel(task.status)),
- renderCardField(i18n.taskPriorityLabel + ":", task.priority),
- renderCardField(i18n.taskVisibilityLabel + ":", visibilityLabel(task.isPublic)),
- renderCardField(i18n.taskStartTimeLabel + ":", task.startTime ? moment(task.startTime).format("YYYY/MM/DD HH:mm:ss") : ""),
- renderCardField(i18n.taskEndTimeLabel + ":", task.endTime ? moment(task.endTime).format("YYYY/MM/DD HH:mm:ss") : ""),
- br(),
- div(
- { class: "card-field" },
- span({ class: "card-label" }, i18n.taskAssignedTo + ":"),
- span(
- { class: "card-value" },
- assignees.length
- ? assignees.map((id, i) => [i > 0 ? ", " : "", a({ class: "user-link", href: `/author/${encodeURIComponent(id)}` }, id)]).flat()
- : i18n.noAssignees
- )
- ),
- br(),
- Array.isArray(task.tags) && task.tags.length
- ? div(
- { class: "card-tags" },
- task.tags.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
- )
- : null,
- div(
- { class: "card-comments-summary" },
- span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
- span({ class: "card-value" }, String(commentCount)),
- br(),
- br(),
- form(
- { method: "GET", action: `/tasks/${encodeURIComponent(task.id)}` },
- input({ type: "hidden", name: "filter", value: currentFilter }),
- button({ type: "submit", class: "filter-btn" }, i18n.voteCommentsForumButton)
- )
- ),
- br(),
- p(
- { class: "card-footer" },
- span({ class: "date-link" }, `${moment(task.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
- a({ href: `/author/${encodeURIComponent(task.author)}`, class: "user-link" }, `${task.author}`)
- )
- );
- };
- exports.taskView = async (tasks, filter, taskId, returnTo) => {
- const list = Array.isArray(tasks) ? tasks : [tasks];
- const currentFilter = filter || "all";
- const title =
- currentFilter === "mine" ? i18n.taskMineSectionTitle :
- currentFilter === "create" ? i18n.taskCreateSectionTitle :
- currentFilter === "edit" ? i18n.taskUpdateSectionTitle :
- currentFilter === "open" ? i18n.taskOpenTitle :
- currentFilter === "in-progress" ? i18n.taskInProgressTitle :
- currentFilter === "closed" ? i18n.taskClosedTitle :
- currentFilter === "assigned" ? i18n.taskAssignedTitle :
- currentFilter === "priority-urgent" ? i18n.taskFilterUrgent :
- currentFilter === "priority-high" ? i18n.taskFilterHigh :
- currentFilter === "priority-medium" ? i18n.taskFilterMedium :
- currentFilter === "priority-low" ? i18n.taskFilterLow :
- i18n.taskAllSectionTitle;
- const canSee = (t) => {
- const vis = String(t.isPublic || "").toUpperCase();
- if (vis === "PUBLIC") return true;
- if (t.author === userId) return true;
- return safeArray(t.assignees).includes(userId);
- };
- const visible = list.filter(canSee);
- let filtered;
- if (currentFilter === "mine") filtered = visible.filter((t) => t.author === userId);
- else if (currentFilter === "assigned") filtered = visible.filter((t) => safeArray(t.assignees).includes(userId));
- else if (currentFilter === "open") filtered = visible.filter((t) => normalizeStatus(t.status) === "OPEN");
- else if (currentFilter === "in-progress") filtered = visible.filter((t) => normalizeStatus(t.status) === "IN-PROGRESS");
- else if (currentFilter === "closed") filtered = visible.filter((t) => normalizeStatus(t.status) === "CLOSED");
- else if (currentFilter === "priority-urgent") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "URGENT");
- else if (currentFilter === "priority-high") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "HIGH");
- else if (currentFilter === "priority-medium") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "MEDIUM");
- else if (currentFilter === "priority-low") filtered = visible.filter((t) => String(t.priority).toUpperCase() === "LOW");
- else filtered = visible;
- filtered = filtered.sort((a, b) => new Date(b.createdAt) - new Date(a.createdAt));
- const editTask = list.find((t) => t.id === taskId) || {};
- const editTags = Array.isArray(editTask.tags) ? editTask.tags : [];
- const minCreate = moment().add(1, "minute").format("YYYY-MM-DDTHH:mm");
- const ret = typeof returnTo === "string" && returnTo.startsWith("/tasks")
- ? returnTo
- : "/tasks?filter=mine";
- return template(
- title,
- section(
- div(
- { class: "tags-header" },
- h2(i18n.tasksTitle),
- p(i18n.tasksDescription)
- ),
- div(
- { class: "filters" },
- form(
- { method: "GET", action: "/tasks" },
- button({ type: "submit", name: "filter", value: "all", class: currentFilter === "all" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAll),
- button({ type: "submit", name: "filter", value: "mine", class: currentFilter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMine),
- button({ type: "submit", name: "filter", value: "assigned", class: currentFilter === "assigned" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAssigned),
- button({ type: "submit", name: "filter", value: "open", class: currentFilter === "open" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterOpen),
- button({ type: "submit", name: "filter", value: "in-progress", class: currentFilter === "in-progress" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterInProgress),
- button({ type: "submit", name: "filter", value: "closed", class: currentFilter === "closed" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterClosed),
- button({ type: "submit", name: "filter", value: "priority-low", class: currentFilter === "priority-low" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterLow),
- button({ type: "submit", name: "filter", value: "priority-medium", class: currentFilter === "priority-medium" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMedium),
- button({ type: "submit", name: "filter", value: "priority-high", class: currentFilter === "priority-high" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterHigh),
- button({ type: "submit", name: "filter", value: "priority-urgent", class: currentFilter === "priority-urgent" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterUrgent),
- button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.taskCreateButton)
- )
- )
- ),
- section(
- currentFilter === "edit" || currentFilter === "create"
- ? div(
- { class: "task-form" },
- form(
- { action: currentFilter === "edit" ? `/tasks/update/${encodeURIComponent(taskId)}` : "/tasks/create", method: "POST" },
- input({ type: "hidden", name: "returnTo", value: ret }),
- label(i18n.taskTitleLabel), br(),
- input({ type: "text", name: "title", required: true, value: currentFilter === "edit" ? (editTask.title || "") : "" }), br(), br(),
- label(i18n.taskDescriptionLabel), br(),
- textarea({ name: "description", required: true, placeholder: i18n.taskDescriptionPlaceholder, rows: "4" }, currentFilter === "edit" ? (editTask.description || "") : ""), br(), br(),
- label(i18n.taskStartTimeLabel), br(),
- input({
- type: "datetime-local",
- name: "startTime",
- required: true,
- min: currentFilter === "create" ? minCreate : undefined,
- value: currentFilter === "edit" && editTask.startTime ? moment(editTask.startTime).format("YYYY-MM-DDTHH:mm") : ""
- }), br(), br(),
- label(i18n.taskEndTimeLabel), br(),
- input({
- type: "datetime-local",
- name: "endTime",
- required: true,
- min: currentFilter === "create" ? minCreate : undefined,
- value: currentFilter === "edit" && editTask.endTime ? moment(editTask.endTime).format("YYYY-MM-DDTHH:mm") : ""
- }), br(), br(),
- label(i18n.taskPriorityLabel), br(),
- select(
- { name: "priority", required: true },
- opt("URGENT", String(editTask.priority || "").toUpperCase() === "URGENT", i18n.taskPriorityUrgent),
- opt("HIGH", String(editTask.priority || "").toUpperCase() === "HIGH", i18n.taskPriorityHigh),
- opt("MEDIUM", String(editTask.priority || "").toUpperCase() === "MEDIUM", i18n.taskPriorityMedium),
- opt("LOW", !editTask.priority || String(editTask.priority || "").toUpperCase() === "LOW", i18n.taskPriorityLow)
- ), br(), br(),
- label(i18n.taskLocationLabel), br(),
- input({ type: "text", name: "location", value: editTask.location || "" }), br(), br(),
- label(i18n.taskTagsLabel), br(),
- input({ type: "text", name: "tags", value: editTags.join(", ") }), br(), br(),
- label(i18n.taskVisibilityLabel), br(),
- select(
- { name: "isPublic", id: "isPublic" },
- opt("PUBLIC", String(editTask.isPublic || "PUBLIC").toUpperCase() === "PUBLIC", i18n.taskPublic),
- opt("PRIVATE", String(editTask.isPublic || "").toUpperCase() === "PRIVATE", i18n.taskPrivate)
- ), br(), br(),
- button({ type: "submit" }, currentFilter === "edit" ? i18n.taskUpdateButton : i18n.taskCreateButton)
- )
- )
- : div(
- { class: "task-list" },
- filtered.length > 0
- ? filtered.map((t) => renderTaskItem(t, currentFilter))
- : p(i18n.notasks)
- )
- )
- );
- };
- exports.singleTaskView = async (task, filter, comments = []) => {
- const currentFilter = filter || "all";
- const assignees = safeArray(task.assignees);
- const commentCount = typeof task.commentCount === "number" ? task.commentCount : 0;
- const topbar = renderTaskTopbar(task, currentFilter, { single: true });
- return template(
- task.title,
- section(
- div(
- { class: "filters" },
- form(
- { method: "GET", action: "/tasks" },
- button({ type: "submit", name: "filter", value: "all", class: currentFilter === "all" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAll),
- button({ type: "submit", name: "filter", value: "mine", class: currentFilter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMine),
- button({ type: "submit", name: "filter", value: "assigned", class: currentFilter === "assigned" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterAssigned),
- button({ type: "submit", name: "filter", value: "open", class: currentFilter === "open" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterOpen),
- button({ type: "submit", name: "filter", value: "in-progress", class: currentFilter === "in-progress" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterInProgress),
- button({ type: "submit", name: "filter", value: "closed", class: currentFilter === "closed" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterClosed),
- button({ type: "submit", name: "filter", value: "priority-low", class: currentFilter === "priority-low" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterLow),
- button({ type: "submit", name: "filter", value: "priority-medium", class: currentFilter === "priority-medium" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterMedium),
- button({ type: "submit", name: "filter", value: "priority-high", class: currentFilter === "priority-high" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterHigh),
- button({ type: "submit", name: "filter", value: "priority-urgent", class: currentFilter === "priority-urgent" ? "filter-btn active" : "filter-btn" }, i18n.taskFilterUrgent),
- button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.taskCreateButton)
- )
- ),
- div(
- { class: "card card-section task" },
- topbar ? topbar : null,
- renderCardField(i18n.taskTitleLabel + ":", task.title),
- renderCardField(i18n.taskDescriptionLabel + ":", ""),
- p(...renderUrl(task.description)),
- renderCardField(i18n.taskStartTimeLabel + ":", task.startTime ? moment(task.startTime).format("YYYY/MM/DD HH:mm:ss") : ""),
- renderCardField(i18n.taskEndTimeLabel + ":", task.endTime ? moment(task.endTime).format("YYYY/MM/DD HH:mm:ss") : ""),
- renderCardField(i18n.taskPriorityLabel + ":", task.priority),
- task.location && String(task.location).trim() ? renderCardField(i18n.taskLocationLabel + ":", task.location) : null,
- renderCardField(i18n.taskCreatedAt + ":", task.createdAt ? moment(task.createdAt).format(i18n.dateFormat) : ""),
- renderCardField(i18n.taskBy + ":", a({ href: `/author/${encodeURIComponent(task.author)}`, class: "user-link" }, task.author)),
- renderCardField(i18n.taskStatus + ":", statusLabel(task.status)),
- renderCardField(i18n.taskVisibilityLabel + ":", visibilityLabel(task.isPublic)),
- div(
- { class: "card-field" },
- span({ class: "card-label" }, i18n.taskAssignedTo + ":"),
- span(
- { class: "card-value" },
- assignees.length
- ? assignees.map((id, i) => [i > 0 ? ", " : "", a({ class: "user-link", href: `/author/${encodeURIComponent(id)}` }, id)]).flat()
- : i18n.noAssignees
- )
- ),
- Array.isArray(task.tags) && task.tags.length
- ? div(
- { class: "card-tags" },
- task.tags.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
- )
- : null,
- div(
- { class: "card-comments-summary" },
- span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
- span({ class: "card-value" }, String(commentCount))
- )
- ),
- renderTaskCommentsSection(task.id, comments, currentFilter)
- )
- );
- };
|