video_view.js 18 KB

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