mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 02:20:54 +01:00
125 lines
4.5 KiB
TypeScript
125 lines
4.5 KiB
TypeScript
import { Constants } from '../constants';
|
|
import { getSocialTextIV } from '../helpers/author';
|
|
import { sanitizeText } from '../helpers/utils';
|
|
|
|
const populateUserLinks = (tweet: APITweet, text: string): string => {
|
|
/* TODO: Maybe we can add username splices to our API so only genuinely valid users are linked? */
|
|
text.match(/@(\w{1,15})/g)?.forEach(match => {
|
|
const username = match.replace('@', '');
|
|
text = text.replace(
|
|
match,
|
|
`<a href="${Constants.TWITTER_ROOT}/${username}" target="_blank" rel="noopener noreferrer">${match}</a>`
|
|
);
|
|
});
|
|
return text;
|
|
};
|
|
|
|
const generateTweetMedia = (tweet: APITweet): string => {
|
|
let media = '';
|
|
if (tweet.media?.all?.length) {
|
|
tweet.media.all.forEach(mediaItem => {
|
|
switch (mediaItem.type) {
|
|
case 'photo':
|
|
media += `<img src="${mediaItem.url}" alt="${tweet.author.name}'s photo" />`;
|
|
break;
|
|
case 'video':
|
|
media += `<video src="${mediaItem.url}" alt="${tweet.author.name}'s video" />`;
|
|
break;
|
|
case 'gif':
|
|
media += `<video src="${mediaItem.url}" alt="${tweet.author.name}'s gif" />`;
|
|
break;
|
|
}
|
|
});
|
|
}
|
|
return media;
|
|
};
|
|
|
|
// const formatDateTime = (date: Date): string => {
|
|
// const yyyy = date.getFullYear();
|
|
// const mm = String(date.getMonth() + 1).padStart(2, '0'); // Months are 0-indexed
|
|
// const dd = String(date.getDate()).padStart(2, '0');
|
|
// const hh = String(date.getHours()).padStart(2, '0');
|
|
// const min = String(date.getMinutes()).padStart(2, '0');
|
|
// return `${hh}:${min} - ${yyyy}/${mm}/${dd}`;
|
|
// }
|
|
|
|
const htmlifyLinks = (input: string): string => {
|
|
const urlPattern = /\bhttps?:\/\/\S+/g;
|
|
return input.replace(urlPattern, url => {
|
|
return `<a href="${url}">${url}</a>`;
|
|
});
|
|
};
|
|
|
|
const htmlifyHashtags = (input: string): string => {
|
|
const hashtagPattern = /#([a-zA-Z_]\w*)/g;
|
|
return input.replace(hashtagPattern, (match, hashtag) => {
|
|
const encodedHashtag = encodeURIComponent(hashtag);
|
|
return `<a href="https://twitter.com/hashtag/${encodedHashtag}?src=hashtag_click">${match}</a>`;
|
|
});
|
|
};
|
|
|
|
/* 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>');
|
|
text = htmlifyLinks(text);
|
|
text = htmlifyHashtags(text);
|
|
text = populateUserLinks(tweet, text);
|
|
|
|
return `
|
|
<!-- Embed profile picture, display name, and screen name in 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 ? `
|
|
<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>` : '')}
|
|
`
|
|
}
|
|
|
|
export const renderInstantView = (properties: RenderProperties): ResponseInstructions => {
|
|
console.log('Generating Instant View...');
|
|
const { tweet } = properties;
|
|
const instructions: ResponseInstructions = { addHeaders: [] };
|
|
/* Use ISO date for Medium template */
|
|
const postDate = new Date(tweet.created_at).toISOString();
|
|
|
|
/* Include Instant-View related headers. This is an unfinished project. Thanks to https://nikstar.me/post/instant-view/ for the help! */
|
|
instructions.addHeaders = [
|
|
`<meta property="al:android:app_name" content="Medium"/>`,
|
|
`<meta property="article:published_time" content="${postDate}"/>`
|
|
];
|
|
|
|
instructions.text = `
|
|
<section class="section-backgroundImage">
|
|
<figure class="graf--layoutFillWidth"></figure>
|
|
</section>
|
|
<section class="section--first" style="font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif; font-size: 64px;">
|
|
If you can see this, your browser is doing something weird with your user agent. <a href="${
|
|
tweet.url
|
|
}">View original post</a>
|
|
</section>
|
|
<article>
|
|
<h1>${tweet.author.name} (@${tweet.author.screen_name})</h1>
|
|
<p>Instant View (✨ Beta) - <a href="${tweet.url}">View original</a></p>
|
|
|
|
<!--blockquote class="twitter-tweet" data-dnt="true"><p lang="en" dir="ltr"> <a href="${
|
|
tweet.url
|
|
}">_</a></blockquote-->
|
|
|
|
${generateTweet(tweet)}
|
|
</article>
|
|
`;
|
|
|
|
return instructions;
|
|
};
|