UNPKG

@amityco/ts-sdk-react-native

Version:

Amity Social Cloud Typescript SDK

171 lines (147 loc) 6.19 kB
import hash from 'object-hash'; import { pullFromCache, pushToCache } from '~/cache/api'; import { CommentPaginationController } from './CommentPaginationController'; import { CommentQueryStreamController } from './CommentQueryStreamController'; import { LiveCollectionController } from '~/core/liveCollection/LiveCollectionController'; import { onCommentCreated, onCommentUpdated, onCommentDeleted, onCommentFlagged, onCommentUnflagged, onCommentReactionAdded, onCommentReactionRemoved, } from '~/commentRepository/events'; import { filterByPropEquality, sortByFirstCreated, sortByLastCreated } from '~/core/query'; import { isNonNullable } from '~/utils'; import { EnumCommentActions } from './enums'; import { LinkedObject } from '~/utils/linkedObject'; import { prepareCommentPayload } from '~/commentRepository/utils/payload'; import { onCommentCreatedLocal } from '~/commentRepository/events/onCommentCreatedLocal'; import { onCommentDeleteLocal } from '~/commentRepository/events/onCommentDeletedLocal'; import { onLocalCommentReactionAdded } from '~/commentRepository/events/onLocalCommentReactionAdded'; import { onLocalCommentReactionRemoved } from '~/commentRepository/events/onLocalCommentReactionRemoved'; export class CommentLiveCollectionController extends LiveCollectionController< 'comment', Amity.CommentLiveCollection, Amity.Comment, CommentPaginationController > { private queryStreamController: CommentQueryStreamController; private query: Amity.CommentLiveCollection; constructor( query: Amity.CommentLiveCollection, callback: Amity.LiveCollectionCallback<Amity.Comment>, ) { const queryStreamId = hash(query); const cacheKey = ['comments', 'collection', queryStreamId]; const paginationController = new CommentPaginationController(query); super(paginationController, queryStreamId, cacheKey, callback); this.query = query; this.queryStreamController = new CommentQueryStreamController( this.query, this.cacheKey, this.notifyChange.bind(this), prepareCommentPayload, ); this.callback = callback.bind(this); this.loadPage({ initial: true }); } protected setup() { const collection = pullFromCache<Amity.CommentLiveCollectionCache>(this.cacheKey)?.data; if (!collection) { pushToCache(this.cacheKey, { data: [], params: {}, }); } } protected async persistModel(queryPayload: Amity.CommentPayload & Amity.Pagination) { await this.queryStreamController.saveToMainDB(queryPayload); } protected persistQueryStream({ response, direction, refresh, }: Amity.LiveCollectionPersistQueryStreamParams<'comment'>) { this.queryStreamController.appendToQueryStream(response, direction, refresh); } startSubscription() { return this.queryStreamController.subscribeRTE([ { fn: onCommentCreatedLocal, action: EnumCommentActions.OnCommentCreated }, { fn: onCommentDeleteLocal, action: EnumCommentActions.OnCommentDeleted }, { fn: onCommentCreated, action: EnumCommentActions.OnCommentCreated }, { fn: onCommentUpdated, action: EnumCommentActions.OnCommentUpdated }, { fn: onCommentDeleted, action: EnumCommentActions.OnCommentDeleted }, { fn: onCommentFlagged, action: EnumCommentActions.OnCommentFlagged }, { fn: onCommentUnflagged, action: EnumCommentActions.OnCommentUnflagged }, { fn: onCommentReactionAdded, action: EnumCommentActions.OnCommentReactionAdded }, { fn: onCommentReactionRemoved, action: EnumCommentActions.OnCommentReactionRemoved }, { fn: onLocalCommentReactionAdded, action: EnumCommentActions.OnCommentReactionAdded }, { fn: onLocalCommentReactionRemoved, action: EnumCommentActions.OnCommentReactionRemoved }, ]); } notifyChange({ origin, loading, error }: Amity.LiveCollectionNotifyParams) { const collection = pullFromCache<Amity.CommentLiveCollectionCache>(this.cacheKey)?.data; if (!collection) return; const data = this.applyFilter( collection.data .map(id => pullFromCache<Amity.InternalComment>(['comment', 'get', id])!) .filter(isNonNullable) .map(({ data }) => data) ?? [], ).map(LinkedObject.comment); if (!this.shouldNotify(data) && origin === 'event') return; this.callback({ onNextPage: () => this.loadPage({ direction: Amity.LiveCollectionPageDirection.NEXT }), data, hasNextPage: !!this.paginationController.getNextToken(), loading, error, }); } applyFilter(data: Amity.InternalComment[]) { let comments = data; if (!this.query.includeDeleted) { comments = filterByPropEquality(comments, 'isDeleted', false); } if (this.query.parentId) { comments = comments.filter(comment => comment.parentId === this.query.parentId); } if (typeof this.query.hasFlag === 'boolean') { if (this.query.hasFlag) { comments = comments.filter(comment => comment.hashFlag != null); } else { comments = comments.filter(comment => comment.hashFlag == null); } } if (this.query.dataTypes) { if (this.query.dataTypes.matchType === 'exact') { comments = comments.filter(comment => { const sortedDataTypesQueryValue = this.query.dataTypes?.values.sort() || []; const sortedDataTypesCommentValue = comment.dataTypes?.sort() || []; if (sortedDataTypesCommentValue.length !== sortedDataTypesQueryValue.length) { return false; } return sortedDataTypesQueryValue.every( (value, index) => value === sortedDataTypesCommentValue[index], ); }); } if (this.query.dataTypes.matchType === 'any') { comments = comments.filter(comment => this.query.dataTypes?.values.some(value => comment.dataTypes?.includes(value)), ); } } switch (this.query.sortBy) { case 'firstCreated': comments = comments.sort(sortByFirstCreated); break; case 'lastCreated': default: comments = comments.sort(sortByLastCreated); break; } return comments; } }