多维表格
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

416 lines
11 KiB

import { acceptHMRUpdate, defineStore } from 'pinia'
import type { BaseType, OracleUi, ProjectUserReqType, RequestParams, SourceType } from 'nocodb-sdk'
import { SqlUiFactory } from 'nocodb-sdk'
import { isString } from '@vue/shared'
// todo: merge with base store
export const useBases = defineStore('basesStore', () => {
const { $api } = useNuxtApp()
const { user: currentUser } = useGlobal()
const { loadRoles } = useRoles()
const { isUIAllowed } = useRoles()
const bases = ref<Map<string, NcProject>>(new Map())
const basesList = computed<NcProject[]>(() =>
Array.from(bases.value.values()).sort(
(a, b) => (a.order != null ? a.order : Infinity) - (b.order != null ? b.order : Infinity),
),
)
const basesUser = ref<Map<string, User[]>>(new Map())
const router = useRouter()
const route = router.currentRoute
const activeProjectId = computed(() => {
if (route.value.params.typeOrId === 'base') {
return basesList.value?.[0]?.id
}
return route.value.params.baseId as string | undefined
})
const openedProject = computed(() => (activeProjectId.value ? bases.value.get(activeProjectId.value) : undefined))
const openedProjectBasesMap = computed(() => {
const basesMap = new Map<string, SourceType>()
if (!openedProject.value) return basesMap
if (!openedProject.value.sources) return basesMap
for (const source of openedProject.value.sources) {
basesMap.set(source.id!, source)
}
return basesMap
})
const isDataSourceLimitReached = computed(() => Number(openedProject.value?.sources?.length) > 9)
const workspaceStore = useWorkspace()
const tableStore = useTablesStore()
const { api } = useApi()
const { getBaseUrl } = useGlobal()
const isProjectsLoading = ref(false)
async function getBaseUsers({ baseId, searchText, force = false }: { baseId: string; searchText?: string; force?: boolean }) {
if (!force && basesUser.value.has(baseId)) {
const users = basesUser.value.get(baseId)
return {
users,
totalRows: users?.length ?? 0,
}
}
const response: any = await api.auth.baseUserList(baseId, {
query: {
query: searchText,
},
} as RequestParams)
const totalRows = response.users.pageInfo.totalRows ?? 0
basesUser.value.set(baseId, response.users.list)
return {
users: response.users.list,
totalRows,
}
}
const clearBasesUser = () => {
basesUser.value.clear()
}
const createProjectUser = async (baseId: string, user: User) => {
await api.auth.baseUserAdd(baseId, user as ProjectUserReqType)
}
const updateProjectUser = async (baseId: string, user: User) => {
await api.auth.baseUserUpdate(baseId, user.id, user as ProjectUserReqType)
// reload roles if updating roles of current user
if (user.id === currentUser.value?.id) {
loadRoles(baseId).catch(() => {
// ignore
})
}
}
const removeProjectUser = async (baseId: string, user: User) => {
await api.auth.baseUserRemove(baseId, user.id)
}
const loadProjects = async (page: 'recent' | 'shared' | 'starred' | 'workspace' = 'recent') => {
// if shared base then get the shared base and create a list
if (route.value.params.typeOrId === 'base' && route.value.params.baseId) {
try {
const { base_id } = await $api.public.sharedBaseGet(route.value.params.baseId as string)
const base: BaseType = await $api.base.read(base_id)
if (!base) return
bases.value = [base].reduce((acc, base) => {
acc.set(base.id!, base)
return acc
}, new Map())
bases.value.set(base.id!, {
...(bases.value.get(base.id!) || {}),
...base,
sources: [...(base.sources ?? bases.value.get(base.id!)?.sources ?? [])],
isExpanded: route.value.params.baseId === base.id || bases.value.get(base.id!)?.isExpanded,
isLoading: false,
})
return
} catch (e: any) {
if (e?.response?.status === 404) {
return router.push('/error/404')
}
throw e
}
}
const activeWorkspace = workspaceStore.activeWorkspace
const workspace = workspaceStore.workspace
if ((!page || page === 'workspace') && !workspace?.id && !activeWorkspace?.id) {
throw new Error('Workspace not selected')
}
let _projects: BaseType[] = []
isProjectsLoading.value = true
try {
const { list } = await $api.base.list({
baseURL: getBaseUrl(activeWorkspace?.id ?? workspace?.id),
})
_projects = list
bases.value = _projects.reduce((acc, base) => {
const existingProjectMeta = bases.value.get(base.id!) || {}
acc.set(base.id!, {
...existingProjectMeta,
...base,
isExpanded: route.value.params.baseId === base.id || bases.value.get(base.id!)?.isExpanded,
isLoading: false,
})
return acc
}, new Map())
await updateIfBaseOrderIsNullOrDuplicate()
} catch (e) {
console.error(e)
message.error(e.message)
} finally {
isProjectsLoading.value = false
}
}
function isProjectEmpty(baseId: string) {
if (!isProjectPopulated(baseId)) return true
const base = bases.value.get(baseId)
if (!base) return false
return tableStore.baseTables.get(baseId)!.length === 0
return false
}
function isProjectPopulated(baseId: string) {
const base = bases.value.get(baseId)
if (!base) return false
return !!(base.sources?.length && tableStore.baseTables.get(baseId))
}
// actions
const loadProject = async (baseId: string, force = false) => {
try {
if (!force && isProjectPopulated(baseId)) return bases.value.get(baseId)
const _project = await api.base.read(baseId)
if (!_project) {
await navigateTo(`/`)
return
}
_project.meta = _project?.meta && typeof _project.meta === 'string' ? JSON.parse(_project.meta) : {}
const existingProject = bases.value.get(baseId) ?? ({} as any)
const base = {
...existingProject,
..._project,
isExpanded: route.value.params.baseId === baseId || existingProject.isExpanded,
// isLoading is managed by Sidebar
isLoading: existingProject.isLoading,
meta: { ...parseProp(existingProject.meta), ...parseProp(_project.meta) },
}
bases.value.set(baseId, base)
} catch (e: any) {
await message.error(await extractSdkResponseErrorMsg(e))
}
}
const getSqlUi = async (baseId: string, sourceId: string) => {
if (!bases.value.get(baseId)) await loadProject(baseId)
let sqlUi = null
const base = bases.value.get(baseId)!
for (const source of base.sources ?? []) {
if (source.id === sourceId) {
sqlUi = SqlUiFactory.create({ client: source.type }) as any
break
}
}
return sqlUi as Exclude<ReturnType<(typeof SqlUiFactory)['create']>, typeof OracleUi>
}
const updateProject = async (baseId: string, baseUpdatePayload: BaseType) => {
11 months ago
const existingProject = bases.value.get(baseId) ?? ({} as any)
const base = {
...existingProject,
...baseUpdatePayload,
}
bases.value.set(baseId, { ...base, meta: parseProp(base.meta) })
11 months ago
await api.base.update(baseId, baseUpdatePayload)
11 months ago
await loadProject(baseId, true)
}
const createProject = async (basePayload: {
title: string
workspaceId?: string
type: string
linkedDbProjectIds?: string[]
meta?: Record<string, unknown>
}) => {
const result = await api.base.create(
{
title: basePayload.title,
linked_db_project_ids: basePayload.linkedDbProjectIds,
meta: JSON.stringify({
...(basePayload.meta || {}),
}),
},
{
baseURL: getBaseUrl('nc'),
},
)
await loadProjects()
return result
}
const deleteProject = async (baseId: string) => {
await api.base.delete(baseId)
bases.value.delete(baseId)
tableStore.baseTables.delete(baseId)
await loadProjects()
}
const getProjectMeta = (baseId: string) => {
const base = bases.value.get(baseId)
if (!base) throw new Error('Base not found')
let meta = {
showNullAndEmptyInFilter: false,
}
try {
meta = (isString(base.meta) ? JSON.parse(base.meta) : base.meta) ?? meta
} catch {}
return meta
}
async function getProjectMetaInfo(baseId: string) {
return await api.base.metaGet(baseId!, {})
}
async function setProject(baseId: string, base: NcProject) {
bases.value.set(baseId, base)
}
async function clearBases() {
bases.value.clear()
}
const navigateToProject = async ({ baseId, page }: { baseId: string; page?: 'collaborators' }) => {
if (!baseId) return
const base = bases.value.get(baseId)
if (!base) return
if (page) {
return await navigateTo(`/nc/${baseId}?page=${page}`)
}
await navigateTo(`/nc/${baseId}`)
}
async function updateIfBaseOrderIsNullOrDuplicate() {
if (!isUIAllowed('baseReorder')) return
11 months ago
const basesArray = Array.from(bases.value.values())
const baseOrderSet = new Set()
let hasNullOrDuplicates = false
// Check if basesArray contains null or duplicate order
for (const base of basesArray) {
if (base.order === null || baseOrderSet.has(base.order)) {
hasNullOrDuplicates = true
break
}
baseOrderSet.add(base.order)
}
if (!hasNullOrDuplicates) return
// update the local state and return updated bases payload
const updatedBasesOrder = basesArray.map((base, i) => {
bases.value.set(base.id!, { ...base, order: i + 1 })
return {
id: base.id,
order: i + 1,
}
})
try {
await Promise.all(
updatedBasesOrder.map(async (base) => {
await api.base.update(base.id!, { order: base.order })
}),
)
} catch (e: any) {
message.error(await extractSdkResponseErrorMsg(e))
}
}
onMounted(() => {
if (!activeProjectId.value) return
if (isProjectPopulated(activeProjectId.value)) return
loadProject(activeProjectId.value)
})
const navigateToFirstProjectOrHome = async () => {
// if active base id is deleted, navigate to first base or home page
if (basesList.value?.length) await navigateToProject({ baseId: basesList.value[0].id! })
else navigateTo('/')
}
const toggleStarred = async (..._args: any) => {}
return {
bases,
basesList,
loadProjects,
loadProject,
getSqlUi,
updateProject,
createProject,
deleteProject,
getProjectMetaInfo,
getProjectMeta,
setProject,
clearBases,
isProjectEmpty,
isProjectPopulated,
isProjectsLoading,
activeProjectId,
openedProject,
openedProjectBasesMap,
getBaseUsers,
createProjectUser,
updateProjectUser,
navigateToProject,
removeProjectUser,
navigateToFirstProjectOrHome,
toggleStarred,
basesUser,
clearBasesUser,
isDataSourceLimitReached,
}
})
if (import.meta.hot) {
import.meta.hot.accept(acceptHMRUpdate(useBases as any, import.meta.hot))
}