/** * Copyright 2014-2017 Riccardo Massera (TheCoder4.Eu) and Stephan Rauh (http://www.beyondjava.net). * * This file is part of BootsFaces. * * 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 net.bootsfaces.component.dataTable; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; import javax.el.ValueExpression; import javax.faces.application.ResourceDependencies; import javax.faces.application.ResourceDependency; import javax.faces.component.FacesComponent; import javax.faces.component.UIComponent; import javax.faces.component.UIComponentBase; import javax.faces.component.UIViewRoot; import javax.faces.component.behavior.ClientBehaviorHolder; import javax.faces.context.FacesContext; import javax.faces.event.FacesEvent; import net.bootsfaces.C; import net.bootsfaces.component.ajax.IAJAXComponent; import net.bootsfaces.listeners.AddResourcesListener; import net.bootsfaces.render.IContentDisabled; import net.bootsfaces.render.IResponsive; import net.bootsfaces.render.Tooltip; import net.bootsfaces.utils.BsfUtils; /** This class holds the attributes of <b:dataTable />. */ @ResourceDependencies({ @ResourceDependency(library = "bsf", name = "js/datatables.min.js", target = "body"), // @ResourceDependency(library = "bsf", name = // "js/datatables-bf-extensions.js", target = "body"), @ResourceDependency(library = "bsf", name = "css/datatables.min.css", target = "head") }) @FacesComponent(DataTable.COMPONENT_TYPE) public class DataTable extends DataTableCore implements IAJAXComponent, ClientBehaviorHolder, net.bootsfaces.render.IHasTooltip, IResponsive, IContentDisabled { public static final String COMPONENT_TYPE = C.BSFCOMPONENT + ".dataTable.DataTable"; public static final String COMPONENT_FAMILY = C.BSFCOMPONENT; public static final String DEFAULT_RENDERER = "net.bootsfaces.component.dataTable.DataTable"; private static final Collection<String> EVENT_NAMES = Collections.unmodifiableCollection(Arrays.asList("click", "dblclick", "dragstart", "dragover", "drop", "mousedown", "mousemove", "mouseout", "mouseover", "mouseup")); /** * This map is used to store the column sort information gathered during * rendering. */ private Map<Integer, String> columnSortOrder; /** * This array ist used to store the column information bits that are used to * initialize the columns using the columns attribute of datatables.net */ private List<String> columnInfo; public enum DataTablePropertyType { pageLength, searchTerm, currentPage } public DataTable() { setRendererType(DEFAULT_RENDERER); Tooltip.addResourceFiles(); AddResourcesListener.addThemedCSSResource("core.css"); //AddResourcesListener.addThemedCSSResource("bsf.css"); } public void setValueExpression(String name, ValueExpression binding) { name = BsfUtils.snakeCaseToCamelCase(name); super.setValueExpression(name, binding); } @Override public boolean getRendersChildren() { return true; } /** * returns the subset of AJAX requests that are implemented by jQuery * callback or other non-standard means (such as the onclick event of * b:tabView, which has to be implemented manually). * * @return */ public Map<String, String> getJQueryEvents() { Map<String, String> result = new HashMap<String, String>(); result.put("order", "order.dt"); result.put("page", "page.dt"); result.put("search", "search.dt"); result.put("select", "select.dt"); result.put("deselect", "deselect.dt"); return result; } /** * Returns the parameter list of jQuery and other non-standard JS callbacks. * If there's no parameter list for a certain event, the default is simply "event". * * @return A hash map containing the events. May be null. */ @Override public Map<String, String> getJQueryEventParameterLists() { Map<String, String> result = new HashMap<String, String>(); result.put("select", "event, datatable, typeOfSelection, indexes"); result.put("deselect", "event, datatable, typeOfSelection, indexes"); return result; } /** * Returns the subset of the parameter list of jQuery and other non-standard JS callbacks which is sent to the server via AJAX. * If there's no parameter list for a certain event, the default is simply null. * * @return A hash map containing the events. May be null. */ @Override public Map<String, String> getJQueryEventParameterListsForAjax() { Map<String, String> result = new HashMap<String, String>(); result.put("select", "'typeOfSelection':typeOfSelection,'indexes':indexes"); result.put("deselect", "'typeOfSelection':typeOfSelection,'indexes':indexes"); return result; } public Collection<String> getEventNames() { return EVENT_NAMES; } public String getDefaultEventName() { return "click"; } public String getFamily() { return COMPONENT_FAMILY; } /** * This map contains all of the default sorting for each column. This map is * used to store the column sort information gathered during rendering. * * @return The map containing the column / sort type pairs */ public Map<Integer, String> getColumnSortOrderMap() { return columnSortOrder; } /** * Called in order to lazily initialize the map. This map is used to store * the column sort information gathered during rendering. */ public void initColumnSortOrderMap() { this.columnSortOrder = new HashMap<Integer, String>(); } /** * This array ist used to store the column information bits that are used to * initialize the columns using the columns attribute of datatables.net */ public List<String> getColumnInfo() { return columnInfo; } /** * This array ist used to store the column information bits that are used to * initialize the columns using the columns attribute of datatables.net */ public void setColumnInfo(List<String> columnInfo) { this.columnInfo = columnInfo; } /** * <p>Queue an event for broadcast at the end of the current request * processing lifecycle phase. The default implementation in * {@link UIComponentBase} must delegate this call to the * <code>queueEvent()</code> method of the parent {@link UIComponent}.</p> * * @param event {@link FacesEvent} to be queued * * @throws IllegalStateException if this component is not a * descendant of a {@link UIViewRoot} * @throws NullPointerException if <code>event</code> * is <code>null</code> */ public void queueEvent(FacesEvent event) { FacesContext context = FacesContext.getCurrentInstance(); String indexes = (String) context.getExternalContext().getRequestParameterMap().get("indexes"); context.getELContext().getELResolver().setValue(context.getELContext(), null, "indexes", indexes); String typeOfSelection = (String) context.getExternalContext().getRequestParameterMap().get("typeOfSelection"); context.getELContext().getELResolver().setValue(context.getELContext(), null, "typeOfSelection", typeOfSelection); try { int oldIndex = getRowIndex(); int index = Integer.valueOf(indexes); setRowIndex(index); super.queueEvent(event); setRowIndex(oldIndex); } catch (Exception multipleIndexes) { super.queueEvent(event); } } }