Change twitter_card when overriding media

This commit is contained in:
dangered wolf 2023-08-22 16:48:34 -04:00
parent 72cf2030e4
commit f1fd8dd7d8
No known key found for this signature in database
GPG key ID: 41E4D37680ED8B58
5 changed files with 34 additions and 14 deletions

View file

@ -163,7 +163,6 @@ const populateTweetProperties = async (
if (tweet.card) {
const card = renderCard(tweet.card);
if (card.external_media) {
apiTweet.twitter_card = 'summary_large_image';
apiTweet.media = apiTweet.media || {};
apiTweet.media.external = card.external_media;
}
@ -172,6 +171,12 @@ const populateTweetProperties = async (
}
}
/* Workaround: Force player card by default for videos */
// @ts-expect-error Inexplicably, twitter_card becomes type of 'tweet' instead of 'tweet' | 'summary' | 'summary_large_image' | 'player'
if (apiTweet.media?.videos && apiTweet.twitter_card !== 'player') {
apiTweet.twitter_card = 'player';
}
/* If a language is specified in API or by user, let's try translating it! */
if (
typeof language === 'string' &&

View file

@ -1,6 +1,6 @@
import { Constants } from '../constants';
import { handleQuote } from '../helpers/quote';
import { formatNumber, sanitizeText } from '../helpers/utils';
import { formatNumber, sanitizeText, truncateWithEllipsis } from '../helpers/utils';
import { Strings } from '../strings';
import { getAuthorText } from '../helpers/author';
import { statusAPI } from '../api/status';
@ -121,9 +121,6 @@ export const handleStatus = async (
`<link rel="canonical" href="${Constants.TWITTER_ROOT}/${tweet.author.screen_name}/status/${tweet.id}"/>`,
`<meta property="og:url" content="${Constants.TWITTER_ROOT}/${tweet.author.screen_name}/status/${tweet.id}"/>`,
`<meta property="theme-color" content="${tweet.color || '#00a8fc'}"/>`,
`<meta property="twitter:card" content="${
tweet.quote?.twitter_card || tweet.twitter_card
}"/>`,
`<meta property="twitter:site" content="@${tweet.author.screen_name}"/>`,
`<meta property="twitter:creator" content="@${tweet.author.screen_name}"/>`,
`<meta property="twitter:title" content="${tweet.author.name} (@${tweet.author.screen_name})"/>`
@ -201,6 +198,10 @@ export const handleStatus = async (
if (instructions.siteName) {
siteName = instructions.siteName;
}
/* Overwrite our Twitter Card if overriding media, so it displays correctly in Discord */
if (tweet.twitter_card === 'player') {
tweet.twitter_card = 'summary_large_image';
}
break;
case 'video':
instructions = renderVideo(
@ -214,6 +215,10 @@ export const handleStatus = async (
if (instructions.siteName) {
siteName = instructions.siteName;
}
/* Overwrite our Twitter Card if overriding media, so it displays correctly in Discord */
if (tweet.twitter_card !== 'player') {
tweet.twitter_card = 'player';
}
/* This Tweet has a video to render. */
break;
}
@ -349,7 +354,10 @@ export const handleStatus = async (
headers.push(
`<meta property="og:title" content="${tweet.author.name} (@${tweet.author.screen_name})"/>`,
`<meta property="og:description" content="${text}"/>`,
`<meta property="og:site_name" content="${siteName}"/>`
`<meta property="og:site_name" content="${siteName}"/>`,
`<meta property="twitter:card" content="${
tweet.quote?.twitter_card || tweet.twitter_card
}"/>`,
);
/* Special reply handling if authorText is not overriden */
@ -366,13 +374,15 @@ export const handleStatus = async (
/* The additional oembed is pulled by Discord to enable improved embeds.
Telegram does not use this. */
headers.push(
`<link rel="alternate" href="${Constants.HOST_URL}/owoembed?text=${encodeURIComponent(
authorText.substring(0, 200)
)}${flags?.deprecated ? '&deprecated=true' : ''}&status=${encodeURIComponent(
status
)}&author=${encodeURIComponent(tweet.author?.screen_name || '')}&useXbranding=${
flags?.isXDomain ? 'true' : 'false'
}" type="application/json+oembed" title="${tweet.author.name}">`
`<link rel="alternate" href="{base}/owoembed?text={text}{deprecatedFlag}&status={status}&author={author}&useXbranding={useXBranding}" type="application/json+oembed" title="{name}">`.format({
base: Constants.HOST_URL,
text: encodeURIComponent(truncateWithEllipsis(authorText, 250)),
deprecatedFlag: flags?.deprecated ? '&deprecated=true' : '',
status: encodeURIComponent(status),
author: encodeURIComponent(tweet.author?.screen_name || ''),
useXBranding: flags?.isXDomain ? 'true' : 'false',
name: tweet.author.name || ''
})
);
/* When dealing with a Tweet of unknown lang, fall back to en */

View file

@ -15,6 +15,8 @@ export const unescapeText = (text: string) => {
.replace(/&amp;/g, '&');
};
export const truncateWithEllipsis = (str: string, maxLength: number): string => str.length > maxLength ? str.substring(0, maxLength - 1) + '…' : str;
const numberFormat = new Intl.NumberFormat('en-US');
export const formatNumber = (num: number) => numberFormat.format(num);

View file

@ -46,9 +46,10 @@ export const renderVideo = (
/* Push the raw video-related headers */
instructions.addHeaders = [
`<meta property="twitter:player:stream:content_type" content="${video.format}"/>`,
`<meta property="twitter:player:height" content="${video.height * sizeMultiplier}"/>`,
`<meta property="twitter:player:width" content="${video.width * sizeMultiplier}"/>`,
`<meta property="twitter:player:stream" content="${video.url}"/>`,
`<meta property="twitter:player:stream:content_type" content="${video.format}"/>`,
`<meta property="og:video" content="${video.url}"/>`,
`<meta property="og:video:secure_url" content="${video.url}"/>`,
`<meta property="og:video:height" content="${video.height * sizeMultiplier}"/>`,

View file

@ -442,10 +442,12 @@ router.get('/:prefix?/:handle/status/:id', statusRequest);
router.get('/:prefix?/:handle/status/:id/photo/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/status/:id/photos/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/status/:id/video/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/status/:id/videos/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/statuses/:id', statusRequest);
router.get('/:prefix?/:handle/statuses/:id/photo/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/statuses/:id/photos/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/statuses/:id/video/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/statuses/:id/videos/:mediaNumber', statusRequest);
router.get('/:prefix?/:handle/status/:id/:language', statusRequest);
router.get('/:prefix?/:handle/statuses/:id/:language', statusRequest);
router.get('/status/:id', statusRequest);