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

View file

@ -495,10 +495,24 @@ export const constructTwitterThread = async (
code: 200
};
threadStatuses.forEach(async status => {
socialThread.thread?.push(
(await buildAPITwitterStatus(c, status, undefined, true, false)) as APITwitterStatus
);
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);
}));
// Sort socialThread.thread by id converted to bigint
socialThread.thread?.sort((a, b) => {
const aId = BigInt(a.id);
const bId = BigInt(b.id);
if (aId < bId) {
return -1;
}
if (aId > bId) {
return 1;
}
return 0;
});
return socialThread;

View file

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

View file

@ -68,7 +68,7 @@ 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="${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 { author } = status;
const generateStatusFooter = (status: APIStatus, isQuote = false, author: APIUser): string => {
let description = author.description;
description = htmlifyLinks(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);
text = htmlifyLinks(text);
text = htmlifyHashtags(text);
@ -180,20 +178,23 @@ const generateStatus = (status: APIStatus, isQuote = false): string => {
<!-- Embed Status text -->
${text}
<!-- Embedded quote status -->
${!isQuote && status.quote ? generateStatus(status.quote, true) : notApplicableComment}
${!isQuote ? generateStatusFooter(status) : ''}
<br>${!isQuote ? `<a href="${status.url}">View original post</a>` : notApplicableComment}
${!isQuote && status.quote ? generateStatus(status.quote, author, true) : notApplicableComment}
`.format({
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 => {
console.log('Generating Instant View...');
const { status, flags } = properties;
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();
@ -210,6 +211,8 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
: ``
];
console.log('thread', thread?.thread)
instructions.text = `
<section class="section-backgroundImage">
<figure class="graf--layoutFillWidth"></figure>
@ -224,7 +227,9 @@ export const renderInstantView = (properties: RenderProperties): ResponseInstruc
<sub><a href="${status.url}">View original</a></sub>
<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>`;
return instructions;

View file

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