001/* 002 * Copyright 2008 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package org.kuali.ole.sys.document.web.renderers; 017 018import java.io.IOException; 019import java.util.ArrayList; 020import java.util.List; 021 022import javax.servlet.jsp.JspException; 023import javax.servlet.jsp.JspWriter; 024import javax.servlet.jsp.PageContext; 025import javax.servlet.jsp.tagext.Tag; 026 027import org.apache.commons.lang.StringUtils; 028import org.kuali.ole.sys.OLEConstants; 029import org.kuali.ole.sys.businessobject.SourceAccountingLine; 030import org.kuali.ole.sys.context.SpringContext; 031import org.kuali.ole.sys.document.AccountingDocument; 032import org.kuali.ole.sys.document.datadictionary.AccountingLineGroupDefinition; 033import org.kuali.ole.sys.document.datadictionary.AccountingLineViewActionDefinition; 034import org.kuali.ole.sys.document.web.AccountingLineViewAction; 035import org.kuali.rice.core.api.config.property.ConfigurationService; 036import org.kuali.rice.kim.api.services.KimApiServiceLocator; 037import org.kuali.rice.kns.web.taglib.html.KNSFileTag; 038import org.kuali.rice.kns.web.taglib.html.KNSImageTag; 039import org.kuali.rice.krad.util.GlobalVariables; 040 041/** 042 * Renders the standard group header/import line 043 */ 044public class GroupTitleLineRenderer implements Renderer, CellCountCurious { 045 private int titleCellSpan = 4; 046 private int cellCount = 1; 047 private AccountingLineGroupDefinition accountingLineGroupDefinition; 048 private AccountingDocument accountingDocument; 049 private String lineCollectionProperty; 050 private KNSFileTag scriptFileTag = new KNSFileTag(); 051 private KNSFileTag noscriptFileTag = new KNSFileTag(); 052 private KNSImageTag uploadButtonTag = new KNSImageTag(); 053 private KNSImageTag cancelButtonTag = new KNSImageTag(); 054 private boolean shouldUpload = true; 055 private boolean canEdit = false; 056 057 private boolean groupActionsRendered = false; 058 059 /** 060 * Constructs a ImportLineRenderer, setting defaults on the tags that will always exist 061 */ 062 public GroupTitleLineRenderer() { 063 scriptFileTag.setSize("30"); 064 noscriptFileTag.setSize("30"); 065 noscriptFileTag.setStyle("font:10px;height:16px;"); 066 uploadButtonTag.setSrc(SpringContext.getBean(ConfigurationService.class).getPropertyValueAsString("externalizable.images.url") + "tinybutton-add1.gif"); 067 uploadButtonTag.setStyleClass("tinybutton"); 068 cancelButtonTag.setProperty("methodToCall.cancel"); 069 cancelButtonTag.setSrc(SpringContext.getBean(ConfigurationService.class).getPropertyValueAsString("externalizable.images.url") + "tinybutton-cancelimport.gif"); 070 cancelButtonTag.setStyleClass("tinybutton"); 071 } 072 073 /** 074 * @see org.kuali.ole.sys.document.web.renderers.Renderer#clear() 075 */ 076 @Override 077 public void clear() { 078 cellCount = 1; 079 accountingLineGroupDefinition = null; 080 titleCellSpan = 4; 081 lineCollectionProperty = null; 082 accountingDocument = null; 083 shouldUpload = true; 084 canEdit = false; 085 086 // clean script file tag 087 scriptFileTag.setPageContext(null); 088 scriptFileTag.setParent(null); 089 scriptFileTag.setProperty(null); 090 091 // clean noscript file tag 092 noscriptFileTag.setPageContext(null); 093 noscriptFileTag.setParent(null); 094 noscriptFileTag.setProperty(null); 095 096 // clean upload button tag 097 uploadButtonTag.setPageContext(null); 098 uploadButtonTag.setParent(null); 099 uploadButtonTag.setProperty(null); 100 uploadButtonTag.setAlt(null); 101 uploadButtonTag.setTitle(null); 102 103 // clean cancel import tag 104 cancelButtonTag.setPageContext(null); 105 cancelButtonTag.setParent(null); 106 cancelButtonTag.setAlt(null); 107 cancelButtonTag.setTitle(null); 108 cancelButtonTag.setOnclick(null); 109 } 110 111 /** 112 * @see org.kuali.ole.sys.document.web.renderers.Renderer#render(javax.servlet.jsp.PageContext, javax.servlet.jsp.tagext.Tag, 113 * org.kuali.core.bo.BusinessObject) 114 */ 115 @Override 116 public void render(PageContext pageContext, Tag parentTag) throws JspException { 117 try { 118 pageContext.getOut().write(buildRowBeginning()); 119 120 pageContext.getOut().write(buildTitleCell()); 121 this.renderGroupLevelActions(pageContext, parentTag); 122 123 pageContext.getOut().write(buildRowEnding()); 124 } 125 catch (IOException ioe) { 126 throw new JspException("Difficulty in rendering import/group header line", ioe); 127 } 128 } 129 130 /** 131 * Builds a tag for the row beginning 132 * 133 * @returns the String with the HTML for the row opening 134 */ 135 protected String buildRowBeginning() { 136 return "<tr>"; 137 } 138 139 /** 140 * Builds the tag for the row beginning 141 * 142 * @returns the String with the HTML for the row beginning 143 */ 144 protected String buildRowEnding() { 145 return "</tr>"; 146 } 147 148 protected void renderGroupLevelActions(PageContext pageContext, Tag parentTag) throws JspException { 149 JspWriter out = pageContext.getOut(); 150 151 try { 152 out.write(this.buildGroupActionsBeginning()); 153 154 this.renderGroupActions(pageContext, parentTag); 155 String nameSpaceCode = OLEConstants.Account.ACCOUNT_NAMESPACE; 156 157 boolean hasPermission = KimApiServiceLocator.getPermissionService().hasPermission( 158 GlobalVariables.getUserSession().getPerson().getPrincipalId(), nameSpaceCode, 159 OLEConstants.Account.UPLOAD_BUDGET); 160 161 if(hasPermission) { 162 this.renderUploadCell(pageContext, parentTag); 163 } 164 165 out.write(this.buildGroupActionsColumnEnding()); 166 } 167 catch (IOException ioe) { 168 throw new JspException("Difficulty rendering group level actions", ioe); 169 } 170 } 171 172 /** 173 * Builds a tag for the row beginning 174 * 175 * @returns the String with the HTML for the row opening 176 */ 177 protected String buildGroupActionsBeginning() { 178 if (this.canUpload() || this.isGroupActionsRendered()) { 179 StringBuilder groupActionsBeginning = new StringBuilder(); 180 final int width = cellCount - titleCellSpan; 181 182 groupActionsBeginning.append("<td "); 183 groupActionsBeginning.append("colspan=\""); 184 groupActionsBeginning.append(Integer.toString(width)); 185 groupActionsBeginning.append("\" "); 186 187 groupActionsBeginning.append("class=\"tab-subhead-import\" "); 188 groupActionsBeginning.append("align=\"right\" "); 189 groupActionsBeginning.append("nowrap=\"nowrap\" "); 190 groupActionsBeginning.append("style=\"border-right: none;\""); 191 groupActionsBeginning.append(">"); 192 193 return groupActionsBeginning.toString(); 194 } 195 196 return StringUtils.EMPTY; 197 } 198 199 /** 200 * Builds the tag for the row beginning 201 * 202 * @returns the String with the HTML for the row beginning 203 */ 204 protected String buildGroupActionsColumnEnding() { 205 return this.canUpload() || this.isGroupActionsRendered() ? "</td>" : StringUtils.EMPTY; 206 } 207 208 /** 209 * Builds the tags for the title cell of the import line 210 * 211 * @return the String with the HTML for the title cell 212 */ 213 protected String buildTitleCell() { 214 StringBuilder titleCell = new StringBuilder(); 215 int colSpan = (this.canUpload() || this.isGroupActionsRendered()) ? titleCellSpan : cellCount; 216 217 titleCell.append("<td "); 218 219 titleCell.append("colspan=\""); 220 titleCell.append(colSpan); 221 titleCell.append("\" "); 222 223 titleCell.append("class=\"tab-subhead\" "); 224 225 titleCell.append("style=\"border-right: none;\""); 226 227 titleCell.append(">"); 228 229 titleCell.append(buildGroupAnchor()); 230 231 titleCell.append(accountingLineGroupDefinition.getGroupLabel()); 232 233 titleCell.append("</td>"); 234 235 return titleCell.toString(); 236 } 237 238 /** 239 * Builds the unique anchor for this group 240 * 241 * @return the unique anchor for this group 242 */ 243 protected String buildGroupAnchor() { 244 return "<a name=\"accounting" + getGroupInfix() + "Anchor\"></a>"; 245 } 246 247 protected void renderGroupActions(PageContext pageContext, Tag parentTag) throws JspException { 248 List<? extends AccountingLineViewActionDefinition> accountingLineGroupActions = accountingLineGroupDefinition.getAccountingLineGroupActions(); 249 if (!this.isGroupActionsRendered() || accountingLineGroupActions == null || accountingLineGroupActions.isEmpty()) { 250 return; 251 } 252 253 List<AccountingLineViewAction> viewActions = new ArrayList<AccountingLineViewAction>(); 254 for (AccountingLineViewActionDefinition action : accountingLineGroupActions) { 255 String actionMethod = action.getActionMethod(); 256 String actionLabel = action.getActionLabel(); 257 String imageName = SpringContext.getBean(ConfigurationService.class).getPropertyValueAsString("externalizable.images.url") + action.getImageName(); 258 259 AccountingLineViewAction viewAction = new AccountingLineViewAction(actionMethod, actionLabel, imageName); 260 viewActions.add(viewAction); 261 } 262 263 if (!viewActions.isEmpty()) { 264 ActionsRenderer actionsRenderer = new ActionsRenderer(); 265 actionsRenderer.setTagBeginning(" "); 266 actionsRenderer.setTagEnding(" "); 267 actionsRenderer.setPostButtonSpacing(" "); 268 actionsRenderer.setActions(viewActions); 269 actionsRenderer.render(pageContext, parentTag); 270 actionsRenderer.clear(); 271 } 272 } 273 274 /** 275 * A dumb way to get the group infix that tries to figure out if it's dealing with a source or target line 276 * 277 * @return the String "source" or "target" to populate the buildGroupAnchor 278 */ 279 protected String getGroupInfix() { 280 Class accountingLineClass = accountingLineGroupDefinition.getAccountingLineClass(); 281 return (accountingLineClass.isAssignableFrom(SourceAccountingLine.class) ? "source" : "target"); 282 } 283 284 /** 285 * Oy, the big one...this one actually renders instead of returning the HTML in a String. This is because it's kind of complex 286 * (and a likely target for future refactoring) 287 * 288 * @param pageContext the page contex to render to 289 * @param parentTag the tag that is requesting all the rendering 290 * @throws JspException thrown if something goes wrong 291 */ 292 protected void renderUploadCell(PageContext pageContext, Tag parentTag) throws JspException { 293 JspWriter out = pageContext.getOut(); 294 295 if (canUpload()) { 296 try { 297 String hideImport = getHideImportName(); 298 String showImport = getShowImportName(); 299 String showLink = getShowLinkName(); 300 String uploadDiv = getUploadDivName(); 301 302 out.write("\n<SCRIPT type=\"text/javascript\">\n"); 303 out.write("<!--\n"); 304 out.write("\tfunction " + hideImport + "() {\n"); 305 out.write("\t\tdocument.getElementById(\"" + showLink + "\").style.display=\"inline\";\n"); 306 out.write("\t\tdocument.getElementById(\"" + uploadDiv + "\").style.display=\"none\";\n"); 307 out.write("\t}\n"); 308 out.write("\tfunction " + showImport + "() {\n"); 309 out.write("\t\tdocument.getElementById(\"" + showLink + "\").style.display=\"none\";\n"); 310 out.write("\t\tdocument.getElementById(\"" + uploadDiv + "\").style.display=\"inline\";\n"); 311 out.write("\t}\n"); 312 out.write("\tdocument.write(\n"); 313 out.write("\t\t'<a id=\"" + showLink + "\" href=\"#\" onclick=\"" + showImport + "();return false;\">' +\n"); 314 out.write("\t\t'<img src=\"" + SpringContext.getBean(ConfigurationService.class).getPropertyValueAsString("externalizable.images.url") + "tinybutton-importlines.gif\" title=\"import file\" alt=\"import file\"' +\n"); 315 out.write("\t\t'width=\"72\" border=\"0\">' +\n"); 316 out.write("\t\t'</a>' +\n"); 317 out.write("\t\t'<div id=\"" + uploadDiv + "\" style=\"display:none;\" >' +\n"); 318 319 out.write("\t\t'"); 320 321 scriptFileTag.setPageContext(pageContext); 322 scriptFileTag.setParent(parentTag); 323 scriptFileTag.setProperty(accountingLineGroupDefinition.getImportedLinePropertyPrefix() + "File"); 324 scriptFileTag.doStartTag(); 325 scriptFileTag.doEndTag(); 326 327 out.write("' +\n"); 328 out.write("\t\t'"); 329 330 uploadButtonTag.setPageContext(pageContext); 331 uploadButtonTag.setParent(parentTag); 332 uploadButtonTag.setProperty("methodToCall.upload" + StringUtils.capitalize(accountingLineGroupDefinition.getImportedLinePropertyPrefix()) + "Lines"); 333 uploadButtonTag.setAlt("insert " + accountingLineGroupDefinition.getGroupLabel() + " accounting lines"); 334 uploadButtonTag.setTitle("insert " + accountingLineGroupDefinition.getGroupLabel() + " accounting lines"); 335 uploadButtonTag.doStartTag(); 336 uploadButtonTag.doEndTag(); 337 338 out.write("' +\n"); 339 340 out.write("\t\t'"); 341 342 cancelButtonTag.setPageContext(pageContext); 343 cancelButtonTag.setParent(parentTag); 344 cancelButtonTag.setAlt("Cancel import of " + accountingLineGroupDefinition.getGroupLabel() + " accounting lines"); 345 cancelButtonTag.setTitle("Cancel import of " + accountingLineGroupDefinition.getGroupLabel() + " accounting lines"); 346 cancelButtonTag.setOnclick(getHideImportName() + "();return false;"); 347 cancelButtonTag.doStartTag(); 348 cancelButtonTag.doEndTag(); 349 350 out.write("' +\n"); 351 352 out.write("\t'</div>');\n"); 353 out.write("\t//-->\n"); 354 out.write("</SCRIPT>\n"); 355 out.write("<NOSCRIPT>\n"); 356 out.write("\tImport " + accountingLineGroupDefinition.getGroupLabel() + " lines\n"); 357 358 noscriptFileTag.setPageContext(pageContext); 359 noscriptFileTag.setParent(parentTag); 360 noscriptFileTag.setProperty(accountingLineGroupDefinition.getImportedLinePropertyPrefix() + "File"); 361 noscriptFileTag.doStartTag(); 362 noscriptFileTag.doEndTag(); 363 364 uploadButtonTag.doStartTag(); 365 uploadButtonTag.doEndTag(); 366 367 out.write("</NOSCRIPT>\n"); 368 } 369 catch (IOException ioe) { 370 throw new JspException("Difficulty rendering accounting lines import upload", ioe); 371 } 372 } 373 } 374 375 /** 376 * @return the name of the line collection property, but in a form that is okay for javascript variable/function naming 377 */ 378 protected String getVariableFriendlyLineCollectionProperty() { 379 return lineCollectionProperty.replaceAll("[^A-Za-z0-9]", "_"); 380 } 381 382 /** 383 * @return the name of the hide import function 384 */ 385 protected String getHideImportName() { 386 return "hide" + getVariableFriendlyLineCollectionProperty() + "Import"; 387 } 388 389 /** 390 * @return the name of the show import function 391 */ 392 protected String getShowImportName() { 393 return "show" + getVariableFriendlyLineCollectionProperty() + "Import"; 394 } 395 396 /** 397 * @return the name of the show link element 398 */ 399 protected String getShowLinkName() { 400 return lineCollectionProperty + "ShowLink"; 401 } 402 403 /** 404 * @return the name of the upload div 405 */ 406 protected String getUploadDivName() { 407 return "upload" + lineCollectionProperty + "Div"; 408 } 409 410 /** 411 * Determines if an upload can proceed for the accounting line group 412 * 413 * @return true if upload is possible, false otherwise 414 */ 415 protected boolean canUpload() { 416 return (canEdit && accountingDocument.getAccountingLineParser() != null && shouldUpload); 417 } 418 419 /** 420 * Allows overriding of whether something can be uploaded - though this serves only to turn uploading more off, never more on 421 * 422 * @param allowUpload should we be allowed to upload? 423 */ 424 public void overrideCanUpload(boolean allowUpload) { 425 this.shouldUpload = allowUpload; 426 } 427 428 /** 429 * Gets the cellCount attribute. 430 * 431 * @return Returns the cellCount. 432 */ 433 public int getCellCount() { 434 return cellCount; 435 } 436 437 /** 438 * Sets the cellCount attribute value. 439 * 440 * @param cellCount The cellCount to set. 441 */ 442 @Override 443 public void setCellCount(int cellCount) { 444 this.cellCount = cellCount; 445 } 446 447 /** 448 * Gets the accountingDocument attribute. 449 * 450 * @return Returns the accountingDocument. 451 */ 452 public AccountingDocument getAccountingDocument() { 453 return accountingDocument; 454 } 455 456 /** 457 * Sets the accountingDocument attribute value. 458 * 459 * @param accountingDocument The accountingDocument to set. 460 */ 461 public void setAccountingDocument(AccountingDocument accountingDocument) { 462 this.accountingDocument = accountingDocument; 463 } 464 465 /** 466 * Gets the accountingLineGroupDefinition attribute. 467 * 468 * @return Returns the accountingLineGroupDefinition. 469 */ 470 public AccountingLineGroupDefinition getAccountingLineGroupDefinition() { 471 return accountingLineGroupDefinition; 472 } 473 474 /** 475 * Sets the accountingLineGroupDefinition attribute value. 476 * 477 * @param accountingLineGroupDefinition The accountingLineGroupDefinition to set. 478 */ 479 public void setAccountingLineGroupDefinition(AccountingLineGroupDefinition accountingLineGroupDefinition) { 480 this.accountingLineGroupDefinition = accountingLineGroupDefinition; 481 } 482 483 /** 484 * Gets the titleCellSpan attribute. 485 * 486 * @return Returns the titleCellSpan. 487 */ 488 public int getTitleCellSpan() { 489 return titleCellSpan; 490 } 491 492 /** 493 * Sets the titleCellSpan attribute value. 494 * 495 * @param titleCellSpan The titleCellSpan to set. 496 */ 497 public void setTitleCellSpan(int titleCellSpan) { 498 this.titleCellSpan = titleCellSpan; 499 } 500 501 /** 502 * Gets the lineCollectionProperty attribute. 503 * 504 * @return Returns the lineCollectionProperty. 505 */ 506 public String getLineCollectionProperty() { 507 return lineCollectionProperty; 508 } 509 510 /** 511 * Sets the lineCollectionProperty attribute value. 512 * 513 * @param lineCollectionProperty The lineCollectionProperty to set. 514 */ 515 public void setLineCollectionProperty(String lineCollectionProperty) { 516 this.lineCollectionProperty = lineCollectionProperty; 517 } 518 519 /** 520 * Gets the groupActionsRendered attribute. 521 * 522 * @return Returns the groupActionsRendered. 523 */ 524 public boolean isGroupActionsRendered() { 525 return groupActionsRendered; 526 } 527 528 /** 529 * Sets the groupActionsRendered attribute value. 530 * 531 * @param groupActionsRendered The groupActionsRendered to set. 532 */ 533 public void setGroupActionsRendered(boolean groupActionsRenderred) { 534 this.groupActionsRendered = groupActionsRenderred; 535 } 536 537 /** 538 * Sets the canEdit attribute value. 539 * @param canEdit The canEdit to set. 540 */ 541 public void setCanEdit(boolean canEdit) { 542 this.canEdit = canEdit; 543 } 544 545}