// @flow import type { Node } from 'react'; import classnames from 'classnames'; import React, { Fragment, useEffect, useState } from 'react'; import { withRouter } from 'react-router'; import * as CS from 'constants/claim_search'; import { createNormalizedClaimSearchKey, MATURE_TAGS } from 'lbry-redux'; import { FormField } from 'component/common/form'; import Button from 'component/button'; import moment from 'moment'; import ClaimList from 'component/claimList'; import ClaimPreview from 'component/claimPreview'; import { toCapitalCase } from 'util/string'; import I18nMessage from 'component/i18nMessage'; import * as ICONS from 'constants/icons'; type Props = { uris: Array, subscribedChannels: Array, doClaimSearch: ({}) => void, loading: boolean, personalView: boolean, doToggleTagFollowDesktop: string => void, meta?: Node, showNsfw: boolean, showReposts: boolean, history: { action: string, push: string => void, replace: string => void }, location: { search: string, pathname: string }, claimSearchByQuery: { [string]: Array, }, hiddenUris: Array, hiddenNsfwMessage?: Node, channelIds?: Array, tags: Array, orderBy?: Array, defaultOrderBy?: string, freshness?: string, defaultFreshness?: string, header?: Node, headerLabel?: string | Node, name?: string, hideBlock?: boolean, hideFilter?: boolean, claimType?: Array, defaultClaimType?: Array, streamType?: string | Array, defaultStreamType?: string | Array, renderProperties?: Claim => Node, includeSupportAction?: boolean, pageSize?: number, }; function ClaimListDiscover(props: Props) { const { doClaimSearch, claimSearchByQuery, tags, loading, meta, channelIds, showNsfw, showReposts, history, location, hiddenUris, hiddenNsfwMessage, defaultOrderBy, orderBy, headerLabel, header, name, claimType, pageSize, hideBlock, defaultClaimType, streamType, defaultStreamType, freshness, defaultFreshness = CS.FRESH_WEEK, renderProperties, includeSupportAction, hideFilter, } = props; const didNavigateForward = history.action === 'PUSH'; const { search } = location; const [page, setPage] = useState(1); const [forceRefresh, setForceRefresh] = useState(); const [expanded, setExpanded] = useState(false); const urlParams = new URLSearchParams(search); const tagsParam = tags || urlParams.get(CS.TAGS_KEY) || null; const orderParam = orderBy || urlParams.get(CS.ORDER_BY_KEY) || defaultOrderBy || CS.ORDER_BY_TRENDING; const freshnessParam = freshness || urlParams.get(CS.FRESH_KEY) || defaultFreshness; const contentTypeParam = urlParams.get(CS.CONTENT_KEY); const claimTypeParam = claimType || (CS.CLAIM_TYPES.includes(contentTypeParam) && contentTypeParam) || defaultClaimType || null; const streamTypeParam = streamType || (CS.FILE_TYPES.includes(contentTypeParam) && contentTypeParam) || defaultStreamType || null; const durationParam = urlParams.get(CS.DURATION_KEY) || null; const showDuration = !(claimType && claimType === CS.CLAIM_CHANNEL); const isFiltered = () => Boolean(urlParams.get(CS.FRESH_KEY) || urlParams.get(CS.CONTENT_KEY) || urlParams.get(CS.DURATION_KEY)); useEffect(() => { if (isFiltered()) setExpanded(true); // eslint-disable-next-line react-hooks/exhaustive-deps }, []); const options: { page_size: number, page: number, no_totals: boolean, any_tags: Array, not_tags: Array, channel_ids: Array, not_channel_ids: Array, order_by: Array, release_time?: string, claim_type?: Array, name?: string, duration?: string, stream_types?: any, } = { page_size: pageSize || CS.PAGE_SIZE, page, name, claim_type: claimType || undefined, // no_totals makes it so the sdk doesn't have to calculate total number pages for pagination // it's faster, but we will need to remove it if we start using total_pages no_totals: true, any_tags: tagsParam || [], channel_ids: channelIds || [], not_channel_ids: // If channelIds were passed in, we don't need not_channel_ids !channelIds && hiddenUris && hiddenUris.length ? hiddenUris.map(hiddenUri => hiddenUri.split('#')[1]) : [], not_tags: !showNsfw ? MATURE_TAGS : [], order_by: orderParam === CS.ORDER_BY_TRENDING ? CS.ORDER_BY_TRENDING_VALUE : orderParam === CS.ORDER_BY_NEW ? CS.ORDER_BY_NEW_VALUE : CS.ORDER_BY_TOP_VALUE, // Sort by top }; if (orderParam === CS.ORDER_BY_TOP && freshnessParam !== CS.FRESH_ALL) { options.release_time = `>${Math.floor( moment() .subtract(1, freshnessParam) .startOf('hour') .unix() )}`; } else if (orderParam === CS.ORDER_BY_NEW || orderParam === CS.ORDER_BY_TRENDING) { // Warning - hack below // If users are following more than 10 channels or tags, limit results to stuff less than a year old // For more than 20, drop it down to 6 months // This helps with timeout issues for users that are following a ton of stuff // https://github.com/lbryio/lbry-sdk/issues/2420 if ( (options.channel_ids && options.channel_ids.length > 20) || (options.any_tags && options.any_tags.length > 20) ) { options.release_time = `>${Math.floor( moment() .subtract(3, CS.FRESH_MONTH) .startOf('week') .unix() )}`; } else if ( (options.channel_ids && options.channel_ids.length > 10) || (options.any_tags && options.any_tags.length > 10) ) { options.release_time = `>${Math.floor( moment() .subtract(1, CS.FRESH_YEAR) .startOf('week') .unix() )}`; } else { // Hack for at least the New page until https://github.com/lbryio/lbry-sdk/issues/2591 is fixed options.release_time = `<${Math.floor( moment() .startOf('minute') .unix() )}`; } } if (durationParam) { if (durationParam === CS.DURATION_SHORT) { options.duration = '<=1800'; } else if (durationParam === CS.DURATION_LONG) { options.duration = '>=1800'; } } if (streamTypeParam) { if (streamTypeParam !== CS.CONTENT_ALL) { options.stream_types = [streamTypeParam]; } } if (claimTypeParam) { if (claimTypeParam !== CS.CONTENT_ALL) { if (Array.isArray(claimTypeParam)) { options.claim_type = claimTypeParam; } else { options.claim_type = [claimTypeParam]; } } } // https://github.com/lbryio/lbry-desktop/issues/3774 // if (!showReposts) { // if (Array.isArray(options.claim_type)) { // options.claim_type = options.claim_type.filter(claimType => claimType !== 'repost'); // } else { // options.claim_type = ['stream', 'channel']; // } // } const hasMatureTags = tags && tags.some(t => MATURE_TAGS.includes(t)); const claimSearchCacheQuery = createNormalizedClaimSearchKey(options); const uris = claimSearchByQuery[claimSearchCacheQuery] || []; const shouldPerformSearch = uris.length === 0 || didNavigateForward || (!loading && uris.length < CS.PAGE_SIZE * page && uris.length % CS.PAGE_SIZE === 0); // Don't use the query from createNormalizedClaimSearchKey for the effect since that doesn't include page & release_time const optionsStringForEffect = JSON.stringify(options); const noResults = (

setForceRefresh(Date.now())} /> ), }} > Sorry, your request returned no results or timed out. Modify your options or %again%

, }} > If you continue to have issues, please %contact_support%.

); function handleChange(change) { const url = buildUrl(change); setPage(1); history.push(url); } function buildUrl(delta) { const newUrlParams = new URLSearchParams(); CS.KEYS.forEach(k => { // $FlowFixMe append() can't take null as second arg, but get() can return null if (urlParams.get(k) !== null) newUrlParams.append(k, urlParams.get(k)); }); switch (delta.key) { case CS.ORDER_BY_KEY: newUrlParams.set(CS.ORDER_BY_KEY, delta.value); break; case CS.FRESH_KEY: if (delta.value === defaultFreshness || delta.value === CS.FRESH_DEFAULT) { newUrlParams.delete(CS.FRESH_KEY); } else { newUrlParams.set(CS.FRESH_KEY, delta.value); } break; case CS.CONTENT_KEY: if (delta.value === CS.CLAIM_CHANNEL || delta.value === CS.CLAIM_REPOST) { newUrlParams.delete(CS.DURATION_KEY); newUrlParams.set(CS.CONTENT_KEY, delta.value); } else if (delta.value === CS.CONTENT_ALL) { newUrlParams.delete(CS.CONTENT_KEY); } else { newUrlParams.set(CS.CONTENT_KEY, delta.value); } break; case CS.DURATION_KEY: if (delta.value === CS.DURATION_ALL) { newUrlParams.delete(CS.DURATION_KEY); } else { newUrlParams.set(CS.DURATION_KEY, delta.value); } break; } return `?${newUrlParams.toString()}`; } function handleScrollBottom() { if (!loading) { setPage(page + 1); } } useEffect(() => { if (shouldPerformSearch) { const searchOptions = JSON.parse(optionsStringForEffect); doClaimSearch(searchOptions); } }, [doClaimSearch, shouldPerformSearch, optionsStringForEffect, forceRefresh]); const defaultHeader = (
{CS.ORDER_BY_TYPES.map(type => (
{!hideFilter && (
{expanded && ( <>
{/* FRESHNESS FIELD */} {orderParam === CS.ORDER_BY_TOP && (
handleChange({ key: CS.FRESH_KEY, value: e.target.value, }) } > {CS.FRESH_TYPES.map(time => ( ))}
)} {/* CONTENT_TYPES FIELD */}
handleChange({ key: CS.CONTENT_KEY, value: e.target.value, }) } > {CS.CONTENT_TYPES.map(type => { if (type !== CS.CLAIM_CHANNEL || (type === CS.CLAIM_CHANNEL && !channelIds)) { return ( ); } })}
{/* DURATIONS FIELD */} {showDuration && (
handleChange({ key: CS.DURATION_KEY, value: e.target.value, }) } > {CS.DURATION_TYPES.map(dur => ( ))}
)}
)}
{hasMatureTags && hiddenNsfwMessage}
); return (
{loading && new Array(CS.PAGE_SIZE).fill(1).map((x, i) => )}
); } export default withRouter(ClaimListDiscover);