mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 18:40:56 +01:00
Start refactoring media render code
This commit is contained in:
parent
bb18cc13d3
commit
1097a6ea3a
5 changed files with 116 additions and 82 deletions
|
@ -4,6 +4,7 @@ import { formatNumber, sanitizeText } from '../helpers/utils';
|
||||||
import { Strings } from '../strings';
|
import { Strings } from '../strings';
|
||||||
import { getAuthorText } from '../helpers/author';
|
import { getAuthorText } from '../helpers/author';
|
||||||
import { statusAPI } from '../api/status';
|
import { statusAPI } from '../api/status';
|
||||||
|
import { renderPhoto } from '../render/photo';
|
||||||
|
|
||||||
export const returnError = (error: string): StatusResponse => {
|
export const returnError = (error: string): StatusResponse => {
|
||||||
return {
|
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
|
// 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]) {
|
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`;
|
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.
|
if (overrideMedia) {
|
||||||
You can still use /video/:number to get a specific video. Otherwise, it'll pick the first. */
|
let instructions: ResponseInstructions;
|
||||||
if (tweet.media?.videos || overrideMedia?.type === 'video') {
|
|
||||||
authorText = newText || '';
|
|
||||||
|
|
||||||
if (tweet?.translation) {
|
switch (overrideMedia.type) {
|
||||||
authorText = tweet.translation?.text || '';
|
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 videos = tweet.media?.videos;
|
||||||
const all = tweet.media?.all || [];
|
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,
|
/* This fix is specific to Discord not wanting to render videos that are too large,
|
||||||
or rendering low quality videos too small.
|
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="og:video:type" content="${video.format}"/>`,
|
||||||
`<meta property="twitter:image" content="0"/>`
|
`<meta property="twitter:image" content="0"/>`
|
||||||
);
|
);
|
||||||
}
|
} else if (tweet.media?.external) {
|
||||||
|
|
||||||
/* 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) {
|
|
||||||
const { external } = tweet.media;
|
const { external } = tweet.media;
|
||||||
authorText = newText || '';
|
authorText = newText || '';
|
||||||
headers.push(
|
headers.push(
|
||||||
|
|
|
@ -29,10 +29,11 @@ export const handleMosaic = async (
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
type: 'mosaic_photo',
|
||||||
formats: {
|
formats: {
|
||||||
jpeg: `${baseUrl}jpeg/${id}${path}`,
|
jpeg: `${baseUrl}jpeg/${id}${path}`,
|
||||||
webp: `${baseUrl}webp/${id}${path}`
|
webp: `${baseUrl}webp/${id}${path}`
|
||||||
}
|
}
|
||||||
} as unknown as APIMosaicPhoto;
|
} as APIMosaicPhoto;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
53
src/render/photo.ts
Normal file
53
src/render/photo.ts
Normal 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;
|
||||||
|
}
|
|
@ -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})`,
|
QUOTE_TEXT: `↘️ Quoting {name} (@{screen_name})`,
|
||||||
TRANSLATE_TEXT: `↘️ Translated from {language}`,
|
TRANSLATE_TEXT: `↘️ Translated from {language}`,
|
||||||
TRANSLATE_TEXT_INTL: `↘️ {source} ➡️ {destination}`,
|
TRANSLATE_TEXT_INTL: `↘️ {source} ➡️ {destination}`,
|
||||||
PHOTO_COUNT: `Photo {number} of {total}`,
|
PHOTO_COUNT: `Photo {number} / {total}`,
|
||||||
VIDEO_COUNT: `Video {number} of {total}`,
|
VIDEO_COUNT: `Video {number} / {total}`,
|
||||||
|
MEDIA_COUNT: `Media {number} / {total}`,
|
||||||
|
|
||||||
SINGULAR_DAY_LEFT: 'day left',
|
SINGULAR_DAY_LEFT: 'day left',
|
||||||
PLURAL_DAYS_LEFT: 'days left',
|
PLURAL_DAYS_LEFT: 'days left',
|
||||||
|
|
45
src/types/types.d.ts
vendored
45
src/types/types.d.ts
vendored
|
@ -15,6 +15,22 @@ interface StatusResponse {
|
||||||
cacheControl?: string | null;
|
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 {
|
interface Request {
|
||||||
params: {
|
params: {
|
||||||
[param: string]: string;
|
[param: string]: string;
|
||||||
|
@ -91,15 +107,26 @@ interface APIPoll {
|
||||||
time_left_en: string;
|
time_left_en: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface APIPhoto {
|
interface APIMedia {
|
||||||
type: 'photo';
|
type: string;
|
||||||
url: string;
|
url: string;
|
||||||
width: number;
|
width: number;
|
||||||
height: number;
|
height: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface APIPhoto extends APIMedia {
|
||||||
|
type: 'photo';
|
||||||
altText: string;
|
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';
|
type: 'mosaic_photo';
|
||||||
formats: {
|
formats: {
|
||||||
webp: string;
|
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 {
|
interface APITweet {
|
||||||
id: string;
|
id: string;
|
||||||
url: string;
|
url: string;
|
||||||
|
@ -140,7 +157,7 @@ interface APITweet {
|
||||||
external?: APIExternalMedia;
|
external?: APIExternalMedia;
|
||||||
photos?: APIPhoto[];
|
photos?: APIPhoto[];
|
||||||
videos?: APIVideo[];
|
videos?: APIVideo[];
|
||||||
all?: (APIPhoto | APIVideo)[];
|
all?: APIMedia[];
|
||||||
mosaic?: APIMosaicPhoto;
|
mosaic?: APIMosaicPhoto;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue