2020-10-06 15:35:13 -04:00
|
|
|
// @flow
|
|
|
|
import * as REACTION_TYPES from 'constants/reactions';
|
|
|
|
import { SORT_COMMENTS_NEW, SORT_COMMENTS_BEST, SORT_COMMENTS_CONTROVERSIAL } from 'constants/comment';
|
|
|
|
|
|
|
|
// Mostly taken from Reddit's sorting functions
|
|
|
|
// https://github.com/reddit-archive/reddit/blob/master/r2/r2/lib/db/_sorts.pyx
|
|
|
|
|
2020-10-07 16:18:16 -04:00
|
|
|
type SortProps = {
|
|
|
|
comments: ?Array<Comment>,
|
|
|
|
reactionsById: {},
|
|
|
|
sort: string,
|
2021-05-17 16:55:23 -03:00
|
|
|
isMyComment: (string) => boolean,
|
2021-07-15 22:23:26 +08:00
|
|
|
justCommented: Array<string>,
|
2020-10-07 16:18:16 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
export function sortComments(sortProps: SortProps): Array<Comment> {
|
2021-07-15 22:23:26 +08:00
|
|
|
const { comments, reactionsById, sort, isMyComment, justCommented } = sortProps;
|
2020-10-07 16:18:16 -04:00
|
|
|
|
2021-05-15 01:56:58 -03:00
|
|
|
if (!comments) return [];
|
2020-10-06 15:35:13 -04:00
|
|
|
|
2020-10-07 16:18:16 -04:00
|
|
|
return comments.slice().sort((a: Comment, b: Comment) => {
|
2020-10-08 11:31:36 -04:00
|
|
|
if (a.is_pinned) {
|
|
|
|
return -1;
|
|
|
|
} else if (b.is_pinned) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-05-15 01:56:58 -03:00
|
|
|
if (sort === SORT_COMMENTS_NEW) return 0;
|
2020-10-06 15:35:13 -04:00
|
|
|
|
2020-10-07 16:18:16 -04:00
|
|
|
const aIsMine = isMyComment(a.channel_id);
|
|
|
|
const bIsMine = isMyComment(b.channel_id);
|
2021-07-15 22:23:26 +08:00
|
|
|
const aIsMyRecent = justCommented.includes(a.comment_id);
|
|
|
|
const bIsMyRecent = justCommented.includes(b.comment_id);
|
2020-10-07 16:18:16 -04:00
|
|
|
|
2021-07-15 22:23:26 +08:00
|
|
|
if (aIsMine && justCommented.length && aIsMyRecent) {
|
2020-10-07 16:18:16 -04:00
|
|
|
return -1;
|
2021-07-15 22:23:26 +08:00
|
|
|
} else if (bIsMine && justCommented.length && bIsMyRecent) {
|
2020-10-07 16:18:16 -04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-10-06 15:35:13 -04:00
|
|
|
const aReactions = reactionsById[a.comment_id];
|
|
|
|
const bReactions = reactionsById[b.comment_id];
|
|
|
|
const aLikes = (aReactions && aReactions[REACTION_TYPES.LIKE]) || 0;
|
|
|
|
const aDislikes = (aReactions && aReactions[REACTION_TYPES.DISLIKE]) || 0;
|
|
|
|
const bLikes = (bReactions && bReactions[REACTION_TYPES.LIKE]) || 0;
|
|
|
|
const bDislikes = (bReactions && bReactions[REACTION_TYPES.DISLIKE]) || 0;
|
|
|
|
|
2020-10-07 16:18:16 -04:00
|
|
|
if (sort === SORT_COMMENTS_CONTROVERSIAL) {
|
2020-10-06 15:35:13 -04:00
|
|
|
if (aLikes === 0 && aDislikes === 0) {
|
|
|
|
return 1;
|
|
|
|
} else if (bLikes === 0 && bDislikes === 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const aMagnitude = aLikes + aDislikes;
|
|
|
|
const bMagnitude = bLikes + bDislikes;
|
|
|
|
|
|
|
|
const aBalance = aLikes > aDislikes ? aDislikes / aLikes : aLikes / aDislikes;
|
|
|
|
const bBalance = bLikes > bDislikes ? bDislikes / bLikes : bLikes / bDislikes;
|
|
|
|
|
|
|
|
return bMagnitude ** bBalance - aMagnitude ** aBalance;
|
|
|
|
}
|
|
|
|
|
2020-10-07 16:18:16 -04:00
|
|
|
if (sort === SORT_COMMENTS_BEST) {
|
2020-10-06 15:35:13 -04:00
|
|
|
const aN = aLikes + aDislikes;
|
|
|
|
const bN = bLikes + bDislikes;
|
|
|
|
|
2020-10-28 12:17:22 -04:00
|
|
|
if (aLikes === 0 && bLikes === 0 && (aDislikes || bDislikes)) {
|
|
|
|
return aDislikes - bDislikes;
|
2020-11-02 17:15:06 -05:00
|
|
|
} else if (aLikes === 0 && bLikes > 0) {
|
|
|
|
return 1;
|
|
|
|
} else if (bLikes === 0 && aLikes > 1) {
|
|
|
|
return -1;
|
2020-10-06 15:35:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const z = 1.281551565545; // 80% confidence
|
|
|
|
const aP = aLikes / aN;
|
|
|
|
const bP = bLikes / bN;
|
|
|
|
|
|
|
|
const aLeft = aP + (1 / (2 * aN)) * z * z;
|
|
|
|
const aRight = z * Math.sqrt((aP * (1 - aP)) / aN + (z * z) / (4 * aN * aN));
|
|
|
|
const aUnder = 1 + (1 / aN) * z * z;
|
|
|
|
|
|
|
|
const bLeft = bP + (1 / (2 * bN)) * z * z;
|
|
|
|
const bRight = z * Math.sqrt((bP * (1 - bP)) / bN + (z * z) / (4 * bN * bN));
|
|
|
|
const bUnder = 1 + (1 / bN) * z * z;
|
|
|
|
|
|
|
|
return (bLeft - bRight) / bUnder - (aLeft - aRight) / aUnder;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
});
|
|
|
|
}
|