Files
speckle-server/packages/frontend-2/components/server-management/DeleteProjectDialog.vue
T
andrewwallacespeckle 671979e012 Feature - Frontend2 - Server Settings Pages (#1764)
* Add server management menu item

* Server Management Page & Settings Dialog

* escape from the precommits hooks

* Add Delete User Dialog

* Add invitations page

* Add Projects Page

* Pulling in GraphQL Data

* Settings Mutations

* Fix cardinfo type

* GraphQL in Active Users

* Add icons

* Infinite Load on Active Users

* Tidy Ups

* TS fixes

* Add Update button functionality

* Add Delete User

* Projects

* Home Done

* New query

* Delete User

* WIP User Roles

* Overflow fix

* Table Update

* Add Onclick to rows

* Fix bugs

* Fix console error

* Admin Middleware

* Role Update Dialog

* TS Fixes

* Latest updates

* Tidyups

* Tidy Up Nav Menu

* Tidy Up Card

* ChangeUserRoleDialog Tidyup

* Fix bubbling issue on Table

* Design fixes

* fix: speckle server version hydration mismatch

* Remove Dialog bg-white

* Rename Query, Fix Export

* fix: allowing FormSelectBase to reject updates

* feat: cache policies for new admin graph fields

* feat: cache updates so no refetch necessary

* PR Changes

* Fix Projects Page

* Updates to Roles

* Project Delete Mutation

* Add Resend & Delete Invite Mutation

* Fix console warnings

* Disable active user role select

* Linting fixes

* Reorder

* Linting fixes

* Changes from PR

* Fixes from PR

* Ui Fixes

* Update Settings Dialog Cache on Save

* Fix colours

* Delete Invitation Mutation Moved

* Delete Project Logic moved

* Delete Mutation moved to component

* Change User Role mutation moved to Dialog

* Reorder

* WIP - Checkbox Bug

* attempted checkbox fix

* Update Caching

* Rollback tsc

* Update to Resend Button

* Add new buttons

* Use defineModel

* Re-add emits

* Move mutation

* serverInfo cache update fix

* fixed delete invite cache update

* Updates to Project Delete Dialog Caching

* Remove unneeded props

* Fix caching in DeleteUserDialog

* Improved error handling in server info update

* Swap Invite modal to new style

* updated evictObjectFields details

* Update cache on successfull invite

* Add Invite dialog

* Add project Dialog Update

* defineModel

* Remove emits from Add Project and Invites

* Add missing Dialog

* Updates from PR

* Adjust query

* Remove need for items in admin data query

---------

Co-authored-by: Kristaps Fabians Geikins <fabis94@live.com>
2023-09-04 11:57:34 +01:00

140 lines
4.0 KiB
Vue

<template>
<LayoutDialog
v-model:open="isOpen"
max-width="sm"
title="Delete Project"
:buttons="dialogButtons"
>
<div class="flex flex-col gap-6">
<p>
Are you sure you want to
<strong>permanently delete</strong>
the selected project?
</p>
<div v-if="project">
<strong>{{ project.name }}</strong>
<p>
{{ project.models.totalCount }} models,
{{ project.versions.totalCount }} versions,
</p>
</div>
<p>
This action
<strong>cannot</strong>
be undone.
</p>
</div>
</LayoutDialog>
</template>
<script setup lang="ts">
import { useMutation } from '@vue/apollo-composable'
import { LayoutDialog } from '@speckle/ui-components'
import { ProjectItem } from '~~/lib/server-management/helpers/types'
import { adminDeleteProjectMutation } from '~~/lib/server-management/graphql/mutations'
import { useGlobalToast, ToastNotificationType } from '~~/lib/common/composables/toast'
import {
ROOT_QUERY,
convertThrowIntoFetchResult,
getCacheId,
getFirstErrorMessage,
modifyObjectFields
} from '~~/lib/common/helpers/graphql'
import { ProjectCollection } from '~~/lib/common/generated/gql/graphql'
const props = defineProps<{
open: boolean
title: string
project: ProjectItem | null
}>()
const { triggerNotification } = useGlobalToast()
const { mutate: adminDeleteMutation } = useMutation(adminDeleteProjectMutation)
const isOpen = defineModel<boolean>('open', { required: true })
const deleteConfirmed = async () => {
const projectId = props.project?.id
if (!projectId) {
return
}
const result = await adminDeleteMutation(
{
ids: [projectId]
},
{
update: (cache, { data }) => {
if (data?.streamsDelete) {
// Remove project from cache
const cacheId = getCacheId('Project', projectId)
cache.evict({
id: cacheId
})
// Modify 'admin' field of ROOT_QUERY so that we can modify all `projectList` instances
modifyObjectFields<undefined, { [key: string]: ProjectCollection }>(
cache,
ROOT_QUERY,
(_fieldName, _variables, value, details) => {
// Find all `projectList` fields (there can be multiple due to differing variables)
const projectListFields = Object.keys(value).filter(
(k) =>
details.revolveFieldNameAndVariables(k).fieldName === 'projectList'
)
// Being careful not to mutate original `value`
const newVal: typeof value = { ...value }
// Iterate over each and adjust `items` and `totalCount`
for (const field of projectListFields) {
const oldItems = value[field]?.items || []
const newItems = oldItems.filter((i) => i.__ref !== cacheId)
newVal[field] = {
...value[field],
...(value[field]?.items ? { items: newItems } : {}),
totalCount: Math.max(0, (value[field]?.totalCount || 0) - 1)
}
}
return newVal
},
{ fieldNameWhitelist: ['admin'] }
)
}
}
}
).catch(convertThrowIntoFetchResult)
if (result?.data?.streamsDelete) {
triggerNotification({
type: ToastNotificationType.Success,
title: 'Project deleted',
description: 'The project has been successfully deleted'
})
isOpen.value = false
} else {
const errorMessage = getFirstErrorMessage(result?.errors)
triggerNotification({
type: ToastNotificationType.Danger,
title: 'Failed to delete project',
description: errorMessage
})
}
}
const dialogButtons = [
{
text: 'Delete',
props: { color: 'danger', fullWidth: true },
onClick: deleteConfirmed
},
{
text: 'Cancel',
props: { color: 'secondary', fullWidth: true, outline: true },
onClick: () => (isOpen.value = false)
}
]
</script>