Start refactoring media render code

This commit is contained in:
dangered wolf 2023-05-31 15:50:19 -04:00
parent bb18cc13d3
commit 1097a6ea3a
No known key found for this signature in database
GPG key ID: 41E4D37680ED8B58
5 changed files with 116 additions and 82 deletions

View file

@ -4,6 +4,7 @@ import { formatNumber, sanitizeText } from '../helpers/utils';
import { Strings } from '../strings';
import { getAuthorText } from '../helpers/author';
import { statusAPI } from '../api/status';
import { renderPhoto } from '../render/photo';
export const returnError = (error: string): StatusResponse => {
return {
@ -43,7 +44,7 @@ export const handleStatus = async (
};
}
let overrideMedia: APIPhoto | APIVideo | undefined;
let overrideMedia: APIMedia | undefined;
// Check if mediaNumber exists, and if that media exists in tweet.media.all. If it does, we'll store overrideMedia variable
if (mediaNumber && tweet.media && tweet.media.all && tweet.media.all[mediaNumber - 1]) {
@ -140,20 +141,36 @@ export const handleStatus = async (
newText = `${formatText}\n\n` + `${translation.text}\n\n`;
}
/* This Tweet has a video to render.
console.log('overrideMedia', JSON.stringify(overrideMedia));
Twitter supports multiple videos in a Tweet now. But we have no mechanism to embed more than one.
You can still use /video/:number to get a specific video. Otherwise, it'll pick the first. */
if (tweet.media?.videos || overrideMedia?.type === 'video') {
authorText = newText || '';
if (overrideMedia) {
let instructions: ResponseInstructions;
if (tweet?.translation) {
authorText = tweet.translation?.text || '';
switch (overrideMedia.type) {
case 'photo':
/* This Tweet has a photo to render. */
instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, isOverrideMedia: true, userAgent: userAgent }, overrideMedia as APIPhoto );
headers.push(...instructions.addHeaders);
if (instructions.authorText) {
authorText = instructions.authorText;
}
if (instructions.siteName) {
siteName = instructions.siteName;
}
break;
case 'video':
/* This Tweet has a video to render. */
break;
}
} else if (tweet.media?.mosaic) {
const instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, userAgent: userAgent }, tweet.media?.mosaic );
headers.push(...instructions.addHeaders);
} else if (tweet.media?.videos) {
authorText = tweet.translation?.text || newText || '';
const videos = tweet.media?.videos;
const all = tweet.media?.all || [];
const video = overrideMedia as APIVideo || videos?.[(mediaNumber || 1) - 1];
const video = videos?.[0];
/* This fix is specific to Discord not wanting to render videos that are too large,
or rendering low quality videos too small.
@ -206,62 +223,7 @@ export const handleStatus = async (
`<meta property="og:video:type" content="${video.format}"/>`,
`<meta property="twitter:image" content="0"/>`
);
}
/* This Tweet has one or more photos to render */
if (tweet.media?.photos || overrideMedia?.type === 'photo') {
let photo: APIPhoto | APIMosaicPhoto = overrideMedia as APIPhoto || tweet.media?.photos?.[0];
/* If there isn't a specified media number and we have a
mosaic response, we'll render it using mosaic */
if (!overrideMedia && tweet.media?.mosaic) {
photo = {
/* Include dummy height/width for TypeScript reasons. We have a check to make sure we don't use these later. */
height: 0,
width: 0,
url: tweet.media.mosaic.formats.jpeg,
type: 'photo',
altText: ''
};
/* If mosaic isn't available or the link calls for a specific photo,
we'll indicate which photo it is out of the total */
} else if (tweet.media?.all && tweet.media.all.length > 1) {
const { all } = tweet.media;
const photoCounter = Strings.PHOTO_COUNT.format({
number: String(all.indexOf(photo) + 1),
total: String(all.length)
});
authorText =
authorText === Strings.DEFAULT_AUTHOR_TEXT
? photoCounter
: `${authorText}${authorText ? ' ― ' : ''}${photoCounter}`;
siteName = `${Constants.BRANDING_NAME} - ${photoCounter}`;
if (engagementText) {
siteName = `${Constants.BRANDING_NAME} - ${engagementText} - ${photoCounter}`;
}
}
/* Push the raw photo-related headers */
headers.push(
`<meta property="twitter:image" content="${photo.url}"/>`,
`<meta property="og:image" content="${photo.url}"/>`
);
if (!tweet.media?.mosaic) {
headers.push(
`<meta property="twitter:image:width" content="${photo.width}"/>`,
`<meta property="twitter:image:height" content="${photo.height}"/>`,
`<meta property="og:image:width" content="${photo.width}"/>`,
`<meta property="og:image:height" content="${photo.height}"/>`
);
}
}
/* We have external media available to us (i.e. YouTube videos) */
if (tweet.media?.external) {
} else if (tweet.media?.external) {
const { external } = tweet.media;
authorText = newText || '';
headers.push(

View file

@ -29,10 +29,11 @@ export const handleMosaic = async (
}
return {
type: 'mosaic_photo',
formats: {
jpeg: `${baseUrl}jpeg/${id}${path}`,
webp: `${baseUrl}webp/${id}${path}`
}
} as unknown as APIMosaicPhoto;
} as APIMosaicPhoto;
}
};

53
src/render/photo.ts Normal file
View file

@ -0,0 +1,53 @@
import { Constants } from "../constants";
import { Strings } from "../strings";
export const renderPhoto = (properties: RenderProperties, photo: APIPhoto | APIMosaicPhoto): ResponseInstructions => {
const { tweet, engagementText, authorText, isOverrideMedia, userAgent } = properties;
const instructions: ResponseInstructions = { addHeaders: [] };
if (!tweet.media?.mosaic || isOverrideMedia) {
photo = photo as APIPhoto;
const all = tweet.media?.all as APIMedia[];
const baseString = all.length === tweet.media?.photos?.length ? Strings.PHOTO_COUNT : Strings.MEDIA_COUNT;
const photoCounter = baseString.format({
number: String(all.indexOf(photo) + 1),
total: String(all.length)
});
console.log('Telegram', userAgent?.indexOf('Telegram'))
if (authorText === Strings.DEFAULT_AUTHOR_TEXT || (userAgent?.indexOf('Telegram') ?? 0) > -1) {
instructions.authorText = photoCounter;
} else {
instructions.authorText = `${authorText}${authorText ? ' ― ' : ''}${photoCounter}`;
}
if (engagementText && (userAgent?.indexOf('Telegram') ?? 0) === -1) {
instructions.siteName = `${Constants.BRANDING_NAME} - ${engagementText} - ${photoCounter}`;
} else {
instructions.siteName = `${Constants.BRANDING_NAME} - ${photoCounter}`;
}
}
if (photo.type === 'mosaic_photo' && !isOverrideMedia) {
console.log('Mosaic object:', tweet.media?.mosaic);
instructions.addHeaders = [
`<meta property="twitter:image" content="${tweet.media?.mosaic?.formats.jpeg}"/>`,
`<meta property="og:image" content="${tweet.media?.mosaic?.formats.jpeg}"/>`
];
} else {
instructions.addHeaders = [
`<meta property="twitter:image" content="${photo.url}"/>`,
`<meta property="og:image" content="${photo.url}"/>`,
`<meta property="twitter:image:width" content="${photo.width}"/>`,
`<meta property="twitter:image:height" content="${photo.height}"/>`,
`<meta property="og:image:width" content="${photo.width}"/>`,
`<meta property="og:image:height" content="${photo.height}"/>`
];
}
console.log('Photo render instructions', JSON.stringify(instructions));
return instructions;
}

View file

@ -131,8 +131,9 @@ This is caused by Twitter API downtime or a new bug. Try again in a little while
QUOTE_TEXT: `↘️ Quoting {name} (@{screen_name})`,
TRANSLATE_TEXT: `↘️ Translated from {language}`,
TRANSLATE_TEXT_INTL: `↘️ {source} ➡️ {destination}`,
PHOTO_COUNT: `Photo {number} of {total}`,
VIDEO_COUNT: `Video {number} of {total}`,
PHOTO_COUNT: `Photo {number} / {total}`,
VIDEO_COUNT: `Video {number} / {total}`,
MEDIA_COUNT: `Media {number} / {total}`,
SINGULAR_DAY_LEFT: 'day left',
PLURAL_DAYS_LEFT: 'days left',

45
src/types/types.d.ts vendored
View file

@ -15,6 +15,22 @@ interface StatusResponse {
cacheControl?: string | null;
}
interface ResponseInstructions {
addHeaders: string[];
authorText?: string;
siteName?: string;
engagementText?: string;
}
interface RenderProperties {
tweet: APITweet;
siteText?: string;
authorText?: string;
engagementText?: string;
isOverrideMedia?: boolean;
userAgent?: string;
}
interface Request {
params: {
[param: string]: string;
@ -91,15 +107,26 @@ interface APIPoll {
time_left_en: string;
}
interface APIPhoto {
type: 'photo';
interface APIMedia {
type: string;
url: string;
width: number;
height: number;
}
interface APIPhoto extends APIMedia {
type: 'photo';
altText: string;
}
interface APIMosaicPhoto {
interface APIVideo extends APIMedia {
type: 'video' | 'gif';
thumbnail_url: string;
format: string;
duration: number;
}
interface APIMosaicPhoto extends APIMedia {
type: 'mosaic_photo';
formats: {
webp: string;
@ -107,16 +134,6 @@ interface APIMosaicPhoto {
};
}
interface APIVideo {
type: 'video' | 'gif';
url: string;
thumbnail_url: string;
width: number;
height: number;
format: string;
duration: number;
}
interface APITweet {
id: string;
url: string;
@ -140,7 +157,7 @@ interface APITweet {
external?: APIExternalMedia;
photos?: APIPhoto[];
videos?: APIVideo[];
all?: (APIPhoto | APIVideo)[];
all?: APIMedia[];
mosaic?: APIMosaicPhoto;
};