|
|
|
<script lang="ts" setup>
|
|
|
|
import { nextTick } from '@vue/runtime-core'
|
|
|
|
import { message } from 'ant-design-vue'
|
|
|
|
import { stringifyRolesObj } from 'nocodb-sdk'
|
|
|
|
import type { BaseType, ProjectType, TableType } from 'nocodb-sdk'
|
|
|
|
import { LoadingOutlined } from '@ant-design/icons-vue'
|
|
|
|
import { useTitle } from '@vueuse/core'
|
|
|
|
import {
|
|
|
|
NcProjectType,
|
|
|
|
ProjectInj,
|
|
|
|
ProjectRoleInj,
|
|
|
|
ToggleDialogInj,
|
|
|
|
extractSdkResponseErrorMsg,
|
|
|
|
openLink,
|
|
|
|
storeToRefs,
|
|
|
|
useProjects,
|
|
|
|
} from '#imports'
|
|
|
|
import type { NcProject } from '#imports'
|
|
|
|
import { useNuxtApp } from '#app'
|
|
|
|
|
|
|
|
const indicator = h(LoadingOutlined, {
|
|
|
|
class: '!text-gray-400',
|
|
|
|
style: {
|
|
|
|
fontSize: '0.85rem',
|
|
|
|
},
|
|
|
|
spin: true,
|
|
|
|
})
|
|
|
|
|
|
|
|
const router = useRouter()
|
|
|
|
const route = router.currentRoute
|
|
|
|
|
|
|
|
const { isSharedBase } = storeToRefs(useProject())
|
|
|
|
const { projectUrl } = useProject()
|
|
|
|
|
|
|
|
const { setMenuContext, openRenameTableDialog, duplicateTable, contextMenuTarget } = inject(TreeViewInj)!
|
|
|
|
|
|
|
|
const project = inject(ProjectInj)!
|
|
|
|
|
|
|
|
const projectsStore = useProjects()
|
|
|
|
|
|
|
|
const { isMobileMode } = useGlobal()
|
|
|
|
|
|
|
|
const { loadProject, loadProjects, createProject: _createProject, updateProject, getProjectMetaInfo } = projectsStore
|
|
|
|
const { projects } = storeToRefs(projectsStore)
|
|
|
|
|
|
|
|
const { loadProjectTables } = useTablesStore()
|
|
|
|
const { activeTable } = storeToRefs(useTablesStore())
|
|
|
|
|
|
|
|
const { appInfo, navigateToProject } = useGlobal()
|
|
|
|
|
|
|
|
const { orgRoles, isUIAllowed } = useRoles()
|
|
|
|
|
|
|
|
useTabs()
|
|
|
|
|
|
|
|
const editMode = ref(false)
|
|
|
|
|
|
|
|
const tempTitle = ref('')
|
|
|
|
|
|
|
|
const activeBaseId = ref('')
|
|
|
|
|
|
|
|
const isErdModalOpen = ref<Boolean>(false)
|
|
|
|
|
|
|
|
const { t } = useI18n()
|
|
|
|
|
|
|
|
const input = ref<HTMLInputElement>()
|
|
|
|
|
|
|
|
const projectRole = inject(ProjectRoleInj)
|
|
|
|
|
|
|
|
const { activeProjectId } = storeToRefs(useProjects())
|
|
|
|
|
|
|
|
const toggleDialog = inject(ToggleDialogInj, () => {})
|
|
|
|
|
|
|
|
const { $e } = useNuxtApp()
|
|
|
|
|
|
|
|
const isOptionsOpen = ref(false)
|
|
|
|
const isBasesOptionsOpen = ref<Record<string, boolean>>({})
|
|
|
|
|
|
|
|
const activeKey = ref<string[]>([])
|
|
|
|
const [searchActive] = useToggle()
|
|
|
|
const filterQuery = ref('')
|
|
|
|
const keys = ref<Record<string, number>>({})
|
|
|
|
const isTableDeleteDialogVisible = ref(false)
|
|
|
|
const isProjectDeleteDialogVisible = ref(false)
|
|
|
|
|
|
|
|
// If only project is open, i.e in case of docs, project view is open and not the page view
|
|
|
|
const projectViewOpen = computed(() => {
|
|
|
|
const routeNameSplit = String(route.value?.name).split('projectId-index-index')
|
|
|
|
if (routeNameSplit.length <= 1) return false
|
|
|
|
|
|
|
|
const routeNameAfterProjectView = routeNameSplit[routeNameSplit.length - 1]
|
|
|
|
return routeNameAfterProjectView.split('-').length === 2 || routeNameAfterProjectView.split('-').length === 1
|
|
|
|
})
|
|
|
|
|
|
|
|
const showBaseOption = computed(() => {
|
|
|
|
return ['airtableImport', 'csvImport', 'jsonImport', 'excelImport'].some((permission) => isUIAllowed(permission))
|
|
|
|
})
|
|
|
|
|
|
|
|
const enableEditMode = () => {
|
|
|
|
editMode.value = true
|
|
|
|
tempTitle.value = project.value.title!
|
|
|
|
nextTick(() => {
|
|
|
|
input.value?.focus()
|
|
|
|
input.value?.select()
|
|
|
|
input.value?.scrollIntoView()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const updateProjectTitle = async () => {
|
|
|
|
if (!tempTitle.value) return
|
|
|
|
|
|
|
|
try {
|
|
|
|
await updateProject(project.value.id!, {
|
|
|
|
title: tempTitle.value,
|
|
|
|
})
|
|
|
|
editMode.value = false
|
|
|
|
tempTitle.value = ''
|
|
|
|
|
|
|
|
$e('a:project:rename')
|
|
|
|
|
|
|
|
useTitle(`${project.value?.title}`)
|
|
|
|
} catch (e: any) {
|
|
|
|
message.error(await extractSdkResponseErrorMsg(e))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const { copy } = useCopy(true)
|
|
|
|
|
|
|
|
const copyProjectInfo = async () => {
|
|
|
|
try {
|
|
|
|
if (
|
|
|
|
await copy(
|
|
|
|
Object.entries(await getProjectMetaInfo(project.value.id!)!)
|
|
|
|
.map(([k, v]) => `${k}: **${v}**`)
|
|
|
|
.join('\n'),
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
// Copied to clipboard
|
|
|
|
message.info(t('msg.info.copiedToClipboard'))
|
|
|
|
}
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error(e)
|
|
|
|
message.error(e.message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
defineExpose({
|
|
|
|
enableEditMode,
|
|
|
|
})
|
|
|
|
|
|
|
|
const setIcon = async (icon: string, project: ProjectType) => {
|
|
|
|
try {
|
|
|
|
const meta = {
|
|
|
|
...((project.meta as object) || {}),
|
|
|
|
icon,
|
|
|
|
}
|
|
|
|
|
|
|
|
projectsStore.updateProject(project.id!, { meta: JSON.stringify(meta) })
|
|
|
|
|
|
|
|
$e('a:project:icon:navdraw', { icon })
|
|
|
|
} catch (e: any) {
|
|
|
|
message.error(await extractSdkResponseErrorMsg(e))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function openTableCreateDialog(baseIndex?: number | undefined) {
|
|
|
|
const isOpen = ref(true)
|
|
|
|
let baseId = project.value!.bases?.[0].id
|
|
|
|
if (typeof baseIndex === 'number') {
|
|
|
|
baseId = project.value!.bases?.[baseIndex].id
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!baseId || !project.value?.id) return
|
|
|
|
|
|
|
|
const { close } = useDialog(resolveComponent('DlgTableCreate'), {
|
|
|
|
'modelValue': isOpen,
|
|
|
|
baseId, // || bases.value[0].id,
|
|
|
|
'projectId': project.value!.id,
|
|
|
|
'onCreate': closeDialog,
|
|
|
|
'onUpdate:modelValue': () => closeDialog(),
|
|
|
|
})
|
|
|
|
|
|
|
|
function closeDialog(table?: TableType) {
|
|
|
|
isOpen.value = false
|
|
|
|
|
|
|
|
if (!table) return
|
|
|
|
|
|
|
|
project.value.isExpanded = true
|
|
|
|
|
|
|
|
if (!activeKey.value || !activeKey.value.includes(`collapse-${baseId}`)) {
|
|
|
|
activeKey.value.push(`collapse-${baseId}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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 isAddNewProjectChildEntityLoading = ref(false)
|
|
|
|
const addNewProjectChildEntity = async () => {
|
|
|
|
if (isAddNewProjectChildEntityLoading.value) return
|
|
|
|
|
|
|
|
isAddNewProjectChildEntityLoading.value = true
|
|
|
|
|
|
|
|
const isProjectPopulated = projectsStore.isProjectPopulated(project.value.id!)
|
|
|
|
if (!isProjectPopulated && project.value.type === NcProjectType.DB) {
|
|
|
|
// We do not wait for tables api, so that add new table is seamless.
|
|
|
|
// Only con would be while saving table duplicate table name FE validation might not work
|
|
|
|
// If the table list api takes time to load before the table name validation
|
|
|
|
loadProjectTables(project.value.id!)
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
openTableCreateDialog()
|
|
|
|
|
|
|
|
if (!project.value.isExpanded && project.value.type !== NcProjectType.DB) {
|
|
|
|
project.value.isExpanded = true
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
isAddNewProjectChildEntityLoading.value = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const onProjectClick = async (project: NcProject, ignoreNavigation?: boolean, toggleIsExpanded?: boolean) => {
|
|
|
|
if (!project) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!toggleIsExpanded) $e('c:base:open')
|
|
|
|
|
|
|
|
ignoreNavigation = isMobileMode.value || ignoreNavigation
|
|
|
|
toggleIsExpanded = isMobileMode.value || toggleIsExpanded
|
|
|
|
|
|
|
|
if (toggleIsExpanded) {
|
|
|
|
project.isExpanded = !project.isExpanded
|
|
|
|
} else {
|
|
|
|
project.isExpanded = true
|
|
|
|
}
|
|
|
|
|
|
|
|
const isProjectPopulated = projectsStore.isProjectPopulated(project.id!)
|
|
|
|
|
|
|
|
if (!isProjectPopulated) project.isLoading = true
|
|
|
|
|
|
|
|
if (!ignoreNavigation) {
|
|
|
|
await navigateTo(
|
|
|
|
projectUrl({
|
|
|
|
id: project.id!,
|
|
|
|
type: 'database',
|
|
|
|
isSharedBase: isSharedBase.value,
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isProjectPopulated) {
|
|
|
|
await loadProjectTables(project.id!)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isProjectPopulated) {
|
|
|
|
const updatedProject = projects.value.get(project.id!)!
|
|
|
|
updatedProject.isLoading = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function openErdView(base: BaseType) {
|
|
|
|
activeBaseId.value = base.id
|
|
|
|
isErdModalOpen.value = !isErdModalOpen.value
|
|
|
|
}
|
|
|
|
|
|
|
|
async function openProjectErdView(_project: ProjectType) {
|
|
|
|
if (!_project.id) return
|
|
|
|
|
|
|
|
if (!projectsStore.isProjectPopulated(_project.id)) {
|
|
|
|
await loadProject(_project.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
const project = projects.value.get(_project.id)
|
|
|
|
|
|
|
|
const base = project?.bases?.[0]
|
|
|
|
if (!base) return
|
|
|
|
openErdView(base)
|
|
|
|
}
|
|
|
|
|
|
|
|
const reloadTables = async () => {
|
|
|
|
$e('a:table:refresh:navdraw')
|
|
|
|
|
|
|
|
// await loadTables()
|
|
|
|
}
|
|
|
|
|
|
|
|
const contextMenuBase = computed(() => {
|
|
|
|
if (contextMenuTarget.type === 'base') {
|
|
|
|
return contextMenuTarget.value
|
|
|
|
} else if (contextMenuTarget.type === 'table') {
|
|
|
|
const base = project.value?.bases?.find((b) => b.id === contextMenuTarget.value.base_id)
|
|
|
|
if (base) return base
|
|
|
|
}
|
|
|
|
return null
|
|
|
|
})
|
|
|
|
|
|
|
|
watch(
|
|
|
|
() => activeTable.value?.id,
|
|
|
|
async () => {
|
|
|
|
if (!activeTable.value) return
|
|
|
|
|
|
|
|
const baseId = activeTable.value.base_id
|
|
|
|
if (!baseId) return
|
|
|
|
|
|
|
|
if (!activeKey.value.includes(`collapse-${baseId}`)) {
|
|
|
|
activeKey.value.push(`collapse-${baseId}`)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
immediate: true,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
onKeyStroke('Escape', () => {
|
|
|
|
if (isOptionsOpen.value) {
|
|
|
|
isOptionsOpen.value = false
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const key of Object.keys(isBasesOptionsOpen.value)) {
|
|
|
|
isBasesOptionsOpen.value[key] = false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const isDuplicateDlgOpen = ref(false)
|
|
|
|
const selectedProjectToDuplicate = ref()
|
|
|
|
|
|
|
|
const duplicateProject = (project: ProjectType) => {
|
|
|
|
selectedProjectToDuplicate.value = project
|
|
|
|
isDuplicateDlgOpen.value = true
|
|
|
|
}
|
|
|
|
const { $poller } = useNuxtApp()
|
|
|
|
|
|
|
|
const DlgProjectDuplicateOnOk = async (jobData: { id: string; project_id: string }) => {
|
|
|
|
await loadProjects('workspace')
|
|
|
|
|
|
|
|
$poller.subscribe(
|
|
|
|
{ id: jobData.id },
|
|
|
|
async (data: {
|
|
|
|
id: string
|
|
|
|
status?: string
|
|
|
|
data?: {
|
|
|
|
error?: {
|
|
|
|
message: string
|
|
|
|
}
|
|
|
|
message?: string
|
|
|
|
result?: any
|
|
|
|
}
|
|
|
|
}) => {
|
|
|
|
if (data.status !== 'close') {
|
|
|
|
if (data.status === JobStatus.COMPLETED) {
|
|
|
|
await loadProjects('workspace')
|
|
|
|
|
|
|
|
const project = projects.value.get(jobData.project_id)
|
|
|
|
|
|
|
|
// open project after duplication
|
|
|
|
if (project) {
|
|
|
|
await navigateToProject({
|
|
|
|
projectId: project.id,
|
|
|
|
type: project.type,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else if (data.status === JobStatus.FAILED) {
|
|
|
|
message.error('Failed to duplicate project')
|
|
|
|
await loadProjects('workspace')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)
|
|
|
|
$e('a:project:duplicate')
|
|
|
|
}
|
|
|
|
|
|
|
|
const tableDelete = () => {
|
|
|
|
isTableDeleteDialogVisible.value = true
|
|
|
|
$e('c:table:delete')
|
|
|
|
}
|
|
|
|
|
|
|
|
const projectDelete = () => {
|
|
|
|
isProjectDeleteDialogVisible.value = true
|
|
|
|
$e('c:project:delete')
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
|
|
|
<NcDropdown :trigger="['contextmenu']" overlay-class-name="nc-dropdown-tree-view-context-menu">
|
|
|
|
<div
|
|
|
|
class="mx-1 nc-project-sub-menu rounded-md"
|
|
|
|
:class="{ active: project.isExpanded }"
|
|
|
|
:data-testid="`nc-sidebar-project-${project.title}`"
|
|
|
|
:data-project-id="project.id"
|
|
|
|
>
|
|
|
|
<div class="flex items-center gap-0.75 py-0.25 cursor-pointer" @contextmenu="setMenuContext('project', project)">
|
|
|
|
<div
|
|
|
|
ref="projectNodeRefs"
|
|
|
|
:class="{
|
|
|
|
'bg-primary-selected active': activeProjectId === project.id && projectViewOpen && !isMobileMode,
|
|
|
|
'hover:bg-gray-200': !(activeProjectId === project.id && projectViewOpen),
|
|
|
|
}"
|
|
|
|
:data-testid="`nc-sidebar-project-title-${project.title}`"
|
|
|
|
class="nc-sidebar-node project-title-node h-7.25 flex-grow rounded-md group flex items-center w-full pr-1"
|
|
|
|
>
|
|
|
|
<NcButton
|
|
|
|
v-e="['c:base:expand']"
|
|
|
|
type="text"
|
|
|
|
size="xxsmall"
|
|
|
|
class="nc-sidebar-node-btn nc-sidebar-expand ml-0.75 !xs:visible"
|
|
|
|
@click="onProjectClick(project, true, true)"
|
|
|
|
>
|
|
|
|
<GeneralIcon
|
|
|
|
icon="triangleFill"
|
|
|
|
class="group-hover:visible cursor-pointer transform transition-transform duration-500 h-1.5 w-1.75 rotate-90 !xs:visible"
|
|
|
|
:class="{ '!rotate-180': project.isExpanded, '!visible': isOptionsOpen }"
|
|
|
|
/>
|
|
|
|
</NcButton>
|
|
|
|
|
|
|
|
<div class="flex items-center mr-1" @click="onProjectClick(project)">
|
|
|
|
<div class="flex items-center select-none w-6 h-full">
|
|
|
|
<a-spin
|
|
|
|
v-if="project.isLoading"
|
|
|
|
class="!ml-1.25 !flex !flex-row !items-center !my-0.5 w-8"
|
|
|
|
:indicator="indicator"
|
|
|
|
/>
|
|
|
|
|
|
|
|
<LazyGeneralEmojiPicker
|
|
|
|
v-else
|
|
|
|
:key="project.meta?.icon"
|
|
|
|
v-e="['c:base:emojiSelect']"
|
|
|
|
:emoji="project.meta?.icon"
|
|
|
|
:readonly="true"
|
|
|
|
size="small"
|
|
|
|
@emoji-selected="setIcon($event, project)"
|
|
|
|
>
|
|
|
|
<template #default>
|
|
|
|
<GeneralProjectIcon :type="project.type" />
|
|
|
|
</template>
|
|
|
|
</LazyGeneralEmojiPicker>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<input
|
|
|
|
v-if="editMode"
|
|
|
|
ref="input"
|
|
|
|
v-model="tempTitle"
|
|
|
|
class="flex-grow leading-1 outline-0 ring-none capitalize !text-inherit !bg-transparent w-4/5"
|
|
|
|
:class="{ 'text-black font-semibold': activeProjectId === project.id && projectViewOpen && !isMobileMode }"
|
|
|
|
@click.stop
|
|
|
|
@keyup.enter="updateProjectTitle"
|
|
|
|
@keyup.esc="updateProjectTitle"
|
|
|
|
@blur="updateProjectTitle"
|
|
|
|
/>
|
|
|
|
<span
|
|
|
|
v-else
|
|
|
|
class="nc-sidebar-node-title capitalize text-ellipsis overflow-hidden select-none"
|
|
|
|
:style="{ wordBreak: 'keep-all', whiteSpace: 'nowrap', display: 'inline' }"
|
|
|
|
:class="{ 'text-black font-semibold': activeProjectId === project.id && projectViewOpen }"
|
|
|
|
@click="onProjectClick(project)"
|
|
|
|
>
|
|
|
|
{{ project.title }}
|
|
|
|
</span>
|
|
|
|
<div :class="{ 'flex flex-grow h-full': !editMode }" @click="onProjectClick(project)"></div>
|
|
|
|
|
|
|
|
<NcDropdown v-model:visible="isOptionsOpen" :trigger="['click']">
|
|
|
|
<NcButton
|
|
|
|
v-e="['c:base:options']"
|
|
|
|
class="nc-sidebar-node-btn"
|
|
|
|
:class="{ '!text-black !opacity-100': isOptionsOpen }"
|
|
|
|
data-testid="nc-sidebar-context-menu"
|
|
|
|
type="text"
|
|
|
|
size="xxsmall"
|
|
|
|
@click.stop
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="threeDotHorizontal" class="text-xl w-4.75" />
|
|
|
|
</NcButton>
|
|
|
|
<template #overlay>
|
|
|
|
<NcMenu
|
|
|
|
class="nc-scrollbar-md"
|
|
|
|
:style="{
|
|
|
|
maxHeight: '70vh',
|
|
|
|
overflow: 'overlay',
|
|
|
|
}"
|
|
|
|
:data-testid="`nc-sidebar-project-${project.title}-options`"
|
|
|
|
@click="isOptionsOpen = false"
|
|
|
|
>
|
|
|
|
<template v-if="!isSharedBase">
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('projectRename')"
|
|
|
|
v-e="['c:base:rename']"
|
|
|
|
data-testid="nc-sidebar-project-rename"
|
|
|
|
@click="enableEditMode"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="edit" class="group-hover:text-black" />
|
|
|
|
{{ $t('general.rename') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('projectDuplicate', { roles: [stringifyRolesObj(orgRoles), projectRole].join() })"
|
|
|
|
v-e="['c:base:duplicate']"
|
|
|
|
data-testid="nc-sidebar-project-duplicate"
|
|
|
|
@click="duplicateProject(project)"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="duplicate" class="text-gray-700" />
|
|
|
|
{{ $t('general.duplicate') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<NcDivider v-if="['projectDuplicate', 'projectRename'].some((permission) => isUIAllowed(permission))" />
|
|
|
|
|
|
|
|
<!-- Copy Project Info -->
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="!isEeUI"
|
|
|
|
key="copy"
|
|
|
|
v-e="['c:base:copy-proj-info']"
|
|
|
|
data-testid="nc-sidebar-project-copy-project-info"
|
|
|
|
@click.stop="copyProjectInfo"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="copy" class="group-hover:text-black" />
|
|
|
|
{{ $t('activity.account.projInfo') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<!-- ERD View -->
|
|
|
|
<NcMenuItem
|
|
|
|
key="erd"
|
|
|
|
v-e="['c:base:erd']"
|
|
|
|
data-testid="nc-sidebar-project-relations"
|
|
|
|
@click="openProjectErdView(project)"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="erd" />
|
|
|
|
{{ $t('title.relations') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<!-- Swagger: Rest APIs -->
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('apiDocs')"
|
|
|
|
key="api"
|
|
|
|
v-e="['c:base:api-docs']"
|
|
|
|
data-testid="nc-sidebar-project-rest-apis"
|
|
|
|
@click.stop="
|
|
|
|
() => {
|
|
|
|
$e('c:base:api-docs')
|
|
|
|
openLink(`/api/v1/db/meta/projects/${project.id}/swagger`, appInfo.ncSiteUrl)
|
|
|
|
}
|
|
|
|
"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="snippet" class="group-hover:text-black !max-w-3.9" />
|
|
|
|
{{ $t('activity.account.swagger') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<template v-if="project.bases && project.bases[0] && showBaseOption">
|
|
|
|
<NcDivider />
|
|
|
|
<DashboardTreeViewBaseOptions v-model:project="project" :base="project.bases[0]" />
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<NcDivider v-if="['projectMiscSettings', 'projectDelete'].some((permission) => isUIAllowed(permission))" />
|
|
|
|
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('projectMiscSettings')"
|
|
|
|
key="teamAndSettings"
|
|
|
|
v-e="['c:base:settings']"
|
|
|
|
data-testid="nc-sidebar-project-settings"
|
|
|
|
class="nc-sidebar-project-project-settings"
|
|
|
|
@click="toggleDialog(true, 'teamAndAuth', undefined, project.id)"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="settings" class="group-hover:text-black" />
|
|
|
|
{{ $t('activity.settings') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('projectDelete', { roles: [stringifyRolesObj(orgRoles), projectRole].join() })"
|
|
|
|
data-testid="nc-sidebar-project-delete"
|
|
|
|
class="!text-red-500 !hover:bg-red-50"
|
|
|
|
@click="projectDelete"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="delete" class="w-4" />
|
|
|
|
{{ $t('general.delete') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
</NcMenu>
|
|
|
|
</template>
|
|
|
|
</NcDropdown>
|
|
|
|
|
|
|
|
<NcButton
|
|
|
|
v-if="isUIAllowed('tableCreate', { roles: projectRole })"
|
|
|
|
v-e="['c:base:create-table']"
|
|
|
|
class="nc-sidebar-node-btn"
|
|
|
|
size="xxsmall"
|
|
|
|
type="text"
|
|
|
|
data-testid="nc-sidebar-add-project-entity"
|
|
|
|
:class="{ '!text-black !visible': isAddNewProjectChildEntityLoading, '!visible': isOptionsOpen }"
|
|
|
|
:loading="isAddNewProjectChildEntityLoading"
|
|
|
|
@click.stop="addNewProjectChildEntity"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="plus" class="text-xl leading-5" style="-webkit-text-stroke: 0.15px" />
|
|
|
|
</NcButton>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div
|
|
|
|
v-if="project.id && !project.isLoading"
|
|
|
|
key="g1"
|
|
|
|
class="overflow-x-hidden transition-max-height"
|
|
|
|
:class="{ 'max-h-0': !project.isExpanded }"
|
|
|
|
>
|
|
|
|
<template v-if="project && project?.bases">
|
|
|
|
<div class="flex-1 overflow-y-auto overflow-x-hidden flex flex-col" :class="{ 'mb-[20px]': isSharedBase }">
|
|
|
|
<div v-if="project?.bases?.[0]?.enabled" class="flex-1">
|
|
|
|
<div class="transition-height duration-200">
|
|
|
|
<DashboardTreeViewTableList :project="project" :base-index="0" />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div v-if="project?.bases?.slice(1).filter((el) => el.enabled)?.length" class="transition-height duration-200">
|
|
|
|
<div class="border-none sortable-list">
|
|
|
|
<div v-for="(base, baseIndex) of project.bases" :key="`base-${base.id}`">
|
|
|
|
<template v-if="baseIndex === 0"></template>
|
|
|
|
<a-collapse
|
|
|
|
v-else-if="base && base.enabled"
|
|
|
|
v-model:activeKey="activeKey"
|
|
|
|
v-e="['c:source:toggle-expand']"
|
|
|
|
class="!mx-0 !px-0 nc-sidebar-base-node"
|
|
|
|
:class="[{ hidden: searchActive && !!filterQuery }]"
|
|
|
|
expand-icon-position="left"
|
|
|
|
:bordered="false"
|
|
|
|
ghost
|
|
|
|
>
|
|
|
|
<template #expandIcon="{ isActive }">
|
|
|
|
<div
|
|
|
|
class="nc-sidebar-expand nc-sidebar-node-btn flex flex-row items-center -mt-2 xs:(mt-3 border-1 border-gray-200 px-2.25 py-0.5 rounded-md !mr-0.25)"
|
|
|
|
>
|
|
|
|
<GeneralIcon
|
|
|
|
icon="triangleFill"
|
|
|
|
class="nc-sidebar-base-node-btns -mt-0.75 invisible xs:visible cursor-pointer transform transition-transform duration-500 h-1.5 w-1.5 text-gray-500 rotate-90"
|
|
|
|
:class="{ '!rotate-180': isActive }"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<a-collapse-panel :key="`collapse-${base.id}`">
|
|
|
|
<template #header>
|
|
|
|
<div class="nc-sidebar-node min-w-20 w-full flex flex-row group py-0.25">
|
|
|
|
<div
|
|
|
|
v-if="baseIndex === 0"
|
|
|
|
class="base-context flex items-center gap-2 text-gray-800 nc-sidebar-node-title"
|
|
|
|
@contextmenu="setMenuContext('base', base)"
|
|
|
|
>
|
|
|
|
<GeneralBaseLogo :base-type="base.type" class="min-w-4 !xs:(min-w-4.25 w-4.25 text-sm)" />
|
|
|
|
{{ $t('general.default') }}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-else
|
|
|
|
class="base-context flex flex-grow items-center gap-1.75 text-gray-800 min-w-1/20 max-w-full"
|
|
|
|
@contextmenu="setMenuContext('base', base)"
|
|
|
|
>
|
|
|
|
<GeneralBaseLogo :base-type="base.type" class="min-w-4 !xs:(min-w-4.25 w-4.25 text-sm)" />
|
|
|
|
<div
|
|
|
|
:data-testid="`nc-sidebar-project-${base.alias}`"
|
|
|
|
class="nc-sidebar-node-title flex capitalize text-ellipsis overflow-hidden select-none"
|
|
|
|
:style="{ wordBreak: 'keep-all', whiteSpace: 'nowrap', display: 'inline' }"
|
|
|
|
>
|
|
|
|
{{ base.alias || '' }}
|
|
|
|
</div>
|
|
|
|
<a-tooltip class="xs:(hidden)">
|
|
|
|
<template #title>{{ $t('objects.externalDb') }}</template>
|
|
|
|
<div>
|
|
|
|
<GeneralIcon icon="info" class="text-gray-400 -mt-0.5 hover:text-gray-700 mr-1" />
|
|
|
|
</div>
|
|
|
|
</a-tooltip>
|
|
|
|
</div>
|
|
|
|
<div class="flex flex-row items-center gap-x-0.25 w-12.25">
|
|
|
|
<NcDropdown
|
|
|
|
:visible="isBasesOptionsOpen[base!.id!]"
|
|
|
|
:trigger="['click']"
|
|
|
|
@update:visible="isBasesOptionsOpen[base!.id!] = $event"
|
|
|
|
>
|
|
|
|
<NcButton
|
|
|
|
v-e="['c:source:options']"
|
|
|
|
class="nc-sidebar-node-btn"
|
|
|
|
:class="{ '!text-black !opacity-100': isBasesOptionsOpen[base!.id!] }"
|
|
|
|
type="text"
|
|
|
|
size="xxsmall"
|
|
|
|
@click.stop="isBasesOptionsOpen[base!.id!] = !isBasesOptionsOpen[base!.id!]"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="threeDotHorizontal" class="text-xl w-4.75" />
|
|
|
|
</NcButton>
|
|
|
|
<template #overlay>
|
|
|
|
<NcMenu
|
|
|
|
class="nc-scrollbar-md"
|
|
|
|
:style="{
|
|
|
|
maxHeight: '70vh',
|
|
|
|
overflow: 'overlay',
|
|
|
|
}"
|
|
|
|
@click="isBasesOptionsOpen[base!.id!] = false"
|
|
|
|
>
|
|
|
|
<!-- ERD View -->
|
|
|
|
<NcMenuItem key="erd" v-e="['c:source:erd']" @click="openErdView(base)">
|
|
|
|
<GeneralIcon icon="erd" />
|
|
|
|
{{ $t('title.relations') }}
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<DashboardTreeViewBaseOptions v-if="showBaseOption" v-model:project="project" :base="base" />
|
|
|
|
</NcMenu>
|
|
|
|
</template>
|
|
|
|
</NcDropdown>
|
|
|
|
|
|
|
|
<NcButton
|
|
|
|
v-if="isUIAllowed('tableCreate', { roles: projectRole })"
|
|
|
|
v-e="['c:source:add-table']"
|
|
|
|
type="text"
|
|
|
|
size="xxsmall"
|
|
|
|
class="nc-sidebar-node-btn"
|
|
|
|
@click.stop="openTableCreateDialog(baseIndex)"
|
|
|
|
>
|
|
|
|
<GeneralIcon icon="plus" class="text-xl leading-5" style="-webkit-text-stroke: 0.15px" />
|
|
|
|
</NcButton>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<div
|
|
|
|
ref="menuRefs"
|
|
|
|
:key="`sortable-${base.id}-${base.id && base.id in keys ? keys[base.id] : '0'}`"
|
|
|
|
:nc-base="base.id"
|
|
|
|
>
|
|
|
|
<DashboardTreeViewTableList :project="project" :base-index="baseIndex" />
|
|
|
|
</div>
|
|
|
|
</a-collapse-panel>
|
|
|
|
</a-collapse>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<template v-if="!isSharedBase" #overlay>
|
|
|
|
<NcMenu class="!py-0 rounded text-sm">
|
|
|
|
<template v-if="contextMenuTarget.type === 'project' && project.type === 'database'"></template>
|
|
|
|
|
|
|
|
<template v-else-if="contextMenuTarget.type === 'base'"></template>
|
|
|
|
|
|
|
|
<template v-else-if="contextMenuTarget.type === 'table'">
|
|
|
|
v-e="['c:table:rename']"
|
|
|
|
<NcMenuItem v-if="isUIAllowed('tableRename')" @click="openRenameTableDialog(contextMenuTarget.value, true)">
|
|
|
|
<div class="nc-project-option-item">
|
|
|
|
<GeneralIcon icon="edit" class="text-gray-700" />
|
|
|
|
{{ $t('general.rename') }}
|
|
|
|
</div>
|
|
|
|
</NcMenuItem>
|
|
|
|
|
|
|
|
<NcMenuItem
|
|
|
|
v-if="isUIAllowed('tableDuplicate') && (contextMenuBase?.is_meta || contextMenuBase?.is_local)"
|
|
|
|
v-e="['c:table:duplicate']"
|
|
|
|
@click="duplicateTable(contextMenuTarget.value)"
|
|
|
|
>
|
|
|
|
<div class="nc-project-option-item">
|
|
|
|
<GeneralIcon icon="duplicate" class="text-gray-700" />
|
|
|
|
{{ $t('general.duplicate') }}
|
|
|
|
</div>
|
|
|
|
</NcMenuItem>
|
|
|
|
<NcDivider />
|
|
|
|
<NcMenuItem v-if="isUIAllowed('table-delete')" class="!hover:bg-red-50" @click="tableDelete">
|
|
|
|
<div class="nc-project-option-item text-red-600">
|
|
|
|
<GeneralIcon icon="delete" />
|
|
|
|
{{ $t('general.delete') }}
|
|
|
|
</div>
|
|
|
|
</NcMenuItem>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<template v-else>
|
|
|
|
<NcMenuItem @click="reloadTables">
|
|
|
|
<div class="nc-project-option-item">
|
|
|
|
{{ $t('general.reload') }}
|
|
|
|
</div>
|
|
|
|
</NcMenuItem>
|
|
|
|
</template>
|
|
|
|
</NcMenu>
|
|
|
|
</template>
|
|
|
|
</NcDropdown>
|
|
|
|
<DlgTableDelete
|
|
|
|
v-if="contextMenuTarget.value?.id && project?.id"
|
|
|
|
v-model:visible="isTableDeleteDialogVisible"
|
|
|
|
:table-id="contextMenuTarget.value?.id"
|
|
|
|
:project-id="project?.id"
|
|
|
|
/>
|
|
|
|
<DlgProjectDelete v-model:visible="isProjectDeleteDialogVisible" :project-id="project?.id" />
|
|
|
|
<DlgProjectDuplicate
|
|
|
|
v-if="selectedProjectToDuplicate"
|
|
|
|
v-model="isDuplicateDlgOpen"
|
|
|
|
:project="selectedProjectToDuplicate"
|
|
|
|
:on-ok="DlgProjectDuplicateOnOk"
|
|
|
|
/>
|
|
|
|
<GeneralModal v-model:visible="isErdModalOpen" size="large">
|
|
|
|
<div class="h-[80vh]">
|
|
|
|
<LazyDashboardSettingsErd :base-id="activeBaseId" />
|
|
|
|
</div>
|
|
|
|
</GeneralModal>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
:deep(.ant-collapse-header) {
|
|
|
|
@apply !mx-0 !pl-8.75 !xs:(pl-8) !pr-0.5 !py-0.5 hover:bg-gray-200 xs:(hover:bg-gray-50 ) !rounded-md;
|
|
|
|
}
|
|
|
|
|
|
|
|
:deep(.ant-collapse-item) {
|
|
|
|
@apply h-full;
|
|
|
|
}
|
|
|
|
|
|
|
|
:deep(.ant-collapse-content-box) {
|
|
|
|
@apply !px-0 !pb-0 !pt-0.25;
|
|
|
|
}
|
|
|
|
|
|
|
|
:deep(.ant-collapse-header:hover .nc-sidebar-base-node-btns) {
|
|
|
|
@apply visible;
|
|
|
|
}
|
|
|
|
</style>
|