mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 18:40:56 +01:00
Implement thread unrolling in Instant View
This commit is contained in:
parent
5dc760638a
commit
90e9c3a1eb
5 changed files with 44 additions and 20 deletions
|
@ -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
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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';
|
||||||
|
|
|
@ -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;
|
||||||
|
|
3
src/types/types.d.ts
vendored
3
src/types/types.d.ts
vendored
|
@ -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;
|
||||||
|
|
Loading…
Add table
Reference in a new issue