lbry-desktop/ui/component/sideNavigation/view.jsx

512 lines
16 KiB
React
Raw Normal View History

2018-03-26 23:32:43 +02:00
// @flow
2020-08-10 22:47:39 +02:00
import type { Node } from 'react';
2019-03-28 17:53:13 +01:00
import * as PAGES from 'constants/pages';
import * as ICONS from 'constants/icons';
2021-08-31 09:05:42 +02:00
import * as KEYCODES from 'constants/keycodes';
import React, { useEffect } from 'react';
2018-03-26 23:32:43 +02:00
import Button from 'component/button';
2020-05-21 17:38:28 +02:00
import classnames from 'classnames';
2020-11-10 06:21:04 +01:00
import Icon from 'component/common/icon';
2020-08-10 22:47:39 +02:00
import NotificationBubble from 'component/notificationBubble';
2020-11-10 06:21:04 +01:00
import I18nMessage from 'component/i18nMessage';
2021-06-09 21:20:19 +02:00
import ChannelThumbnail from 'component/channelThumbnail';
import { GetLinksData } from 'util/buildHomepage';
import { DOMAIN, ENABLE_UI_NOTIFICATIONS, ENABLE_NO_SOURCE_CLAIMS, CHANNEL_STAKED_LEVEL_LIVESTREAM } from 'config';
const FOLLOWED_ITEM_INITIAL_LIMIT = 10;
type SideNavLink = {
title: string,
link?: string,
route?: string,
onClick?: () => any,
icon: string,
extra?: Node,
hideForUnauth?: boolean,
};
2020-01-03 20:11:47 +01:00
2020-08-21 17:18:47 +02:00
const HOME = {
title: 'Home',
link: `/`,
2020-08-21 17:18:47 +02:00
icon: ICONS.HOME,
onClick: () => {
if (window.location.pathname === '/') window.location.reload();
},
2020-08-21 17:18:47 +02:00
};
const RECENT_FROM_FOLLOWING = {
title: 'Following --[sidebar button]--',
link: `/$/${PAGES.CHANNELS_FOLLOWING}`,
2020-08-21 17:18:47 +02:00
icon: ICONS.SUBSCRIBE,
};
const PLAYLISTS = {
title: 'Lists',
link: `/$/${PAGES.LISTS}`,
icon: ICONS.STACK,
hideForUnauth: true,
};
const UNAUTH_LINKS: Array<SideNavLink> = [
{
title: 'Log In',
link: `/$/${PAGES.AUTH_SIGNIN}`,
icon: ICONS.SIGN_IN,
},
{
title: 'Sign Up',
link: `/$/${PAGES.AUTH}`,
icon: ICONS.SIGN_UP,
},
{
title: 'Settings',
link: `/$/${PAGES.SETTINGS}`,
icon: ICONS.SETTINGS,
},
{
title: 'Help',
link: `/$/${PAGES.HELP}`,
icon: ICONS.HELP,
},
];
// ****************************************************************************
// ****************************************************************************
2018-03-26 23:32:43 +02:00
type Props = {
2019-06-11 20:10:58 +02:00
subscriptions: Array<Subscription>,
followedTags: Array<Tag>,
2019-08-21 22:54:44 +02:00
email: ?string,
uploadCount: number,
2019-12-18 06:27:08 +01:00
doSignOut: () => void,
2020-08-10 22:47:39 +02:00
sidebarOpen: boolean,
setSidebarOpen: (boolean) => void,
2020-08-10 22:47:39 +02:00
isMediumScreen: boolean,
isOnFilePage: boolean,
unseenCount: number,
2020-05-21 17:38:28 +02:00
purchaseSuccess: boolean,
doClearPurchasedUriSuccess: () => void,
2020-08-11 22:32:03 +02:00
user: ?User,
homepageData: any,
activeChannelStakedLevel: number,
};
2020-01-02 17:30:27 +01:00
function SideNavigation(props: Props) {
2019-12-18 06:27:08 +01:00
const {
subscriptions,
2020-08-20 17:10:29 +02:00
doSignOut,
2019-12-18 06:27:08 +01:00
email,
2020-05-21 17:38:28 +02:00
purchaseSuccess,
doClearPurchasedUriSuccess,
2020-08-10 22:47:39 +02:00
sidebarOpen,
setSidebarOpen,
isMediumScreen,
isOnFilePage,
unseenCount,
homepageData,
2020-08-11 22:32:03 +02:00
user,
2021-05-31 08:26:45 +02:00
followedTags,
activeChannelStakedLevel,
2019-12-18 06:27:08 +01:00
} = props;
2020-08-20 17:10:29 +02:00
const EXTRA_SIDEBAR_LINKS = GetLinksData(homepageData).map(({ pinnedUrls, ...theRest }) => theRest);
const MOBILE_LINKS: Array<SideNavLink> = [
{
title: 'Notifications',
link: `/$/${PAGES.NOTIFICATIONS}`,
icon: ICONS.NOTIFICATION,
extra: <NotificationBubble inline />,
hideForUnauth: true,
},
2020-08-20 17:10:29 +02:00
{
title: 'Upload',
link: `/$/${PAGES.UPLOAD}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.PUBLISH,
},
{
title: 'New Channel',
link: `/$/${PAGES.CHANNEL_NEW}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.CHANNEL,
hideForUnauth: true,
},
{
title: 'Uploads',
link: `/$/${PAGES.UPLOADS}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.PUBLISH,
hideForUnauth: true,
},
{
title: 'Channels',
link: `/$/${PAGES.CHANNELS}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.CHANNEL,
hideForUnauth: true,
},
{
title: 'Creator Analytics',
link: `/$/${PAGES.CREATOR_DASHBOARD}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.ANALYTICS,
hideForUnauth: true,
},
{
title: 'Wallet',
link: `/$/${PAGES.WALLET}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.WALLET,
hideForUnauth: true,
},
{
title: 'Rewards',
2020-11-10 18:21:31 +01:00
link: `/$/${PAGES.REWARDS}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.REWARDS,
hideForUnauth: true,
},
{
title: 'Invites',
link: `/$/${PAGES.INVITE}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.INVITE,
hideForUnauth: true,
},
{
title: 'Settings',
link: `/$/${PAGES.SETTINGS}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.SETTINGS,
hideForUnauth: true,
},
{
title: 'Help',
link: `/$/${PAGES.HELP}`,
2020-08-20 17:10:29 +02:00
icon: ICONS.HELP,
hideForUnauth: true,
},
{
title: 'Sign Out',
2020-08-20 17:10:29 +02:00
onClick: doSignOut,
icon: ICONS.SIGN_OUT,
hideForUnauth: true,
},
];
const notificationsEnabled = ENABLE_UI_NOTIFICATIONS || (user && user.experimental_ui);
2019-12-18 06:27:08 +01:00
const isAuthenticated = Boolean(email);
// SIDE LINKS: FOLLOWING, HOME, [FULL,] [EXTRA]
let SIDE_LINKS: Array<SideNavLink> = [];
SIDE_LINKS.push(HOME);
SIDE_LINKS.push(RECENT_FROM_FOLLOWING);
SIDE_LINKS.push(PLAYLISTS);
wip wip wip - everything but publish, autoplay, and styling collection publishing add channel to collection publish cleanup wip bump clear mass add after success move collection item management controls redirect replace to published collection id bump playlist selector on create bump use new collection add ui element bump wip gitignore add content json wip bump context add to playlist basic collections page style pass wip wip: edits, buttons, styles... change fileAuthor to claimAuthor update, pending bugfixes, delete modal progress, collection header, other bugfixes bump cleaning show page bugfix builtin collection headers no playlists, no grid title wip style tweaks use normal looking claim previews for collection tiles add collection changes style library previews collection menulist for delete/view on library delete modal works for unpublished rearrange collection publish tabs clean up collection publishing and items show on odysee begin collectoin edit header and css renaming better thumbnails bump fix collection publish redirect view collection in menu does something copy and thumbs list previews, pending, context menus, list page enter to add collection, lists page empty state playable lists only, delete feature, bump put fileListDownloaded back better collection titles improve collection claim details fix horiz more icon fix up channel page style, copy, bump refactor preview overlay properties, fix reposts showing as floppydisk add watch later toast, small overlay properties on wunderbar results, fix collection actions buttons bump cleanup cleaning, refactoring bump preview thumb styling, cleanup support discover page lists search sync, bump bump, fix sync more enforce builtin order for now new lists page empty state try to indicate unpublished edits in lists bump fix autoplay and linting consts, fix autoplay bugs fixes cleanup fix, bump lists experimental ui, fixes refactor listIndex out hack in collection fallback thumb bump
2021-02-06 08:03:51 +01:00
if (EXTRA_SIDEBAR_LINKS) {
// $FlowFixMe
wip wip wip - everything but publish, autoplay, and styling collection publishing add channel to collection publish cleanup wip bump clear mass add after success move collection item management controls redirect replace to published collection id bump playlist selector on create bump use new collection add ui element bump wip gitignore add content json wip bump context add to playlist basic collections page style pass wip wip: edits, buttons, styles... change fileAuthor to claimAuthor update, pending bugfixes, delete modal progress, collection header, other bugfixes bump cleaning show page bugfix builtin collection headers no playlists, no grid title wip style tweaks use normal looking claim previews for collection tiles add collection changes style library previews collection menulist for delete/view on library delete modal works for unpublished rearrange collection publish tabs clean up collection publishing and items show on odysee begin collectoin edit header and css renaming better thumbnails bump fix collection publish redirect view collection in menu does something copy and thumbs list previews, pending, context menus, list page enter to add collection, lists page empty state playable lists only, delete feature, bump put fileListDownloaded back better collection titles improve collection claim details fix horiz more icon fix up channel page style, copy, bump refactor preview overlay properties, fix reposts showing as floppydisk add watch later toast, small overlay properties on wunderbar results, fix collection actions buttons bump cleanup cleaning, refactoring bump preview thumb styling, cleanup support discover page lists search sync, bump bump, fix sync more enforce builtin order for now new lists page empty state try to indicate unpublished edits in lists bump fix autoplay and linting consts, fix autoplay bugs fixes cleanup fix, bump lists experimental ui, fixes refactor listIndex out hack in collection fallback thumb bump
2021-02-06 08:03:51 +01:00
SIDE_LINKS.push(...EXTRA_SIDEBAR_LINKS);
const WILD_WEST = {
title: 'Wild West',
link: `/$/${PAGES.WILD_WEST}`,
icon: ICONS.WILD_WEST,
};
SIDE_LINKS.push(WILD_WEST);
}
2020-11-16 20:26:23 +01:00
const livestreamEnabled = Boolean(
ENABLE_NO_SOURCE_CLAIMS &&
user &&
!user.odysee_live_disabled &&
(activeChannelStakedLevel >= CHANNEL_STAKED_LEVEL_LIVESTREAM || user.odysee_live_enabled)
);
2020-05-21 17:38:28 +02:00
const [pulseLibrary, setPulseLibrary] = React.useState(false);
const [expandSubscriptions, setExpandSubscriptions] = React.useState(false);
const [expandTags, setExpandTags] = React.useState(false);
const isPersonalized = !IS_WEB || isAuthenticated;
2020-08-10 22:47:39 +02:00
const isAbsolute = isOnFilePage || isMediumScreen;
2020-08-11 22:32:03 +02:00
const microNavigation = !sidebarOpen || isMediumScreen;
const subLinks = email
? MOBILE_LINKS.filter((link) => {
2020-08-11 22:32:03 +02:00
if (!notificationsEnabled && link.icon === ICONS.NOTIFICATION) {
return false;
}
return true;
})
: UNAUTH_LINKS;
const showSubscriptionSection = sidebarOpen && isPersonalized && subscriptions && subscriptions.length > 0;
const showTagSection = sidebarOpen && isPersonalized && followedTags && followedTags.length;
let displayedSubscriptions = subscriptions;
if (showSubscriptionSection && subscriptions.length > FOLLOWED_ITEM_INITIAL_LIMIT && !expandSubscriptions) {
displayedSubscriptions = subscriptions.slice(0, FOLLOWED_ITEM_INITIAL_LIMIT);
}
let displayedFollowedTags = followedTags;
if (showTagSection && followedTags.length > FOLLOWED_ITEM_INITIAL_LIMIT && !expandTags) {
displayedFollowedTags = followedTags.slice(0, FOLLOWED_ITEM_INITIAL_LIMIT);
}
function getSubscriptionSection() {
if (showSubscriptionSection) {
return (
<>
<ul className="navigation__secondary navigation-links">
{displayedSubscriptions.map((subscription) => (
<SubscriptionListItem key={subscription.uri} subscription={subscription} />
))}
</ul>
{subscriptions.length > FOLLOWED_ITEM_INITIAL_LIMIT && (
<Button
label={expandSubscriptions ? __('Show less') : __('Show more')}
className="navigation-link"
onClick={() => setExpandSubscriptions(!expandSubscriptions)}
/>
)}
</>
);
}
return null;
}
function getFollowedTagsSection() {
if (showTagSection) {
return (
<>
<ul className="navigation__secondary navigation-links navigation-links--small">
{displayedFollowedTags.map(({ name }, key) => (
<li key={name} className="navigation-link__wrapper">
<Button navigate={`/$/discover?t=${name}`} label={`#${name}`} className="navigation-link" />
</li>
))}
</ul>
{followedTags.length > FOLLOWED_ITEM_INITIAL_LIMIT && (
<Button
label={expandTags ? __('Show less') : __('Show more')}
className="navigation-link"
onClick={() => setExpandTags(!expandTags)}
/>
)}
</>
);
}
return null;
}
2020-05-21 17:38:28 +02:00
React.useEffect(() => {
if (purchaseSuccess) {
setPulseLibrary(true);
let timeout = setTimeout(() => {
setPulseLibrary(false);
doClearPurchasedUriSuccess();
}, 2500);
return () => clearTimeout(timeout);
}
}, [setPulseLibrary, purchaseSuccess, doClearPurchasedUriSuccess]);
2020-08-10 22:47:39 +02:00
React.useEffect(() => {
function handleKeydown(e: SyntheticKeyboardEvent<*>) {
2021-08-31 09:05:42 +02:00
if (e.keyCode === KEYCODES.ESCAPE && isAbsolute) {
2020-08-10 22:47:39 +02:00
setSidebarOpen(false);
2021-08-31 09:05:42 +02:00
} else if (e.keyCode === KEYCODES.BACKSLASH) {
const hasActiveInput = document.querySelector('input:focus, textarea:focus');
2020-08-10 22:47:39 +02:00
if (!hasActiveInput) {
setSidebarOpen(!sidebarOpen);
}
}
}
window.addEventListener('keydown', handleKeydown);
return () => window.removeEventListener('keydown', handleKeydown);
}, [sidebarOpen, setSidebarOpen, isAbsolute]);
2019-04-19 23:15:41 +02:00
useEffect(() => {
if (!window.sp) {
const gdprDiv = document.getElementById('gdprPrivacyFooter');
if (gdprDiv) {
gdprDiv.style.display = 'none';
}
}
}, [sidebarOpen]);
const unAuthNudge =
DOMAIN === 'lbry.tv' ? null : (
<div className="navigation__auth-nudge">
<span>
<I18nMessage tokens={{ lbc: <Icon icon={ICONS.LBC} /> }}>
Sign up to earn %lbc% for you and your favorite creators.
</I18nMessage>
</span>
2021-06-18 00:52:21 +02:00
<Button
button="secondary"
label={__('Sign Up')}
navigate={`/$/${PAGES.AUTH}?src=sidenav_nudge`}
disabled={user === null}
/>{' '}
</div>
);
2020-11-10 06:21:04 +01:00
2020-11-10 17:10:09 +01:00
const helpLinks = (
<ul className="navigation__tertiary navigation-links--small">
<li className="navigation-link">
2021-10-08 21:22:03 +02:00
<Button label={__('FAQ and Support')} href="https://odysee.com/@OdyseeHelp:b" />
2020-11-10 17:10:09 +01:00
</li>
<li className="navigation-link">
<Button label={__('Community Guidelines')} href="https://odysee.com/@OdyseeHelp:b/Community-Guidelines:c" />
</li>
2020-11-10 17:10:09 +01:00
<li className="navigation-link">
2021-10-08 21:22:03 +02:00
<Button label={__('Terms')} href="https://odysee.com/$/tos" />
2020-11-10 17:10:09 +01:00
</li>
<li className="navigation-link">
2021-10-08 21:22:03 +02:00
<Button label={__('Privacy Policy')} href="https://odysee.com/$/privacypolicy" />
2020-11-10 17:10:09 +01:00
</li>
<li className="navigation-link" id="gdprPrivacyFooter">
<Button label={__('Cookies')} onClick={() => window.sp && window.sp.showPrivacyBanner()} />
</li>
2020-11-10 17:10:09 +01:00
</ul>
);
2020-01-24 19:31:49 +01:00
return (
2020-08-11 22:32:03 +02:00
<div
className={classnames('navigation__wrapper', {
'navigation__wrapper--micro': microNavigation && !isOnFilePage,
'navigation__wrapper--absolute': isAbsolute,
})}
>
2020-08-10 22:47:39 +02:00
{!isOnFilePage && (
<nav
aria-label={'Sidebar'}
className={classnames('navigation', {
'navigation--micro': microNavigation,
})}
>
2020-08-26 23:16:45 +02:00
<div>
<ul className={classnames('navigation-links', { 'navigation-links--micro': !sidebarOpen })}>
{SIDE_LINKS.map((linkProps) => {
// $FlowFixMe
const { hideForUnauth, ...passedProps } = linkProps;
2020-08-26 23:16:45 +02:00
return !email && linkProps.hideForUnauth && IS_WEB ? null : (
2021-05-31 08:26:45 +02:00
<li key={linkProps.route || linkProps.link}>
2020-08-11 22:32:03 +02:00
<Button
{...passedProps}
label={__(linkProps.title)}
title={__(linkProps.title)}
// $FlowFixMe
navigate={linkProps.route || linkProps.link}
2020-08-11 22:32:03 +02:00
icon={pulseLibrary && linkProps.icon === ICONS.LIBRARY ? ICONS.PURCHASED : linkProps.icon}
className={classnames('navigation-link', {
'navigation-link--pulse': linkProps.icon === ICONS.LIBRARY && pulseLibrary,
'navigation-link--highlighted': linkProps.icon === ICONS.NOTIFICATION && unseenCount > 0,
2020-08-11 22:32:03 +02:00
})}
activeClass="navigation-link--active"
/>
{linkProps.extra && linkProps.extra}
2020-08-11 22:32:03 +02:00
</li>
);
})}
2020-08-10 22:47:39 +02:00
</ul>
{getSubscriptionSection()}
{getFollowedTagsSection()}
2020-12-04 15:12:16 +01:00
{!isAuthenticated && sidebarOpen && unAuthNudge}
2020-08-26 23:16:45 +02:00
</div>
2020-11-10 17:10:09 +01:00
{sidebarOpen && helpLinks}
2020-08-26 23:16:45 +02:00
</nav>
)}
{(isOnFilePage || isMediumScreen) && sidebarOpen && (
<>
<nav className="navigation--absolute">
2020-08-26 23:16:45 +02:00
<div>
<ul className="navigation-links--absolute mobile-only">
{email && livestreamEnabled && (
<li key={'Go Live'} className="mobile-only">
<Button
icon={ICONS.VIDEO}
navigate={`/$/${PAGES.LIVESTREAM}`}
label={__('Go Live')}
title={__('Go Live')}
className="navigation-link"
activeClass="navigation-link--active"
/>
</li>
)}
</ul>
2020-08-26 23:16:45 +02:00
<ul className="navigation-links--absolute">
{SIDE_LINKS.map((linkProps) => {
// $FlowFixMe
const { hideForUnauth, link, route, ...passedProps } = linkProps;
return !email && linkProps.hideForUnauth && IS_WEB ? null : (
2020-12-17 19:43:47 +01:00
<li key={route || link}>
2020-08-26 23:16:45 +02:00
<Button
{...passedProps}
navigate={route || link}
label={__(linkProps.title)}
title={__(linkProps.title)}
2020-08-26 23:16:45 +02:00
icon={pulseLibrary && linkProps.icon === ICONS.LIBRARY ? ICONS.PURCHASED : linkProps.icon}
className={classnames('navigation-link', {
'navigation-link--pulse': linkProps.icon === ICONS.LIBRARY && pulseLibrary,
'navigation-link--highlighted': linkProps.icon === ICONS.NOTIFICATION && unseenCount > 0,
2020-08-26 23:16:45 +02:00
})}
activeClass="navigation-link--active"
/>
{linkProps.extra && linkProps.extra}
2020-08-26 23:16:45 +02:00
</li>
);
})}
</ul>
<ul className="navigation-links--absolute mobile-only">
{subLinks.map((linkProps) => {
2020-08-26 23:16:45 +02:00
const { hideForUnauth, ...passedProps } = linkProps;
return !email && hideForUnauth && IS_WEB ? null : (
2020-12-17 19:43:47 +01:00
<li key={linkProps.title} className="mobile-only">
2020-08-26 23:16:45 +02:00
<Button
{...passedProps}
navigate={linkProps.link}
label={__(linkProps.title)}
title={__(linkProps.title)}
2020-08-26 23:16:45 +02:00
className="navigation-link"
activeClass="navigation-link--active"
/>
{linkProps.extra}
</li>
);
})}
</ul>
{getSubscriptionSection()}
{getFollowedTagsSection()}
2020-11-10 06:21:04 +01:00
{!isAuthenticated && unAuthNudge}
{helpLinks}
2020-08-26 23:16:45 +02:00
</div>
2020-08-10 22:47:39 +02:00
</nav>
<div className="navigation__overlay" onClick={() => setSidebarOpen(false)} />
2020-08-10 22:47:39 +02:00
</>
)}
2020-08-11 22:32:03 +02:00
</div>
2019-06-11 20:10:58 +02:00
);
}
2018-03-26 23:32:43 +02:00
2021-06-09 21:20:19 +02:00
function SubscriptionListItem({ subscription }: { subscription: Subscription }) {
const { uri, channelName } = subscription;
return (
<li className="navigation-link__wrapper">
<Button
navigate={uri}
className="navigation-link navigation-link--with-thumbnail"
activeClass="navigation-link--active"
>
ChannelThumbnail improvements - [x] (6332) The IntersectionObserver method of lazy-loading loads cached images visibly late on slower devices. Previously, it was also showing the "broken image" icon briefly, which we mended by placing a dummy transparent image as the initial src. - Reverted that ugly transparent image fix. - Use the browser's built-in `loading="lazy"` instead. Sorry, Safari. - [x] Size-optimization did not take "device pixel ratio" into account. - When resizing an image through the CDN, we can't just take the dimensions of the tag in pixels directly -- we need to take zooming into account, otherwise the image ends up blurry. - Previously, we quickly disabled optimization for the channel avatar in the Channel Page because of this. Now that we know the root-cause, the change was reverted and we now go through the CDN with appropriate sizes. This also improves our Web Vital scores. - [x] Size-optimization wasn't really implemented for all ChannelThumbnail instances. - The CDN-optimized size was hardcoded to the largest instance, so small images like sidebar thumbnails are still loading images that are unnecessarily larger. - There's a little-bit of hardcoding of values from CSS here, but I think it's a ok compromise (not something we change often). It also doesn't need to be exact -- the "device pixel ratio" calculate will ensure it's slightly larger than what we need. - [x] Set `width` and `height` of `<img>` to improve CLS. - Addresses Ligthhouse complaints, although technically the shifting was addressed at the `ClaimPreviewTile` level (sub-container dimensions are well defined). - Notes: the values don't need to be the final CSS-adjusted sizes. It just needs to be in the right aspect ratio to help the browser pre-allocate space to avoid shifts. - [x] Add option to disable lazy-load Channel Thumbnails - The guidelines mentioned that items that are already in the viewport should not enable `loading="lazy"`. - We have a few areas where it doesn't make sense to lazy-load (e.g. thumbnail in Header, channel selector dropdown, publish preview, etc.).
2021-07-05 07:20:40 +02:00
<ChannelThumbnail xsmall uri={uri} hideStakedIndicator />
2021-06-09 21:20:19 +02:00
<span dir="auto" className="button__label">
{channelName}
</span>
</Button>
</li>
);
}
2020-08-10 22:47:39 +02:00
export default SideNavigation;