fix: improve BaseAssignment modal UI and UX

This commit is contained in:
DarkPhoenix2704
2026-03-31 13:02:59 +00:00
parent 654d5e254f
commit 9ccaa0c39e
2 changed files with 101 additions and 52 deletions

View File

@@ -25,18 +25,42 @@ const { activeWorkspaceId } = storeToRefs(workspaceStore)
const basesStore = useBases()
const { basesList } = storeToRefs(basesStore)
const isLoading = ref(false)
const isLoading = ref(true)
const isSaving = ref(false)
const allBases = ref(true)
const selectedBaseIds = ref<Set<string>>(new Set())
const initialAllBases = ref(true)
const initialBaseIds = ref<Set<string>>(new Set())
const hasChanges = computed(() => {
if (allBases.value !== initialAllBases.value) return true
if (allBases.value) return false
if (selectedBaseIds.value.size !== initialBaseIds.value.size) return true
for (const id of selectedBaseIds.value) {
if (!initialBaseIds.value.has(id)) return true
}
return false
})
const listRef = ref<HTMLDivElement>()
const hasScrollableContent = ref(false)
function checkScrollable() {
if (!listRef.value) return
const { scrollTop, scrollHeight, clientHeight } = listRef.value
hasScrollableContent.value = scrollTop + clientHeight < scrollHeight - 4
}
const isOpen = computed({
get: () => visible.value,
set: (val) => emits('update:visible', val),
})
async function loadCurrentState() {
if (!activeWorkspaceId.value || !integration.value?.id) return
if (!activeWorkspaceId.value || !integration.value?.id) {
isLoading.value = false
return
}
try {
isLoading.value = true
@@ -52,6 +76,9 @@ async function loadCurrentState() {
allBases.value = false
selectedBaseIds.value = new Set((result?.bases || []).map((b: any) => b.id))
}
initialAllBases.value = allBases.value
initialBaseIds.value = new Set(selectedBaseIds.value)
} catch (e: any) {
message.error(await extractSdkResponseErrorMsg(e))
} finally {
@@ -65,9 +92,7 @@ async function save() {
try {
isSaving.value = true
const payload = allBases.value
? { all_bases: true }
: { base_ids: Array.from(selectedBaseIds.value) }
const payload = allBases.value ? { all_bases: true } : { base_ids: Array.from(selectedBaseIds.value) }
await $api.internal.postOperation(
activeWorkspaceId.value,
@@ -95,53 +120,70 @@ function toggleBase(baseId: string) {
selectedBaseIds.value = new Set(selectedBaseIds.value)
}
watch(visible, (val) => {
if (val) {
loadCurrentState()
}
watch(
visible,
(val) => {
if (val) {
loadCurrentState()
}
},
{ immediate: true },
)
watch(allBases, () => {
nextTick(checkScrollable)
})
</script>
<template>
<NcModal v-model:visible="isOpen" size="sm" :body-style="{ padding: 0 }">
<div class="flex flex-col">
<!-- Header -->
<div class="flex items-center justify-between p-4 border-b-1 border-nc-border-gray-medium">
<h3 class="text-sm font-semibold text-nc-content-gray m-0">
{{ t('labels.manageBaseAccess') }}
</h3>
<NcButton size="xs" type="text" @click="isOpen = false">
<GeneralIcon icon="close" />
</NcButton>
<NcModal v-model:visible="isOpen" size="sm" wrap-class-name="nc-modal-base-assignment">
<template #header>
<span class="text-heading3">
{{ t('labels.manageBaseAccess') }}
</span>
</template>
<div v-if="isLoading" class="flex items-center justify-center py-8">
<GeneralLoader />
</div>
<div v-else class="flex flex-col flex-1 min-h-0 overflow-hidden">
<div
class="flex items-center justify-between p-3 rounded-lg border-1 cursor-pointer"
:class="allBases ? 'border-nc-border-brand bg-nc-bg-brand-soft' : 'border-nc-border-gray-medium'"
@click="allBases = !allBases"
>
<div class="flex flex-col gap-1">
<span class="text-sm font-semibold text-nc-content-gray">{{ t('activity.allBases') }}</span>
<span class="text-bodySm text-nc-content-gray-subtle">{{ t('labels.grantAccessToAllBases') }}</span>
</div>
<span @click.stop>
<NcSwitch v-model:checked="allBases" size="small" :disabled="isLoading" />
</span>
</div>
<div v-if="isLoading" class="flex items-center justify-center py-12">
<GeneralLoader />
</div>
<template v-else>
<!-- Toggle -->
<div class="flex flex-col gap-3 p-4">
<div class="flex items-center gap-2">
<NcSwitch v-model:checked="allBases" size="small" />
<span class="text-sm text-nc-content-gray">{{ t('activity.allBases') }}</span>
</div>
<!-- Base list (when specific bases selected) -->
<div v-if="!allBases" class="flex flex-col gap-1 max-h-64 overflow-auto nc-scrollbar-thin">
<template v-if="!allBases">
<div class="flex items-center justify-between mt-4 mb-2">
<span class="text-captionSm text-nc-content-gray-subtle2 uppercase tracking-wide">
{{ t('labels.selectBases') }}
</span>
<span class="text-bodySm text-nc-content-gray-subtle">
{{ selectedBaseIds.size }} / {{ basesList.length }}
</span>
</div>
<div class="relative flex-1 min-h-0">
<div
ref="listRef"
class="flex flex-col gap-1 h-full overflow-auto nc-scrollbar-thin"
@scroll="checkScrollable"
>
<div
v-for="base in basesList"
:key="base.id"
class="flex items-center gap-2 p-2 rounded-md hover:bg-nc-bg-gray-light cursor-pointer"
class="flex items-center gap-2.5 p-2 rounded-lg hover:bg-nc-bg-gray-light cursor-pointer"
@click="toggleBase(base.id!)"
>
<NcCheckbox :checked="selectedBaseIds.has(base.id!)" />
<GeneralProjectIcon
:color="parseProp(base.meta).iconColor"
:type="base.type"
class="h-4 w-4 flex-none"
/>
<NcTooltip show-on-truncate-only class="truncate text-sm text-nc-content-gray">
<GeneralProjectIcon :color="parseProp(base.meta).iconColor" :type="base.type" class="h-4.5 w-4.5 flex-none" />
<NcTooltip show-on-truncate-only class="truncate text-sm font-medium text-nc-content-gray">
{{ base.title }}
</NcTooltip>
</div>
@@ -149,18 +191,23 @@ watch(visible, (val) => {
{{ t('labels.noData') }}
</div>
</div>
</div>
<!-- Footer -->
<div class="flex items-center justify-end gap-2 p-4 border-t-1 border-nc-border-gray-medium">
<NcButton size="small" type="secondary" @click="isOpen = false">
{{ $t('general.cancel') }}
</NcButton>
<NcButton size="small" type="primary" :loading="isSaving" @click="save">
{{ $t('general.save') }}
</NcButton>
<div
v-if="hasScrollableContent"
class="absolute bottom-0 left-0 right-0 h-5 pointer-events-none"
style="background: linear-gradient(transparent, var(--nc-bg-default))"
/>
</div>
</template>
<!-- Footer -->
<div class="flex items-center justify-end gap-2 mt-auto pt-4">
<NcButton size="small" type="secondary" @click="isOpen = false">
{{ $t('general.cancel') }}
</NcButton>
<NcButton size="small" type="primary" :loading="isSaving" :disabled="!hasChanges" @click="save">
{{ $t('general.save') }}
</NcButton>
</div>
</div>
</NcModal>
</template>