Incremental selector memoization fixes (#92)
This commit is contained in:
commit
9041e5e38d
29 changed files with 358 additions and 304 deletions
1
flow-typed/Comment.js
vendored
1
flow-typed/Comment.js
vendored
|
@ -45,6 +45,7 @@ declare type CommentsState = {
|
||||||
isLoading: boolean,
|
isLoading: boolean,
|
||||||
isLoadingById: boolean,
|
isLoadingById: boolean,
|
||||||
isLoadingByParentId: { [string]: boolean },
|
isLoadingByParentId: { [string]: boolean },
|
||||||
|
isCommenting: boolean,
|
||||||
myComments: ?Set<string>,
|
myComments: ?Set<string>,
|
||||||
isFetchingReacts: boolean,
|
isFetchingReacts: boolean,
|
||||||
myReactsByCommentId: ?{ [string]: Array<string> }, // {"CommentId:MyChannelId": ["like", "dislike", ...]}
|
myReactsByCommentId: ?{ [string]: Array<string> }, // {"CommentId:MyChannelId": ["like", "dislike", ...]}
|
||||||
|
|
|
@ -50,7 +50,6 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ungap/from-entries": "^0.2.1",
|
"@ungap/from-entries": "^0.2.1",
|
||||||
"proxy-polyfill": "0.1.6",
|
|
||||||
"auto-launch": "^5.0.5",
|
"auto-launch": "^5.0.5",
|
||||||
"electron-dl": "^1.11.0",
|
"electron-dl": "^1.11.0",
|
||||||
"electron-log": "^2.2.12",
|
"electron-log": "^2.2.12",
|
||||||
|
@ -61,6 +60,8 @@
|
||||||
"if-env": "^1.0.4",
|
"if-env": "^1.0.4",
|
||||||
"match-sorter": "^6.3.0",
|
"match-sorter": "^6.3.0",
|
||||||
"parse-duration": "^1.0.0",
|
"parse-duration": "^1.0.0",
|
||||||
|
"proxy-polyfill": "0.1.6",
|
||||||
|
"re-reselect": "^4.0.0",
|
||||||
"react-datetime-picker": "^3.2.1",
|
"react-datetime-picker": "^3.2.1",
|
||||||
"react-plastic": "^1.1.1",
|
"react-plastic": "^1.1.1",
|
||||||
"react-top-loading-bar": "^2.0.1",
|
"react-top-loading-bar": "^2.0.1",
|
||||||
|
|
|
@ -4,12 +4,12 @@ import { selectSubscriptions } from 'redux/selectors/subscriptions';
|
||||||
import { withRouter } from 'react-router';
|
import { withRouter } from 'react-router';
|
||||||
import { makeSelectClaimForUri } from 'redux/selectors/claims';
|
import { makeSelectClaimForUri } from 'redux/selectors/claims';
|
||||||
import { doResolveUris } from 'redux/actions/claims';
|
import { doResolveUris } from 'redux/actions/claims';
|
||||||
import { makeSelectTopLevelCommentsForUri } from 'redux/selectors/comments';
|
import { selectTopLevelCommentsForUri } from 'redux/selectors/comments';
|
||||||
import ChannelMentionSuggestions from './view';
|
import ChannelMentionSuggestions from './view';
|
||||||
|
|
||||||
const select = (state, props) => {
|
const select = (state, props) => {
|
||||||
const subscriptionUris = selectSubscriptions(state).map(({ uri }) => uri);
|
const subscriptionUris = selectSubscriptions(state).map(({ uri }) => uri);
|
||||||
const topLevelComments = makeSelectTopLevelCommentsForUri(props.uri)(state);
|
const topLevelComments = selectTopLevelCommentsForUri(state, props.uri);
|
||||||
|
|
||||||
const commentorUris = [];
|
const commentorUris = [];
|
||||||
// Avoid repeated commentors
|
// Avoid repeated commentors
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import {
|
import {
|
||||||
makeSelectClaimForUri,
|
selectClaimForUri,
|
||||||
makeSelectIsUriResolving,
|
makeSelectIsUriResolving,
|
||||||
makeSelectClaimIsMine,
|
makeSelectClaimIsMine,
|
||||||
makeSelectClaimIsPending,
|
makeSelectClaimIsPending,
|
||||||
|
@ -9,7 +9,7 @@ import {
|
||||||
makeSelectClaimWasPurchased,
|
makeSelectClaimWasPurchased,
|
||||||
makeSelectClaimIsStreamPlaceholder,
|
makeSelectClaimIsStreamPlaceholder,
|
||||||
makeSelectTitleForUri,
|
makeSelectTitleForUri,
|
||||||
makeSelectDateForUri,
|
selectDateForUri,
|
||||||
} from 'redux/selectors/claims';
|
} from 'redux/selectors/claims';
|
||||||
import { makeSelectStreamingUrlForUri } from 'redux/selectors/file_info';
|
import { makeSelectStreamingUrlForUri } from 'redux/selectors/file_info';
|
||||||
import {
|
import {
|
||||||
|
@ -32,14 +32,14 @@ import ClaimPreview from './view';
|
||||||
import formatMediaDuration from 'util/formatMediaDuration';
|
import formatMediaDuration from 'util/formatMediaDuration';
|
||||||
|
|
||||||
const select = (state, props) => {
|
const select = (state, props) => {
|
||||||
const claim = props.uri && makeSelectClaimForUri(props.uri)(state);
|
const claim = props.uri && selectClaimForUri(state, props.uri);
|
||||||
const media = claim && claim.value && (claim.value.video || claim.value.audio);
|
const media = claim && claim.value && (claim.value.video || claim.value.audio);
|
||||||
const mediaDuration = media && media.duration && formatMediaDuration(media.duration, { screenReader: true });
|
const mediaDuration = media && media.duration && formatMediaDuration(media.duration, { screenReader: true });
|
||||||
|
|
||||||
return {
|
return {
|
||||||
claim,
|
claim,
|
||||||
mediaDuration,
|
mediaDuration,
|
||||||
date: props.uri && makeSelectDateForUri(props.uri)(state),
|
date: props.uri && selectDateForUri(state, props.uri),
|
||||||
title: props.uri && makeSelectTitleForUri(props.uri)(state),
|
title: props.uri && makeSelectTitleForUri(props.uri)(state),
|
||||||
pending: props.uri && makeSelectClaimIsPending(props.uri)(state),
|
pending: props.uri && makeSelectClaimIsPending(props.uri)(state),
|
||||||
reflectingProgress: props.uri && makeSelectReflectingClaimForUri(props.uri)(state),
|
reflectingProgress: props.uri && makeSelectReflectingClaimForUri(props.uri)(state),
|
||||||
|
@ -47,7 +47,6 @@ const select = (state, props) => {
|
||||||
claimIsMine: props.uri && makeSelectClaimIsMine(props.uri)(state),
|
claimIsMine: props.uri && makeSelectClaimIsMine(props.uri)(state),
|
||||||
isResolvingUri: props.uri && makeSelectIsUriResolving(props.uri)(state),
|
isResolvingUri: props.uri && makeSelectIsUriResolving(props.uri)(state),
|
||||||
isResolvingRepost: props.uri && makeSelectIsUriResolving(props.repostUrl)(state),
|
isResolvingRepost: props.uri && makeSelectIsUriResolving(props.repostUrl)(state),
|
||||||
repostClaim: props.uri && makeSelectClaimForUri(props.uri)(state),
|
|
||||||
nsfw: props.uri && makeSelectClaimIsNsfw(props.uri)(state),
|
nsfw: props.uri && makeSelectClaimIsNsfw(props.uri)(state),
|
||||||
blackListedOutpoints: selectBlackListedOutpoints(state),
|
blackListedOutpoints: selectBlackListedOutpoints(state),
|
||||||
filteredOutpoints: selectFilteredOutpoints(state),
|
filteredOutpoints: selectFilteredOutpoints(state),
|
||||||
|
|
|
@ -7,7 +7,7 @@ import {
|
||||||
makeSelectChannelForClaimUri,
|
makeSelectChannelForClaimUri,
|
||||||
makeSelectClaimIsNsfw,
|
makeSelectClaimIsNsfw,
|
||||||
makeSelectClaimIsStreamPlaceholder,
|
makeSelectClaimIsStreamPlaceholder,
|
||||||
makeSelectDateForUri,
|
selectDateForUri,
|
||||||
} from 'redux/selectors/claims';
|
} from 'redux/selectors/claims';
|
||||||
import { doFileGet } from 'redux/actions/file';
|
import { doFileGet } from 'redux/actions/file';
|
||||||
import { doResolveUri } from 'redux/actions/claims';
|
import { doResolveUri } from 'redux/actions/claims';
|
||||||
|
@ -26,7 +26,7 @@ const select = (state, props) => {
|
||||||
return {
|
return {
|
||||||
claim,
|
claim,
|
||||||
mediaDuration,
|
mediaDuration,
|
||||||
date: props.uri && makeSelectDateForUri(props.uri)(state),
|
date: props.uri && selectDateForUri(state, props.uri),
|
||||||
channel: props.uri && makeSelectChannelForClaimUri(props.uri)(state),
|
channel: props.uri && makeSelectChannelForClaimUri(props.uri)(state),
|
||||||
isResolvingUri: props.uri && makeSelectIsUriResolving(props.uri)(state),
|
isResolvingUri: props.uri && makeSelectIsUriResolving(props.uri)(state),
|
||||||
thumbnail: props.uri && makeSelectThumbnailForUri(props.uri)(state),
|
thumbnail: props.uri && makeSelectThumbnailForUri(props.uri)(state),
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { makeSelectTagsForUri } from 'redux/selectors/claims';
|
import { selectTagsForUri } from 'redux/selectors/claims';
|
||||||
import { selectFollowedTags } from 'redux/selectors/tags';
|
import { selectFollowedTags } from 'redux/selectors/tags';
|
||||||
import ClaimTags from './view';
|
import ClaimTags from './view';
|
||||||
|
|
||||||
const select = (state, props) => ({
|
const select = (state, props) => ({
|
||||||
tags: makeSelectTagsForUri(props.uri)(state),
|
tags: selectTagsForUri(state, props.uri),
|
||||||
followedTags: selectFollowedTags(state),
|
followedTags: selectFollowedTags(state),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,7 @@ import { doSetPlayingUri } from 'redux/actions/content';
|
||||||
import { selectUserVerifiedEmail } from 'redux/selectors/user';
|
import { selectUserVerifiedEmail } from 'redux/selectors/user';
|
||||||
import {
|
import {
|
||||||
selectLinkedCommentAncestors,
|
selectLinkedCommentAncestors,
|
||||||
makeSelectOthersReactionsForComment,
|
selectOthersReactsForComment,
|
||||||
makeSelectTotalReplyPagesForParentId,
|
makeSelectTotalReplyPagesForParentId,
|
||||||
} from 'redux/selectors/comments';
|
} from 'redux/selectors/comments';
|
||||||
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
||||||
|
@ -29,7 +29,7 @@ const select = (state, props) => {
|
||||||
thumbnail: props.authorUri && makeSelectThumbnailForUri(props.authorUri)(state),
|
thumbnail: props.authorUri && makeSelectThumbnailForUri(props.authorUri)(state),
|
||||||
channelIsBlocked: props.authorUri && makeSelectChannelIsMuted(props.authorUri)(state),
|
channelIsBlocked: props.authorUri && makeSelectChannelIsMuted(props.authorUri)(state),
|
||||||
commentingEnabled: IS_WEB ? Boolean(selectUserVerifiedEmail(state)) : true,
|
commentingEnabled: IS_WEB ? Boolean(selectUserVerifiedEmail(state)) : true,
|
||||||
othersReacts: makeSelectOthersReactionsForComment(reactionKey)(state),
|
othersReacts: selectOthersReactsForComment(state, reactionKey),
|
||||||
activeChannelClaim,
|
activeChannelClaim,
|
||||||
myChannels: selectMyChannelClaims(state),
|
myChannels: selectMyChannelClaims(state),
|
||||||
playingUri: selectPlayingUri(state),
|
playingUri: selectPlayingUri(state),
|
||||||
|
|
|
@ -3,7 +3,7 @@ import Comment from './view';
|
||||||
import { makeSelectClaimIsMine, makeSelectClaimForUri } from 'redux/selectors/claims';
|
import { makeSelectClaimIsMine, makeSelectClaimForUri } from 'redux/selectors/claims';
|
||||||
import { doResolveUri } from 'redux/actions/claims';
|
import { doResolveUri } from 'redux/actions/claims';
|
||||||
import { doToast } from 'redux/actions/notifications';
|
import { doToast } from 'redux/actions/notifications';
|
||||||
import { makeSelectMyReactionsForComment, makeSelectOthersReactionsForComment } from 'redux/selectors/comments';
|
import { selectMyReactsForComment, selectOthersReactsForComment } from 'redux/selectors/comments';
|
||||||
import { doCommentReact } from 'redux/actions/comments';
|
import { doCommentReact } from 'redux/actions/comments';
|
||||||
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
||||||
|
|
||||||
|
@ -15,8 +15,8 @@ const select = (state, props) => {
|
||||||
return {
|
return {
|
||||||
claim: makeSelectClaimForUri(props.uri)(state),
|
claim: makeSelectClaimForUri(props.uri)(state),
|
||||||
claimIsMine: makeSelectClaimIsMine(props.uri)(state),
|
claimIsMine: makeSelectClaimIsMine(props.uri)(state),
|
||||||
myReacts: makeSelectMyReactionsForComment(reactionKey)(state),
|
myReacts: selectMyReactsForComment(state, reactionKey),
|
||||||
othersReacts: makeSelectOthersReactionsForComment(reactionKey)(state),
|
othersReacts: selectOthersReactsForComment(state, reactionKey),
|
||||||
activeChannelId,
|
activeChannelId,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -73,12 +73,12 @@ export default function CommentReactions(props: Props) {
|
||||||
const shouldHide = !canCreatorReact && hideCreatorLike;
|
const shouldHide = !canCreatorReact && hideCreatorLike;
|
||||||
const creatorLiked = getCountForReact(REACTION_TYPES.CREATOR_LIKE) > 0;
|
const creatorLiked = getCountForReact(REACTION_TYPES.CREATOR_LIKE) > 0;
|
||||||
const likeIcon = SIMPLE_SITE
|
const likeIcon = SIMPLE_SITE
|
||||||
? myReacts.includes(REACTION_TYPES.LIKE)
|
? myReacts && myReacts.includes(REACTION_TYPES.LIKE)
|
||||||
? ICONS.FIRE_ACTIVE
|
? ICONS.FIRE_ACTIVE
|
||||||
: ICONS.FIRE
|
: ICONS.FIRE
|
||||||
: ICONS.UPVOTE;
|
: ICONS.UPVOTE;
|
||||||
const dislikeIcon = SIMPLE_SITE
|
const dislikeIcon = SIMPLE_SITE
|
||||||
? myReacts.includes(REACTION_TYPES.DISLIKE)
|
? myReacts && myReacts.includes(REACTION_TYPES.DISLIKE)
|
||||||
? ICONS.SLIME_ACTIVE
|
? ICONS.SLIME_ACTIVE
|
||||||
: ICONS.SLIME
|
: ICONS.SLIME
|
||||||
: ICONS.DOWNVOTE;
|
: ICONS.DOWNVOTE;
|
||||||
|
|
|
@ -7,17 +7,17 @@ import {
|
||||||
selectMyChannelClaims,
|
selectMyChannelClaims,
|
||||||
} from 'redux/selectors/claims';
|
} from 'redux/selectors/claims';
|
||||||
import {
|
import {
|
||||||
makeSelectTopLevelCommentsForUri,
|
selectTopLevelCommentsForUri,
|
||||||
makeSelectTopLevelTotalPagesForUri,
|
makeSelectTopLevelTotalPagesForUri,
|
||||||
selectIsFetchingComments,
|
selectIsFetchingComments,
|
||||||
selectIsFetchingCommentsById,
|
selectIsFetchingCommentsById,
|
||||||
selectIsFetchingReacts,
|
selectIsFetchingReacts,
|
||||||
makeSelectTotalCommentsCountForUri,
|
makeSelectTotalCommentsCountForUri,
|
||||||
selectOthersReactsById,
|
selectOthersReacts,
|
||||||
selectMyReactionsByCommentId,
|
selectMyReacts,
|
||||||
makeSelectCommentIdsForUri,
|
makeSelectCommentIdsForUri,
|
||||||
selectSettingsByChannelId,
|
selectSettingsByChannelId,
|
||||||
makeSelectPinnedCommentsForUri,
|
selectPinnedCommentsForUri,
|
||||||
} from 'redux/selectors/comments';
|
} from 'redux/selectors/comments';
|
||||||
import { doCommentReset, doCommentList, doCommentById, doCommentReactList } from 'redux/actions/comments';
|
import { doCommentReset, doCommentList, doCommentById, doCommentReactList } from 'redux/actions/comments';
|
||||||
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
||||||
|
@ -25,7 +25,7 @@ import CommentsList from './view';
|
||||||
|
|
||||||
const select = (state, props) => {
|
const select = (state, props) => {
|
||||||
const activeChannelClaim = selectActiveChannelClaim(state);
|
const activeChannelClaim = selectActiveChannelClaim(state);
|
||||||
const topLevelComments = makeSelectTopLevelCommentsForUri(props.uri)(state);
|
const topLevelComments = selectTopLevelCommentsForUri(state, props.uri);
|
||||||
|
|
||||||
const resolvedComments =
|
const resolvedComments =
|
||||||
topLevelComments && topLevelComments.length > 0
|
topLevelComments && topLevelComments.length > 0
|
||||||
|
@ -37,7 +37,7 @@ const select = (state, props) => {
|
||||||
resolvedComments,
|
resolvedComments,
|
||||||
myChannels: selectMyChannelClaims(state),
|
myChannels: selectMyChannelClaims(state),
|
||||||
allCommentIds: makeSelectCommentIdsForUri(props.uri)(state),
|
allCommentIds: makeSelectCommentIdsForUri(props.uri)(state),
|
||||||
pinnedComments: makeSelectPinnedCommentsForUri(props.uri)(state),
|
pinnedComments: selectPinnedCommentsForUri(state, props.uri),
|
||||||
topLevelTotalPages: makeSelectTopLevelTotalPagesForUri(props.uri)(state),
|
topLevelTotalPages: makeSelectTopLevelTotalPagesForUri(props.uri)(state),
|
||||||
totalComments: makeSelectTotalCommentsCountForUri(props.uri)(state),
|
totalComments: makeSelectTotalCommentsCountForUri(props.uri)(state),
|
||||||
claim: makeSelectClaimForUri(props.uri)(state),
|
claim: makeSelectClaimForUri(props.uri)(state),
|
||||||
|
@ -47,8 +47,8 @@ const select = (state, props) => {
|
||||||
isFetchingReacts: selectIsFetchingReacts(state),
|
isFetchingReacts: selectIsFetchingReacts(state),
|
||||||
fetchingChannels: selectFetchingMyChannels(state),
|
fetchingChannels: selectFetchingMyChannels(state),
|
||||||
settingsByChannelId: selectSettingsByChannelId(state),
|
settingsByChannelId: selectSettingsByChannelId(state),
|
||||||
myReactsByCommentId: selectMyReactionsByCommentId(state),
|
myReactsByCommentId: selectMyReacts(state),
|
||||||
othersReactsById: selectOthersReactsById(state),
|
othersReactsById: selectOthersReacts(state),
|
||||||
activeChannelId: activeChannelClaim && activeChannelClaim.claim_id,
|
activeChannelId: activeChannelClaim && activeChannelClaim.claim_id,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { doResolveUris } from 'redux/actions/claims';
|
import { doResolveUris } from 'redux/actions/claims';
|
||||||
import { makeSelectClaimIsMine, selectMyChannelClaims, makeSelectClaimForUri } from 'redux/selectors/claims';
|
import { makeSelectClaimIsMine, selectMyChannelClaims, makeSelectClaimForUri } from 'redux/selectors/claims';
|
||||||
import { selectIsFetchingCommentsByParentId, makeSelectRepliesForParentId } from 'redux/selectors/comments';
|
import { selectIsFetchingCommentsByParentId, selectRepliesForParentId } from 'redux/selectors/comments';
|
||||||
import { selectUserVerifiedEmail } from 'redux/selectors/user';
|
import { selectUserVerifiedEmail } from 'redux/selectors/user';
|
||||||
import CommentsReplies from './view';
|
import CommentsReplies from './view';
|
||||||
|
|
||||||
const select = (state, props) => {
|
const select = (state, props) => {
|
||||||
const fetchedReplies = makeSelectRepliesForParentId(props.parentId)(state);
|
const fetchedReplies = selectRepliesForParentId(state, props.parentId);
|
||||||
const resolvedReplies =
|
const resolvedReplies =
|
||||||
fetchedReplies && fetchedReplies.length > 0
|
fetchedReplies && fetchedReplies.length > 0
|
||||||
? fetchedReplies.filter(({ channel_url }) => makeSelectClaimForUri(channel_url)(state) !== undefined)
|
? fetchedReplies.filter(({ channel_url }) => makeSelectClaimForUri(channel_url)(state) !== undefined)
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { makeSelectDateForUri } from 'redux/selectors/claims';
|
import { selectDateForUri } from 'redux/selectors/claims';
|
||||||
import * as SETTINGS from 'constants/settings';
|
import * as SETTINGS from 'constants/settings';
|
||||||
import { makeSelectClientSetting } from 'redux/selectors/settings';
|
import { makeSelectClientSetting } from 'redux/selectors/settings';
|
||||||
import DateTime from './view';
|
import DateTime from './view';
|
||||||
|
|
||||||
const select = (state, props) => ({
|
const select = (state, props) => ({
|
||||||
date: props.date || makeSelectDateForUri(props.uri)(state),
|
date: props.date || selectDateForUri(state, props.uri),
|
||||||
clock24h: makeSelectClientSetting(SETTINGS.CLOCK_24H)(state),
|
clock24h: makeSelectClientSetting(SETTINGS.CLOCK_24H)(state),
|
||||||
});
|
});
|
||||||
export default connect(select)(DateTime);
|
export default connect(select)(DateTime);
|
||||||
|
|
|
@ -2,7 +2,6 @@ import { connect } from 'react-redux';
|
||||||
import {
|
import {
|
||||||
makeSelectClaimForUri,
|
makeSelectClaimForUri,
|
||||||
makeSelectMetadataForUri,
|
makeSelectMetadataForUri,
|
||||||
makeSelectTagsForUri,
|
|
||||||
makeSelectClaimIsMine,
|
makeSelectClaimIsMine,
|
||||||
} from 'redux/selectors/claims';
|
} from 'redux/selectors/claims';
|
||||||
import { makeSelectPendingAmountByUri } from 'redux/selectors/wallet';
|
import { makeSelectPendingAmountByUri } from 'redux/selectors/wallet';
|
||||||
|
@ -15,7 +14,6 @@ const select = (state, props) => ({
|
||||||
claimIsMine: makeSelectClaimIsMine(props.uri)(state),
|
claimIsMine: makeSelectClaimIsMine(props.uri)(state),
|
||||||
metadata: makeSelectMetadataForUri(props.uri)(state),
|
metadata: makeSelectMetadataForUri(props.uri)(state),
|
||||||
user: selectUser(state),
|
user: selectUser(state),
|
||||||
tags: makeSelectTagsForUri(props.uri)(state),
|
|
||||||
pendingAmount: makeSelectPendingAmountByUri(props.uri)(state),
|
pendingAmount: makeSelectPendingAmountByUri(props.uri)(state),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -16,7 +16,6 @@ type Props = {
|
||||||
claim: StreamClaim,
|
claim: StreamClaim,
|
||||||
metadata: StreamMetadata,
|
metadata: StreamMetadata,
|
||||||
user: ?any,
|
user: ?any,
|
||||||
tags: any,
|
|
||||||
pendingAmount: number,
|
pendingAmount: number,
|
||||||
doOpenModal: (id: string, {}) => void,
|
doOpenModal: (id: string, {}) => void,
|
||||||
claimIsMine: boolean,
|
claimIsMine: boolean,
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { makeSelectStakedLevelForChannelUri, makeSelectClaimForUri } from 'redux/selectors/claims';
|
import { makeSelectStakedLevelForChannelUri, selectClaimForUri } from 'redux/selectors/claims';
|
||||||
import LivestreamComment from './view';
|
import LivestreamComment from './view';
|
||||||
|
|
||||||
const select = (state, props) => ({
|
const select = (state, props) => ({
|
||||||
claim: makeSelectClaimForUri(props.uri)(state),
|
claim: selectClaimForUri(state, props.uri),
|
||||||
stakedLevel: makeSelectStakedLevelForChannelUri(props.authorUri)(state),
|
stakedLevel: makeSelectStakedLevelForChannelUri(props.authorUri)(state),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,20 +1,22 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { makeSelectClaimForUri, selectMyChannelClaims } from 'redux/selectors/claims';
|
import { selectClaimForUri, selectMyChannelClaims } from 'redux/selectors/claims';
|
||||||
import { doCommentSocketConnect, doCommentSocketDisconnect } from 'redux/actions/websocket';
|
import { doCommentSocketConnect, doCommentSocketDisconnect } from 'redux/actions/websocket';
|
||||||
import { doCommentList, doSuperChatList } from 'redux/actions/comments';
|
import { doCommentList, doSuperChatList } from 'redux/actions/comments';
|
||||||
import {
|
import {
|
||||||
makeSelectTopLevelCommentsForUri,
|
selectTopLevelCommentsForUri,
|
||||||
selectIsFetchingComments,
|
selectIsFetchingComments,
|
||||||
makeSelectSuperChatsForUri,
|
makeSelectSuperChatsForUri,
|
||||||
makeSelectSuperChatTotalAmountForUri,
|
makeSelectSuperChatTotalAmountForUri,
|
||||||
makeSelectPinnedCommentsForUri,
|
selectPinnedCommentsForUri,
|
||||||
} from 'redux/selectors/comments';
|
} from 'redux/selectors/comments';
|
||||||
import LivestreamComments from './view';
|
import LivestreamComments from './view';
|
||||||
|
|
||||||
|
const MAX_LIVESTREAM_COMMENTS = 75;
|
||||||
|
|
||||||
const select = (state, props) => ({
|
const select = (state, props) => ({
|
||||||
claim: makeSelectClaimForUri(props.uri)(state),
|
claim: selectClaimForUri(state, props.uri),
|
||||||
comments: makeSelectTopLevelCommentsForUri(props.uri)(state).slice(0, 75),
|
comments: selectTopLevelCommentsForUri(state, props.uri, MAX_LIVESTREAM_COMMENTS),
|
||||||
pinnedComments: makeSelectPinnedCommentsForUri(props.uri)(state),
|
pinnedComments: selectPinnedCommentsForUri(state, props.uri),
|
||||||
fetchingComments: selectIsFetchingComments(state),
|
fetchingComments: selectIsFetchingComments(state),
|
||||||
superChats: makeSelectSuperChatsForUri(props.uri)(state),
|
superChats: makeSelectSuperChatsForUri(props.uri)(state),
|
||||||
superChatsTotalAmount: makeSelectSuperChatTotalAmountForUri(props.uri)(state),
|
superChatsTotalAmount: makeSelectSuperChatTotalAmountForUri(props.uri)(state),
|
||||||
|
|
|
@ -63,11 +63,11 @@ export default function LivestreamComments(props: Props) {
|
||||||
|
|
||||||
const pinnedComment = pinnedComments.length > 0 ? pinnedComments[0] : null;
|
const pinnedComment = pinnedComments.length > 0 ? pinnedComments[0] : null;
|
||||||
|
|
||||||
function restoreScrollPos() {
|
const restoreScrollPos = React.useCallback(() => {
|
||||||
if (discussionElement) {
|
if (discussionElement) {
|
||||||
discussionElement.scrollTop = 0;
|
discussionElement.scrollTop = 0;
|
||||||
}
|
}
|
||||||
}
|
}, [discussionElement]);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
if (claimId) {
|
if (claimId) {
|
||||||
|
@ -139,10 +139,10 @@ export default function LivestreamComments(props: Props) {
|
||||||
const clonedSuperchats = JSON.parse(JSON.stringify(superChatsByTipAmount));
|
const clonedSuperchats = JSON.parse(JSON.stringify(superChatsByTipAmount));
|
||||||
|
|
||||||
// for top to bottom display, oldest superchat on top most recent on bottom
|
// for top to bottom display, oldest superchat on top most recent on bottom
|
||||||
superChatsReversed = clonedSuperchats
|
superChatsReversed = clonedSuperchats.sort((a, b) => {
|
||||||
.sort((a, b) => {
|
return b.timestamp - a.timestamp;
|
||||||
return b.timestamp - a.timestamp;
|
});
|
||||||
}); }
|
}
|
||||||
|
|
||||||
// todo: implement comment_list --mine in SDK so redux can grab with selectCommentIsMine
|
// todo: implement comment_list --mine in SDK so redux can grab with selectCommentIsMine
|
||||||
function isMyComment(channelId: string) {
|
function isMyComment(channelId: string) {
|
||||||
|
|
|
@ -8,7 +8,7 @@ import {
|
||||||
selectAbandonClaimSupportError,
|
selectAbandonClaimSupportError,
|
||||||
} from 'redux/selectors/wallet';
|
} from 'redux/selectors/wallet';
|
||||||
|
|
||||||
import { makeSelectMetadataForUri, makeSelectClaimForUri } from 'redux/selectors/claims';
|
import { makeSelectClaimForUri } from 'redux/selectors/claims';
|
||||||
import { doSupportAbandonForClaim } from 'redux/actions/wallet';
|
import { doSupportAbandonForClaim } from 'redux/actions/wallet';
|
||||||
import SupportsLiquidate from './view';
|
import SupportsLiquidate from './view';
|
||||||
|
|
||||||
|
@ -19,7 +19,6 @@ const select = (state, props) => ({
|
||||||
supportsBalance: selectSupportsBalance(state) || undefined,
|
supportsBalance: selectSupportsBalance(state) || undefined,
|
||||||
tipsBalance: selectTipsBalance(state) || undefined,
|
tipsBalance: selectTipsBalance(state) || undefined,
|
||||||
claim: makeSelectClaimForUri(props.uri)(state),
|
claim: makeSelectClaimForUri(props.uri)(state),
|
||||||
metadata: makeSelectMetadataForUri(props.uri)(state),
|
|
||||||
abandonClaimError: selectAbandonClaimSupportError(state),
|
abandonClaimError: selectAbandonClaimSupportError(state),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -15,7 +15,6 @@ type Props = {
|
||||||
supportsBalance: number,
|
supportsBalance: number,
|
||||||
tipsBalance: number,
|
tipsBalance: number,
|
||||||
claim: any,
|
claim: any,
|
||||||
metaData: any,
|
|
||||||
handleClose: () => void,
|
handleClose: () => void,
|
||||||
abandonSupportForClaim: (string, string, boolean | string, boolean) => any,
|
abandonSupportForClaim: (string, string, boolean | string, boolean) => any,
|
||||||
abandonClaimError: ?string,
|
abandonClaimError: ?string,
|
||||||
|
@ -37,14 +36,14 @@ const SupportsLiquidate = (props: Props) => {
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (claimId && abandonSupportForClaim) {
|
if (claimId && abandonSupportForClaim) {
|
||||||
abandonSupportForClaim(claimId, type, false, true).then(r => {
|
abandonSupportForClaim(claimId, type, false, true).then((r) => {
|
||||||
setPreviewBalance(r.total_input);
|
setPreviewBalance(r.total_input);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [abandonSupportForClaim, claimId, type, setPreviewBalance]);
|
}, [abandonSupportForClaim, claimId, type, setPreviewBalance]);
|
||||||
|
|
||||||
function handleSubmit() {
|
function handleSubmit() {
|
||||||
abandonSupportForClaim(claimId, type, keep, false).then(r => {
|
abandonSupportForClaim(claimId, type, keep, false).then((r) => {
|
||||||
if (r) {
|
if (r) {
|
||||||
handleClose();
|
handleClose();
|
||||||
}
|
}
|
||||||
|
@ -141,7 +140,7 @@ const SupportsLiquidate = (props: Props) => {
|
||||||
step={0.01}
|
step={0.01}
|
||||||
max={previewBalance}
|
max={previewBalance}
|
||||||
value={Number(amount) >= 0 ? amount : previewBalance / 4} // by default, set it to 25% of available
|
value={Number(amount) >= 0 ? amount : previewBalance / 4} // by default, set it to 25% of available
|
||||||
onChange={e => handleChange(e.target.value)}
|
onChange={(e) => handleChange(e.target.value)}
|
||||||
/>
|
/>
|
||||||
<label className="range__label">
|
<label className="range__label">
|
||||||
<span>0</span>
|
<span>0</span>
|
||||||
|
@ -152,7 +151,7 @@ const SupportsLiquidate = (props: Props) => {
|
||||||
type="text"
|
type="text"
|
||||||
value={amount >= 0 ? amount || '' : previewBalance && previewBalance / 4}
|
value={amount >= 0 ? amount || '' : previewBalance && previewBalance / 4}
|
||||||
helper={message}
|
helper={message}
|
||||||
onChange={e => handleChange(e.target.value)}
|
onChange={(e) => handleChange(e.target.value)}
|
||||||
/>
|
/>
|
||||||
</Form>
|
</Form>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -2,7 +2,6 @@ import { connect } from 'react-redux';
|
||||||
import { doSetContentHistoryItem, doSetPrimaryUri, clearPosition } from 'redux/actions/content';
|
import { doSetContentHistoryItem, doSetPrimaryUri, clearPosition } from 'redux/actions/content';
|
||||||
import { withRouter } from 'react-router-dom';
|
import { withRouter } from 'react-router-dom';
|
||||||
import {
|
import {
|
||||||
makeSelectMetadataForUri,
|
|
||||||
makeSelectClaimIsNsfw,
|
makeSelectClaimIsNsfw,
|
||||||
makeSelectTagInClaimOrChannelForUri,
|
makeSelectTagInClaimOrChannelForUri,
|
||||||
makeSelectClaimIsStreamPlaceholder,
|
makeSelectClaimIsStreamPlaceholder,
|
||||||
|
@ -27,7 +26,6 @@ const select = (state, props) => {
|
||||||
return {
|
return {
|
||||||
linkedCommentId: urlParams.get('lc'),
|
linkedCommentId: urlParams.get('lc'),
|
||||||
costInfo: makeSelectCostInfoForUri(props.uri)(state),
|
costInfo: makeSelectCostInfoForUri(props.uri)(state),
|
||||||
metadata: makeSelectMetadataForUri(props.uri)(state),
|
|
||||||
obscureNsfw: !selectShowMatureContent(state),
|
obscureNsfw: !selectShowMatureContent(state),
|
||||||
isMature: makeSelectClaimIsNsfw(props.uri)(state),
|
isMature: makeSelectClaimIsNsfw(props.uri)(state),
|
||||||
fileInfo: makeSelectFileInfoForUri(props.uri)(state),
|
fileInfo: makeSelectFileInfoForUri(props.uri)(state),
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { doCommentListOwn, doCommentReset } from 'redux/actions/comments';
|
||||||
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
import { selectActiveChannelClaim } from 'redux/selectors/app';
|
||||||
import {
|
import {
|
||||||
selectIsFetchingComments,
|
selectIsFetchingComments,
|
||||||
makeSelectCommentsForUri,
|
selectCommentsForUri,
|
||||||
makeSelectTotalCommentsCountForUri,
|
makeSelectTotalCommentsCountForUri,
|
||||||
makeSelectTopLevelTotalPagesForUri,
|
makeSelectTopLevelTotalPagesForUri,
|
||||||
} from 'redux/selectors/comments';
|
} from 'redux/selectors/comments';
|
||||||
|
@ -17,7 +17,7 @@ const select = (state) => {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
activeChannelClaim,
|
activeChannelClaim,
|
||||||
allComments: makeSelectCommentsForUri(uri)(state),
|
allComments: selectCommentsForUri(state, uri),
|
||||||
totalComments: makeSelectTotalCommentsCountForUri(uri)(state),
|
totalComments: makeSelectTotalCommentsCountForUri(uri)(state),
|
||||||
topLevelTotalPages: makeSelectTopLevelTotalPagesForUri(uri)(state),
|
topLevelTotalPages: makeSelectTopLevelTotalPagesForUri(uri)(state),
|
||||||
isFetchingComments: selectIsFetchingComments(state),
|
isFetchingComments: selectIsFetchingComments(state),
|
||||||
|
|
|
@ -9,8 +9,8 @@ import { selectClaimsByUri, selectMyChannelClaims } from 'redux/selectors/claims
|
||||||
import { doClaimSearch } from 'redux/actions/claims';
|
import { doClaimSearch } from 'redux/actions/claims';
|
||||||
import { doToast, doSeeNotifications } from 'redux/actions/notifications';
|
import { doToast, doSeeNotifications } from 'redux/actions/notifications';
|
||||||
import {
|
import {
|
||||||
makeSelectMyReactionsForComment,
|
selectMyReactsForComment,
|
||||||
makeSelectOthersReactionsForComment,
|
selectOthersReactsForComment,
|
||||||
selectPendingCommentReacts,
|
selectPendingCommentReacts,
|
||||||
selectModerationBlockList,
|
selectModerationBlockList,
|
||||||
selectModerationDelegatorsById,
|
selectModerationDelegatorsById,
|
||||||
|
@ -466,8 +466,8 @@ export function doCommentReact(commentId: string, type: string) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const reactKey = `${commentId}:${activeChannelClaim.claim_id}`;
|
const reactKey = `${commentId}:${activeChannelClaim.claim_id}`;
|
||||||
const myReacts = makeSelectMyReactionsForComment(reactKey)(state);
|
const myReacts = selectMyReactsForComment(state, reactKey) || [];
|
||||||
const othersReacts = makeSelectOthersReactionsForComment(reactKey)(state);
|
const othersReacts = selectOthersReactsForComment(state, reactKey) || {};
|
||||||
|
|
||||||
const signatureData = await channelSignName(activeChannelClaim.claim_id, activeChannelClaim.name);
|
const signatureData = await channelSignName(activeChannelClaim.claim_id, activeChannelClaim.name);
|
||||||
if (!signatureData) {
|
if (!signatureData) {
|
||||||
|
|
|
@ -26,6 +26,14 @@ export default handleActions(
|
||||||
[ACTIONS.USER_STATE_POPULATE]: (state: BlocklistState, action: { data: { blocked: ?Array<string> } }) => {
|
[ACTIONS.USER_STATE_POPULATE]: (state: BlocklistState, action: { data: { blocked: ?Array<string> } }) => {
|
||||||
const { blocked } = action.data;
|
const { blocked } = action.data;
|
||||||
const sanitizedBlocked = blocked && blocked.filter((e) => typeof e === 'string');
|
const sanitizedBlocked = blocked && blocked.filter((e) => typeof e === 'string');
|
||||||
|
|
||||||
|
const next = sanitizedBlocked;
|
||||||
|
const prev = state.blockedChannels;
|
||||||
|
|
||||||
|
if (next && prev && prev.length === next.length && prev.every((value, index) => value === next[index])) {
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...state,
|
...state,
|
||||||
blockedChannels: sanitizedBlocked && sanitizedBlocked.length ? sanitizedBlocked : state.blockedChannels,
|
blockedChannels: sanitizedBlocked && sanitizedBlocked.length ? sanitizedBlocked : state.blockedChannels,
|
||||||
|
|
|
@ -65,11 +65,22 @@ export default handleActions(
|
||||||
[ACTIONS.USER_STATE_POPULATE]: (state: TagState, action: { data: { tags: ?Array<string> } }) => {
|
[ACTIONS.USER_STATE_POPULATE]: (state: TagState, action: { data: { tags: ?Array<string> } }) => {
|
||||||
const { tags } = action.data;
|
const { tags } = action.data;
|
||||||
if (Array.isArray(tags)) {
|
if (Array.isArray(tags)) {
|
||||||
|
const next = tags && tags.filter((tag) => typeof tag === 'string');
|
||||||
|
const prev = state.followedTags;
|
||||||
|
|
||||||
|
if (next && prev && prev.length === next.length && prev.every((value, index) => value === next[index])) {
|
||||||
|
// No changes
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
// New state
|
||||||
return {
|
return {
|
||||||
...state,
|
...state,
|
||||||
followedTags: tags,
|
followedTags: next || [],
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Purge 'followedTags'
|
||||||
return {
|
return {
|
||||||
...state,
|
...state,
|
||||||
};
|
};
|
||||||
|
|
|
@ -2,11 +2,11 @@
|
||||||
import { createSelector } from 'reselect';
|
import { createSelector } from 'reselect';
|
||||||
import { splitBySeparator } from 'util/lbryURI';
|
import { splitBySeparator } from 'util/lbryURI';
|
||||||
|
|
||||||
const selectState = (state: { blocked: BlocklistState }) => state.blocked || {};
|
type State = { blocked: BlocklistState };
|
||||||
|
|
||||||
export const selectMutedChannels = createSelector(selectState, (state: BlocklistState) => {
|
const selectState = (state: State) => state.blocked || {};
|
||||||
return state.blockedChannels.filter((e) => typeof e === 'string');
|
|
||||||
});
|
export const selectMutedChannels = (state: State) => selectState(state).blockedChannels;
|
||||||
|
|
||||||
export const makeSelectChannelIsMuted = (uri: string) =>
|
export const makeSelectChannelIsMuted = (uri: string) =>
|
||||||
createSelector(selectMutedChannels, (state: Array<string>) => {
|
createSelector(selectMutedChannels, (state: Array<string>) => {
|
||||||
|
|
|
@ -2,20 +2,22 @@
|
||||||
import { normalizeURI, parseURI, isURIValid } from 'util/lbryURI';
|
import { normalizeURI, parseURI, isURIValid } from 'util/lbryURI';
|
||||||
import { selectSupportsByOutpoint } from 'redux/selectors/wallet';
|
import { selectSupportsByOutpoint } from 'redux/selectors/wallet';
|
||||||
import { createSelector } from 'reselect';
|
import { createSelector } from 'reselect';
|
||||||
|
import { createCachedSelector } from 're-reselect';
|
||||||
import { isClaimNsfw, filterClaims } from 'util/claim';
|
import { isClaimNsfw, filterClaims } from 'util/claim';
|
||||||
import * as CLAIM from 'constants/claim';
|
import * as CLAIM from 'constants/claim';
|
||||||
|
|
||||||
|
type State = { claims: any };
|
||||||
|
|
||||||
const selectState = (state) => state.claims || {};
|
const selectState = (state) => state.claims || {};
|
||||||
|
|
||||||
export const selectById = createSelector(selectState, (state) => state.byId || {});
|
export const selectById = (state: State) => selectState(state).byId || {};
|
||||||
|
export const selectPendingClaimsById = (state: State) => selectState(state).pendingById || {};
|
||||||
export const selectPendingClaimsById = createSelector(selectState, (state) => state.pendingById || {});
|
|
||||||
|
|
||||||
export const selectClaimsById = createSelector(selectById, selectPendingClaimsById, (byId, pendingById) => {
|
export const selectClaimsById = createSelector(selectById, selectPendingClaimsById, (byId, pendingById) => {
|
||||||
return Object.assign(byId, pendingById); // do I need merged to keep metadata?
|
return Object.assign(byId, pendingById); // do I need merged to keep metadata?
|
||||||
});
|
});
|
||||||
|
|
||||||
export const selectClaimIdsByUri = createSelector(selectState, (state) => state.claimsByUri || {});
|
export const selectClaimIdsByUri = (state: State) => selectState(state).claimsByUri || {};
|
||||||
|
|
||||||
export const selectCurrentChannelPage = createSelector(selectState, (state) => state.currentChannelPage || 1);
|
export const selectCurrentChannelPage = createSelector(selectState, (state) => state.currentChannelPage || 1);
|
||||||
|
|
||||||
|
@ -74,6 +76,43 @@ export const selectReflectingById = createSelector(selectState, (state) => state
|
||||||
|
|
||||||
export const makeSelectClaimForClaimId = (claimId: string) => createSelector(selectClaimsById, (byId) => byId[claimId]);
|
export const makeSelectClaimForClaimId = (claimId: string) => createSelector(selectClaimsById, (byId) => byId[claimId]);
|
||||||
|
|
||||||
|
export const selectClaimForUri = createCachedSelector(
|
||||||
|
selectClaimIdsByUri,
|
||||||
|
selectClaimsById,
|
||||||
|
(state, uri) => uri,
|
||||||
|
(state, uri, returnRepost = true) => returnRepost,
|
||||||
|
(byUri, byId, uri, returnRepost) => {
|
||||||
|
const validUri = isURIValid(uri);
|
||||||
|
|
||||||
|
if (validUri && byUri) {
|
||||||
|
const claimId = uri && byUri[normalizeURI(uri)];
|
||||||
|
const claim = byId[claimId];
|
||||||
|
|
||||||
|
// Make sure to return the claim as is so apps can check if it's been resolved before (null) or still needs to be resolved (undefined)
|
||||||
|
if (claimId === null) {
|
||||||
|
return null;
|
||||||
|
} else if (claimId === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const repostedClaim = claim && claim.reposted_claim;
|
||||||
|
if (repostedClaim && returnRepost) {
|
||||||
|
const channelUrl =
|
||||||
|
claim.signing_channel && (claim.signing_channel.canonical_url || claim.signing_channel.permanent_url);
|
||||||
|
|
||||||
|
return {
|
||||||
|
...repostedClaim,
|
||||||
|
repost_url: normalizeURI(uri),
|
||||||
|
repost_channel_url: channelUrl,
|
||||||
|
repost_bid_amount: claim && claim.meta && claim.meta.effective_amount,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return claim;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)((state, uri, returnRepost = true) => `${uri}:${returnRepost ? '1' : '0'}`);
|
||||||
|
|
||||||
export const makeSelectClaimForUri = (uri: string, returnRepost: boolean = true) =>
|
export const makeSelectClaimForUri = (uri: string, returnRepost: boolean = true) =>
|
||||||
createSelector(selectClaimIdsByUri, selectClaimsById, (byUri, byId) => {
|
createSelector(selectClaimIdsByUri, selectClaimsById, (byUri, byId) => {
|
||||||
const validUri = isURIValid(uri);
|
const validUri = isURIValid(uri);
|
||||||
|
@ -231,6 +270,11 @@ export const makeSelectTotalPagesInChannelSearch = (uri: string) =>
|
||||||
return byChannel['pageCount'];
|
return byChannel['pageCount'];
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const selectMetadataForUri = createCachedSelector(selectClaimForUri, (claim, uri) => {
|
||||||
|
const metadata = claim && claim.value;
|
||||||
|
return metadata || (claim === undefined ? undefined : null);
|
||||||
|
})((state, uri) => uri);
|
||||||
|
|
||||||
export const makeSelectMetadataForUri = (uri: string) =>
|
export const makeSelectMetadataForUri = (uri: string) =>
|
||||||
createSelector(makeSelectClaimForUri(uri), (claim) => {
|
createSelector(makeSelectClaimForUri(uri), (claim) => {
|
||||||
const metadata = claim && claim.value;
|
const metadata = claim && claim.value;
|
||||||
|
@ -245,8 +289,9 @@ export const makeSelectMetadataItemForUri = (uri: string, key: string) =>
|
||||||
export const makeSelectTitleForUri = (uri: string) =>
|
export const makeSelectTitleForUri = (uri: string) =>
|
||||||
createSelector(makeSelectMetadataForUri(uri), (metadata) => metadata && metadata.title);
|
createSelector(makeSelectMetadataForUri(uri), (metadata) => metadata && metadata.title);
|
||||||
|
|
||||||
export const makeSelectDateForUri = (uri: string) =>
|
export const selectDateForUri = createCachedSelector(
|
||||||
createSelector(makeSelectClaimForUri(uri), (claim) => {
|
selectClaimForUri, // input: (state, uri, ?returnRepost)
|
||||||
|
(claim) => {
|
||||||
const timestamp =
|
const timestamp =
|
||||||
claim &&
|
claim &&
|
||||||
claim.value &&
|
claim.value &&
|
||||||
|
@ -260,7 +305,8 @@ export const makeSelectDateForUri = (uri: string) =>
|
||||||
}
|
}
|
||||||
const dateObj = new Date(timestamp);
|
const dateObj = new Date(timestamp);
|
||||||
return dateObj;
|
return dateObj;
|
||||||
});
|
}
|
||||||
|
)((state, uri) => uri);
|
||||||
|
|
||||||
export const makeSelectAmountForUri = (uri: string) =>
|
export const makeSelectAmountForUri = (uri: string) =>
|
||||||
createSelector(makeSelectClaimForUri(uri), (claim) => {
|
createSelector(makeSelectClaimForUri(uri), (claim) => {
|
||||||
|
@ -496,6 +542,10 @@ export const makeSelectMyChannelPermUrlForName = (name: string) =>
|
||||||
return matchingClaim ? matchingClaim.permanent_url : null;
|
return matchingClaim ? matchingClaim.permanent_url : null;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const selectTagsForUri = createCachedSelector(selectMetadataForUri, (metadata: ?GenericMetadata) => {
|
||||||
|
return (metadata && metadata.tags) || [];
|
||||||
|
})((state, uri) => uri);
|
||||||
|
|
||||||
export const makeSelectTagsForUri = (uri: string) =>
|
export const makeSelectTagsForUri = (uri: string) =>
|
||||||
createSelector(makeSelectMetadataForUri(uri), (metadata: ?GenericMetadata) => {
|
createSelector(makeSelectMetadataForUri(uri), (metadata: ?GenericMetadata) => {
|
||||||
return (metadata && metadata.tags) || [];
|
return (metadata && metadata.tags) || [];
|
||||||
|
|
|
@ -1,43 +1,79 @@
|
||||||
// @flow
|
// @flow
|
||||||
import { createSelector } from 'reselect';
|
import { createSelector } from 'reselect';
|
||||||
|
import { createCachedSelector } from 're-reselect';
|
||||||
import { selectMutedChannels } from 'redux/selectors/blocked';
|
import { selectMutedChannels } from 'redux/selectors/blocked';
|
||||||
import { selectShowMatureContent } from 'redux/selectors/settings';
|
import { selectShowMatureContent } from 'redux/selectors/settings';
|
||||||
import { selectBlacklistedOutpointMap, selectFilteredOutpointMap } from 'lbryinc';
|
import { selectBlacklistedOutpointMap, selectFilteredOutpointMap } from 'lbryinc';
|
||||||
import { selectClaimsById, selectMyActiveClaims } from 'redux/selectors/claims';
|
import { selectClaimsById, selectMyActiveClaims } from 'redux/selectors/claims';
|
||||||
import { isClaimNsfw } from 'util/claim';
|
import { isClaimNsfw } from 'util/claim';
|
||||||
|
|
||||||
|
type State = { comments: CommentsState };
|
||||||
|
|
||||||
const selectState = (state) => state.comments || {};
|
const selectState = (state) => state.comments || {};
|
||||||
|
|
||||||
export const selectCommentsById = createSelector(selectState, (state) => state.commentById || {});
|
export const selectCommentsById = (state: State) => selectState(state).commentById || {};
|
||||||
export const selectIsFetchingComments = createSelector(selectState, (state) => state.isLoading);
|
export const selectIsFetchingComments = (state: State) => selectState(state).isLoading;
|
||||||
export const selectIsFetchingCommentsById = createSelector(selectState, (state) => state.isLoadingById);
|
export const selectIsFetchingCommentsById = (state: State) => selectState(state).isLoadingById;
|
||||||
export const selectIsFetchingCommentsByParentId = createSelector(selectState, (state) => state.isLoadingByParentId);
|
export const selectIsFetchingCommentsByParentId = (state: State) => selectState(state).isLoadingByParentId;
|
||||||
export const selectIsFetchingReacts = createSelector(selectState, (state) => state.isFetchingReacts);
|
export const selectIsFetchingReacts = (state: State) => selectState(state).isFetchingReacts;
|
||||||
export const selectOthersReactsById = createSelector(selectState, (state) => state.othersReactsByCommentId);
|
|
||||||
|
|
||||||
export const selectPinnedCommentsById = createSelector(selectState, (state) => state.pinnedCommentsById);
|
export const selectMyReacts = (state: State) => state.comments.myReactsByCommentId;
|
||||||
export const makeSelectPinnedCommentsForUri = (uri: string) =>
|
export const selectMyReactsForComment = (state: State, commentIdChannelId: string) => {
|
||||||
createSelector(
|
// @commentIdChannelId: Format = 'commentId:MyChannelId'
|
||||||
selectCommentsByUri,
|
return state.comments.myReactsByCommentId && state.comments.myReactsByCommentId[commentIdChannelId];
|
||||||
selectCommentsById,
|
};
|
||||||
selectPinnedCommentsById,
|
|
||||||
(byUri, byId, pinnedCommentsById) => {
|
|
||||||
const claimId = byUri[uri];
|
|
||||||
const pinnedCommentIds = pinnedCommentsById && pinnedCommentsById[claimId];
|
|
||||||
const pinnedComments = [];
|
|
||||||
|
|
||||||
if (pinnedCommentIds) {
|
export const selectOthersReacts = (state: State) => state.comments.othersReactsByCommentId;
|
||||||
pinnedCommentIds.forEach((commentId) => {
|
export const selectOthersReactsForComment = (state: State, id: string) => {
|
||||||
pinnedComments.push(byId[commentId]);
|
return state.comments.othersReactsByCommentId && state.comments.othersReactsByCommentId[id];
|
||||||
});
|
};
|
||||||
}
|
|
||||||
|
|
||||||
return pinnedComments;
|
// previously this used a mapping from claimId -> Array<Comments>
|
||||||
|
/* export const selectCommentsById = createSelector(
|
||||||
|
selectState,
|
||||||
|
state => state.byId || {}
|
||||||
|
); */
|
||||||
|
export const selectCommentsByUri = createSelector(selectState, (state) => {
|
||||||
|
const byUri = state.commentsByUri || {};
|
||||||
|
const comments = {};
|
||||||
|
Object.keys(byUri).forEach((uri) => {
|
||||||
|
const claimId = byUri[uri];
|
||||||
|
if (claimId === null) {
|
||||||
|
comments[uri] = null;
|
||||||
|
} else {
|
||||||
|
comments[uri] = claimId;
|
||||||
}
|
}
|
||||||
);
|
});
|
||||||
|
|
||||||
export const selectModerationBlockList = createSelector(selectState, (state) =>
|
return comments;
|
||||||
state.moderationBlockList ? state.moderationBlockList.reverse() : []
|
});
|
||||||
|
|
||||||
|
export const selectPinnedCommentsById = (state: State) => selectState(state).pinnedCommentsById;
|
||||||
|
export const selectPinnedCommentsForUri = createCachedSelector(
|
||||||
|
selectCommentsByUri,
|
||||||
|
selectCommentsById,
|
||||||
|
selectPinnedCommentsById,
|
||||||
|
(state, uri) => uri,
|
||||||
|
(byUri, byId, pinnedCommentsById, uri) => {
|
||||||
|
const claimId = byUri[uri];
|
||||||
|
const pinnedCommentIds = pinnedCommentsById && pinnedCommentsById[claimId];
|
||||||
|
const pinnedComments = [];
|
||||||
|
|
||||||
|
if (pinnedCommentIds) {
|
||||||
|
pinnedCommentIds.forEach((commentId) => {
|
||||||
|
pinnedComments.push(byId[commentId]);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return pinnedComments;
|
||||||
|
}
|
||||||
|
)((state, uri) => uri);
|
||||||
|
|
||||||
|
export const selectModerationBlockList = createSelector(
|
||||||
|
(state) => selectState(state).moderationBlockList,
|
||||||
|
(moderationBlockList) => {
|
||||||
|
return moderationBlockList ? moderationBlockList.reverse() : [];
|
||||||
|
}
|
||||||
);
|
);
|
||||||
export const selectAdminBlockList = createSelector(selectState, (state) =>
|
export const selectAdminBlockList = createSelector(selectState, (state) =>
|
||||||
state.adminBlockList ? state.adminBlockList.reverse() : []
|
state.adminBlockList ? state.adminBlockList.reverse() : []
|
||||||
|
@ -46,35 +82,19 @@ export const selectModeratorBlockList = createSelector(selectState, (state) =>
|
||||||
state.moderatorBlockList ? state.moderatorBlockList.reverse() : []
|
state.moderatorBlockList ? state.moderatorBlockList.reverse() : []
|
||||||
);
|
);
|
||||||
|
|
||||||
export const selectPersonalTimeoutMap = createSelector(selectState, (state) => state.personalTimeoutMap);
|
export const selectPersonalTimeoutMap = (state: State) => selectState(state).personalTimeoutMap;
|
||||||
export const selectAdminTimeoutMap = createSelector(selectState, (state) => state.adminTimeoutMap);
|
export const selectAdminTimeoutMap = (state: State) => selectState(state).adminTimeoutMap;
|
||||||
export const selectModeratorTimeoutMap = createSelector(selectState, (state) => state.moderatorTimeoutMap);
|
export const selectModeratorTimeoutMap = (state: State) => selectState(state).moderatorTimeoutMap;
|
||||||
|
export const selectModeratorBlockListDelegatorsMap = (state: State) =>
|
||||||
export const selectModeratorBlockListDelegatorsMap = createSelector(
|
selectState(state).moderatorBlockListDelegatorsMap;
|
||||||
selectState,
|
export const selectTogglingForDelegatorMap = (state: State) => selectState(state).togglingForDelegatorMap;
|
||||||
(state) => state.moderatorBlockListDelegatorsMap
|
export const selectBlockingByUri = (state: State) => selectState(state).blockingByUri;
|
||||||
);
|
export const selectUnBlockingByUri = (state: State) => selectState(state).unBlockingByUri;
|
||||||
|
export const selectFetchingModerationBlockList = (state: State) => selectState(state).fetchingModerationBlockList;
|
||||||
export const selectTogglingForDelegatorMap = createSelector(selectState, (state) => state.togglingForDelegatorMap);
|
export const selectModerationDelegatesById = (state: State) => selectState(state).moderationDelegatesById;
|
||||||
|
export const selectIsFetchingModerationDelegates = (state: State) => selectState(state).fetchingModerationDelegates;
|
||||||
export const selectBlockingByUri = createSelector(selectState, (state) => state.blockingByUri);
|
export const selectModerationDelegatorsById = (state: State) => selectState(state).moderationDelegatorsById;
|
||||||
export const selectUnBlockingByUri = createSelector(selectState, (state) => state.unBlockingByUri);
|
export const selectIsFetchingModerationDelegators = (state: State) => selectState(state).fetchingModerationDelegators;
|
||||||
export const selectFetchingModerationBlockList = createSelector(
|
|
||||||
selectState,
|
|
||||||
(state) => state.fetchingModerationBlockList
|
|
||||||
);
|
|
||||||
|
|
||||||
export const selectModerationDelegatesById = createSelector(selectState, (state) => state.moderationDelegatesById);
|
|
||||||
export const selectIsFetchingModerationDelegates = createSelector(
|
|
||||||
selectState,
|
|
||||||
(state) => state.fetchingModerationDelegates
|
|
||||||
);
|
|
||||||
|
|
||||||
export const selectModerationDelegatorsById = createSelector(selectState, (state) => state.moderationDelegatorsById);
|
|
||||||
export const selectIsFetchingModerationDelegators = createSelector(
|
|
||||||
selectState,
|
|
||||||
(state) => state.fetchingModerationDelegators
|
|
||||||
);
|
|
||||||
|
|
||||||
export const selectHasAdminChannel = createSelector(selectState, (state) => {
|
export const selectHasAdminChannel = createSelector(selectState, (state) => {
|
||||||
const myChannelIds = Object.keys(state.moderationDelegatorsById);
|
const myChannelIds = Object.keys(state.moderationDelegatorsById);
|
||||||
|
@ -107,24 +127,28 @@ export const selectCommentsByClaimId = createSelector(selectState, selectComment
|
||||||
return comments;
|
return comments;
|
||||||
});
|
});
|
||||||
|
|
||||||
export const selectSuperchatsByUri = createSelector(selectState, (state) => state.superChatsByUri);
|
export const selectSuperchatsByUri = (state: State) => selectState(state).superChatsByUri;
|
||||||
|
|
||||||
export const selectTopLevelCommentsByClaimId = createSelector(selectState, selectCommentsById, (state, byId) => {
|
export const selectTopLevelCommentsByClaimId = createSelector(
|
||||||
const byClaimId = state.topLevelCommentsById || {};
|
(state) => selectState(state).topLevelCommentsById,
|
||||||
const comments = {};
|
selectCommentsById,
|
||||||
|
(topLevelCommentsById, byId) => {
|
||||||
|
const byClaimId = topLevelCommentsById || {};
|
||||||
|
const comments = {};
|
||||||
|
|
||||||
// replace every comment_id in the list with the actual comment object
|
// replace every comment_id in the list with the actual comment object
|
||||||
Object.keys(byClaimId).forEach((claimId) => {
|
Object.keys(byClaimId).forEach((claimId) => {
|
||||||
const commentIds = byClaimId[claimId];
|
const commentIds = byClaimId[claimId];
|
||||||
|
|
||||||
comments[claimId] = Array(commentIds === null ? 0 : commentIds.length);
|
comments[claimId] = Array(commentIds === null ? 0 : commentIds.length);
|
||||||
for (let i = 0; i < commentIds.length; i++) {
|
for (let i = 0; i < commentIds.length; i++) {
|
||||||
comments[claimId][i] = byId[commentIds[i]];
|
comments[claimId][i] = byId[commentIds[i]];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
return comments;
|
return comments;
|
||||||
});
|
}
|
||||||
|
);
|
||||||
|
|
||||||
export const makeSelectCommentForCommentId = (commentId: string) =>
|
export const makeSelectCommentForCommentId = (commentId: string) =>
|
||||||
createSelector(selectCommentsById, (comments) => comments[commentId]);
|
createSelector(selectCommentsById, (comments) => comments[commentId]);
|
||||||
|
@ -146,27 +170,7 @@ export const selectRepliesByParentId = createSelector(selectState, selectComment
|
||||||
return comments;
|
return comments;
|
||||||
});
|
});
|
||||||
|
|
||||||
// previously this used a mapping from claimId -> Array<Comments>
|
export const selectLinkedCommentAncestors = (state: State) => selectState(state).linkedCommentAncestors;
|
||||||
/* export const selectCommentsById = createSelector(
|
|
||||||
selectState,
|
|
||||||
state => state.byId || {}
|
|
||||||
); */
|
|
||||||
export const selectCommentsByUri = createSelector(selectState, (state) => {
|
|
||||||
const byUri = state.commentsByUri || {};
|
|
||||||
const comments = {};
|
|
||||||
Object.keys(byUri).forEach((uri) => {
|
|
||||||
const claimId = byUri[uri];
|
|
||||||
if (claimId === null) {
|
|
||||||
comments[uri] = null;
|
|
||||||
} else {
|
|
||||||
comments[uri] = claimId;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return comments;
|
|
||||||
});
|
|
||||||
|
|
||||||
export const selectLinkedCommentAncestors = createSelector(selectState, (state) => state.linkedCommentAncestors);
|
|
||||||
|
|
||||||
export const makeSelectCommentIdsForUri = (uri: string) =>
|
export const makeSelectCommentIdsForUri = (uri: string) =>
|
||||||
createSelector(selectState, selectCommentsByUri, selectClaimsById, (state, byUri) => {
|
createSelector(selectState, selectCommentsByUri, selectClaimsById, (state, byUri) => {
|
||||||
|
@ -174,62 +178,48 @@ export const makeSelectCommentIdsForUri = (uri: string) =>
|
||||||
return state.byId[claimId];
|
return state.byId[claimId];
|
||||||
});
|
});
|
||||||
|
|
||||||
export const selectMyReactionsByCommentId = createSelector(selectState, (state) => state.myReactsByCommentId);
|
const filterCommentsDepOnList = {
|
||||||
|
claimsById: selectClaimsById,
|
||||||
|
myClaims: selectMyActiveClaims,
|
||||||
|
mutedChannels: selectMutedChannels,
|
||||||
|
personalBlockList: selectModerationBlockList,
|
||||||
|
blacklistedMap: selectBlacklistedOutpointMap,
|
||||||
|
filteredMap: selectFilteredOutpointMap,
|
||||||
|
showMatureContent: selectShowMatureContent,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
const filterCommentsPropKeys = Object.keys(filterCommentsDepOnList);
|
||||||
* makeSelectMyReactionsForComment
|
|
||||||
*
|
|
||||||
* @param commentIdChannelId Format = "commentId:MyChannelId".
|
|
||||||
*/
|
|
||||||
export const makeSelectMyReactionsForComment = (commentIdChannelId: string) =>
|
|
||||||
createSelector(selectState, (state) => {
|
|
||||||
if (!state.myReactsByCommentId) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return state.myReactsByCommentId[commentIdChannelId] || [];
|
export const selectPendingCommentReacts = (state: State) => selectState(state).pendingCommentReactions;
|
||||||
});
|
export const selectSettingsByChannelId = (state: State) => selectState(state).settingsByChannelId;
|
||||||
|
export const selectFetchingCreatorSettings = (state: State) => selectState(state).fetchingSettings;
|
||||||
|
export const selectFetchingBlockedWords = (state: State) => selectState(state).fetchingBlockedWords;
|
||||||
|
|
||||||
export const makeSelectOthersReactionsForComment = (commentId: string) =>
|
export const selectCommentsForUri = createCachedSelector(
|
||||||
createSelector(selectState, (state) => {
|
(state, uri) => uri,
|
||||||
if (!state.othersReactsByCommentId) {
|
selectCommentsByClaimId,
|
||||||
return {};
|
selectCommentsByUri,
|
||||||
}
|
...Object.values(filterCommentsDepOnList),
|
||||||
|
(uri, byClaimId, byUri, ...filterInputs) => {
|
||||||
|
const claimId = byUri[uri];
|
||||||
|
const comments = byClaimId && byClaimId[claimId];
|
||||||
|
return filterComments(comments, claimId, filterInputs);
|
||||||
|
}
|
||||||
|
)((state, uri) => uri);
|
||||||
|
|
||||||
return state.othersReactsByCommentId[commentId] || {};
|
export const selectTopLevelCommentsForUri = createCachedSelector(
|
||||||
});
|
(state, uri) => uri,
|
||||||
|
(state, uri, maxCount) => maxCount,
|
||||||
export const selectPendingCommentReacts = createSelector(selectState, (state) => state.pendingCommentReactions);
|
selectTopLevelCommentsByClaimId,
|
||||||
|
selectCommentsByUri,
|
||||||
export const selectSettingsByChannelId = createSelector(selectState, (state) => state.settingsByChannelId);
|
...Object.values(filterCommentsDepOnList),
|
||||||
|
(uri, maxCount = -1, byClaimId, byUri, ...filterInputs) => {
|
||||||
export const selectFetchingCreatorSettings = createSelector(selectState, (state) => state.fetchingSettings);
|
const claimId = byUri[uri];
|
||||||
|
const comments = byClaimId && byClaimId[claimId];
|
||||||
export const selectFetchingBlockedWords = createSelector(selectState, (state) => state.fetchingBlockedWords);
|
const filtered = filterComments(comments, claimId, filterInputs);
|
||||||
|
return maxCount > 0 ? filtered.slice(0, maxCount) : filtered;
|
||||||
export const makeSelectCommentsForUri = (uri: string) =>
|
}
|
||||||
createSelector(
|
)((state, uri, maxCount = -1) => `${uri}:${maxCount}`);
|
||||||
(state) => state,
|
|
||||||
selectCommentsByClaimId,
|
|
||||||
selectCommentsByUri,
|
|
||||||
(state, byClaimId, byUri) => {
|
|
||||||
const claimId = byUri[uri];
|
|
||||||
const comments = byClaimId && byClaimId[claimId];
|
|
||||||
return makeSelectFilteredComments(comments, claimId)(state);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
export const makeSelectTopLevelCommentsForUri = (uri: string) =>
|
|
||||||
createSelector(
|
|
||||||
(state) => state,
|
|
||||||
selectTopLevelCommentsByClaimId,
|
|
||||||
selectCommentsByUri,
|
|
||||||
(state, byClaimId, byUri) => {
|
|
||||||
const claimId = byUri[uri];
|
|
||||||
const comments = byClaimId && byClaimId[claimId];
|
|
||||||
return makeSelectFilteredComments(comments, claimId)(state);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
export const makeSelectTopLevelTotalCommentsForUri = (uri: string) =>
|
export const makeSelectTopLevelTotalCommentsForUri = (uri: string) =>
|
||||||
createSelector(selectState, selectCommentsByUri, (state, byUri) => {
|
createSelector(selectState, selectCommentsByUri, (state, byUri) => {
|
||||||
|
@ -243,99 +233,93 @@ export const makeSelectTopLevelTotalPagesForUri = (uri: string) =>
|
||||||
return state.topLevelTotalPagesById[claimId] || 0;
|
return state.topLevelTotalPagesById[claimId] || 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
export const makeSelectRepliesForParentId = (id: string) =>
|
export const selectRepliesForParentId = createCachedSelector(
|
||||||
createSelector(
|
(state, id) => id,
|
||||||
(state) => state,
|
(state) => selectState(state).repliesByParentId,
|
||||||
selectCommentsById,
|
selectCommentsById,
|
||||||
(state, commentsById) => {
|
...Object.values(filterCommentsDepOnList),
|
||||||
// const claimId = byUri[uri]; // just parentId (id)
|
(id, repliesByParentId, commentsById, ...filterInputs) => {
|
||||||
const replyIdsByParentId = state.comments.repliesByParentId;
|
// const claimId = byUri[uri]; // just parentId (id)
|
||||||
const replyIdsForParent = replyIdsByParentId[id] || [];
|
const replyIdsForParent = repliesByParentId[id] || [];
|
||||||
if (!replyIdsForParent.length) return null;
|
if (!replyIdsForParent.length) return null;
|
||||||
|
|
||||||
const comments = [];
|
const comments = [];
|
||||||
replyIdsForParent.forEach((cid) => {
|
replyIdsForParent.forEach((cid) => {
|
||||||
comments.push(commentsById[cid]);
|
comments.push(commentsById[cid]);
|
||||||
});
|
});
|
||||||
// const comments = byParentId && byParentId[id];
|
// const comments = byParentId && byParentId[id];
|
||||||
|
|
||||||
return makeSelectFilteredComments(comments)(state);
|
return filterComments(comments, undefined, filterInputs);
|
||||||
}
|
}
|
||||||
);
|
)((state, id: string) => id);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* makeSelectFilteredComments
|
* filterComments
|
||||||
*
|
*
|
||||||
* @param comments List of comments to filter.
|
* @param comments List of comments to filter.
|
||||||
* @param claimId The claim that `comments` reside in.
|
* @param claimId The claim that `comments` reside in.
|
||||||
|
* @oaram filterInputs Values returned by filterCommentsDepOnList.
|
||||||
*/
|
*/
|
||||||
const makeSelectFilteredComments = (comments: Array<Comment>, claimId?: string) =>
|
const filterComments = (comments: Array<Comment>, claimId?: string, filterInputs: any) => {
|
||||||
createSelector(
|
const filterProps = filterInputs.reduce(function (acc, cur, i) {
|
||||||
selectClaimsById,
|
acc[filterCommentsPropKeys[i]] = cur;
|
||||||
selectMyActiveClaims,
|
return acc;
|
||||||
selectMutedChannels,
|
}, {});
|
||||||
selectModerationBlockList,
|
|
||||||
selectAdminBlockList,
|
const {
|
||||||
selectModeratorBlockList,
|
claimsById,
|
||||||
selectBlacklistedOutpointMap,
|
myClaims,
|
||||||
selectFilteredOutpointMap,
|
mutedChannels,
|
||||||
selectShowMatureContent,
|
personalBlockList,
|
||||||
(
|
blacklistedMap,
|
||||||
claimsById,
|
filteredMap,
|
||||||
myClaims,
|
showMatureContent,
|
||||||
mutedChannels,
|
} = filterProps;
|
||||||
personalBlockList,
|
|
||||||
adminBlockList,
|
return comments
|
||||||
moderatorBlockList,
|
? comments.filter((comment) => {
|
||||||
blacklistedMap,
|
if (!comment) {
|
||||||
filteredMap,
|
// It may have been recently deleted after being blocked
|
||||||
showMatureContent
|
return false;
|
||||||
) => {
|
}
|
||||||
return comments
|
|
||||||
? comments.filter((comment) => {
|
const channelClaim = claimsById[comment.channel_id];
|
||||||
if (!comment) {
|
|
||||||
// It may have been recently deleted after being blocked
|
// Return comment if `channelClaim` doesn't exist so the component knows to resolve the author
|
||||||
|
if (channelClaim) {
|
||||||
|
if (myClaims && myClaims.size > 0) {
|
||||||
|
const claimIsMine = channelClaim.is_my_output || myClaims.has(channelClaim.claim_id);
|
||||||
|
if (claimIsMine) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const outpoint = `${channelClaim.txid}:${channelClaim.nout}`;
|
||||||
|
if (blacklistedMap[outpoint] || filteredMap[outpoint]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!showMatureContent) {
|
||||||
|
const claimIsMature = isClaimNsfw(channelClaim);
|
||||||
|
if (claimIsMature) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const channelClaim = claimsById[comment.channel_id];
|
if (claimId) {
|
||||||
|
const claimIdIsMine = myClaims && myClaims.size > 0 && myClaims.has(claimId);
|
||||||
// Return comment if `channelClaim` doesn't exist so the component knows to resolve the author
|
if (!claimIdIsMine) {
|
||||||
if (channelClaim) {
|
if (personalBlockList.includes(comment.channel_url)) {
|
||||||
if (myClaims && myClaims.size > 0) {
|
return false;
|
||||||
const claimIsMine = channelClaim.is_my_output || myClaims.has(channelClaim.claim_id);
|
|
||||||
if (claimIsMine) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const outpoint = `${channelClaim.txid}:${channelClaim.nout}`;
|
|
||||||
if (blacklistedMap[outpoint] || filteredMap[outpoint]) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!showMatureContent) {
|
|
||||||
const claimIsMature = isClaimNsfw(channelClaim);
|
|
||||||
if (claimIsMature) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (claimId) {
|
return !mutedChannels.includes(comment.channel_url);
|
||||||
const claimIdIsMine = myClaims && myClaims.size > 0 && myClaims.has(claimId);
|
})
|
||||||
if (!claimIdIsMine) {
|
: [];
|
||||||
if (personalBlockList.includes(comment.channel_url) || adminBlockList.includes(comment.channel_url)) {
|
};
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return !mutedChannels.includes(comment.channel_url);
|
|
||||||
})
|
|
||||||
: [];
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
export const makeSelectTotalReplyPagesForParentId = (parentId: string) =>
|
export const makeSelectTotalReplyPagesForParentId = (parentId: string) =>
|
||||||
createSelector(selectState, (state) => {
|
createSelector(selectState, (state) => {
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
// @flow
|
// @flow
|
||||||
import { createSelector } from 'reselect';
|
import { createSelector } from 'reselect';
|
||||||
|
|
||||||
const selectState = (state: { tags: TagState }) => state.tags || {};
|
type State = { tags: TagState };
|
||||||
|
|
||||||
|
const selectState = (state: State) => state.tags || {};
|
||||||
|
|
||||||
export const selectKnownTagsByName = createSelector(selectState, (state: TagState): KnownTags => state.knownTags);
|
export const selectKnownTagsByName = createSelector(selectState, (state: TagState): KnownTags => state.knownTags);
|
||||||
|
|
||||||
export const selectFollowedTagsList = createSelector(selectState, (state: TagState): Array<string> =>
|
export const selectFollowedTagsList = (state: State) => selectState(state).followedTags;
|
||||||
state.followedTags.filter(tag => typeof tag === 'string')
|
|
||||||
);
|
|
||||||
|
|
||||||
export const selectFollowedTags = createSelector(selectFollowedTagsList, (followedTags: Array<string>): Array<Tag> =>
|
export const selectFollowedTags = createSelector(selectFollowedTagsList, (followedTags: Array<string>): Array<Tag> =>
|
||||||
followedTags.map(tag => ({ name: tag.toLowerCase() })).sort((a, b) => a.name.localeCompare(b.name))
|
followedTags.map((tag) => ({ name: tag.toLowerCase() })).sort((a, b) => a.name.localeCompare(b.name))
|
||||||
);
|
);
|
||||||
|
|
||||||
export const selectUnfollowedTags = createSelector(
|
export const selectUnfollowedTags = createSelector(
|
||||||
|
@ -19,7 +19,7 @@ export const selectUnfollowedTags = createSelector(
|
||||||
(tagsByName: KnownTags, followedTags: Array<string>): Array<Tag> => {
|
(tagsByName: KnownTags, followedTags: Array<string>): Array<Tag> => {
|
||||||
const followedTagsSet = new Set(followedTags);
|
const followedTagsSet = new Set(followedTags);
|
||||||
let tagsToReturn = [];
|
let tagsToReturn = [];
|
||||||
Object.keys(tagsByName).forEach(key => {
|
Object.keys(tagsByName).forEach((key) => {
|
||||||
if (!followedTagsSet.has(key)) {
|
if (!followedTagsSet.has(key)) {
|
||||||
const { name } = tagsByName[key];
|
const { name } = tagsByName[key];
|
||||||
tagsToReturn.push({ name: name.toLowerCase() });
|
tagsToReturn.push({ name: name.toLowerCase() });
|
||||||
|
@ -31,6 +31,6 @@ export const selectUnfollowedTags = createSelector(
|
||||||
);
|
);
|
||||||
|
|
||||||
export const makeSelectIsFollowingTag = (tag: string) =>
|
export const makeSelectIsFollowingTag = (tag: string) =>
|
||||||
createSelector(selectFollowedTags, followedTags => {
|
createSelector(selectFollowedTags, (followedTags) => {
|
||||||
return followedTags.some(followedTag => followedTag.name === tag.toLowerCase());
|
return followedTags.some((followedTag) => followedTag.name === tag.toLowerCase());
|
||||||
});
|
});
|
||||||
|
|
|
@ -13317,6 +13317,11 @@ rc@^1.0.1, rc@^1.1.6, rc@^1.2.8:
|
||||||
minimist "^1.2.0"
|
minimist "^1.2.0"
|
||||||
strip-json-comments "~2.0.1"
|
strip-json-comments "~2.0.1"
|
||||||
|
|
||||||
|
re-reselect@^4.0.0:
|
||||||
|
version "4.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/re-reselect/-/re-reselect-4.0.0.tgz#9ddec4c72c4d952f68caa5aa4b76a9ed38b75cac"
|
||||||
|
integrity sha512-wuygyq8TXUlSdVXv2kigXxQNOgdb9m7LbIjwfTNGSpaY1riLd5e+VeQjlQMyUtrk0oiyhi1AqIVynworl3qxHA==
|
||||||
|
|
||||||
react-async-script@^1.1.1:
|
react-async-script@^1.1.1:
|
||||||
version "1.1.1"
|
version "1.1.1"
|
||||||
resolved "https://registry.yarnpkg.com/react-async-script/-/react-async-script-1.1.1.tgz#f481c6c5f094bf4b94a9d52da0d0dda2e1a74bdf"
|
resolved "https://registry.yarnpkg.com/react-async-script/-/react-async-script-1.1.1.tgz#f481c6c5f094bf4b94a9d52da0d0dda2e1a74bdf"
|
||||||
|
|
Loading…
Reference in a new issue