UNPKG

@amityco/ts-sdk-react-native

Version:

Amity Social Cloud Typescript SDK

145 lines (117 loc) 4.82 kB
/* eslint-disable no-use-before-define */ import { QueryStreamController } from '~/core/liveCollection/QueryStreamController'; import { pullFromCache, pushToCache } from '~/cache/api'; import { ingestInCache } from '~/cache/api/ingestInCache'; import { getActiveClient } from '~/client'; import { MessagePaginationController } from './MessagePaginationController'; import { getResolver } from '~/core/model'; export class MessageQueryStreamController extends QueryStreamController< Amity.MessagePayload, Amity.MessagesLiveCollection > { private notifyChange: (params: Amity.LiveCollectionNotifyParams) => void; private preparePayload: ( response: Amity.MessagePayload, ) => Promise<Amity.ProcessedMessagePayload>; private paginationController: MessagePaginationController; constructor( query: Amity.MessagesLiveCollection, cacheKey: string[], notifyChange: (params: Amity.LiveCollectionNotifyParams) => void, preparePayload: (response: Amity.MessagePayload) => Promise<Amity.ProcessedMessagePayload>, paginationController: MessagePaginationController, ) { super(query, cacheKey); this.notifyChange = notifyChange; this.preparePayload = preparePayload; this.paginationController = paginationController; } async saveToMainDB(response: Amity.MessagePayload) { const processedPayload = await this.preparePayload(response); const client = getActiveClient(); const cachedAt = client.cache && Date.now(); if (client.cache) { ingestInCache(processedPayload, { cachedAt }); } } appendToQueryStream( response: Amity.MessagePayload & Partial<Amity.Pagination>, direction: Amity.LiveCollectionPageDirection, refresh = false, ) { if (refresh) { pushToCache(this.cacheKey, { data: response.messages.map(getResolver('message')), query: this.query, }); } else { const collection = pullFromCache<Amity.MessageLiveCollectionCache>(this.cacheKey)?.data; const messages = collection?.data ?? []; pushToCache(this.cacheKey, { ...collection, data: direction === 'next' ? [...new Set([...messages, ...response.messages.map(getResolver('message'))])] : [...new Set([...response.messages.map(getResolver('message')), ...messages])], }); } } reactor(action: string) { return (payload: Amity.InternalMessage) => { if (action === 'onCreate') { const collection = pullFromCache<Amity.MessageLiveCollectionCache>(this.cacheKey)?.data; const { referenceId } = payload; if (!collection) return; if (this.query.subChannelId !== payload?.subChannelId || !collection) return; if (this.query.type && this.query.type !== payload.dataType) return; if ( this.query.excludingTags && this.query.excludingTags?.some(value => payload.tags?.includes(value)) ) return; if (!!this.query.hasFlags !== !!payload.flagCount) return; if (this.query.parentId && this.query.parentId !== payload.parentId) return; if ( this.query.hasOwnProperty('includeDeleted') && !this.query.includeDeleted && payload.isDeleted ) return; if ( this.query.includingTags && !this.query.includingTags?.some(value => payload.tags?.includes(value)) ) return; if ( (!this.query.sortBy || this.query.sortBy === 'segmentDesc') && !this.paginationController.getPrevToken() ) { collection.data = [...new Set([referenceId ?? payload.messageId, ...collection.data])]; } if (this.query.sortBy === 'segmentAsc' && !this.paginationController.getNextToken()) { collection.data = [...new Set([...collection.data, referenceId ?? payload.messageId])]; } pushToCache(this.cacheKey, collection); } if (action === 'onDelete' && payload.syncState === Amity.SyncState.Error) { const collection = pullFromCache<Amity.MessageLiveCollectionCache>(this.cacheKey)?.data; if (!collection) return; if (collection.data.includes(payload.messageId)) { const newCollectionData = collection.data.filter( messageId => messageId !== payload.messageId, ); pushToCache(this.cacheKey, { ...collection, data: newCollectionData }); } } this.notifyChange({ origin: Amity.LiveDataOrigin.EVENT, loading: false }); }; } subscribeRTE( createSubscriber: { fn: (reactor: Amity.Listener<Amity.InternalMessage>) => Amity.Unsubscriber; action: string; }[], ) { return createSubscriber.map(subscriber => subscriber.fn(this.reactor(subscriber.action))); } }