2018-03-26 14:32:43 -07:00
|
|
|
// @flow
|
2020-02-12 14:10:35 -05:00
|
|
|
import * as ICONS from 'constants/icons';
|
2020-09-03 16:05:38 -04:00
|
|
|
import * as PAGES from 'constants/pages';
|
2020-06-30 01:51:15 -04:00
|
|
|
import React from 'react';
|
2019-05-06 22:35:04 -04:00
|
|
|
import { parseURI } from 'lbry-redux';
|
2020-09-03 16:05:38 -04:00
|
|
|
import { YOUTUBE_STATUSES } from 'lbryinc';
|
2018-03-26 14:32:43 -07:00
|
|
|
import Page from 'component/page';
|
2019-05-06 22:35:04 -04:00
|
|
|
import SubscribeButton from 'component/subscribeButton';
|
|
|
|
import ShareButton from 'component/shareButton';
|
|
|
|
import { Tabs, TabList, Tab, TabPanels, TabPanel } from 'component/common/tabs';
|
2020-06-30 01:51:15 -04:00
|
|
|
import { useHistory } from 'react-router';
|
2019-06-28 13:00:29 -04:00
|
|
|
import Button from 'component/button';
|
2019-12-02 12:30:08 -05:00
|
|
|
import { formatLbryUrlForWeb } from 'util/url';
|
2019-05-06 22:35:04 -04:00
|
|
|
import ChannelContent from 'component/channelContent';
|
|
|
|
import ChannelAbout from 'component/channelAbout';
|
2019-10-14 18:21:40 -04:00
|
|
|
import ChannelDiscussion from 'component/channelDiscussion';
|
2019-05-06 22:35:04 -04:00
|
|
|
import ChannelThumbnail from 'component/channelThumbnail';
|
2019-07-02 13:54:42 -04:00
|
|
|
import ChannelEdit from 'component/channelEdit';
|
2019-08-01 20:56:25 -04:00
|
|
|
import classnames from 'classnames';
|
2019-10-03 17:40:54 -04:00
|
|
|
import HelpLink from 'component/common/help-link';
|
2020-06-08 14:42:29 -04:00
|
|
|
import ClaimSupportButton from 'component/claimSupportButton';
|
2021-02-16 16:09:20 -05:00
|
|
|
import ChannelStakedIndicator from 'component/channelStakedIndicator';
|
2021-03-03 13:50:16 -05:00
|
|
|
import ClaimMenuList from 'component/claimMenuList';
|
|
|
|
import Yrbl from 'component/yrbl';
|
2019-05-06 22:35:04 -04:00
|
|
|
|
2020-08-31 13:12:34 -04:00
|
|
|
export const PAGE_VIEW_QUERY = `view`;
|
2019-05-06 22:35:04 -04:00
|
|
|
const ABOUT_PAGE = `about`;
|
2020-08-31 13:12:34 -04:00
|
|
|
export const DISCUSSION_PAGE = `discussion`;
|
2020-06-30 01:51:15 -04:00
|
|
|
const EDIT_PAGE = 'edit';
|
2017-05-03 23:44:08 -04:00
|
|
|
|
2018-03-26 14:32:43 -07:00
|
|
|
type Props = {
|
|
|
|
uri: string,
|
2019-07-17 16:49:06 -04:00
|
|
|
claim: ChannelClaim,
|
2019-05-06 22:35:04 -04:00
|
|
|
title: ?string,
|
|
|
|
cover: ?string,
|
|
|
|
thumbnail: ?string,
|
2019-05-14 01:12:24 -04:00
|
|
|
page: number,
|
2019-05-06 22:35:04 -04:00
|
|
|
match: { params: { attribute: ?string } },
|
2019-06-28 13:00:29 -04:00
|
|
|
channelIsMine: boolean,
|
2019-07-08 16:54:58 -04:00
|
|
|
isSubscribed: boolean,
|
|
|
|
channelIsBlocked: boolean,
|
2019-08-28 21:39:21 -04:00
|
|
|
blackListedOutpoints: Array<{
|
|
|
|
txid: string,
|
|
|
|
nout: number,
|
|
|
|
}>,
|
2021-02-16 16:09:20 -05:00
|
|
|
fetchSubCount: (string) => void,
|
2019-09-24 23:42:51 -04:00
|
|
|
subCount: number,
|
2020-06-21 12:51:06 -04:00
|
|
|
pending: boolean,
|
2020-09-03 16:05:38 -04:00
|
|
|
youtubeChannels: ?Array<{ channel_claim_id: string, sync_status: string, transfer_state: string }>,
|
2021-03-03 13:50:16 -05:00
|
|
|
blockedChannels: Array<string>,
|
2021-03-04 12:55:16 -05:00
|
|
|
mutedChannels: Array<string>,
|
2018-03-26 14:32:43 -07:00
|
|
|
};
|
|
|
|
|
2019-03-28 12:53:13 -04:00
|
|
|
function ChannelPage(props: Props) {
|
2019-07-08 16:54:58 -04:00
|
|
|
const {
|
|
|
|
uri,
|
2020-06-29 15:54:07 -04:00
|
|
|
claim,
|
2019-07-08 16:54:58 -04:00
|
|
|
title,
|
|
|
|
cover,
|
|
|
|
page,
|
|
|
|
channelIsMine,
|
|
|
|
isSubscribed,
|
2019-08-28 21:39:21 -04:00
|
|
|
blackListedOutpoints,
|
2019-09-24 23:42:51 -04:00
|
|
|
fetchSubCount,
|
|
|
|
subCount,
|
2020-06-21 12:51:06 -04:00
|
|
|
pending,
|
2020-09-03 16:05:38 -04:00
|
|
|
youtubeChannels,
|
2021-03-03 13:50:16 -05:00
|
|
|
blockedChannels,
|
2021-03-04 12:55:16 -05:00
|
|
|
mutedChannels,
|
2019-07-08 16:54:58 -04:00
|
|
|
} = props;
|
2020-06-30 01:51:15 -04:00
|
|
|
const {
|
|
|
|
push,
|
|
|
|
goBack,
|
|
|
|
location: { search },
|
|
|
|
} = useHistory();
|
2021-03-03 13:50:16 -05:00
|
|
|
const [viewBlockedChannel, setViewBlockedChannel] = React.useState(false);
|
2019-03-28 12:53:13 -04:00
|
|
|
const urlParams = new URLSearchParams(search);
|
2019-05-06 22:35:04 -04:00
|
|
|
const currentView = urlParams.get(PAGE_VIEW_QUERY) || undefined;
|
2020-07-24 19:35:22 +08:00
|
|
|
const [discussionWasMounted, setDiscussionWasMounted] = React.useState(false);
|
2020-07-10 17:04:36 -04:00
|
|
|
const editing = urlParams.get(PAGE_VIEW_QUERY) === EDIT_PAGE;
|
2020-06-30 01:51:15 -04:00
|
|
|
const { channelName } = parseURI(uri);
|
2019-07-17 16:49:06 -04:00
|
|
|
const { permanent_url: permanentUrl } = claim;
|
2019-09-26 23:52:04 -04:00
|
|
|
const claimId = claim.claim_id;
|
2020-05-25 12:19:15 -04:00
|
|
|
const formattedSubCount = Number(subCount).toLocaleString();
|
2021-03-03 13:50:16 -05:00
|
|
|
const isBlocked = claim && blockedChannels.includes(claim.permanent_url);
|
2021-03-04 12:55:16 -05:00
|
|
|
const isMuted = claim && mutedChannels.includes(claim.permanent_url);
|
2020-09-03 16:05:38 -04:00
|
|
|
const isMyYouTubeChannel =
|
|
|
|
claim &&
|
|
|
|
youtubeChannels &&
|
|
|
|
youtubeChannels.some(({ channel_claim_id, sync_status, transfer_state }) => {
|
|
|
|
if (
|
|
|
|
channel_claim_id === claim.claim_id &&
|
|
|
|
sync_status !== YOUTUBE_STATUSES.YOUTUBE_SYNC_ABANDONDED &&
|
|
|
|
transfer_state !== YOUTUBE_STATUSES.YOUTUBE_SYNC_COMPLETED_TRANSFER
|
|
|
|
) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
2020-06-30 01:51:15 -04:00
|
|
|
let channelIsBlackListed = false;
|
|
|
|
|
|
|
|
if (claim && blackListedOutpoints) {
|
|
|
|
channelIsBlackListed = blackListedOutpoints.some(
|
2021-02-16 16:09:20 -05:00
|
|
|
(outpoint) => outpoint.txid === claim.txid && outpoint.nout === claim.nout
|
2020-06-30 01:51:15 -04:00
|
|
|
);
|
|
|
|
}
|
2019-06-28 13:00:29 -04:00
|
|
|
|
2019-05-06 22:35:04 -04:00
|
|
|
// If a user changes tabs, update the url so it stays on the same page if they refresh.
|
|
|
|
// We don't want to use links here because we can't animate the tab change and using links
|
|
|
|
// would alter the Tab label's role attribute, which should stay role="tab" to work with keyboards/screen readers.
|
2019-10-14 18:21:40 -04:00
|
|
|
const tabIndex = currentView === ABOUT_PAGE || editing ? 1 : currentView === DISCUSSION_PAGE ? 2 : 0;
|
|
|
|
|
2019-09-16 01:32:02 -04:00
|
|
|
function onTabChange(newTabIndex) {
|
2019-12-02 12:30:08 -05:00
|
|
|
let url = formatLbryUrlForWeb(uri);
|
2019-05-14 01:12:24 -04:00
|
|
|
let search = '?';
|
2019-10-14 18:21:40 -04:00
|
|
|
|
|
|
|
if (newTabIndex === 0) {
|
2019-05-14 01:12:24 -04:00
|
|
|
search += `page=${page}`;
|
2019-10-14 18:21:40 -04:00
|
|
|
} else if (newTabIndex === 1) {
|
|
|
|
search += `${PAGE_VIEW_QUERY}=${ABOUT_PAGE}`;
|
|
|
|
} else {
|
|
|
|
search += `${PAGE_VIEW_QUERY}=${DISCUSSION_PAGE}`;
|
2017-08-24 17:12:23 -04:00
|
|
|
}
|
2020-06-30 01:51:15 -04:00
|
|
|
|
|
|
|
push(`${url}${search}`);
|
2019-09-16 01:32:02 -04:00
|
|
|
}
|
|
|
|
|
2020-07-24 19:35:22 +08:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (currentView === DISCUSSION_PAGE) {
|
|
|
|
setDiscussionWasMounted(true);
|
|
|
|
}
|
|
|
|
}, [currentView]);
|
|
|
|
|
2019-09-23 09:57:03 -04:00
|
|
|
React.useEffect(() => {
|
2019-09-26 23:52:04 -04:00
|
|
|
fetchSubCount(claimId);
|
|
|
|
}, [uri, fetchSubCount, claimId]);
|
2019-09-23 09:57:03 -04:00
|
|
|
|
2020-06-29 15:54:07 -04:00
|
|
|
if (editing) {
|
|
|
|
return (
|
|
|
|
<Page
|
|
|
|
noFooter
|
|
|
|
noSideNavigation={editing}
|
2020-06-30 01:51:15 -04:00
|
|
|
backout={{
|
|
|
|
title: __('Editing @%channel%', { channel: channelName }),
|
|
|
|
simpleTitle: __('Editing'),
|
|
|
|
}}
|
2020-06-29 15:54:07 -04:00
|
|
|
>
|
2020-07-10 17:04:36 -04:00
|
|
|
<ChannelEdit uri={uri} onDone={() => goBack()} />
|
2020-06-29 15:54:07 -04:00
|
|
|
</Page>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-03-28 12:53:13 -04:00
|
|
|
return (
|
2020-05-15 10:19:37 -04:00
|
|
|
<Page noFooter>
|
2020-01-03 11:36:15 -05:00
|
|
|
<header className="channel-cover">
|
|
|
|
<div className="channel__quick-actions">
|
2020-09-03 16:05:38 -04:00
|
|
|
{isMyYouTubeChannel && (
|
|
|
|
<Button
|
|
|
|
button="alt"
|
|
|
|
label={__('Claim Your Channel')}
|
|
|
|
icon={ICONS.YOUTUBE}
|
|
|
|
navigate={`/$/${PAGES.CHANNELS}`}
|
|
|
|
/>
|
|
|
|
)}
|
2021-03-04 12:55:16 -05:00
|
|
|
{!channelIsBlackListed && <ShareButton uri={uri} />}
|
|
|
|
{!(isBlocked || isMuted) && <ClaimSupportButton uri={uri} />}
|
|
|
|
{!(isBlocked || isMuted) && (!channelIsBlackListed || isSubscribed) && <SubscribeButton uri={permanentUrl} />}
|
2021-03-03 13:50:16 -05:00
|
|
|
<ClaimMenuList uri={claim.permanent_url} inline />
|
2020-01-03 11:36:15 -05:00
|
|
|
</div>
|
2021-03-04 12:55:16 -05:00
|
|
|
{cover && <img className={classnames('channel-cover__custom')} src={cover} />}
|
2020-01-03 11:36:15 -05:00
|
|
|
<div className="channel__primary-info">
|
2021-03-04 12:55:16 -05:00
|
|
|
<ChannelThumbnail className="channel__thumbnail--channel-page" uri={uri} allowGifs hideStakedIndicator />
|
2021-02-16 16:09:20 -05:00
|
|
|
<h1 className="channel__title">
|
|
|
|
{title || '@' + channelName}
|
|
|
|
<ChannelStakedIndicator uri={uri} large />
|
|
|
|
</h1>
|
2020-01-03 11:36:15 -05:00
|
|
|
<div className="channel__meta">
|
|
|
|
<span>
|
2020-05-25 12:19:15 -04:00
|
|
|
{formattedSubCount} {subCount !== 1 ? __('Followers') : __('Follower')}
|
2020-01-03 11:36:15 -05:00
|
|
|
<HelpLink href="https://lbry.com/faq/views" />
|
|
|
|
</span>
|
2020-06-29 15:54:07 -04:00
|
|
|
{channelIsMine && (
|
2020-06-21 12:51:06 -04:00
|
|
|
<>
|
|
|
|
{pending ? (
|
|
|
|
<span>{__('Your changes will be live in a few minutes')}</span>
|
|
|
|
) : (
|
|
|
|
<Button
|
|
|
|
button="alt"
|
|
|
|
title={__('Edit')}
|
2020-07-10 17:04:36 -04:00
|
|
|
onClick={() => push(`?${PAGE_VIEW_QUERY}=${EDIT_PAGE}`)}
|
2020-06-21 12:51:06 -04:00
|
|
|
icon={ICONS.EDIT}
|
|
|
|
iconSize={18}
|
|
|
|
disabled={pending}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
)}
|
2020-01-03 11:36:15 -05:00
|
|
|
</div>
|
|
|
|
</div>
|
2020-06-18 11:28:30 -04:00
|
|
|
<div className="channel-cover__gradient" />
|
2020-01-03 11:36:15 -05:00
|
|
|
</header>
|
2021-02-16 16:09:20 -05:00
|
|
|
|
2021-03-04 12:55:16 -05:00
|
|
|
{(isBlocked || isMuted) && !viewBlockedChannel ? (
|
2021-03-03 13:50:16 -05:00
|
|
|
<div className="main--empty">
|
|
|
|
<Yrbl
|
2021-03-04 12:55:16 -05:00
|
|
|
title={isBlocked ? __('This channel is blocked') : __('This channel is muted')}
|
|
|
|
subtitle={
|
|
|
|
isBlocked
|
|
|
|
? __('Are you sure you want to view this content? Viewing will not unblock @%channel%', {
|
|
|
|
channel: channelName,
|
|
|
|
})
|
|
|
|
: __('Are you sure you want to view this content? Viewing will not unmute @%channel%', {
|
|
|
|
channel: channelName,
|
|
|
|
})
|
|
|
|
}
|
2021-03-03 13:50:16 -05:00
|
|
|
actions={
|
|
|
|
<div className="section__actions">
|
|
|
|
<Button button="primary" label={__('View Content')} onClick={() => setViewBlockedChannel(true)} />
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<Tabs onChange={onTabChange} index={tabIndex}>
|
|
|
|
<TabList className="tabs__list--channel-page">
|
|
|
|
<Tab disabled={editing}>{__('Content')}</Tab>
|
|
|
|
<Tab>{editing ? __('Editing Your Channel') : __('About --[tab title in Channel Page]--')}</Tab>
|
|
|
|
<Tab disabled={editing}>{__('Community')}</Tab>
|
|
|
|
</TabList>
|
|
|
|
<TabPanels>
|
|
|
|
<TabPanel>
|
2021-03-04 12:55:16 -05:00
|
|
|
<ChannelContent uri={uri} channelIsBlackListed={channelIsBlackListed} viewHiddenChannels />
|
2021-03-03 13:50:16 -05:00
|
|
|
</TabPanel>
|
|
|
|
<TabPanel>
|
|
|
|
<ChannelAbout uri={uri} />
|
|
|
|
</TabPanel>
|
|
|
|
<TabPanel>
|
|
|
|
{(discussionWasMounted || currentView === DISCUSSION_PAGE) && <ChannelDiscussion uri={uri} />}
|
|
|
|
</TabPanel>
|
|
|
|
</TabPanels>
|
|
|
|
</Tabs>
|
|
|
|
)}
|
2019-03-28 12:53:13 -04:00
|
|
|
</Page>
|
|
|
|
);
|
2017-05-03 23:44:08 -04:00
|
|
|
}
|
|
|
|
|
2020-06-30 01:51:15 -04:00
|
|
|
export default ChannelPage;
|