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

com.ocs.dynamo.ui.composite.grid.GridWrapper 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.composite.grid;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Supplier;

import com.ocs.dynamo.dao.FetchJoinInformation;
import com.ocs.dynamo.domain.AbstractEntity;
import com.ocs.dynamo.domain.model.AttributeModel;
import com.ocs.dynamo.domain.model.EntityModel;
import com.ocs.dynamo.service.BaseService;
import com.ocs.dynamo.ui.Searchable;
import com.ocs.dynamo.ui.composite.ComponentContext;
import com.ocs.dynamo.ui.composite.export.ExportDelegate;
import com.ocs.dynamo.ui.composite.layout.BaseCustomComponent;
import com.ocs.dynamo.ui.composite.layout.FormOptions;
import com.ocs.dynamo.ui.provider.QueryType;
import com.vaadin.flow.component.grid.Grid;
import com.vaadin.flow.component.grid.GridSortOrderBuilder;
import com.vaadin.flow.data.binder.Validator;
import com.vaadin.flow.data.converter.Converter;
import com.vaadin.flow.data.provider.DataProvider;
import com.vaadin.flow.data.provider.SortDirection;
import com.vaadin.flow.data.provider.SortOrder;
import com.vaadin.flow.function.SerializablePredicate;

import lombok.Getter;
import lombok.Setter;

public abstract class GridWrapper, U> extends BaseCustomComponent
		implements Searchable {

	private static final long serialVersionUID = -7839990909007399928L;

	/**
	 * The entity model used to create the container
	 */
	private EntityModel entityModel;

	/**
	 * The export service used for generating XLS and CSV exports
	 */
	private ExportDelegate exportDelegate = getService(ExportDelegate.class);

	/**
	 * The entity model to use when exporting
	 */
	private EntityModel exportEntityModel;

	/**
	 * The joins to use when exporting (needed when using exportmode FULL)
	 */
	private FetchJoinInformation[] exportJoins;

	/**
	 * The search filter that is applied to the grid
	 */
	private SerializablePredicate filter;

	/**
	 * The form options
	 */
	@Getter
	private FormOptions formOptions;

	@Getter
	private ComponentContext componentContext;

	/**
	 * The fetch joins to use when querying
	 */
	private FetchJoinInformation[] joins;

	/**
	 * The type of the query
	 */
	private final QueryType queryType;

	/**
	 * The service used to query the database
	 */
	private final BaseService service;

	@Getter
	@Setter
	private Map>> customConverters = new HashMap<>();

	@Getter
	@Setter
	private Map>> customValidators = new HashMap<>();

	/**
	 * The sort orders
	 */
	private List> sortOrders = new ArrayList<>();

	protected GridWrapper(BaseService service, EntityModel entityModel, QueryType queryType,
			FormOptions formOptions, ComponentContext componentContext, SerializablePredicate filter,
			List> sortOrders, FetchJoinInformation... joins) {
		setSpacing(false);
		setPadding(false);
		this.entityModel = entityModel;
		this.filter = filter;
		this.service = service;
		this.queryType = queryType;
		this.formOptions = formOptions;
		this.componentContext = componentContext;
		this.sortOrders = sortOrders != null ? sortOrders : new ArrayList<>();
		this.joins = joins;
	}

	/**
	 * Perform any actions that are necessary before carrying out a search
	 * 
	 * @param filter the current search filter
	 */
	protected SerializablePredicate beforeSearchPerformed(SerializablePredicate filter) {
		// overwrite in subclasses
		return null;
	}

	public abstract DataProvider> getDataProvider();

	public abstract int getDataProviderSize();

	/**
	 * @return the entityModel
	 */
	public EntityModel getEntityModel() {
		return entityModel;
	}

	public ExportDelegate getExportDelegate() {
		return exportDelegate;
	}

	public EntityModel getExportEntityModel() {
		return exportEntityModel;
	}

	public FetchJoinInformation[] getExportJoins() {
		return exportJoins;
	}

	protected SerializablePredicate getFilter() {
		return filter;
	}

	public abstract Grid getGrid();

	public FetchJoinInformation[] getJoins() {
		return joins;
	}

	public QueryType getQueryType() {
		return queryType;
	}

	public BaseService getService() {
		return service;
	}

	/**
	 * 
	 * @return the sort orders
	 */
	public List> getSortOrders() {
		return Collections.unmodifiableList(sortOrders);
	}

	protected List> initSortingAndFiltering() {

		List> fallBackOrders = new ArrayList<>();
		GridSortOrderBuilder builder = new GridSortOrderBuilder<>();
		boolean missing = false;

		if (getSortOrders() != null && !getSortOrders().isEmpty()) {
			missing = initExplicitSortOrders(fallBackOrders, builder);
		} else if (getEntityModel().getSortOrder() != null && !getEntityModel().getSortOrder().keySet().isEmpty()) {
			missing = initDefaultSortOrders(fallBackOrders, builder);
		}

		// use grid sorting only if all columns are present, otherwise use fallback
		if (!missing) {
			getGrid().sort(builder.build());
		}

		return fallBackOrders;

	}

	private boolean initDefaultSortOrders(List> fallBackOrders, GridSortOrderBuilder builder) {
		boolean missing = false;
		for (AttributeModel am : getEntityModel().getSortOrder().keySet()) {
			boolean asc = getEntityModel().getSortOrder().get(am);
			if (getGrid().getColumnByKey(am.getPath()) != null) {
				if (asc) {
					builder.thenAsc(getGrid().getColumnByKey(am.getPath()));
				} else {
					builder.thenDesc(getGrid().getColumnByKey(am.getPath()));
				}
			} else {
				missing = true;
			}
			fallBackOrders.add(new SortOrder<>(am.getActualSortPath(),
					asc ? SortDirection.ASCENDING : SortDirection.DESCENDING));
		}
		return missing;
	}

	private boolean initExplicitSortOrders(List> fallBackOrders, GridSortOrderBuilder builder) {
		boolean missing = false;
		for (SortOrder o : getSortOrders()) {
			if (getGrid().getColumnByKey((String) o.getSorted()) != null) {
				// only include column in sort order if it is present in the grid
				if (SortDirection.ASCENDING.equals(o.getDirection())) {
					builder.thenAsc(getGrid().getColumnByKey(o.getSorted().toString()));
				} else {
					builder.thenDesc(getGrid().getColumnByKey(o.getSorted().toString()));
				}
			} else {
				missing = true;
			}
			fallBackOrders.add(o);
		}
		return missing;
	}

	public abstract void reloadDataProvider();

	@Override
	@SuppressWarnings({ "unchecked", "rawtypes" })
	public void search(SerializablePredicate filter) {
		SerializablePredicate temp = beforeSearchPerformed(filter);
		this.filter = temp != null ? temp : filter;
		getGrid().getDataCommunicator().setDataProvider((DataProvider) getDataProvider(), temp != null ? temp : filter);
	}

	public void setExportEntityModel(EntityModel exportEntityModel) {
		this.exportEntityModel = exportEntityModel;
	}

	public void setExportJoins(FetchJoinInformation[] exportJoins) {
		this.exportJoins = exportJoins;
	}

	/**
	 * Sets the provided filter as the component filter and then refreshes the
	 * container
	 * 
	 * @param filter
	 */
	public void setFilter(SerializablePredicate filter) {
		this.filter = filter;
		search(filter);
	}

	public void setJoins(FetchJoinInformation[] joins) {
		this.joins = joins;
	}

	public void setSortOrders(List> sortOrders) {
		this.sortOrders = sortOrders;
	}
}