mirror of
https://github.com/CompeyDev/fxtwitter-docker.git
synced 2025-04-05 18:40:56 +01:00
Updated Instant View author section
This commit is contained in:
parent
b143797f77
commit
202472fa34
6 changed files with 102 additions and 30 deletions
|
@ -13,7 +13,7 @@ import { isGraphQLTweet } from '../utils/graphql';
|
||||||
and using it to create FixTweet's streamlined API responses */
|
and using it to create FixTweet's streamlined API responses */
|
||||||
const populateTweetProperties = async (
|
const populateTweetProperties = async (
|
||||||
tweet: GraphQLTweet,
|
tweet: GraphQLTweet,
|
||||||
conversation: any, // TimelineBlobPartial,
|
conversation: TweetResultsByRestIdResult, // TimelineBlobPartial,
|
||||||
language: string | undefined
|
language: string | undefined
|
||||||
// eslint-disable-next-line sonarjs/cognitive-complexity
|
// eslint-disable-next-line sonarjs/cognitive-complexity
|
||||||
): Promise<APITweet> => {
|
): Promise<APITweet> => {
|
||||||
|
@ -44,6 +44,8 @@ const populateTweetProperties = async (
|
||||||
const graphQLUser = tweet.core.user_results.result;
|
const graphQLUser = tweet.core.user_results.result;
|
||||||
const apiUser = convertToApiUser(graphQLUser);
|
const apiUser = convertToApiUser(graphQLUser);
|
||||||
|
|
||||||
|
console.log(JSON.stringify(graphQLUser))
|
||||||
|
|
||||||
/* Populating a lot of the basics */
|
/* Populating a lot of the basics */
|
||||||
apiTweet.url = `${Constants.TWITTER_ROOT}/${apiUser.screen_name}/status/${tweet.rest_id}`;
|
apiTweet.url = `${Constants.TWITTER_ROOT}/${apiUser.screen_name}/status/${tweet.rest_id}`;
|
||||||
apiTweet.id = tweet.rest_id;
|
apiTweet.id = tweet.rest_id;
|
||||||
|
@ -53,15 +55,24 @@ const populateTweetProperties = async (
|
||||||
name: apiUser.name,
|
name: apiUser.name,
|
||||||
screen_name: apiUser.screen_name,
|
screen_name: apiUser.screen_name,
|
||||||
avatar_url: (apiUser.avatar_url || '').replace('_normal', '_200x200') || '',
|
avatar_url: (apiUser.avatar_url || '').replace('_normal', '_200x200') || '',
|
||||||
avatar_color: '0000FF' /* colorFromPalette(
|
avatar_color: null,
|
||||||
tweet.user?.profile_image_extensions_media_color?.palette || []
|
banner_url: apiUser.banner_url || '',
|
||||||
),*/,
|
description: apiUser.description || '',
|
||||||
banner_url: apiUser.banner_url || ''
|
location: apiUser.location || '',
|
||||||
|
url: apiUser.url || '',
|
||||||
|
followers: apiUser.followers,
|
||||||
|
following: apiUser.following,
|
||||||
|
joined: apiUser.joined,
|
||||||
|
tweets: apiUser.tweets,
|
||||||
|
likes: apiUser.likes,
|
||||||
|
protected: apiUser.protected,
|
||||||
|
birthday: apiUser.birthday,
|
||||||
|
website: apiUser.website,
|
||||||
};
|
};
|
||||||
apiTweet.replies = tweet.legacy.reply_count;
|
apiTweet.replies = tweet.legacy.reply_count;
|
||||||
apiTweet.retweets = tweet.legacy.retweet_count;
|
apiTweet.retweets = tweet.legacy.retweet_count;
|
||||||
apiTweet.likes = tweet.legacy.favorite_count;
|
apiTweet.likes = tweet.legacy.favorite_count;
|
||||||
apiTweet.color = apiTweet.author.avatar_color;
|
apiTweet.color = null;
|
||||||
apiTweet.twitter_card = 'tweet';
|
apiTweet.twitter_card = 'tweet';
|
||||||
apiTweet.created_at = tweet.legacy.created_at;
|
apiTweet.created_at = tweet.legacy.created_at;
|
||||||
apiTweet.created_timestamp = new Date(tweet.legacy.created_at).getTime() / 1000;
|
apiTweet.created_timestamp = new Date(tweet.legacy.created_at).getTime() / 1000;
|
||||||
|
|
|
@ -41,6 +41,16 @@ export const convertToApiUser = (user: GraphQLUser): APIUser => {
|
||||||
if (typeof birthdate.month === 'number') apiUser.birthday.month = birthdate.month;
|
if (typeof birthdate.month === 'number') apiUser.birthday.month = birthdate.month;
|
||||||
if (typeof birthdate.year === 'number') apiUser.birthday.year = birthdate.year;
|
if (typeof birthdate.year === 'number') apiUser.birthday.year = birthdate.year;
|
||||||
}
|
}
|
||||||
|
const website = user.legacy.entities?.url?.urls?.[0];
|
||||||
|
|
||||||
|
if (website) {
|
||||||
|
apiUser.website = {
|
||||||
|
url: website.expanded_url,
|
||||||
|
display_url: website.display_url
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
apiUser.website = null;
|
||||||
|
}
|
||||||
|
|
||||||
return apiUser;
|
return apiUser;
|
||||||
};
|
};
|
||||||
|
@ -61,7 +71,7 @@ const populateUserProperties = async (
|
||||||
export const userAPI = async (
|
export const userAPI = async (
|
||||||
username: string,
|
username: string,
|
||||||
event: FetchEvent,
|
event: FetchEvent,
|
||||||
flags?: InputFlags
|
// flags?: InputFlags
|
||||||
): Promise<UserAPIResponse> => {
|
): Promise<UserAPIResponse> => {
|
||||||
const userResponse = await fetchUser(username, event);
|
const userResponse = await fetchUser(username, event);
|
||||||
if (!userResponse || !Object.keys(userResponse).length) {
|
if (!userResponse || !Object.keys(userResponse).length) {
|
||||||
|
|
|
@ -113,7 +113,8 @@ export const handleStatus = async (
|
||||||
|
|
||||||
/* Base headers included in all responses */
|
/* Base headers included in all responses */
|
||||||
const headers = [
|
const headers = [
|
||||||
`<link rel="canonical" href="https://twitter.com/${tweet.author.screen_name}/status/${tweet.id}"/>`,
|
`<link rel="canonical" href="${Constants.TWITTER_ROOT}/${tweet.author.screen_name}/status/${tweet.id}"/>`,
|
||||||
|
`<meta property="og:url" content="${Constants.TWITTER_ROOT}/${tweet.author.screen_name}/status/${tweet.id}"/>`,
|
||||||
`<meta property="theme-color" content="${tweet.color}"/>`,
|
`<meta property="theme-color" content="${tweet.color}"/>`,
|
||||||
`<meta property="twitter:card" content="${
|
`<meta property="twitter:card" content="${
|
||||||
tweet.quote?.twitter_card || tweet.twitter_card
|
tweet.quote?.twitter_card || tweet.twitter_card
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
/* eslint-disable no-irregular-whitespace */
|
||||||
import { Constants } from '../constants';
|
import { Constants } from '../constants';
|
||||||
import { getSocialTextIV } from '../helpers/author';
|
import { getSocialTextIV } from '../helpers/author';
|
||||||
import { sanitizeText } from '../helpers/utils';
|
import { sanitizeText } from '../helpers/utils';
|
||||||
|
@ -44,6 +45,13 @@ const generateTweetMedia = (tweet: APITweet): string => {
|
||||||
// return `${hh}:${min} - ${yyyy}/${mm}/${dd}`;
|
// 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 htmlifyLinks = (input: string): string => {
|
||||||
const urlPattern = /\bhttps?:\/\/\S+/g;
|
const urlPattern = /\bhttps?:\/\/\S+/g;
|
||||||
return input.replace(urlPattern, url => {
|
return input.replace(urlPattern, url => {
|
||||||
|
@ -93,6 +101,50 @@ function getTranslatedText(tweet: APITweet, isQuote = false): string | null {
|
||||||
|
|
||||||
const notApplicableComment = '<!-- N/A -->';
|
const notApplicableComment = '<!-- N/A -->';
|
||||||
|
|
||||||
|
// 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 generateTweetFooter = (tweet: APITweet, isQuote = false): string => {
|
||||||
|
const { author } = tweet;
|
||||||
|
|
||||||
|
let description = author.description;
|
||||||
|
description = htmlifyLinks(description);
|
||||||
|
description = htmlifyHashtags(description);
|
||||||
|
description = populateUserLinks(tweet, description);
|
||||||
|
|
||||||
|
return `
|
||||||
|
<p>${getSocialTextIV(tweet)}</p>
|
||||||
|
<!-- Embed profile picture, display name, and screen name in table -->
|
||||||
|
<h3>About author</h3>
|
||||||
|
${
|
||||||
|
!isQuote
|
||||||
|
? `<table>
|
||||||
|
<img src="${author.avatar_url?.replace('_200x200', '_400x400')}" alt="${
|
||||||
|
author.name
|
||||||
|
}'s profile picture" />
|
||||||
|
<h2>${author.name}</h2>
|
||||||
|
<p><a href="${author.url}">@${author.screen_name}</a></p>
|
||||||
|
<p><b>${description}</b></p>
|
||||||
|
<p>${author.location ? `📌 ${author.location}` : ''}${' '
|
||||||
|
}${author.website ? `🔗 <a href=${author.website.url}>${author.website.display_url}</a>` : ''}${' '
|
||||||
|
}${author.joined ? `📆 ${formatDate(new Date(author.joined))}` : ''}</p>
|
||||||
|
<p>${truncateSocialCount(author.following)} <b>Following</b>
|
||||||
|
${truncateSocialCount(author.followers)} <b>Followers</b>
|
||||||
|
${truncateSocialCount(author.tweets)} <b>Posts</b></p>
|
||||||
|
</table>`
|
||||||
|
: ''
|
||||||
|
}`;
|
||||||
|
}
|
||||||
|
|
||||||
const generateTweet = (tweet: APITweet, isQuote = false): string => {
|
const generateTweet = (tweet: APITweet, isQuote = false): string => {
|
||||||
let text = paragraphify(sanitizeText(tweet.text), isQuote);
|
let text = paragraphify(sanitizeText(tweet.text), isQuote);
|
||||||
text = htmlifyLinks(text);
|
text = htmlifyLinks(text);
|
||||||
|
@ -102,19 +154,6 @@ const generateTweet = (tweet: APITweet, isQuote = false): string => {
|
||||||
const translatedText = getTranslatedText(tweet, isQuote);
|
const translatedText = getTranslatedText(tweet, isQuote);
|
||||||
|
|
||||||
return `<!-- Telegram Instant View -->
|
return `<!-- Telegram Instant View -->
|
||||||
<!-- Embed profile picture, display name, and screen name in table -->
|
|
||||||
${
|
|
||||||
!isQuote
|
|
||||||
? `<table>
|
|
||||||
<img src="${tweet.author.avatar_url?.replace('_200x200', '_400x400')}" alt="${
|
|
||||||
tweet.author.name
|
|
||||||
}'s profile picture" />
|
|
||||||
<h2>${tweet.author.name}</h2>
|
|
||||||
<p>@${tweet.author.screen_name}</p>
|
|
||||||
<p>${getSocialTextIV(tweet)}</p>
|
|
||||||
</table>`
|
|
||||||
: ''
|
|
||||||
}
|
|
||||||
${
|
${
|
||||||
isQuote
|
isQuote
|
||||||
? `
|
? `
|
||||||
|
@ -123,14 +162,15 @@ const generateTweet = (tweet: APITweet, isQuote = false): string => {
|
||||||
`
|
`
|
||||||
: ''
|
: ''
|
||||||
}
|
}
|
||||||
|
<!-- Embed Tweet media -->
|
||||||
|
${generateTweetMedia(tweet)}
|
||||||
<!-- Translated text (if applicable) -->
|
<!-- Translated text (if applicable) -->
|
||||||
${translatedText ? translatedText : notApplicableComment}
|
${translatedText ? translatedText : notApplicableComment}
|
||||||
<!-- Embed Tweet text -->
|
<!-- Embed Tweet text -->
|
||||||
${text}
|
${text}
|
||||||
<!-- Embed Tweet media -->
|
|
||||||
${generateTweetMedia(tweet)}
|
|
||||||
<!-- Embedded quote tweet -->
|
<!-- Embedded quote tweet -->
|
||||||
${!isQuote && tweet.quote ? generateTweet(tweet.quote, true) : notApplicableComment}
|
${!isQuote && tweet.quote ? generateTweet(tweet.quote, true) : notApplicableComment}
|
||||||
|
${generateTweetFooter(tweet)}
|
||||||
<br>${!isQuote ? `<a href="${tweet.url}">View original</a>` : notApplicableComment}
|
<br>${!isQuote ? `<a href="${tweet.url}">View original</a>` : notApplicableComment}
|
||||||
`;
|
`;
|
||||||
};
|
};
|
||||||
|
|
9
src/types/twitterTypes.d.ts
vendored
9
src/types/twitterTypes.d.ts
vendored
|
@ -239,6 +239,14 @@ type GraphQLUser = {
|
||||||
indices: [0, 23];
|
indices: [0, 23];
|
||||||
}[];
|
}[];
|
||||||
};
|
};
|
||||||
|
url?: {
|
||||||
|
urls?: {
|
||||||
|
display_url: string; // "about.twitter.com",
|
||||||
|
expanded_url: string; // "https://about.twitter.com/",
|
||||||
|
url: string; // "https://t.co/DAtOo6uuHk",
|
||||||
|
indices: [0, 23];
|
||||||
|
}[];
|
||||||
|
}
|
||||||
};
|
};
|
||||||
fast_followers_count: 0;
|
fast_followers_count: 0;
|
||||||
favourites_count: number; // 126708,
|
favourites_count: number; // 126708,
|
||||||
|
@ -502,6 +510,7 @@ type GraphQLTweetFoundResponse = {
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
type TweetResultsByRestIdResult = {
|
type TweetResultsByRestIdResult = {
|
||||||
|
guestToken?: string;
|
||||||
errors?: unknown[];
|
errors?: unknown[];
|
||||||
data?: {
|
data?: {
|
||||||
tweetResult?: {
|
tweetResult?: {
|
||||||
|
|
15
src/types/types.d.ts
vendored
15
src/types/types.d.ts
vendored
|
@ -87,10 +87,6 @@ interface BaseUser {
|
||||||
banner_url?: string;
|
banner_url?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface APITweetAuthor extends BaseUser {
|
|
||||||
avatar_color: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface APIExternalMedia {
|
interface APIExternalMedia {
|
||||||
type: 'video';
|
type: 'video';
|
||||||
url: string;
|
url: string;
|
||||||
|
@ -155,7 +151,7 @@ interface APITweet {
|
||||||
quote?: APITweet;
|
quote?: APITweet;
|
||||||
poll?: APIPoll;
|
poll?: APIPoll;
|
||||||
translation?: APITranslate;
|
translation?: APITranslate;
|
||||||
author: APITweetAuthor;
|
author: APIUser;
|
||||||
|
|
||||||
media?: {
|
media?: {
|
||||||
external?: APIExternalMedia;
|
external?: APIExternalMedia;
|
||||||
|
@ -179,17 +175,22 @@ interface APITweet {
|
||||||
}
|
}
|
||||||
|
|
||||||
interface APIUser extends BaseUser {
|
interface APIUser extends BaseUser {
|
||||||
|
// verified: 'legacy' | 'blue'| 'business' | 'government';
|
||||||
|
// verified_label: string;
|
||||||
description: string;
|
description: string;
|
||||||
location: string;
|
location: string;
|
||||||
url: string;
|
url: string;
|
||||||
|
avatar_color?: string | null;
|
||||||
protected: boolean;
|
protected: boolean;
|
||||||
// verified: 'legacy' | 'blue'| 'business' | 'government';
|
|
||||||
// verified_label: string;
|
|
||||||
followers: number;
|
followers: number;
|
||||||
following: number;
|
following: number;
|
||||||
tweets: number;
|
tweets: number;
|
||||||
likes: number;
|
likes: number;
|
||||||
joined: string;
|
joined: string;
|
||||||
|
website: {
|
||||||
|
url: string;
|
||||||
|
display_url: string;
|
||||||
|
} | null;
|
||||||
birthday: {
|
birthday: {
|
||||||
day?: number;
|
day?: number;
|
||||||
month?: number;
|
month?: number;
|
||||||
|
|
Loading…
Add table
Reference in a new issue