image_view.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  1. const { form, button, div, h2, p, section, input, label, br, a, img, span, textarea, select, option } =
  2. require("../server/node_modules/hyperaxe");
  3. const moment = require("../server/node_modules/moment");
  4. const { template, i18n } = require("./main_views");
  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 `/images?${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 renderTags = (tags) => {
  31. const list = safeArr(tags).map((t) => String(t || "").trim()).filter(Boolean);
  32. return list.length
  33. ? div(
  34. { class: "card-tags" },
  35. list.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
  36. )
  37. : null;
  38. };
  39. const renderImageFavoriteToggle = (imgObj, returnTo = "") =>
  40. form(
  41. {
  42. method: "POST",
  43. action: imgObj.isFavorite
  44. ? `/images/favorites/remove/${encodeURIComponent(imgObj.key)}`
  45. : `/images/favorites/add/${encodeURIComponent(imgObj.key)}`
  46. },
  47. returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
  48. button(
  49. { type: "submit", class: "filter-btn" },
  50. imgObj.isFavorite ? i18n.imageRemoveFavoriteButton : i18n.imageAddFavoriteButton
  51. )
  52. );
  53. const renderImageMedia = (imgObj, filter, params = {}) => {
  54. const src = imgObj?.url ? `/blob/${encodeURIComponent(imgObj.url)}` : "";
  55. return imgObj?.url
  56. ? div(
  57. { class: "image-container", style: "display:flex;gap:8px;align-items:center;flex-wrap:wrap;" },
  58. a(
  59. {
  60. href: `/images/${encodeURIComponent(imgObj.key)}?filter=${encodeURIComponent(filter || "all")}${
  61. params.q ? `&q=${encodeURIComponent(params.q)}` : ""
  62. }${params.sort ? `&sort=${encodeURIComponent(params.sort)}` : ""}`
  63. },
  64. img({ src, alt: imgObj.title || "", class: "media-preview", loading: "lazy" })
  65. )
  66. )
  67. : p(i18n.imageNoFile);
  68. };
  69. const renderImageOwnerActions = (filter, imgObj, params = {}) => {
  70. const returnTo = buildReturnTo(filter, params);
  71. const isAuthor = String(imgObj.author) === String(userId);
  72. const hasOpinions = Object.keys(imgObj.opinions || {}).length > 0;
  73. if (!isAuthor) return [];
  74. const items = [];
  75. if (!hasOpinions) {
  76. items.push(
  77. form(
  78. { method: "GET", action: `/images/edit/${encodeURIComponent(imgObj.key)}` },
  79. input({ type: "hidden", name: "returnTo", value: returnTo }),
  80. button({ class: "update-btn", type: "submit" }, i18n.imageUpdateButton)
  81. )
  82. );
  83. }
  84. items.push(
  85. form(
  86. { method: "POST", action: `/images/delete/${encodeURIComponent(imgObj.key)}` },
  87. input({ type: "hidden", name: "returnTo", value: returnTo }),
  88. button({ class: "delete-btn", type: "submit" }, i18n.imageDeleteButton)
  89. )
  90. );
  91. return items;
  92. };
  93. const renderImageList = (images, filter, params = {}) => {
  94. const returnTo = buildReturnTo(filter, params);
  95. return images.length
  96. ? images.map((imgObj) => {
  97. const commentCount = typeof imgObj.commentCount === "number" ? imgObj.commentCount : 0;
  98. const title = safeText(imgObj.title);
  99. const ownerActions = renderImageOwnerActions(filter, imgObj, params);
  100. return div(
  101. { class: "tags-header image-card" },
  102. div(
  103. { class: "bookmark-topbar" },
  104. div(
  105. { class: "bookmark-topbar-left" },
  106. form(
  107. { method: "GET", action: `/images/${encodeURIComponent(imgObj.key)}` },
  108. input({ type: "hidden", name: "returnTo", value: returnTo }),
  109. input({ type: "hidden", name: "filter", value: filter || "all" }),
  110. params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
  111. params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
  112. button({ type: "submit", class: "filter-btn" }, i18n.viewDetails)
  113. ),
  114. renderImageFavoriteToggle(imgObj, returnTo),
  115. renderPMButton(imgObj.author)
  116. ),
  117. ownerActions.length ? div({ class: "bookmark-actions" }, ...ownerActions) : null
  118. ),
  119. title ? h2(title) : null,
  120. renderImageMedia(imgObj, filter, params),
  121. div(
  122. { class: "card-comments-summary" },
  123. span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
  124. span({ class: "card-value" }, String(commentCount)),
  125. br(),
  126. br(),
  127. form(
  128. { method: "GET", action: `/images/${encodeURIComponent(imgObj.key)}` },
  129. input({ type: "hidden", name: "returnTo", value: returnTo }),
  130. input({ type: "hidden", name: "filter", value: filter || "all" }),
  131. params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
  132. params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
  133. button({ type: "submit", class: "filter-btn" }, i18n.voteCommentsForumButton)
  134. )
  135. ),
  136. br(),
  137. (() => {
  138. const createdTs = imgObj.createdAt ? new Date(imgObj.createdAt).getTime() : NaN;
  139. const updatedTs = imgObj.updatedAt ? new Date(imgObj.updatedAt).getTime() : NaN;
  140. const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
  141. return p(
  142. { class: "card-footer" },
  143. span({ class: "date-link" }, `${moment(imgObj.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
  144. a({ href: `/author/${encodeURIComponent(imgObj.author)}`, class: "user-link" }, `${imgObj.author}`),
  145. showUpdated
  146. ? span(
  147. { class: "votations-comment-date" },
  148. ` | ${i18n.imageUpdatedAt}: ${moment(imgObj.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
  149. )
  150. : null
  151. );
  152. })()
  153. );
  154. })
  155. : p(params.q ? i18n.imageNoMatch : i18n.noImages);
  156. };
  157. const renderImageForm = (filter, imageId, imageToEdit, params = {}) => {
  158. const returnFilter = filter === "create" ? "all" : params.filter || "all";
  159. const returnTo = safeText(params.returnTo) || buildReturnTo(returnFilter, params);
  160. const tagsValue = safeArr(imageToEdit?.tags).join(", ");
  161. return div(
  162. { class: "div-center image-form" },
  163. form(
  164. {
  165. action: filter === "edit" ? `/images/update/${encodeURIComponent(imageId)}` : "/images/create",
  166. method: "POST",
  167. enctype: "multipart/form-data"
  168. },
  169. input({ type: "hidden", name: "returnTo", value: returnTo }),
  170. label(i18n.imageFileLabel),
  171. br(),
  172. input({ type: "file", name: "image", required: filter !== "edit" }),
  173. br(),
  174. br(),
  175. imageToEdit?.url
  176. ? img({ src: `/blob/${encodeURIComponent(imageToEdit.url)}`, class: "media-preview", alt: imageToEdit?.title || "" })
  177. : null,
  178. label(i18n.imageTitleLabel),
  179. br(),
  180. input({ type: "text", name: "title", placeholder: i18n.imageTitlePlaceholder, value: imageToEdit?.title || "" }),
  181. br(),
  182. label(i18n.imageDescriptionLabel),
  183. br(),
  184. textarea({ name: "description", placeholder: i18n.imageDescriptionPlaceholder, rows: "4" }, imageToEdit?.description || ""),
  185. br(),
  186. input({ type: "hidden", name: "meme", value: "0" }),
  187. label(i18n.imageTagsLabel),
  188. br(),
  189. input({ type: "text", name: "tags", placeholder: i18n.imageTagsPlaceholder, value: tagsValue }),
  190. br(),
  191. label(i18n.imageMemeLabel),
  192. input({
  193. id: "meme-checkbox",
  194. type: "checkbox",
  195. name: "meme",
  196. value: "1",
  197. class: "meme-checkbox",
  198. ...(imageToEdit?.meme ? { checked: true } : {})
  199. }),
  200. br(),
  201. br(),
  202. button({ type: "submit" }, filter === "edit" ? i18n.imageUpdateButton : i18n.imageCreateButton)
  203. )
  204. );
  205. };
  206. const renderGallery = (images) => {
  207. if (!images.length) return div(i18n.noImages);
  208. return div(
  209. { class: "gallery" },
  210. images.map((imgObj) => {
  211. const src = imgObj.url ? `/image/256/${encodeURIComponent(imgObj.url)}` : "";
  212. return a(
  213. { href: `#img-${encodeURIComponent(imgObj.key)}`, class: "gallery-item" },
  214. img({ src, alt: imgObj.title || "", class: "gallery-image", loading: "lazy" })
  215. );
  216. })
  217. );
  218. };
  219. const renderLightbox = (images) =>
  220. images.map((imgObj) => {
  221. const src = imgObj.url ? `/blob/${encodeURIComponent(imgObj.url)}` : "";
  222. return div(
  223. { id: `img-${encodeURIComponent(imgObj.key)}`, class: "lightbox" },
  224. a({ href: "#", class: "lightbox-close" }, "×"),
  225. img({ src, class: "lightbox-image", alt: imgObj.title || "" })
  226. );
  227. });
  228. const renderImageCommentsSection = (imageKey, comments = [], returnTo = null) => {
  229. const list = safeArr(comments);
  230. const commentsCount = list.length;
  231. return div(
  232. { class: "vote-comments-section" },
  233. div(
  234. { class: "comments-count" },
  235. span({ class: "card-label" }, i18n.voteCommentsLabel + ": "),
  236. span({ class: "card-value" }, String(commentsCount))
  237. ),
  238. div(
  239. { class: "comment-form-wrapper" },
  240. h2({ class: "comment-form-title" }, i18n.voteNewCommentLabel),
  241. form(
  242. { method: "POST", action: `/images/${encodeURIComponent(imageKey)}/comments`, class: "comment-form", enctype: "multipart/form-data" },
  243. returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
  244. textarea({
  245. id: "comment-text",
  246. name: "text",
  247. rows: 4,
  248. class: "comment-textarea",
  249. placeholder: i18n.voteNewCommentPlaceholder
  250. }),
  251. div({ class: "comment-file-upload" }, label(i18n.uploadMedia), input({ type: "file", name: "blob" })),
  252. br(),
  253. button({ type: "submit", class: "comment-submit-btn" }, i18n.voteNewCommentButton)
  254. )
  255. ),
  256. list.length
  257. ? div(
  258. { class: "comments-list" },
  259. list.map((c) => {
  260. const author = c?.value?.author || "";
  261. const ts = c?.value?.timestamp || c?.timestamp;
  262. const absDate = ts ? moment(ts).format("YYYY/MM/DD HH:mm:ss") : "";
  263. const relDate = ts ? moment(ts).fromNow() : "";
  264. const userName = author && author.includes("@") ? author.split("@")[1] : author;
  265. const content = c?.value?.content || {};
  266. const text = content.text || "";
  267. const threadRoot = content.fork || content.root || null;
  268. return div(
  269. { class: "votations-comment-card" },
  270. span(
  271. { class: "created-at" },
  272. span(i18n.createdBy),
  273. author ? a({ href: `/author/${encodeURIComponent(author)}` }, `@${userName}`) : span("(unknown)"),
  274. absDate ? span(" | ") : "",
  275. absDate ? span({ class: "votations-comment-date" }, absDate) : "",
  276. relDate ? span({ class: "votations-comment-date" }, " | ", i18n.sendTime) : "",
  277. relDate && threadRoot ? a({ href: `/thread/${encodeURIComponent(threadRoot)}#${encodeURIComponent(c.key)}` }, relDate) : ""
  278. ),
  279. p({ class: "votations-comment-text" }, ...renderUrl(text))
  280. );
  281. })
  282. )
  283. : p({ class: "votations-no-comments" }, i18n.voteNoCommentsYet)
  284. );
  285. };
  286. exports.imageView = async (images, filter = "all", imageId = null, params = {}) => {
  287. const title =
  288. filter === "mine"
  289. ? i18n.imageMineSectionTitle
  290. : filter === "create"
  291. ? i18n.imageCreateSectionTitle
  292. : filter === "edit"
  293. ? i18n.imageUpdateSectionTitle
  294. : filter === "gallery"
  295. ? i18n.imageGallerySectionTitle
  296. : filter === "meme"
  297. ? i18n.imageMemeSectionTitle
  298. : filter === "recent"
  299. ? i18n.imageRecentSectionTitle
  300. : filter === "top"
  301. ? i18n.imageTopSectionTitle
  302. : filter === "favorites"
  303. ? i18n.imageFavoritesSectionTitle
  304. : i18n.imageAllSectionTitle;
  305. const q = safeText(params.q || "");
  306. const sort = safeText(params.sort || "recent");
  307. const list = safeArr(images);
  308. const imageToEdit = imageId ? list.find((im) => im.key === imageId) : null;
  309. return template(
  310. title,
  311. section(
  312. div({ class: "tags-header" }, h2(title), p(i18n.imageDescription)),
  313. div(
  314. { class: "filters" },
  315. form(
  316. { method: "GET", action: "/images", class: "ui-toolbar ui-toolbar--filters" },
  317. input({ type: "hidden", name: "q", value: q }),
  318. input({ type: "hidden", name: "sort", value: sort }),
  319. button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterAll),
  320. button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMine),
  321. button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterRecent),
  322. button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterTop),
  323. button(
  324. { type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" },
  325. i18n.imageFilterFavorites
  326. ),
  327. button(
  328. { type: "submit", name: "filter", value: "gallery", class: filter === "gallery" ? "filter-btn active" : "filter-btn" },
  329. i18n.imageFilterGallery
  330. ),
  331. button({ type: "submit", name: "filter", value: "meme", class: filter === "meme" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMeme),
  332. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.imageCreateButton)
  333. )
  334. )
  335. ),
  336. section(
  337. filter === "create" || filter === "edit"
  338. ? renderImageForm(filter, imageId, imageToEdit, { ...params, filter })
  339. : section(
  340. div(
  341. { class: "images-search" },
  342. form(
  343. { method: "GET", action: "/images", class: "filter-box" },
  344. input({ type: "hidden", name: "filter", value: filter }),
  345. input({
  346. type: "text",
  347. name: "q",
  348. value: q,
  349. placeholder: i18n.imageSearchPlaceholder,
  350. class: "filter-box__input"
  351. }),
  352. div(
  353. { class: "filter-box__controls" },
  354. select(
  355. { name: "sort", class: "filter-box__select" },
  356. option({ value: "recent", selected: sort === "recent" }, i18n.imageSortRecent),
  357. option({ value: "oldest", selected: sort === "oldest" }, i18n.imageSortOldest),
  358. option({ value: "top", selected: sort === "top" }, i18n.imageSortTop)
  359. ),
  360. button({ type: "submit", class: "filter-box__button" }, i18n.imageSearchButton)
  361. )
  362. )
  363. ),
  364. filter === "gallery" ? renderGallery(list) : div({ class: "images-list" }, renderImageList(list, filter, { q, sort }))
  365. )
  366. ),
  367. ...(filter === "gallery" ? renderLightbox(list) : [])
  368. );
  369. };
  370. exports.singleImageView = async (imageObj, filter = "all", comments = [], params = {}) => {
  371. const q = safeText(params.q || "");
  372. const sort = safeText(params.sort || "recent");
  373. const returnTo = safeText(params.returnTo) || buildReturnTo(filter, { q, sort });
  374. const title = safeText(imageObj.title);
  375. const ownerActions = renderImageOwnerActions(filter, imageObj, { q, sort });
  376. const topbar = div(
  377. { class: "bookmark-topbar" },
  378. div(
  379. { class: "bookmark-topbar-left" },
  380. renderImageFavoriteToggle(imageObj, returnTo),
  381. renderPMButton(imageObj.author)
  382. ),
  383. ownerActions.length ? div({ class: "bookmark-actions" }, ...ownerActions) : null
  384. );
  385. return template(
  386. i18n.imageTitle,
  387. section(
  388. div(
  389. { class: "filters" },
  390. form(
  391. { method: "GET", action: "/images", class: "ui-toolbar ui-toolbar--filters" },
  392. input({ type: "hidden", name: "q", value: q }),
  393. input({ type: "hidden", name: "sort", value: sort }),
  394. button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterAll),
  395. button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMine),
  396. button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterRecent),
  397. button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterTop),
  398. button(
  399. { type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" },
  400. i18n.imageFilterFavorites
  401. ),
  402. button(
  403. { type: "submit", name: "filter", value: "gallery", class: filter === "gallery" ? "filter-btn active" : "filter-btn" },
  404. i18n.imageFilterGallery
  405. ),
  406. button({ type: "submit", name: "filter", value: "meme", class: filter === "meme" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMeme),
  407. button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.imageCreateButton)
  408. )
  409. ),
  410. div(
  411. { class: "bookmark-item card" },
  412. topbar,
  413. title ? h2(title) : null,
  414. imageObj?.url
  415. ? div(
  416. { class: "image-container", style: "display:flex;gap:8px;align-items:center;flex-wrap:wrap;" },
  417. img({
  418. src: `/blob/${encodeURIComponent(imageObj.url)}`,
  419. alt: imageObj.title || "",
  420. class: "media-preview",
  421. loading: "lazy"
  422. })
  423. )
  424. : p(i18n.imageNoFile),
  425. safeText(imageObj.description) ? p(...renderUrl(imageObj.description)) : null,
  426. renderTags(imageObj.tags),
  427. br(),
  428. (() => {
  429. const createdTs = imageObj.createdAt ? new Date(imageObj.createdAt).getTime() : NaN;
  430. const updatedTs = imageObj.updatedAt ? new Date(imageObj.updatedAt).getTime() : NaN;
  431. const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
  432. return p(
  433. { class: "card-footer" },
  434. span({ class: "date-link" }, `${moment(imageObj.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
  435. a({ href: `/author/${encodeURIComponent(imageObj.author)}`, class: "user-link" }, `${imageObj.author}`),
  436. showUpdated
  437. ? span(
  438. { class: "votations-comment-date" },
  439. ` | ${i18n.imageUpdatedAt}: ${moment(imageObj.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
  440. )
  441. : null
  442. );
  443. })(),
  444. div(
  445. { class: "voting-buttons" },
  446. opinionCategories.map((category) =>
  447. form(
  448. { method: "POST", action: `/images/opinions/${encodeURIComponent(imageObj.key)}/${category}` },
  449. input({ type: "hidden", name: "returnTo", value: returnTo }),
  450. button(
  451. { class: "vote-btn" },
  452. `${i18n[`vote${category.charAt(0).toUpperCase() + category.slice(1)}`] || category} [${
  453. imageObj.opinions?.[category] || 0
  454. }]`
  455. )
  456. )
  457. )
  458. )
  459. ),
  460. div({ id: "comments" }, renderImageCommentsSection(imageObj.key, comments, returnTo))
  461. )
  462. );
  463. };