| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 |
- const { form, button, div, h2, p, section, input, label, br, a, img, span, textarea, select, option } =
- 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 opinionCategories = require("../backend/opinion_categories");
- const userId = config.keys.id;
- const safeArr = (v) => (Array.isArray(v) ? v : []);
- const safeText = (v) => String(v || "").trim();
- const buildReturnTo = (filter, params = {}) => {
- const f = safeText(filter || "all");
- const q = safeText(params.q || "");
- const sort = safeText(params.sort || "recent");
- const parts = [`filter=${encodeURIComponent(f)}`];
- if (q) parts.push(`q=${encodeURIComponent(q)}`);
- if (sort) parts.push(`sort=${encodeURIComponent(sort)}`);
- return `/images?${parts.join("&")}`;
- };
- const renderPMButton = (recipient, className = "filter-btn") => {
- const r = safeText(recipient);
- if (!r) return null;
- if (String(r) === String(userId)) return null;
- return form(
- { method: "GET", action: "/pm" },
- input({ type: "hidden", name: "recipients", value: r }),
- button({ type: "submit", class: className }, i18n.privateMessage)
- );
- };
- const renderTags = (tags) => {
- const list = safeArr(tags).map((t) => String(t || "").trim()).filter(Boolean);
- return list.length
- ? div(
- { class: "card-tags" },
- list.map((tag) => a({ href: `/search?query=%23${encodeURIComponent(tag)}`, class: "tag-link" }, `#${tag}`))
- )
- : null;
- };
- const renderImageFavoriteToggle = (imgObj, returnTo = "") =>
- form(
- {
- method: "POST",
- action: imgObj.isFavorite
- ? `/images/favorites/remove/${encodeURIComponent(imgObj.key)}`
- : `/images/favorites/add/${encodeURIComponent(imgObj.key)}`
- },
- returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
- button(
- { type: "submit", class: "filter-btn" },
- imgObj.isFavorite ? i18n.imageRemoveFavoriteButton : i18n.imageAddFavoriteButton
- )
- );
- const renderImageMedia = (imgObj, filter, params = {}) => {
- const src = imgObj?.url ? `/blob/${encodeURIComponent(imgObj.url)}` : "";
- return imgObj?.url
- ? div(
- { class: "image-container", style: "display:flex;gap:8px;align-items:center;flex-wrap:wrap;" },
- a(
- {
- href: `/images/${encodeURIComponent(imgObj.key)}?filter=${encodeURIComponent(filter || "all")}${
- params.q ? `&q=${encodeURIComponent(params.q)}` : ""
- }${params.sort ? `&sort=${encodeURIComponent(params.sort)}` : ""}`
- },
- img({ src, alt: imgObj.title || "", class: "media-preview", loading: "lazy" })
- )
- )
- : p(i18n.imageNoFile);
- };
- const renderImageOwnerActions = (filter, imgObj, params = {}) => {
- const returnTo = buildReturnTo(filter, params);
- const isAuthor = String(imgObj.author) === String(userId);
- const hasOpinions = Object.keys(imgObj.opinions || {}).length > 0;
- if (!isAuthor) return [];
- const items = [];
- if (!hasOpinions) {
- items.push(
- form(
- { method: "GET", action: `/images/edit/${encodeURIComponent(imgObj.key)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button({ class: "update-btn", type: "submit" }, i18n.imageUpdateButton)
- )
- );
- }
- items.push(
- form(
- { method: "POST", action: `/images/delete/${encodeURIComponent(imgObj.key)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button({ class: "delete-btn", type: "submit" }, i18n.imageDeleteButton)
- )
- );
- return items;
- };
- const renderImageList = (images, filter, params = {}) => {
- const returnTo = buildReturnTo(filter, params);
- return images.length
- ? images.map((imgObj) => {
- const commentCount = typeof imgObj.commentCount === "number" ? imgObj.commentCount : 0;
- const title = safeText(imgObj.title);
- const ownerActions = renderImageOwnerActions(filter, imgObj, params);
- return div(
- { class: "tags-header image-card" },
- div(
- { class: "bookmark-topbar" },
- div(
- { class: "bookmark-topbar-left" },
- form(
- { method: "GET", action: `/images/${encodeURIComponent(imgObj.key)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- input({ type: "hidden", name: "filter", value: filter || "all" }),
- params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
- params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
- button({ type: "submit", class: "filter-btn" }, i18n.viewDetails)
- ),
- renderImageFavoriteToggle(imgObj, returnTo),
- renderPMButton(imgObj.author)
- ),
- ownerActions.length ? div({ class: "bookmark-actions" }, ...ownerActions) : null
- ),
- title ? h2(title) : null,
- renderImageMedia(imgObj, filter, params),
- div(
- { class: "card-comments-summary" },
- span({ class: "card-label" }, i18n.voteCommentsLabel + ":"),
- span({ class: "card-value" }, String(commentCount)),
- br(),
- br(),
- form(
- { method: "GET", action: `/images/${encodeURIComponent(imgObj.key)}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- input({ type: "hidden", name: "filter", value: filter || "all" }),
- params.q ? input({ type: "hidden", name: "q", value: params.q }) : null,
- params.sort ? input({ type: "hidden", name: "sort", value: params.sort }) : null,
- button({ type: "submit", class: "filter-btn" }, i18n.voteCommentsForumButton)
- )
- ),
- br(),
- (() => {
- const createdTs = imgObj.createdAt ? new Date(imgObj.createdAt).getTime() : NaN;
- const updatedTs = imgObj.updatedAt ? new Date(imgObj.updatedAt).getTime() : NaN;
- const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
- return p(
- { class: "card-footer" },
- span({ class: "date-link" }, `${moment(imgObj.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
- a({ href: `/author/${encodeURIComponent(imgObj.author)}`, class: "user-link" }, `${imgObj.author}`),
- showUpdated
- ? span(
- { class: "votations-comment-date" },
- ` | ${i18n.imageUpdatedAt}: ${moment(imgObj.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
- )
- : null
- );
- })()
- );
- })
- : p(params.q ? i18n.imageNoMatch : i18n.noImages);
- };
- const renderImageForm = (filter, imageId, imageToEdit, params = {}) => {
- const returnFilter = filter === "create" ? "all" : params.filter || "all";
- const returnTo = safeText(params.returnTo) || buildReturnTo(returnFilter, params);
- const tagsValue = safeArr(imageToEdit?.tags).join(", ");
- return div(
- { class: "div-center image-form" },
- form(
- {
- action: filter === "edit" ? `/images/update/${encodeURIComponent(imageId)}` : "/images/create",
- method: "POST",
- enctype: "multipart/form-data"
- },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- label(i18n.imageFileLabel),
- br(),
- input({ type: "file", name: "image", required: filter !== "edit" }),
- br(),
- br(),
- imageToEdit?.url
- ? img({ src: `/blob/${encodeURIComponent(imageToEdit.url)}`, class: "media-preview", alt: imageToEdit?.title || "" })
- : null,
- label(i18n.imageTitleLabel),
- br(),
- input({ type: "text", name: "title", placeholder: i18n.imageTitlePlaceholder, value: imageToEdit?.title || "" }),
- br(),
- label(i18n.imageDescriptionLabel),
- br(),
- textarea({ name: "description", placeholder: i18n.imageDescriptionPlaceholder, rows: "4" }, imageToEdit?.description || ""),
- br(),
- input({ type: "hidden", name: "meme", value: "0" }),
- label(i18n.imageTagsLabel),
- br(),
- input({ type: "text", name: "tags", placeholder: i18n.imageTagsPlaceholder, value: tagsValue }),
- br(),
- label(i18n.imageMemeLabel),
- input({
- id: "meme-checkbox",
- type: "checkbox",
- name: "meme",
- value: "1",
- class: "meme-checkbox",
- ...(imageToEdit?.meme ? { checked: true } : {})
- }),
- br(),
- br(),
- button({ type: "submit" }, filter === "edit" ? i18n.imageUpdateButton : i18n.imageCreateButton)
- )
- );
- };
- const renderGallery = (images) => {
- if (!images.length) return div(i18n.noImages);
- return div(
- { class: "gallery" },
- images.map((imgObj) => {
- const src = imgObj.url ? `/image/256/${encodeURIComponent(imgObj.url)}` : "";
- return a(
- { href: `#img-${encodeURIComponent(imgObj.key)}`, class: "gallery-item" },
- img({ src, alt: imgObj.title || "", class: "gallery-image", loading: "lazy" })
- );
- })
- );
- };
- const renderLightbox = (images) =>
- images.map((imgObj) => {
- const src = imgObj.url ? `/blob/${encodeURIComponent(imgObj.url)}` : "";
- return div(
- { id: `img-${encodeURIComponent(imgObj.key)}`, class: "lightbox" },
- a({ href: "#", class: "lightbox-close" }, "×"),
- img({ src, class: "lightbox-image", alt: imgObj.title || "" })
- );
- });
- const renderImageCommentsSection = (imageKey, comments = [], returnTo = null) => {
- const list = safeArr(comments);
- const commentsCount = list.length;
- 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: `/images/${encodeURIComponent(imageKey)}/comments`, class: "comment-form", enctype: "multipart/form-data" },
- returnTo ? input({ type: "hidden", name: "returnTo", value: returnTo }) : null,
- 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)
- )
- ),
- list.length
- ? div(
- { class: "comments-list" },
- list.map((c) => {
- const author = c?.value?.author || "";
- const ts = 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?.content || {};
- const text = content.text || "";
- const threadRoot = content.fork || content.root || null;
- 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 && threadRoot ? a({ href: `/thread/${encodeURIComponent(threadRoot)}#${encodeURIComponent(c.key)}` }, relDate) : ""
- ),
- p({ class: "votations-comment-text" }, ...renderUrl(text))
- );
- })
- )
- : p({ class: "votations-no-comments" }, i18n.voteNoCommentsYet)
- );
- };
- exports.imageView = async (images, filter = "all", imageId = null, params = {}) => {
- const title =
- filter === "mine"
- ? i18n.imageMineSectionTitle
- : filter === "create"
- ? i18n.imageCreateSectionTitle
- : filter === "edit"
- ? i18n.imageUpdateSectionTitle
- : filter === "gallery"
- ? i18n.imageGallerySectionTitle
- : filter === "meme"
- ? i18n.imageMemeSectionTitle
- : filter === "recent"
- ? i18n.imageRecentSectionTitle
- : filter === "top"
- ? i18n.imageTopSectionTitle
- : filter === "favorites"
- ? i18n.imageFavoritesSectionTitle
- : i18n.imageAllSectionTitle;
- const q = safeText(params.q || "");
- const sort = safeText(params.sort || "recent");
- const list = safeArr(images);
- const imageToEdit = imageId ? list.find((im) => im.key === imageId) : null;
- return template(
- title,
- section(
- div({ class: "tags-header" }, h2(title), p(i18n.imageDescription)),
- div(
- { class: "filters" },
- form(
- { method: "GET", action: "/images", class: "ui-toolbar ui-toolbar--filters" },
- input({ type: "hidden", name: "q", value: q }),
- input({ type: "hidden", name: "sort", value: sort }),
- button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterAll),
- button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMine),
- button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterRecent),
- button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterTop),
- button(
- { type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" },
- i18n.imageFilterFavorites
- ),
- button(
- { type: "submit", name: "filter", value: "gallery", class: filter === "gallery" ? "filter-btn active" : "filter-btn" },
- i18n.imageFilterGallery
- ),
- button({ type: "submit", name: "filter", value: "meme", class: filter === "meme" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMeme),
- button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.imageCreateButton)
- )
- )
- ),
- section(
- filter === "create" || filter === "edit"
- ? renderImageForm(filter, imageId, imageToEdit, { ...params, filter })
- : section(
- div(
- { class: "images-search" },
- form(
- { method: "GET", action: "/images", class: "filter-box" },
- input({ type: "hidden", name: "filter", value: filter }),
- input({
- type: "text",
- name: "q",
- value: q,
- placeholder: i18n.imageSearchPlaceholder,
- class: "filter-box__input"
- }),
- div(
- { class: "filter-box__controls" },
- select(
- { name: "sort", class: "filter-box__select" },
- option({ value: "recent", selected: sort === "recent" }, i18n.imageSortRecent),
- option({ value: "oldest", selected: sort === "oldest" }, i18n.imageSortOldest),
- option({ value: "top", selected: sort === "top" }, i18n.imageSortTop)
- ),
- button({ type: "submit", class: "filter-box__button" }, i18n.imageSearchButton)
- )
- )
- ),
- filter === "gallery" ? renderGallery(list) : div({ class: "images-list" }, renderImageList(list, filter, { q, sort }))
- )
- ),
- ...(filter === "gallery" ? renderLightbox(list) : [])
- );
- };
- exports.singleImageView = async (imageObj, filter = "all", comments = [], params = {}) => {
- const q = safeText(params.q || "");
- const sort = safeText(params.sort || "recent");
- const returnTo = safeText(params.returnTo) || buildReturnTo(filter, { q, sort });
- const title = safeText(imageObj.title);
- const ownerActions = renderImageOwnerActions(filter, imageObj, { q, sort });
- const topbar = div(
- { class: "bookmark-topbar" },
- div(
- { class: "bookmark-topbar-left" },
- renderImageFavoriteToggle(imageObj, returnTo),
- renderPMButton(imageObj.author)
- ),
- ownerActions.length ? div({ class: "bookmark-actions" }, ...ownerActions) : null
- );
- return template(
- i18n.imageTitle,
- section(
- div(
- { class: "filters" },
- form(
- { method: "GET", action: "/images", class: "ui-toolbar ui-toolbar--filters" },
- input({ type: "hidden", name: "q", value: q }),
- input({ type: "hidden", name: "sort", value: sort }),
- button({ type: "submit", name: "filter", value: "all", class: filter === "all" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterAll),
- button({ type: "submit", name: "filter", value: "mine", class: filter === "mine" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMine),
- button({ type: "submit", name: "filter", value: "recent", class: filter === "recent" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterRecent),
- button({ type: "submit", name: "filter", value: "top", class: filter === "top" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterTop),
- button(
- { type: "submit", name: "filter", value: "favorites", class: filter === "favorites" ? "filter-btn active" : "filter-btn" },
- i18n.imageFilterFavorites
- ),
- button(
- { type: "submit", name: "filter", value: "gallery", class: filter === "gallery" ? "filter-btn active" : "filter-btn" },
- i18n.imageFilterGallery
- ),
- button({ type: "submit", name: "filter", value: "meme", class: filter === "meme" ? "filter-btn active" : "filter-btn" }, i18n.imageFilterMeme),
- button({ type: "submit", name: "filter", value: "create", class: "create-button" }, i18n.imageCreateButton)
- )
- ),
- div(
- { class: "bookmark-item card" },
- topbar,
- title ? h2(title) : null,
- imageObj?.url
- ? div(
- { class: "image-container", style: "display:flex;gap:8px;align-items:center;flex-wrap:wrap;" },
- img({
- src: `/blob/${encodeURIComponent(imageObj.url)}`,
- alt: imageObj.title || "",
- class: "media-preview",
- loading: "lazy"
- })
- )
- : p(i18n.imageNoFile),
- safeText(imageObj.description) ? p(...renderUrl(imageObj.description)) : null,
- renderTags(imageObj.tags),
- br(),
- (() => {
- const createdTs = imageObj.createdAt ? new Date(imageObj.createdAt).getTime() : NaN;
- const updatedTs = imageObj.updatedAt ? new Date(imageObj.updatedAt).getTime() : NaN;
- const showUpdated = Number.isFinite(updatedTs) && (!Number.isFinite(createdTs) || updatedTs !== createdTs);
- return p(
- { class: "card-footer" },
- span({ class: "date-link" }, `${moment(imageObj.createdAt).format("YYYY/MM/DD HH:mm:ss")} ${i18n.performed} `),
- a({ href: `/author/${encodeURIComponent(imageObj.author)}`, class: "user-link" }, `${imageObj.author}`),
- showUpdated
- ? span(
- { class: "votations-comment-date" },
- ` | ${i18n.imageUpdatedAt}: ${moment(imageObj.updatedAt).format("YYYY/MM/DD HH:mm:ss")}`
- )
- : null
- );
- })(),
- div(
- { class: "voting-buttons" },
- opinionCategories.map((category) =>
- form(
- { method: "POST", action: `/images/opinions/${encodeURIComponent(imageObj.key)}/${category}` },
- input({ type: "hidden", name: "returnTo", value: returnTo }),
- button(
- { class: "vote-btn" },
- `${i18n[`vote${category.charAt(0).toUpperCase() + category.slice(1)}`] || category} [${
- imageObj.opinions?.[category] || 0
- }]`
- )
- )
- )
- )
- ),
- div({ id: "comments" }, renderImageCommentsSection(imageObj.key, comments, returnTo))
- )
- );
- };
|