|
|
|
<script lang="ts" setup>
|
|
|
|
import type { SourceType, TableType } from 'nocodb-sdk'
|
|
|
|
import dayjs from 'dayjs'
|
|
|
|
import NcTooltip from '~/components/nc/Tooltip.vue'
|
|
|
|
|
|
|
|
const { activeTables } = storeToRefs(useTablesStore())
|
|
|
|
const { openTable } = useTablesStore()
|
|
|
|
const { openedProject, isDataSourceLimitReached } = storeToRefs(useBases())
|
|
|
|
|
|
|
|
const { base } = useBase()
|
|
|
|
|
|
|
|
const isNewBaseModalOpen = ref(false)
|
|
|
|
|
|
|
|
const { isUIAllowed } = useRoles()
|
|
|
|
|
|
|
|
const { $e } = useNuxtApp()
|
|
|
|
|
|
|
|
const isImportModalOpen = ref(false)
|
|
|
|
|
|
|
|
const defaultBase = computed(() => {
|
|
|
|
return openedProject.value?.sources?.[0]
|
|
|
|
})
|
|
|
|
|
|
|
|
const sources = computed(() => {
|
|
|
|
// Convert array of sources to map of sources
|
|
|
|
|
|
|
|
const baseMap = new Map<string, SourceType>()
|
|
|
|
|
|
|
|
openedProject.value?.sources?.forEach((source) => {
|
|
|
|
baseMap.set(source.id!, source)
|
|
|
|
})
|
|
|
|
|
|
|
|
return baseMap
|
|
|
|
})
|
|
|
|
|
|
|
|
function openTableCreateDialog(baseIndex?: number | undefined) {
|
|
|
|
$e('c:table:create:navdraw')
|
|
|
|
|
|
|
|
const isOpen = ref(true)
|
|
|
|
let sourceId = openedProject.value!.sources?.[0].id
|
|
|
|
if (typeof baseIndex === 'number') {
|
|
|
|
sourceId = openedProject.value!.sources?.[baseIndex].id
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!sourceId || !openedProject.value?.id) return
|
|
|
|
|
|
|
|
const { close } = useDialog(resolveComponent('DlgTableCreate'), {
|
|
|
|
'modelValue': isOpen,
|
|
|
|
sourceId, // || sources.value[0].id,
|
|
|
|
'baseId': openedProject.value.id,
|
|
|
|
'onCreate': closeDialog,
|
|
|
|
'onUpdate:modelValue': () => closeDialog(),
|
|
|
|
})
|
|
|
|
|
|
|
|
function closeDialog(table?: TableType) {
|
|
|
|
isOpen.value = false
|
|
|
|
|
|
|
|
if (!table) return
|
|
|
|
|
|
|
|
// TODO: Better way to know when the table node dom is available
|
|
|
|
setTimeout(() => {
|
|
|
|
const newTableDom = document.querySelector(`[data-table-id="${table.id}"]`)
|
|
|
|
if (!newTableDom) return
|
|
|
|
|
|
|
|
newTableDom?.scrollIntoView({ behavior: 'smooth', block: 'nearest' })
|
|
|
|
}, 1000)
|
|
|
|
|
|
|
|
close(1000)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const onCreateBaseClick = () => {
|
|
|
|
if (isDataSourceLimitReached.value) return
|
|
|
|
|
|
|
|
isNewBaseModalOpen.value = true
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
|
|
|
<div class="nc-all-tables-view">
|
|
|
|
<div
|
|
|
|
class="flex flex-row gap-x-6 pb-3 pt-6"
|
|
|
|
:class="{
|
|
|
|
'pointer-events-none': base?.isLoading,
|
|
|
|
}"
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
v-if="isUIAllowed('tableCreate')"
|
|
|
|
role="button"
|
|
|
|
class="nc-base-view-all-table-btn"
|
|
|
|
data-testid="proj-view-btn__add-new-table"
|
|
|
|
@click="openTableCreateDialog()"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="addOutlineBox" />
|
|
|
|
<div class="label">{{ $t('general.new') }} {{ $t('objects.table') }}</div>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-if="isUIAllowed('tableCreate')"
|
|
|
|
v-e="['c:table:import']"
|
|
|
|
role="button"
|
|
|
|
class="nc-base-view-all-table-btn"
|
|
|
|
data-testid="proj-view-btn__import-data"
|
|
|
|
@click="isImportModalOpen = true"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="download" />
|
|
|
|
<div class="label">{{ $t('activity.import') }} {{ $t('general.data') }}</div>
|
|
|
|
</div>
|
|
|
|
<component :is="isDataSourceLimitReached ? NcTooltip : 'div'" v-if="isUIAllowed('sourceCreate')">
|
|
|
|
<template #title>
|
|
|
|
<div>
|
|
|
|
{{ $t('tooltip.reachedSourceLimit') }}
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<div
|
|
|
|
v-e="['c:table:create-source']"
|
|
|
|
role="button"
|
|
|
|
class="nc-base-view-all-table-btn"
|
|
|
|
data-testid="proj-view-btn__create-source"
|
|
|
|
:class="{
|
|
|
|
disabled: isDataSourceLimitReached,
|
|
|
|
}"
|
|
|
|
@click="onCreateBaseClick"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="dataSource" />
|
|
|
|
<div class="label">{{ $t('labels.connectDataSource') }}</div>
|
|
|
|
</div>
|
|
|
|
</component>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-if="base?.isLoading"
|
|
|
|
class="flex items-center justify-center text-center"
|
|
|
|
:style="{
|
|
|
|
height: 'calc(100vh - var(--topbar-height) - 18rem)',
|
|
|
|
}"
|
|
|
|
>
|
|
|
|
<div>
|
|
|
|
<GeneralLoader size="xlarge" />
|
|
|
|
<div class="mt-2">
|
|
|
|
{{ $t('general.loading') }}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<template v-else-if="activeTables.length">
|
|
|
|
<div class="flex flex-row w-full text-gray-400 border-b-1 border-gray-50 py-3 px-2.5">
|
|
|
|
<div class="w-2/5">{{ $t('objects.table') }}</div>
|
|
|
|
<div class="w-1/5">{{ $t('general.source') }}</div>
|
|
|
|
<div class="w-1/5">{{ $t('labels.createdOn') }}</div>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
class="nc-base-view-all-table-list nc-scrollbar-md"
|
|
|
|
:style="{
|
|
|
|
height: 'calc(100vh - var(--topbar-height) - 18rem)',
|
|
|
|
}"
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
v-for="table in [...activeTables].sort(
|
|
|
|
(a, b) => a.source_id!.localeCompare(b.source_id!) * 20
|
|
|
|
)"
|
|
|
|
:key="table.id"
|
|
|
|
class="py-4 flex flex-row w-full cursor-pointer hover:bg-gray-100 border-b-1 border-gray-100 px-2.25"
|
|
|
|
data-testid="proj-view-list__item"
|
|
|
|
@click="openTable(table)"
|
|
|
|
>
|
|
|
|
<div class="flex flex-row w-2/5 items-center gap-x-2" data-testid="proj-view-list__item-title">
|
|
|
|
<div class="min-w-5 flex items-center justify-center">
|
|
|
|
<GeneralTableIcon :meta="table" class="text-gray-500" />
|
|
|
|
</div>
|
|
|
|
{{ table?.title }}
|
|
|
|
</div>
|
|
|
|
<div class="w-1/5 text-gray-600" data-testid="proj-view-list__item-type">
|
|
|
|
<div v-if="table.source_id === defaultBase?.id" class="ml-0.75">-</div>
|
|
|
|
<div v-else class="capitalize flex flex-row items-center gap-x-0.5">
|
|
|
|
<GeneralBaseLogo class="w-4 mr-1" />
|
|
|
|
{{ sources.get(table.source_id!)?.alias }}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="w-1/5 text-gray-400 ml-0.25" data-testid="proj-view-list__item-created-at">
|
|
|
|
{{ dayjs(table?.created_at).fromNow() }}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<div v-else class="py-3 flex items-center gap-6 <lg:flex-col">
|
|
|
|
<img src="~assets/img/placeholder/table.png" class="!w-[23rem] flex-none" />
|
|
|
|
<div class="text-center lg:text-left">
|
|
|
|
<div class="text-2xl text-gray-800 font-bold">{{ $t('placeholder.createTable') }}</div>
|
|
|
|
<div class="text-sm text-gray-700 pt-6">
|
|
|
|
{{ $t('placeholder.createTableLabel') }}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<ProjectImportModal v-if="defaultBase" v-model:visible="isImportModalOpen" :source="defaultBase" />
|
|
|
|
<LazyDashboardSettingsDataSourcesCreateBase v-model:open="isNewBaseModalOpen" />
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.nc-base-view-all-table-btn {
|
|
|
|
@apply flex flex-col gap-y-6 p-4 bg-gray-100 rounded-xl w-56 cursor-pointer text-gray-600 hover:(bg-gray-200 text-black);
|
|
|
|
|
|
|
|
.nc-icon {
|
|
|
|
@apply h-10 w-10;
|
|
|
|
}
|
|
|
|
|
|
|
|
.label {
|
|
|
|
@apply text-base font-medium;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-base-view-all-table-btn.disabled {
|
|
|
|
@apply bg-gray-50 text-gray-400 hover:(bg-gray-50 text-gray-400) cursor-not-allowed;
|
|
|
|
}
|
|
|
|
</style>
|