mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-06 19:10:54 +01:00
Implement new video implementation
This commit is contained in:
parent
1097a6ea3a
commit
d90b39ab11
5 changed files with 81 additions and 57 deletions
|
@ -71,6 +71,8 @@ const populateTweetProperties = async (
|
||||||
tweet.extended_entities?.media || tweet.entities?.media || []
|
tweet.extended_entities?.media || tweet.entities?.media || []
|
||||||
);
|
);
|
||||||
|
|
||||||
|
console.log('tweet', JSON.stringify(tweet))
|
||||||
|
|
||||||
/* Populate this Tweet's media */
|
/* Populate this Tweet's media */
|
||||||
mediaList.forEach(media => {
|
mediaList.forEach(media => {
|
||||||
const mediaObject = processMedia(media);
|
const mediaObject = processMedia(media);
|
||||||
|
|
|
@ -29,10 +29,13 @@ export const Constants = {
|
||||||
'include_quote_count=true',
|
'include_quote_count=true',
|
||||||
'include_reply_count=1',
|
'include_reply_count=1',
|
||||||
'tweet_mode=extended',
|
'tweet_mode=extended',
|
||||||
|
'include_entities=true',
|
||||||
'include_ext_media_color=true',
|
'include_ext_media_color=true',
|
||||||
'include_ext_media_availability=true',
|
'include_ext_media_availability=true',
|
||||||
'include_ext_sensitive_media_warning=true',
|
'include_ext_sensitive_media_warning=true',
|
||||||
'simple_quoted_tweet=true'
|
'include_ext_has_birdwatch_notes=true',
|
||||||
|
'simple_quoted_tweet=true',
|
||||||
|
'ext=mediaStats%2ChighlightedLabel'
|
||||||
].join('&'),
|
].join('&'),
|
||||||
BASE_HEADERS: {
|
BASE_HEADERS: {
|
||||||
'DNT': `1`,
|
'DNT': `1`,
|
||||||
|
|
|
@ -5,6 +5,7 @@ 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';
|
import { renderPhoto } from '../render/photo';
|
||||||
|
import { renderVideo } from '../render/video';
|
||||||
|
|
||||||
export const returnError = (error: string): StatusResponse => {
|
export const returnError = (error: string): StatusResponse => {
|
||||||
return {
|
return {
|
||||||
|
@ -149,7 +150,7 @@ export const handleStatus = async (
|
||||||
switch (overrideMedia.type) {
|
switch (overrideMedia.type) {
|
||||||
case 'photo':
|
case 'photo':
|
||||||
/* This Tweet has a photo to render. */
|
/* This Tweet has a photo to render. */
|
||||||
instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, isOverrideMedia: true, userAgent: userAgent }, overrideMedia as APIPhoto );
|
instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, userAgent: userAgent, isOverrideMedia: true }, overrideMedia as APIPhoto );
|
||||||
headers.push(...instructions.addHeaders);
|
headers.push(...instructions.addHeaders);
|
||||||
if (instructions.authorText) {
|
if (instructions.authorText) {
|
||||||
authorText = instructions.authorText;
|
authorText = instructions.authorText;
|
||||||
|
@ -159,6 +160,14 @@ export const handleStatus = async (
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'video':
|
case 'video':
|
||||||
|
instructions = renderVideo( {tweet: tweet, userAgent: userAgent, text: newText, isOverrideMedia: true }, overrideMedia as APIVideo );
|
||||||
|
headers.push(...instructions.addHeaders);
|
||||||
|
if (instructions.authorText) {
|
||||||
|
authorText = instructions.authorText;
|
||||||
|
}
|
||||||
|
if (instructions.siteName) {
|
||||||
|
siteName = instructions.siteName;
|
||||||
|
}
|
||||||
/* This Tweet has a video to render. */
|
/* This Tweet has a video to render. */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -166,63 +175,14 @@ export const handleStatus = async (
|
||||||
const instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, userAgent: userAgent }, tweet.media?.mosaic );
|
const instructions = renderPhoto( {tweet: tweet, authorText: authorText, engagementText: engagementText, userAgent: userAgent }, tweet.media?.mosaic );
|
||||||
headers.push(...instructions.addHeaders);
|
headers.push(...instructions.addHeaders);
|
||||||
} else if (tweet.media?.videos) {
|
} else if (tweet.media?.videos) {
|
||||||
authorText = tweet.translation?.text || newText || '';
|
const instructions = renderVideo( {tweet: tweet, userAgent: userAgent, text: newText }, tweet.media?.videos[0] );
|
||||||
|
headers.push(...instructions.addHeaders);
|
||||||
const videos = tweet.media?.videos;
|
if (instructions.authorText) {
|
||||||
const all = tweet.media?.all || [];
|
authorText = instructions.authorText;
|
||||||
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.
|
|
||||||
|
|
||||||
Basically, our solution is to cut the dimensions in half if the video is too big (> 1080p),
|
|
||||||
or double them if it's too small. (<400p)
|
|
||||||
|
|
||||||
We check both height and width so we can apply this to both horizontal and vertical videos equally*/
|
|
||||||
|
|
||||||
let sizeMultiplier = 1;
|
|
||||||
|
|
||||||
if (video.width > 1920 || video.height > 1920) {
|
|
||||||
sizeMultiplier = 0.5;
|
|
||||||
}
|
}
|
||||||
if (video.width < 400 && video.height < 400) {
|
if (instructions.siteName) {
|
||||||
sizeMultiplier = 2;
|
siteName = instructions.siteName;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Like photos when picking a specific one (not using mosaic),
|
|
||||||
we'll put an indicator if there are more than one video */
|
|
||||||
if (all && all.length > 1) {
|
|
||||||
const videoCounter = Strings.VIDEO_COUNT.format({
|
|
||||||
number: String(all.indexOf(video) + 1),
|
|
||||||
total: String(all.length)
|
|
||||||
});
|
|
||||||
|
|
||||||
authorText =
|
|
||||||
authorText === Strings.DEFAULT_AUTHOR_TEXT
|
|
||||||
? videoCounter
|
|
||||||
: `${authorText}${authorText ? ' ― ' : ''}${videoCounter}`;
|
|
||||||
|
|
||||||
siteName = `${Constants.BRANDING_NAME} - ${videoCounter}`;
|
|
||||||
|
|
||||||
if (engagementText) {
|
|
||||||
siteName = `${Constants.BRANDING_NAME} - ${engagementText} - ${videoCounter}`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Push the raw video-related headers */
|
|
||||||
headers.push(
|
|
||||||
`<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="og:video" content="${video.url}"/>`,
|
|
||||||
`<meta property="og:video:secure_url" content="${video.url}"/>`,
|
|
||||||
`<meta property="og:video:height" content="${video.height * sizeMultiplier}"/>`,
|
|
||||||
`<meta property="og:video:width" content="${video.width * sizeMultiplier}"/>`,
|
|
||||||
`<meta property="og:video:type" content="${video.format}"/>`,
|
|
||||||
`<meta property="twitter:image" content="0"/>`
|
|
||||||
);
|
|
||||||
} else if (tweet.media?.external) {
|
} else if (tweet.media?.external) {
|
||||||
const { external } = tweet.media;
|
const { external } = tweet.media;
|
||||||
authorText = newText || '';
|
authorText = newText || '';
|
||||||
|
|
57
src/render/video.ts
Normal file
57
src/render/video.ts
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
import { Constants } from "../constants";
|
||||||
|
import { Strings } from "../strings";
|
||||||
|
|
||||||
|
export const renderVideo = (properties: RenderProperties, video: APIVideo): ResponseInstructions => {
|
||||||
|
const { tweet, userAgent, text } = properties;
|
||||||
|
const instructions: ResponseInstructions = { addHeaders: [] };
|
||||||
|
|
||||||
|
const all = tweet.media?.all as APIMedia[];
|
||||||
|
|
||||||
|
/* This fix is specific to Discord not wanting to render videos that are too large,
|
||||||
|
or rendering low quality videos too small.
|
||||||
|
|
||||||
|
Basically, our solution is to cut the dimensions in half if the video is too big (> 1080p),
|
||||||
|
or double them if it's too small. (<400p)
|
||||||
|
|
||||||
|
We check both height and width so we can apply this to both horizontal and vertical videos equally*/
|
||||||
|
|
||||||
|
let sizeMultiplier = 1;
|
||||||
|
|
||||||
|
if (video.width > 1920 || video.height > 1920) {
|
||||||
|
sizeMultiplier = 0.5;
|
||||||
|
}
|
||||||
|
if (video.width < 400 && video.height < 400) {
|
||||||
|
sizeMultiplier = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Like photos when picking a specific one (not using mosaic),
|
||||||
|
we'll put an indicator if there are more than one video */
|
||||||
|
if (all && all.length > 1 && (userAgent?.indexOf('Telegram') ?? 0) > -1) {
|
||||||
|
const baseString = all.length === tweet.media?.videos?.length ? Strings.VIDEO_COUNT : Strings.MEDIA_COUNT;
|
||||||
|
const videoCounter = baseString.format({
|
||||||
|
number: String(all.indexOf(video) + 1),
|
||||||
|
total: String(all.length)
|
||||||
|
});
|
||||||
|
|
||||||
|
instructions.siteName = `${Constants.BRANDING_NAME} - ${videoCounter}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
instructions.authorText = tweet.translation?.text || text || '';
|
||||||
|
|
||||||
|
/* 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="og:video" content="${video.url}"/>`,
|
||||||
|
`<meta property="og:video:secure_url" content="${video.url}"/>`,
|
||||||
|
`<meta property="og:video:height" content="${video.height * sizeMultiplier}"/>`,
|
||||||
|
`<meta property="og:video:width" content="${video.width * sizeMultiplier}"/>`,
|
||||||
|
`<meta property="og:video:type" content="${video.format}"/>`,
|
||||||
|
`<meta property="twitter:image" content="0"/>`
|
||||||
|
];
|
||||||
|
|
||||||
|
return instructions;
|
||||||
|
}
|
2
src/types/types.d.ts
vendored
2
src/types/types.d.ts
vendored
|
@ -20,6 +20,7 @@ interface ResponseInstructions {
|
||||||
authorText?: string;
|
authorText?: string;
|
||||||
siteName?: string;
|
siteName?: string;
|
||||||
engagementText?: string;
|
engagementText?: string;
|
||||||
|
text?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface RenderProperties {
|
interface RenderProperties {
|
||||||
|
@ -29,6 +30,7 @@ interface RenderProperties {
|
||||||
engagementText?: string;
|
engagementText?: string;
|
||||||
isOverrideMedia?: boolean;
|
isOverrideMedia?: boolean;
|
||||||
userAgent?: string;
|
userAgent?: string;
|
||||||
|
text?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Request {
|
interface Request {
|
||||||
|
|
Loading…
Add table
Reference in a new issue