2022-03-15 12:07:31 -07:00
|
|
|
// @flow
|
|
|
|
import React from 'react';
|
|
|
|
import Button from 'component/button';
|
|
|
|
import HelpLink from 'component/common/help-link';
|
|
|
|
import Icon from 'component/common/icon';
|
|
|
|
import ClaimList from 'component/claimList';
|
|
|
|
import { URL, SHARE_DOMAIN_URL } from 'config';
|
|
|
|
import * as ICONS from 'constants/icons';
|
|
|
|
import * as PAGES from 'constants/pages';
|
2022-03-17 08:23:16 +08:00
|
|
|
import * as SETTINGS from 'constants/settings';
|
2022-03-15 12:07:31 -07:00
|
|
|
import { useIsLargeScreen, useIsMediumScreen } from 'effects/use-screensize';
|
|
|
|
|
|
|
|
// TODO: recsysFyp will be moved into 'RecSys', so the redux import in a jsx
|
|
|
|
// violation is just temporary.
|
|
|
|
import { recsysFyp } from 'redux/actions/search';
|
|
|
|
|
|
|
|
// ****************************************************************************
|
|
|
|
// SectionHeader (TODO: DRY)
|
|
|
|
// ****************************************************************************
|
|
|
|
|
|
|
|
type SectionHeaderProps = {
|
|
|
|
title: string,
|
|
|
|
navigate?: string,
|
|
|
|
icon?: string,
|
|
|
|
help?: string,
|
2022-03-17 08:23:16 +08:00
|
|
|
onHide?: () => void,
|
2022-03-15 12:07:31 -07:00
|
|
|
};
|
|
|
|
|
2022-03-17 08:23:16 +08:00
|
|
|
const SectionHeader = ({ title, icon = '', help, onHide }: SectionHeaderProps) => {
|
2022-03-15 12:07:31 -07:00
|
|
|
const SHARE_DOMAIN = SHARE_DOMAIN_URL || URL;
|
|
|
|
return (
|
|
|
|
<h1 className="claim-grid__header">
|
|
|
|
<Icon className="claim-grid__header-icon" sectionIcon icon={icon} size={20} />
|
|
|
|
<span className="claim-grid__title">{title}</span>
|
|
|
|
{help}
|
|
|
|
<HelpLink href={`${SHARE_DOMAIN}/$/${PAGES.FYP}`} iconSize={24} description={__('Learn more')} />
|
2022-03-17 08:23:16 +08:00
|
|
|
<Button button="link" label={__('Hide')} onClick={onHide} className={'ml-m text-s'} />
|
2022-03-15 12:07:31 -07:00
|
|
|
</h1>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
// ****************************************************************************
|
|
|
|
// RecommendedPersonal
|
|
|
|
// ****************************************************************************
|
|
|
|
|
|
|
|
const VIEW = { ALL_VISIBLE: 0, COLLAPSED: 1, EXPANDED: 2 };
|
|
|
|
|
|
|
|
function getSuitablePageSizeForScreen(defaultSize, isLargeScreen, isMediumScreen) {
|
|
|
|
return isMediumScreen ? 6 : isLargeScreen ? Math.ceil(defaultSize * (3 / 2)) : defaultSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
onLoad: (displayed: boolean) => void,
|
|
|
|
// --- redux ---
|
|
|
|
userId: ?string,
|
|
|
|
personalRecommendations: { gid: string, uris: Array<string> },
|
|
|
|
hasMembership: boolean,
|
2022-03-17 08:23:16 +08:00
|
|
|
hideFyp: boolean,
|
2022-03-15 12:07:31 -07:00
|
|
|
doFetchPersonalRecommendations: () => void,
|
2022-03-17 08:23:16 +08:00
|
|
|
doSetClientSetting: (key: string, value: any, pushPreferences: boolean) => void,
|
|
|
|
doToast: ({ isError?: boolean, message: string }) => void,
|
2022-03-15 12:07:31 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
export default function RecommendedPersonal(props: Props) {
|
2022-03-17 08:23:16 +08:00
|
|
|
const {
|
|
|
|
onLoad,
|
|
|
|
userId,
|
|
|
|
personalRecommendations,
|
|
|
|
hasMembership,
|
|
|
|
hideFyp,
|
|
|
|
doFetchPersonalRecommendations,
|
|
|
|
doSetClientSetting,
|
|
|
|
doToast,
|
|
|
|
} = props;
|
2022-03-22 09:42:38 +08:00
|
|
|
|
|
|
|
const ref = React.useRef();
|
2022-03-15 12:07:31 -07:00
|
|
|
const [markedGid, setMarkedGid] = React.useState('');
|
|
|
|
const [view, setView] = React.useState(VIEW.ALL_VISIBLE);
|
|
|
|
const isLargeScreen = useIsLargeScreen();
|
|
|
|
const isMediumScreen = useIsMediumScreen();
|
|
|
|
|
|
|
|
const count = personalRecommendations.uris.length;
|
|
|
|
const countCollapsed = getSuitablePageSizeForScreen(8, isLargeScreen, isMediumScreen);
|
|
|
|
const finalCount = view === VIEW.ALL_VISIBLE ? count : view === VIEW.COLLAPSED ? countCollapsed : count;
|
|
|
|
|
2022-03-17 08:23:16 +08:00
|
|
|
function doHideFyp() {
|
|
|
|
doSetClientSetting(SETTINGS.HIDE_FYP, true, true);
|
|
|
|
doToast({ message: __('Recommendations hidden; you can re-enable them in Settings.') });
|
|
|
|
}
|
|
|
|
|
2022-03-16 10:48:40 +08:00
|
|
|
// **************************************************************************
|
|
|
|
// Effects
|
|
|
|
// **************************************************************************
|
|
|
|
|
2022-03-15 12:07:31 -07:00
|
|
|
React.useEffect(() => {
|
2022-03-16 10:48:40 +08:00
|
|
|
// -- Update parent's callback request
|
2022-03-16 05:49:44 -07:00
|
|
|
if (typeof onLoad === 'function') {
|
|
|
|
onLoad(count > 0);
|
|
|
|
}
|
2022-03-15 12:07:31 -07:00
|
|
|
}, [count, onLoad]);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
2022-03-16 10:48:40 +08:00
|
|
|
// -- Resolve the view state:
|
2022-03-15 12:07:31 -07:00
|
|
|
let newView;
|
|
|
|
if (count <= countCollapsed) {
|
|
|
|
newView = VIEW.ALL_VISIBLE;
|
|
|
|
} else {
|
|
|
|
if (view === VIEW.ALL_VISIBLE) {
|
|
|
|
newView = VIEW.COLLAPSED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newView && newView !== view) {
|
|
|
|
setView(newView);
|
|
|
|
}
|
|
|
|
}, [count, countCollapsed, view, setView]);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
2022-03-16 10:48:40 +08:00
|
|
|
// -- Mark recommendations when rendered:
|
2022-03-15 12:07:31 -07:00
|
|
|
if (userId && markedGid !== personalRecommendations.gid) {
|
|
|
|
setMarkedGid(personalRecommendations.gid);
|
|
|
|
recsysFyp.markPersonalRecommendations(userId, personalRecommendations.gid);
|
|
|
|
}
|
|
|
|
}, [userId, markedGid, personalRecommendations.gid]);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
2022-03-16 10:48:40 +08:00
|
|
|
// -- Fetch FYP
|
2022-03-17 08:23:16 +08:00
|
|
|
if (hasMembership) {
|
2022-03-16 10:48:40 +08:00
|
|
|
doFetchPersonalRecommendations();
|
|
|
|
}
|
|
|
|
}, [hasMembership, doFetchPersonalRecommendations]);
|
|
|
|
|
|
|
|
// **************************************************************************
|
|
|
|
// **************************************************************************
|
2022-03-15 12:07:31 -07:00
|
|
|
|
2022-03-17 08:23:16 +08:00
|
|
|
if (hideFyp || !hasMembership || count < 1) {
|
2022-03-15 12:07:31 -07:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2022-03-22 09:42:38 +08:00
|
|
|
<div ref={ref}>
|
2022-03-17 08:23:16 +08:00
|
|
|
<SectionHeader title={__('Recommended For You')} icon={ICONS.WEB} onHide={doHideFyp} />
|
|
|
|
|
2022-03-15 12:07:31 -07:00
|
|
|
<ClaimList
|
|
|
|
tileLayout
|
|
|
|
uris={personalRecommendations.uris.slice(0, finalCount)}
|
|
|
|
fypId={personalRecommendations.gid}
|
|
|
|
/>
|
|
|
|
|
|
|
|
{view !== VIEW.ALL_VISIBLE && (
|
|
|
|
<div className="livestream-list--view-more">
|
|
|
|
<Button
|
|
|
|
label={view === VIEW.COLLAPSED ? __('Show more') : __('Show less')}
|
|
|
|
button="link"
|
|
|
|
iconRight={view === VIEW.COLLAPSED ? ICONS.DOWN : ICONS.UP}
|
|
|
|
className="claim-grid__title--secondary"
|
|
|
|
onClick={() => {
|
|
|
|
if (view === VIEW.COLLAPSED) {
|
|
|
|
setView(VIEW.EXPANDED);
|
|
|
|
} else {
|
|
|
|
setView(VIEW.COLLAPSED);
|
2022-03-22 09:42:38 +08:00
|
|
|
if (ref.current) {
|
|
|
|
ref.current.scrollIntoView({ block: 'start', behavior: 'smooth' });
|
|
|
|
} else {
|
|
|
|
// Unlikely to happen, but should have a fallback (just go all the way up)
|
|
|
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
|
|
|
}
|
2022-03-15 12:07:31 -07:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
2022-03-22 09:42:38 +08:00
|
|
|
</div>
|
2022-03-15 12:07:31 -07:00
|
|
|
);
|
|
|
|
}
|