Source code
Revision control
Copy as Markdown
Other Tools
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
import { actionCreators as ac, actionTypes as at } from "common/Actions.mjs";
import { DSImage } from "../DSImage/DSImage.jsx";
import { DSLinkMenu } from "../DSLinkMenu/DSLinkMenu";
import { ImpressionStats } from "../../DiscoveryStreamImpressionStats/ImpressionStats";
import { getActiveCardSize } from "../../../lib/utils";
import React from "react";
import { SafeAnchor } from "../SafeAnchor/SafeAnchor";
import {
DSContextFooter,
SponsorLabel,
DSMessageFooter,
} from "../DSContextFooter/DSContextFooter.jsx";
import { DSThumbsUpDownButtons } from "../DSThumbsUpDownButtons/DSThumbsUpDownButtons.jsx";
import { FluentOrText } from "../../FluentOrText/FluentOrText.jsx";
import { connect } from "react-redux";
import { LinkMenuOptions } from "content-src/lib/link-menu-options";
const READING_WPM = 220;
const PREF_OHTTP_MERINO = "discoverystream.merino-provider.ohttp.enabled";
const PREF_OHTTP_UNIFIED_ADS = "unifiedAds.ohttp.enabled";
const PREF_CONTEXTUAL_ADS = "discoverystream.sections.contextualAds.enabled";
const PREF_INFERRED_PERSONALIZATION_SYSTEM =
"discoverystream.sections.personalization.inferred.enabled";
const PREF_INFERRED_PERSONALIZATION_USER =
"discoverystream.sections.personalization.inferred.user.enabled";
const PREF_SECTIONS_ENABLED = "discoverystream.sections.enabled";
const PREF_FAVICONS_ENABLED = "discoverystream.publisherFavicon.enabled";
/**
* READ TIME FROM WORD COUNT
* @param {int} wordCount number of words in an article
* @returns {int} number of words per minute in minutes
*/
export function readTimeFromWordCount(wordCount) {
if (!wordCount) {
return false;
}
return Math.ceil(parseInt(wordCount, 10) / READING_WPM);
}
export const DSSource = ({
source,
timeToRead,
newSponsoredLabel,
context,
sponsor,
sponsored_by_override,
icon_src,
refinedCardsLayout,
}) => {
// refinedCard styles will have a larger favicon size
const faviconSize = refinedCardsLayout ? 20 : 16;
// First try to display sponsored label or time to read here.
if (newSponsoredLabel) {
// If we can display something for spocs, do so.
if (sponsored_by_override || sponsor || context) {
return (
<SponsorLabel
context={context}
sponsor={sponsor}
sponsored_by_override={sponsored_by_override}
newSponsoredLabel="new-sponsored-label"
/>
);
}
}
// If we are not a spoc, and can display a time to read value.
if (source && timeToRead) {
return (
<p className="source clamp time-to-read">
<FluentOrText
message={{
id: `newtab-label-source-read-time`,
values: { source, timeToRead },
}}
/>
</p>
);
}
// Otherwise display a default source.
return (
<div className="source-wrapper">
{icon_src && (
<img src={icon_src} height={faviconSize} width={faviconSize} alt="" />
)}
<p className="source clamp">{source}</p>
</div>
);
};
export const DefaultMeta = ({
source,
title,
excerpt,
timeToRead,
newSponsoredLabel,
context,
context_type,
sponsor,
sponsored_by_override,
ctaButtonVariant,
dispatch,
mayHaveSectionsCards,
mayHaveThumbsUpDown,
onThumbsUpClick,
onThumbsDownClick,
isListCard,
state,
format,
topic,
isSectionsCard,
showTopics,
icon_src,
refinedCardsLayout,
}) => {
const shouldHaveThumbs =
!isListCard &&
format !== "rectangle" &&
mayHaveSectionsCards &&
mayHaveThumbsUpDown;
const shouldHaveFooterSection =
isSectionsCard && (shouldHaveThumbs || showTopics);
return (
<div className="meta">
<div className="info-wrap">
{ctaButtonVariant !== "variant-b" &&
format !== "rectangle" &&
!refinedCardsLayout && (
<DSSource
source={source}
timeToRead={timeToRead}
newSponsoredLabel={newSponsoredLabel}
context={context}
sponsor={sponsor}
sponsored_by_override={sponsored_by_override}
icon_src={icon_src}
/>
)}
<h3 className="title clamp">
{format === "rectangle" ? "Sponsored" : title}
</h3>
{format === "rectangle" ? (
<p className="excerpt clamp">
Sponsored content supports our mission to build a better web.
</p>
) : (
excerpt && <p className="excerpt clamp">{excerpt}</p>
)}
</div>
{!isListCard &&
format !== "rectangle" &&
!mayHaveSectionsCards &&
mayHaveThumbsUpDown &&
!refinedCardsLayout && (
<DSThumbsUpDownButtons
onThumbsDownClick={onThumbsDownClick}
onThumbsUpClick={onThumbsUpClick}
sponsor={sponsor}
isThumbsDownActive={state.isThumbsDownActive}
isThumbsUpActive={state.isThumbsUpActive}
/>
)}
{(shouldHaveFooterSection || refinedCardsLayout) && (
<div className="sections-card-footer">
{refinedCardsLayout &&
format !== "rectangle" &&
format !== "spoc" && (
<DSSource
source={source}
timeToRead={timeToRead}
newSponsoredLabel={newSponsoredLabel}
context={context}
sponsor={sponsor}
sponsored_by_override={sponsored_by_override}
icon_src={icon_src}
refinedCardsLayout={refinedCardsLayout}
/>
)}
{(shouldHaveThumbs || refinedCardsLayout) && (
<DSThumbsUpDownButtons
onThumbsDownClick={onThumbsDownClick}
onThumbsUpClick={onThumbsUpClick}
sponsor={sponsor}
isThumbsDownActive={state.isThumbsDownActive}
isThumbsUpActive={state.isThumbsUpActive}
refinedCardsLayout={refinedCardsLayout}
/>
)}
{showTopics && (
<span
className="ds-card-topic"
data-l10n-id={`newtab-topic-label-${topic}`}
/>
)}
</div>
)}
{!newSponsoredLabel && (
<DSContextFooter
context_type={context_type}
context={context}
sponsor={sponsor}
sponsored_by_override={sponsored_by_override}
cta_button_variant={ctaButtonVariant}
source={source}
dispatch={dispatch}
mayHaveSectionsCards={mayHaveSectionsCards}
/>
)}
{/* Sponsored label is normally in the way of any message.
newSponsoredLabel cards sponsored label is moved to just under the thumbnail,
so we can display both, so we specifically don't pass in context. */}
{newSponsoredLabel && (
<DSMessageFooter context_type={context_type} context={null} />
)}
</div>
);
};
export class _DSCard extends React.PureComponent {
constructor(props) {
super(props);
this.onLinkClick = this.onLinkClick.bind(this);
this.doesLinkTopicMatchSelectedTopic =
this.doesLinkTopicMatchSelectedTopic.bind(this);
this.onMenuUpdate = this.onMenuUpdate.bind(this);
this.onMenuShow = this.onMenuShow.bind(this);
this.onThumbsUpClick = this.onThumbsUpClick.bind(this);
this.onThumbsDownClick = this.onThumbsDownClick.bind(this);
const refinedCardsLayout =
this.props.Prefs.values["discoverystream.refinedCardsLayout.enabled"];
this.setContextMenuButtonHostRef = element => {
this.contextMenuButtonHostElement = element;
};
this.setPlaceholderRef = element => {
this.placeholderElement = element;
};
this.state = {
isSeen: false,
isThumbsUpActive: false,
isThumbsDownActive: false,
};
// If this is for the about:home startup cache, then we always want
// to render the DSCard, regardless of whether or not its been seen.
if (props.App.isForStartupCache.App) {
this.state.isSeen = true;
}
// We want to choose the optimal thumbnail for the underlying DSImage, but
// want to do it in a performant way. The breakpoints used in the
// CSS of the page are, unfortuntely, not easy to retrieve without
// causing a style flush. To avoid that, we hardcode them here.
//
// The values chosen here were the dimensions of the card thumbnails as
// computed by getBoundingClientRect() for each type of viewport width
// across both high-density and normal-density displays.
this.standardCardImageSizes = [
{
mediaMatcher: "default",
width: 296,
height: 148,
},
];
this.listCardImageSizes = [
{
mediaMatcher: "(min-width: 1122px)",
width: 75,
height: 75,
},
{
mediaMatcher: "default",
width: 50,
height: 50,
},
];
this.sectionsCardImagesSizes = {
small: {
width: 110,
height: 117,
},
medium: {
width: 300,
height: refinedCardsLayout ? 160 : 150,
},
large: {
width: 190,
height: 250,
},
};
this.sectionsColumnMediaMatcher = {
1: "default",
2: "(min-width: 724px)",
3: "(min-width: 1122px)",
4: "(min-width: 1390px)",
};
}
getSectionImageSize(column, size) {
const cardImageSize = {
mediaMatcher: this.sectionsColumnMediaMatcher[column],
width: this.sectionsCardImagesSizes[size].width,
height: this.sectionsCardImagesSizes[size].height,
};
return cardImageSize;
}
doesLinkTopicMatchSelectedTopic() {
// Edge case for clicking on a card when topic selections have not be set
if (!this.props.selectedTopics) {
return "not-set";
}
// Edge case the topic of the card is not one of the available topics
if (!this.props.availableTopics.includes(this.props.topic)) {
return "topic-not-selectable";
}
if (this.props.selectedTopics.includes(this.props.topic)) {
return "true";
}
return "false";
}
onLinkClick() {
const matchesSelectedTopic = this.doesLinkTopicMatchSelectedTopic();
if (this.props.dispatch) {
if (this.props.isFakespot) {
this.props.dispatch(
ac.DiscoveryStreamUserEvent({
event: "FAKESPOT_CLICK",
value: {
product_id: this.props.id,
category: this.props.category || "",
},
})
);
} else {
this.props.dispatch(
ac.DiscoveryStreamUserEvent({
event: "CLICK",
source: this.props.type.toUpperCase(),
action_position: this.props.pos,
value: {
event_source: "card",
card_type: this.props.flightId ? "spoc" : "organic",
recommendation_id: this.props.recommendation_id,
tile_id: this.props.id,
...(this.props.shim && this.props.shim.click
? { shim: this.props.shim.click }
: {}),
fetchTimestamp: this.props.fetchTimestamp,
firstVisibleTimestamp: this.props.firstVisibleTimestamp,
corpus_item_id: this.props.corpus_item_id,
scheduled_corpus_item_id: this.props.scheduled_corpus_item_id,
recommended_at: this.props.recommended_at,
received_rank: this.props.received_rank,
topic: this.props.topic,
features: this.props.features,
matches_selected_topic: matchesSelectedTopic,
selected_topics: this.props.selectedTopics,
is_list_card: this.props.isListCard,
...(this.props.format
? { format: this.props.format }
: {
format: getActiveCardSize(
window.innerWidth,
this.props.sectionsClassNames,
this.props.section,
this.props.flightId
),
}),
...(this.props.section
? {
section: this.props.section,
section_position: this.props.sectionPosition,
is_section_followed: this.props.sectionFollowed,
layout_name: this.props.sectionLayoutName,
}
: {}),
},
})
);
this.props.dispatch(
ac.ImpressionStats({
source: this.props.type.toUpperCase(),
click: 0,
window_inner_width: this.props.windowObj.innerWidth,
window_inner_height: this.props.windowObj.innerHeight,
tiles: [
{
id: this.props.id,
pos: this.props.pos,
...(this.props.shim && this.props.shim.click
? { shim: this.props.shim.click }
: {}),
type: this.props.flightId ? "spoc" : "organic",
recommendation_id: this.props.recommendation_id,
topic: this.props.topic,
selected_topics: this.props.selectedTopics,
is_list_card: this.props.isListCard,
...(this.props.format
? { format: this.props.format }
: {
format: getActiveCardSize(
window.innerWidth,
this.props.sectionsClassNames,
this.props.section,
this.props.flightId
),
}),
...(this.props.section
? {
section: this.props.section,
section_position: this.props.sectionPosition,
is_section_followed: this.props.sectionFollowed,
}
: {}),
},
],
})
);
}
}
}
onThumbsUpClick(event) {
event.stopPropagation();
event.preventDefault();
// Toggle active state for thumbs up button to show CSS animation
const currentState = this.state.isThumbsUpActive;
// If thumbs up has been clicked already, do nothing.
if (currentState) {
return;
}
this.setState({ isThumbsUpActive: !currentState });
// Record thumbs up telemetry event
this.props.dispatch(
ac.DiscoveryStreamUserEvent({
event: "POCKET_THUMBS_UP",
source: "THUMBS_UI",
value: {
recommendation_id: this.props.recommendation_id,
tile_id: this.props.id,
corpus_item_id: this.props.corpus_item_id,
scheduled_corpus_item_id: this.props.scheduled_corpus_item_id,
recommended_at: this.props.recommended_at,
received_rank: this.props.received_rank,
thumbs_up: true,
thumbs_down: false,
topic: this.props.topic,
format: getActiveCardSize(
window.innerWidth,
this.props.sectionsClassNames,
this.props.section,
false // (thumbs up/down only exist on organic content)
),
...(this.props.section
? {
section: this.props.section,
section_position: this.props.sectionPosition,
is_section_followed: this.props.sectionFollowed,
}
: {}),
},
})
);
// Show Toast
this.props.dispatch(
ac.OnlyToOneContent(
{
type: at.SHOW_TOAST_MESSAGE,
data: {
showNotifications: true,
toastId: "thumbsUpToast",
},
},
"ActivityStream:Content"
)
);
}
onThumbsDownClick(event) {
event.stopPropagation();
event.preventDefault();
// Toggle active state for thumbs down button to show CSS animation
const currentState = this.state.isThumbsDownActive;
this.setState({ isThumbsDownActive: !currentState });
// Run dismiss event after 0.5 second delay
if (
this.props.dispatch &&
this.props.type &&
this.props.id &&
this.props.url
) {
const index = this.props.pos;
const source = this.props.type.toUpperCase();
const spocData = {
url: this.props.url,
guid: this.props.id,
type: "CardGrid",
card_type: "organic",
recommendation_id: this.props.recommendation_id,
tile_id: this.props.id,
corpus_item_id: this.props.corpus_item_id,
scheduled_corpus_item_id: this.props.scheduled_corpus_item_id,
recommended_at: this.props.recommended_at,
received_rank: this.props.received_rank,
};
const blockUrlOption = LinkMenuOptions.BlockUrl(spocData, index, source);
const { action, impression, userEvent } = blockUrlOption;
setTimeout(() => {
this.props.dispatch(action);
this.props.dispatch(
ac.DiscoveryStreamUserEvent({
event: userEvent,
source,
action_position: index,
})
);
}, 500);
if (impression) {
this.props.dispatch(impression);
}
// Record thumbs down telemetry event
this.props.dispatch(
ac.DiscoveryStreamUserEvent({
event: "POCKET_THUMBS_DOWN",
source: "THUMBS_UI",
value: {
recommendation_id: this.props.recommendation_id,
tile_id: this.props.id,
corpus_item_id: this.props.corpus_item_id,
scheduled_corpus_item_id: this.props.scheduled_corpus_item_id,
recommended_at: this.props.recommended_at,
received_rank: this.props.received_rank,
thumbs_up: false,
thumbs_down: true,
topic: this.props.topic,
format: getActiveCardSize(
window.innerWidth,
this.props.sectionsClassNames,
this.props.section,
false // (thumbs up/down only exist on organic content)
),
...(this.props.section
? {
section: this.props.section,
section_position: this.props.sectionPosition,
is_section_followed: this.props.sectionFollowed,
}
: {}),
},
})
);
// Show Toast
this.props.dispatch(
ac.OnlyToOneContent(
{
type: at.SHOW_TOAST_MESSAGE,
data: {
showNotifications: true,
toastId: "thumbsDownToast",
},
},
"ActivityStream:Content"
)
);
}
}
onMenuUpdate(showContextMenu) {
if (!showContextMenu) {
const dsLinkMenuHostDiv = this.contextMenuButtonHostElement;
if (dsLinkMenuHostDiv) {
dsLinkMenuHostDiv.classList.remove("active", "last-item");
}
}
}
async onMenuShow() {
const dsLinkMenuHostDiv = this.contextMenuButtonHostElement;
if (dsLinkMenuHostDiv) {
// Force translation so we can be sure it's ready before measuring.
await this.props.windowObj.document.l10n.translateFragment(
dsLinkMenuHostDiv
);
if (this.props.windowObj.scrollMaxX > 0) {
dsLinkMenuHostDiv.classList.add("last-item");
}
dsLinkMenuHostDiv.classList.add("active");
}
}
onSeen(entries) {
if (this.state) {
const entry = entries.find(e => e.isIntersecting);
if (entry) {
if (this.placeholderElement) {
this.observer.unobserve(this.placeholderElement);
}
// Stop observing since element has been seen
this.setState({
isSeen: true,
});
}
}
}
onIdleCallback() {
if (!this.state.isSeen) {
// To improve responsiveness without impacting performance,
// we start rendering stories on idle.
// To reduce the number of requests for secure OHTTP images,
// we skip idle-time loading.
if (!this.secureImage) {
if (this.observer && this.placeholderElement) {
this.observer.unobserve(this.placeholderElement);
}
this.setState({
isSeen: true,
});
}
}
}
componentDidMount() {
this.idleCallbackId = this.props.windowObj.requestIdleCallback(
this.onIdleCallback.bind(this)
);
if (this.placeholderElement) {
this.observer = new IntersectionObserver(this.onSeen.bind(this));
this.observer.observe(this.placeholderElement);
}
}
componentWillUnmount() {
// Remove observer on unmount
if (this.observer && this.placeholderElement) {
this.observer.unobserve(this.placeholderElement);
}
if (this.idleCallbackId) {
this.props.windowObj.cancelIdleCallback(this.idleCallbackId);
}
}
// Wraps the image URL with the moz-cached-ohttp:// protocol.
// This enables Firefox to load resources over Oblivious HTTP (OHTTP),
// providing privacy-preserving resource loading.
// Applied only when inferred personalization is enabled.
secureImageURL(url) {
}
getRawImageSrc() {
let rawImageSrc = "";
// There is no point in fetching images for startup cache.
if (!this.props.App.isForStartupCache.App) {
rawImageSrc = this.props.raw_image_src;
}
return rawImageSrc;
}
getFaviconSrc() {
let faviconSrc = "";
const faviconEnabled = this.props.Prefs.values[PREF_FAVICONS_ENABLED];
// There is no point in fetching favicons for startup cache.
if (
!this.props.App.isForStartupCache.App &&
faviconEnabled &&
this.props.icon_src
) {
faviconSrc = this.props.icon_src;
if (this.secureImage) {
faviconSrc = this.secureImageURL(this.props.icon_src);
}
}
return faviconSrc;
}
get secureImage() {
const { Prefs, flightId } = this.props;
let ohttpEnabled = false;
if (flightId) {
ohttpEnabled =
Prefs.values[PREF_CONTEXTUAL_ADS] &&
Prefs.values[PREF_OHTTP_UNIFIED_ADS];
} else {
ohttpEnabled = Prefs.values[PREF_OHTTP_MERINO];
}
const inferredPersonalizationUser =
Prefs.values[PREF_INFERRED_PERSONALIZATION_USER];
const inferredPersonalizationSystem =
Prefs.values[PREF_INFERRED_PERSONALIZATION_SYSTEM];
const inferredPersonalization =
inferredPersonalizationSystem && inferredPersonalizationUser;
const ohttpImagesEnabled = Prefs.values.ohttpImagesConfig?.enabled;
const includeTopStoriesSection =
Prefs.values.ohttpImagesConfig?.includeTopStoriesSection;
const sectionsEnabled = Prefs.values[PREF_SECTIONS_ENABLED];
const nonPersonalizedSections = ["top_stories_section"];
const sectionPersonalized =
!nonPersonalizedSections.includes(this.props.section) ||
includeTopStoriesSection;
const secureImage =
sectionsEnabled &&
ohttpImagesEnabled &&
ohttpEnabled &&
sectionPersonalized &&
inferredPersonalization;
return secureImage;
}
renderImage({ sizes = [], classNames = "" } = {}) {
const { Prefs } = this.props;
const rawImageSrc = this.getRawImageSrc();
const smartCrop = Prefs.values["images.smart"];
return (
<DSImage
extraClassNames={`img ${classNames}`}
source={this.props.image_src}
rawSource={rawImageSrc}
sizes={sizes}
url={this.props.url}
title={this.props.title}
isRecentSave={this.props.isRecentSave}
alt_text={this.props.alt_text}
smartCrop={smartCrop}
secureImage={this.secureImage}
/>
);
}
renderSectionCardImages() {
const { sectionsCardImageSizes } = this.props;
const columns = ["1", "2", "3", "4"];
const images = [];
for (const column of columns) {
const size = sectionsCardImageSizes[column];
const sizes = [this.getSectionImageSize(column, size)];
const image = this.renderImage({ sizes, classNames: `image-${column}` });
images.push(image);
}
return <>{images}</>;
}
render() {
const {
isRecentSave,
DiscoveryStream,
Prefs,
isListCard,
isFakespot,
mayHaveSectionsCards,
format,
} = this.props;
const refinedCardsLayout =
Prefs.values["discoverystream.refinedCardsLayout.enabled"];
const refinedCardsClassName = refinedCardsLayout ? `refined-cards` : ``;
if (this.props.placeholder || !this.state.isSeen) {
// placeholder-seen is used to ensure the loading animation is only used if the card is visible.
const placeholderClassName = this.state.isSeen ? `placeholder-seen` : ``;
let placeholderElements = (
<>
<div className="placeholder-image placeholder-fill" />
<div className="placeholder-label placeholder-fill" />
<div className="placeholder-header placeholder-fill" />
<div className="placeholder-description placeholder-fill" />
</>
);
if (refinedCardsLayout) {
placeholderElements = (
<>
<div className="placeholder-image placeholder-fill" />
<div className="placeholder-description placeholder-fill" />
<div className="placeholder-header placeholder-fill" />
</>
);
}
return (
<div
className={`ds-card placeholder ${placeholderClassName} ${
isListCard ? "list-card-placeholder" : ""
} ${refinedCardsClassName}`}
ref={this.setPlaceholderRef}
>
{placeholderElements}
</div>
);
}
let source = this.props.source || this.props.publisher;
if (!source) {
try {
source = new URL(this.props.url).hostname;
} catch (e) {}
}
const {
hideDescriptions,
compactImages,
imageGradient,
newSponsoredLabel,
titleLines = 3,
descLines = 3,
readTime: displayReadTime,
} = DiscoveryStream;
const sectionsEnabled = Prefs.values[PREF_SECTIONS_ENABLED];
// Refined cards have their own excerpt hiding logic.
// We can ignore hideDescriptions if we are in sections and refined cards.
const excerpt =
!hideDescriptions || (sectionsEnabled && refinedCardsLayout)
? this.props.excerpt
: "";
let timeToRead;
if (displayReadTime) {
timeToRead =
this.props.time_to_read || readTimeFromWordCount(this.props.word_count);
}
const ctaButtonEnabled = this.props.ctaButtonSponsors?.includes(
this.props.sponsor?.toLowerCase()
);
let ctaButtonVariant = "";
if (ctaButtonEnabled) {
ctaButtonVariant = this.props.ctaButtonVariant;
}
let ctaButtonVariantClassName = ctaButtonVariant;
const ctaButtonClassName = ctaButtonEnabled ? `ds-card-cta-button` : ``;
const compactImagesClassName = compactImages ? `ds-card-compact-image` : ``;
const imageGradientClassName = imageGradient
? `ds-card-image-gradient`
: ``;
const listCardClassName = isListCard ? `list-feed-card` : ``;
const fakespotClassName = isFakespot ? `fakespot` : ``;
const sectionsCardsClassName = [
mayHaveSectionsCards ? `sections-card-ui` : ``,
this.props.sectionsClassNames,
].join(" ");
const titleLinesName = `ds-card-title-lines-${titleLines}`;
const descLinesClassName = `ds-card-desc-lines-${descLines}`;
const isMediumRectangle = format === "rectangle";
const spocFormatClassName = isMediumRectangle ? `ds-spoc-rectangle` : ``;
const faviconSrc = this.getFaviconSrc();
let images = this.renderImage({ sizes: this.standardCardImageSizes });
if (isMediumRectangle) {
images = this.renderImage();
} else if (isListCard) {
images = this.renderImage({ sizes: this.listCardImageSizes });
} else if (sectionsEnabled) {
images = this.renderSectionCardImages();
}
return (
<article
className={`ds-card ${listCardClassName} ${fakespotClassName} ${sectionsCardsClassName} ${compactImagesClassName} ${imageGradientClassName} ${titleLinesName} ${descLinesClassName} ${spocFormatClassName} ${ctaButtonClassName} ${ctaButtonVariantClassName} ${refinedCardsClassName}`}
ref={this.setContextMenuButtonHostRef}
data-position-one={this.props["data-position-one"]}
data-position-two={this.props["data-position-one"]}
data-position-three={this.props["data-position-one"]}
data-position-four={this.props["data-position-one"]}
>
<SafeAnchor
className="ds-card-link"
dispatch={this.props.dispatch}
onLinkClick={!this.props.placeholder ? this.onLinkClick : undefined}
url={this.props.url}
title={this.props.title}
isSponsored={!!this.props.flightId}
>
{this.props.showTopics &&
!this.props.mayHaveSectionsCards &&
this.props.topic &&
!isListCard &&
!refinedCardsLayout && (
<span
className="ds-card-topic"
data-l10n-id={`newtab-topic-label-${this.props.topic}`}
/>
)}
<div className="img-wrapper">{images}</div>
<ImpressionStats
flightId={this.props.flightId}
rows={[
{
id: this.props.id,
pos: this.props.pos,
...(this.props.shim && this.props.shim.impression
? { shim: this.props.shim.impression }
: {}),
recommendation_id: this.props.recommendation_id,
fetchTimestamp: this.props.fetchTimestamp,
corpus_item_id: this.props.corpus_item_id,
scheduled_corpus_item_id: this.props.scheduled_corpus_item_id,
recommended_at: this.props.recommended_at,
received_rank: this.props.received_rank,
topic: this.props.topic,
features: this.props.features,
is_list_card: isListCard,
...(format ? { format } : {}),
isFakespot,
category: this.props.category,
...(this.props.section
? {
section: this.props.section,
section_position: this.props.sectionPosition,
is_section_followed: this.props.sectionFollowed,
sectionLayoutName: this.props.sectionLayoutName,
}
: {}),
...(!format && this.props.section
? // Note: sectionsCardsClassName is passed to ImpressionStats.jsx in order to calculate format
{ class_names: sectionsCardsClassName }
: {}),
},
]}
dispatch={this.props.dispatch}
isFakespot={isFakespot}
source={this.props.type}
firstVisibleTimestamp={this.props.firstVisibleTimestamp}
/>
{ctaButtonVariant === "variant-b" && (
<div className="cta-header">Shop Now</div>
)}
{isFakespot ? (
<div className="meta">
<div className="info-wrap">
<h3 className="title clamp">{this.props.title}</h3>
</div>
</div>
) : (
<DefaultMeta
source={source}
title={this.props.title}
excerpt={excerpt}
newSponsoredLabel={newSponsoredLabel}
timeToRead={timeToRead}
context={this.props.context}
context_type={this.props.context_type}
sponsor={this.props.sponsor}
sponsored_by_override={this.props.sponsored_by_override}
ctaButtonVariant={ctaButtonVariant}
dispatch={this.props.dispatch}
mayHaveThumbsUpDown={this.props.mayHaveThumbsUpDown}
mayHaveSectionsCards={this.props.mayHaveSectionsCards}
onThumbsUpClick={this.onThumbsUpClick}
onThumbsDownClick={this.onThumbsDownClick}
state={this.state}
isListCard={isListCard}
showTopics={!refinedCardsLayout && this.props.showTopics}
isSectionsCard={
this.props.mayHaveSectionsCards &&
this.props.topic &&
!isListCard
}
format={format}
topic={this.props.topic}
icon_src={faviconSrc}
refinedCardsLayout={refinedCardsLayout}
/>
)}
</SafeAnchor>
<div className="card-stp-button-hover-background">
<div className="card-stp-button-position-wrapper">
{!isFakespot && (
<DSLinkMenu
id={this.props.id}
index={this.props.pos}
dispatch={this.props.dispatch}
url={this.props.url}
title={this.props.title}
source={source}
type={this.props.type}
card_type={this.props.flightId ? "spoc" : "organic"}
pocket_id={this.props.pocket_id}
shim={this.props.shim}
bookmarkGuid={this.props.bookmarkGuid}
flightId={this.props.flightId}
showPrivacyInfo={!!this.props.flightId}
onMenuUpdate={this.onMenuUpdate}
onMenuShow={this.onMenuShow}
isRecentSave={isRecentSave}
recommendation_id={this.props.recommendation_id}
tile_id={this.props.id}
block_key={this.props.id}
corpus_item_id={this.props.corpus_item_id}
scheduled_corpus_item_id={this.props.scheduled_corpus_item_id}
recommended_at={this.props.recommended_at}
received_rank={this.props.received_rank}
is_list_card={this.props.isListCard}
section={this.props.section}
section_position={this.props.sectionPosition}
is_section_followed={this.props.sectionFollowed}
fetchTimestamp={this.props.fetchTimestamp}
firstVisibleTimestamp={this.props.firstVisibleTimestamp}
format={
format
? format
: getActiveCardSize(
window.innerWidth,
this.props.sectionsClassNames,
this.props.section,
this.props.flightId
)
}
isSectionsCard={this.props.mayHaveSectionsCards}
topic={this.props.topic}
selected_topics={this.props.selected_topics}
/>
)}
</div>
</div>
</article>
);
}
}
_DSCard.defaultProps = {
windowObj: window, // Added to support unit tests
};
export const DSCard = connect(state => ({
App: state.App,
DiscoveryStream: state.DiscoveryStream,
Prefs: state.Prefs,
}))(_DSCard);
export const PlaceholderDSCard = () => <DSCard placeholder={true} />;