Add paragraphs and bugfix

This commit is contained in:
dangered wolf 2023-08-18 04:58:00 -04:00
parent 5c55fe47b6
commit 40d79d0d05
No known key found for this signature in database
GPG key ID: 41E4D37680ED8B58
2 changed files with 125 additions and 108 deletions

View file

@ -75,7 +75,7 @@ export const handleStatus = async (
}
/* Catch direct media request (d.fxtwitter.com, or .mp4 / .jpg) */
if (flags?.direct && tweet.media) {
if (flags?.direct && !flags?.textOnly && tweet.media) {
let redirectUrl: string | null = null;
const all = tweet.media.all || [];
// if (tweet.media.videos) {
@ -98,16 +98,6 @@ export const handleStatus = async (
}
}
/* Use quote media if there is no media in this Tweet */
if (!tweet.media && tweet.quote?.media) {
tweet.media = tweet.quote.media;
tweet.twitter_card = tweet.quote.twitter_card;
}
if (flags?.textOnly) {
tweet.media = undefined;
}
/* At this point, we know we're going to have to create a
regular embed because it's not an API or direct media request */
@ -121,7 +111,9 @@ export const handleStatus = async (
const headers = [
`<link rel="canonical" href="https://twitter.com/${tweet.author.screen_name}/status/${tweet.id}"/>`,
`<meta property="theme-color" content="${tweet.color}"/>`,
`<meta property="twitter:card" content="${tweet.twitter_card}"/>`,
`<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})"/>`
@ -165,92 +157,95 @@ export const handleStatus = async (
console.log('overrideMedia', JSON.stringify(overrideMedia));
if (overrideMedia) {
let instructions: ResponseInstructions;
if (!flags?.textOnly) {
const media = tweet.media || tweet.quote?.media;
if (overrideMedia) {
let instructions: ResponseInstructions;
switch (overrideMedia.type) {
case 'photo':
/* This Tweet has a photo to render. */
instructions = renderPhoto(
{
tweet: tweet,
authorText: authorText,
engagementText: engagementText,
userAgent: userAgent,
isOverrideMedia: true
},
overrideMedia as APIPhoto
);
headers.push(...instructions.addHeaders);
if (instructions.authorText) {
authorText = instructions.authorText;
}
if (instructions.siteName) {
siteName = instructions.siteName;
}
break;
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. */
break;
switch (overrideMedia.type) {
case 'photo':
/* This Tweet has a photo to render. */
instructions = renderPhoto(
{
tweet: tweet,
authorText: authorText,
engagementText: engagementText,
userAgent: userAgent,
isOverrideMedia: true
},
overrideMedia as APIPhoto
);
headers.push(...instructions.addHeaders);
if (instructions.authorText) {
authorText = instructions.authorText;
}
if (instructions.siteName) {
siteName = instructions.siteName;
}
break;
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. */
break;
}
} else if (media?.mosaic) {
const instructions = renderPhoto(
{
tweet: tweet,
authorText: authorText,
engagementText: engagementText,
userAgent: userAgent
},
media.mosaic
);
headers.push(...instructions.addHeaders);
} else if (media?.videos) {
const instructions = renderVideo(
{ tweet: tweet, userAgent: userAgent, text: newText },
media.videos[0]
);
headers.push(...instructions.addHeaders);
if (instructions.authorText) {
authorText = instructions.authorText;
}
if (instructions.siteName) {
siteName = instructions.siteName;
}
} else if (media?.photos) {
const instructions = renderPhoto(
{
tweet: tweet,
authorText: authorText,
engagementText: engagementText,
userAgent: userAgent
},
media.photos[0]
);
headers.push(...instructions.addHeaders);
} else if (media?.external) {
const { external } = media;
authorText = newText || '';
headers.push(
`<meta property="twitter:player" content="${external.url}">`,
`<meta property="twitter:player:width" content="${external.width}">`,
`<meta property="twitter:player:height" content="${external.height}">`,
`<meta property="og:type" content="video.other">`,
`<meta property="og:video:url" content="${external.url}">`,
`<meta property="og:video:secure_url" content="${external.url}">`,
`<meta property="og:video:width" content="${external.width}">`,
`<meta property="og:video:height" content="${external.height}">`
);
}
} 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) {
const instructions = renderVideo(
{ tweet: tweet, userAgent: userAgent, text: newText },
tweet.media?.videos[0]
);
headers.push(...instructions.addHeaders);
if (instructions.authorText) {
authorText = instructions.authorText;
}
if (instructions.siteName) {
siteName = instructions.siteName;
}
} else if (tweet.media?.photos) {
const instructions = renderPhoto(
{
tweet: tweet,
authorText: authorText,
engagementText: engagementText,
userAgent: userAgent
},
tweet.media?.photos[0]
);
headers.push(...instructions.addHeaders);
} else if (tweet.media?.external) {
const { external } = tweet.media;
authorText = newText || '';
headers.push(
`<meta property="twitter:player" content="${external.url}">`,
`<meta property="twitter:player:width" content="${external.width}">`,
`<meta property="twitter:player:height" content="${external.height}">`,
`<meta property="og:type" content="video.other">`,
`<meta property="og:video:url" content="${external.url}">`,
`<meta property="og:video:secure_url" content="${external.url}">`,
`<meta property="og:video:width" content="${external.width}">`,
`<meta property="og:video:height" content="${external.height}">`
);
}
/* This Tweet contains a poll, so we'll render it */
@ -292,7 +287,13 @@ export const handleStatus = async (
}
/* If we have no media to display, instead we'll display the user profile picture in the embed */
if (!tweet.media?.videos && !tweet.media?.photos && !flags?.textOnly) {
if (
!tweet.media?.videos &&
!tweet.media?.photos &&
!tweet.quote?.media?.photos &&
!tweet.quote?.media?.videos &&
!flags?.textOnly
) {
/* Use a slightly higher resolution image for profile pics */
const avatar = tweet.author.avatar_url;
if (!useIV) {

View file

@ -58,34 +58,50 @@ const htmlifyHashtags = (input: string): string => {
});
};
function paragraphify(text: string, isQuote = false): string {
const tag = isQuote ? 'blockquote' : 'p';
return text
.split('\n')
.map(line => line.trim())
.filter(line => line.length > 0)
.map(line => `<${tag}>${line}</${tag}>`)
.join('\n');
}
/* TODO: maybe refactor so all tweets pull from this */
const generateTweet = (tweet: APITweet, isQuote = false): string => {
let text = sanitizeText(tweet.text).replace(/\n/g, '<br>');
let text = paragraphify(sanitizeText(tweet.text));
text = htmlifyLinks(text);
text = htmlifyHashtags(text);
text = populateUserLinks(tweet, text);
return `
<!-- Embed profile picture, display name, and screen name in table -->
${!isQuote ? `<table>
${
!isQuote
? `<table>
<img src="${tweet.author.avatar_url}" alt="${tweet.author.name}'s profile picture" />
<h2>${tweet.author.name}</h2>
<p>@${tweet.author.screen_name}</p>
<p>${getSocialTextIV(tweet)}</p>
</table>` : ''}
${isQuote ? `
</table>`
: ''
}
${
isQuote
? `
<h4>Quoting <a href="${Constants.TWITTER_ROOT}/${tweet.author.screen_name}">${tweet.author.name}</a> (@${tweet.author.screen_name})</h4>
` : ''}
`
: ''
}
<!-- Embed Tweet text -->
${isQuote ? '<blockquote>' : ''}
${text}
${isQuote ? '</blockquote>' : ''}
${generateTweetMedia(tweet)}
${(!isQuote && tweet.quote) ? generateTweet(tweet.quote, true) : ''}
<br>${(!isQuote ? `<a href="${tweet.url}">View original</a>` : '')}
`
}
${!isQuote && tweet.quote ? generateTweet(tweet.quote, true) : ''}
<br>${!isQuote ? `<a href="${tweet.url}">View original</a>` : ''}
`;
};
export const renderInstantView = (properties: RenderProperties): ResponseInstructions => {
console.log('Generating Instant View...');