6fb9849007
* Allow to add/remove from groups on market page * remove lib * Fix Sinclair's relative import from May * Clean
173 lines
4.7 KiB
TypeScript
173 lines
4.7 KiB
TypeScript
import {
|
|
deleteDoc,
|
|
doc,
|
|
getDocs,
|
|
query,
|
|
updateDoc,
|
|
where,
|
|
} from 'firebase/firestore'
|
|
import { sortBy, uniq } from 'lodash'
|
|
import { Group, GROUP_CHAT_SLUG } from 'common/group'
|
|
import { updateContract } from './contracts'
|
|
import {
|
|
coll,
|
|
getValue,
|
|
getValues,
|
|
listenForValue,
|
|
listenForValues,
|
|
} from './utils'
|
|
import { Contract } from 'common/contract'
|
|
|
|
export const groups = coll<Group>('groups')
|
|
|
|
export function groupPath(
|
|
groupSlug: string,
|
|
subpath?:
|
|
| 'edit'
|
|
| 'markets'
|
|
| 'about'
|
|
| typeof GROUP_CHAT_SLUG
|
|
| 'leaderboards'
|
|
) {
|
|
return `/group/${groupSlug}${subpath ? `/${subpath}` : ''}`
|
|
}
|
|
|
|
export function updateGroup(group: Group, updates: Partial<Group>) {
|
|
return updateDoc(doc(groups, group.id), updates)
|
|
}
|
|
|
|
export function deleteGroup(group: Group) {
|
|
return deleteDoc(doc(groups, group.id))
|
|
}
|
|
|
|
export async function listAllGroups() {
|
|
return getValues<Group>(groups)
|
|
}
|
|
|
|
export async function listGroups(groupSlugs: string[]) {
|
|
return Promise.all(groupSlugs.map(getGroupBySlug))
|
|
}
|
|
|
|
export function listenForGroups(setGroups: (groups: Group[]) => void) {
|
|
return listenForValues(groups, setGroups)
|
|
}
|
|
|
|
export function getGroup(groupId: string) {
|
|
return getValue<Group>(doc(groups, groupId))
|
|
}
|
|
|
|
export async function getGroupBySlug(slug: string) {
|
|
const q = query(groups, where('slug', '==', slug))
|
|
const docs = (await getDocs(q)).docs
|
|
return docs.length === 0 ? null : docs[0].data()
|
|
}
|
|
|
|
export function listenForGroup(
|
|
groupId: string,
|
|
setGroup: (group: Group | null) => void
|
|
) {
|
|
return listenForValue(doc(groups, groupId), setGroup)
|
|
}
|
|
|
|
export function listenForMemberGroups(
|
|
userId: string,
|
|
setGroups: (groups: Group[]) => void,
|
|
sort?: { by: 'mostRecentChatActivityTime' | 'mostRecentContractAddedTime' }
|
|
) {
|
|
const q = query(groups, where('memberIds', 'array-contains', userId))
|
|
const sorter = (group: Group) => {
|
|
if (sort?.by === 'mostRecentChatActivityTime') {
|
|
return group.mostRecentChatActivityTime ?? group.createdTime
|
|
}
|
|
if (sort?.by === 'mostRecentContractAddedTime') {
|
|
return group.mostRecentContractAddedTime ?? group.createdTime
|
|
}
|
|
return group.mostRecentActivityTime
|
|
}
|
|
return listenForValues<Group>(q, (groups) => {
|
|
const sorted = sortBy(groups, [(group) => -sorter(group)])
|
|
setGroups(sorted)
|
|
})
|
|
}
|
|
|
|
export async function listenForGroupsWithContractId(
|
|
contractId: string,
|
|
setGroups: (groups: Group[]) => void
|
|
) {
|
|
const q = query(groups, where('contractIds', 'array-contains', contractId))
|
|
return listenForValues<Group>(q, setGroups)
|
|
}
|
|
|
|
export async function addUserToGroupViaId(groupId: string, userId: string) {
|
|
// get group to get the member ids
|
|
const group = await getGroup(groupId)
|
|
if (!group) {
|
|
console.error(`Group not found: ${groupId}`)
|
|
return
|
|
}
|
|
return await joinGroup(group, userId)
|
|
}
|
|
|
|
export async function joinGroup(group: Group, userId: string): Promise<void> {
|
|
const { memberIds } = group
|
|
if (memberIds.includes(userId)) return // already a member
|
|
|
|
const newMemberIds = [...memberIds, userId]
|
|
return await updateGroup(group, { memberIds: uniq(newMemberIds) })
|
|
}
|
|
|
|
export async function leaveGroup(group: Group, userId: string): Promise<void> {
|
|
const { memberIds } = group
|
|
if (!memberIds.includes(userId)) return // not a member
|
|
|
|
const newMemberIds = memberIds.filter((id) => id !== userId)
|
|
return await updateGroup(group, { memberIds: uniq(newMemberIds) })
|
|
}
|
|
|
|
export async function addContractToGroup(group: Group, contract: Contract) {
|
|
await updateContract(contract.id, {
|
|
groupSlugs: [...(contract.groupSlugs ?? []), group.slug],
|
|
})
|
|
return await updateGroup(group, {
|
|
contractIds: uniq([...group.contractIds, contract.id]),
|
|
})
|
|
.then(() => group)
|
|
.catch((err) => {
|
|
console.error('error adding contract to group', err)
|
|
return err
|
|
})
|
|
}
|
|
|
|
export async function removeContractFromGroup(
|
|
group: Group,
|
|
contract: Contract
|
|
) {
|
|
const newGroupSlugs = contract.groupSlugs?.filter(
|
|
(slug) => slug !== group.slug
|
|
)
|
|
await updateContract(contract.id, {
|
|
groupSlugs: uniq(newGroupSlugs ?? []),
|
|
})
|
|
const newContractIds = group.contractIds.filter((id) => id !== contract.id)
|
|
return await updateGroup(group, {
|
|
contractIds: uniq(newContractIds),
|
|
})
|
|
.then(() => group)
|
|
.catch((err) => {
|
|
console.error('error removing contract from group', err)
|
|
return err
|
|
})
|
|
}
|
|
|
|
export async function setContractGroupSlugs(group: Group, contractId: string) {
|
|
await updateContract(contractId, { groupSlugs: [group.slug] })
|
|
return await updateGroup(group, {
|
|
contractIds: uniq([...group.contractIds, contractId]),
|
|
})
|
|
.then(() => group)
|
|
.catch((err) => {
|
|
console.error('error adding contract to group', err)
|
|
return err
|
|
})
|
|
}
|