このリポジトリは2023-09-09にアーカイブされています。 ファイルの閲覧とクローンは可能ですが、プッシュ、イシューの作成、プルリクエストはできません。
Nitter-mod/src/views/tweet.nim

379 行
13 KiB
Nim
Raw 通常表示 履歴

2021-12-27 10:37:38 +09:00
# SPDX-License-Identifier: AGPL-3.0-only
import strutils, sequtils, strformat, options, algorithm
import karax/[karaxdsl, vdom, vstyles]
from jester import Request
import renderutils
2019-09-06 09:42:35 +09:00
import ".."/[types, utils, formatters]
2022-01-15 02:01:47 +09:00
import general
const
doctype = "<!DOCTYPE html>\n"
2020-06-07 15:26:39 +09:00
proc getSmallPic(url: string): string =
result = url
if "?" notin url and not url.endsWith("placeholder.png"):
result &= "?name=small"
2020-06-07 15:26:39 +09:00
result = getPicUrl(result)
proc renderMiniAvatar(user: User; prefs: Prefs): VNode =
let url = getPicUrl(user.getUserPic("_mini"))
2020-06-11 00:04:48 +09:00
buildHtml():
2022-01-14 11:16:09 +09:00
img(class=(prefs.getAvatarClass & " mini"), src=url)
2020-06-11 00:04:48 +09:00
proc renderHeader(tweet: Tweet; retweet: string; prefs: Prefs): VNode =
buildHtml(tdiv):
2020-06-01 09:22:22 +09:00
if retweet.len > 0:
tdiv(class="retweet-header"):
2020-06-01 09:22:22 +09:00
span: icon "retweet", retweet & " retweeted"
if tweet.pinned:
tdiv(class="pinned"):
span: icon "pin", "Pinned Tweet"
tdiv(class="tweet-header"):
a(class="tweet-avatar", href=("/" & tweet.user.username)):
2020-11-08 10:56:06 +09:00
var size = "_bigger"
if not prefs.autoplayGifs and tweet.user.userPic.endsWith("gif"):
size = "_400x400"
genImg(tweet.user.getUserPic(size), class=prefs.getAvatarClass)
2019-08-13 00:02:07 +09:00
tdiv(class="tweet-name-row"):
tdiv(class="fullname-and-username"):
linkUser(tweet.user, class="fullname")
linkUser(tweet.user, class="username")
span(class="tweet-date"):
2020-06-01 09:22:22 +09:00
a(href=getLink(tweet), title=tweet.getTime):
text tweet.getShortTime
proc renderAlbum(tweet: Tweet): VNode =
let
groups = if tweet.photos.len < 3: @[tweet.photos]
else: tweet.photos.distribute(2)
2020-06-01 09:22:22 +09:00
buildHtml(tdiv(class="attachments")):
for i, photos in groups:
let margin = if i > 0: ".25em" else: ""
tdiv(class="gallery-row", style={marginTop: margin}):
for photo in photos:
tdiv(class="attachment image"):
let
named = "name=" in photo
orig = photo
small = if named: photo else: photo & "?name=small"
a(href=getOrigPicUrl(orig), class="still-image", target="_blank"):
genImg(small)
proc isPlaybackEnabled(prefs: Prefs; playbackType: VideoType): bool =
case playbackType
2019-08-19 10:28:04 +09:00
of mp4: prefs.mp4Playback
of m3u8, vmap: prefs.hlsPlayback
proc hasMp4Url(video: Video): bool =
video.variants.anyIt(it.contentType == mp4)
2022-06-04 08:32:02 +09:00
proc renderVideoDisabled(playbackType: VideoType; path: string): VNode =
2022-01-03 11:27:04 +09:00
buildHtml(tdiv(class="video-overlay")):
2022-06-04 08:32:02 +09:00
case playbackType
2022-01-03 11:27:04 +09:00
of mp4:
p: text "mp4 playback disabled in preferences"
of m3u8, vmap:
buttonReferer "/enablehls", "Enable hls playback", path
2019-08-19 10:28:04 +09:00
2019-08-20 05:03:00 +09:00
proc renderVideoUnavailable(video: Video): VNode =
2022-01-03 11:27:04 +09:00
buildHtml(tdiv(class="video-overlay")):
case video.reason
of "dmcaed":
p: text "This media has been disabled in response to a report by the copyright owner"
else:
p: text "This media is unavailable"
2019-08-20 05:03:00 +09:00
2019-12-06 23:15:56 +09:00
proc renderVideo*(video: Video; prefs: Prefs; path: string): VNode =
2022-06-04 08:16:37 +09:00
let
container = if video.description.len == 0 and video.title.len == 0: ""
else: " card-container"
playbackType = if not prefs.proxyVideos and video.hasMp4Url: mp4
else: video.playbackType
2022-01-03 11:27:04 +09:00
buildHtml(tdiv(class="attachments card")):
tdiv(class="gallery-video" & container):
tdiv(class="attachment video-container"):
2020-06-07 15:26:39 +09:00
let thumb = getSmallPic(video.thumb)
2019-08-20 05:03:00 +09:00
if not video.available:
2022-01-03 11:27:04 +09:00
img(src=thumb)
2019-08-20 05:03:00 +09:00
renderVideoUnavailable(video)
elif not prefs.isPlaybackEnabled(playbackType):
2022-01-03 11:27:04 +09:00
img(src=thumb)
2022-06-04 08:32:02 +09:00
renderVideoDisabled(playbackType, path)
2019-08-20 05:03:00 +09:00
else:
let
vars = video.variants.filterIt(it.contentType == playbackType)
vidUrl = vars.sortedByIt(it.resolution)[^1].url
source = if prefs.proxyVideos: getVidUrl(vidUrl)
else: vidUrl
case playbackType
2019-08-19 10:28:04 +09:00
of mp4:
if prefs.muteVideos:
video(poster=thumb, controls="", muted=""):
source(src=source, `type`="video/mp4")
else:
video(poster=thumb, controls=""):
source(src=source, `type`="video/mp4")
of m3u8, vmap:
2019-08-20 03:25:00 +09:00
video(poster=thumb, data-url=source, data-autoload="false")
verbatim "<div class=\"video-overlay\" onclick=\"playVideo(this)\">"
2019-11-12 18:57:28 +09:00
tdiv(class="overlay-circle"): span(class="overlay-triangle")
verbatim "</div>"
if container.len > 0:
tdiv(class="card-content"):
h2(class="card-title"): text video.title
if video.description.len > 0:
p(class="card-description"): text video.description
2019-08-14 02:44:29 +09:00
proc renderGif(gif: Gif; prefs: Prefs): VNode =
buildHtml(tdiv(class="attachments media-gif")):
2020-05-26 21:24:41 +09:00
tdiv(class="gallery-gif", style={maxHeight: "unset"}):
tdiv(class="attachment"):
2020-06-07 15:26:39 +09:00
let thumb = getSmallPic(gif.thumb)
let url = getPicUrl(gif.url)
2019-08-14 02:44:29 +09:00
if prefs.autoplayGifs:
2020-03-29 15:05:09 +09:00
video(class="gif", poster=thumb, controls="", autoplay="", muted="", loop=""):
2019-08-14 02:44:29 +09:00
source(src=url, `type`="video/mp4")
else:
video(class="gif", poster=thumb, controls="", muted="", loop=""):
source(src=url, `type`="video/mp4")
proc renderPoll(poll: Poll): VNode =
buildHtml(tdiv(class="poll")):
for i in 0 ..< poll.options.len:
2020-05-26 21:24:41 +09:00
let
leader = if poll.leader == i: " leader" else: ""
val = poll.values[i]
perc = if val > 0: val / poll.votes * 100 else: 0
2020-05-26 21:24:41 +09:00
percStr = (&"{perc:>3.0f}").strip(chars={'.'}) & '%'
tdiv(class=("poll-meter" & leader)):
2020-05-26 21:24:41 +09:00
span(class="poll-choice-bar", style={width: percStr})
span(class="poll-choice-value"): text percStr
span(class="poll-choice-option"): text poll.options[i]
span(class="poll-info"):
2022-06-04 09:18:26 +09:00
text &"{insertSep($poll.votes, ',')} votes • {poll.status}"
2019-07-15 23:03:01 +09:00
proc renderCardImage(card: Card): VNode =
buildHtml(tdiv(class="card-image-container")):
tdiv(class="card-image"):
2020-06-01 09:22:22 +09:00
img(src=getPicUrl(card.image), alt="")
2019-07-15 23:03:01 +09:00
if card.kind == player:
tdiv(class="card-overlay"):
2019-08-20 04:27:28 +09:00
tdiv(class="overlay-circle"):
span(class="overlay-triangle")
2019-07-15 23:03:01 +09:00
2019-08-20 03:53:57 +09:00
proc renderCardContent(card: Card): VNode =
buildHtml(tdiv(class="card-content")):
h2(class="card-title"): text card.title
2020-06-10 23:13:40 +09:00
if card.text.len > 0:
p(class="card-description"): text card.text
if card.dest.len > 0:
span(class="card-destination"): text card.dest
2019-08-20 03:53:57 +09:00
proc renderCard(card: Card; prefs: Prefs; path: string): VNode =
const smallCards = {app, player, summary, storeLink}
2020-06-01 09:22:22 +09:00
let large = if card.kind notin smallCards: " large" else: ""
2021-12-27 10:27:49 +09:00
let url = replaceUrls(card.url, prefs)
2019-07-15 20:41:27 +09:00
buildHtml(tdiv(class=("card" & large))):
2019-08-20 03:53:57 +09:00
if card.video.isSome:
tdiv(class="card-container"):
renderVideo(get(card.video), prefs, path)
2019-08-20 03:53:57 +09:00
a(class="card-content-container", href=url):
renderCardContent(card)
else:
a(class="card-container", href=url):
2020-06-01 09:22:22 +09:00
if card.image.len > 0:
2019-08-20 03:53:57 +09:00
renderCardImage(card)
tdiv(class="card-content-container"):
renderCardContent(card)
2019-07-15 20:41:27 +09:00
func formatStat(stat: int): string =
if stat > 0: insertSep($stat, ',')
else: ""
2019-08-20 04:18:18 +09:00
proc renderStats(stats: TweetStats; views: string): VNode =
buildHtml(tdiv(class="tweet-stats")):
span(class="tweet-stat"): icon "comment", formatStat(stats.replies)
span(class="tweet-stat"): icon "retweet", formatStat(stats.retweets)
span(class="tweet-stat"): icon "quote", formatStat(stats.quotes)
span(class="tweet-stat"): icon "heart", formatStat(stats.likes)
2019-08-20 04:18:18 +09:00
if views.len > 0:
2020-05-26 21:24:41 +09:00
span(class="tweet-stat"): icon "play", insertSep(views, ',')
proc renderReply(tweet: Tweet): VNode =
buildHtml(tdiv(class="replying-to")):
2021-12-28 10:55:07 +09:00
text "返事者:"
for i, u in tweet.reply:
if i > 0: text " "
a(href=("/" & u)): text "@" & u
proc renderAttribution(user: User; prefs: Prefs): VNode =
buildHtml(a(class="attribution", href=("/" & user.username))):
renderMiniAvatar(user, prefs)
strong: text user.fullname
if user.verified:
icon "ok", class="verified-icon", title="Verified account"
2019-10-26 23:37:58 +09:00
proc renderMediaTags(tags: seq[User]): VNode =
2019-12-21 13:07:50 +09:00
buildHtml(tdiv(class="media-tag-block")):
icon "user"
for i, p in tags:
a(class="media-tag", href=("/" & p.username), title=p.username):
text p.fullname
if i < tags.high:
text ", "
2020-06-01 09:22:22 +09:00
proc renderQuoteMedia(quote: Tweet; prefs: Prefs; path: string): VNode =
buildHtml(tdiv(class="quote-media-container")):
2020-06-01 09:22:22 +09:00
if quote.photos.len > 0:
renderAlbum(quote)
elif quote.video.isSome:
renderVideo(quote.video.get(), prefs, path)
elif quote.gif.isSome:
renderGif(quote.gif.get(), prefs)
proc renderQuote(quote: Tweet; prefs: Prefs; path: string): VNode =
if not quote.available:
return buildHtml(tdiv(class="quote unavailable")):
tdiv(class="unavailable-quote"):
if quote.tombstone.len > 0:
text quote.tombstone
elif quote.text.len > 0:
text quote.text
else:
2021-12-28 10:55:07 +09:00
text "このツイートが見つけられません"
2020-06-01 09:22:22 +09:00
buildHtml(tdiv(class="quote quote-big")):
a(class="quote-link", href=getLink(quote))
2020-06-01 09:22:22 +09:00
tdiv(class="tweet-name-row"):
tdiv(class="fullname-and-username"):
renderMiniAvatar(quote.user, prefs)
linkUser(quote.user, class="fullname")
linkUser(quote.user, class="username")
2020-06-01 09:22:22 +09:00
span(class="tweet-date"):
a(href=getLink(quote), title=quote.getTime):
text quote.getShortTime
if quote.reply.len > 0:
renderReply(quote)
2020-06-01 09:22:22 +09:00
if quote.text.len > 0:
2020-11-08 08:06:37 +09:00
tdiv(class="quote-text", dir="auto"):
2021-12-27 10:27:49 +09:00
verbatim replaceUrls(quote.text, prefs)
if quote.hasThread:
a(class="show-thread", href=getLink(quote)):
2021-12-28 10:55:07 +09:00
text "スレットの表示"
2020-06-01 09:22:22 +09:00
if quote.photos.len > 0 or quote.video.isSome or quote.gif.isSome:
renderQuoteMedia(quote, prefs, path)
2019-12-21 13:44:58 +09:00
proc renderLocation*(tweet: Tweet): string =
let (place, url) = tweet.getLocation()
if place.len == 0: return
let node = buildHtml(span(class="tweet-geo")):
text " at "
if url.len > 1:
a(href=url): text place
else:
text place
return $node
2020-06-01 09:22:22 +09:00
proc renderTweet*(tweet: Tweet; prefs: Prefs; path: string; class=""; index=0;
last=false; showThread=false; mainTweet=false; afterTweet=false): VNode =
var divClass = class
2020-06-01 09:22:22 +09:00
if index == -1 or last:
divClass = "thread-last " & class
if not tweet.available:
2019-09-19 10:19:06 +09:00
return buildHtml(tdiv(class=divClass & "unavailable timeline-item")):
tdiv(class="unavailable-box"):
if tweet.tombstone.len > 0:
text tweet.tombstone
elif tweet.text.len > 0:
text tweet.text
2019-09-19 10:19:06 +09:00
else:
2021-12-28 10:55:07 +09:00
text "このツイートが見つけられません"
if tweet.quote.isSome:
renderQuote(tweet.quote.get(), prefs, path)
2020-06-01 09:22:22 +09:00
let fullTweet = tweet
var retweet: string
var tweet = fullTweet
if tweet.retweet.isSome:
tweet = tweet.retweet.get
retweet = fullTweet.user.fullname
2020-06-01 09:22:22 +09:00
2019-09-14 02:57:27 +09:00
buildHtml(tdiv(class=("timeline-item " & divClass))):
if not mainTweet:
a(class="tweet-link", href=getLink(tweet))
2019-09-14 02:57:27 +09:00
tdiv(class="tweet-body"):
var views = ""
renderHeader(tweet, retweet, prefs)
2019-09-14 02:57:27 +09:00
2020-06-01 09:22:22 +09:00
if not afterTweet and index == 0 and tweet.reply.len > 0 and
(tweet.reply.len > 1 or tweet.reply[0] != tweet.user.username):
2019-09-14 02:57:27 +09:00
renderReply(tweet)
var tweetClass = "tweet-content media-body"
if prefs.bidiSupport:
tweetClass &= " tweet-bidi"
tdiv(class=tweetClass, dir="auto"):
2021-12-27 10:27:49 +09:00
verbatim replaceUrls(tweet.text, prefs) & renderLocation(tweet)
2019-09-14 02:57:27 +09:00
2019-10-26 23:37:58 +09:00
if tweet.attribution.isSome:
2022-01-14 11:16:09 +09:00
renderAttribution(tweet.attribution.get(), prefs)
2019-10-26 23:37:58 +09:00
2019-09-14 02:57:27 +09:00
if tweet.card.isSome:
renderCard(tweet.card.get(), prefs, path)
2020-06-01 09:22:22 +09:00
if tweet.photos.len > 0:
2019-09-14 02:57:27 +09:00
renderAlbum(tweet)
elif tweet.video.isSome:
renderVideo(tweet.video.get(), prefs, path)
views = tweet.video.get().views
elif tweet.gif.isSome:
renderGif(tweet.gif.get(), prefs)
2020-03-29 15:05:09 +09:00
views = "GIF"
2020-06-01 09:22:22 +09:00
if tweet.poll.isSome:
2019-09-14 02:57:27 +09:00
renderPoll(tweet.poll.get())
2020-05-26 21:24:41 +09:00
if tweet.quote.isSome:
2020-06-01 09:22:22 +09:00
renderQuote(tweet.quote.get(), prefs, path)
2020-05-26 21:24:41 +09:00
if mainTweet:
2022-02-27 09:00:06 +09:00
p(class="tweet-published"): text &"{getTime(tweet)} · {tweet.source}"
2019-12-21 13:07:50 +09:00
if tweet.mediaTags.len > 0:
renderMediaTags(tweet.mediaTags)
2019-09-14 02:57:27 +09:00
if not prefs.hideTweetStats:
renderStats(tweet.stats, views)
2019-09-19 10:51:15 +09:00
if showThread:
a(class="show-thread", href=("/i/status/" & $tweet.threadId)):
2019-09-14 02:57:27 +09:00
text "Show this thread"
2022-01-15 03:11:51 +09:00
proc renderTweetEmbed*(tweet: Tweet; path: string; prefs: Prefs; cfg: Config; req: Request): string =
let node = buildHtml(html(lang="en")):
2022-01-15 03:11:51 +09:00
renderHead(prefs, cfg, req)
body:
tdiv(class="tweet-embed"):
renderTweet(tweet, prefs, path, mainTweet=true)
result = doctype & $node