// Copyright (C) The Arvados Authors. All rights reserved. // // SPDX-License-Identifier: AGPL-3.0 import React from 'react'; import { WrappedFieldProps, Field, formValues, FormName, WrappedFieldInputProps, WrappedFieldMetaProps, change } from 'redux-form'; import { compose } from 'redux'; import { Autocomplete } from 'components/autocomplete/autocomplete'; import { Vocabulary, isStrictTag, getTagValues, getTagValueID, getTagValueLabel, PropFieldSuggestion, getPreferredTagValues } from 'models/vocabulary'; import { PROPERTY_KEY_FIELD_ID, PROPERTY_KEY_FIELD_NAME } from 'views-components/resource-properties-form/property-key-field'; import { handleSelect, handleBlur, VocabularyProp, ValidationProp, connectVocabulary, buildProps } from 'views-components/resource-properties-form/property-field-common'; import { TAG_VALUE_VALIDATION } from 'validators/validators'; import { escapeRegExp } from 'common/regexp'; import { ChangeEvent } from 'react'; interface PropertyKeyProp { propertyKeyId: string; propertyKeyName: string; } interface PropertyValueInputProp { disabled: boolean; } type PropertyValueFieldProps = VocabularyProp & PropertyKeyProp & ValidationProp & PropertyValueInputProp; export const PROPERTY_VALUE_FIELD_NAME = 'value'; export const PROPERTY_VALUE_FIELD_ID = 'valueID'; const connectVocabularyAndPropertyKey = compose( connectVocabulary, formValues({ propertyKeyId: PROPERTY_KEY_FIELD_ID, propertyKeyName: PROPERTY_KEY_FIELD_NAME, }), ); export const PropertyValueField = connectVocabularyAndPropertyKey( ({ skipValidation, ...props }: PropertyValueFieldProps) => ); const PropertyValueInput = ({ vocabulary, propertyKeyId, propertyKeyName, ...props }: WrappedFieldProps & PropertyValueFieldProps) => ( s.synonyms && s.synonyms.length > 0 ? `${s.label} (${s.synonyms.join('; ')})` : s.label } onSelect={handleSelect(PROPERTY_VALUE_FIELD_ID, data.form, props.input, props.meta)} onBlur={() => { // Case-insensitive search for the value in the vocabulary const foundValueID = getTagValueID(propertyKeyId, props.input.value, vocabulary); if (foundValueID !== '') { props.input.value = getTagValueLabel(propertyKeyId, foundValueID, vocabulary); } handleBlur(PROPERTY_VALUE_FIELD_ID, data.form, props.meta, props.input, foundValueID)(); }} onChange={(e: ChangeEvent) => { const newValue = e.currentTarget.value; const tagValueID = getTagValueID(propertyKeyId, newValue, vocabulary); handleChange(data.form, tagValueID, props.input, props.meta, newValue); }} /> )} />; const getValidation = (props: PropertyValueFieldProps) => isStrictTag(props.propertyKeyId, props.vocabulary) ? [...TAG_VALUE_VALIDATION, matchTagValues(props)] : TAG_VALUE_VALIDATION; const matchTagValues = ({ vocabulary, propertyKeyId }: PropertyValueFieldProps) => (value: string) => getTagValues(propertyKeyId, vocabulary).find(v => !value || v.label === value) ? undefined : 'Incorrect value'; const getSuggestions = (value: string, tagName: string, vocabulary: Vocabulary) => { const re = new RegExp(escapeRegExp(value), "i"); return getPreferredTagValues(tagName, vocabulary, value).filter( val => (val.label !== value && re.test(val.label)) || (val.synonyms && val.synonyms.some(s => re.test(s)))); }; const handleChange = ( formName: string, tagValueID: string, { onChange }: WrappedFieldInputProps, { dispatch }: WrappedFieldMetaProps, value: string) => { onChange(value); dispatch(change(formName, PROPERTY_VALUE_FIELD_NAME, value)); dispatch(change(formName, PROPERTY_VALUE_FIELD_ID, tagValueID)); };