diff --git a/client/components/share/mastodon-share-preview/index.jsx b/client/components/share/mastodon-share-preview/index.jsx
index 48d67ef19e666..2482576b3ebbb 100644
--- a/client/components/share/mastodon-share-preview/index.jsx
+++ b/client/components/share/mastodon-share-preview/index.jsx
@@ -13,6 +13,7 @@ export class MastodonSharePreview extends PureComponent {
articleTitle,
articleContent,
imageUrl,
+ media,
message,
hidePostPreview,
} = this.props;
@@ -25,6 +26,7 @@ export class MastodonSharePreview extends PureComponent {
description={ decodeEntities( articleContent ) }
customText={ decodeEntities( message ) }
image={ imageUrl }
+ media={ media }
user={ {
displayName: externalName,
avatarUrl: externalProfilePicture,
diff --git a/packages/social-previews/package.json b/packages/social-previews/package.json
index 1236cbaa27a38..0f674e0b906f3 100644
--- a/packages/social-previews/package.json
+++ b/packages/social-previews/package.json
@@ -1,6 +1,6 @@
{
"name": "@automattic/social-previews",
- "version": "2.0.1-beta.12",
+ "version": "2.0.1-beta.13",
"description": "A suite of components to generate previews for a post for both social and search engines.",
"main": "dist/cjs/index.js",
"module": "dist/esm/index.js",
diff --git a/packages/social-previews/src/helpers.tsx b/packages/social-previews/src/helpers.tsx
index d5369e892c914..24ff9e6d1dbd6 100644
--- a/packages/social-previews/src/helpers.tsx
+++ b/packages/social-previews/src/helpers.tsx
@@ -67,6 +67,13 @@ export const formatTweetDate = new Intl.DateTimeFormat( 'en-US', {
day: 'numeric',
} ).format;
+export const formatMastodonDate = new Intl.DateTimeFormat( 'en-US', {
+ // Result: "Apr 7, 2024", "Dec 31, 2023"
+ month: 'short',
+ day: 'numeric',
+ year: 'numeric',
+} ).format;
+
export type Platform = 'twitter' | 'facebook' | 'linkedin' | 'instagram' | 'mastodon' | 'nextdoor';
type PreviewTextOptions = {
diff --git a/packages/social-previews/src/mastodon-preview/constants.ts b/packages/social-previews/src/mastodon-preview/constants.ts
index bb6ffb41ac0df..bbe2311063805 100644
--- a/packages/social-previews/src/mastodon-preview/constants.ts
+++ b/packages/social-previews/src/mastodon-preview/constants.ts
@@ -1 +1,4 @@
export const DEFAULT_MASTODON_INSTANCE = 'mastodon.social';
+
+export const DEFAULT_AVATAR =
+ '';
diff --git a/packages/social-previews/src/mastodon-preview/post-preview.tsx b/packages/social-previews/src/mastodon-preview/post-preview.tsx
index 3ee5c1d48b49f..2767adda14c8d 100644
--- a/packages/social-previews/src/mastodon-preview/post-preview.tsx
+++ b/packages/social-previews/src/mastodon-preview/post-preview.tsx
@@ -1,28 +1,42 @@
-import { __ } from '@wordpress/i18n';
+import classNames from 'classnames';
import MastodonPostActions from './post/actions';
import MastonPostBody from './post/body';
+import MastodonPostCard from './post/card';
import MastodonPostHeader from './post/header';
import type { MastodonPreviewProps } from './types';
import './styles.scss';
export const MastodonPostPreview: React.FC< MastodonPreviewProps > = ( props ) => {
- const { user, customImage, image } = props;
-
- const img = customImage || image;
+ const { user, media } = props;
return (
- { img && (
-
- ) }
+ { media?.length ? (
+ 1 } ) }
+ >
+ { media.map( ( mediaItem, index ) => (
+
+ { mediaItem.type.startsWith( 'video/' ) ? (
+ // eslint-disable-next-line jsx-a11y/media-has-caption
+
+ ) : (
+
+ ) }
+
+ ) ) }
+
+ ) : null }
+ { ! media?.length ?
: null }
);
diff --git a/packages/social-previews/src/mastodon-preview/post/actions/index.tsx b/packages/social-previews/src/mastodon-preview/post/actions/index.tsx
index 2d0f00aa13679..cb4e56b0c1774 100644
--- a/packages/social-previews/src/mastodon-preview/post/actions/index.tsx
+++ b/packages/social-previews/src/mastodon-preview/post/actions/index.tsx
@@ -1,49 +1,64 @@
-import { __ } from '@wordpress/i18n';
-import MastodonPostIcon from '../icons';
-
import './styles.scss';
const MastodonPostActions: React.FC = () => (
-
- { [
- {
- icon: 'reply',
- // translators: "Reply" action on a Mastodon post
- label: __( 'Reply', 'social-previews' ),
- text: 0,
- },
- {
- icon: 'boost',
- // translators: "Boost" action on a Mastodon post
- label: __( 'Boost', 'social-previews' ),
- },
- {
- icon: 'like',
- // translators: "Favourite" action on a Mastodon post
- label: __( 'Favourite', 'social-previews' ),
- },
- {
- icon: 'bookmark',
- // translators: "Bookmark" action on a Mastodon post
- label: __( 'Bookmark', 'social-previews' ),
- },
- {
- icon: 'more',
- // translators: "More" menu on a Mastodon post
- label: __( 'More', 'social-previews' ),
- },
- ].map( ( { icon, label, text } ) => (
- -
-
-
-
- { ( typeof text === 'number' || text ) && (
- { text }
- ) }
-
- ) ) }
-
+
+
+
+
+
);
diff --git a/packages/social-previews/src/mastodon-preview/post/actions/styles.scss b/packages/social-previews/src/mastodon-preview/post/actions/styles.scss
index 18ccb7d6a9039..57a6e81539127 100644
--- a/packages/social-previews/src/mastodon-preview/post/actions/styles.scss
+++ b/packages/social-previews/src/mastodon-preview/post/actions/styles.scss
@@ -2,34 +2,19 @@
.mastodon-preview__post-actions {
margin-top: 1rem;
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
- ul {
+ & > div {
display: flex;
- justify-content: space-between;
align-items: center;
-
- margin: 0;
- padding: 0;
-
- list-style-type: none;
-
- > li {
- display: flex;
- align-items: center;
- gap: 0.25rem;
-
- margin: 0;
- }
+ color: #606984;
}
-}
-.mastodon-preview__post-icon-wrapper {
- display: flex;
- justify-content: center;
- align-items: center;
-
- width: 24px;
- height: 24px;
+ svg {
+ fill: currentColor;
+ }
}
.mastodon-preview__post-icon-text {
diff --git a/packages/social-previews/src/mastodon-preview/post/card/index.tsx b/packages/social-previews/src/mastodon-preview/post/card/index.tsx
index ab32f364dc004..5944cfc881329 100644
--- a/packages/social-previews/src/mastodon-preview/post/card/index.tsx
+++ b/packages/social-previews/src/mastodon-preview/post/card/index.tsx
@@ -1,5 +1,6 @@
import { __ } from '@wordpress/i18n';
-import { baseDomain, getTitleFromDescription } from '../../../helpers';
+import classNames from 'classnames';
+import { baseDomain, getTitleFromDescription, stripHtmlTags } from '../../../helpers';
import { mastodonTitle } from '../../helpers';
import { MastodonPreviewProps } from '../../types';
@@ -14,20 +15,33 @@ const MastodonPostCard: React.FC< MastodonPreviewProps > = ( {
customImage,
} ) => {
return (
-
+
- { ( image || customImage ) && (
+ { image || customImage ? (
+ ) : (
+
) }
+ { siteName || baseDomain( url ) }
{ mastodonTitle( title ) || getTitleFromDescription( description ) }
- { siteName || baseDomain( url ) }
+ { stripHtmlTags( description ) }
);
diff --git a/packages/social-previews/src/mastodon-preview/post/card/styles.scss b/packages/social-previews/src/mastodon-preview/post/card/styles.scss
index 3196c3bc8ad3f..d019ced28ceed 100644
--- a/packages/social-previews/src/mastodon-preview/post/card/styles.scss
+++ b/packages/social-previews/src/mastodon-preview/post/card/styles.scss
@@ -6,18 +6,24 @@
margin-top: 1rem;
margin-bottom: 1rem;
- border: solid 1px #ccd7e0;
- border-radius: 4px;
+ border: solid 1px #d9e1e8;
+ /* stylelint-disable-next-line scales/radii */
+ border-radius: 8px;
+ overflow: hidden;
color: $mastodon-body-text-color;
+
+ &.has-image {
+ flex-direction: column;
+ }
+
+ &.has-image &-img {
+ width: 100%;
+ }
}
.mastodon-preview__card-img {
- width: 60px;
min-height: 60px;
- border-start-start-radius: inherit;
- border-end-start-radius: inherit;
-
img {
display: block;
@@ -28,25 +34,43 @@
border-start-start-radius: inherit;
border-end-start-radius: inherit;
}
+
+ &--fallback {
+ aspect-ratio: 1;
+ background: #c0cdd9;
+ position: relative;
+ width: 120px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ }
}
.mastodon-preview__card-text {
display: flex;
flex-direction: column;
- gap: 0.25rem;
-
- padding: 0.625rem 0.5rem;
+ gap: 0.5rem;
+ padding: 1rem;
+ overflow: hidden;
}
.mastodon-preview__card-title {
color: #282c37;
- font-size: 0.875rem;
+ /* stylelint-disable-next-line scales/font-sizes */
+ font-size: 1.187rem;
font-weight: 700;
- line-height: 1.3;
+ line-height: 1.4;
}
.mastodon-preview__card-site {
- /* stylelint-disable-next-line scales/font-sizes */
- font-size: 0.8125rem;
+ font-size: 0.875rem;
+ display: block;
+}
+
+.mastodon-preview__card-description {
+ font-size: 0.875rem;
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
}
diff --git a/packages/social-previews/src/mastodon-preview/post/header/index.tsx b/packages/social-previews/src/mastodon-preview/post/header/index.tsx
index 6205e28a28a40..b2e9c8d5a45e8 100644
--- a/packages/social-previews/src/mastodon-preview/post/header/index.tsx
+++ b/packages/social-previews/src/mastodon-preview/post/header/index.tsx
@@ -1,6 +1,7 @@
import { __ } from '@wordpress/i18n';
-import { DEFAULT_MASTODON_INSTANCE } from '../../constants';
-import MastodonPostIcon from '../icons';
+import { formatMastodonDate } from '../../../helpers';
+import { DEFAULT_AVATAR, DEFAULT_MASTODON_INSTANCE } from '../../constants';
+import { GlobeIcon } from '../icons';
import type { MastodonPreviewProps } from '../../types';
import './styles.scss';
@@ -13,7 +14,7 @@ const MastodonPostHeader: React.FC< Props > = ( { user } ) => {
return (
- { avatarUrl &&
}
+
{ displayName ||
@@ -21,16 +22,13 @@ const MastodonPostHeader: React.FC< Props > = ( { user } ) => {
__( 'anonymous-user', 'social-previews' ) }
- { address?.replace( `@${ DEFAULT_MASTODON_INSTANCE }`, '' ) }
+ { address?.replace( `@${ DEFAULT_MASTODON_INSTANCE }`, '' ) || '@username' }
-
- {
- // translators: time elapsed since post was published (1 hour)
- __( '1h', 'social-previews' )
- }
+
+ { formatMastodonDate( new Date() ) }
);
diff --git a/packages/social-previews/src/mastodon-preview/post/icons/index.tsx b/packages/social-previews/src/mastodon-preview/post/icons/index.tsx
index 40e2d0b8c9860..cf897d30392b2 100644
--- a/packages/social-previews/src/mastodon-preview/post/icons/index.tsx
+++ b/packages/social-previews/src/mastodon-preview/post/icons/index.tsx
@@ -1,7 +1,13 @@
-import './styles.scss';
-
-const MastodonPostIcon: React.FC< { name: string } > = ( { name } ) => (
-
-);
-
-export default MastodonPostIcon;
+export function GlobeIcon() {
+ return (
+
+ );
+}
diff --git a/packages/social-previews/src/mastodon-preview/post/icons/styles.scss b/packages/social-previews/src/mastodon-preview/post/icons/styles.scss
deleted file mode 100644
index 1d475cc7faf41..0000000000000
--- a/packages/social-previews/src/mastodon-preview/post/icons/styles.scss
+++ /dev/null
@@ -1,50 +0,0 @@
-.mastodon-preview__post-icon {
- display: inline-block;
-
- /* stylelint-disable-next-line function-url-quotes */
- background-image: url("");
- background-repeat: no-repeat;
- background-size: 99px 17px;
-}
-
-.mastodon-preview__post-icon-reply {
- width: 19px;
- height: 17px;
-
- background-position: 0 0;
-}
-
-.mastodon-preview__post-icon-boost {
- width: 21px;
- height: 13px;
-
- background-position: -19px 0;
-}
-
-.mastodon-preview__post-icon-like {
- width: 17px;
- height: 17px;
-
- background-position: -40px 0;
-}
-
-.mastodon-preview__post-icon-bookmark {
- width: 14px;
- height: 16px;
-
- background-position: -57px 0;
-}
-
-.mastodon-preview__post-icon-more {
- width: 15px;
- height: 4px;
-
- background-position: -71px 0;
-}
-
-.mastodon-preview__post-icon-globe {
- width: 13px;
- height: 14px;
-
- background-position: -86px 0;
-}
diff --git a/packages/social-previews/src/mastodon-preview/styles.scss b/packages/social-previews/src/mastodon-preview/styles.scss
index c1bbcf3fac326..68132865d073b 100644
--- a/packages/social-previews/src/mastodon-preview/styles.scss
+++ b/packages/social-previews/src/mastodon-preview/styles.scss
@@ -40,3 +40,34 @@
aspect-ratio: 16 / 9;
object-fit: cover;
}
+
+.mastodon-preview__media {
+
+ margin-top: 8px;
+ min-height: 64px;
+ /* stylelint-disable-next-line scales/radii */
+ border-radius: 8px;
+
+ &.as-grid {
+ display: grid;
+ gap: 2px;
+ grid-template-columns: 50% 50%;
+ width: 100%;
+ overflow: hidden;
+ }
+
+ &-item {
+ display: flex;
+ border: 0;
+ /* stylelint-disable-next-line scales/radii */
+ border-radius: 8px;
+ overflow: hidden;
+
+ img,
+ video {
+ width: 100%;
+ object-fit: cover;
+ }
+ }
+
+}
diff --git a/packages/social-previews/src/tumblr-preview/previews.tsx b/packages/social-previews/src/tumblr-preview/previews.tsx
index 72444f907b722..76fc48b7913e7 100644
--- a/packages/social-previews/src/tumblr-preview/previews.tsx
+++ b/packages/social-previews/src/tumblr-preview/previews.tsx
@@ -14,6 +14,8 @@ export const TumblrPreviews: React.FC< TumblrPreviewsProps > = ( {
hidePostPreview,
...props
} ) => {
+ const hasMedia = !! props.media?.length;
+
return (
{ ! hidePostPreview && (
@@ -27,7 +29,7 @@ export const TumblrPreviews: React.FC< TumblrPreviewsProps > = ( {
{ __( 'This is what your social post will look like on Tumblr:', 'social-previews' ) }
-
+ { hasMedia ?
:
}
) }
{ ! hideLinkPreview && (