|
|
@ -1,17 +1,17 @@ |
|
|
|
<script lang="ts" setup> |
|
|
|
<script lang="ts" setup> |
|
|
|
import { Empty, Modal } from 'ant-design-vue' |
|
|
|
import { Empty, Modal } from 'ant-design-vue' |
|
|
|
import type { ProjectType } from 'nocodb-sdk' |
|
|
|
import type { ProjectType } from 'nocodb-sdk' |
|
|
|
import tinycolor from 'tinycolor2' |
|
|
|
|
|
|
|
import { |
|
|
|
import { |
|
|
|
computed, |
|
|
|
computed, |
|
|
|
definePageMeta, |
|
|
|
definePageMeta, |
|
|
|
extractSdkResponseErrorMsg, |
|
|
|
extractSdkResponseErrorMsg, |
|
|
|
message, |
|
|
|
message, |
|
|
|
navigateTo, |
|
|
|
navigateTo, |
|
|
|
|
|
|
|
onBeforeMount, |
|
|
|
projectThemeColors, |
|
|
|
projectThemeColors, |
|
|
|
ref, |
|
|
|
ref, |
|
|
|
themeV2Colors, |
|
|
|
themeV2Colors, |
|
|
|
useLazyAsyncData, |
|
|
|
useApi, |
|
|
|
useNuxtApp, |
|
|
|
useNuxtApp, |
|
|
|
useSidebar, |
|
|
|
useSidebar, |
|
|
|
useUIPermission, |
|
|
|
useUIPermission, |
|
|
@ -23,15 +23,7 @@ definePageMeta({ |
|
|
|
|
|
|
|
|
|
|
|
const { $api, $e } = useNuxtApp() |
|
|
|
const { $api, $e } = useNuxtApp() |
|
|
|
|
|
|
|
|
|
|
|
const { |
|
|
|
const { api, isLoading } = useApi() |
|
|
|
data: projects, |
|
|
|
|
|
|
|
pending: isLoading, |
|
|
|
|
|
|
|
execute, |
|
|
|
|
|
|
|
} = useLazyAsyncData(async () => { |
|
|
|
|
|
|
|
const response = await $api.project.list({}) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return response.list |
|
|
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const { isUIAllowed } = useUIPermission() |
|
|
|
const { isUIAllowed } = useUIPermission() |
|
|
|
|
|
|
|
|
|
|
@ -39,6 +31,13 @@ useSidebar('nc-left-sidebar', { hasSidebar: false, isOpen: true }) |
|
|
|
|
|
|
|
|
|
|
|
const filterQuery = ref('') |
|
|
|
const filterQuery = ref('') |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const projects = ref<ProjectType[]>() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const loadProjects = async () => { |
|
|
|
|
|
|
|
const response = await api.project.list({}) |
|
|
|
|
|
|
|
projects.value = response.list |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const filteredProjects = computed( |
|
|
|
const filteredProjects = computed( |
|
|
|
() => |
|
|
|
() => |
|
|
|
projects.value?.filter( |
|
|
|
projects.value?.filter( |
|
|
@ -57,7 +56,7 @@ const deleteProject = (project: ProjectType) => { |
|
|
|
cancelText: 'No', |
|
|
|
cancelText: 'No', |
|
|
|
async onOk() { |
|
|
|
async onOk() { |
|
|
|
try { |
|
|
|
try { |
|
|
|
await $api.project.delete(project.id as string) |
|
|
|
await api.project.delete(project.id as string) |
|
|
|
|
|
|
|
|
|
|
|
$e('a:project:delete') |
|
|
|
$e('a:project:delete') |
|
|
|
|
|
|
|
|
|
|
@ -69,10 +68,9 @@ const deleteProject = (project: ProjectType) => { |
|
|
|
}) |
|
|
|
}) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
await execute() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const handleProjectColor = async (projectId: string, color: string) => { |
|
|
|
const handleProjectColor = async (projectId: string, color: string) => { |
|
|
|
const tcolor = tinycolor(color) |
|
|
|
const tinyColor = await import('tinycolor2').then((d) => d.default) |
|
|
|
|
|
|
|
const tcolor = tinyColor(color) |
|
|
|
|
|
|
|
|
|
|
|
if (tcolor.isValid()) { |
|
|
|
if (tcolor.isValid()) { |
|
|
|
const complement = tcolor.complement() |
|
|
|
const complement = tcolor.complement() |
|
|
@ -97,6 +95,7 @@ const handleProjectColor = async (projectId: string, color: string) => { |
|
|
|
|
|
|
|
|
|
|
|
if (localProject) { |
|
|
|
if (localProject) { |
|
|
|
localProject.color = color |
|
|
|
localProject.color = color |
|
|
|
|
|
|
|
|
|
|
|
localProject.meta = JSON.stringify({ |
|
|
|
localProject.meta = JSON.stringify({ |
|
|
|
...meta, |
|
|
|
...meta, |
|
|
|
theme: { |
|
|
|
theme: { |
|
|
@ -124,6 +123,8 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
}, |
|
|
|
}, |
|
|
|
class: ['group'], |
|
|
|
class: ['group'], |
|
|
|
}) |
|
|
|
}) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
onBeforeMount(loadProjects) |
|
|
|
</script> |
|
|
|
</script> |
|
|
|
|
|
|
|
|
|
|
|
<template> |
|
|
|
<template> |
|
|
@ -143,7 +144,7 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
v-e="['a:project:refresh']" |
|
|
|
v-e="['a:project:refresh']" |
|
|
|
class="text-xl text-gray-500 group-hover:text-accent cursor-pointer" |
|
|
|
class="text-xl text-gray-500 group-hover:text-accent cursor-pointer" |
|
|
|
:class="isLoading ? '!text-primary' : ''" |
|
|
|
:class="isLoading ? '!text-primary' : ''" |
|
|
|
@click="execute" |
|
|
|
@click="loadProjects" |
|
|
|
/> |
|
|
|
/> |
|
|
|
</span> |
|
|
|
</span> |
|
|
|
</a-tooltip> |
|
|
|
</a-tooltip> |
|
|
@ -196,18 +197,12 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
</a-dropdown> |
|
|
|
</a-dropdown> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
|
|
|
|
|
|
|
|
<TransitionGroup name="layout" mode="out-in"> |
|
|
|
<Transition name="layout" mode="out-in"> |
|
|
|
<div v-if="isLoading" key="skeleton"> |
|
|
|
<div v-if="isLoading"> |
|
|
|
<a-skeleton /> |
|
|
|
<a-skeleton /> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
|
|
|
|
|
|
|
|
<a-table |
|
|
|
<a-table v-else :custom-row="customRow" :data-source="filteredProjects" :pagination="{ position: ['bottomCenter'] }"> |
|
|
|
v-else |
|
|
|
|
|
|
|
key="table" |
|
|
|
|
|
|
|
:custom-row="customRow" |
|
|
|
|
|
|
|
:data-source="filteredProjects" |
|
|
|
|
|
|
|
:pagination="{ position: ['bottomCenter'] }" |
|
|
|
|
|
|
|
> |
|
|
|
|
|
|
|
<template #emptyText> |
|
|
|
<template #emptyText> |
|
|
|
<a-empty :image="Empty.PRESENTED_IMAGE_SIMPLE" :description="$t('labels.noData')" /> |
|
|
|
<a-empty :image="Empty.PRESENTED_IMAGE_SIMPLE" :description="$t('labels.noData')" /> |
|
|
|
</template> |
|
|
|
</template> |
|
|
@ -233,12 +228,13 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
|
|
|
|
|
|
|
|
<template #expandIcon></template> |
|
|
|
<template #expandIcon></template> |
|
|
|
|
|
|
|
|
|
|
|
<GeneralColorPicker |
|
|
|
<LazyGeneralColorPicker |
|
|
|
:colors="projectThemeColors" |
|
|
|
:colors="projectThemeColors" |
|
|
|
:row-size="9" |
|
|
|
:row-size="9" |
|
|
|
:advanced="false" |
|
|
|
:advanced="false" |
|
|
|
@input="handleProjectColor(record.id, $event)" |
|
|
|
@input="handleProjectColor(record.id, $event)" |
|
|
|
/> |
|
|
|
/> |
|
|
|
|
|
|
|
|
|
|
|
<a-sub-menu key="pick-primary"> |
|
|
|
<a-sub-menu key="pick-primary"> |
|
|
|
<template #title> |
|
|
|
<template #title> |
|
|
|
<div class="nc-project-menu-item group !py-0"> |
|
|
|
<div class="nc-project-menu-item group !py-0"> |
|
|
@ -247,7 +243,8 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
<template #expandIcon></template> |
|
|
|
<template #expandIcon></template> |
|
|
|
<GeneralChromeWrapper @input="handleProjectColor(record.id, $event)" /> |
|
|
|
|
|
|
|
|
|
|
|
<LazyGeneralChromeWrapper @input="handleProjectColor(record.id, $event)" /> |
|
|
|
</a-sub-menu> |
|
|
|
</a-sub-menu> |
|
|
|
</a-sub-menu> |
|
|
|
</a-sub-menu> |
|
|
|
</template> |
|
|
|
</template> |
|
|
@ -273,7 +270,7 @@ const customRow = (record: ProjectType) => ({ |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
</a-table-column> |
|
|
|
</a-table-column> |
|
|
|
</a-table> |
|
|
|
</a-table> |
|
|
|
</TransitionGroup> |
|
|
|
</Transition> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
|
|
|
|
|
|
|
|