import { Constants } from "./constants"; import { fetchUsingGuest } from "./fetch"; import { Html } from "./html"; import { colorFromPalette } from "./palette"; import { renderPoll } from "./poll"; export const handleStatus = async (handle: string, id: string, mediaNumber?: number): Promise => { const tweet = await fetchUsingGuest(id); console.log(tweet); /* Try to deep link to mobile apps, just like Twitter does. No idea if this actually works.*/ let headers: string[] = [ ``, ``, ``, ``, ``, ``, ``, ``, ]; // Fallback for if Tweet did not load if (typeof tweet.full_text === "undefined") { headers.push( ``, `` ); return Html.BASE_HTML.format({ lang: '', headers: headers.join(''), tweet: JSON.stringify(tweet), }); } let text = tweet.full_text; const user = tweet.user; const screenName = user?.screen_name || ''; const name = user?.name || ''; const mediaList = tweet.extended_entities?.media || tweet.entities?.media || []; let authorText = 'Twitter'; // This is used to chop off the end if it's like pic.twitter.com or something if (tweet.display_text_range) { const [start, end] = tweet.display_text_range; // We ignore start because it cuts off reply handles // text = text.substring(0, end + 1); } if (tweet.card) { text += await renderPoll(tweet.card); } // Replace t.co links with their full counterparts if (typeof tweet.entities?.urls !== 'undefined') { tweet.entities?.urls.forEach((url: TcoExpansion) => { text = text.replace(url.url, url.expanded_url); }); text = text.replace(/ ?https\:\/\/t\.co\/\w{10}/, ''); } if (typeof tweet.extended_entities?.media === 'undefined' && typeof tweet.entities?.media === 'undefined') { let palette = user?.profile_image_extensions_media_color?.palette; let colorOverride: string = Constants.DEFAULT_COLOR; // for loop for palettes if (palette) { colorOverride = colorFromPalette(palette); } headers.push( ``, ``, ``, ``, ``, ``, `` ); } else { let firstMedia = mediaList[0]; let palette = firstMedia?.ext_media_color?.palette; let colorOverride: string = Constants.DEFAULT_COLOR; let pushedCardType = false; // for loop for palettes if (palette) { colorOverride = colorFromPalette(palette); } headers.push( `` ) const processMedia = (media: TweetMedia) => { if (media.type === 'photo') { headers.push( `` ); if (!pushedCardType) { headers.push(``); pushedCardType = true; } } else if (media.type === 'video') { headers.push( `` ); authorText = encodeURIComponent(text); // Find the variant with the highest bitrate let bestVariant = media.video_info?.variants?.reduce?.((a, b) => (a.bitrate || 0) > (b.bitrate || 0) ? a : b); headers.push( ``, ``, ``, ``, ``, ``, ``, ``, ``, `` ); } } let actualMediaNumber = 1; /* You can specify a specific photo in the URL and we'll pull the correct one, otherwise it falls back to first */ if (typeof mediaNumber === "number" && mediaList[mediaNumber]) { actualMediaNumber = mediaNumber; processMedia(mediaList[mediaNumber]); } else { /* I wish Telegram respected multiple photos in a tweet, and that Discord could do the same for 3rd party providers like us */ // media.forEach(media => processMedia(media)); processMedia(firstMedia); } if (mediaList.length > 1) { authorText = `Photo ${actualMediaNumber} of ${mediaList.length}`; } headers.push( ``, `` ); } /* */ if (tweet.in_reply_to_screen_name) { authorText = `↪ Replying to @${tweet.in_reply_to_screen_name}`; } /* The additional oembed is pulled by Discord to enable improved embeds. Telegram does not use this. */ headers.push(``) console.log(JSON.stringify(tweet)) /* When dealing with a Tweet of unknown lang, fall back to en */ let lang = tweet.lang === 'unk' ? 'en' : tweet.lang || 'en'; return Html.BASE_HTML.format({ lang: `lang="${lang}"`, headers: headers.join('') }); };