diff --git a/src/application/types.js b/src/application/types.js index 7ee0ba774..ca275675b 100644 --- a/src/application/types.js +++ b/src/application/types.js @@ -33,6 +33,9 @@ export type FormSection = { id: number; identifier: string; title: string; + title_fi: string; + title_en: string | null; + title_sv: string | null; visible: boolean; sort_order: number; add_new_allowed: boolean; @@ -50,7 +53,13 @@ export type FormField = { identifier: string; type: number; label: string; - hint_text?: string; + label_fi: string; + label_en: string | null; + label_sv: string | null; + hint_text: string | null; + hint_text_fi: string | null; + hint_text_en: string | null; + hint_text_sv: string | null; enabled: boolean; required: boolean; validation?: string | null; @@ -64,6 +73,9 @@ export type FormField = { export type FormFieldChoice = { id: number; text: string; + text_fi: string; + text_en: string | null; + text_sv: string | null; value: string; action?: string | null; has_text_input: boolean; diff --git a/src/components/button/IconButton.js b/src/components/button/IconButton.js index 4264ea159..e8a6b9b9e 100644 --- a/src/components/button/IconButton.js +++ b/src/components/button/IconButton.js @@ -10,21 +10,34 @@ type Props = { style?: Object, title?: string, type?: string, + id?: string, } -const IconButton = ({children, className, disabled, onClick, style, title, type = 'button'}: Props) => { - return ( - - {children} - - ); -}; +const IconButton = (React.forwardRef( + ({ + children, + className, + disabled, + onClick, + style, + title, + id, + type = 'button', + }: Props, ref) => { + return ( + + {children} + + ); + }): React$AbstractComponent); export default IconButton; diff --git a/src/components/collapse/Collapse.js b/src/components/collapse/Collapse.js index 4192e8d84..f7222fdc4 100644 --- a/src/components/collapse/Collapse.js +++ b/src/components/collapse/Collapse.js @@ -32,6 +32,7 @@ type Props = { showTitleOnOpen?: boolean, tooltipStyle?: Object, uiDataKey?: ?string, + isOpen?: boolean, } type State = { @@ -53,10 +54,13 @@ class Collapse extends PureComponent { }; state: State = { - contentHeight: this.props.defaultOpen ? null : 0, + contentHeight: (this.props.isOpen !== undefined + ? this.props.isOpen + : this.props.defaultOpen + ) ? null : 0, isCollapsing: false, isExpanding: false, - isOpen: this.props.defaultOpen, + isOpen: this.props.isOpen !== undefined ? this.props.isOpen : this.props.defaultOpen, }; setComponentRef: (any) => void = (el) => { @@ -76,6 +80,10 @@ class Collapse extends PureComponent { } componentDidUpdate(prevProps: Object, prevState: Object) { + if (this.props.isOpen !== undefined && this.props.isOpen !== this.state.isOpen) { + this.handleToggleStateChange(this.props.isOpen); + } + if ((this.state.isOpen && !this.state.contentHeight) || (this.state.isOpen !== prevState.isOpen)) { this.calculateHeight(); @@ -101,35 +109,43 @@ class Collapse extends PureComponent { } handleToggle: (SyntheticEvent) => void = (e) => { - const {onToggle} = this.props; + const {onToggle, isOpen: externalIsOpen} = this.props; const {isOpen} = this.state; const target = e.currentTarget; const tooltipEl = ReactDOM.findDOMNode(this.tooltip); + const isExternallyControlled = externalIsOpen !== undefined; + if (!tooltipEl || (tooltipEl && target !== tooltipEl && !tooltipEl.contains(target))) { - if(isOpen) { - this.setState({ - isCollapsing: true, - isExpanding: false, - isOpen: false, - }); - } else { - this.setState({ - isCollapsing: false, - isExpanding: true, - isOpen: true, - }); + if (!isExternallyControlled) { + this.handleToggleStateChange(!isOpen); } - if(onToggle) { + if (onToggle) { onToggle(!isOpen); } } }; + handleToggleStateChange: (boolean) => void = (newIsOpen) => { + if (newIsOpen) { + this.setState({ + isCollapsing: false, + isExpanding: true, + isOpen: true, + }); + } else { + this.setState({ + isCollapsing: true, + isExpanding: false, + isOpen: false, + }); + } + } + handleKeyDown: (SyntheticKeyboardEvent) => void = (e) => { - if(e.keyCode === 13) { + if (e.keyCode === 13) { e.preventDefault(); this.handleToggle(e); } diff --git a/src/components/form/FieldTypeMultiSelect.js b/src/components/form/FieldTypeMultiSelect.js index f8546dd35..a03547312 100644 --- a/src/components/form/FieldTypeMultiSelect.js +++ b/src/components/form/FieldTypeMultiSelect.js @@ -32,7 +32,7 @@ const FieldTypeMultiSelect = ({ options={options} onBlur={handleBlur} onSelectedChanged={onChange} - selected={value} + selected={value instanceof Array ? value : []} disabled={disabled} isLoading={isLoading} /> diff --git a/src/components/form/FormField.js b/src/components/form/FormField.js index 450bb44c2..8f2fee7a5 100644 --- a/src/components/form/FormField.js +++ b/src/components/form/FormField.js @@ -283,6 +283,7 @@ type Props = { filterOption?: Function, invisibleLabel?: boolean, isLoading?: boolean, + isMulti?: boolean, language?: string, name: string, onBlur?: Function, diff --git a/src/components/icons/MoveDownIcon.js b/src/components/icons/MoveDownIcon.js new file mode 100644 index 000000000..69876939f --- /dev/null +++ b/src/components/icons/MoveDownIcon.js @@ -0,0 +1,18 @@ +// @flow +import React from 'react'; +import classNames from 'classnames'; + +type Props = { + className?: string, +} + +const MoveDownIcon = ({className}: Props): React$Element<'svg'> => + + Siirrä alas + + + + ; + + +export default MoveDownIcon; diff --git a/src/components/icons/MoveUpIcon.js b/src/components/icons/MoveUpIcon.js new file mode 100644 index 000000000..b104895b0 --- /dev/null +++ b/src/components/icons/MoveUpIcon.js @@ -0,0 +1,17 @@ +// @flow +import React from 'react'; +import classNames from 'classnames'; + +type Props = { + className?: string, +} + +const MoveUpIcon = ({className}: Props): React$Element<'svg'> => + + Siirrä ylös + + + + ; + +export default MoveUpIcon; diff --git a/src/components/icons/TrashIcon.js b/src/components/icons/TrashIcon.js index d0076225d..b66d4053e 100644 --- a/src/components/icons/TrashIcon.js +++ b/src/components/icons/TrashIcon.js @@ -6,7 +6,7 @@ type Props = { className?: string, } -const TrashIcon = ({className}: Props) => +const TrashIcon = ({className}: Props): React$Element<'svg'> => Poista diff --git a/src/components/icons/_icons.scss b/src/components/icons/_icons.scss index 95170bca4..3e21be881 100644 --- a/src/components/icons/_icons.scss +++ b/src/components/icons/_icons.scss @@ -43,6 +43,18 @@ stroke: none !important; } + &.icons__move-up, + &.icons__move-down { + width: rem-calc(16px); + + &.icon-medium { + width: rem-calc(12px); + } + &.icon-small { + width: rem-calc(9px); + } + } + &.icon-medium { height: rem-calc(20px); width: rem-calc(20px); diff --git a/src/components/multi-select/MultiSelect.js b/src/components/multi-select/MultiSelect.js index 4f7032299..d99574f1c 100644 --- a/src/components/multi-select/MultiSelect.js +++ b/src/components/multi-select/MultiSelect.js @@ -43,7 +43,7 @@ const MultiSelect = ({ hasSelectAll = true, onSelectedChanged, valueRenderer, -}: Props) => { +}: Props): React$Node => { const getSelectedText = () => { const selectedOptions = selected .map(s => options.find(o => o.value === s)); diff --git a/src/enums.js b/src/enums.js index 650cca571..e9407cdb6 100644 --- a/src/enums.js +++ b/src/enums.js @@ -386,6 +386,11 @@ export const ConfirmationModalTexts = { LABEL: 'Haluatko varmasti poistaa kentän?', TITLE: 'Poista kenttä', }, + DELETE_SECTION_SUBSECTION: { + BUTTON: 'Poista aliosio', + LABEL: 'Haluatko varmasti poistaa aliosion?', + TITLE: 'Poista aliosio', + }, DELETE_APPLICATION_TARGET_PROPOSED_MANAGEMENT: { BUTTON: DELETE_MODAL_BUTTON_TEXT, LABEL: 'Haluatko varmasti poistaa hallinta- ja rahoitusmuotoehdotuksen?', diff --git a/src/plotSearch/actions.js b/src/plotSearch/actions.js index adeb80957..63b089cb5 100644 --- a/src/plotSearch/actions.js +++ b/src/plotSearch/actions.js @@ -66,6 +66,9 @@ import type { ResetPlanUnitDecisionsAction, ShowEditModeAction, TemplateFormsNotFoundAction, + ClearSectionEditorCollapseStatesAction, + SetSectionEditorCollapseStateAction, + InitializeSectionEditorCollapseStatesAction, } from '$src/plotSearch/types'; export const fetchAttributes = (): FetchAttributesAction => @@ -238,3 +241,12 @@ export const directReservationLinkCreated = (): DirectReservationLinkCreatedActi export const directReservationLinkCreationFailed = (payload: any): DirectReservationLinkCreationFailedAction => createAction('mvj/plotSearch/DIRECT_RESERVATION_LINK_CREATION_FAILED')(payload); + +export const clearSectionEditorCollapseStates = (): ClearSectionEditorCollapseStatesAction => + createAction('mvj/plotSearch/CLEAR_SECTION_EDITOR_COLLAPSE_STATES')(); + +export const setSectionEditorCollapseState = (key: string, isOpen: boolean): SetSectionEditorCollapseStateAction => + createAction('mvj/plotSearch/SET_SECTION_EDITOR_COLLAPSE_STATE')({key, state: isOpen}); + +export const initializeSectionEditorCollapseStates = (states: {[key: string]: boolean}): InitializeSectionEditorCollapseStatesAction => + createAction('mvj/plotSearch/INITIALIZE_SECTION_EDITOR_COLLAPSE_STATES')(states); diff --git a/src/plotSearch/components/plotSearchSections/application/ApplicationEdit.js b/src/plotSearch/components/plotSearchSections/application/ApplicationEdit.js index af10951fb..4c3c39d21 100644 --- a/src/plotSearch/components/plotSearchSections/application/ApplicationEdit.js +++ b/src/plotSearch/components/plotSearchSections/application/ApplicationEdit.js @@ -80,7 +80,7 @@ type State = { class ApplicationEdit extends PureComponent { state = { isModalOpen: false, - modalSectionIndex: 0, + modalSectionIndex: -1, } componentDidMount() { @@ -271,18 +271,6 @@ class ApplicationEdit extends PureComponent { uiDataKey={getUiDataLeaseKey(ApplicationFieldPaths.NAME)} /> - {/* - */} {formData.sections.map((section, index) => diff --git a/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionFieldForm.js b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionFieldForm.js new file mode 100644 index 000000000..29725afd5 --- /dev/null +++ b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionFieldForm.js @@ -0,0 +1,786 @@ +// @flow +import React, {Fragment, useEffect, useMemo, useRef, useState} from 'react'; +import {connect} from 'react-redux'; +import {Column, Row} from 'react-foundation'; +import get from 'lodash/get'; +import {change, FieldArray, formValueSelector} from 'redux-form'; + +import FormField from '$components/form/FormField'; +import InfoIcon from '$components/icons/InfoIcon'; +import Tooltip from '$components/tooltip/Tooltip'; +import {FieldTypes, FormNames} from '$src/enums'; +import TooltipToggleButton from '$components/tooltip/TooltipToggleButton'; +import TooltipWrapper from '$components/tooltip/TooltipWrapper'; +import {getFieldTypeMapping, getFormAttributes} from '$src/application/selectors'; +import SubTitle from '$components/content/SubTitle'; +import {generateFieldIdentifierFromName} from '$src/plotSearch/helpers'; +import {FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME, FieldTypeFeatures, FieldTypeLabels} from '$src/plotSearch/constants'; +import IconButton from '$components/button/IconButton'; +import TrashIcon from '$components/icons/TrashIcon'; +import AddIcon from '$components/icons/AddIcon'; +import EditIcon from '$components/icons/EditIcon'; +import MoveUpIcon from '$components/icons/MoveUpIcon'; +import MoveDownIcon from '$components/icons/MoveDownIcon'; + +import type {Attributes} from '$src/types'; +import type {Fields} from 'redux-form/lib/FieldArrayProps.types'; + +type ChoiceProps = { + attributes: Attributes, + disabled: boolean, + field: string, + fields: Fields, + change: Function, + onChoiceValuesChanged: Function, + onChoiceDeleted: Function, + autoFillValues: boolean, + protectedValues: Array, +} + +const EditPlotApplicationSectionFieldChoice = ({ + fields, + attributes, + disabled, + change, + onChoiceValuesChanged, + autoFillValues, + protectedValues, +}: ChoiceProps): React$Node => { + const choiceRefs = useRef({}); + + const getDataMapBase = (): {[key: string]: any} => fields.getAll().reduce((acc, item) => ({ + ...acc, + [item.value]: item.value, + }), {}); + + useEffect(() => { + if (autoFillValues) { + const dataMap = getDataMapBase(); + + fields.forEach((choiceField, choiceIndex) => { + const autoValue = (choiceIndex + 1).toString(); + const currentValue = fields.get(choiceIndex).value; + + if (currentValue !== autoValue) { + dataMap[currentValue] = autoValue; + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${choiceField}.value`, + autoValue, + ); + } + }); + + onChoiceValuesChanged(dataMap); + } + }, [autoFillValues]); + + const setOtherChoicesTextInputOff = (changedField: string): void => { + fields.forEach((choiceField) => { + if (choiceField !== changedField) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${choiceField}.has_text_input`, + false + ); + } + }); + }; + + const handleRemove = (index: number) => { + if (autoFillValues) { + const dataMap = getDataMapBase(); + const deletedValue = fields.get(index).value; + dataMap[deletedValue] = null; + + fields.forEach((choiceField, choiceIndex) => { + if (index >= choiceIndex) { + return; + } + + const autoValue = choiceIndex.toString(); + const currentValue = fields.get(choiceIndex).value; + if (currentValue !== autoValue) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${choiceField}.value`, + autoValue, + ); + dataMap[currentValue] = autoValue; + } + }); + + onChoiceValuesChanged(dataMap); + } + + fields.remove(index); + }; + + const handleAdd = () => { + const dataMap = getDataMapBase(); + const initialValue = autoFillValues ? String(fields.length + 1) : ''; + + fields.push({ + text: '', + text_fi: '', + text_en: '', + text_sv: '', + value: initialValue, + has_text_input: false, + protected_values: [], + }); + + dataMap[initialValue] = initialValue; + onChoiceValuesChanged(dataMap); + }; + + const handleMoveUp = (index: number) => { + if (autoFillValues) { + handleSwapAutoValues(index, index - 1); + } + fields.move(index, index - 1); + setImmediate(() => { + + if (index - 1 !== 0) { + choiceRefs.current[`SectionEditorMoveUpButton_Choice_${index - 1}`]?.focus(); + } else { + choiceRefs.current[`SectionEditorMoveDownButton_Choice_${index - 1}`]?.focus(); + } + }); + }; + + const handleMoveDown = (index: number) => { + if (autoFillValues) { + handleSwapAutoValues(index, index + 1); + } + fields.move(index, index + 1); + setImmediate(() => { + if (index + 1 < fields.length - 1) { + choiceRefs.current[`SectionEditorMoveDownButton_Choice_${index + 1}`]?.focus(); + } else { + choiceRefs.current[`SectionEditorMoveUpButton_Choice_${index + 1}`]?.focus(); + } + }); + }; + + const setRef = (index, el) => { + choiceRefs.current[index] = el; + }; + + const handleSwapAutoValues = (index1: number, index2: number) => { + const dataMap = getDataMapBase(); + dataMap[(index2 + 1).toString()] = (index1 + 1).toString(); + dataMap[(index1 + 1).toString()] = (index2 + 1).toString(); + onChoiceValuesChanged(dataMap); + + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${fields.name}[${index1}].value`, + (index2 + 1).toString(), + ); + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${fields.name}[${index2}].value`, + (index1 + 1).toString(), + ); + }; + + const handleSingleValueChange = (index: number, value: any) => { + const dataMap = getDataMapBase(); + const currentValue = fields.get(index).value; + dataMap[currentValue] = value; + + onChoiceValuesChanged(dataMap); + }; + + return <> + {fields.map((field, i) => { + const isProtected = protectedValues.includes(fields.get(i).value); + + return + + + #{i + 1} + + + + + + + + + + + + + + + + handleRemove(i)} disabled={isProtected}> + + + handleMoveUp(i)} + disabled={i === 0} + id={`SectionEditorMoveUpButton_Choice_${i}`} + ref={(el) => setRef(`SectionEditorMoveUpButton_Choice_${i}`, el)} + > + + + handleMoveDown(i)} + disabled={(i + 1) >= fields.length} + id={`SectionEditorMoveDownButton_Choice_${i}`} + ref={(el) => setRef(`SectionEditorMoveDownButton_Choice_${i}`, el)} + > + + + + + + + + + + handleSingleValueChange(i, newValue)} + /> + + + setOtherChoicesTextInputOff(field)} + /> + + + + + + ; + })} + + Lisää vaihtoehto + + >; +}; + +type OwnProps = { + disabled: boolean, + field: any, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, +}; + +type Props = { + ...OwnProps, + attributes: Attributes, + fieldValues: Object, + fieldTypeMapping: {[number]: string}, + change: Function, + fieldIdentifiers: Array, + index: number, + onDelete: Function, + onMoveUp: Function | null, + onMoveDown: Function | null, + fieldRefs: any, +} + +const EditPlotApplicationSectionFieldForm = ({ + disabled, + field, + attributes, + fieldValues, + fieldTypeMapping, + change, + fieldIdentifiers, + onDelete, + onMoveUp, + onMoveDown, + collapseStates, + setSectionEditorCollapseState, + fieldRefs, +}: Props) => { + const [isHintPopupOpen, setIsHintPopupOpen] = useState(false); + const id = fieldValues.id ?? fieldValues.temporary_id; + const upButtonId = `SectionEditorMoveUpButton_Field_${id}`; + const downButtonId = `SectionEditorMoveDownButton_Field_${id}`; + + const isOpen = collapseStates[`field-${id}`]; + + useEffect(() => { + if (isOpen === undefined) { + setSectionEditorCollapseState(`field-${id}`, !fieldValues.id || !fieldValues.type); + } + }, []); + + const type = fieldTypeMapping[fieldValues.type]; + + const typeChoices = useMemo>(() => { + return get(attributes, 'sections.child.children.fields.child.children.type.choices')?.map((type) => ({ + value: type.value, + label: FieldTypeLabels[type.display_name] || type.display_name, + })); + }, [fieldTypeMapping]); + + const fieldFeatures = useMemo>(() => { + return FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME[type] || []; + }, [fieldValues.type]); + + const listSelectionDefaultValueType = useMemo(() => { + if (fieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS)) { + if (fieldValues.choices.length === 0) { + return FieldTypes.BOOLEAN; + } else { + return FieldTypes.MULTISELECT; + } + } + + return FieldTypes.CHOICE; + }, [ + fieldValues.type, + fieldValues.choices, + ]); + + const updateAutoIdentifier = (shouldChange: boolean, newName: string): void => { + if (shouldChange && !fieldValues.is_protected) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.identifier`, + generateFieldIdentifierFromName(newName, fieldIdentifiers) + ); + } + }; + + const handleTypeChanged = (newType: number): void => { + const newFieldFeatures = FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME[fieldTypeMapping[newType]] || []; + + const prevIsMultiSelect = fieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS); + const newIsMultiSelect = newFieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS); + + if (prevIsMultiSelect && !newIsMultiSelect) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + fieldValues.default_value?.[0] || '' + ); + } else if (newIsMultiSelect && !prevIsMultiSelect) { + if (fieldValues.choices.length > 0) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + fieldValues.default_value ? [fieldValues.default_value] : [] + ); + } else { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + false + ); + } + } + }; + + const handleChoiceValuesChanged = (dataMap: {[string]: string}) => { + if (fieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS)) { + if (Object.values(dataMap).filter((choiceValue) => choiceValue !== null).length === 0) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + false, + ); + } else { + if (fieldValues.default_value instanceof Array) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + fieldValues.default_value.map((choiceValue) => dataMap[choiceValue]) + .filter((choiceValue) => choiceValue !== null) || [], + ); + } else { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + [], + ); + } + } + } else { + if (dataMap[fieldValues.default_value] !== fieldValues.default_value) { + change( + FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING, + `${field}.default_value`, + dataMap[fieldValues.default_value], + ); + } + } + }; + + const optionChoices = useMemo(() => { + const options = fieldValues.choices?.map((option) => ({ + label: option.text, + value: option.value, + })); + + if (fieldFeatures.includes(FieldTypeFeatures.SINGLE_SELECTION_OPTIONS)) { + options.unshift({ + label: '', + value: null, + }); + } + return options; + }, [ + fieldValues.choices, + fieldValues.type, + ]); + + const setRef = (id, el) => { + fieldRefs.current[id] = el; + }; + + return ( + + + + + + {fieldValues.hint_text && + + setIsHintPopupOpen(true)}> + + + setIsHintPopupOpen(false)}> + {fieldValues.hint_text} + + + } + + + + + + setSectionEditorCollapseState(`field-${id}`, !isOpen)}> + + + + + + onMoveUp?.(id)} + id={upButtonId} + ref={(el) => setRef(upButtonId, el)} + > + + + onMoveDown?.(id)} + id={downButtonId} + ref={(el) => setRef(downButtonId, el)} + > + + + + + {isOpen && + + + + + updateAutoIdentifier( + fieldValues.auto_fill_identifier, + newName, + )} + /> + + + + + + + + + + + + + + + { + if (value) { + updateAutoIdentifier( + true, + fieldValues.label, + ); + } + }} + /> + + + + + + + + + + + + + + {( + fieldFeatures.includes(FieldTypeFeatures.FREEFORM_DEFAULT_VALUE) || + fieldFeatures.includes(FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE) + ) && ( + + Kentän arvot + + )} + + {fieldFeatures.includes(FieldTypeFeatures.FREEFORM_DEFAULT_VALUE) && + + } + {fieldFeatures.includes(FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE) && + + } + + {( + fieldFeatures.includes(FieldTypeFeatures.SINGLE_SELECTION_OPTIONS) || + fieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS) + ) && <> + + + Vaihtoehdot + + + + + 0} + invisibleLabel + overrideValues={{ + options: [{ + label: 'Täytä arvot automaattisesti', + value: true, + }], + }} + /> + + + + + + + + >} + + + } + + ); +}; + +const selector = formValueSelector(FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING); + +export default (connect( + (state, props: OwnProps) => { + return { + attributes: getFormAttributes(state), + fieldValues: selector(state, props.field), + fieldTypeMapping: getFieldTypeMapping(state), + }; + }, { + change, + } +)(EditPlotApplicationSectionFieldForm): React$ComponentType); diff --git a/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionForm.js b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionForm.js index bf4da2b4a..4d8d714d5 100644 --- a/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionForm.js +++ b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionForm.js @@ -1,7 +1,7 @@ // @flow -import React, {Component} from 'react'; +import React, {Component, useEffect, useRef} from 'react'; import {connect} from 'react-redux'; -import {formValueSelector, reduxForm, getFormValues} from 'redux-form'; +import {formValueSelector, reduxForm, getFormValues, change} from 'redux-form'; import {Row, Column} from 'react-foundation'; import flowRight from 'lodash/flowRight'; import get from 'lodash/get'; @@ -11,68 +11,223 @@ import {FieldArray} from 'redux-form'; import Button from '$components/button/Button'; import FormField from '$components/form/FormField'; import ModalButtonWrapper from '$components/modal/ModalButtonWrapper'; -import {FormNames} from '$src/enums'; +import {ConfirmationModalTexts, FieldTypes, FormNames} from '$src/enums'; import {ButtonColors} from '$components/enums'; -import SectionField from '$src/plotSearch/components/plotSearchSections/application/SectionField'; +import EditPlotApplicationSectionFieldForm from '$src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionFieldForm'; import Collapse from '$src/components/collapse/Collapse'; import SubTitle from '$src/components/content/SubTitle'; -import {getFormAttributes} from '$src/application/selectors'; +import {getFieldTypeMapping, getFormAttributes} from '$src/application/selectors'; +import {ActionTypes, AppConsumer} from '$src/app/AppContext'; +import IconButton from '$components/button/IconButton'; +import TrashIcon from '$components/icons/TrashIcon'; +import MoveUpIcon from '$components/icons/MoveUpIcon'; +import MoveDownIcon from '$components/icons/MoveDownIcon'; import type {Attributes} from '$src/types'; import type {FormSection} from '$src/application/types'; +import { + generateSectionIdentifierFromName, + getDefaultNewFormField, + getDefaultNewFormSection, + getInitialFormSectionEditorData, + transformCommittedFormSectionEditorData, +} from '$src/plotSearch/helpers'; +import {APPLICANT_SECTION_IDENTIFIER} from '$src/application/constants'; +import {getSectionEditorCollapseStates} from '$src/plotSearch/selectors'; +import { + clearSectionEditorCollapseStates, + initializeSectionEditorCollapseStates, + setSectionEditorCollapseState, +} from '$src/plotSearch/actions'; +import {uniq} from 'lodash/array'; +import ErrorBlock from '$components/form/ErrorBlock'; type SectionFieldProps = { disabled: boolean, fields: any, formName: string, isSaveClicked: Boolean, - form: string + form: string, + fieldIdentifiers: Array, + dispatch: Function, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, + meta: Object, } const EditPlotApplicationSectionFormSectionFields = ({ disabled, fields, form, + fieldIdentifiers, + dispatch, + collapseStates, + setSectionEditorCollapseState, + meta, // usersPermissions, -}: SectionFieldProps): React$Element<*> => { +}: SectionFieldProps): React$Node => { + const fieldRefs = useRef({}); + const handleRemove = (index: number) => { + dispatch({ + type: ActionTypes.SHOW_CONFIRMATION_MODAL, + confirmationFunction: () => { + fields.remove(index); + }, + confirmationModalButtonClassName: ButtonColors.ALERT, + confirmationModalButtonText: ConfirmationModalTexts.DELETE_SECTION_FIELD.BUTTON, + confirmationModalLabel: ConfirmationModalTexts.DELETE_SECTION_FIELD.LABEL, + confirmationModalTitle: ConfirmationModalTexts.DELETE_SECTION_FIELD.TITLE, + }); + }; + return ( {!!fields.length && fields.map((field, index) => { - return { + fields.move(index, index - 1); + setImmediate(() => { + + if (index - 1 !== 0) { + fieldRefs.current[`SectionEditorMoveUpButton_Field_${id}`]?.focus(); + } else { + fieldRefs.current[`SectionEditorMoveDownButton_Field_${id}`]?.focus(); + } + }); + }; + + const handleMoveDown = (id) => { + fields.move(index, index + 1); + setImmediate(() => { + if (index + 1 < fields.length - 1) { + fieldRefs.current[`SectionEditorMoveDownButton_Field_${id}`]?.focus(); + } else { + fieldRefs.current[`SectionEditorMoveUpButton_Field_${id}`]?.focus(); + } + }); + }; + return index !== i)} + onDelete={() => handleRemove(index)} + onMoveUp={(index > 0) ? handleMoveUp : null} + onMoveDown={(index < (fields.length - 1)) ? handleMoveDown : null} + collapseStates={collapseStates} + setSectionEditorCollapseState={setSectionEditorCollapseState} + fieldRefs={fieldRefs} />; })} + + {meta.error && } + + { + fields.push(getDefaultNewFormField(fieldIdentifiers)); + }} text="Lisää kenttä" /> + + + ); }; type SectionSubsectionProps = { fields: any, + sectionPath: string, form: string, attributes: Attributes, level: number, - stagedSectionValues: Object + stagedSectionValues: Object, + change: Function, + dispatch: Function, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, + meta: Object, } const EditPlotApplicationSectionFormSectionSubsections = ({ fields, + sectionPath, form, attributes, level, stagedSectionValues, -}: SectionSubsectionProps): React$Element<*> => { - return fields.map( - (ss, i) => ); + change, + dispatch, + collapseStates, + setSectionEditorCollapseState, + meta, +}: SectionSubsectionProps): React$Node => { + const handleRemove = (index: number) => { + dispatch({ + type: ActionTypes.SHOW_CONFIRMATION_MODAL, + confirmationFunction: () => { + fields.remove(index); + }, + confirmationModalButtonClassName: ButtonColors.ALERT, + confirmationModalButtonText: ConfirmationModalTexts.DELETE_SECTION_SUBSECTION.BUTTON, + confirmationModalLabel: ConfirmationModalTexts.DELETE_SECTION_SUBSECTION.LABEL, + confirmationModalTitle: ConfirmationModalTexts.DELETE_SECTION_SUBSECTION.TITLE, + }); + }; + const sectionRefs = useRef({}); + + const section = get(stagedSectionValues, sectionPath); + const subsectionIdentifiers = section.subsections.map((subsection) => subsection.identifier); + + return <> + {fields.map( + (ss, index) => { + const handleMoveUp = (id) => { + fields.move(index, index - 1); + setImmediate(() => { + if (index - 1 !== 0) { + sectionRefs.current[`SectionEditorMoveUpButton_Section_${id}`]?.focus(); + } else { + sectionRefs.current[`SectionEditorMoveDownButton_Section_${id}`]?.focus(); + } + }); + }; + + const handleMoveDown = (id) => { + fields.move(index, index + 1); + setImmediate(() => { + if (index + 1 < fields.length - 1) { + sectionRefs.current[`SectionEditorMoveDownButton_Section_${id}`]?.focus(); + } else { + sectionRefs.current[`SectionEditorMoveUpButton_Section_${id}`]?.focus(); + } + }); + }; + + return index !== i)} + onDelete={() => handleRemove(index)} + onMoveUp={(index > 0) ? handleMoveUp : null} + onMoveDown={(index < (fields.length - 1)) ? handleMoveDown : null} + dispatch={dispatch} + collapseStates={collapseStates} + setSectionEditorCollapseState={setSectionEditorCollapseState} + sectionRefs={sectionRefs} + />; + })} + + + {meta.error && } + { + fields.push(getDefaultNewFormSection(subsectionIdentifiers)); + }} text={`Lisää aliosio osioon ${section.title || ''}`} /> + + + >; }; type SubsectionProps = { @@ -81,6 +236,15 @@ type SubsectionProps = { sectionPath: string, form: string, stagedSectionValues: Object, + change: Function, + peerSectionIdentifiers: Array, + onMoveUp: ?Function, + onMoveDown: ?Function, + onDelete: ?Function, + dispatch: Function, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, + sectionRefs?: any, } type SubsectionWrapperProps = { @@ -89,6 +253,16 @@ type SubsectionWrapperProps = { sectionPath: string, subsection: FormSection, children: React$Node, + stagedSectionValues: Object, + peerSectionIdentifiers: Array, + change: Function, + onMoveUp: Function, + onMoveDown: Function, + onDelete: Function, + isApplicantSecondLevelSubsection: boolean, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, + sectionRefs: any, } const EditPlotApplicationSectionFormSubsectionFirstLevelWrapper = ({children, level}: SubsectionWrapperProps) => - - {subsection.show_duplication_check ? : } - } - className={classNames( - 'edit-plot-application-section-form__section', - `edit-plot-application-section-form__section--level-${level}`, - { - 'collapse__secondary': level === 2, - 'collapse__third': level > 2, + stagedSectionValues, + peerSectionIdentifiers, + change, + onDelete, + onMoveUp, + onMoveDown, + isApplicantSecondLevelSubsection, + collapseStates, + setSectionEditorCollapseState, + sectionRefs, +}: SubsectionWrapperProps) => { + const autoIdentifier = get(stagedSectionValues, `${sectionPath}.auto_fill_identifier`); + const isProtected = get(stagedSectionValues, `${sectionPath}.is_protected`); + const id = get(stagedSectionValues, `${sectionPath}.id`) ?? get(stagedSectionValues, `${sectionPath}.temporary_id`); + const upButtonId = `SectionEditorMoveUpButton_Section_${id}`; + const downButtonId = `SectionEditorMoveDownButton_Section_${id}`; + + const isOpen = collapseStates[`section-${id}`]; + + useEffect(() => { + if (isOpen === undefined) { + setSectionEditorCollapseState(`section-${id}`, true); } - )} ->{children}; + }, []); + + const updateAutoIdentifier = (shouldChange: boolean, sectionPath: string, newName: string): void => { + if (shouldChange && !isProtected) { + change( + `${sectionPath}.identifier`, + generateSectionIdentifierFromName(newName, peerSectionIdentifiers) + ); + } + }; + + const handleMoveUp = () => { + onMoveUp(id); + }; + + const handleMoveDown = () => { + onMoveDown(id); + }; + + const setRef = (id, el) => { + if (sectionRefs) { + sectionRefs.current[id] = el; + } + }; + + const sectionValuesColumnWidth = isApplicantSecondLevelSubsection ? 4 : 3; + + return setSectionEditorCollapseState(`section-${id}`, newIsOpen)} + className={classNames( + 'edit-plot-application-section-form__section', + `edit-plot-application-section-form__section--level-${level}`, + { + 'collapse__secondary': level === 2, + 'collapse__third': level > 2, + }, + )} + headerTitle={subsection.title || '-'} + headerExtras={ + + {subsection.show_duplication_check ? : } + + + + setRef(upButtonId, el)} + > + + + setRef(downButtonId, el)} + > + + + } + > + + + updateAutoIdentifier( + autoIdentifier, + sectionPath, + newName, + )} + /> + + + + + + + + + + { + if (value) { + updateAutoIdentifier( + true, + sectionPath, + get(stagedSectionValues, `${sectionPath}.title`), + ); + } + }} + /> + + {isApplicantSecondLevelSubsection && + + } + + {children} + ; +}; const EditPlotApplicationSectionFormSubsection: React$ComponentType = ({ sectionPath, @@ -146,19 +474,59 @@ const EditPlotApplicationSectionFormSubsection: React$ComponentType { const subsection = get(stagedSectionValues, sectionPath); + if (Object.keys(subsection || {}).length === 0) { + return null; + } + + const fieldIdentifiers = subsection.fields.map((field) => field.identifier); + const Wrapper = (level > 1) ? EditPlotApplicationSectionFormSubsectionSecondLevelWrapper : EditPlotApplicationSectionFormSubsectionFirstLevelWrapper; - return + return { + if (value && uniq(value.map((v) => v.identifier)).length < value.length) { + return 'Kahdella kentällä ei saa olla samaa sisäistä tunnusta!'; + } + }} /> { + if (value && uniq(value.map((v) => v.identifier)).length < value.length) { + return 'Kahdella osiolla ei saa olla samaa sisäistä tunnusta!'; + } + }} /> ; }; @@ -176,6 +554,7 @@ type OwnProps = { onClose: Function, onSubmit: Function, sectionIndex: number, + isOpen: boolean, }; type Props = { @@ -187,7 +566,13 @@ type Props = { parentFormSection: Object, initialize: Function, form: string, - stagedSectionValues: Object + stagedSectionValues: Object, + change: Function, + collapseStates: {[key: string]: boolean}, + setSectionEditorCollapseState: Function, + clearSectionEditorCollapseStates: Function, + initializeSectionEditorCollapseStates: Function, + fieldTypeMapping: Object, } class EditPlotApplicationSectionForm extends Component { @@ -198,44 +583,56 @@ class EditPlotApplicationSectionForm extends Component { } setFocus = (): void => { - if(this.firstField) { + if (this.firstField) { this.firstField.focus(); } } componentDidMount(): void { - const { - sectionIndex, - parentFormSection, - initialize, - } = this.props; + const {sectionIndex} = this.props; + if (sectionIndex !== undefined) { - initialize({ - section: parentFormSection, - }); + this.initializeData(); } } componentDidUpdate(prevProps: Props): void { const { sectionIndex, - parentFormSection, - initialize, + isOpen, } = this.props; - if (sectionIndex !== prevProps.sectionIndex) { - initialize({ - section: parentFormSection, - }); + if (sectionIndex !== prevProps.sectionIndex || (isOpen && !prevProps.isOpen)) { + this.initializeData(); } } + initializeData = (): void => { + const { + parentFormSection, + initialize, + clearSectionEditorCollapseStates, + initializeSectionEditorCollapseStates, + fieldTypeMapping, + } = this.props; + + const {sectionData, collapseInitialState} = getInitialFormSectionEditorData(fieldTypeMapping, parentFormSection); + + clearSectionEditorCollapseStates(); + initialize({ + section: sectionData, + identifier: parentFormSection.identifier, + }); + + initializeSectionEditorCollapseStates(collapseInitialState); + }; + handleSubmit = (): void => { const { onSubmit, onClose, stagedSectionValues, } = this.props; - onSubmit(stagedSectionValues.section); + onSubmit(transformCommittedFormSectionEditorData(stagedSectionValues.section)); onClose(); }; @@ -247,6 +644,9 @@ class EditPlotApplicationSectionForm extends Component { parentFormSection, stagedSectionValues, form, + change, + collapseStates, + setSectionEditorCollapseState, } = this.props; if (!parentFormSection) { @@ -254,50 +654,63 @@ class EditPlotApplicationSectionForm extends Component { } return ( - - - {parentFormSection.title} - - - - + {({dispatch}) => + + {parentFormSection.title} + + + + + + + + + + + + - - - - - - - - - - - + + } + ); } } @@ -314,9 +727,16 @@ export default (flowRight( attributes: getFormAttributes(state), parentFormSection: parentSelector(state, `form.sections[${props.sectionIndex}]`), stagedSectionValues: getFormValues(formName)(state), + collapseStates: getSectionEditorCollapseStates(state), + fieldTypeMapping: getFieldTypeMapping(state), }; }, - null, + { + change, + setSectionEditorCollapseState, + clearSectionEditorCollapseStates, + initializeSectionEditorCollapseStates, + }, null, {forwardRef: true} ), diff --git a/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionModal.js b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionModal.js index 830795654..f1aedf235 100644 --- a/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionModal.js +++ b/src/plotSearch/components/plotSearchSections/application/EditPlotApplicationSectionModal.js @@ -32,6 +32,10 @@ class EditPlotApplicationSectionModal extends Component { sectionIndex, } = this.props; + if (sectionIndex === -1) { + return null; + } + return ( { onClose={onClose} onSubmit={onSubmit} sectionIndex={sectionIndex} + isOpen={isOpen} /> ); diff --git a/src/plotSearch/components/plotSearchSections/application/SectionField.js b/src/plotSearch/components/plotSearchSections/application/SectionField.js deleted file mode 100644 index 81cd14a99..000000000 --- a/src/plotSearch/components/plotSearchSections/application/SectionField.js +++ /dev/null @@ -1,106 +0,0 @@ -// @flow -import React, {Fragment, useState} from 'react'; -import {connect} from 'react-redux'; -import {Row, Column} from 'react-foundation'; -import get from 'lodash/get'; -import {formValueSelector} from 'redux-form'; - -import FormField from '$components/form/FormField'; -import InfoIcon from '$components/icons/InfoIcon'; -import Tooltip from '$components/tooltip/Tooltip'; -import {FormNames} from '$src/enums'; -import TooltipToggleButton from '$components/tooltip/TooltipToggleButton'; -import TooltipWrapper from '$components/tooltip/TooltipWrapper'; -import {getFormAttributes} from '$src/application/selectors'; - -import type {Attributes} from '$src/types'; - -type OwnProps = { - disabled: boolean, - field: any, -}; - -type Props = { - ...OwnProps, - attributes: Attributes, - fieldValues: Object, -} - -const SectionField = ({ - disabled, - field, - attributes, - fieldValues, -}: Props) => { - const [isHintPopupOpen, setIsHintPopupOpen] = useState(false); - - return ( - - - - - - {fieldValues.hint_text && - - setIsHintPopupOpen(true)}> - - - setIsHintPopupOpen(false)}> - {fieldValues.hint_text} - - - } - - - - - - - ); -}; - -const selector = formValueSelector(FormNames.PLOT_SEARCH_APPLICATION_SECTION_STAGING); - -export default (connect( - (state, props: OwnProps) => { - return { - attributes: getFormAttributes(state), - fieldValues: selector(state, props.field), - }; - } -)(SectionField): React$ComponentType); diff --git a/src/plotSearch/components/plotSearchSections/application/_application.scss b/src/plotSearch/components/plotSearchSections/application/_application.scss index 56a39fb54..0230e5623 100644 --- a/src/plotSearch/components/plotSearchSections/application/_application.scss +++ b/src/plotSearch/components/plotSearchSections/application/_application.scss @@ -58,6 +58,8 @@ } .edit-plot-application-section-form__section { + border-bottom: 1px solid $gray-separator; + .section-field { padding: .75rem 0; margin-right: rem-calc(-10px); @@ -72,6 +74,11 @@ justify-content: flex-start; } + > .form-field__error-block { + display: block; + margin: rem-calc(10px) rem-calc(15px); + } + + .section-field { border-top: 0; } @@ -81,9 +88,33 @@ } } + .section-editor, + .section-field-editor { + padding: .75rem 0; + margin-right: rem-calc(-10px); + margin-left: rem-calc(-10px); + border-bottom: 1px solid $gray-separator; + + > * { + .row .form-field { + margin-left: 0; + } + + > .form-field__error-block { + display: block; + margin: rem-calc(10px) rem-calc(15px); + } + } + } + .section-editor { + background-color: $light-gray; + } + .section-field-editor { + background-color: $white; + } + .collapse__header { background-color: $light-gray; - border-bottom: 1px solid $gray-separator; } .collapse.collapse__secondary > .collapse__content > .collapse__content-wrapper, @@ -181,8 +212,22 @@ } } - > .collapse__content > .collapse__content-wrapper > .edit-plot-application-form__section-fields-container .columns:first-child .form-field { - margin-left: ($level - 1) * $section-indent-amount; + > .collapse__content > .collapse__content-wrapper { + > .edit-plot-application-form__section-fields-container { + > .section-field .columns:first-child { + .form-field, .mvj-button { + margin-left: ($level - 1) * $section-indent-amount; + } + } + + > .section-field-editor { + padding-left: ($level - 1) * $section-indent-amount; + } + } + + > .section-editor { + padding-left: ($level - 1) * $section-indent-amount; + } } } } @@ -195,4 +240,26 @@ .edit-plot-application-section-form__field-required-field { margin-right: rem-calc(2px); } + + .icon-button-component { + .icons__move-up { + margin-left: rem-calc(4px); + } + + .icons, .icons__trash #Artboard { + fill: $black; + } + + &:hover { + .icons, .icons__trash #Artboard { + fill: $blue; + } + } + + &:disabled { + .icons, .icons__trash #Artboard { + fill: $medium-gray !important; + } + } + } } diff --git a/src/plotSearch/constants.js b/src/plotSearch/constants.js index bd2378c19..da6892d2f 100644 --- a/src/plotSearch/constants.js +++ b/src/plotSearch/constants.js @@ -1,6 +1,7 @@ // @flow import {TableSortOrder} from '$src/enums'; import {PlotSearchStageTypes} from '$src/plotSearch/enums'; +import type {ProtectedFormPathsSections} from '$src/plotSearch/types'; /** * Default plotSearch states value for plotSearch list search @@ -44,3 +45,93 @@ export const FIELDS_LOCKED_FOR_EDITING = [ 'modified_at', 'form', ]; + +export const FieldTypeFeatures = { + FREEFORM_DEFAULT_VALUE: 'free-form-default-value', + LIST_SELECTION_DEFAULT_VALUE: 'list-selection-default-value', + MULTIPLE_SELECTION_OPTIONS: 'multiple-selection-options', + SINGLE_SELECTION_OPTIONS: 'single-selection-options', + TEXT_AREA_INPUT: 'text-area-input', + UNCHANGEABLE_VALUE: 'unchangeable-value', +}; + +export const FieldTypeLabels = { + textbox: 'Tekstikenttä', + textarea: 'Tekstialue', + dropdown: 'Pudotusvalikko', + checkbox: 'Valintaruutu', + radiobutton: 'Radiopainike', + radiobuttoninline: 'Radiopainike linjassa', + uploadfiles: 'Tiedoston lataus', + fractional: 'Murtoluku', + hidden: 'Piilotettu', +}; + +export const FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME: {[string]: Array} = { + textbox: [ + FieldTypeFeatures.FREEFORM_DEFAULT_VALUE, + ], + textarea: [ + FieldTypeFeatures.FREEFORM_DEFAULT_VALUE, + FieldTypeFeatures.TEXT_AREA_INPUT, + ], + dropdown: [ + FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE, + FieldTypeFeatures.SINGLE_SELECTION_OPTIONS, + ], + checkbox: [ + FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE, + FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS, + ], + radiobutton: [ + FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE, + FieldTypeFeatures.SINGLE_SELECTION_OPTIONS, + ], + radiobuttoninline: [ + FieldTypeFeatures.LIST_SELECTION_DEFAULT_VALUE, + FieldTypeFeatures.SINGLE_SELECTION_OPTIONS, + ], + uploadfiles: [], + fractional: [ + FieldTypeFeatures.FREEFORM_DEFAULT_VALUE, + ], + hidden: [ + FieldTypeFeatures.FREEFORM_DEFAULT_VALUE, + FieldTypeFeatures.UNCHANGEABLE_VALUE, + ], +}; + +export const PROTECTED_FORM_PATHS: ProtectedFormPathsSections = { + 'hakijan-tiedot': { + subsections: { + 'henkilon-tiedot': { + subsections: {}, + fields: [ + 'etunimi', + 'Sukunimi', + 'henkilotunnus', + ], + }, + 'yrityksen-tiedot': { + subsections: {}, + fields: [ + 'yrityksen-nimi', + 'y-tunnus', + ], + }, + }, + fields: [ + 'hakija', + ], + fieldChoices: { + hakija: [ + '1', + '2', + ], + }, + }, + 'kohteen-tiedot': { + subsections: {}, + fields: [], + }, +}; diff --git a/src/plotSearch/helpers.js b/src/plotSearch/helpers.js index f4f7c8397..3b4abd3ba 100644 --- a/src/plotSearch/helpers.js +++ b/src/plotSearch/helpers.js @@ -1,6 +1,8 @@ // @flow import get from 'lodash/get'; import isEmpty from 'lodash/isEmpty'; +import {kebabCase} from 'lodash/string'; +import {cloneDeep} from 'lodash/lang'; import {formValueSelector} from 'redux-form'; import {getContentUser} from '$src/users/helpers'; @@ -8,10 +10,15 @@ import {removeSessionStorageItem} from '$util/storage'; import {FormNames} from '$src/enums'; import {formatDate, getApiResponseResults} from '$util/helpers'; import {PlotSearchTargetType} from '$src/plotSearch/enums'; +import { + FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME, + FieldTypeFeatures, + PROTECTED_FORM_PATHS, +} from '$src/plotSearch/constants'; -import type {PlotSearch, PlotSearchState} from '$src/plotSearch/types'; +import type {PlotSearch, PlotSearchState, ProtectedFormPathsSectionNode} from '$src/plotSearch/types'; import type {Attributes} from '$src/types'; -import type {Form} from '$src/application/types'; +import type {Form, FormField, FormSection} from '$src/application/types'; /** * Get plotSearch basic information content @@ -208,3 +215,232 @@ export const getInfoLinkLanguageDisplayText = (key: string, attributes: Attribut return languages?.find((language) => language.value === key)?.display_name || key; }; +class UniqueTemporaryIdFactory { + i; + + constructor() { + this.i = 0; + } + + next() { + this.i++; + return 't' + this.i.toString(); + } +} + +const uniqueTemporaryIdFactory = new UniqueTemporaryIdFactory(); +const getUniqueTemporaryId: () => string = () => uniqueTemporaryIdFactory.next(); + +export const getDefaultNewFormSection = (peerIdentifiers: Array): Object => ({ + temporary_id: getUniqueTemporaryId(), + identifier: generateSectionIdentifierFromName('', peerIdentifiers), + title: '', + title_en: '', + title_sv: '', + visible: true, + required: false, + add_new_allowed: false, + show_duplication_check: true, + fields: [], + subsections: [], + auto_fill_identifier: true, + is_protected: false, +}); + +export const getDefaultNewFormField = (peerIdentifiers: Array): Object => ({ + temporary_id: getUniqueTemporaryId(), + identifier: generateFieldIdentifierFromName('', peerIdentifiers), + label: '', + label_en: '', + label_sv: '', + hint_text: '', + hint_text_en: '', + hint_text_sv: '', + enabled: true, + required: false, + type: null, + default_value: '', + choices: [], + auto_fill_identifier: true, + is_protected: false, + protected_values: [], +}); + +export const getInitialFormSectionEditorData = (fieldTypeMapping: Object, section: FormSection): Object => { + const protectedPathsRoot = PROTECTED_FORM_PATHS; + + const collapseInitialState = {}; + + const addUIPropertiesToField = ( + field: FormField, + peerFieldIdentifiers: Array, + isProtected: boolean, + protectedValues: Array, + ): Object => { + let defaultValue = field.default_value; + const fieldFeatures = FIELD_TYPE_FEATURES_BY_FIELD_TYPE_NAME[fieldTypeMapping[field.type]] || []; + let temporaryId; + + if (field.id) { + collapseInitialState[`field-${field.id}`] = false; + } else { + temporaryId = getUniqueTemporaryId(); + collapseInitialState[`field-${temporaryId}`] = false; + } + + if (typeof defaultValue === 'string' && fieldFeatures.includes(FieldTypeFeatures.MULTIPLE_SELECTION_OPTIONS)) { + try { + defaultValue = JSON.parse(defaultValue); + if (!(defaultValue instanceof Array)) { + defaultValue = [defaultValue]; + } + } catch { + defaultValue = [defaultValue]; + } + } + return { + ...cloneDeep(field), + temporary_id: temporaryId, + auto_fill_identifier: + generateFieldIdentifierFromName(field.label, peerFieldIdentifiers) === field.identifier, + auto_fill_choice_values: + protectedValues.length === 0 && field.choices.every( + (choice, index) => choice.value.toString() === (index + 1).toString() + ), + is_protected: isProtected, + protected_values: protectedValues, + default_value: defaultValue, + }; + }; + const addUIPropertiesToSection = ( + section: FormSection, + peerSectionIdentifiers: Array, + protectedPaths: ProtectedFormPathsSectionNode, + ): Object => { + const subsectionIdentifiers = section.subsections.map((subsection) => subsection.identifier); + const fieldIdentifiers = section.fields.map((field) => field.identifier); + let temporaryId; + + if (section.id) { + collapseInitialState[`section-${section.id}`] = true; + } else { + temporaryId = getUniqueTemporaryId(); + collapseInitialState[`section-${temporaryId}`] = true; + } + + return { + ...cloneDeep(section), + temporary_id: temporaryId, + is_protected: !!protectedPaths.fields || !!protectedPaths.subsections, + subsections: section.subsections.map( + (subsection, index) => addUIPropertiesToSection( + subsection, + subsectionIdentifiers.filter((_, i) => i !== index), + protectedPaths?.subsections?.[subsection.identifier] || {}, + )), + fields: section.fields.map( + (field, index) => addUIPropertiesToField( + field, + fieldIdentifiers.filter((_, i) => i !== index), + protectedPaths?.fields?.includes(field.identifier) || false, + protectedPaths?.fieldChoices?.[field.identifier] || [], + )), + auto_fill_identifier: + generateFieldIdentifierFromName( + section.title, + peerSectionIdentifiers, + ) === section.identifier, + }; + }; + + return { + sectionData: addUIPropertiesToSection(section, [], protectedPathsRoot[section.identifier] || {}), + collapseInitialState, + }; +}; + +export const generateIdentifierFromName = (name: string, peerIdentifiers: Array, emptyBaseName: string): string => { + const baseAutoName = kebabCase(name.length > 0 ? name : emptyBaseName); + let autoName = baseAutoName; + let idx = 0; + + while (peerIdentifiers.includes(autoName)) { + if (idx > 1e5) { + throw new Error('too many fields with the same label'); + } + + ++idx; + autoName = `${baseAutoName}-${idx}`; + } + + return autoName; +}; + +export const generateSectionIdentifierFromName = (name: string, peerIdentifiers: Array): string => + generateIdentifierFromName(name, peerIdentifiers, 'Uusi osio'); + +export const generateFieldIdentifierFromName = (name: string, peerIdentifiers: Array): string => + generateIdentifierFromName(name, peerIdentifiers, 'Uusi kenttä'); + +export const transformCommittedFormSectionEditorData = (section: Object): FormSection => { + const transformChoice = (choice: Object, index: number): Object => { + const { + text, + default_value, + ...rest + } = choice; + + return { + ...cloneDeep(rest), + sort_order: index, + text, + default_value: default_value instanceof Array ? JSON.stringify(default_value) : default_value, + text_fi: text, + }; + }; + const transformField = (field: Object, index: number): Object => { + const { + // UI fields to be removed + // eslint-disable-next-line no-unused-vars + auto_fill_identifier, auto_fill_choice_values, is_protected, protected_values, temporary_id, + + choices, + label, + ...rest + } = field; + + return { + ...cloneDeep(rest), + sort_order: index, + choices: choices?.map(transformChoice) || null, + label, + label_fi: label, + }; + }; + const transformSection = (section: Object, index: number): Object => { + const { + // UI fields to be removed + // eslint-disable-next-line no-unused-vars + auto_fill_identifier, sort_order, is_protected, temporary_id, + + subsections, + fields, + label, + help_text, + ...rest + } = section; + + return { + ...cloneDeep(rest), + subsections: subsections.map(transformSection), + fields: fields.map(transformField), + sort_order: index, + label, + help_text, + label_fi: label, + help_text_fi: help_text, + }; + }; + + return transformSection(section, section.sort_order); +}; diff --git a/src/plotSearch/reducer.js b/src/plotSearch/reducer.js index 49fab758b..00663d3a7 100644 --- a/src/plotSearch/reducer.js +++ b/src/plotSearch/reducer.js @@ -275,6 +275,18 @@ const isCreatingDirectReservationLinkReducer: Reducer = handleActions({ ['mvj/plotSearch/DIRECT_RESERVATION_LINK_CREATION_FAILED']: () => false, }, false); +const sectionEditorCollapseStatesReducer: Reducer<{[key: string]: boolean}> = handleActions({ + ['mvj/plotSearch/CLEAR_SECTION_EDITOR_COLLAPSE_STATES']: () => ({}), + ['mvj/plotSearch/SET_SECTION_EDITOR_COLLAPSE_STATE']: (state: {[key: string]: boolean}, {payload}) => ({ + ...state, + [payload.key]: payload.state, + }), + ['mvj/plotSearch/INITIALIZE_SECTION_EDITOR_COLLAPSE_STATES']: (state, {payload}) => ({ + ...state, + ...payload, + }), +}, {}); + export default (combineReducers({ attributes: attributesReducer, @@ -309,4 +321,5 @@ export default (combineReducers({ isFetchingReservationIdentifierUnitLists: isFetchingReservationIdentifierUnitListsReducer, reservationIdentifierUnitLists: reservationIdentifierUnitListsReducer, isCreatingDirectReservationLink: isCreatingDirectReservationLinkReducer, + sectionEditorCollapseStates: sectionEditorCollapseStatesReducer, }): CombinedReducer>); diff --git a/src/plotSearch/selectors.js b/src/plotSearch/selectors.js index 241445b06..aaf9c8f12 100644 --- a/src/plotSearch/selectors.js +++ b/src/plotSearch/selectors.js @@ -140,3 +140,6 @@ export const getReservationIdentifierUnitLists: Selector = (state: export const getIsCreatingDirectReservationLink: Selector = (state: RootState): boolean => state.plotSearch.isCreatingDirectReservationLink; + +export const getSectionEditorCollapseStates: Selector<{[key: string]: boolean}, void> = (state: RootState): {[key: string]: boolean} => + state.plotSearch.sectionEditorCollapseStates; diff --git a/src/plotSearch/spec.js b/src/plotSearch/spec.js index e08587ce2..ba8e054d0 100644 --- a/src/plotSearch/spec.js +++ b/src/plotSearch/spec.js @@ -88,6 +88,7 @@ const baseState: PlotSearchState = { isFetchingReservationIdentifierUnitLists: false, reservationIdentifierUnitLists: null, isCreatingDirectReservationLink: false, + sectionEditorCollapseStates: {}, }; diff --git a/src/plotSearch/types.js b/src/plotSearch/types.js index 7d8910426..2c8e5f047 100644 --- a/src/plotSearch/types.js +++ b/src/plotSearch/types.js @@ -35,6 +35,7 @@ export type PlotSearchState = { isFetchingReservationIdentifierUnitLists: boolean, reservationIdentifierUnitLists: null | Object, isCreatingDirectReservationLink: boolean, + sectionEditorCollapseStates: {[key: string]: boolean}, }; export type CustomDetailedPlan = Object; @@ -48,6 +49,14 @@ export type FetchSinglePlotSearchAfterEditPayload = { callbackFunctions?: Array, } +export type ProtectedFormPathsSections = {[key: string]: ProtectedFormPathsSectionNode}; + +export type ProtectedFormPathsSectionNode = { + subsections?: ProtectedFormPathsSections, + fields?: Array, + fieldChoices?: {[key: string]: Array}, +}; + export type FetchAttributesAction = Action<'mvj/plotSearch/FETCH_ATTRIBUTES', void>; export type ReceiveAttributesAction = Action<'mvj/plotSearch/RECEIVE_ATTRIBUTES', Attributes>; export type ReceiveMethodsAction = Action<'mvj/plotSearch/RECEIVE_METHODS', Methods>; @@ -121,3 +130,7 @@ export type ReservationIdentifierUnitListsNotFoundAction = Action<'mvj/plotSearc export type CreateDirectReservationLinkAction = Action<'mvj/plotSearch/CREATE_DIRECT_RESERVATION_LINK', {data: Object, callBack: Function}>; export type DirectReservationLinkCreatedAction = Action<'mvj/plotSearch/DIRECT_RESERVATION_LINK_CREATED', void>; export type DirectReservationLinkCreationFailedAction = Action<'mvj/plotSearch/DIRECT_RESERVATION_LINK_CREATION_FAILED', any>; + +export type ClearSectionEditorCollapseStatesAction = Action<'mvj/plotSearch/CLEAR_SECTION_EDITOR_COLLAPSE_STATES', void>; +export type SetSectionEditorCollapseStateAction = Action<'mvj/plotSearch/SET_SECTION_EDITOR_COLLAPSE_STATE', {key: string, state: boolean}>; +export type InitializeSectionEditorCollapseStatesAction = Action<'mvj/plotSearch/INITIALIZE_SECTION_EDITOR_COLLAPSE_STATES', {[key: string]: boolean}>;