verdnatura-chat/app/lib/methods/subscriptions/room.js

241 lines
6.5 KiB
JavaScript
Raw Normal View History

2019-04-08 12:35:28 +00:00
import EJSON from 'ejson';
import { sanitizedRaw } from '@nozbe/watermelondb/RawRecord';
import { InteractionManager } from 'react-native';
2019-04-08 12:35:28 +00:00
import log from '../../../utils/log';
2019-04-08 12:35:28 +00:00
import protectedFunction from '../helpers/protectedFunction';
import buildMessage from '../helpers/buildMessage';
import database from '../../database';
import reduxStore from '../../createStore';
import { addUserTyping, removeUserTyping, clearUserTyping } from '../../../actions/usersTyping';
2019-12-11 23:01:12 +00:00
import debounce from '../../../utils/debounce';
2020-02-19 19:43:47 +00:00
import RocketChat from '../../rocketchat';
export default class RoomSubscription {
constructor(rid) {
this.rid = rid;
this.isAlive = true;
}
subscribe = async() => {
console.log(`[RCRN] Subscribing to room ${ this.rid }`);
if (this.promises) {
await this.unsubscribe();
}
this.promises = RocketChat.subscribeRoom(this.rid);
this.connectedListener = RocketChat.onStreamData('connected', this.handleConnection);
this.disconnectedListener = RocketChat.onStreamData('close', this.handleConnection);
this.notifyRoomListener = RocketChat.onStreamData('stream-notify-room', this.handleNotifyRoomReceived);
this.messageReceivedListener = RocketChat.onStreamData('stream-room-messages', this.handleMessageReceived);
if (!this.isAlive) {
this.unsubscribe();
}
}
Beta (#265) * Fabric iOS * Fabric configured on iOS and Android * - react-native-fabric configured - login tracked * README updated * Run scripts from README updated * README scripts * get rooms and messages by rest * user status * more improves * more improves * send pong on timeout * fix some methods * more tests * rest messages * Room actions (#266) * Toggle notifications * Search messages * Invite users * Mute/Unmute users in room * rocket.cat messages * Room topic layout fixed * Starred messages loading onEndReached * Room actions onEndReached * Unnecessary login request * Login loading * Login services fixed * User presence layout * ïmproves on room actions view * Removed unnecessary data from SelectedUsersView * load few messages on open room, search message improve * fix loading messages forever * Removed state from search * Custom message time format * secureTextEntry layout * Reduce android app size * Roles subscription fix * Public routes navigation * fix reconnect * - New login/register, login, register * proguard * Login flux * App init/restore * Android layout fixes * Multiple meteor connection requests fixed * Nested attachments * Nested attachments * fix check status * New login layout (#269) * Public routes navigation * New login/register, login, register * Multiple meteor connection requests fixed * Nested attachments * Button component * TextInput android layout fixed * Register fixed * Thinner close modal button * Requests /me after login only one time * Static images moved * fix reconnect * fix ddp * fix custom emoji * New message layout (#273) * Grouping messages * Message layout * Users typing animation * Image attachment layout
2018-04-24 19:34:03 +00:00
2020-02-19 19:43:47 +00:00
unsubscribe = async() => {
console.log(`[RCRN] Unsubscribing from room ${ this.rid }`);
this.isAlive = false;
if (this.promises) {
try {
const subscriptions = await this.promises || [];
subscriptions.forEach(sub => sub.unsubscribe().catch(() => console.log('unsubscribeRoom')));
} catch (e) {
// do nothing
}
}
this.removeListener(this.connectedListener);
this.removeListener(this.disconnectedListener);
this.removeListener(this.notifyRoomListener);
this.removeListener(this.messageReceivedListener);
reduxStore.dispatch(clearUserTyping());
}
2020-01-29 20:20:36 +00:00
2020-02-19 19:43:47 +00:00
removeListener = async(promise) => {
if (promise) {
try {
const listener = await promise;
listener.stop();
} catch (e) {
// do nothing
}
}
};
Beta (#265) * Fabric iOS * Fabric configured on iOS and Android * - react-native-fabric configured - login tracked * README updated * Run scripts from README updated * README scripts * get rooms and messages by rest * user status * more improves * more improves * send pong on timeout * fix some methods * more tests * rest messages * Room actions (#266) * Toggle notifications * Search messages * Invite users * Mute/Unmute users in room * rocket.cat messages * Room topic layout fixed * Starred messages loading onEndReached * Room actions onEndReached * Unnecessary login request * Login loading * Login services fixed * User presence layout * ïmproves on room actions view * Removed unnecessary data from SelectedUsersView * load few messages on open room, search message improve * fix loading messages forever * Removed state from search * Custom message time format * secureTextEntry layout * Reduce android app size * Roles subscription fix * Public routes navigation * fix reconnect * - New login/register, login, register * proguard * Login flux * App init/restore * Android layout fixes * Multiple meteor connection requests fixed * Nested attachments * Nested attachments * fix check status * New login layout (#269) * Public routes navigation * New login/register, login, register * Multiple meteor connection requests fixed * Nested attachments * Button component * TextInput android layout fixed * Register fixed * Thinner close modal button * Requests /me after login only one time * Static images moved * fix reconnect * fix ddp * fix custom emoji * New message layout (#273) * Grouping messages * Message layout * Users typing animation * Image attachment layout
2018-04-24 19:34:03 +00:00
2020-02-19 19:43:47 +00:00
handleConnection = () => {
RocketChat.loadMissedMessages({ rid: this.rid }).catch(e => console.log(e));
2019-04-08 12:35:28 +00:00
};
2020-02-19 19:43:47 +00:00
handleNotifyRoomReceived = protectedFunction((ddpMessage) => {
2019-04-08 12:35:28 +00:00
const [_rid, ev] = ddpMessage.fields.eventName.split('/');
2020-02-19 19:43:47 +00:00
if (this.rid !== _rid) {
2019-04-08 12:35:28 +00:00
return;
}
if (ev === 'typing') {
const [username, typing] = ddpMessage.fields.args;
if (typing) {
reduxStore.dispatch(addUserTyping(username));
2019-04-08 12:35:28 +00:00
} else {
reduxStore.dispatch(removeUserTyping(username));
2019-04-08 12:35:28 +00:00
}
} else if (ev === 'deleteMessage') {
InteractionManager.runAfterInteractions(async() => {
2019-04-08 12:35:28 +00:00
if (ddpMessage && ddpMessage.fields && ddpMessage.fields.args.length > 0) {
try {
const { _id } = ddpMessage.fields.args[0];
const db = database.active;
const msgCollection = db.collections.get('messages');
const threadsCollection = db.collections.get('threads');
const threadMessagesCollection = db.collections.get('thread_messages');
let deleteMessage;
let deleteThread;
let deleteThreadMessage;
// Delete message
try {
const m = await msgCollection.find(_id);
deleteMessage = m.prepareDestroyPermanently();
} catch (e) {
// Do nothing
}
// Delete thread
try {
const m = await threadsCollection.find(_id);
deleteThread = m.prepareDestroyPermanently();
} catch (e) {
// Do nothing
}
// Delete thread message
try {
const m = await threadMessagesCollection.find(_id);
deleteThreadMessage = m.prepareDestroyPermanently();
} catch (e) {
// Do nothing
}
await db.action(async() => {
await db.batch(
deleteMessage, deleteThread, deleteThreadMessage
);
2019-04-17 17:01:03 +00:00
});
} catch (e) {
log(e);
2019-04-17 17:01:03 +00:00
}
2019-04-08 12:35:28 +00:00
}
});
}
});
2020-02-19 19:43:47 +00:00
read = debounce((lastOpen) => {
RocketChat.readMessages(this.rid, lastOpen);
2019-12-11 23:01:12 +00:00
}, 300);
2020-02-19 19:43:47 +00:00
handleMessageReceived = protectedFunction((ddpMessage) => {
2019-04-17 17:01:03 +00:00
const message = buildMessage(EJSON.fromJSONValue(ddpMessage.fields.args[0]));
const lastOpen = new Date();
2020-02-19 19:43:47 +00:00
if (this.rid !== message.rid) {
2019-04-08 12:35:28 +00:00
return;
}
InteractionManager.runAfterInteractions(async() => {
const db = database.active;
const batch = [];
const msgCollection = db.collections.get('messages');
const threadsCollection = db.collections.get('threads');
const threadMessagesCollection = db.collections.get('thread_messages');
2019-12-11 23:01:12 +00:00
let messageRecord;
let threadRecord;
let threadMessageRecord;
// Create or update message
2019-04-08 12:35:28 +00:00
try {
2019-12-11 23:01:12 +00:00
messageRecord = await msgCollection.find(message._id);
} catch (error) {
2019-12-11 23:01:12 +00:00
// Do nothing
}
if (messageRecord) {
try {
const update = messageRecord.prepareUpdate((m) => {
Object.assign(m, message);
});
batch.push(update);
} catch (e) {
console.log(e);
}
} else {
batch.push(
msgCollection.prepareCreate(protectedFunction((m) => {
m._raw = sanitizedRaw({ id: message._id }, msgCollection.schema);
2020-02-19 19:43:47 +00:00
m.subscription.id = this.rid;
Object.assign(m, message);
}))
);
}
// Create or update thread
if (message.tlm) {
try {
2019-12-11 23:01:12 +00:00
threadRecord = await threadsCollection.find(message._id);
} catch (error) {
// Do nothing
}
if (threadRecord) {
batch.push(
2019-12-11 23:01:12 +00:00
threadRecord.prepareUpdate(protectedFunction((t) => {
Object.assign(t, message);
2019-12-11 23:01:12 +00:00
}))
);
2019-12-11 23:01:12 +00:00
} else {
batch.push(
threadsCollection.prepareCreate(protectedFunction((t) => {
t._raw = sanitizedRaw({ id: message._id }, threadsCollection.schema);
2020-02-19 19:43:47 +00:00
t.subscription.id = this.rid;
Object.assign(t, message);
}))
);
}
}
2019-04-08 12:35:28 +00:00
// Create or update thread message
if (message.tmid) {
try {
2019-12-11 23:01:12 +00:00
threadMessageRecord = await threadMessagesCollection.find(message._id);
} catch (error) {
// Do nothing
}
if (threadMessageRecord) {
batch.push(
2019-12-11 23:01:12 +00:00
threadMessageRecord.prepareUpdate(protectedFunction((tm) => {
Object.assign(tm, message);
tm.rid = message.tmid;
delete tm.tmid;
2019-12-11 23:01:12 +00:00
}))
);
2019-12-11 23:01:12 +00:00
} else {
batch.push(
threadMessagesCollection.prepareCreate(protectedFunction((tm) => {
tm._raw = sanitizedRaw({ id: message._id }, threadMessagesCollection.schema);
Object.assign(tm, message);
2020-02-19 19:43:47 +00:00
tm.subscription.id = this.rid;
tm.rid = message.tmid;
delete tm.tmid;
}))
);
}
}
2020-02-19 19:43:47 +00:00
this.read(lastOpen);
try {
await db.action(async() => {
await db.batch(...batch);
});
} catch (e) {
log(e);
2019-04-08 12:35:28 +00:00
}
});
});
Beta (#265) * Fabric iOS * Fabric configured on iOS and Android * - react-native-fabric configured - login tracked * README updated * Run scripts from README updated * README scripts * get rooms and messages by rest * user status * more improves * more improves * send pong on timeout * fix some methods * more tests * rest messages * Room actions (#266) * Toggle notifications * Search messages * Invite users * Mute/Unmute users in room * rocket.cat messages * Room topic layout fixed * Starred messages loading onEndReached * Room actions onEndReached * Unnecessary login request * Login loading * Login services fixed * User presence layout * ïmproves on room actions view * Removed unnecessary data from SelectedUsersView * load few messages on open room, search message improve * fix loading messages forever * Removed state from search * Custom message time format * secureTextEntry layout * Reduce android app size * Roles subscription fix * Public routes navigation * fix reconnect * - New login/register, login, register * proguard * Login flux * App init/restore * Android layout fixes * Multiple meteor connection requests fixed * Nested attachments * Nested attachments * fix check status * New login layout (#269) * Public routes navigation * New login/register, login, register * Multiple meteor connection requests fixed * Nested attachments * Button component * TextInput android layout fixed * Register fixed * Thinner close modal button * Requests /me after login only one time * Static images moved * fix reconnect * fix ddp * fix custom emoji * New message layout (#273) * Grouping messages * Message layout * Users typing animation * Image attachment layout
2018-04-24 19:34:03 +00:00
}