mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 10:30:55 +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 { 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(
|
||||
|
|
|
@ -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
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})`,
|
||||
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
45
src/types/types.d.ts
vendored
|
@ -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;
|
||||
};
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue