2018-03-26 14:32:43 -07:00
|
|
|
// @flow
|
2020-02-12 14:10:35 -05:00
|
|
|
import * as ICONS from 'constants/icons';
|
|
|
|
import * as MODALS from 'constants/modal_types';
|
2020-01-16 00:50:05 -06:00
|
|
|
import React, { useState, useEffect } from 'react';
|
2019-05-06 22:35:04 -04:00
|
|
|
import { parseURI } from 'lbry-redux';
|
2020-02-12 14:10:35 -05:00
|
|
|
import { Lbryio } 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';
|
2019-07-08 16:54:58 -04:00
|
|
|
import BlockButton from 'component/blockButton';
|
2019-05-06 22:35:04 -04:00
|
|
|
import ShareButton from 'component/shareButton';
|
|
|
|
import { Tabs, TabList, Tab, TabPanels, TabPanel } from 'component/common/tabs';
|
|
|
|
import { withRouter } 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-07-08 14:58:33 -04:00
|
|
|
import ClaimUri from 'component/claimUri';
|
2019-08-01 20:56:25 -04:00
|
|
|
import classnames from 'classnames';
|
2019-09-16 01:32:02 -04:00
|
|
|
import { Form, FormField } from 'component/common/form';
|
|
|
|
import Icon from 'component/common/icon';
|
2019-10-03 17:40:54 -04:00
|
|
|
import HelpLink from 'component/common/help-link';
|
2019-10-19 14:43:16 +01:00
|
|
|
import { DEBOUNCE_WAIT_DURATION_MS } from 'constants/search';
|
2020-01-19 01:06:44 -06:00
|
|
|
import ClaimList from 'component/claimList';
|
2020-02-12 14:10:35 -05:00
|
|
|
import relativeDate from 'tiny-relative-date';
|
2019-05-06 22:35:04 -04:00
|
|
|
|
|
|
|
const PAGE_VIEW_QUERY = `view`;
|
|
|
|
const ABOUT_PAGE = `about`;
|
2019-10-14 18:21:40 -04:00
|
|
|
const DISCUSSION_PAGE = `discussion`;
|
2019-09-16 01:32:02 -04:00
|
|
|
const LIGHTHOUSE_URL = 'https://lighthouse.lbry.com/search';
|
2020-01-15 00:13:37 -06:00
|
|
|
const ARROW_LEFT_KEYCODE = 37;
|
|
|
|
const ARROW_RIGHT_KEYCODE = 39;
|
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
|
|
|
location: { search: string },
|
2019-04-04 17:05:23 -04:00
|
|
|
history: { push: string => void },
|
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,
|
|
|
|
}>,
|
2019-08-30 12:56:35 -04:00
|
|
|
openModal: (id: string, { uri: string, claimIsMine?: boolean, isSupport?: boolean }) => void,
|
|
|
|
supportOption: boolean,
|
2019-09-24 23:42:51 -04:00
|
|
|
fetchSubCount: string => void,
|
|
|
|
subCount: number,
|
2020-01-28 17:05:44 -05:00
|
|
|
showMature: boolean,
|
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,
|
|
|
|
title,
|
|
|
|
cover,
|
|
|
|
history,
|
|
|
|
location,
|
|
|
|
page,
|
|
|
|
channelIsMine,
|
|
|
|
thumbnail,
|
|
|
|
claim,
|
|
|
|
isSubscribed,
|
|
|
|
channelIsBlocked,
|
2019-08-28 21:39:21 -04:00
|
|
|
blackListedOutpoints,
|
2019-08-30 12:56:35 -04:00
|
|
|
openModal,
|
|
|
|
supportOption,
|
2020-01-28 17:05:44 -05:00
|
|
|
showMature,
|
2019-09-24 23:42:51 -04:00
|
|
|
fetchSubCount,
|
|
|
|
subCount,
|
2019-07-08 16:54:58 -04:00
|
|
|
} = props;
|
2019-09-19 16:32:45 -04:00
|
|
|
|
2019-07-08 14:58:33 -04:00
|
|
|
const { channelName } = parseURI(uri);
|
2019-03-28 12:53:13 -04:00
|
|
|
const { search } = location;
|
|
|
|
const urlParams = new URLSearchParams(search);
|
2019-05-06 22:35:04 -04:00
|
|
|
const currentView = urlParams.get(PAGE_VIEW_QUERY) || undefined;
|
2019-07-17 16:49:06 -04:00
|
|
|
const { permanent_url: permanentUrl } = claim;
|
2019-06-28 13:00:29 -04:00
|
|
|
const [editing, setEditing] = useState(false);
|
|
|
|
const [thumbPreview, setThumbPreview] = useState(thumbnail);
|
|
|
|
const [coverPreview, setCoverPreview] = useState(cover);
|
2019-09-16 01:32:02 -04:00
|
|
|
const [searchQuery, setSearchQuery] = useState('');
|
|
|
|
const [searchResults, setSearchResults] = useState(undefined);
|
2020-02-12 14:10:35 -05:00
|
|
|
const [lastYtSyncDate, setLastYtSyncDate] = useState();
|
2019-09-26 23:52:04 -04:00
|
|
|
const claimId = claim.claim_id;
|
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-09-16 01:32:02 -04:00
|
|
|
setSearchResults(null);
|
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
|
|
|
}
|
2019-05-14 01:12:24 -04:00
|
|
|
history.push(`${url}${search}`);
|
2019-09-16 01:32:02 -04:00
|
|
|
}
|
|
|
|
|
2019-10-19 14:43:16 +01:00
|
|
|
function getResults(fetchUrl) {
|
2019-09-16 01:32:02 -04:00
|
|
|
fetch(fetchUrl)
|
|
|
|
.then(res => res.json())
|
|
|
|
.then(results => {
|
|
|
|
const urls = results.map(({ name, claimId }) => {
|
|
|
|
return `lbry://${name}#${claimId}`;
|
|
|
|
});
|
|
|
|
setSearchResults(urls);
|
|
|
|
})
|
|
|
|
.catch(() => {
|
|
|
|
setSearchResults(null);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-01-16 00:50:05 -06:00
|
|
|
useEffect(() => {
|
|
|
|
const timer = setTimeout(() => {
|
|
|
|
if (searchQuery === '') {
|
|
|
|
// In order to display original search results, search results must be set to null. A query of '' should display original results.
|
|
|
|
return setSearchResults(null);
|
|
|
|
} else {
|
2020-01-28 17:05:44 -05:00
|
|
|
getResults(
|
|
|
|
`${LIGHTHOUSE_URL}?s=${encodeURIComponent(searchQuery)}&channel_id=${encodeURIComponent(claimId)}${
|
|
|
|
!showMature ? '&nsfw=false' : ''
|
|
|
|
}`
|
|
|
|
);
|
2020-01-16 00:50:05 -06:00
|
|
|
}
|
|
|
|
}, DEBOUNCE_WAIT_DURATION_MS);
|
|
|
|
return () => clearTimeout(timer);
|
|
|
|
}, [claimId, searchQuery]);
|
2019-10-19 16:24:42 +01:00
|
|
|
|
2020-02-12 14:10:35 -05:00
|
|
|
useEffect(() => {
|
|
|
|
Lbryio.call('yt', 'get_youtuber', { channel_claim_id: claimId }).then(response => {
|
|
|
|
if (response.is_verified_youtuber) {
|
|
|
|
setLastYtSyncDate(response.last_synced);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, [claimId]);
|
|
|
|
|
2019-09-16 01:32:02 -04:00
|
|
|
function handleInputChange(e) {
|
|
|
|
const { value } = e.target;
|
|
|
|
setSearchQuery(value);
|
|
|
|
}
|
2019-03-28 12:53:13 -04:00
|
|
|
|
2020-01-15 00:13:37 -06:00
|
|
|
/*
|
|
|
|
Since the search is inside of TabList, the left and right arrow keys change the tabIndex.
|
|
|
|
This results in the user not being able to navigate the search string by using arrow keys.
|
|
|
|
This function allows the event to change cursor position and then stops propagation to prevent tab changing.
|
|
|
|
*/
|
|
|
|
function handleSearchArrowKeys(e) {
|
|
|
|
if (e.keyCode === ARROW_LEFT_KEYCODE || e.keyCode === ARROW_RIGHT_KEYCODE) {
|
|
|
|
e.stopPropagation();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-28 21:39:21 -04:00
|
|
|
let channelIsBlackListed = false;
|
|
|
|
|
|
|
|
if (claim && blackListedOutpoints) {
|
|
|
|
channelIsBlackListed = blackListedOutpoints.some(
|
|
|
|
outpoint => outpoint.txid === claim.txid && outpoint.nout === claim.nout
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-23 09:57:03 -04:00
|
|
|
React.useEffect(() => {
|
|
|
|
setSearchResults(null);
|
|
|
|
setSearchQuery('');
|
2019-09-26 23:52:04 -04:00
|
|
|
|
|
|
|
fetchSubCount(claimId);
|
|
|
|
}, [uri, fetchSubCount, claimId]);
|
2019-09-23 09:57:03 -04:00
|
|
|
|
2020-01-09 22:52:53 -06:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (!channelIsMine && editing) {
|
|
|
|
setEditing(false);
|
|
|
|
}
|
|
|
|
}, [channelIsMine, editing]);
|
|
|
|
|
2019-03-28 12:53:13 -04:00
|
|
|
return (
|
2019-06-11 14:10:58 -04:00
|
|
|
<Page>
|
2020-01-02 17:30:58 -05:00
|
|
|
<ClaimUri uri={uri} />
|
|
|
|
|
2020-02-12 14:10:35 -05:00
|
|
|
{lastYtSyncDate && (
|
|
|
|
<div className="media__uri--right">
|
|
|
|
<Icon icon={ICONS.VALIDATED} size={12} />
|
|
|
|
{__('Official YouTube Creator - Last updated %time_ago%', { time_ago: relativeDate(lastYtSyncDate) })}
|
|
|
|
</div>
|
|
|
|
)}
|
2020-01-03 11:36:15 -05:00
|
|
|
<header className="channel-cover">
|
|
|
|
<div className="channel__quick-actions">
|
|
|
|
{!channelIsBlocked && !channelIsBlackListed && <ShareButton uri={uri} isChannel />}
|
|
|
|
{!channelIsMine && (
|
|
|
|
<Button
|
|
|
|
button="alt"
|
|
|
|
icon={ICONS.TIP}
|
|
|
|
label={__('Tip')}
|
|
|
|
title={__('Send a tip to this creator')}
|
|
|
|
onClick={() => openModal(MODALS.SEND_TIP, { uri, channelIsMine, isSupport: false })}
|
2019-08-01 20:56:25 -04:00
|
|
|
/>
|
|
|
|
)}
|
2020-01-03 11:36:15 -05:00
|
|
|
{(channelIsMine || (!channelIsMine && supportOption)) && (
|
|
|
|
<Button
|
|
|
|
button="alt"
|
|
|
|
icon={ICONS.SUPPORT}
|
|
|
|
label={__('Support')}
|
|
|
|
title={__('Support this creator')}
|
|
|
|
onClick={() => openModal(MODALS.SEND_TIP, { uri, channelIsMine, isSupport: true })}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
{!channelIsBlocked && (!channelIsBlackListed || isSubscribed) && <SubscribeButton uri={permanentUrl} />}
|
|
|
|
{!isSubscribed && <BlockButton uri={permanentUrl} />}
|
|
|
|
</div>
|
|
|
|
{!editing && cover && (
|
|
|
|
<img
|
|
|
|
className={classnames('channel-cover__custom', { 'channel__image--blurred': channelIsBlocked })}
|
|
|
|
src={cover}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
{editing && <img className="channel-cover__custom" src={coverPreview} />}
|
|
|
|
{/* component that offers select/upload */}
|
|
|
|
<div className="channel__primary-info">
|
|
|
|
{!editing && (
|
|
|
|
<ChannelThumbnail className="channel__thumbnail--channel-page" uri={uri} obscure={channelIsBlocked} />
|
|
|
|
)}
|
|
|
|
{editing && (
|
|
|
|
<ChannelThumbnail className="channel__thumbnail--channel-page" uri={uri} thumbnailPreview={thumbPreview} />
|
|
|
|
)}
|
|
|
|
<h1 className="channel__title">{title || '@' + channelName}</h1>
|
|
|
|
{channelIsMine && !editing && (
|
|
|
|
<Button button="alt" title={__('Edit')} onClick={() => setEditing(!editing)} icon={ICONS.EDIT} />
|
|
|
|
)}
|
|
|
|
<div className="channel__meta">
|
|
|
|
<span>
|
|
|
|
{subCount} {subCount !== 1 ? __('Followers') : __('Follower')}
|
|
|
|
<HelpLink href="https://lbry.com/faq/views" />
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
<Tabs onChange={onTabChange} index={tabIndex}>
|
|
|
|
<TabList className="tabs__list--channel-page">
|
|
|
|
<Tab disabled={editing}>{__('Content')}</Tab>
|
|
|
|
<Tab>{editing ? __('Editing Your Channel') : __('About')}</Tab>
|
|
|
|
<Tab disabled={editing}>{__('Comments')}</Tab>
|
|
|
|
{/* only render searchbar on content page (tab index 0 === content page) */}
|
|
|
|
{tabIndex === 0 ? (
|
2020-01-16 00:50:05 -06:00
|
|
|
<Form onSubmit={() => {}} className="wunderbar--inline">
|
2020-01-03 11:36:15 -05:00
|
|
|
<Icon icon={ICONS.SEARCH} />
|
|
|
|
<FormField
|
|
|
|
className="wunderbar__input"
|
|
|
|
value={searchQuery}
|
|
|
|
onChange={handleInputChange}
|
2020-01-15 00:13:37 -06:00
|
|
|
onKeyDown={handleSearchArrowKeys}
|
2020-01-03 11:36:15 -05:00
|
|
|
type="text"
|
|
|
|
placeholder={__('Search')}
|
|
|
|
/>
|
|
|
|
</Form>
|
|
|
|
) : (
|
|
|
|
<div />
|
|
|
|
)}
|
|
|
|
</TabList>
|
|
|
|
|
|
|
|
<TabPanels>
|
|
|
|
<TabPanel>
|
|
|
|
{searchResults ? (
|
2020-01-19 01:06:44 -06:00
|
|
|
<ClaimList
|
|
|
|
header={false}
|
|
|
|
headerAltControls={null}
|
|
|
|
id={`search-results-for-${claimId}`}
|
|
|
|
loading={false}
|
|
|
|
showHiddenByUser={false}
|
|
|
|
uris={searchResults}
|
|
|
|
/>
|
2020-01-03 11:36:15 -05:00
|
|
|
) : (
|
|
|
|
<ChannelContent uri={uri} channelIsBlackListed={channelIsBlackListed} />
|
2019-08-01 20:56:25 -04:00
|
|
|
)}
|
2020-01-03 11:36:15 -05:00
|
|
|
</TabPanel>
|
|
|
|
<TabPanel>
|
|
|
|
{editing ? (
|
|
|
|
<ChannelEdit
|
2019-06-28 13:00:29 -04:00
|
|
|
uri={uri}
|
2020-01-03 11:36:15 -05:00
|
|
|
setEditing={setEditing}
|
|
|
|
updateThumb={v => setThumbPreview(v)}
|
|
|
|
updateCover={v => setCoverPreview(v)}
|
2019-06-28 13:00:29 -04:00
|
|
|
/>
|
2019-10-18 18:28:43 +01:00
|
|
|
) : (
|
2020-01-03 11:36:15 -05:00
|
|
|
<ChannelAbout uri={uri} />
|
2019-10-18 18:28:43 +01:00
|
|
|
)}
|
2020-01-03 11:36:15 -05:00
|
|
|
</TabPanel>
|
|
|
|
<TabPanel>
|
|
|
|
<ChannelDiscussion uri={uri} />
|
|
|
|
</TabPanel>
|
|
|
|
</TabPanels>
|
|
|
|
</Tabs>
|
2019-03-28 12:53:13 -04:00
|
|
|
</Page>
|
|
|
|
);
|
2017-05-03 23:44:08 -04:00
|
|
|
}
|
|
|
|
|
2019-04-04 17:05:23 -04:00
|
|
|
export default withRouter(ChannelPage);
|