2019-08-06 19:16:31 +00:00
|
|
|
import strutils, sequtils
|
2019-07-10 22:42:31 +00:00
|
|
|
import karax/[karaxdsl, vdom, vstyles]
|
|
|
|
|
|
|
|
import renderutils
|
2019-08-15 02:00:40 +00:00
|
|
|
import ../types, ../utils, ../formatters
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
proc renderHeader(tweet: Tweet): VNode =
|
|
|
|
buildHtml(tdiv):
|
|
|
|
if tweet.retweet.isSome:
|
|
|
|
tdiv(class="retweet"):
|
2019-08-15 02:00:40 +00:00
|
|
|
span: icon "retweet-1", get(tweet.retweet).by & " retweeted"
|
|
|
|
|
2019-07-10 22:42:31 +00:00
|
|
|
if tweet.pinned:
|
|
|
|
tdiv(class="pinned"):
|
2019-08-15 02:00:40 +00:00
|
|
|
span: icon "pin", "Pinned Tweet"
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
tdiv(class="tweet-header"):
|
2019-08-12 15:02:07 +00:00
|
|
|
a(class="tweet-avatar", href=("/" & tweet.profile.username)):
|
|
|
|
genImg(tweet.profile.getUserpic("_bigger"), class="avatar")
|
2019-07-10 22:42:31 +00:00
|
|
|
|
2019-08-12 15:02:07 +00:00
|
|
|
tdiv(class="tweet-name-row"):
|
2019-07-10 22:42:31 +00:00
|
|
|
tdiv(class="fullname-and-username"):
|
|
|
|
linkUser(tweet.profile, class="fullname")
|
|
|
|
linkUser(tweet.profile, class="username")
|
|
|
|
|
|
|
|
span(class="tweet-date"):
|
|
|
|
a(href=getLink(tweet), title=tweet.getTime()):
|
|
|
|
text tweet.shortTime
|
|
|
|
|
|
|
|
proc renderAlbum(tweet: Tweet): VNode =
|
|
|
|
let
|
|
|
|
groups = if tweet.photos.len < 3: @[tweet.photos]
|
|
|
|
else: tweet.photos.distribute(2)
|
|
|
|
class = if groups.len == 1 and groups[0].len == 1: "single-image"
|
|
|
|
else: ""
|
|
|
|
|
|
|
|
buildHtml(tdiv(class=("attachments " & class))):
|
|
|
|
for i, photos in groups:
|
|
|
|
let margin = if i > 0: ".25em" else: ""
|
|
|
|
let flex = if photos.len > 1 or groups.len > 1: "flex" else: "block"
|
|
|
|
tdiv(class="gallery-row", style={marginTop: margin}):
|
|
|
|
for photo in photos:
|
|
|
|
tdiv(class="attachment image"):
|
|
|
|
a(href=getSigUrl(photo & "?name=orig", "pic"), class="still-image",
|
|
|
|
target="_blank", style={display: flex}):
|
|
|
|
genImg(photo)
|
|
|
|
|
2019-08-19 01:28:04 +00:00
|
|
|
proc isPlaybackEnabled(prefs: Prefs; video: Video): bool =
|
|
|
|
case video.playbackType
|
|
|
|
of mp4: prefs.mp4Playback
|
|
|
|
of m3u8, vmap: prefs.hlsPlayback
|
|
|
|
|
|
|
|
proc renderVideoDisabled(video: Video): VNode =
|
|
|
|
buildHtml(tdiv):
|
|
|
|
img(src=video.thumb.getSigUrl("pic"))
|
|
|
|
tdiv(class="video-overlay"):
|
|
|
|
case video.playbackType
|
|
|
|
of mp4:
|
|
|
|
p: text "mp4 playback disabled in preferences"
|
|
|
|
of m3u8, vmap:
|
|
|
|
p: text "hls playback disabled in preferences"
|
|
|
|
|
2019-08-13 17:44:29 +00:00
|
|
|
proc renderVideo(video: Video; prefs: Prefs): VNode =
|
2019-07-10 22:42:31 +00:00
|
|
|
buildHtml(tdiv(class="attachments")):
|
|
|
|
tdiv(class="gallery-video"):
|
|
|
|
tdiv(class="attachment video-container"):
|
2019-08-19 01:28:04 +00:00
|
|
|
if prefs.isPlaybackEnabled(video):
|
|
|
|
let thumb = video.thumb.getSigUrl("pic")
|
|
|
|
let source = video.url.getSigUrl("video")
|
|
|
|
case video.playbackType
|
|
|
|
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-19 18:25:00 +00:00
|
|
|
video(poster=thumb, data-url=source, data-autoload="false")
|
|
|
|
verbatim "<div class=\"video-overlay\" onclick=\"playVideo(this)\">"
|
|
|
|
verbatim "<div class=\"card-overlay-circle\">"
|
|
|
|
verbatim "<span class=\"card-overlay-triangle\"</span></div></div>"
|
2019-08-19 01:28:04 +00:00
|
|
|
else:
|
|
|
|
renderVideoDisabled(video)
|
2019-08-13 17:44:29 +00:00
|
|
|
|
|
|
|
proc renderGif(gif: Gif; prefs: Prefs): VNode =
|
2019-07-10 22:42:31 +00:00
|
|
|
buildHtml(tdiv(class="attachments media-gif")):
|
|
|
|
tdiv(class="gallery-gif", style=style(maxHeight, "unset")):
|
|
|
|
tdiv(class="attachment"):
|
2019-08-13 17:44:29 +00:00
|
|
|
let thumb = gif.thumb.getSigUrl("pic")
|
|
|
|
let url = gif.url.getSigUrl("video")
|
|
|
|
if prefs.autoplayGifs:
|
|
|
|
video(class="gif", poster=thumb, autoplay="", muted="", loop=""):
|
|
|
|
source(src=url, `type`="video/mp4")
|
|
|
|
else:
|
|
|
|
video(class="gif", poster=thumb, controls="", muted="", loop=""):
|
|
|
|
source(src=url, `type`="video/mp4")
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
proc renderPoll(poll: Poll): VNode =
|
|
|
|
buildHtml(tdiv(class="poll")):
|
|
|
|
for i in 0 ..< poll.options.len:
|
|
|
|
let leader = if poll.leader == i: " leader" else: ""
|
|
|
|
let perc = $poll.values[i] & "%"
|
|
|
|
tdiv(class=("poll-meter" & leader)):
|
|
|
|
span(class="poll-choice-bar", style=style(width, perc))
|
|
|
|
span(class="poll-choice-value"): text perc
|
|
|
|
span(class="poll-choice-option"): text poll.options[i]
|
|
|
|
span(class="poll-info"):
|
|
|
|
text $poll.votes & " votes • " & poll.status
|
|
|
|
|
2019-07-15 14:03:01 +00:00
|
|
|
proc renderCardImage(card: Card): VNode =
|
|
|
|
buildHtml(tdiv(class="card-image-container")):
|
|
|
|
tdiv(class="card-image"):
|
2019-08-15 13:51:20 +00:00
|
|
|
img(src=getSigUrl(get(card.image), "pic"))
|
2019-07-15 14:03:01 +00:00
|
|
|
if card.kind == player:
|
|
|
|
tdiv(class="card-overlay"):
|
|
|
|
tdiv(class="card-overlay-circle"):
|
|
|
|
span(class="card-overlay-triangle")
|
|
|
|
|
2019-08-19 18:53:57 +00:00
|
|
|
proc renderCardContent(card: Card): VNode =
|
|
|
|
buildHtml(tdiv(class="card-content")):
|
|
|
|
h2(class="card-title"): text card.title
|
|
|
|
p(class="card-description"): text card.text
|
|
|
|
span(class="card-destination"): text card.dest
|
|
|
|
|
2019-08-13 17:44:29 +00:00
|
|
|
proc renderCard(card: Card; prefs: Prefs): VNode =
|
2019-07-15 14:03:01 +00:00
|
|
|
const largeCards = {summaryLarge, liveEvent, promoWebsite, promoVideo}
|
2019-07-15 11:41:27 +00:00
|
|
|
let large = if card.kind in largeCards: " large" else: ""
|
2019-08-19 18:53:57 +00:00
|
|
|
let url = replaceUrl(card.url, prefs)
|
2019-07-15 11:41:27 +00:00
|
|
|
|
|
|
|
buildHtml(tdiv(class=("card" & large))):
|
2019-08-19 18:53:57 +00:00
|
|
|
if card.video.isSome:
|
|
|
|
tdiv(class="card-container"):
|
2019-08-13 17:44:29 +00:00
|
|
|
renderVideo(get(card.video), prefs)
|
2019-08-19 18:53:57 +00:00
|
|
|
a(class="card-content-container", href=url):
|
|
|
|
renderCardContent(card)
|
|
|
|
else:
|
|
|
|
a(class="card-container", href=url):
|
|
|
|
if card.image.isSome:
|
|
|
|
renderCardImage(card)
|
|
|
|
tdiv(class="card-content-container"):
|
|
|
|
renderCardContent(card)
|
2019-07-15 11:41:27 +00:00
|
|
|
|
2019-07-10 22:42:31 +00:00
|
|
|
proc renderStats(stats: TweetStats): VNode =
|
|
|
|
buildHtml(tdiv(class="tweet-stats")):
|
2019-08-15 02:00:40 +00:00
|
|
|
span(class="tweet-stat"): icon "comment", $stats.replies
|
|
|
|
span(class="tweet-stat"): icon "retweet-1", $stats.retweets
|
|
|
|
span(class="tweet-stat"): icon "thumbs-up-alt", $stats.likes
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
proc renderReply(tweet: Tweet): VNode =
|
|
|
|
buildHtml(tdiv(class="replying-to")):
|
|
|
|
text "Replying to "
|
|
|
|
for i, u in tweet.reply:
|
|
|
|
if i > 0: text " "
|
|
|
|
a(href=("/" & u)): text "@" & u
|
|
|
|
|
|
|
|
proc renderReply(quote: Quote): VNode =
|
|
|
|
buildHtml(tdiv(class="replying-to")):
|
|
|
|
text "Replying to "
|
|
|
|
for i, u in quote.reply:
|
|
|
|
if i > 0: text " "
|
|
|
|
a(href=("/" & u)): text "@" & u
|
|
|
|
|
|
|
|
proc renderQuoteMedia(quote: Quote): VNode =
|
|
|
|
buildHtml(tdiv(class="quote-media-container")):
|
|
|
|
if quote.thumb.len > 0:
|
|
|
|
tdiv(class="quote-media"):
|
|
|
|
genImg(quote.thumb)
|
|
|
|
if quote.badge.len > 0:
|
|
|
|
tdiv(class="quote-badge"):
|
|
|
|
tdiv(class="quote-badge-text"): text quote.badge
|
|
|
|
elif quote.sensitive:
|
|
|
|
tdiv(class="quote-sensitive"):
|
2019-08-15 02:00:40 +00:00
|
|
|
icon "attention", class="quote-sensitive-icon"
|
2019-07-10 22:42:31 +00:00
|
|
|
|
2019-08-15 13:51:20 +00:00
|
|
|
proc renderQuote(quote: Quote; prefs: Prefs): VNode =
|
2019-07-10 22:42:31 +00:00
|
|
|
if not quote.available:
|
|
|
|
return buildHtml(tdiv(class="quote unavailable")):
|
|
|
|
tdiv(class="unavailable-quote"):
|
|
|
|
text "This tweet is unavailable"
|
|
|
|
|
|
|
|
buildHtml(tdiv(class="quote")):
|
|
|
|
a(class="quote-link", href=getLink(quote))
|
|
|
|
|
|
|
|
if quote.thumb.len > 0 or quote.sensitive:
|
|
|
|
renderQuoteMedia(quote)
|
|
|
|
|
|
|
|
tdiv(class="fullname-and-username"):
|
|
|
|
linkUser(quote.profile, class="fullname")
|
|
|
|
linkUser(quote.profile, class="username")
|
|
|
|
|
|
|
|
if quote.reply.len > 0:
|
|
|
|
renderReply(quote)
|
|
|
|
|
|
|
|
tdiv(class="quote-text"):
|
2019-08-15 13:51:20 +00:00
|
|
|
verbatim linkifyText(quote.text, prefs)
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
if quote.hasThread:
|
2019-08-13 19:21:54 +00:00
|
|
|
a(class="show-thread", href=getLink(quote)):
|
2019-07-10 22:42:31 +00:00
|
|
|
text "Show this thread"
|
|
|
|
|
2019-08-13 17:44:29 +00:00
|
|
|
proc renderTweet*(tweet: Tweet; prefs: Prefs; class="";
|
|
|
|
index=0; total=(-1); last=false): VNode =
|
2019-07-10 22:42:31 +00:00
|
|
|
var divClass = class
|
|
|
|
if index == total or last:
|
|
|
|
divClass = "thread-last " & class
|
|
|
|
|
|
|
|
if not tweet.available:
|
|
|
|
return buildHtml(tdiv(class=divClass)):
|
|
|
|
tdiv(class="status-el unavailable"):
|
|
|
|
tdiv(class="unavailable-box"):
|
|
|
|
text "This tweet is unavailable"
|
|
|
|
|
|
|
|
buildHtml(tdiv(class=divClass)):
|
|
|
|
tdiv(class="status-el"):
|
|
|
|
tdiv(class="status-body"):
|
|
|
|
renderHeader(tweet)
|
|
|
|
|
|
|
|
if index == 0 and tweet.reply.len > 0:
|
|
|
|
renderReply(tweet)
|
|
|
|
|
|
|
|
tdiv(class="status-content media-body"):
|
2019-08-15 13:51:20 +00:00
|
|
|
verbatim linkifyText(tweet.text, prefs)
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
if tweet.quote.isSome:
|
2019-08-15 13:51:20 +00:00
|
|
|
renderQuote(tweet.quote.get(), prefs)
|
2019-07-10 22:42:31 +00:00
|
|
|
|
2019-07-15 14:03:01 +00:00
|
|
|
if tweet.card.isSome:
|
2019-08-13 17:44:29 +00:00
|
|
|
renderCard(tweet.card.get(), prefs)
|
2019-07-15 14:03:01 +00:00
|
|
|
elif tweet.photos.len > 0:
|
2019-07-10 22:42:31 +00:00
|
|
|
renderAlbum(tweet)
|
|
|
|
elif tweet.video.isSome:
|
2019-08-13 17:44:29 +00:00
|
|
|
renderVideo(tweet.video.get(), prefs)
|
2019-07-10 22:42:31 +00:00
|
|
|
elif tweet.gif.isSome:
|
2019-08-13 17:44:29 +00:00
|
|
|
renderGif(tweet.gif.get(), prefs)
|
2019-07-10 22:42:31 +00:00
|
|
|
elif tweet.poll.isSome:
|
|
|
|
renderPoll(tweet.poll.get())
|
|
|
|
|
2019-08-13 19:25:29 +00:00
|
|
|
if not prefs.hideTweetStats:
|
|
|
|
renderStats(tweet.stats)
|
2019-07-10 22:42:31 +00:00
|
|
|
|
|
|
|
if tweet.hasThread and "timeline" in class:
|
2019-08-13 19:21:54 +00:00
|
|
|
a(class="show-thread", href=getLink(tweet)):
|
2019-07-10 22:42:31 +00:00
|
|
|
text "Show this thread"
|