diff --git a/src/api/status.ts b/src/api/status.ts
index d554f1b..ce71e6c 100644
--- a/src/api/status.ts
+++ b/src/api/status.ts
@@ -75,14 +75,17 @@ const populateTweetProperties = async (
mediaList.forEach(media => {
const mediaObject = processMedia(media);
if (mediaObject) {
+
+ apiTweet.media = apiTweet.media || {};
+ apiTweet.media.all = apiTweet.media?.all || [];
+ apiTweet.media.all.push(mediaObject);
+
if (mediaObject.type === 'photo') {
apiTweet.twitter_card = 'summary_large_image';
- apiTweet.media = apiTweet.media || {};
apiTweet.media.photos = apiTweet.media.photos || [];
apiTweet.media.photos.push(mediaObject);
} else if (mediaObject.type === 'video' || mediaObject.type === 'gif') {
apiTweet.twitter_card = 'player';
- apiTweet.media = apiTweet.media || {};
apiTweet.media.videos = apiTweet.media.videos || [];
apiTweet.media.videos.push(mediaObject);
}
diff --git a/src/embed/status.ts b/src/embed/status.ts
index f3c56e2..42ca7a2 100644
--- a/src/embed/status.ts
+++ b/src/embed/status.ts
@@ -43,6 +43,13 @@ export const handleStatus = async (
};
}
+ let overrideMedia: APIPhoto | APIVideo | undefined;
+
+ // Check if mediaNumber exists, and if that media exists in tweet.media.all. If it does, we'll store overrideMedia variable
+ if (mediaNumber && tweet.media && tweet.media.all && tweet.media.all[mediaNumber - 1]) {
+ overrideMedia = tweet.media.all[mediaNumber - 1];
+ }
+
/* If there was any errors fetching the Tweet, we'll return it */
switch (api.code) {
case 401:
@@ -56,13 +63,22 @@ export const handleStatus = async (
/* Catch direct media request (d.fxtwitter.com, or .mp4 / .jpg) */
if (flags?.direct && tweet.media) {
let redirectUrl: string | null = null;
- if (tweet.media.videos) {
- const { videos } = tweet.media;
- redirectUrl = (videos[(mediaNumber || 1) - 1] || videos[0]).url;
- } else if (tweet.media.photos) {
- const { photos } = tweet.media;
- redirectUrl = (photos[(mediaNumber || 1) - 1] || photos[0]).url;
+ const all = tweet.media.all || [];
+ // if (tweet.media.videos) {
+ // const { videos } = tweet.media;
+ // redirectUrl = (videos[(mediaNumber || 1) - 1] || videos[0]).url;
+ // } else if (tweet.media.photos) {
+ // const { photos } = tweet.media;
+ // redirectUrl = (photos[(mediaNumber || 1) - 1] || photos[0]).url;
+ // }
+
+ const selectedMedia = all[(mediaNumber || 1) - 1];
+ if (selectedMedia) {
+ redirectUrl = selectedMedia.url;
+ } else if (all.length > 0) {
+ redirectUrl = all[0].url;
}
+
if (redirectUrl) {
return { response: Response.redirect(redirectUrl, 302) };
}
@@ -128,15 +144,16 @@ export const handleStatus = async (
Twitter supports multiple videos in a Tweet now. But we have no mechanism to embed more than one.
You can still use /video/:number to get a specific video. Otherwise, it'll pick the first. */
- if (tweet.media?.videos) {
+ if (tweet.media?.videos || overrideMedia?.type === 'video') {
authorText = newText || '';
if (tweet?.translation) {
authorText = tweet.translation?.text || '';
}
- const { videos } = tweet.media;
- const video = videos[(mediaNumber || 1) - 1];
+ const videos = tweet.media?.videos;
+ const all = tweet.media?.all || [];
+ const video = overrideMedia as APIVideo || videos?.[(mediaNumber || 1) - 1];
/* This fix is specific to Discord not wanting to render videos that are too large,
or rendering low quality videos too small.
@@ -157,10 +174,10 @@ export const handleStatus = async (
/* Like photos when picking a specific one (not using mosaic),
we'll put an indicator if there are more than one video */
- if (videos.length > 1) {
+ if (all && all.length > 1) {
const videoCounter = Strings.VIDEO_COUNT.format({
- number: String(videos.indexOf(video) + 1),
- total: String(videos.length)
+ number: String(all.indexOf(video) + 1),
+ total: String(all.length)
});
authorText =
@@ -192,13 +209,12 @@ export const handleStatus = async (
}
/* This Tweet has one or more photos to render */
- if (tweet.media?.photos) {
- const { photos } = tweet.media;
- let photo: APIPhoto | APIMosaicPhoto = photos[(mediaNumber || 1) - 1];
+ if (tweet.media?.photos || overrideMedia?.type === 'photo') {
+ let photo: APIPhoto | APIMosaicPhoto = overrideMedia as APIPhoto || tweet.media?.photos?.[0];
/* If there isn't a specified media number and we have a
mosaic response, we'll render it using mosaic */
- if (typeof mediaNumber !== 'number' && tweet.media.mosaic) {
+ if (!overrideMedia && tweet.media?.mosaic) {
photo = {
/* Include dummy height/width for TypeScript reasons. We have a check to make sure we don't use these later. */
height: 0,
@@ -209,10 +225,11 @@ export const handleStatus = async (
};
/* If mosaic isn't available or the link calls for a specific photo,
we'll indicate which photo it is out of the total */
- } else if (photos.length > 1) {
+ } else if (tweet.media?.all && tweet.media.all.length > 1) {
+ const { all } = tweet.media;
const photoCounter = Strings.PHOTO_COUNT.format({
- number: String(photos.indexOf(photo) + 1),
- total: String(photos.length)
+ number: String(all.indexOf(photo) + 1),
+ total: String(all.length)
});
authorText =
@@ -233,7 +250,7 @@ export const handleStatus = async (
``
);
- if (!tweet.media.mosaic) {
+ if (!tweet.media?.mosaic) {
headers.push(
``,
``,
diff --git a/src/types/types.d.ts b/src/types/types.d.ts
index 476e3bd..574c2c6 100644
--- a/src/types/types.d.ts
+++ b/src/types/types.d.ts
@@ -140,6 +140,7 @@ interface APITweet {
external?: APIExternalMedia;
photos?: APIPhoto[];
videos?: APIVideo[];
+ all?: (APIPhoto | APIVideo)[];
mosaic?: APIMosaicPhoto;
};