mirror of
https://github.com/GenderDysphoria/GenderDysphoria.fyi.git
synced 2025-01-31 07:16:17 +00:00
170 lines
4.9 KiB
JavaScript
170 lines
4.9 KiB
JavaScript
var twemoji = require('twemoji' );
|
|
const { deepPick, has } = require('./util');
|
|
const path = require('path');
|
|
|
|
const schema = {
|
|
id_str: true,
|
|
created_at: true,
|
|
user: {
|
|
screen_name: true,
|
|
avatar: true,
|
|
name_html: true,
|
|
verified: true,
|
|
protected: true,
|
|
},
|
|
html: true,
|
|
quoted_status_id_str: true,
|
|
entities: { media: [ {
|
|
type: true,
|
|
media_url_https: true,
|
|
dimensions: true,
|
|
video_info: { variants: [ {
|
|
url: true,
|
|
content_type: true,
|
|
} ] },
|
|
} ] },
|
|
media: true,
|
|
in_reply_to_status_id_str: true,
|
|
};
|
|
|
|
var entityProcessors = {
|
|
hashtags (tags, tweet) {
|
|
tags.forEach((tagObj) => {
|
|
tweet.html = tweet.html.replace('#' + tagObj.text, `<a href="https://twitter.com/hashtag/{tagObj.text}" class="hashtag">#${tagObj.text}</a>`);
|
|
});
|
|
},
|
|
|
|
symbols (/* symbols, tweet */) {
|
|
|
|
},
|
|
|
|
user_mentions (users, tweet) {
|
|
users.forEach((userObj) => {
|
|
var regex = new RegExp('@' + userObj.screen_name, 'gi' );
|
|
tweet.html = tweet.html.replace(regex, `<a href="https://twitter.com/${userObj.screen_name}" class="mention">@${userObj.screen_name}</a>`);
|
|
});
|
|
},
|
|
|
|
urls (urls, tweet) {
|
|
urls.forEach(({ url, expanded_url, display_url }) => {
|
|
const isQT = tweet.quoted_status_permalink && url === tweet.quoted_status_permalink.url;
|
|
const className = isQT ? 'quoted-tweet' : 'url';
|
|
tweet.html = tweet.html.replace(url, isQT ? '' : `<a href="${expanded_url}" class="${className}">${display_url}</a>`);
|
|
});
|
|
},
|
|
|
|
media (media, tweet) {
|
|
media.forEach((m) => {
|
|
tweet.html = tweet.html.replace(m.url, '');
|
|
let width, height;
|
|
|
|
if (has(m, 'video_info.aspect_ratio')) {
|
|
[ width, height ] = m.video_info.aspect_ratio;
|
|
} else if (has(m, 'sizes.medium')) {
|
|
({ width, height } = m.sizes.medium);
|
|
}
|
|
|
|
if (width && height) {
|
|
const ratioH = Math.round((height / width) * 100);
|
|
const ratioW = Math.round((width / height) * 100);
|
|
let orientation = 'wide';
|
|
if (ratioH > 100) {
|
|
orientation = 'tall';
|
|
} else if (ratioH === 100) {
|
|
orientation = 'square';
|
|
}
|
|
|
|
m.dimensions = {
|
|
width,
|
|
height,
|
|
ratioH,
|
|
ratioW,
|
|
orientation,
|
|
};
|
|
}
|
|
return;
|
|
});
|
|
},
|
|
};
|
|
|
|
module.exports = exports = function (tweets) {
|
|
return tweets.length ? tweets.map(parseTweet) : parseTweet(tweets);
|
|
|
|
function parseTweet (tweet) {
|
|
// clone the tweet so we're not altering the original
|
|
tweet = JSON.parse(JSON.stringify(tweet));
|
|
|
|
tweet.user.avatar = {
|
|
input: tweet.user.profile_image_url_https,
|
|
output: `tweets/${tweet.user.screen_name}.jpg`,
|
|
cache: `twitter-avatars/${tweet.user.screen_name}.jpg`,
|
|
};
|
|
|
|
tweet.media = [
|
|
tweet.user.avatar,
|
|
];
|
|
|
|
// Copying text value to a new property html. The final output will be set to this property
|
|
tweet.html = (tweet.full_text || tweet.text)
|
|
.split(/(\r\n|\n\r|\r|\n)+/)
|
|
.map((s) => s.trim() && '<p>' + s + '</p>')
|
|
.filter(Boolean)
|
|
.join('');
|
|
|
|
if (tweet.quoted_status) {
|
|
tweet.quoted_status = parseTweet(tweet.quoted_status);
|
|
}
|
|
|
|
if (has(tweet, 'entities.media') && has(tweet, 'extended_entities.media')) {
|
|
tweet.entities.media = tweet.extended_entities.media.map((media) => {
|
|
media = { ...media };
|
|
if (media.media_url_https) {
|
|
const mediaItem = {
|
|
input: media.media_url_https,
|
|
output: `tweets/${tweet.id_str}/${path.basename(media.media_url_https)}`,
|
|
cache: `twitter-entities/${tweet.id_str}/${path.basename(media.media_url_https)}`,
|
|
};
|
|
if (media.type === 'photo') mediaItem.input += '?name=medium';
|
|
tweet.media.push(mediaItem);
|
|
media.media_url_https = '/' + mediaItem.output;
|
|
}
|
|
|
|
if (media.video_info && media.video_info.variants) {
|
|
media.video_info.variants = media.video_info.variants.map((variant) => {
|
|
if (!variant.url || !variant.bitrate) return variant;
|
|
|
|
const fname = path.basename(variant.url).split('?')[0];
|
|
const mediaItem = {
|
|
input: variant.url,
|
|
output: `tweets/${tweet.id_str}/${fname}`,
|
|
cache: `twitter-entities/${tweet.id_str}/${fname}`,
|
|
};
|
|
tweet.media.push(mediaItem);
|
|
variant.url = '/' + mediaItem.output;
|
|
|
|
return variant;
|
|
});
|
|
}
|
|
|
|
return media;
|
|
});
|
|
|
|
delete tweet.extended_entities;
|
|
}
|
|
|
|
// Process entities
|
|
if (Object.getOwnPropertyNames(tweet.entities).length) {
|
|
for (let [ entityType, entity ] of Object.entries(tweet.entities)) { // eslint-disable-line prefer-const
|
|
entityProcessors[entityType](entity, tweet);
|
|
}
|
|
}
|
|
|
|
// Process Emoji's
|
|
tweet.html = twemoji.parse(tweet.html);
|
|
tweet.user.name_html = twemoji.parse(tweet.user.name);
|
|
|
|
return deepPick(tweet, schema);
|
|
}
|
|
|
|
};
|