/* eslint-disable no-irregular-whitespace */
import { Constants } from '../constants';
import { getSocialTextIV } from '../helpers/socialproof';
import { sanitizeText } from '../helpers/utils';
import { Strings } from '../strings';
const populateUserLinks = (status: APIStatus, 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,
`${match}`
);
});
return text;
};
const generateStatusMedia = (status: APIStatus): string => {
let media = '';
if (status.media?.all?.length) {
status.media.all.forEach(mediaItem => {
switch (mediaItem.type) {
case 'photo':
// eslint-disable-next-line no-case-declarations
const { altText } = mediaItem as APIPhoto;
media += `
`.format({
altText: altText ? `alt="${altText}"` : '',
url: mediaItem.url
});
break;
case 'video':
media += ``;
break;
case 'gif':
media += ``;
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 formatDate = (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');
return `${yyyy}/${mm}/${dd}`;
};
const htmlifyLinks = (input: string): string => {
const urlPattern = /\bhttps?:\/\/\S+/g;
return input.replace(urlPattern, url => {
return `${url}`;
});
};
const htmlifyHashtags = (input: string): string => {
const hashtagPattern = /#([a-zA-Z_]\w*)/g;
return input.replace(hashtagPattern, (match, hashtag) => {
const encodedHashtag = encodeURIComponent(hashtag);
return ` ${match} `;
});
};
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');
}
function getTranslatedText(status: APITwitterStatus, isQuote = false): string | null {
if (!status.translation) {
return null;
}
let text = paragraphify(sanitizeText(status.translation?.text), isQuote);
text = htmlifyLinks(text);
text = htmlifyHashtags(text);
text = populateUserLinks(status, text);
const formatText =
status.translation.target_lang === 'en'
? Strings.TRANSLATE_TEXT.format({
language: status.translation.source_lang_en
})
: Strings.TRANSLATE_TEXT_INTL.format({
source: status.translation.source_lang.toUpperCase(),
destination: status.translation.target_lang.toUpperCase()
});
return `
${formatText}
${text}Original
`;
}
const notApplicableComment = '';
// 1100 -> 1.1K, 1100000 -> 1.1M
const truncateSocialCount = (count: number): string => {
if (count >= 1000000) {
return `${(count / 1000000).toFixed(1)}M`;
} else if (count >= 1000) {
return `${(count / 1000).toFixed(1)}K`;
} else {
return String(count);
}
};
const generateStatusFooter = (status: APIStatus, isQuote = false, author: APIUser): string => {
let description = author.description;
description = htmlifyLinks(description);
description = htmlifyHashtags(description);
description = populateUserLinks(status, description);
return `
{socialText}
{viewOriginal}
{aboutSection}
`.format({
socialText: getSocialTextIV(status as APITwitterStatus) || '',
viewOriginal: !isQuote
? `View original post`
: notApplicableComment,
aboutSection: isQuote
? ''
: `About author
{pfp}
${author.name}
@${author.screen_name}
${description}
{location} {website} {joined}
{following} Following
{followers} Followers
{statuses} Posts
`.format({
pfp: `
`,
location: author.location ? `📌 ${author.location}` : '',
website: author.website
? `🔗 ${author.website.display_url}`
: '',
joined: author.joined ? `📆 ${formatDate(new Date(author.joined))}` : '',
following: truncateSocialCount(author.following),
followers: truncateSocialCount(author.followers),
statuses: truncateSocialCount(author.statuses)
})
});
};
const generateStatus = (status: APIStatus, author: APIUser, isQuote = false): string => {
let text = paragraphify(sanitizeText(status.text), isQuote);
text = htmlifyLinks(text);
text = htmlifyHashtags(text);
text = populateUserLinks(status, text);
const translatedText = getTranslatedText(status as APITwitterStatus, isQuote);
return `
{quoteHeader}
${generateStatusMedia(status)}
${translatedText ? translatedText : notApplicableComment}
${text}
${!isQuote && status.quote ? generateStatus(status.quote, author, true) : notApplicableComment}
`.format({
quoteHeader: isQuote
? ``
: ''
});
};
export const renderInstantView = (properties: RenderProperties): ResponseInstructions => {
console.log('Generating Instant View...');
const { status, thread, flags } = properties;
const instructions: ResponseInstructions = { addHeaders: [] };
if (!status) {
throw new Error('Status is undefined');
}
/* Use ISO date for Medium template */
const statusDate = new Date(status.created_at).toISOString();
/* Pretend to be Medium to allow Instant View to work.
Thanks to https://nikstar.me/post/instant-view/ for the help!
If you work for Telegram and want to let us build our own templates
contact me https://t.me/dangeredwolf */
instructions.addHeaders = [
``,
``,
flags?.archive
? ``
: ``
];
console.log('thread', thread?.thread)
instructions.text = `
${
flags?.archive
? `${Constants.BRANDING_NAME} archive`
: 'If you can see this, your browser is doing something weird with your user agent.'
} View original post
View original
${status.author.name} (@${status.author.screen_name})
${thread?.thread?.map(status => generateStatus(status, thread?.author ?? status.author, false)).join('')}
${generateStatusFooter(status, false, thread?.author ?? status.author)}
${`View original post`}
`;
return instructions;
};