tabletag.java
来自「displaytag-1.0修正版」· Java 代码 · 共 1,827 行 · 第 1/4 页
JAVA
1,827 行
// don't remove parameters added by the table tag
if (!this.paramEncoder.isParameterEncoded(key))
{
normalHref.removeParameter(key);
}
}
}
else
{
for (int j = 0; j < splittedExcludedParams.length; j++)
{
normalHref.removeParameter(splittedExcludedParams[j]);
}
}
}
if (this.requestUri != null)
{
// if user has added a requestURI create a new href
String fullURI = requestUri;
if (!this.dontAppendContext)
{
String contextPath = ((HttpServletRequest) this.pageContext.getRequest()).getContextPath();
// prepend the context path if any.
// actually checks if context path is already there for people which manually add it
if (!StringUtils.isEmpty(contextPath)
&& requestUri != null
&& requestUri.startsWith("/")
&& !requestUri.startsWith(contextPath))
{
fullURI = contextPath + this.requestUri;
}
}
// call encodeURL to preserve session id when cookies are disabled
fullURI = ((HttpServletResponse) this.pageContext.getResponse()).encodeURL(fullURI);
this.baseHref = new Href(fullURI);
// ... and copy parameters from the current request
Map parameterMap = normalHref.getParameterMap();
this.baseHref.addParameterMap(parameterMap);
}
else
{
// simply copy href
this.baseHref = normalHref;
}
}
/**
* Draw the table. This is where everything happens, we figure out what values we are supposed to be showing, we
* figure out how we are supposed to be showing them, then we draw them.
* @return int
* @throws JspException generic exception
* @see javax.servlet.jsp.tagext.Tag#doEndTag()
*/
public int doEndTag() throws JspException
{
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] doEndTag called");
}
if (!this.doAfterBodyExecuted)
{
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] tag body is empty.");
}
// first row (created in doStartTag)
if (this.currentRow != null)
{
// if yes add to table model and remove
this.tableModel.addRow(this.currentRow);
}
// other rows
while (this.tableIterator.hasNext())
{
Object iteratedObject = this.tableIterator.next();
this.rowNumber++;
// Row object for Cell values
this.currentRow = new Row(iteratedObject, this.rowNumber);
this.tableModel.addRow(this.currentRow);
}
}
// if no rows are defined automatically get all properties from bean
if (this.tableModel.isEmpty())
{
describeEmptyTable();
}
TableDecorator tableDecorator = DecoratorFactory.loadTableDecorator(this.decoratorName);
if (tableDecorator != null)
{
tableDecorator.init(this.pageContext, this.list);
this.tableModel.setTableDecorator(tableDecorator);
}
setupViewableData();
// Figure out how we should sort this data, typically we just sort
// the data being shown, but the programmer can override this behavior
if (!this.tableModel.isSortFullTable())
{
this.tableModel.sortPageList();
}
// Get the data back in the representation that the user is after, do they want HTML/XML/CSV/EXCEL/etc...
int returnValue = EVAL_PAGE;
// check for nested tables
Object previousMediaType = this.pageContext.getAttribute(PAGE_ATTRIBUTE_MEDIA);
if (MediaTypeEnum.HTML.equals(this.currentMediaType)
&& (previousMediaType == null || MediaTypeEnum.HTML.equals(previousMediaType)))
{
writeHTMLData();
}
else if (!MediaTypeEnum.HTML.equals(this.currentMediaType))
{
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] doEndTag - exporting");
}
returnValue = doExport();
}
// do not remove media attribute! if the table is nested in other tables this is still needed
// this.pageContext.removeAttribute(PAGE_ATTRIBUTE_MEDIA);
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] doEndTag - end");
}
cleanUp();
return returnValue;
}
/**
* clean up instance variables, but not the ones representing tag attributes.
*/
private void cleanUp()
{
// reset instance variables (non attributes)
this.currentMediaType = null;
this.baseHref = null;
this.caption = null;
this.currentRow = null;
this.doAfterBodyExecuted = false;
this.footer = null;
this.listHelper = null;
this.nextRow = null;
this.pageNumber = 0;
this.paramEncoder = null;
this.previousRow = null;
this.properties = null;
this.rowNumber = 1;
this.tableIterator = null;
this.tableModel = null;
this.list = null;
}
/**
* If no columns are provided, automatically add them from bean properties. Get the first object in the list and get
* all the properties (except the "class" property which is automatically skipped). Of course this isn't possible
* for empty lists.
*/
private void describeEmptyTable()
{
this.tableIterator = IteratorUtils.getIterator(this.list);
if (this.tableIterator.hasNext())
{
Object iteratedObject = this.tableIterator.next();
Map objectProperties = new HashMap();
// if it's a String don't add the "Bytes" column
if (iteratedObject instanceof String)
{
return;
}
// if it's a map already use key names for column headers
if (iteratedObject instanceof Map)
{
objectProperties = (Map) iteratedObject;
}
else
{
try
{
objectProperties = BeanUtils.describe(iteratedObject);
}
catch (Exception e)
{
log.warn("Unable to automatically add columns: " + e.getMessage(), e);
}
}
// iterator on properties names
Iterator propertiesIterator = objectProperties.keySet().iterator();
while (propertiesIterator.hasNext())
{
// get the property name
String propertyName = (String) propertiesIterator.next();
// dont't want to add the standard "class" property
if (!"class".equals(propertyName)) //$NON-NLS-1$
{
// creates a new header and add to the table model
HeaderCell headerCell = new HeaderCell();
headerCell.setBeanPropertyName(propertyName);
// handle title i18n
headerCell.setTitle(this.properties.geResourceProvider().getResource(
null,
propertyName,
this,
this.pageContext));
this.tableModel.addColumnHeader(headerCell);
}
}
}
}
/**
* Called when data are not displayed in a html page but should be exported.
* @return int SKIP_PAGE
* @throws JspException generic exception
*/
protected int doExport() throws JspException
{
boolean exportFullList = this.properties.getExportFullList();
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] currentMediaType=" + this.currentMediaType);
}
boolean exportHeader = this.properties.getExportHeader(this.currentMediaType);
boolean exportDecorated = this.properties.getExportDecorated();
ExportView exportView = ExportViewFactory.getInstance().getView(
this.currentMediaType,
this.tableModel,
exportFullList,
exportHeader,
exportDecorated);
try
{
writeExport(exportView);
}
catch (IOException e)
{
throw new WrappedRuntimeException(getClass(), e);
}
return SKIP_PAGE;
}
/**
* Will write the export. The default behavior is to write directly to the response. If the ResponseOverrideFilter
* is configured for this request, will instead write the exported content to a map in the Request object.
* @param exportView export view
* @throws JspException for problem in clearing the response or for invalid export views
* @throws IOException exception thrown when writing content to the response
*/
protected void writeExport(ExportView exportView) throws IOException, JspException
{
String filename = properties.getExportFileName(this.currentMediaType);
HttpServletResponse response = (HttpServletResponse) this.pageContext.getResponse();
HttpServletRequest request = (HttpServletRequest) this.pageContext.getRequest();
Map bean = (Map) request.getAttribute(FILTER_CONTENT_OVERRIDE_BODY);
boolean usingFilter = bean != null;
String mimeType = exportView.getMimeType();
// original encoding, be sure to add it back after reset()
String characterEncoding = response.getCharacterEncoding();
if (usingFilter)
{
if (!bean.containsKey(TableTagParameters.BEAN_BUFFER))
{
// We are running under the export filter, call it
log.debug("Exportfilter enabled in unbuffered mode, setting headers");
response.addHeader(TableTagParameters.PARAMETER_EXPORTING, TagConstants.EMPTY_STRING);
}
else
{
// We are running under the export filter in buffered mode
bean.put(TableTagParameters.BEAN_CONTENTTYPE, mimeType);
bean.put(TableTagParameters.BEAN_FILENAME, filename);
if (exportView instanceof TextExportView)
{
StringWriter writer = new StringWriter();
((TextExportView) exportView).doExport(writer);
bean.put(TableTagParameters.BEAN_BODY, writer.toString());
}
else if (exportView instanceof BinaryExportView)
{
ByteArrayOutputStream stream = new ByteArrayOutputStream();
((BinaryExportView) exportView).doExport(stream);
bean.put(TableTagParameters.BEAN_BODY, stream.toByteArray());
}
else
{
throw new JspTagException("Export view "
+ exportView.getClass().getName()
+ " must implement TextExportView or BinaryExportView");
}
return;
}
}
else
{
log.debug("Exportfilter NOT enabled");
// response can't be already committed at this time
if (response.isCommitted())
{
throw new ExportException(getClass());
}
try
{
response.reset();
pageContext.getOut().clearBuffer();
}
catch (Exception e)
{
throw new ExportException(getClass());
}
}
if (!usingFilter && characterEncoding != null && mimeType.indexOf("charset") == -1) //$NON-NLS-1$
{
mimeType += "; charset=" + characterEncoding; //$NON-NLS-1$
}
response.setContentType(mimeType);
if (StringUtils.isNotEmpty(filename))
{
response.setHeader("Content-Disposition", //$NON-NLS-1$
"attachment; filename=\"" + filename + "\""); //$NON-NLS-1$ //$NON-NLS-2$
}
if (exportView instanceof TextExportView)
{
Writer writer;
if (usingFilter)
{
writer = response.getWriter();
}
else
{
writer = pageContext.getOut();
}
((TextExportView) exportView).doExport(writer);
}
else if (exportView instanceof BinaryExportView)
{
// dealing with binary content
// note that this is not assured to work on any application server if the filter is not enabled. According
// to the jsp specs response.getOutputStream() should no be called in jsps.
((BinaryExportView) exportView).doExport(response.getOutputStream());
}
else
{
throw new JspTagException("Export view "
+ exportView.getClass().getName()
+ " must implement TextExportView or BinaryExportView");
}
log.debug("Export completed");
}
/**
* This sets the list of all of the data that will be displayed on the page via the table tag. This might include
* just a subset of the total data in the list due to to paging being active, or the user asking us to just show a
* subset, etc...
*/
protected void setupViewableData()
{
// If the user has changed the way our default behavior works, then we need to look for it now, and resort
// things if needed before we ask for the viewable part. (this is a bad place for this, this should be
// refactored and moved somewhere else).
if (this.tableModel.isSortFullTable())
{
// Sort the total list...
this.tableModel.sortFullList();
}
Object originalData = this.tableModel.getRowListFull();
// If they have asked for a subset of the list via the length
// attribute, then only fetch those items out of the master list.
List fullList = CollectionUtil.getListFromObject(originalData, this.offset, this.length);
int pageOffset = this.offset;
// If they have asked for just a page of the data, then use the
// SmartListHelper to figure out what page they are after, etc...
if (this.pagesize > 0)
{
this.listHelper = new SmartListHelper(fullList, fullList.size(), this.pagesize, this.properties);
this.listHelper.setCurrentPage(this.pageNumber);
pageOffset = this.listHelper.getFirstIndexForCurrentPage();
fullList = this.listHelper.getListForCurrentPage();
}
this.tableModel.setRowListPage(fullList);
this.tableModel.setPageOffset(pageOffset);
}
/**
* called when data have to be displayed in a html page.
* @throws JspException generic exception
*/
private void writeHTMLData() throws JspException
{
JspWriter out = this.pageContext.getOut();
if (log.isDebugEnabled())
{
log.debug("[" + getUid() + "] getHTMLData called for table [" + getUid() + "]");
}
boolean noItems = this.tableModel.getRowListPage().size() == 0;
if (noItems && !this.properties.getEmptyListShowTable())
{
write(this.properties.getEmptyListMessage(), out);
return;
}
// variables to hold the previous row columns values.
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?