182 lines
5.7 KiB
Vue
182 lines
5.7 KiB
Vue
<template>
|
|
<section>
|
|
<div class="md:max-w-xl md:mx-auto pb-6 md:pb-0">
|
|
<SettingsSectionHeader title="General" text="Manage your workspace settings" />
|
|
<SettingsSectionHeader title="Workspace details" subheading />
|
|
<div class="grid md:grid-cols-2 pt-4">
|
|
<div class="flex items-center justify-center">
|
|
<SettingsWorkspacesGeneralEditAvatar
|
|
v-if="workspaceResult?.workspace"
|
|
:workspace="workspaceResult?.workspace"
|
|
:disabled="!isAdmin"
|
|
size="xxl"
|
|
/>
|
|
</div>
|
|
<div class="pt-6 md:pt-0">
|
|
<FormTextInput
|
|
v-model="name"
|
|
color="foundation"
|
|
label="Name"
|
|
name="name"
|
|
placeholder="Workspace name"
|
|
show-label
|
|
:disabled="!isAdmin"
|
|
:rules="[isRequired, isStringOfLength({ maxLength: 512 })]"
|
|
validate-on-value-update
|
|
@change="save()"
|
|
/>
|
|
<hr class="mt-4 mb-2" />
|
|
<FormTextInput
|
|
v-model="description"
|
|
color="foundation"
|
|
label="Description"
|
|
name="description"
|
|
placeholder="Workspace description"
|
|
show-label
|
|
:disabled="!isAdmin"
|
|
:rules="[isStringOfLength({ maxLength: 512 })]"
|
|
@change="save()"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<hr class="my-6 md:my-8" />
|
|
<div class="flex flex-col space-y-6">
|
|
<SettingsSectionHeader title="Leave workspace" subheading />
|
|
<CommonCard class="bg-foundation">
|
|
By clicking the button below you will leave this workspace.
|
|
</CommonCard>
|
|
<div>
|
|
<FormButton color="danger" @click="showLeaveDialog = true">
|
|
Leave workspace
|
|
</FormButton>
|
|
</div>
|
|
</div>
|
|
<template v-if="isAdmin">
|
|
<hr class="my-6 md:my-8" />
|
|
<div class="flex flex-col space-y-6">
|
|
<SettingsSectionHeader title="Delete workspace" subheading />
|
|
<CommonCard class="bg-foundation">
|
|
We will delete all content of this workspace, and any associated data. We
|
|
will ask you to type in your workspace name and press the delete button.
|
|
</CommonCard>
|
|
<div>
|
|
<FormButton color="danger" @click="showDeleteDialog = true">
|
|
Delete workspace
|
|
</FormButton>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<SettingsWorkspacesGeneralLeaveDialog
|
|
v-if="workspaceResult"
|
|
v-model:open="showLeaveDialog"
|
|
:workspace="workspaceResult.workspace"
|
|
/>
|
|
|
|
<SettingsWorkspacesGeneralDeleteDialog
|
|
v-if="workspaceResult && isAdmin"
|
|
v-model:open="showDeleteDialog"
|
|
:workspace="workspaceResult.workspace"
|
|
/>
|
|
</section>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { Roles } from '@speckle/shared'
|
|
import { graphql } from '~~/lib/common/generated/gql'
|
|
import { useForm } from 'vee-validate'
|
|
import { useQuery, useMutation } from '@vue/apollo-composable'
|
|
import { settingsUpdateWorkspaceMutation } from '~/lib/settings/graphql/mutations'
|
|
import { settingsWorkspaceGeneralQuery } from '~/lib/settings/graphql/queries'
|
|
import type { WorkspaceUpdateInput } from '~~/lib/common/generated/gql/graphql'
|
|
import { ToastNotificationType, useGlobalToast } from '~~/lib/common/composables/toast'
|
|
import {
|
|
getFirstErrorMessage,
|
|
convertThrowIntoFetchResult
|
|
} from '~~/lib/common/helpers/graphql'
|
|
import { isRequired, isStringOfLength } from '~~/lib/common/helpers/validation'
|
|
import { useMixpanel } from '~/lib/core/composables/mp'
|
|
|
|
graphql(`
|
|
fragment SettingsWorkspacesGeneral_Workspace on Workspace {
|
|
...SettingsWorkspacesGeneralEditAvatar_Workspace
|
|
...SettingsWorkspaceGeneralDeleteDialog_Workspace
|
|
id
|
|
name
|
|
description
|
|
logo
|
|
role
|
|
}
|
|
`)
|
|
|
|
type FormValues = { name: string; description: string }
|
|
|
|
const props = defineProps<{
|
|
workspaceId: string
|
|
}>()
|
|
|
|
const mixpanel = useMixpanel()
|
|
const { handleSubmit } = useForm<FormValues>()
|
|
const { triggerNotification } = useGlobalToast()
|
|
const { mutate: updateMutation } = useMutation(settingsUpdateWorkspaceMutation)
|
|
const { result: workspaceResult } = useQuery(settingsWorkspaceGeneralQuery, () => ({
|
|
id: props.workspaceId
|
|
}))
|
|
|
|
const name = ref('')
|
|
const description = ref('')
|
|
const showDeleteDialog = ref(false)
|
|
const showLeaveDialog = ref(false)
|
|
|
|
const isAdmin = computed(
|
|
() => workspaceResult.value?.workspace?.role === Roles.Workspace.Admin
|
|
)
|
|
|
|
const save = handleSubmit(async () => {
|
|
if (!workspaceResult.value?.workspace) return
|
|
|
|
const input: WorkspaceUpdateInput = {
|
|
id: workspaceResult.value.workspace.id
|
|
}
|
|
if (name.value !== workspaceResult.value.workspace.name) input.name = name.value
|
|
if (description.value !== workspaceResult.value.workspace.description)
|
|
input.description = description.value
|
|
|
|
const result = await updateMutation({ input }).catch(convertThrowIntoFetchResult)
|
|
|
|
if (result?.data) {
|
|
mixpanel.track('Workspace General Settings Updated', {
|
|
fields: (Object.keys(input) as Array<keyof WorkspaceUpdateInput>).filter(
|
|
(key) => key !== 'id'
|
|
),
|
|
// eslint-disable-next-line camelcase
|
|
workspace_id: props.workspaceId
|
|
})
|
|
|
|
triggerNotification({
|
|
type: ToastNotificationType.Success,
|
|
title: 'Workspace updated'
|
|
})
|
|
} else {
|
|
const errorMessage = getFirstErrorMessage(result?.errors)
|
|
triggerNotification({
|
|
type: ToastNotificationType.Danger,
|
|
title: 'Workspace update failed',
|
|
description: errorMessage
|
|
})
|
|
}
|
|
})
|
|
|
|
watch(
|
|
() => workspaceResult,
|
|
() => {
|
|
if (workspaceResult.value?.workspace) {
|
|
name.value = workspaceResult.value.workspace.name
|
|
description.value = workspaceResult.value.workspace.description ?? ''
|
|
}
|
|
},
|
|
{ deep: true, immediate: true }
|
|
)
|
|
</script>
|