bookmark_view.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487
  1. const { form, button, div, h2, p, section, input, label, textarea, br, a, span, select, option } =
  2. require("../server/node_modules/hyperaxe");
  3. const { template, i18n, userLink} = require("./main_views");
  4. const moment = require("../server/node_modules/moment");
  5. const { config } = require("../server/SSB_server.js");
  6. const { renderUrl } = require("../backend/renderUrl");
  7. const opinionCategories = require("../backend/opinion_categories");
  8. const userId = config.keys.id;
  9. const safeArr = (v) => (Array.isArray(v) ? v : []);
  10. const safeText = (v) => String(v || "").trim();
  11. const buildReturnTo = (filter, params = {}) => {
  12. const f = safeText(filter || "all");
  13. const q = safeText(params.q || "");
  14. const sort = safeText(params.sort || "recent");
  15. const parts = [`filter=${encodeURIComponent(f)}`];
  16. if (q) parts.push(`q=${encodeURIComponent(q)}`);
  17. if (sort) parts.push(`sort=${encodeURIComponent(sort)}`);
  18. return `/bookmarks?${parts.join("&")}`;
  19. };
  20. const renderPMButton = (recipient, className = "filter-btn") => {
  21. const r = safeText(recipient);
  22. if (!r) return null;
  23. if (String(r) === String(userId)) return null;
  24. return form(
  25. { method: "GET", action: "/pm" },
  26. input({ type: "hidden", name: "recipients", value: r }),
  27. button({ type: "submit", class: className }, i18n.privateMessage)
  28. );
  29. };
  30. const renderBookmarkActions = (filter, bookmark, params = {}) => {
  31. const returnTo = buildReturnTo(filter, params);
  32. const isAuthor = String(bookmark.author) === String(userId);
  33. const hasOpinions = Object.keys(bookmark.opinions || {}).length > 0;
  34. return isAuthor
  35. ? div(
  36. { class: "bookmark-actions" },
  37. !hasOpinions
  38. ? form(
  39. { method: "GET", action: `/bookmarks/edit/${encodeURIComponent(bookmark.id)}` },
  40. input({ type: "hidden", name: "returnTo", value: returnTo }),
  41. button({ class: "update-btn", type: "submit" }, i18n.bookmarkUpdateButton)
  42. )
  43. : null,
  44. form(
  45. { method: "POST", action: `/bookmarks/delete/${encodeURIComponent(bookmark.id)}` },
  46. input({ type: "hidden", name: "returnTo", value: returnTo }),
  47. button({ class: "delete-btn", type: "submit" }, i18n.bookmarkDeleteButton)
  48. )
  49. )
  50. : null;
  51. };
  52. const renderBookmarkCommentsSection = (bookmarkId, rootId, comments = [], returnTo = null) => {
  53. const list = safeArr(comments).filter(c => {
  54. const t = c && c.value && c.value.content && c.value.content.text;
  55. return t && String(t).trim();
  56. });
  57. const commentsCount = list.length;
  58. return div(
  59. { class: "vote-comments-section" },
  60. div(
  61. { class: "comments-count" },
  62. span({ class: "card-label" }, i18n.voteCommentsLabel + ": "),
  63. span({ class: "card-value" }, String(commentsCount))
  64. ),
  65. div(
  66. { class: "comment-form-wrapper" },
  67. h2({ class: "comment-form-title" }, i18n.voteNewCommentLabel),
  68. form(
  69. { method: "POST", action: `/bookmarks/${encodeURIComponent(bookmarkId)}/comments`, class: "comment-form", enctype: "multipart/form-data" },
  70. returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
  71. rootId ? input({ type: "hidden", name: "rootId", value: rootId }) : null,
  72. textarea({
  73. id: "comment-text",
  74. name: "text",
  75. rows: 4,
  76. class: "comment-textarea",
  77. placeholder: i18n.voteNewCommentPlaceholder
  78. }),
  79. div({ class: "comment-file-upload" }, label(i18n.uploadMedia), input({ type: "file", name: "blob" })),
  80. br(),
  81. button({ type: "submit", class: "comment-submit-btn" }, i18n.voteNewCommentButton)
  82. )
  83. ),
  84. list.length
  85. ? div(
  86. { class: "comments-list" },
  87. list.map((c) => {
  88. const author = c?.value?.author || "";
  89. const ts = c?.value?.timestamp || c?.timestamp;
  90. const absDate = ts ? moment(ts).format("YYYY/MM/DD HH:mm:ss") : "";
  91. const relDate = ts ? moment(ts).fromNow() : "";
  92. const userName = author && author.includes("@") ? author.split("@")[1] : author;
  93. const content = c?.value?.content || {};
  94. const text = content.text || "";
  95. const threadRoot = content.fork || content.root || null;
  96. return div(
  97. { class: "votations-comment-card" },
  98. span(
  99. { class: "created-at" },
  100. span(i18n.createdBy),
  101. author ? a({ href: `/author/${encodeURIComponent(author)}` }, `@${userName}`) : span("(unknown)"),
  102. absDate ? span(" | ") : "",
  103. absDate ? span({ class: "votations-comment-date" }, absDate) : "",
  104. relDate ? span({ class: "votations-comment-date" }, " | ", i18n.sendTime) : "",
  105. relDate && threadRoot
  106. ? a({ href: `/thread/${encodeURIComponent(threadRoot)}#${encodeURIComponent(c.key)}` }, relDate)
  107. : ""
  108. ),
  109. p({ class: "votations-comment-text" }, ...renderUrl(text))
  110. );
  111. })
  112. )
  113. : p({ class: "votations-no-comments" }, i18n.voteNoCommentsYet)
  114. );
  115. };
  116. const renderCardField = (labelText, value) =>
  117. div(
  118. { class: "card-field" },
  119. span({ class: "card-label" }, labelText),
  120. span({ class: "card-value" }, value)
  121. );
  122. const renderFavoriteToggle = (bookmark, returnTo) =>
  123. form(
  124. {
  125. method: "POST",
  126. action: bookmark.isFavorite
  127. ? `/bookmarks/favorites/remove/${encodeURIComponent(bookmark.id)}`
  128. : `/bookmarks/favorites/add/${encodeURIComponent(bookmark.id)}`,
  129. class: "bookmark-favorite-form"
  130. },
  131. returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
  132. button(
  133. { type: "submit", class: "filter-btn" },
  134. bookmark.isFavorite ? i18n.bookmarkRemoveFavoriteButton : i18n.bookmarkAddFavoriteButton
  135. )
  136. );
  137. const renderTags = (tags) => {
  138. const list = safeArr(tags).map((t) => String(t || "").trim()).filter(Boolean);
  139. return list.length
  140. ? div(
  141. { class: "card-tags" },
  142. list.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
  143. )
  144. : null;
  145. };
  146. const renderBookmarkList = (filteredBookmarks, filter, params = {}) => {
  147. const returnTo = buildReturnTo(filter, params);
  148. return filteredBookmarks.length
  149. ? filteredBookmarks.map((bookmark) => {
  150. const commentCount = typeof bookmark.commentCount === "number" ? bookmark.commentCount : 0;
  151. const lastVisit = bookmark.lastVisit ? moment(bookmark.lastVisit) : null;
  152. const lastVisitTxt =
  153. lastVisit && lastVisit.isValid()
  154. ? `${lastVisit.format("YYYY/MM/DD HH:mm:ss")} (${lastVisit.fromNow()})`
  155. : i18n.noLastVisit;
  156. const urlLink = bookmark.url
  157. ? a({ href: bookmark.url, target: "_blank", rel: "noreferrer noopener", class: "bookmark-url" }, bookmark.url)
  158. : i18n.noUrl;
  159. return div(
  160. { class: "tags-header bookmark-card" },
  161. div(
  162. { class: "bookmark-topbar" },
  163. div(
  164. { class: "bookmark-topbar-left" },
  165. form(
  166. { method: "GET", action: `/bookmarks/${encodeURIComponent(bookmark.id)}` },
  167. input({ type: "hidden", name: "returnTo", value: returnTo }),
  168. input({ type: "hidden", name: "filter", value: filter || "all" }),
  169. params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
  170. params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
  171. button({ type: "submit", class: "filter-btn" }, i18n.viewDetails)
  172. ),
  173. renderPMButton(bookmark.author),
  174. renderFavoriteToggle(bookmark, returnTo)
  175. ),
  176. renderBookmarkActions(filter, bookmark, params)
  177. ),
  178. h2({ class: "bookmark-title" }, bookmark.category || bookmark.url || ""),
  179. renderCardField(i18n.bookmarkUrlLabel + ":", urlLink),
  180. renderCardField(i18n.bookmarkLastVisitLabel + ":", lastVisitTxt),
  181. renderCardField(i18n.bookmarkCategoryLabel + ":", safeText(bookmark.category) || i18n.noCategory),
  182. br,
  183. div(
  184. { class: "card-comments-summary" },
  185. span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
  186. span({ class: "card-value" }, String(commentCount)),
  187. br(),
  188. br(),
  189. form(
  190. { method: "GET", action: `/bookmarks/${encodeURIComponent(bookmark.id)}` },
  191. input({ type: "hidden", name: "returnTo", value: returnTo }),
  192. input({ type: "hidden", name: "filter", value: filter || "all" }),
  193. params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
  194. params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
  195. button({ type: "submit", class: "filter-btn" }, i18n.voteCommentsForumButton)
  196. )
  197. ),
  198. (() => {
  199. const createdTs = bookmark.createdAt ? new Date(bookmark.createdAt).getTime() : NaN;
  200. const updatedTs = bookmark.updatedAt ? new Date(bookmark.updatedAt).getTime() : NaN;
  201. const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
  202. return p(
  203. { class: "card-footer" },
  204. span({ class: "date-link" }, `${moment(bookmark.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
  205. userLink(bookmark.author),
  206. showUpdated
  207. ? span(
  208. { class: "votations-comment-date" },
  209. ` | ${i18n.bookmarkUpdatedAt}: ${moment(bookmark.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
  210. )
  211. : null
  212. );
  213. })()
  214. );
  215. })
  216. : p(params.q ? i18n.bookmarkNoMatch : i18n.noBookmarks);
  217. };
  218. const renderBookmarkForm = (filter, bookmarkId, bookmarkToEdit, tags, params = {}) => {
  219. const returnFilter = filter === "create" ? "all" : params.filter || "all";
  220. const returnTo = params.returnTo || buildReturnTo(returnFilter, params);
  221. const lastVisitValue =
  222. bookmarkToEdit?.lastVisit && moment(bookmarkToEdit.lastVisit).isValid()
  223. ? moment(bookmarkToEdit.lastVisit).format("YYYY-MM-DDTHH:mm")
  224. : "";
  225. const lastVisitMax = moment().format("YYYY-MM-DDTHH:mm");
  226. return div(
  227. { class: "div-center bookmark-form" },
  228. form(
  229. { action: filter === "edit" ? `/bookmarks/update/${encodeURIComponent(bookmarkId)}` : "/bookmarks/create", method: "POST" },
  230. input({ type: "hidden", name: "returnTo", value: returnTo }),
  231. label(i18n.bookmarkUrlLabel),
  232. br(),
  233. input({
  234. type: "url",
  235. name: "url",
  236. id: "url",
  237. required: true,
  238. placeholder: i18n.bookmarkUrlPlaceholder,
  239. value: filter === "edit" ? bookmarkToEdit.url || "" : ""
  240. }),
  241. br(),
  242. br(),
  243. label(i18n.bookmarkDescriptionLabel),
  244. br(),
  245. textarea(
  246. { name: "description", id: "description", placeholder: i18n.bookmarkDescriptionPlaceholder, rows: "4" },
  247. filter === "edit" ? bookmarkToEdit.description || "" : ""
  248. ),
  249. br(),
  250. label(i18n.bookmarkLastVisitLabel),
  251. br(),
  252. input({
  253. type: "datetime-local",
  254. name: "lastVisit",
  255. max: lastVisitMax,
  256. value: filter === "edit" ? lastVisitValue : ""
  257. }),
  258. br(),
  259. br(),
  260. label(i18n.bookmarkCategoryLabel),
  261. br(),
  262. input({
  263. type: "text",
  264. name: "category",
  265. id: "category",
  266. placeholder: i18n.bookmarkCategoryPlaceholder,
  267. value: filter === "edit" ? bookmarkToEdit.category || "" : ""
  268. }),
  269. br(),
  270. label(i18n.bookmarkTagsLabel),
  271. br(),
  272. input({
  273. type: "text",
  274. name: "tags",
  275. id: "tags",
  276. placeholder: i18n.bookmarkTagsPlaceholder,
  277. value: filter === "edit" ? safeArr(tags).join(", ") : ""
  278. }),
  279. br(),
  280. br(),
  281. button({ type: "submit" }, filter === "edit" ? i18n.bookmarkUpdateButton : i18n.bookmarkCreateButton)
  282. )
  283. );
  284. };
  285. exports.bookmarkView = async (bookmarks, filter = "all", bookmarkId = null, params = {}) => {
  286. const title =
  287. filter === "mine"
  288. ? i18n.bookmarkMineSectionTitle
  289. : filter === "create"
  290. ? i18n.bookmarkCreateSectionTitle
  291. : filter === "edit"
  292. ? i18n.bookmarkUpdateSectionTitle
  293. : filter === "recent"
  294. ? i18n.bookmarkRecentSectionTitle
  295. : filter === "top"
  296. ? i18n.bookmarkTopSectionTitle
  297. : filter === "favorites"
  298. ? i18n.bookmarkFavoritesSectionTitle
  299. : i18n.bookmarkAllSectionTitle;
  300. const q = safeText(params.q || "");
  301. const sort = safeText(params.sort || "recent");
  302. const list = safeArr(bookmarks);
  303. const bookmarkToEdit = bookmarkId ? list.find((b) => b.id === bookmarkId) : null;
  304. const tags = bookmarkToEdit && Array.isArray(bookmarkToEdit.tags) ? bookmarkToEdit.tags : [];
  305. return template(
  306. title,
  307. section(
  308. div({ class: "tags-header" }, h2(title), p(i18n.bookmarkDescription)),
  309. div(
  310. { class: "filters" },
  311. form(
  312. { method: "GET", action: "/bookmarks", class: "ui-toolbar ui-toolbar--filters" },
  313. input({ type: "hidden", name: "q", value: q }),
  314. input({ type: "hidden", name: "sort", value: sort }),
  315. button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterAll),
  316. button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterMine),
  317. button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterTop),
  318. button({ type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterFavorites),
  319. button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterRecent),
  320. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.bookmarkCreateButton)
  321. )
  322. )
  323. ),
  324. section(
  325. filter === "edit" || filter === "create"
  326. ? renderBookmarkForm(filter, bookmarkId, bookmarkToEdit || {}, tags, { ...params, filter })
  327. : section(
  328. div(
  329. { class: "bookmarks-search" },
  330. form(
  331. { method: "GET", action: "/bookmarks", class: "filter-box" },
  332. input({ type: "hidden", name: "filter", value: filter }),
  333. input({ type: "text", name: "q", value: q, placeholder: i18n.bookmarkSearchPlaceholder, class: "filter-box__input" }),
  334. div(
  335. { class: "filter-box__controls" },
  336. select(
  337. { name: "sort", class: "filter-box__select" },
  338. option({ value: "recent", selected: sort === "recent" }, i18n.bookmarkSortRecent),
  339. option({ value: "oldest", selected: sort === "oldest" }, i18n.bookmarkSortOldest),
  340. option({ value: "top", selected: sort === "top" }, i18n.bookmarkSortTop)
  341. ),
  342. button({ type: "submit", class: "filter-box__button" }, i18n.bookmarkSearchButton)
  343. )
  344. )
  345. ),
  346. div({ class: "bookmark-list" }, renderBookmarkList(list, filter, { q, sort }))
  347. )
  348. )
  349. );
  350. };
  351. exports.singleBookmarkView = async (bookmark, filter = "all", comments = [], params = {}) => {
  352. const q = safeText(params.q || "");
  353. const sort = safeText(params.sort || "recent");
  354. const returnTo = params.returnTo || buildReturnTo(filter, { q, sort });
  355. const isAuthor = String(bookmark.author) === String(userId);
  356. const hasOpinions = Object.keys(bookmark.opinions || {}).length > 0;
  357. const lastVisit = bookmark.lastVisit ? moment(bookmark.lastVisit) : null;
  358. const lastVisitTxt =
  359. lastVisit && lastVisit.isValid()
  360. ? `${lastVisit.format("YYYY/MM/DD HH:mm:ss")} (${lastVisit.fromNow()})`
  361. : i18n.noLastVisit;
  362. const urlLink = bookmark.url
  363. ? a({ href: bookmark.url, target: "_blank", rel: "noreferrer noopener", class: "bookmark-url" }, bookmark.url)
  364. : i18n.noUrl;
  365. const pmBtn = renderPMButton(bookmark.author);
  366. const actions =
  367. isAuthor
  368. ? div(
  369. { class: "bookmark-actions" },
  370. renderFavoriteToggle(bookmark, returnTo),
  371. !hasOpinions
  372. ? form(
  373. { method: "GET", action: `/bookmarks/edit/${encodeURIComponent(bookmark.id)}` },
  374. input({ type: "hidden", name: "returnTo", value: returnTo }),
  375. button({ class: "update-btn", type: "submit" }, i18n.bookmarkUpdateButton)
  376. )
  377. : null,
  378. form(
  379. { method: "POST", action: `/bookmarks/delete/${encodeURIComponent(bookmark.id)}` },
  380. input({ type: "hidden", name: "returnTo", value: returnTo }),
  381. button({ class: "delete-btn", type: "submit" }, i18n.bookmarkDeleteButton)
  382. )
  383. )
  384. : div(
  385. { class: "bookmark-actions" },
  386. pmBtn,
  387. renderFavoriteToggle(bookmark, returnTo)
  388. );
  389. return template(
  390. i18n.bookmarkTitle,
  391. section(
  392. div(
  393. { class: "filters" },
  394. form(
  395. { method: "GET", action: "/bookmarks", class: "ui-toolbar ui-toolbar--filters" },
  396. input({ type: "hidden", name: "q", value: q }),
  397. input({ type: "hidden", name: "sort", value: sort }),
  398. button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterAll),
  399. button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterMine),
  400. button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterTop),
  401. button({ type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterFavorites),
  402. button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.bookmarkFilterRecent),
  403. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.bookmarkCreateButton)
  404. )
  405. ),
  406. div(
  407. { class: "bookmark-item card" },
  408. actions,
  409. h2({ class: "bookmark-title" }, bookmark.category || bookmark.url || ""),
  410. renderCardField(i18n.bookmarkUrlLabel + ":", urlLink),
  411. renderCardField(i18n.bookmarkLastVisitLabel + ":", lastVisitTxt),
  412. renderCardField(i18n.bookmarkCategoryLabel + ":", safeText(bookmark.category) || i18n.noCategory),
  413. safeText(bookmark.description) ? p(...renderUrl(bookmark.description)) : null,
  414. renderTags(bookmark.tags),
  415. br(),
  416. (() => {
  417. const createdTs = bookmark.createdAt ? new Date(bookmark.createdAt).getTime() : NaN;
  418. const updatedTs = bookmark.updatedAt ? new Date(bookmark.updatedAt).getTime() : NaN;
  419. const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
  420. return p(
  421. { class: "card-footer" },
  422. span({ class: "date-link" }, `${moment(bookmark.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
  423. userLink(bookmark.author),
  424. showUpdated
  425. ? span(
  426. { class: "votations-comment-date" },
  427. ` | ${i18n.bookmarkUpdatedAt}: ${moment(bookmark.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
  428. )
  429. : null
  430. );
  431. })(),
  432. div(
  433. { class: "voting-buttons" },
  434. opinionCategories.map((category) =>
  435. form(
  436. { method: "POST", action: `/bookmarks/opinions/${encodeURIComponent(bookmark.id)}/${category}` },
  437. input({ type: "hidden", name: "returnTo", value: returnTo }),
  438. button(
  439. { class: "vote-btn" },
  440. `${i18n[`vote${category.charAt(0).toUpperCase() + category.slice(1)}`] || category} [${bookmark.opinions?.[category] || 0}]`
  441. )
  442. )
  443. )
  444. )
  445. ),
  446. renderBookmarkCommentsSection(bookmark.id, bookmark.rootId, comments, returnTo)
  447. )
  448. );
  449. };