parent
f9c4acabf8
commit
ebd7afe464
|
@ -94,6 +94,7 @@ proc getVideoFetch(tweet: Tweet; agent, token: string; retry=true): Future[Optio
|
||||||
tokenUses.inc
|
tokenUses.inc
|
||||||
|
|
||||||
proc getVideo*(tweet: Tweet; agent, token: string; force=false) {.async.} =
|
proc getVideo*(tweet: Tweet; agent, token: string; force=false) {.async.} =
|
||||||
|
let token = if token.len == 0: guestToken else: token
|
||||||
var video = getCachedVideo(tweet.id)
|
var video = getCachedVideo(tweet.id)
|
||||||
if video.isNone:
|
if video.isNone:
|
||||||
video = await getVideoFetch(tweet, agent, token)
|
video = await getVideoFetch(tweet, agent, token)
|
||||||
|
|
|
@ -49,8 +49,8 @@ proc getUserpic*(userpic: string; style=""): string =
|
||||||
proc getUserpic*(profile: Profile; style=""): string =
|
proc getUserpic*(profile: Profile; style=""): string =
|
||||||
getUserPic(profile.userpic, style)
|
getUserPic(profile.userpic, style)
|
||||||
|
|
||||||
proc getVideoEmbed*(id: int): string =
|
proc getVideoEmbed*(cfg: Config; id: int): string =
|
||||||
&"https://twitter.com/i/videos/{id}?embed_source=facebook"
|
&"https://{cfg.hostname}/i/videos/{id}"
|
||||||
|
|
||||||
proc pageTitle*(profile: Profile): string =
|
proc pageTitle*(profile: Profile): string =
|
||||||
&"{profile.fullname} (@{profile.username})"
|
&"{profile.fullname} (@{profile.username})"
|
||||||
|
|
|
@ -6,7 +6,7 @@ import jester
|
||||||
import types, config, prefs, formatters
|
import types, config, prefs, formatters
|
||||||
import views/[general, about]
|
import views/[general, about]
|
||||||
import routes/[
|
import routes/[
|
||||||
preferences, timeline, status, media, search, rss, list, unsupported]
|
preferences, timeline, status, media, search, rss, list, unsupported, embed]
|
||||||
|
|
||||||
const configPath {.strdefine.} = "./nitter.conf"
|
const configPath {.strdefine.} = "./nitter.conf"
|
||||||
let cfg = getConfig(configPath)
|
let cfg = getConfig(configPath)
|
||||||
|
@ -20,6 +20,7 @@ createListRouter(cfg)
|
||||||
createStatusRouter(cfg)
|
createStatusRouter(cfg)
|
||||||
createSearchRouter(cfg)
|
createSearchRouter(cfg)
|
||||||
createMediaRouter(cfg)
|
createMediaRouter(cfg)
|
||||||
|
createEmbedRouter(cfg)
|
||||||
createRssRouter(cfg)
|
createRssRouter(cfg)
|
||||||
|
|
||||||
settings:
|
settings:
|
||||||
|
@ -56,3 +57,4 @@ routes:
|
||||||
extend list, ""
|
extend list, ""
|
||||||
extend status, ""
|
extend status, ""
|
||||||
extend media, ""
|
extend media, ""
|
||||||
|
extend embed, ""
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
import asyncdispatch, strutils, sequtils, uri, options
|
||||||
|
|
||||||
|
import jester
|
||||||
|
|
||||||
|
import router_utils
|
||||||
|
import ".."/[api, types, agents]
|
||||||
|
import ../views/[embed]
|
||||||
|
|
||||||
|
export embed
|
||||||
|
|
||||||
|
proc createEmbedRouter*(cfg: Config) =
|
||||||
|
router embed:
|
||||||
|
get "/i/videos/tweet/@id":
|
||||||
|
let tweet = Tweet(id: @"id".parseInt, video: some Video())
|
||||||
|
await getVideo(tweet, getAgent(), "")
|
||||||
|
resp renderVideoEmbed(cfg, tweet)
|
|
@ -31,7 +31,7 @@ proc createStatusRouter*(cfg: Config) =
|
||||||
|
|
||||||
if conversation.tweet.video.isSome():
|
if conversation.tweet.video.isSome():
|
||||||
let thumb = get(conversation.tweet.video).thumb
|
let thumb = get(conversation.tweet.video).thumb
|
||||||
let vidUrl = getVideoEmbed(conversation.tweet.id)
|
let vidUrl = getVideoEmbed(cfg, conversation.tweet.id)
|
||||||
resp renderMain(html, request, cfg, title, desc, images = @[thumb],
|
resp renderMain(html, request, cfg, title, desc, images = @[thumb],
|
||||||
`type`="video", video=vidUrl)
|
`type`="video", video=vidUrl)
|
||||||
elif conversation.tweet.gif.isSome():
|
elif conversation.tweet.gif.isSome():
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
@import 'thread';
|
@import 'thread';
|
||||||
@import 'media';
|
@import 'media';
|
||||||
@import 'video';
|
@import 'video';
|
||||||
|
@import 'embed';
|
||||||
@import 'card';
|
@import 'card';
|
||||||
@import 'poll';
|
@import 'poll';
|
||||||
@import 'quote';
|
@import 'quote';
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
@import '_variables';
|
||||||
|
@import '_mixins';
|
||||||
|
|
||||||
|
.embed-video {
|
||||||
|
.gallery-video {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
position: absolute;
|
||||||
|
background-color: black;
|
||||||
|
top: 0%;
|
||||||
|
left: 0%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-container {
|
||||||
|
max-height: unset;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
import options
|
||||||
|
import karax/[karaxdsl, vdom]
|
||||||
|
|
||||||
|
import ".."/[types, formatters]
|
||||||
|
import general, tweet
|
||||||
|
|
||||||
|
const doctype = "<!DOCTYPE html>\n"
|
||||||
|
|
||||||
|
proc renderVideoEmbed*(cfg: Config; tweet: Tweet): string =
|
||||||
|
let thumb = get(tweet.video).thumb
|
||||||
|
let vidUrl = getVideoEmbed(cfg, tweet.id)
|
||||||
|
let prefs = Prefs(hlsPlayback: true)
|
||||||
|
let node = buildHtml(html(lang="en")):
|
||||||
|
renderHead(prefs, cfg, `type`="video", images = @[thumb], video=vidUrl)
|
||||||
|
|
||||||
|
tdiv(class="embed-video"):
|
||||||
|
renderVideo(get(tweet.video), prefs, "")
|
||||||
|
|
||||||
|
result = doctype & $node
|
|
@ -28,49 +28,54 @@ proc renderNavbar*(title, rss: string; req: Request): VNode =
|
||||||
icon "info-circled", title="About", href="/about"
|
icon "info-circled", title="About", href="/about"
|
||||||
iconReferer "cog", "/settings", path, title="Preferences"
|
iconReferer "cog", "/settings", path, title="Preferences"
|
||||||
|
|
||||||
|
proc renderHead*(prefs: Prefs; cfg: Config; titleText=""; desc=""; `type`="article";
|
||||||
|
video=""; images: seq[string] = @[]): VNode =
|
||||||
|
buildHtml(head):
|
||||||
|
link(rel="stylesheet", `type`="text/css", href="/css/style.css")
|
||||||
|
link(rel="stylesheet", `type`="text/css", href="/css/fontello.css")
|
||||||
|
link(rel="apple-touch-icon", sizes="180x180", href="/apple-touch-icon.png")
|
||||||
|
link(rel="icon", type="image/png", sizes="32x32", href="/favicon-32x32.png")
|
||||||
|
link(rel="icon", type="image/png", sizes="16x16", href="/favicon-16x16.png")
|
||||||
|
link(rel="manifest", href="/site.webmanifest")
|
||||||
|
link(rel="mask-icon", href="/safari-pinned-tab.svg", color="#ff6c60")
|
||||||
|
|
||||||
|
if prefs.hlsPlayback:
|
||||||
|
script(src="/js/hls.light.min.js")
|
||||||
|
script(src="/js/hlsPlayback.js")
|
||||||
|
|
||||||
|
title:
|
||||||
|
if titleText.len > 0:
|
||||||
|
text titleText & " | " & cfg.title
|
||||||
|
else:
|
||||||
|
text cfg.title
|
||||||
|
|
||||||
|
meta(name="viewport", content="width=device-width, initial-scale=1.0")
|
||||||
|
meta(property="og:type", content=`type`)
|
||||||
|
meta(property="og:title", content=titleText)
|
||||||
|
meta(property="og:description", content=stripHtml(desc))
|
||||||
|
meta(property="og:site_name", content="Nitter")
|
||||||
|
|
||||||
|
for url in images:
|
||||||
|
meta(property="og:image", content=getPicUrl(url))
|
||||||
|
|
||||||
|
if video.len > 0:
|
||||||
|
meta(property="og:video:url", content=video)
|
||||||
|
meta(property="og:video:secure_url", content=video)
|
||||||
|
meta(property="og:video:type", content="text/html")
|
||||||
|
|
||||||
proc renderMain*(body: VNode; req: Request; cfg: Config; titleText=""; desc="";
|
proc renderMain*(body: VNode; req: Request; cfg: Config; titleText=""; desc="";
|
||||||
rss=""; `type`="article"; video=""; images: seq[string] = @[]): string =
|
rss=""; `type`="article"; video=""; images: seq[string] = @[]): string =
|
||||||
let prefs = getPrefs(req.cookies.getOrDefault("preferences"), cfg)
|
let prefs = getPrefs(req.cookies.getOrDefault("preferences"), cfg)
|
||||||
let theme = toLowerAscii(prefs.theme).replace(" ", "_")
|
let theme = toLowerAscii(prefs.theme).replace(" ", "_")
|
||||||
|
|
||||||
let node = buildHtml(html(lang="en")):
|
let node = buildHtml(html(lang="en")):
|
||||||
head:
|
renderHead(prefs, cfg, titleText, desc, `type`, video, images):
|
||||||
link(rel="stylesheet", `type`="text/css", href="/css/style.css")
|
|
||||||
link(rel="stylesheet", `type`="text/css", href="/css/fontello.css")
|
|
||||||
if theme.len > 0:
|
if theme.len > 0:
|
||||||
link(rel="stylesheet", `type`="text/css", href=(&"/css/themes/{theme}.css"))
|
link(rel="stylesheet", `type`="text/css", href=(&"/css/themes/{theme}.css"))
|
||||||
|
|
||||||
link(rel="apple-touch-icon", sizes="180x180", href="/apple-touch-icon.png")
|
|
||||||
link(rel="icon", type="image/png", sizes="32x32", href="/favicon-32x32.png")
|
|
||||||
link(rel="icon", type="image/png", sizes="16x16", href="/favicon-16x16.png")
|
|
||||||
link(rel="manifest", href="/site.webmanifest")
|
|
||||||
link(rel="mask-icon", href="/safari-pinned-tab.svg", color="#ff6c60")
|
|
||||||
|
|
||||||
if rss.len > 0:
|
if rss.len > 0:
|
||||||
link(rel="alternate", `type`="application/rss+xml", href=rss, title="RSS feed")
|
link(rel="alternate", `type`="application/rss+xml", href=rss, title="RSS feed")
|
||||||
|
|
||||||
if prefs.hlsPlayback:
|
|
||||||
script(src="/js/hls.light.min.js")
|
|
||||||
script(src="/js/hlsPlayback.js")
|
|
||||||
|
|
||||||
title:
|
|
||||||
if titleText.len > 0:
|
|
||||||
text titleText & " | " & cfg.title
|
|
||||||
else:
|
|
||||||
text cfg.title
|
|
||||||
|
|
||||||
meta(name="viewport", content="width=device-width, initial-scale=1.0")
|
|
||||||
meta(property="og:type", content=`type`)
|
|
||||||
meta(property="og:title", content=titleText)
|
|
||||||
meta(property="og:description", content=stripHtml(desc))
|
|
||||||
meta(property="og:site_name", content="Nitter")
|
|
||||||
|
|
||||||
for url in images:
|
|
||||||
meta(property="og:image", content=getPicUrl(url))
|
|
||||||
|
|
||||||
if video.len > 0:
|
|
||||||
meta(property="og:video:url", content=video)
|
|
||||||
meta(property="og:video:secure_url", content=video)
|
|
||||||
|
|
||||||
body:
|
body:
|
||||||
renderNavbar(cfg.title, rss, req)
|
renderNavbar(cfg.title, rss, req)
|
||||||
|
|
||||||
|
|
|
@ -75,7 +75,7 @@ proc renderVideoUnavailable(video: Video): VNode =
|
||||||
else:
|
else:
|
||||||
p: text "This media is unavailable"
|
p: text "This media is unavailable"
|
||||||
|
|
||||||
proc renderVideo(video: Video; prefs: Prefs; path: string): VNode =
|
proc renderVideo*(video: Video; prefs: Prefs; path: string): VNode =
|
||||||
let container =
|
let container =
|
||||||
if video.description.len > 0 or video.title.len > 0: " card-container"
|
if video.description.len > 0 or video.title.len > 0: " card-container"
|
||||||
else: ""
|
else: ""
|
||||||
|
|
Loading…
Reference in New Issue