2019-03-04 23:46:57 -05:00
|
|
|
// @flow
|
|
|
|
import { createSelector } from 'reselect';
|
2019-08-02 02:28:14 -04:00
|
|
|
import {
|
|
|
|
makeSelectClaimForUri,
|
|
|
|
selectClaimsByUri,
|
|
|
|
makeSelectClaimsInChannelForCurrentPageState,
|
|
|
|
makeSelectClaimIsNsfw,
|
2019-09-09 13:31:00 -04:00
|
|
|
makeSelectRecommendedContentForUri,
|
2019-12-10 14:45:41 -05:00
|
|
|
makeSelectMediaTypeForUri,
|
2020-01-28 13:17:07 -05:00
|
|
|
selectBlockedChannels,
|
|
|
|
parseURI,
|
2019-08-02 02:28:14 -04:00
|
|
|
} from 'lbry-redux';
|
2020-01-28 13:17:07 -05:00
|
|
|
import { selectAllCostInfoByUri } from 'lbryinc';
|
2019-08-02 02:28:14 -04:00
|
|
|
import { selectShowMatureContent } from 'redux/selectors/settings';
|
2019-03-31 19:04:01 -04:00
|
|
|
|
|
|
|
const RECENT_HISTORY_AMOUNT = 10;
|
|
|
|
const HISTORY_ITEMS_PER_PAGE = 50;
|
2019-03-04 23:46:57 -05:00
|
|
|
|
|
|
|
export const selectState = (state: any) => state.content || {};
|
|
|
|
|
|
|
|
export const selectPlayingUri = createSelector(
|
|
|
|
selectState,
|
|
|
|
state => state.playingUri
|
|
|
|
);
|
|
|
|
|
2019-08-02 02:28:14 -04:00
|
|
|
export const makeSelectIsPlaying = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
selectPlayingUri,
|
|
|
|
playingUri => playingUri === uri
|
|
|
|
);
|
|
|
|
|
2019-03-04 23:46:57 -05:00
|
|
|
export const makeSelectContentPositionForUri = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
selectState,
|
|
|
|
makeSelectClaimForUri(uri),
|
|
|
|
(state, claim) => {
|
|
|
|
if (!claim) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const outpoint = `${claim.txid}:${claim.nout}`;
|
|
|
|
const id = claim.claim_id;
|
|
|
|
return state.positions[id] ? state.positions[id][outpoint] : null;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2019-03-31 19:04:01 -04:00
|
|
|
export const selectHistory = createSelector(
|
2019-03-04 23:46:57 -05:00
|
|
|
selectState,
|
2019-03-31 19:04:01 -04:00
|
|
|
state => state.history || []
|
|
|
|
);
|
|
|
|
|
|
|
|
export const selectHistoryPageCount = createSelector(
|
|
|
|
selectHistory,
|
|
|
|
history => Math.ceil(history.length / HISTORY_ITEMS_PER_PAGE)
|
2019-03-04 23:46:57 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
export const makeSelectHistoryForPage = (page: number) =>
|
|
|
|
createSelector(
|
2019-03-31 19:04:01 -04:00
|
|
|
selectHistory,
|
2019-03-04 23:46:57 -05:00
|
|
|
selectClaimsByUri,
|
2019-03-31 19:04:01 -04:00
|
|
|
(history, claimsByUri) => {
|
2019-03-04 23:46:57 -05:00
|
|
|
const left = page * HISTORY_ITEMS_PER_PAGE;
|
2019-03-31 19:04:01 -04:00
|
|
|
const historyItemsForPage = history.slice(left, left + HISTORY_ITEMS_PER_PAGE);
|
|
|
|
return historyItemsForPage;
|
2019-03-04 23:46:57 -05:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
export const makeSelectHistoryForUri = (uri: string) =>
|
|
|
|
createSelector(
|
2019-03-31 19:04:01 -04:00
|
|
|
selectHistory,
|
|
|
|
history => history.find(i => i.uri === uri)
|
2019-03-04 23:46:57 -05:00
|
|
|
);
|
|
|
|
|
2019-07-11 14:06:25 -04:00
|
|
|
export const makeSelectHasVisitedUri = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
makeSelectHistoryForUri(uri),
|
|
|
|
history => Boolean(history)
|
|
|
|
);
|
|
|
|
|
2019-09-09 13:31:00 -04:00
|
|
|
export const makeSelectNextUnplayedRecommended = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
makeSelectRecommendedContentForUri(uri),
|
|
|
|
selectHistory,
|
2020-01-28 13:17:07 -05:00
|
|
|
selectClaimsByUri,
|
|
|
|
selectAllCostInfoByUri,
|
|
|
|
selectBlockedChannels,
|
|
|
|
(
|
|
|
|
recommendedForUri: Array<string>,
|
|
|
|
history: Array<{ uri: string }>,
|
|
|
|
claimsByUri: { [string]: ?Claim },
|
|
|
|
costInfoByUri: { [string]: { cost: 0 | string } },
|
|
|
|
blockedChannels: Array<string>
|
|
|
|
) => {
|
|
|
|
if (recommendedForUri) {
|
|
|
|
// Make sure we don't autoplay paid content, channels, or content from blocked channels
|
|
|
|
for (let i = 0; i < recommendedForUri.length; i++) {
|
|
|
|
const recommendedUri = recommendedForUri[i];
|
2020-02-05 10:07:48 -05:00
|
|
|
const claim = claimsByUri[recommendedUri];
|
|
|
|
|
|
|
|
if (!claim) {
|
|
|
|
continue;
|
|
|
|
}
|
2020-01-28 13:17:07 -05:00
|
|
|
|
|
|
|
const { isChannel } = parseURI(recommendedUri);
|
|
|
|
if (isChannel) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const costInfo = costInfoByUri[recommendedUri];
|
|
|
|
if (!costInfo || costInfo.cost !== 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-02-05 10:07:48 -05:00
|
|
|
// We already check if it's a channel above
|
|
|
|
// $FlowFixMe
|
|
|
|
const isVideo = claim.value && claim.value.stream_type === 'video';
|
2020-02-05 15:46:44 -05:00
|
|
|
// $FlowFixMe
|
|
|
|
const isAudio = claim.value && claim.value.stream_type === 'audio';
|
2020-02-07 01:22:48 -05:00
|
|
|
if (!isVideo && !isAudio) {
|
2020-02-05 10:07:48 -05:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-28 13:17:07 -05:00
|
|
|
const channel = claim && claim.signing_channel;
|
2020-01-29 11:16:03 -05:00
|
|
|
if (channel && blockedChannels.some(blockedUri => blockedUri === channel.permanent_url)) {
|
2020-01-28 13:17:07 -05:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-29 11:16:03 -05:00
|
|
|
if (!history.some(item => item.uri === recommendedForUri[i])) {
|
2020-01-28 13:17:07 -05:00
|
|
|
return recommendedForUri[i];
|
2019-09-09 13:31:00 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2019-03-31 19:04:01 -04:00
|
|
|
export const selectRecentHistory = createSelector(
|
|
|
|
selectHistory,
|
|
|
|
history => {
|
|
|
|
return history.slice(0, RECENT_HISTORY_AMOUNT);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2019-03-04 23:46:57 -05:00
|
|
|
export const makeSelectCategoryListUris = (uris: ?Array<string>, channel: string) =>
|
|
|
|
createSelector(
|
2019-03-28 12:53:13 -04:00
|
|
|
makeSelectClaimsInChannelForCurrentPageState(channel),
|
2019-03-04 23:46:57 -05:00
|
|
|
channelClaims => {
|
|
|
|
if (uris) return uris;
|
|
|
|
|
|
|
|
if (channelClaims) {
|
|
|
|
const CATEGORY_LIST_SIZE = 10;
|
2019-05-07 17:38:29 -04:00
|
|
|
return channelClaims.slice(0, CATEGORY_LIST_SIZE).map(({ name, claim_id: claimId }) => `${name}#${claimId}`);
|
2019-03-04 23:46:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
);
|
2019-08-02 02:28:14 -04:00
|
|
|
|
|
|
|
export const makeSelectShouldObscurePreview = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
selectShowMatureContent,
|
|
|
|
makeSelectClaimIsNsfw(uri),
|
|
|
|
(showMatureContent, isClaimMature) => {
|
|
|
|
return isClaimMature && !showMatureContent;
|
|
|
|
}
|
|
|
|
);
|
2019-12-10 14:45:41 -05:00
|
|
|
|
|
|
|
export const makeSelectCanAutoplay = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
makeSelectMediaTypeForUri(uri),
|
|
|
|
mediaType => {
|
|
|
|
const canAutoPlay = ['audio', 'video', 'image', 'text', 'document'].includes(mediaType);
|
|
|
|
return canAutoPlay;
|
|
|
|
}
|
|
|
|
);
|
2020-01-06 15:57:49 -05:00
|
|
|
|
|
|
|
export const makeSelectIsText = (uri: string) =>
|
|
|
|
createSelector(
|
|
|
|
makeSelectMediaTypeForUri(uri),
|
|
|
|
mediaType => {
|
2020-01-26 17:09:23 +13:00
|
|
|
const isText = ['text', 'document', 'script'].includes(mediaType);
|
2020-01-06 15:57:49 -05:00
|
|
|
return isText;
|
|
|
|
}
|
|
|
|
);
|