/* * The Kuali Financial System, a comprehensive financial management system for higher education. * * Copyright 2005-2014 The Kuali Foundation * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package org.kuali.kfs.module.purap.document.validation.impl; import org.apache.commons.lang.StringUtils; import org.kuali.kfs.module.purap.PurapConstants.AccountDistributionMethodCodes; import org.kuali.kfs.module.purap.businessobject.PurApAccountingLine; import org.kuali.kfs.module.purap.businessobject.PurchaseOrderAccount; import org.kuali.kfs.module.purap.document.PurchaseOrderAmendmentDocument; import org.kuali.kfs.module.purap.document.PurchasingAccountsPayableDocument; import org.kuali.kfs.module.purap.document.service.PurapService; import org.kuali.kfs.sys.businessobject.AccountingLine; import org.kuali.kfs.sys.context.SpringContext; import org.kuali.kfs.sys.document.validation.event.AttributedDocumentEvent; import org.kuali.kfs.sys.document.validation.event.UpdateAccountingLineEvent; import org.kuali.kfs.sys.service.FinancialSystemWorkflowHelperService; import org.kuali.rice.krad.util.GlobalVariables; /** * A validation that checks whether the given accounting line is accessible to the given user or not */ public class PurchaseOrderAmendmentAccountingLineAccessibleValidation extends PurchasingAccountsPayableAccountingLineAccessibleValidation { protected PurapService purapService; /** * Validates that the given accounting line is accessible for editing by the current user. * <strong>This method expects a document as the first parameter and an accounting line as the second</strong> * @see org.kuali.kfs.sys.document.validation.Validation#validate(java.lang.Object[]) */ @Override public boolean validate(AttributedDocumentEvent event) { if( purapService.isDocumentStoppedInRouteNode((PurchasingAccountsPayableDocument)event.getDocument(), "New Unordered Items") ){ //DO NOTHING: do not check that user owns acct lines; at this level, they can edit all accounts on PO amendment return true; } else if (SpringContext.getBean(FinancialSystemWorkflowHelperService.class).isAdhocApprovalRequestedForPrincipal(event.getDocument().getDocumentHeader().getWorkflowDocument(), GlobalVariables.getUserSession().getPrincipalId())) { return true; } else { // KFSCNTRB-1433 // if it's UpdateAccountingLineEvent and only amount changed due to the proportional distribution, that's ok, // since this is the result of item quantity or price change, and the amount change is from re-distributed, not by user. if (event instanceof UpdateAccountingLineEvent) { PurchaseOrderAmendmentDocument poa = (PurchaseOrderAmendmentDocument)event.getDocument(); boolean isProportional = StringUtils.equals(poa.getAccountDistributionMethod(), AccountDistributionMethodCodes.PROPORTIONAL_CODE); boolean onlyAmountChanged = onlyAmountChanged(((UpdateAccountingLineEvent)event).getAccountingLine(), ((UpdateAccountingLineEvent)event).getUpdatedAccountingLine()); if (isProportional && onlyAmountChanged) { return true; } } boolean result = false; boolean setDummyAccountIdentifier = false; if (needsDummyAccountIdentifier()) { ((PurApAccountingLine)getAccountingLineForValidation()).setAccountIdentifier(Integer.MAX_VALUE); // avoid conflicts with any accouting identifier on any other accounting lines in the doc because, you know, you never know... setDummyAccountIdentifier = true; } result = super.validate(event); if (setDummyAccountIdentifier) { ((PurApAccountingLine)getAccountingLineForValidation()).setAccountIdentifier(null); } return result; } } /** * Checks to see if the amount is the only difference between the original accounting line and the updated accounting line. * * @param accountingLine * @param updatedAccountingLine * @return true if only the object code has changed on the accounting line, false otherwise */ private boolean onlyAmountChanged(AccountingLine accountingLine, AccountingLine updatedAccountingLine) { // no changes, return false; this should never happen though, since UpdateAccountingLineEvent means something is changed. if (accountingLine.isLike(updatedAccountingLine)) { return false; } // copy the updatedAccountLine so we can set the amount on the copy of the updated accounting line // to be the original value for comparison purposes AccountingLine updatedLine = null; updatedLine = new PurchaseOrderAccount(); updatedLine.copyFrom(updatedAccountingLine); updatedLine.setAmount(accountingLine.getAmount()); // if they're the same, the only change was the amount return (accountingLine.isLike(updatedLine)); } public void setPurapService(PurapService purapService) { this.purapService = purapService; } }