From ec5f67123ad0361dcc09b36487c8a220de432ef9 Mon Sep 17 00:00:00 2001 From: Zed Date: Wed, 7 Aug 2019 22:02:19 +0200 Subject: [PATCH] Implement link previews --- README.md | 1 - public/style.css | 2 +- src/formatters.nim | 6 ++++++ src/nitter.nim | 14 ++++++++++++-- src/views/general.nim | 30 +++++++++++++++++++++++++----- 5 files changed, 44 insertions(+), 9 deletions(-) diff --git a/README.md b/README.md index 7612d6d..8c3ad5d 100644 --- a/README.md +++ b/README.md @@ -27,7 +27,6 @@ is on implementing missing features. - Search (images/videos, hashtags, etc.) - Custom timeline filter -- Nitter link previews - More caching (waiting for [moigagoo/norm#19](https://github.com/moigagoo/norm/pull/19)) - Simple account system with customizable feed - Video support with hls.js diff --git a/public/style.css b/public/style.css index 8845422..d4b2dc6 100644 --- a/public/style.css +++ b/public/style.css @@ -140,7 +140,7 @@ a:hover { .replying-to { color: hsla(240,1%,73%,.9); - margin: -4px 0 4px 0; + margin: -2px 0 4px 0; } .status-el .status-content { diff --git a/src/formatters.nim b/src/formatters.nim index e5ffc76..5e5d134 100644 --- a/src/formatters.nim +++ b/src/formatters.nim @@ -68,9 +68,15 @@ proc getUserpic*(userpic: string; style=""): string = proc getUserpic*(profile: Profile; style=""): string = getUserPic(profile.userpic, style) +proc getVideoEmbed*(video: Video): string = + &"https://twitter.com/i/videos/{video.videoId}?embed_source=facebook" + proc pageTitle*(profile: Profile): string = &"{profile.fullname} (@{profile.username})" +proc pageDesc*(profile: Profile): string = + "The latest tweets from " & profile.fullname + proc getTime*(tweet: Tweet): string = tweet.time.format("d/M/yyyy', ' HH:mm:ss") diff --git a/src/nitter.nim b/src/nitter.nim index e541baa..19b00a7 100644 --- a/src/nitter.nim +++ b/src/nitter.nim @@ -24,7 +24,7 @@ proc showSingleTimeline(name, after, agent: string; query: Option[Query]): Futur return "" let profileHtml = renderProfile(profile, await timelineFut, await railFut) - return renderMain(profileHtml, title=cfg.title, titleText=pageTitle(profile)) + return renderMain(profileHtml, title=cfg.title, titleText=pageTitle(profile), desc=pageDesc(profile)) proc showMultiTimeline(names: seq[string]; after, agent: string; query: Option[Query]): Future[string] {.async.} = var q = query @@ -91,7 +91,17 @@ routes: resp Http404, showError("Tweet not found", cfg.title) let title = pageTitle(conversation.tweet.profile) - resp renderMain(renderConversation(conversation), title=cfg.title, titleText=title) + let desc = conversation.tweet.text + let html = renderConversation(conversation) + + if conversation.tweet.video.isSome(): + let thumb = get(conversation.tweet.video).thumb + let vidUrl = getVideoEmbed(get(conversation.tweet.video)) + resp renderMain(html, title=cfg.title, titleText=title, desc=desc, + images = @[thumb], `type`="video", video=vidUrl) + else: + resp renderMain(html, title=cfg.title, titleText=title, + desc=desc, images=conversation.tweet.photos) get "/pic/@sig/@url": cond "http" in @"url" diff --git a/src/views/general.nim b/src/views/general.nim index 622514e..c86dc34 100644 --- a/src/views/general.nim +++ b/src/views/general.nim @@ -1,16 +1,36 @@ import karax/[karaxdsl, vdom] +import ../utils + const doctype = "\n" -proc renderMain*(body: VNode; title="Nitter"; titleText=""): string = +proc renderMain*(body: VNode; title="Nitter"; titleText=""; desc=""; + `type`="article"; video=""; images: seq[string] = @[]): string = let node = buildHtml(html(lang="en")): head: - if titleText.len > 0: - title: text titleText & " | " & title - else: - title: text title link(rel="stylesheet", `type`="text/css", href="/style.css") + title: + if titleText.len > 0: + text titleText & " | " & title + else: + text title + + meta(name="og:type", content=`type`) + meta(name="og:title", content=titleText) + meta(name="og:description", content=desc) + meta(name="og:site_name", content="Twitter") + + for url in images: + meta(name="og:image", content=getSigUrl(url, "pic")) + + if video.len > 0: + meta(name="og:video:url", content=video) + meta(name="og:video:secure_url", content=video) + meta(name="og:video:type", content="text/html") + meta(name="og:video:width", content="1200") + meta(name="og:video:height", content="675") + body: nav(id="nav", class="nav-bar container"): tdiv(class="inner-nav"):