|
|
|
import useVuelidate from '@vuelidate/core'
|
|
|
|
import { minLength, required } from '@vuelidate/validators'
|
|
|
|
import { message } from 'ant-design-vue'
|
|
|
|
import type { ColumnType, LinkToAnotherRecordType, TableType } from 'nocodb-sdk'
|
|
|
|
import { ErrorMessages, RelationTypes, UITypes, isVirtualCol } from 'nocodb-sdk'
|
|
|
|
import { extractSdkResponseErrorMsg } from '~/utils'
|
|
|
|
import { useInjectionState } from '#imports'
|
|
|
|
|
|
|
|
const [useProvideSharedFormStore, useSharedFormStore] = useInjectionState(() => {
|
|
|
|
const progress = ref(false)
|
|
|
|
const notFound = ref(false)
|
|
|
|
const showPasswordModal = ref(false)
|
|
|
|
const submitted = ref(false)
|
|
|
|
const password = ref(null)
|
|
|
|
const secondsRemain = ref(0)
|
|
|
|
|
|
|
|
// todo: type
|
|
|
|
const sharedView = ref<any>()
|
|
|
|
const meta = ref<TableType>()
|
|
|
|
const columns = ref<(ColumnType & { required?: boolean })[]>()
|
|
|
|
|
|
|
|
const { $api } = useNuxtApp()
|
|
|
|
const { metas, setMeta } = useMetas()
|
|
|
|
const formState = ref({})
|
|
|
|
|
|
|
|
const { state: additionalState } = useProvideSmartsheetRowStore(
|
|
|
|
meta,
|
|
|
|
ref({
|
|
|
|
row: formState,
|
|
|
|
rowMeta: { new: true },
|
|
|
|
oldRow: {},
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
|
|
|
|
const formColumns = computed(
|
|
|
|
() =>
|
|
|
|
columns?.value
|
|
|
|
?.filter?.(
|
|
|
|
(f: Record<string, any>) =>
|
|
|
|
f.show && f.uidt !== UITypes.Rollup && f.uidt !== UITypes.Lookup && f.uidt !== UITypes.Formula,
|
|
|
|
)
|
|
|
|
.sort((a: Record<string, any>, b: Record<string, any>) => a.order - b.order)
|
|
|
|
.map<ColumnType & { required: boolean }>((c: ColumnType & { required?: boolean }) => ({
|
|
|
|
...c,
|
|
|
|
required: !!(c.required || 0),
|
|
|
|
})) ?? [],
|
|
|
|
)
|
|
|
|
const loadSharedView = async (viewId: string) => {
|
|
|
|
try {
|
|
|
|
// todo: swagget type correction
|
|
|
|
const viewMeta: any = await $api.public.sharedViewMetaGet(viewId)
|
|
|
|
sharedView.value = viewMeta
|
|
|
|
meta.value = viewMeta.model
|
|
|
|
columns.value = viewMeta.model.columns
|
|
|
|
|
|
|
|
setMeta(viewMeta.model)
|
|
|
|
|
|
|
|
const relatedMetas = { ...viewMeta.relatedMetas }
|
|
|
|
Object.keys(relatedMetas).forEach((key) => setMeta(relatedMetas[key]))
|
|
|
|
} catch (e: any) {
|
|
|
|
if (e.response && e.response.status === 404) {
|
|
|
|
notFound.value = true
|
|
|
|
} else if ((await extractSdkResponseErrorMsg(e)) === ErrorMessages.INVALID_SHARED_VIEW_PASSWORD) {
|
|
|
|
showPasswordModal.value = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const validators = computed(() => {
|
|
|
|
const obj: any = {
|
|
|
|
localState: {},
|
|
|
|
virtual: {},
|
|
|
|
}
|
|
|
|
for (const column of formColumns?.value ?? []) {
|
|
|
|
if (
|
|
|
|
!isVirtualCol(column) &&
|
|
|
|
((column.rqd && !column.cdf) || (column.pk && !(column.ai || column.cdf)) || (column as any).required)
|
|
|
|
) {
|
|
|
|
obj.localState[column.title!] = { required }
|
|
|
|
} else if (
|
|
|
|
column.uidt === UITypes.LinkToAnotherRecord &&
|
|
|
|
column.colOptions &&
|
|
|
|
(column.colOptions as LinkToAnotherRecordType).type === RelationTypes.BELONGS_TO
|
|
|
|
) {
|
|
|
|
const col = columns.value?.find((c) => c.id === (column?.colOptions as LinkToAnotherRecordType)?.fk_child_column_id)
|
|
|
|
|
|
|
|
if ((col && col.rqd && !col.cdf) || column.required) {
|
|
|
|
if (col) {
|
|
|
|
obj.virtual[column.title!] = { required }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (isVirtualCol(column) && column.required) {
|
|
|
|
obj.virtual[column.title!] = {
|
|
|
|
minLength: minLength(1),
|
|
|
|
required,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return obj
|
|
|
|
})
|
|
|
|
|
|
|
|
const v$ = useVuelidate(
|
|
|
|
validators,
|
|
|
|
computed(() => ({ localState: formState?.value, virtual: additionalState?.value })),
|
|
|
|
)
|
|
|
|
|
|
|
|
const submitForm = async (formState: Record<string, any>, additionalState: Record<string, any>) => {
|
|
|
|
try {
|
|
|
|
if (!(await v$.value?.$validate())) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
progress.value = true
|
|
|
|
const data = { ...formState, ...additionalState }
|
|
|
|
const attachment: Record<string, any> = {}
|
|
|
|
|
|
|
|
for (const col of metas?.value?.[sharedView?.value?.fk_model_id]?.columns ?? []) {
|
|
|
|
if (col.uidt === UITypes.Attachment) {
|
|
|
|
attachment[`_${col.title}`] = data[col.title!]
|
|
|
|
delete data[col.title!]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
await $api.public.dataCreate(
|
|
|
|
sharedView?.value?.uuid,
|
|
|
|
{
|
|
|
|
data,
|
|
|
|
...attachment,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// todo: add password support
|
|
|
|
// headers: { 'xc-password': password },
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
submitted.value = true
|
|
|
|
progress.value = false
|
|
|
|
|
|
|
|
additionalState.value = {}
|
|
|
|
formState.value = {}
|
|
|
|
|
|
|
|
await message.success(sharedView.value.success_msg || 'Saved successfully.')
|
|
|
|
} catch (e: any) {
|
|
|
|
console.log(e)
|
|
|
|
throw e
|
|
|
|
await message.error(await extractSdkResponseErrorMsg(e))
|
|
|
|
}
|
|
|
|
progress.value = false
|
|
|
|
}
|
|
|
|
|
|
|
|
/** reset form if show_blank_form is true */
|
|
|
|
watch(submitted, (nextVal: boolean) => {
|
|
|
|
if (nextVal && sharedView.value.show_blank_form) {
|
|
|
|
secondsRemain.value = 5
|
|
|
|
const intvl = setInterval(() => {
|
|
|
|
if (--secondsRemain.value < 0) {
|
|
|
|
submitted.value = false
|
|
|
|
clearInterval(intvl)
|
|
|
|
}
|
|
|
|
}, 1000)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return {
|
|
|
|
sharedView,
|
|
|
|
loadSharedView,
|
|
|
|
columns,
|
|
|
|
submitForm,
|
|
|
|
progress,
|
|
|
|
meta,
|
|
|
|
validators,
|
|
|
|
v$,
|
|
|
|
formColumns,
|
|
|
|
formState,
|
|
|
|
notFound,
|
|
|
|
password,
|
|
|
|
submitted,
|
|
|
|
secondsRemain,
|
|
|
|
}
|
|
|
|
}, 'expanded-form-store')
|
|
|
|
|
|
|
|
export { useProvideSharedFormStore }
|
|
|
|
|
|
|
|
export function useSharedFormStoreOrThrow() {
|
|
|
|
const sharedFormStore = useSharedFormStore()
|
|
|
|
if (sharedFormStore == null) throw new Error('Please call `useProvideSharedFormStore` on the appropriate parent component')
|
|
|
|
return sharedFormStore
|
|
|
|
}
|