blob: 93648c4710bc102146aa906dfccd6c9f426da4cc [file] [log] [blame]
// Copyright 2007, 2008 The Apache Software Foundation
//
// 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 org.apache.tapestry5.corelib.components;
import org.apache.tapestry5.ComponentAction;
import org.apache.tapestry5.annotations.Environmental;
import org.apache.tapestry5.annotations.Parameter;
import org.apache.tapestry5.annotations.Property;
import org.apache.tapestry5.beaneditor.PropertyModel;
import org.apache.tapestry5.grid.GridConstants;
import org.apache.tapestry5.grid.GridDataSource;
import org.apache.tapestry5.grid.GridModel;
import org.apache.tapestry5.internal.TapestryInternalUtils;
import org.apache.tapestry5.ioc.internal.util.CollectionFactory;
import org.apache.tapestry5.services.FormSupport;
import java.util.List;
/**
* Renders out a series of rows within the table.
* <p/>
* Inside a {@link Form}, a series of row index numbers are stored into the form ( {@linkplain FormSupport#store(Object,
* ComponentAction) as ComponentActions}). This is not ideal ... in a situation where the data set can shift between the
* form render and the form submission, this can cause unexpected results, including applying changes to the wrong
* objects.
*/
public class GridRows
{
private int startRow;
static class SetupForRow implements ComponentAction<GridRows>
{
private static final long serialVersionUID = -3216282071752371975L;
private final int rowIndex;
public SetupForRow(int rowIndex)
{
this.rowIndex = rowIndex;
}
public void execute(GridRows component)
{
component.setupForRow(rowIndex);
}
}
/**
* Parameter used to set the CSS class for each row (each &lt;tr&gt; element) within the &lt;tbody&gt;). This is not
* cached, so it will be recomputed for each row.
*/
@Parameter(cache = false)
private String rowClass;
/**
* Object that provides access to the bean and data models used to render the Grid.
*/
@Parameter(value = "componentResources.container")
private GridModel gridModel;
/**
* Number of rows displayed on each page. Long result sets are split across multiple pages.
*/
@Parameter(required = true)
private int rowsPerPage;
/**
* The current page number within the available pages (indexed from 1).
*/
@Parameter(required = true)
private int currentPage;
/**
* The current row being rendered, this is primarily an output parameter used to allow the Grid, and the Grid's
* container, to know what object is being rendered.
*/
@Parameter(required = true)
@Property(write = false)
private Object row;
/**
* If true, then the CSS class on each &lt;TD&gt; cell will be omitted, which can reduce the amount of output from
* the component overall by a considerable amount. Leave this as false, the default, when you are leveraging the CSS
* to customize the look and feel of particular columns.
*/
@Parameter
private boolean lean;
/**
* If true and the Loop is enclosed by a Form, then the normal state saving logic is turned off. Defaults to false,
* enabling state saving logic within Forms.
*/
@Parameter(name = "volatile")
private boolean volatileState;
@Environmental(false)
private FormSupport formSupport;
private boolean recordingStateInsideForm;
private int endRow;
private int rowIndex;
private String propertyName;
@Property(write = false)
private PropertyModel columnModel;
public String getRowClass()
{
List<String> classes = CollectionFactory.newList();
// Not a cached parameter, so careful to only access it once.
String rc = rowClass;
if (rc != null) classes.add(rc);
if (rowIndex == startRow) classes.add(GridConstants.FIRST_CLASS);
if (rowIndex == endRow) classes.add(GridConstants.LAST_CLASS);
return TapestryInternalUtils.toClassAttributeValue(classes);
}
public String getCellClass()
{
List<String> classes = CollectionFactory.newList();
String id = gridModel.getDataModel().get(propertyName).getId();
if (!lean)
{
classes.add(id);
switch (gridModel.getSortModel().getColumnSort(id))
{
case ASCENDING:
classes.add(GridConstants.SORT_ASCENDING_CLASS);
break;
case DESCENDING:
classes.add(GridConstants.SORT_DESCENDING_CLASS);
break;
default:
}
}
return TapestryInternalUtils.toClassAttributeValue(classes);
}
void setupRender()
{
GridDataSource dataSource = gridModel.getDataSource();
int availableRows = dataSource.getAvailableRows();
int maxPages = ((availableRows - 1) / rowsPerPage) + 1;
// This can sometimes happen when the number of items shifts between requests.
if (currentPage > maxPages) currentPage = maxPages;
startRow = (currentPage - 1) * rowsPerPage;
endRow = Math.min(availableRows - 1, startRow + rowsPerPage - 1);
rowIndex = startRow;
recordingStateInsideForm = !volatileState && formSupport != null;
}
/**
* Callback method, used when recording state to a form, or called directly when not recording state.
*/
void setupForRow(int rowIndex)
{
row = gridModel.getDataSource().getRowValue(rowIndex);
}
void beginRender()
{
// When needed, store a callback used when the form is submitted.
if (recordingStateInsideForm) formSupport.store(this, new SetupForRow(rowIndex));
// And do it now for the render.
setupForRow(rowIndex);
}
boolean afterRender()
{
rowIndex++;
return rowIndex > endRow;
}
public List<String> getPropertyNames()
{
return gridModel.getDataModel().getPropertyNames();
}
public String getPropertyName()
{
return propertyName;
}
public void setPropertyName(String propertyName)
{
this.propertyName = propertyName;
columnModel = gridModel.getDataModel().get(propertyName);
}
}