2018-03-26 23:32:43 +02:00
|
|
|
// @flow
|
2020-07-02 07:38:25 +02:00
|
|
|
import { MAIN_CLASS } from 'component/page/view';
|
2019-06-17 22:32:38 +02:00
|
|
|
import type { Node } from 'react';
|
2020-06-23 17:22:30 +02:00
|
|
|
import React, { useEffect } from 'react';
|
2019-06-11 20:10:58 +02:00
|
|
|
import classnames from 'classnames';
|
2019-06-28 09:27:55 +02:00
|
|
|
import ClaimPreview from 'component/claimPreview';
|
2019-06-11 20:10:58 +02:00
|
|
|
import Spinner from 'component/spinner';
|
|
|
|
import { FormField } from 'component/common/form';
|
2019-09-27 20:56:15 +02:00
|
|
|
import usePersistedState from 'effects/use-persisted-state';
|
2020-06-23 14:56:38 +02:00
|
|
|
import debounce from 'util/debounce';
|
2020-08-21 17:49:13 +02:00
|
|
|
import ClaimPreviewTile from 'component/claimPreviewTile';
|
2019-06-11 20:10:58 +02:00
|
|
|
|
2020-06-23 14:56:38 +02:00
|
|
|
const DEBOUNCE_SCROLL_HANDLER_MS = 150;
|
2019-06-11 20:10:58 +02:00
|
|
|
const SORT_NEW = 'new';
|
|
|
|
const SORT_OLD = 'old';
|
2018-03-26 23:32:43 +02:00
|
|
|
|
|
|
|
type Props = {
|
2019-06-11 20:10:58 +02:00
|
|
|
uris: Array<string>,
|
2019-06-19 07:05:43 +02:00
|
|
|
header: Node | boolean,
|
2019-06-17 22:32:38 +02:00
|
|
|
headerAltControls: Node,
|
2019-06-11 20:10:58 +02:00
|
|
|
loading: boolean,
|
2019-06-19 07:05:43 +02:00
|
|
|
type: string,
|
2019-06-11 20:10:58 +02:00
|
|
|
empty?: string,
|
2019-06-28 09:33:07 +02:00
|
|
|
defaultSort?: boolean,
|
2019-06-27 08:18:45 +02:00
|
|
|
onScrollBottom?: any => void,
|
2019-07-01 06:35:36 +02:00
|
|
|
page?: number,
|
2019-07-02 04:54:11 +02:00
|
|
|
pageSize?: number,
|
2019-07-23 10:05:51 +02:00
|
|
|
id?: string,
|
2019-06-11 20:10:58 +02:00
|
|
|
// If using the default header, this is a unique ID needed to persist the state of the filter setting
|
|
|
|
persistedStorageKey?: string,
|
2019-08-02 02:56:25 +02:00
|
|
|
showHiddenByUser: boolean,
|
2020-02-05 19:01:07 +01:00
|
|
|
showUnresolvedClaims?: boolean,
|
2020-02-11 20:04:51 +01:00
|
|
|
renderProperties: ?(Claim) => Node,
|
2020-02-12 19:59:48 +01:00
|
|
|
includeSupportAction?: boolean,
|
2020-02-26 20:20:38 +01:00
|
|
|
hideBlock: boolean,
|
2020-03-26 22:47:07 +01:00
|
|
|
injectedItem: ?Node,
|
2020-03-31 18:07:57 +02:00
|
|
|
timedOutMessage?: Node,
|
2020-08-21 17:49:13 +02:00
|
|
|
tileLayout?: boolean,
|
2018-03-26 23:32:43 +02:00
|
|
|
};
|
|
|
|
|
2019-06-19 07:05:43 +02:00
|
|
|
export default function ClaimList(props: Props) {
|
2019-06-27 08:18:45 +02:00
|
|
|
const {
|
|
|
|
uris,
|
|
|
|
headerAltControls,
|
|
|
|
loading,
|
|
|
|
persistedStorageKey,
|
|
|
|
empty,
|
2019-06-28 09:33:07 +02:00
|
|
|
defaultSort,
|
2019-06-27 08:18:45 +02:00
|
|
|
type,
|
|
|
|
header,
|
|
|
|
onScrollBottom,
|
2019-07-02 04:54:11 +02:00
|
|
|
pageSize,
|
2019-07-23 10:05:51 +02:00
|
|
|
page,
|
2019-08-02 02:56:25 +02:00
|
|
|
showHiddenByUser,
|
2020-02-05 19:01:07 +01:00
|
|
|
showUnresolvedClaims,
|
2020-02-11 20:04:51 +01:00
|
|
|
renderProperties,
|
2020-02-12 19:59:48 +01:00
|
|
|
includeSupportAction,
|
2020-02-26 20:20:38 +01:00
|
|
|
hideBlock,
|
2020-03-26 22:47:07 +01:00
|
|
|
injectedItem,
|
2020-03-31 18:07:57 +02:00
|
|
|
timedOutMessage,
|
2020-08-21 17:49:13 +02:00
|
|
|
tileLayout = false,
|
2019-06-27 08:18:45 +02:00
|
|
|
} = props;
|
2020-08-21 17:49:13 +02:00
|
|
|
|
2019-06-11 20:38:08 +02:00
|
|
|
const [currentSort, setCurrentSort] = usePersistedState(persistedStorageKey, SORT_NEW);
|
2020-03-31 18:07:57 +02:00
|
|
|
const timedOut = uris === null;
|
2019-07-17 22:49:06 +02:00
|
|
|
const urisLength = (uris && uris.length) || 0;
|
|
|
|
const sortedUris = (urisLength > 0 && (currentSort === SORT_NEW ? uris : uris.slice().reverse())) || [];
|
2018-03-26 23:32:43 +02:00
|
|
|
|
2019-06-11 20:10:58 +02:00
|
|
|
function handleSortChange() {
|
|
|
|
setCurrentSort(currentSort === SORT_NEW ? SORT_OLD : SORT_NEW);
|
2017-04-30 18:01:43 +02:00
|
|
|
}
|
2017-04-23 18:10:45 +02:00
|
|
|
|
2019-06-27 08:18:45 +02:00
|
|
|
useEffect(() => {
|
2020-06-23 14:56:38 +02:00
|
|
|
const handleScroll = debounce(e => {
|
2020-06-23 17:22:30 +02:00
|
|
|
if (page && pageSize && onScrollBottom) {
|
2020-07-02 07:38:25 +02:00
|
|
|
const mainEl = document.querySelector(`.${MAIN_CLASS}`);
|
2020-06-15 12:08:17 +02:00
|
|
|
|
2020-07-02 07:38:25 +02:00
|
|
|
if (mainEl && !loading && urisLength >= pageSize) {
|
|
|
|
const contentWrapperAtBottomOfPage = mainEl.getBoundingClientRect().bottom - 0.5 <= window.innerHeight;
|
2019-06-28 09:33:07 +02:00
|
|
|
|
2020-06-23 14:56:38 +02:00
|
|
|
if (contentWrapperAtBottomOfPage) {
|
|
|
|
onScrollBottom();
|
2020-02-26 21:48:28 +01:00
|
|
|
}
|
2019-06-28 09:33:07 +02:00
|
|
|
}
|
2020-06-23 14:56:38 +02:00
|
|
|
}
|
|
|
|
}, DEBOUNCE_SCROLL_HANDLER_MS);
|
2019-06-28 09:33:07 +02:00
|
|
|
|
2019-06-27 08:18:45 +02:00
|
|
|
if (onScrollBottom) {
|
|
|
|
window.addEventListener('scroll', handleScroll);
|
2020-06-23 14:56:38 +02:00
|
|
|
return () => window.removeEventListener('scroll', handleScroll);
|
2019-06-27 08:18:45 +02:00
|
|
|
}
|
2020-06-23 17:22:30 +02:00
|
|
|
}, [loading, onScrollBottom, urisLength, pageSize, page]);
|
2019-06-27 08:18:45 +02:00
|
|
|
|
2020-08-21 17:49:13 +02:00
|
|
|
return tileLayout && !header ? (
|
|
|
|
<section className="claim-grid">
|
|
|
|
{urisLength > 0 && uris.map(uri => <ClaimPreviewTile key={uri} uri={uri} />)}
|
2020-09-15 16:05:04 +02:00
|
|
|
{!timedOut && urisLength === 0 && !loading && (
|
|
|
|
<div className="empty main--empty">{empty || __('No results')}</div>
|
|
|
|
)}
|
|
|
|
{timedOut && timedOutMessage && <div className="empty main--empty">{timedOutMessage}</div>}
|
2020-08-21 17:49:13 +02:00
|
|
|
</section>
|
|
|
|
) : (
|
2019-06-28 09:33:07 +02:00
|
|
|
<section
|
|
|
|
className={classnames('claim-list', {
|
|
|
|
'claim-list--small': type === 'small',
|
|
|
|
})}
|
|
|
|
>
|
2019-06-19 07:05:43 +02:00
|
|
|
{header !== false && (
|
2020-01-02 21:36:03 +01:00
|
|
|
<React.Fragment>
|
2020-03-19 17:54:37 +01:00
|
|
|
{header && (
|
|
|
|
<div className={classnames('claim-list__header', { 'section__title--small': type === 'small' })}>
|
|
|
|
{header}
|
|
|
|
{loading && <Spinner type="small" />}
|
|
|
|
<div className="claim-list__alt-controls">
|
|
|
|
{headerAltControls}
|
|
|
|
{defaultSort && (
|
|
|
|
<FormField
|
|
|
|
className="claim-list__dropdown"
|
|
|
|
type="select"
|
|
|
|
name="file_sort"
|
|
|
|
value={currentSort}
|
|
|
|
onChange={handleSortChange}
|
|
|
|
>
|
|
|
|
<option value={SORT_NEW}>{__('Newest First')}</option>
|
|
|
|
<option value={SORT_OLD}>{__('Oldest First')}</option>
|
|
|
|
</FormField>
|
|
|
|
)}
|
|
|
|
</div>
|
2020-01-02 21:36:03 +01:00
|
|
|
</div>
|
2020-03-19 17:54:37 +01:00
|
|
|
)}
|
2020-01-02 21:36:03 +01:00
|
|
|
</React.Fragment>
|
2019-06-11 20:10:58 +02:00
|
|
|
)}
|
2019-08-19 06:49:54 +02:00
|
|
|
|
2019-07-17 22:49:06 +02:00
|
|
|
{urisLength > 0 && (
|
2020-04-29 21:31:11 +02:00
|
|
|
<ul
|
|
|
|
className={classnames('ul--no-style', {
|
2020-09-08 18:52:51 +02:00
|
|
|
card: !(tileLayout || type === 'small'),
|
2020-08-21 17:49:13 +02:00
|
|
|
'claim-list--card-body': tileLayout,
|
2020-04-29 21:31:11 +02:00
|
|
|
})}
|
|
|
|
>
|
2019-06-11 20:10:58 +02:00
|
|
|
{sortedUris.map((uri, index) => (
|
2020-03-26 22:47:07 +01:00
|
|
|
<React.Fragment key={uri}>
|
|
|
|
{injectedItem && index === 4 && <li>{injectedItem}</li>}
|
|
|
|
<ClaimPreview
|
|
|
|
uri={uri}
|
|
|
|
type={type}
|
|
|
|
includeSupportAction={includeSupportAction}
|
|
|
|
showUnresolvedClaim={showUnresolvedClaims}
|
|
|
|
properties={renderProperties || (type !== 'small' ? undefined : false)}
|
|
|
|
showUserBlocked={showHiddenByUser}
|
|
|
|
hideBlock={hideBlock}
|
|
|
|
customShouldHide={(claim: StreamClaim) => {
|
|
|
|
// Hack to hide spee.ch thumbnail publishes
|
|
|
|
// If it meets these requirements, it was probably uploaded here:
|
|
|
|
// https://github.com/lbryio/lbry-redux/blob/master/src/redux/actions/publish.js#L74-L79
|
|
|
|
if (claim.name.length === 24 && !claim.name.includes(' ') && claim.value.author === 'Spee.ch') {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</React.Fragment>
|
2019-06-11 20:10:58 +02:00
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
)}
|
2020-08-21 17:49:13 +02:00
|
|
|
|
2020-03-31 18:07:57 +02:00
|
|
|
{!timedOut && urisLength === 0 && !loading && (
|
2020-04-01 20:43:50 +02:00
|
|
|
<div className="empty empty--centered">{empty || __('No results')}</div>
|
2020-01-03 21:43:49 +01:00
|
|
|
)}
|
2020-04-01 20:43:50 +02:00
|
|
|
{timedOut && timedOutMessage && <div className="empty empty--centered">{timedOutMessage}</div>}
|
2019-06-11 20:10:58 +02:00
|
|
|
</section>
|
|
|
|
);
|
2017-04-30 18:01:43 +02:00
|
|
|
}
|