import type { BaseType, OracleUi, SourceType, TableType } from 'nocodb-sdk' import { SqlUiFactory } from 'nocodb-sdk' import { isString } from '@vue/shared' import { acceptHMRUpdate, defineStore } from 'pinia' export const useBase = defineStore('baseStore', () => { const { $e } = useNuxtApp() const { api, isLoading } = useApi() const router = useRouter() const route = router.currentRoute const { setTheme, theme } = useTheme() const { loadRoles } = useRoles() const { refreshCommandPalette } = useCommandPalette() const forcedProjectId = ref() const baseId = computed(() => forcedProjectId.value || (route.value.params.baseId as string)) const basesStore = useBases() const tablesStore = useTablesStore() const idUserMap = computed(() => { return (basesStore.basesUser.get(baseId.value) || []).reduce((acc, user) => { acc[user.id] = user acc[user.email] = user return acc }, {} as Record) }) // todo: refactor const sharedProject = ref() const openedProject = computed(() => basesStore.bases.get(baseId.value)) // todo: new-layout const base = computed(() => basesStore.bases.get(baseId.value) || sharedProject.value || {}) const tables = computed(() => tablesStore.baseTables.get(baseId.value) || []) const baseLoadedHook = createEventHook() const sources = computed(() => base.value?.sources || []) const baseMetaInfo = ref() const lastOpenedViewMap = ref>({}) // todo: refactor path param name and variable name const baseType = computed(() => route.value.params.typeOrId as string) const baseMeta = computed>(() => { const defaultMeta = { showNullAndEmptyInFilter: false, } try { return (isString(base.value.meta) ? JSON.parse(base.value.meta) : base.value.meta) ?? defaultMeta } catch (e) { return defaultMeta } }) const sqlUis = computed(() => { const temp: Record = {} for (const source of sources.value) { if (source.id) { temp[source.id] = SqlUiFactory.create({ client: source.type }) as Exclude< ReturnType<(typeof SqlUiFactory)['create']>, typeof OracleUi > } } return temp }) function getBaseType(sourceId?: string) { return sources.value.find((source) => source.id === sourceId)?.type || ClientType.MYSQL } function isMysql(sourceId?: string) { return ['mysql', ClientType.MYSQL].includes(getBaseType(sourceId)) } function isSqlite(sourceId?: string) { return getBaseType(sourceId) === ClientType.SQLITE } function isMssql(sourceId?: string) { return getBaseType(sourceId) === 'mssql' } function isPg(sourceId?: string) { return getBaseType(sourceId) === 'pg' } function isSnowflake(sourceId?: string) { return getBaseType(sourceId) === 'snowflake' } function isDatabricks(sourceId?: string) { return getBaseType(sourceId) === 'databricks' } function isXcdbBase(sourceId?: string) { const source = sources.value.find((source) => source.id === sourceId) return (source?.is_meta as boolean) || (source?.is_local as boolean) || false } const isSharedBase = computed(() => baseType.value === 'base') const isSharedErd = computed(() => baseType.value === 'ERD') async function loadProjectMetaInfo(force?: boolean) { if (!baseMetaInfo.value || force) { baseMetaInfo.value = await api.base.metaGet(base.value.id!, {}) } } // todo: add force parameter async function loadTables() { if (base.value.id) { await tablesStore.loadProjectTables(base.value.id, true) // tables.value = basesStore.baseTableList[base.value.id] // await api.dbTable.list(base.value.id, { // includeM2M: includeM2M.value, // }) // if (tablesResponse.list) { // tables.value = tablesResponse.list // } } } async function loadProject(_withTheme = true, forcedId?: string) { if (forcedId) forcedProjectId.value = forcedId if (baseType.value === 'base') { try { const baseData = await api.public.sharedBaseGet(route.value.params.baseId as string) forcedProjectId.value = baseData.base_id sharedProject.value = await api.base.read(baseData.base_id!) } catch (e: any) { if (e?.response?.status === 404) { return router.push('/error/404') } throw e } } else if (baseId.value) { await basesStore.loadProject(baseId.value) // base.value = basesStore.bases[baseId.value] // await api.base.read(baseId.value) } else { console.warn('Base id not found') return } if (isSharedBase.value) { await loadRoles(base.value.id || baseId.value, { isSharedBase: isSharedBase.value, sharedBaseId: route.value.params.baseId as string, }) } else if (isSharedErd.value) { await loadRoles(base.value.id || baseId.value, { isSharedErd: isSharedErd.value, sharedErdId: route.value.params.erdUuid as string, }) } else { await loadRoles(base.value.id || baseId.value) } await loadTables() await basesStore.getBaseUsers({ baseId: base.value.id || baseId.value, }) // if (withTheme) setTheme(baseMeta.value?.theme) return baseLoadedHook.trigger(base.value) } async function updateProject(data: Partial) { if (baseType.value === 'base') { return } if (data.meta && typeof data.meta === 'string') { await api.base.update(baseId.value, data) } else { await api.base.update(baseId.value, { ...data, meta: stringifyProp(data.meta) }) } refreshCommandPalette() } async function saveTheme(_theme: Partial) { const fullTheme = { primaryColor: theme.value.primaryColor, accentColor: theme.value.accentColor, ..._theme, } await updateProject({ color: fullTheme.primaryColor, meta: { ...baseMeta.value, theme: fullTheme, }, }) // setTheme(fullTheme) $e('c:themes:change') } async function hasEmptyOrNullFilters() { return await api.base.hasEmptyOrNullFilters(baseId.value) } const reset = () => { // base.value = {} // tables.value = [] baseMetaInfo.value = undefined setTheme() } const setProject = (baseVal: BaseType) => { sharedProject.value = baseVal } const baseUrl = ({ id, type: _type, isSharedBase }: { id: string; type: 'database'; isSharedBase?: boolean }) => { if (isSharedBase) { const typeOrId = route.value.params.typeOrId as string const baseId = route.value.params.baseId as string return `/${typeOrId}/${baseId}` } return `/nc/${id}` } watch( () => route.value.params.baseType, (n) => { if (!n) reset() }, { immediate: true }, ) watch( () => openedProject.value?.id, () => { if (!openedProject.value) return if (openedProject.value.isExpanded) return openedProject.value.isExpanded = true }, ) const navigateToProjectPage = async ({ page }: { page: 'all-table' | 'collaborator' | 'data-source' }) => { await router.push({ name: 'index-typeOrId-baseId-index-index', params: { typeOrId: route.value.params.typeOrId, baseId: route.value.params.baseId, }, query: { page, }, }) } return { base, sources, tables, loadRoles, loadProject, updateProject, loadTables, isMysql, isMssql, isPg, isSqlite, isSnowflake, isDatabricks, sqlUis, isSharedBase, isSharedErd, loadProjectMetaInfo, baseMetaInfo, baseMeta, saveTheme, baseLoadedHook: baseLoadedHook.on, reset, isLoading, lastOpenedViewMap, isXcdbBase, hasEmptyOrNullFilters, setProject, baseUrl, getBaseType, navigateToProjectPage, idUserMap, } }) if (import.meta.hot) { import.meta.hot.accept(acceptHMRUpdate(useBase as any, import.meta.hot)) }