databag/app/mobile/src/context/useConversationContext.hook.js

331 lines
10 KiB
JavaScript
Raw Normal View History

2022-09-28 22:09:10 +00:00
import { useState, useEffect, useRef, useContext } from 'react';
import { StoreContext } from 'context/StoreContext';
import { CardContext } from 'context/CardContext';
import { ChannelContext } from 'context/ChannelContext';
import { ProfileContext } from 'context/ProfileContext';
export function useConversationContext() {
const [state, setState] = useState({
subject: null,
logo: null,
2022-10-04 21:00:49 +00:00
revision: null,
2022-09-28 22:09:10 +00:00
contacts: [],
2022-09-30 05:34:31 +00:00
topics: new Map(),
2022-09-28 22:09:10 +00:00
});
const store = useContext(StoreContext);
const card = useContext(CardContext);
const channel = useContext(ChannelContext);
const profile = useContext(ProfileContext);
2022-09-30 05:34:31 +00:00
const topics = useRef(null);
2022-09-28 22:09:10 +00:00
const revision = useRef(0);
2022-10-06 22:34:29 +00:00
const force = useRef(false);
2022-09-29 18:31:55 +00:00
const detailRevision = useRef(0);
2022-09-28 22:09:10 +00:00
const syncing = useRef(false);
2022-09-29 18:31:55 +00:00
const conversationId = useRef(null);
2022-09-30 05:34:31 +00:00
const reset = useRef(false);
2022-09-28 22:09:10 +00:00
const setView = useRef(0);
const updateState = (value) => {
setState((s) => ({ ...s, ...value }))
}
2022-09-29 18:31:55 +00:00
const getTopicItems = async (cardId, channelId) => {
if (cardId) {
return await card.actions.getChannelTopicItems(cardId, channelId);
}
return await channel.actions.getTopicItems(channelId);
}
2022-09-30 05:34:31 +00:00
const setTopicItem = async (cardId, channelId, topic) => {
2022-09-29 18:31:55 +00:00
if (cardId) {
2022-09-30 05:34:31 +00:00
return await card.actions.setChannelTopicItem(cardId, channelId, topic);
2022-09-29 18:31:55 +00:00
}
2022-09-30 05:34:31 +00:00
return await channel.actions.setTopicItem(channelId, topic);
2022-09-29 18:31:55 +00:00
}
const clearTopicItem = async (cardId, channelId, topicId) => {
if (cardId) {
return await card.actions.clearChannelTopicItem(cardId, channelId, topicId);
}
return await channel.actions.clearTopicItem(channelId, topicId);
}
const getTopic = async (cardId, channelId, topicId) => {
2022-09-29 22:21:18 +00:00
if (cardId) {
return await card.actions.getChannelTopic(cardId, channelId, topicId);
}
return await channel.actions.getTopic(channelId, topicId);
2022-09-29 18:31:55 +00:00
}
2022-09-29 22:21:18 +00:00
const getTopics = async (cardId, channelId, revision) => {
if (cardId) {
return await card.actions.getChannelTopics(cardId, channelId, revision);
}
return await channel.actions.getTopics(channelId, revision)
2022-09-29 18:31:55 +00:00
}
2022-09-29 22:21:18 +00:00
const getTopicAssetUrl = (cardId, channelId, assetId) => {
if (cardId) {
return card.actions.getChannelTopicAssetUrl(cardId, channelId, topicId, assetId);
}
return channel.actions.getTopicAssetUrl(channelId, assetId);
2022-09-29 18:31:55 +00:00
}
const addTopic = async (cardId, channelId, message, asssets) => {
2022-09-29 22:21:18 +00:00
if (cardId) {
return await card.actions.addChannelTopic(cardId, channelId, message, assetId);
}
return await channel.actions.addTopic(channelId, message, assetId);
2022-09-29 18:31:55 +00:00
}
const setTopicSubject = async (cardId, channelId, topicId, data) => {
2022-09-29 22:21:18 +00:00
if (cardId) {
return await card.actions.setChannelTopicSubject(cardId, channelId, topicId, data);
}
return await channel.actions.setTopicSubject(channelId, topicId, data);
2022-09-29 18:31:55 +00:00
}
2022-09-29 22:21:18 +00:00
const remove = async (cardId, channelId) => {
if (cardId) {
return await card.actions.removeChannel(cardId, channelId);
}
return await channel.actions.remove(channelId);
2022-09-29 18:31:55 +00:00
}
2022-09-29 22:21:18 +00:00
const removeTopic = async (cardId, channelId, topicId) => {
if (cardId) {
return await card.actions.removeChannelTopic(cardId, channelId, topicId);
}
return await channel.actions.remvoeTopic(channelId, topicId);
2022-09-29 18:31:55 +00:00
}
2022-09-30 05:34:31 +00:00
const setSyncRevision = async (cardId, channelId, revision) => {
if (cardId) {
return await card.actions.setSyncRevision(cardId, channelId, revision);
}
return await channel.actions.setSyncRevision(channelId, revision);
}
2022-09-29 18:31:55 +00:00
2022-09-28 22:09:10 +00:00
const sync = async () => {
const curView = setView.current;
2022-09-30 05:34:31 +00:00
if (!syncing.current) {
if (reset.current) {
revision.current = null;
detailRevision.current = null;
topics.current = null;
reset.current = false;
}
if (conversationId.current) {
const { cardId, channelId } = conversationId.current;
const channelItem = getChannel(cardId, channelId);
2022-10-06 22:34:29 +00:00
if (channelItem && (channelItem.revision !== revision.current || force.current)) {
2022-09-30 05:34:31 +00:00
syncing.current = true;
2022-09-29 18:31:55 +00:00
2022-09-30 05:34:31 +00:00
try {
// set channel details
if (detailRevision.current != channelItem.detailRevision) {
if (curView === setView.current) {
setChannel(channelItem);
detailRevision.current = channelItem.detailRevision;
}
2022-09-29 23:00:48 +00:00
}
2022-09-28 22:09:10 +00:00
2022-09-30 05:34:31 +00:00
// initial load from store
if (!topics.current) {
topics.current = new Map();
const items = await getTopicItems(cardId, channelId);
items.forEach(item => {
topics.current.set(item.topicId, item);
});
2022-09-29 23:00:48 +00:00
}
2022-09-30 05:34:31 +00:00
// sync from server
2022-10-06 22:34:29 +00:00
if (channelItem.topicRevision != channelItem.syncRevision || force.current) {
force.current = false;
2022-09-30 05:34:31 +00:00
const res = await getTopics(cardId, channelId, channelItem.syncRevision)
for (const topic of res.topics) {
if (!topic.data) {
topics.current.delete(topic.id);
await clearTopicItem(cardId, channelId, topic.id);
}
2022-10-07 05:52:28 +00:00
else {
const cached = topics.current.get(topic.id);
if (!cached || cached.detailRevision != topic.data.detailRevision) {
if (!topic.data.topicDetail) {
const updated = await getTopic(cardId, channelId, topic.id);
topic.data = updated.data;
}
if (!topic.data) {
topics.current.delete(topic.id);
await clearTopicItem(cardId, channelId, topic.id);
}
else {
await setTopicItem(cardId, channelId, topic);
const { id, revision, data } = topic;
topics.current.set(id, { topicId: id, revision: revision, detailRevision: topic.data.detailRevision, detail: topic.data.topicDetail });
}
2022-09-30 05:34:31 +00:00
}
}
}
2022-10-06 22:34:29 +00:00
await setSyncRevision(cardId, channelId, res.revision);
2022-09-29 23:00:48 +00:00
}
2022-09-30 05:34:31 +00:00
// update revision
revision.current = channelItem.revision;
if (curView == setView.current) {
2022-10-04 21:00:49 +00:00
if (cardId) {
card.actions.setChannelReadRevision(cardId, channelId, revision.current);
}
else {
channel.actions.setReadRevision(channelId, revision.current);
}
2022-09-30 05:34:31 +00:00
updateState({ topics: topics.current });
2022-09-29 23:00:48 +00:00
}
2022-09-30 05:34:31 +00:00
syncing.current = false;
sync();
2022-09-29 18:31:55 +00:00
}
2022-09-30 05:34:31 +00:00
catch(err) {
console.log(err);
syncing.current = false;
//TODO set to unsynced state
2022-09-29 18:31:55 +00:00
}
}
2022-09-28 22:09:10 +00:00
}
}
}
const getCard = (guid) => {
let contact = null
card.state.cards.forEach((card, cardId, map) => {
if (card?.profile?.guid === guid) {
contact = card;
}
});
return contact;
}
const getChannel = (cardId, channelId) => {
if (cardId) {
const entry = card.state.cards.get(cardId);
return entry?.channels.get(channelId);
}
return channel.state.channels.get(channelId);
}
const setChannel = (item) => {
let contacts = [];
let logo = null;
let subject = null;
if (!item) {
updateState({ contacts, logo, subject });
return;
}
if (item.cardId) {
contacts.push(card.state.cards.get(item.cardId));
}
if (item?.detail?.members) {
const profileGuid = profile.state.profile.guid;
item.detail.members.forEach(guid => {
if (profileGuid !== guid) {
const contact = getCard(guid);
contacts.push(contact);
}
})
}
if (contacts.length === 0) {
logo = 'solution';
}
else if (contacts.length === 1) {
if (contacts[0]?.profile?.imageSet) {
logo = card.actions.getCardLogo(contacts[0].cardId, contacts[0].profileRevision);
}
else {
logo = 'avatar';
}
}
else {
logo = 'appstore';
}
if (item?.detail?.data) {
try {
subject = JSON.parse(item?.detail?.data).subject;
}
catch (err) {
console.log(err);
}
}
if (!subject) {
if (contacts.length) {
let names = [];
for (let contact of contacts) {
if (contact?.profile?.name) {
names.push(contact.profile.name);
}
else if (contact?.profile?.handle) {
names.push(contact?.profile?.handle);
}
}
subject = names.join(', ');
}
else {
subject = "Notes";
}
}
updateState({ subject, logo, contacts });
}
useEffect(() => {
sync();
}, [card, channel]);
const actions = {
2022-10-04 18:52:22 +00:00
setChannel: (selected) => {
if (selected == null) {
2022-09-29 18:31:55 +00:00
setView.current++;
conversationId.current = null;
2022-09-30 05:34:31 +00:00
reset.current = true;
updateState({ subject: null, logo: null, contacts: [], topics: new Map() });
2022-09-29 18:31:55 +00:00
}
2022-10-04 18:52:22 +00:00
else if (selected.cardId !== conversationId.current?.cardId || selected.channelId !== conversationId.current?.channelId) {
2022-09-28 22:09:10 +00:00
setView.current++;
2022-10-04 18:52:22 +00:00
conversationId.current = selected;
2022-09-30 05:34:31 +00:00
reset.current = true;
updateState({ subject: null, logo: null, contacts: [], topics: new Map() });
2022-09-28 22:09:10 +00:00
sync();
2022-10-04 18:52:22 +00:00
const { cardId, channelId, revision } = selected;
if (cardId) {
card.actions.setChannelReadRevision(cardId, channelId, revision);
}
else {
channel.actions.setReadRevision(channelId, revision);
}
2022-09-28 22:09:10 +00:00
}
},
2022-10-05 07:09:57 +00:00
getTopicAssetUrl: (topicId, assetId) => {
if (conversationId.current) {
const { cardId, channelId } = conversationId.current;
if (cardId) {
return card.actions.getChannelTopicAssetUrl(cardId, channelId, topicId, assetId);
}
else {
return channel.actions.getTopicAssetUrl(channelId, topicId, assetId);
}
}
return null;
},
2022-10-06 22:34:29 +00:00
addTopic: async (message, files) => {
if (conversationId.current) {
const { cardId, channelId } = conversationId.current;
if (cardId) {
await card.actions.addChannelTopic(cardId, channelId, message, files);
}
else {
await channel.actions.addTopic(channelId, message, files);
}
force.current = true;
sync();
}
},
2022-09-28 22:09:10 +00:00
}
return { state, actions }
}