Implement thread unrolling in Instant View

This commit is contained in:
dangered wolf 2024-01-14 19:37:43 -05:00
parent 5dc760638a
commit 90e9c3a1eb
No known key found for this signature in database
GPG key ID: 41E4D37680ED8B58
5 changed files with 44 additions and 20 deletions

View file

@ -35,9 +35,8 @@ export const handleStatus = async (
let fetchWithThreads = false; let fetchWithThreads = false;
/* TODO: Enable actually pulling threads once we can actually do something with them */ if (c.req.header('user-agent')?.includes('Telegram') && !flags?.direct && flags.instantViewUnrollThreads) {
if (c.req.header('user-agent')?.includes('Telegram') && !flags?.direct) { fetchWithThreads = true;
fetchWithThreads = false;
} }
const thread = await constructTwitterThread( const thread = await constructTwitterThread(
@ -48,6 +47,8 @@ export const handleStatus = async (
flags?.api ?? false flags?.api ?? false
); );
console.log('twitterThread', thread);
const status = thread?.status as APITwitterStatus; const status = thread?.status as APITwitterStatus;
const api = { const api = {
@ -191,6 +192,7 @@ export const handleStatus = async (
try { try {
const instructions = renderInstantView({ const instructions = renderInstantView({
status: status, status: status,
thread: thread,
text: newText, text: newText,
flags: flags flags: flags
}); });

View file

@ -494,11 +494,25 @@ export const constructTwitterThread = async (
author: author, author: author,
code: 200 code: 200
}; };
await Promise.all(threadStatuses.map(async status => {
console.log('Processing status for', status)
const builtStatus = await buildAPITwitterStatus(c, status, undefined, true, false) as APITwitterStatus;
console.log('builtStatus', builtStatus);
socialThread.thread?.push(builtStatus);
}));
threadStatuses.forEach(async status => { // Sort socialThread.thread by id converted to bigint
socialThread.thread?.push( socialThread.thread?.sort((a, b) => {
(await buildAPITwitterStatus(c, status, undefined, true, false)) as APITwitterStatus const aId = BigInt(a.id);
); const bId = BigInt(b.id);
if (aId < bId) {
return -1;
}
if (aId > bId) {
return 1;
}
return 0;
}); });
return socialThread; return socialThread;

View file

@ -94,7 +94,9 @@ export const buildAPITwitterStatus = async (
delete apiStatus.author.global_screen_name; delete apiStatus.author.global_screen_name;
} else { } else {
apiStatus.reposts = status.legacy.retweet_count; apiStatus.reposts = status.legacy.retweet_count;
apiStatus.author.global_screen_name = apiUser.global_screen_name; if (!threadPiece) {
apiStatus.author.global_screen_name = apiUser.global_screen_name;
}
} }
apiStatus.likes = status.legacy.favorite_count; apiStatus.likes = status.legacy.favorite_count;
apiStatus.embed_card = 'tweet'; apiStatus.embed_card = 'tweet';

View file

@ -68,7 +68,7 @@ const htmlifyHashtags = (input: string): string => {
const hashtagPattern = /#([a-zA-Z_]\w*)/g; const hashtagPattern = /#([a-zA-Z_]\w*)/g;
return input.replace(hashtagPattern, (match, hashtag) => { return input.replace(hashtagPattern, (match, hashtag) => {
const encodedHashtag = encodeURIComponent(hashtag); const encodedHashtag = encodeURIComponent(hashtag);
return `<a href="${Constants.TWITTER_ROOT}/hashtag/${encodedHashtag}?src=hashtag_click">${match}</a>`; return ` <a href="${Constants.TWITTER_ROOT}/hashtag/${encodedHashtag}?src=hashtag_click">${match}</a> `;
}); });
}; };
@ -117,9 +117,7 @@ const truncateSocialCount = (count: number): string => {
} }
}; };
const generateStatusFooter = (status: APIStatus, isQuote = false): string => { const generateStatusFooter = (status: APIStatus, isQuote = false, author: APIUser): string => {
const { author } = status;
let description = author.description; let description = author.description;
description = htmlifyLinks(description); description = htmlifyLinks(description);
description = htmlifyHashtags(description); description = htmlifyHashtags(description);
@ -163,7 +161,7 @@ const generateStatusFooter = (status: APIStatus, isQuote = false): string => {
}); });
}; };
const generateStatus = (status: APIStatus, isQuote = false): string => { const generateStatus = (status: APIStatus, author: APIUser, isQuote = false): string => {
let text = paragraphify(sanitizeText(status.text), isQuote); let text = paragraphify(sanitizeText(status.text), isQuote);
text = htmlifyLinks(text); text = htmlifyLinks(text);
text = htmlifyHashtags(text); text = htmlifyHashtags(text);
@ -180,20 +178,23 @@ const generateStatus = (status: APIStatus, isQuote = false): string => {
<!-- Embed Status text --> <!-- Embed Status text -->
${text} ${text}
<!-- Embedded quote status --> <!-- Embedded quote status -->
${!isQuote && status.quote ? generateStatus(status.quote, true) : notApplicableComment} ${!isQuote && status.quote ? generateStatus(status.quote, author, true) : notApplicableComment}
${!isQuote ? generateStatusFooter(status) : ''}
<br>${!isQuote ? `<a href="${status.url}">View original post</a>` : notApplicableComment}
`.format({ `.format({
quoteHeader: isQuote quoteHeader: isQuote
? `<h4><a href="${status.url}">Quoting</a> ${status.author.name} (<a href="${Constants.TWITTER_ROOT}/${status.author.screen_name}">@${status.author.screen_name}</a>)</h4>` ? `<h4><a href="${status.url}">Quoting</a> ${author.name} (<a href="${Constants.TWITTER_ROOT}/${author.screen_name}">@${author.screen_name}</a>)</h4>`
: '' : ''
}); });
}; };
export const renderInstantView = (properties: RenderProperties): ResponseInstructions => { export const renderInstantView = (properties: RenderProperties): ResponseInstructions => {
console.log('Generating Instant View...'); console.log('Generating Instant View...');
const { status, flags } = properties; const { status, thread, flags } = properties;
const instructions: ResponseInstructions = { addHeaders: [] }; const instructions: ResponseInstructions = { addHeaders: [] };
if (!status) {
throw new Error('Status is undefined');
}
/* Use ISO date for Medium template */ /* Use ISO date for Medium template */
const statusDate = new Date(status.created_at).toISOString(); const statusDate = new Date(status.created_at).toISOString();
@ -210,6 +211,8 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
: `` : ``
]; ];
console.log('thread', thread?.thread)
instructions.text = ` instructions.text = `
<section class="section-backgroundImage"> <section class="section-backgroundImage">
<figure class="graf--layoutFillWidth"></figure> <figure class="graf--layoutFillWidth"></figure>
@ -224,7 +227,9 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
<sub><a href="${status.url}">View original</a></sub> <sub><a href="${status.url}">View original</a></sub>
<h1>${status.author.name} (@${status.author.screen_name})</h1> <h1>${status.author.name} (@${status.author.screen_name})</h1>
${generateStatus(status)} ${thread?.thread?.map(status => generateStatus(status, thread?.author ?? status.author, false)).join('')}
${generateStatusFooter(status, false, thread?.author ?? status.author)}
<br>${`<a href="${status.url}">View original post</a>`}
</article>`; </article>`;
return instructions; return instructions;

View file

@ -28,7 +28,8 @@ interface ResponseInstructions {
} }
interface RenderProperties { interface RenderProperties {
status: APITwitterStatus; status?: APITwitterStatus;
thread?: SocialThread;
siteText?: string; siteText?: string;
authorText?: string; authorText?: string;
engagementText?: string; engagementText?: string;