All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.ocs.dynamo.ui.component.ElementCollectionGrid Maven / Gradle / Ivy

/*
   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
 */
package com.ocs.dynamo.ui.component;

import com.ocs.dynamo.domain.AbstractEntity;
import com.ocs.dynamo.domain.model.AttributeModel;
import com.ocs.dynamo.service.MessageService;
import com.ocs.dynamo.service.ServiceLocatorFactory;
import com.ocs.dynamo.ui.CanAssignEntity;
import com.ocs.dynamo.ui.NestedComponent;
import com.ocs.dynamo.ui.composite.layout.FormOptions;
import com.ocs.dynamo.ui.converter.ConverterFactory;
import com.ocs.dynamo.ui.converter.TrimSpacesConverter;
import com.ocs.dynamo.ui.utils.ConvertUtils;
import com.ocs.dynamo.ui.utils.VaadinUtils;
import com.ocs.dynamo.util.SystemPropertyUtils;
import com.ocs.dynamo.utils.ClassUtils;
import com.ocs.dynamo.utils.NumberUtils;
import com.vaadin.flow.component.Component;
import com.vaadin.flow.component.button.Button;
import com.vaadin.flow.component.customfield.CustomField;
import com.vaadin.flow.component.grid.Grid;
import com.vaadin.flow.component.grid.Grid.Column;
import com.vaadin.flow.component.grid.Grid.SelectionMode;
import com.vaadin.flow.component.icon.VaadinIcon;
import com.vaadin.flow.component.orderedlayout.HorizontalLayout;
import com.vaadin.flow.component.orderedlayout.VerticalLayout;
import com.vaadin.flow.component.textfield.TextField;
import com.vaadin.flow.data.binder.*;
import com.vaadin.flow.data.binder.Binder.BindingBuilder;
import com.vaadin.flow.data.provider.ListDataProvider;
import com.vaadin.flow.data.validator.BigDecimalRangeValidator;
import com.vaadin.flow.data.validator.IntegerRangeValidator;
import com.vaadin.flow.data.validator.LongRangeValidator;
import com.vaadin.flow.data.validator.StringLengthValidator;
import com.vaadin.flow.function.ValueProvider;
import lombok.Getter;
import lombok.Setter;
import org.apache.commons.lang3.StringUtils;

import java.io.Serializable;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Consumer;

/**
 * A grid for editing a collection of simple values stored in a JPA collection
 * table
 * 
 * @author Bas Rutten
 *
 * @param  the type of the key of the entity on which the element collection
 *             is defined
 * @param   the type of the entity on which the collection is stored
 * @param   the type of the elements in the collection
 */
public class ElementCollectionGrid, T extends Serializable>
		extends CustomField> implements NestedComponent, CanAssignEntity {

	private static final long serialVersionUID = -1203245694503350276L;

	private static final String VALUE_TOO_HIGH = "ocs.value.too.high";

	private static final String VALUE_TOO_LONG = "ocs.value.too.long";

	private static final String VALUE_TOO_LOW = "ocs.value.too.low";

	private static final String VALUE_TOO_SHORT = "ocs.value.too.short";

	@Getter
	private Button addButton;

	private final AttributeModel attributeModel;

	/**
	 * Map of bindings. Contains one binding for each row
	 */
	private final Map, Binder>> binders = new HashMap<>();

	/**
	 * The entity on to which to store the values. This should not normally be
	 * needed but for some reason the normal binding mechanisms don't work, so we
	 * need to set the values ourselves
	 */
	@Getter
	private U entity;

	@Getter
	private final FormOptions formOptions;

	@Getter
	private Grid> grid;

	private final String gridHeight = SystemPropertyUtils.getDefaultEditGridHeight();

	private final MessageService messageService;

	private final ListDataProvider> provider;

	@Getter
	@Setter
	private Consumer postProcessButtonBar;

	/**
	 * The currently selected item in the grid
	 */
	@Getter
	@Setter
	private T selectedItem;

	/**
	 * Constructor
	 * 
	 * @param attributeModel the attribute model
	 * @param formOptions    the form options
	 */
	public ElementCollectionGrid(AttributeModel attributeModel, FormOptions formOptions) {
		this.messageService = ServiceLocatorFactory.getServiceLocator().getMessageService();
		this.formOptions = formOptions;
		this.attributeModel = attributeModel;
		this.provider = new ListDataProvider<>(new ArrayList<>());
		setSizeFull();
		initContent();
	}

	private void addBigDecimalConverters(BindingBuilder, String> builder) {
		BindingBuilder, BigDecimal> iBuilder = builder
				.withConverter(ConverterFactory.createBigDecimalConverter(attributeModel.isCurrency(),
						attributeModel.isPercentage(), attributeModel.useThousandsGroupingInEditMode(),
						attributeModel.getPrecision(), attributeModel.getCurrencySymbol()));
		if (attributeModel.getMaxValue() != null) {
			iBuilder.withValidator(new BigDecimalRangeValidator(message(VALUE_TOO_HIGH), null,
					BigDecimal.valueOf(attributeModel.getMaxValue())));
		}
		if (attributeModel.getMinValue() != null) {
			iBuilder.withValidator(new BigDecimalRangeValidator(message(VALUE_TOO_LOW),
					BigDecimal.valueOf(attributeModel.getMinValue()), null));
		}
	}

	private void addIntegerConverters(BindingBuilder, String> builder) {
		BindingBuilder, Integer> iBuilder = builder
				.withConverter(ConverterFactory.createIntegerConverter(attributeModel.useThousandsGroupingInEditMode(),
						attributeModel.isPercentage()));
		if (attributeModel.getMaxValue() != null) {
			iBuilder.withValidator(new IntegerRangeValidator(message(VALUE_TOO_HIGH, attributeModel.getMaxValue()),
					null, attributeModel.getMaxValue().intValue()));
		}
		if (attributeModel.getMinValue() != null) {
			iBuilder.withValidator(new IntegerRangeValidator(message(VALUE_TOO_LOW, attributeModel.getMinValue()),
					attributeModel.getMinValue().intValue(), null));
		}
	}

	private void addLongConverters(BindingBuilder, String> builder) {
		BindingBuilder, Long> iBuilder = builder.withConverter(ConverterFactory
				.createLongConverter(attributeModel.useThousandsGroupingInEditMode(), attributeModel.isPercentage()));
		if (attributeModel.getMaxValue() != null) {
			iBuilder.withValidator(new LongRangeValidator(message(VALUE_TOO_HIGH, attributeModel.getMaxValue()), null,
					attributeModel.getMaxValue()));
		}
		if (attributeModel.getMinValue() != null) {
			iBuilder.withValidator(new LongRangeValidator(message(VALUE_TOO_LOW, attributeModel.getMinValue()),
					attributeModel.getMinValue(), null));
		}
	}

	private void addStringConverters(BindingBuilder, String> builder) {
		if (attributeModel.getMaxLength() != null) {
			builder.withValidator(new StringLengthValidator(message(VALUE_TOO_LONG, attributeModel.getMaxLength()),
					null, attributeModel.getMaxLength()));
		}
		if (attributeModel.getMinLength() != null) {
			builder.withValidator(new StringLengthValidator(message(VALUE_TOO_SHORT, attributeModel.getMinLength()),
					attributeModel.getMinLength(), null));
		}
		if (attributeModel.isTrimSpaces()) {
			builder.withConverter(new TrimSpacesConverter());
		}
	}

	@Override
	public void assignEntity(U entity) {
		this.entity = entity;
		provider.refreshAll();

	}

	/**
	 * Constructs the button that is used for adding new items
	 *
	 * @param buttonBar the button bar to add the button to
	 */
	protected void constructAddButton(HorizontalLayout buttonBar) {
		addButton = new Button(messageService.getMessage("ocs.add", VaadinUtils.getLocale()));
		addButton.setIcon(VaadinIcon.PLUS.create());
		addButton.addClickListener(event -> {

			ValueHolder vh = new ValueHolder<>(null);
			provider.getItems().add(vh);

			@SuppressWarnings({ "rawtypes", "unchecked" })
			Binder> binder = new BeanValidationBinder(ValueHolder.class);
			binder.setBean(vh);
			binders.put(vh, binder);
			provider.refreshAll();

		});
		buttonBar.add(addButton);
	}

	/**
	 * Constructs the button bar
	 * 
	 * @param parent the parent layout to which to add the button bar
	 */
	protected void constructButtonBar(VerticalLayout parent) {
		HorizontalLayout buttonBar = new DefaultHorizontalLayout();
		parent.add(buttonBar);

		if (formOptions.isShowAddButton()) {
			constructAddButton(buttonBar);
		}

		if (postProcessButtonBar != null) {
			postProcessButtonBar.accept(buttonBar);
		}
	}

	/**
	 * Constructs the column that holds the "remove" button
	 */
	private void constructRemoveColumn() {
		if (formOptions.isShowRemoveButton()) {
			String removeMsg = message("ocs.detail.remove");
			grid.addComponentColumn((ValueProvider, Component>) entity -> {
				Button remove = new Button();
				remove.setIcon(VaadinIcon.TRASH.create());
				remove.addClickListener(event -> {
					binders.remove(entity);
					provider.getItems().remove(entity);
					provider.refreshAll();
				});
				return remove;
			}).setHeader(removeMsg).setKey(removeMsg);
		}
	}

	@Override
	protected Collection generateModelValue() {
		return getValueInner();
	}

	public Integer getMaxLength() {
		return attributeModel.getMaxLength();
	}

	public Long getMaxValue() {
		return attributeModel.getMaxValue();
	}

	public Integer getMinLength() {
		return attributeModel.getMinLength();
	}

	public Long getMinValue() {
		return attributeModel.getMinValue();
	}

	@Override
	public Collection getValue() {
		return getValueInner();
	}

	private Collection getValueInner() {
		Collection col = provider.getItems().stream().map(ValueHolder::getValue).toList();
		Collection converted = ConvertUtils.convertCollection(col, attributeModel);
		if (entity != null) {
			ClassUtils.setFieldValue(entity, attributeModel.getPath(), converted);
		}
		return converted;
	}

	/**
	 * Constructs the actual component
	 */
	protected void initContent() {

		VerticalLayout main = new VerticalLayout();

		grid = new Grid<>();
		grid.setItems(provider);
		main.add(grid);

		Column> column = grid.addComponentColumn(vh -> {
			TextField tf = new TextField("");
			Binder> binder = binders.get(vh);

			BindingBuilder, String> builder = binder.forField(tf);
			builder.withNullRepresentation("");

			// custom validator since the normal one apparently doesn't work properly here
			// (must be added before the converters)
			Validator notEmptyValidator = (String value, ValueContext v) -> {
				if (StringUtils.isEmpty(value)) {
					return ValidationResult
							.error(messageService.getMessage("ocs.may.not.be.null", VaadinUtils.getLocale()));
				}
				return ValidationResult.ok();
			};
			builder.asRequired(notEmptyValidator);

			if (String.class.equals(attributeModel.getMemberType())) {
				addStringConverters(builder);
			} else if (NumberUtils.isInteger(attributeModel.getMemberType())) {
				addIntegerConverters(builder);
			} else if (NumberUtils.isLong(attributeModel.getMemberType())) {
				addLongConverters(builder);
			} else if (BigDecimal.class.equals(attributeModel.getMemberType())) {
				addBigDecimalConverters(builder);
			}
			builder.bind("value");
			tf.setSizeFull();
			return tf;
		});

		column.setHeader(messageService.getMessage("ocs.value", VaadinUtils.getLocale()));

		grid.setHeight(gridHeight);
		grid.setSelectionMode(SelectionMode.SINGLE);

		// add a change listener (to make sure the buttons are correctly
		// enabled/disabled)
		grid.addSelectionListener(event -> {
			if (grid.getSelectedItems().iterator().hasNext()) {
				ValueHolder vh = grid.getSelectedItems().iterator().next();
				onSelect(vh.getValue());
			}
		});

		// add a remove button directly in the grid
		constructRemoveColumn();
		constructButtonBar(main);

		add(main);
	}

	private String message(String key, Object... values) {
		return messageService.getMessage(key, VaadinUtils.getLocale(), values);
	}

	/**
	 * Callback method that is executed after an item is selected in the grid
	 */
	protected void onSelect(T selected) {
		// overwrite in subclass if needed
	}

	public void setEntity(U entity) {
		this.entity = entity;
		provider.refreshAll();
	}

	@Override
	protected void setPresentationValue(Collection value) {
		provider.getItems().clear();
		provider.refreshAll();
		binders.clear();
		for (T entity : value) {
			ValueHolder vh = new ValueHolder<>(entity);
			provider.getItems().add(vh);
			@SuppressWarnings({ "rawtypes", "unchecked" })
			Binder> binder = new BeanValidationBinder(ValueHolder.class);
			binder.setBean(vh);
			binders.put(vh, binder);
		}
	}

	@Override
	public boolean validateAllFields() {
		return binders.entrySet().stream().anyMatch(entry -> !entry.getValue().validate().isOk());
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy