// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { email as emailValidation } from '@formkit/rules' import { ref, type Ref } from 'vue' import stopEvent from '#shared/utils/events.ts' import type { DropdownOptionsAction } from '#desktop/components/CommonSelect/types.ts' import type { AutoCompleteOptionValueDictionary, ClearFilterInputFunction, SelectOptionFunction, } from '#desktop/components/Form/fields/FieldAutoComplete/types.ts' import type { FormKitNode } from '@formkit/core' export const emailFilterValueValidator = (filter: string) => emailValidation({ value: filter } as FormKitNode) export const phoneFilterValueValidator = (filter: string) => /^\+?[1-9]\d+$/.test(filter) export const useAddUnknownValueAction = ( label?: Ref, filterValueValidator?: (filter: string) => boolean | Promise, ) => { const actions = ref([]) const actionLabel = label ?? ref(__('add new email address')) const isValidFilterValue = filterValueValidator ?? emailFilterValueValidator const isNewOption = ( filter: string, optionValues: AutoCompleteOptionValueDictionary, ) => !optionValues[filter] const addUnknownValue = ( filter: string, selectOption: SelectOptionFunction, clearFilter: ClearFilterInputFunction, focus: boolean, ) => { const newOption = { value: filter, label: filter, } selectOption(newOption, focus) clearFilter() } const onSearchInteractionUpdate = ( filter: string, optionValues: AutoCompleteOptionValueDictionary, selectOption: SelectOptionFunction, clearFilter: ClearFilterInputFunction, ) => { if (!isNewOption(filter, optionValues) || !isValidFilterValue(filter)) { actions.value = [] return } actions.value = [ { key: 'addUnknownValue', label: actionLabel.value, icon: 'plus-square-fill', onClick: (focus) => { addUnknownValue(filter, selectOption, clearFilter, focus) // Reset actions after current filter was added. actions.value = [] }, }, ] } const onKeydownFilterInput = ( event: KeyboardEvent, filter: string, optionValues: AutoCompleteOptionValueDictionary, selectOption: SelectOptionFunction, clearFilter: ClearFilterInputFunction, ) => { const { key } = event if (!filter) return if (['Enter', 'Tab'].includes(key)) { stopEvent(event) if (!isNewOption(filter, optionValues) || !isValidFilterValue(filter)) return addUnknownValue(filter, selectOption, clearFilter, true) } } return { actions, isNewOption, isValidFilterValue, addUnknownValue, onSearchInteractionUpdate, onKeydownFilterInput, } }