View Javadoc

1   /*
2    * Copyright 2006 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  
17  package org.kuali.ole.module.purap.businessobject;
18  
19  import org.kuali.ole.module.purap.PurapConstants;
20  import org.kuali.ole.module.purap.PurapPropertyConstants;
21  import org.kuali.ole.module.purap.document.PaymentRequestDocument;
22  import org.kuali.ole.module.purap.document.PurchaseOrderDocument;
23  import org.kuali.ole.module.purap.document.service.AccountsPayableService;
24  import org.kuali.ole.module.purap.document.service.PurapService;
25  import org.kuali.ole.module.purap.exception.PurError;
26  import org.kuali.ole.module.purap.util.ExpiredOrClosedAccountEntry;
27  import org.kuali.ole.module.purap.util.PurApItemUtils;
28  import org.kuali.ole.module.purap.util.PurApObjectUtils;
29  import org.kuali.ole.select.businessobject.OleInvoiceItem;
30  import org.kuali.ole.select.document.OlePaymentRequestDocument;
31  import org.kuali.ole.sys.context.SpringContext;
32  import org.kuali.rice.core.api.util.type.KualiDecimal;
33  import org.kuali.rice.krad.util.ObjectUtils;
34  
35  import java.math.BigDecimal;
36  import java.util.ArrayList;
37  import java.util.HashMap;
38  import java.util.List;
39  
40  /**
41   * Payment Request Item Business Object.
42   */
43  public class PaymentRequestItem extends AccountsPayableItemBase {
44      private static org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(PaymentRequestItem.class);
45  
46      protected BigDecimal purchaseOrderItemUnitPrice;
47      private KualiDecimal itemOutstandingInvoiceQuantity;
48      private KualiDecimal itemOutstandingInvoiceAmount;
49  
50      /**
51       * Default constructor.
52       */
53      public PaymentRequestItem() {
54  
55      }
56  
57      /**
58       * preq item constructor - Delegate
59       *
60       * @param poi  - purchase order item
61       * @param preq - payment request document
62       */
63      public PaymentRequestItem(PurchaseOrderItem poi, PaymentRequestDocument preq) {
64          this(poi, preq, new HashMap<String, ExpiredOrClosedAccountEntry>());
65      }
66  
67      /**
68       * Constructs a new payment request item, but also merges expired accounts.
69       *
70       * @param poi                        - purchase order item
71       * @param preq                       - payment request document
72       * @param expiredOrClosedAccountList - list of expired or closed accounts to merge
73       */
74      public PaymentRequestItem(PurchaseOrderItem poi, PaymentRequestDocument preq, HashMap<String, ExpiredOrClosedAccountEntry> expiredOrClosedAccountList) {
75  
76          // copy base attributes w/ extra array of fields not to be copied
77          PurApObjectUtils.populateFromBaseClass(PurApItemBase.class, poi, this, PurapConstants.PREQ_ITEM_UNCOPYABLE_FIELDS);
78  
79          setItemDescription(poi.getItemDescription());
80  
81          //New Source Line should be set for PaymentRequestItem
82          resetAccount();
83  
84          // set up accounts
85          List accounts = new ArrayList();
86          for (PurApAccountingLine account : poi.getSourceAccountingLines()) {
87              PurchaseOrderAccount poa = (PurchaseOrderAccount) account;
88  
89              // check if this account is expired/closed and replace as needed
90              SpringContext.getBean(AccountsPayableService.class).processExpiredOrClosedAccount(poa, expiredOrClosedAccountList);
91  
92              //KFSMI-4522 copy an accounting line with zero dollar amount if system parameter allows
93              if (poa.getAmount().isZero()) {
94                  if (SpringContext.getBean(AccountsPayableService.class).canCopyAccountingLinesWithZeroAmount()) {
95                      accounts.add(new PaymentRequestAccount(this, poa));
96                  }
97              } else {
98                  accounts.add(new PaymentRequestAccount(this, poa));
99              }
100         }
101 
102         this.setSourceAccountingLines(accounts);
103         this.getUseTaxItems().clear();
104         //List<PurApItemUseTax> newUseTaxItems = new ArrayList<PurApItemUseTax>();
105         /// this.setUseTaxItems(newUseTaxItems);
106         //copy use tax items over, and blank out keys (useTaxId and itemIdentifier)
107         /*
108         this.getUseTaxItems().clear();
109         for (PurApItemUseTax useTaxItem : poi.getUseTaxItems()) {
110             PaymentRequestItemUseTax newItemUseTax = new PaymentRequestItemUseTax(useTaxItem);
111             this.getUseTaxItems().add(newItemUseTax);
112 
113         }
114         */
115 
116         // clear amount and desc on below the line - we probably don't need that null
117         // itemType check but it's there just in case remove if it causes problems
118         // also do this if of type service
119         if ((ObjectUtils.isNotNull(this.getItemType()) && this.getItemType().isAmountBasedGeneralLedgerIndicator())) {
120             // setting unit price to be null to be more consistent with other below the line
121             this.setItemUnitPrice(null);
122         }
123 
124         // copy custom
125         this.purchaseOrderItemUnitPrice = poi.getItemUnitPrice();
126 //        this.purchaseOrderCommodityCode = poi.getPurchaseOrderCommodityCd();
127 
128         // set doc fields
129         this.setPurapDocumentIdentifier(preq.getPurapDocumentIdentifier());
130         this.setPurapDocument(preq);
131     }
132 
133     /**
134      * Retrieves a purchase order item by inspecting the item type to see if its above the line or below the line and returns the
135      * appropriate type.
136      *
137      * @return - purchase order item
138      */
139     @Override
140     public PurchaseOrderItem getPurchaseOrderItem() {
141         if (ObjectUtils.isNotNull(this.getPurapDocumentIdentifier())) {
142             if (ObjectUtils.isNull(this.getPaymentRequest())) {
143                 this.refreshReferenceObject(PurapPropertyConstants.PURAP_DOC);
144             }
145         }
146         // ideally we should do this a different way - maybe move it all into the service or save this info somehow (make sure and
147         // update though)
148         if (getPaymentRequest() != null) {
149             PurchaseOrderDocument po = getPaymentRequest().getPurchaseOrderDocument();
150             PurchaseOrderItem poi = null;
151             if (this.getItemType().isLineItemIndicator()) {
152                 List<PurchaseOrderItem> items = po.getItems();
153                 poi = items.get(this.getItemLineNumber().intValue() - 1);
154                 // throw error if line numbers don't match
155                 // MSU Contribution DTT-3014 OLEMI-8483 OLECNTRB-974
156                 /*
157                  * List items = po.getItems(); if (items != null) { for (Object object : items) { PurchaseOrderItem item =
158                  * (PurchaseOrderItem) object; if (item != null && item.getItemLineNumber().equals(this.getItemLineNumber())) { poi
159                  * = item; break; } } }
160                  */
161             } else {
162                 poi = (PurchaseOrderItem) SpringContext.getBean(PurapService.class).getBelowTheLineByType(po, this.getItemType());
163             }
164             if (poi != null) {
165                 return poi;
166             } else {
167                 if (LOG.isDebugEnabled()) {
168                     LOG.debug("getPurchaseOrderItem() Returning null because PurchaseOrderItem object for line number" + getItemLineNumber() + "or itemType " + getItemTypeCode() + " is null");
169                 }
170                 return null;
171             }
172         } else {
173 
174             LOG.error("getPurchaseOrderItem() Returning null because paymentRequest object is null");
175             throw new PurError("Payment Request Object in Purchase Order item line number " + getItemLineNumber() + "or itemType " + getItemTypeCode() + " is null");
176         }
177     }
178 
179     public KualiDecimal getPoOutstandingAmount() {
180         PurchaseOrderItem poi = getPurchaseOrderItem();
181         if (ObjectUtils.isNull(this.getPurchaseOrderItemUnitPrice()) || KualiDecimal.ZERO.equals(this.getPurchaseOrderItemUnitPrice())) {
182             return null;
183         } else {
184             return this.getPoOutstandingAmount(poi);
185         }
186     }
187 
188     private KualiDecimal getPoOutstandingAmount(PurchaseOrderItem poi) {
189         if (poi == null) {
190             return KualiDecimal.ZERO;
191         } else {
192             return poi.getItemOutstandingEncumberedAmount();
193         }
194     }
195 
196     public KualiDecimal getPoOriginalAmount() {
197         PurchaseOrderItem poi = getPurchaseOrderItem();
198         if (poi == null) {
199             return null;
200         } else {
201             return poi.getExtendedPrice();
202         }
203     }
204 
205     /**
206      * Exists due to a setter requirement by the htmlControlAttribute
207      *
208      * @param amount - po outstanding amount
209      * @deprecated
210      */
211     @Deprecated
212     public void setPoOutstandingAmount(KualiDecimal amount) {
213         // do nothing
214     }
215 
216 
217     public KualiDecimal getPoOutstandingQuantity() {
218         PurchaseOrderItem poi = getPurchaseOrderItem();
219         if (poi == null) {
220             return null;
221         } else {
222             if (PurapConstants.ItemTypeCodes.ITEM_TYPE_SERVICE_CODE.equals(this.getItemTypeCode())) {
223                 return null;
224             } else {
225                 return poi.getOutstandingQuantity();
226             }
227         }
228     }
229 
230     /**
231      * Exists due to a setter requirement by the htmlControlAttribute
232      *
233      * @param qty - po outstanding quantity
234      * @deprecated
235      */
236     @Deprecated
237     public void setPoOutstandingQuantity(KualiDecimal qty) {
238         // do nothing
239     }
240 
241     public BigDecimal getPurchaseOrderItemUnitPrice() {
242         return purchaseOrderItemUnitPrice;
243     }
244 
245     public BigDecimal getOriginalAmountfromPO() {
246         return purchaseOrderItemUnitPrice;
247     }
248 
249     public void setOriginalAmountfromPO(BigDecimal purchaseOrderItemUnitPrice) {
250         // Do nothing
251     }
252 
253     public void setPurchaseOrderItemUnitPrice(BigDecimal purchaseOrderItemUnitPrice) {
254         this.purchaseOrderItemUnitPrice = purchaseOrderItemUnitPrice;
255     }
256 
257     public KualiDecimal getItemOutstandingInvoiceAmount() {
258         return itemOutstandingInvoiceAmount;
259     }
260 
261     public void setItemOutstandingInvoiceAmount(KualiDecimal itemOutstandingInvoiceAmount) {
262         this.itemOutstandingInvoiceAmount = itemOutstandingInvoiceAmount;
263     }
264 
265     public KualiDecimal getItemOutstandingInvoiceQuantity() {
266         return itemOutstandingInvoiceQuantity;
267     }
268 
269     public void setItemOutstandingInvoiceQuantity(KualiDecimal itemOutstandingInvoiceQuantity) {
270         this.itemOutstandingInvoiceQuantity = itemOutstandingInvoiceQuantity;
271     }
272 
273     public PaymentRequestDocument getPaymentRequest() {
274         if (ObjectUtils.isNotNull(getPurapDocumentIdentifier())) {
275             if (ObjectUtils.isNull(getPurapDocument())) {
276                 this.refreshReferenceObject(PurapPropertyConstants.PURAP_DOC);
277             }
278         }
279         return super.getPurapDocument();
280     }
281 
282     public void setPaymentRequest(PaymentRequestDocument paymentRequest) {
283         this.setPurapDocument(paymentRequest);
284     }
285 
286     public void generateAccountListFromPoItemAccounts(List<PurApAccountingLine> accounts) {
287         for (PurApAccountingLine line : accounts) {
288             PurchaseOrderAccount poa = (PurchaseOrderAccount) line;
289             if (!line.isEmpty()) {
290                 getSourceAccountingLines().add(new PaymentRequestAccount(this, poa));
291             }
292         }
293     }
294 
295     /**
296      * @see org.kuali.ole.module.purap.businessobject.PurApItem#getAccountingLineClass()
297      */
298     @Override
299     public Class getAccountingLineClass() {
300         return PaymentRequestAccount.class;
301     }
302 
303     public boolean isDisplayOnPreq() {
304         PurchaseOrderItem poi = getPurchaseOrderItem();
305         if (ObjectUtils.isNull(poi)) {
306             LOG.debug("poi was null");
307             return false;
308         }
309 
310         // if the po item is not active... skip it
311         if (!poi.isItemActiveIndicator()) {
312             if (LOG.isDebugEnabled()) {
313                 LOG.debug("poi was not active: " + poi.toString());
314             }
315             return false;
316         }
317 
318         ItemType poiType = poi.getItemType();
319 
320         if (poiType.isQuantityBasedGeneralLedgerIndicator()) {
321             if (poi.getItemQuantity().isGreaterThan(poi.getItemInvoicedTotalQuantity())) {
322                 return true;
323             } else {
324                 if (ObjectUtils.isNotNull(this.getItemQuantity()) && this.getItemQuantity().isGreaterThan(KualiDecimal.ZERO)) {
325                     return true;
326                 }
327             }
328 
329             return false;
330         } else { // not quantity based
331             if (poi.getItemOutstandingEncumberedAmount().isGreaterThan(KualiDecimal.ZERO)) {
332                 return true;
333             } else {
334                 if (PurApItemUtils.isNonZeroExtended(this)) {
335                     return true;
336                 }
337                 return false;
338             }
339 
340         }
341     }
342 
343     /**
344      * sets account line percentage to zero.
345      *
346      * @see org.kuali.ole.module.purap.businessobject.PurApItem#resetAccount()
347      */
348     @Override
349     public void resetAccount() {
350         super.resetAccount();
351         this.getNewSourceLine().setAmount(null);
352         this.getNewSourceLine().setAccountLinePercent(new BigDecimal(0));
353     }
354 
355     /**
356      * Added for electronic invoice
357      */
358     public void addToUnitPrice(BigDecimal addThisValue) {
359         if (getItemUnitPrice() == null) {
360             setItemUnitPrice(BigDecimal.ZERO);
361         }
362         BigDecimal addedPrice = getItemUnitPrice().add(addThisValue);
363         setItemUnitPrice(addedPrice);
364     }
365 
366     public void addToExtendedPrice(KualiDecimal addThisValue) {
367         if (getExtendedPrice() == null) {
368             setExtendedPrice(KualiDecimal.ZERO);
369         }
370         KualiDecimal addedPrice = getExtendedPrice().add(addThisValue);
371         setExtendedPrice(addedPrice);
372     }
373 
374     @Override
375     public Class getUseTaxClass() {
376         return PaymentRequestItemUseTax.class;
377     }
378 
379     /**
380      * preq item constructor - Delegate
381      *
382      * @param poi  - purchase order item
383      * @param preq - payment request document
384      */
385     public PaymentRequestItem(OleInvoiceItem poi, OlePaymentRequestDocument preq) {
386         this(poi, preq, new HashMap<String, ExpiredOrClosedAccountEntry>());
387     }
388 
389 
390     /**
391      * Constructs a new payment request item, but also merges expired accounts.
392      *
393      * @param poi                        - purchase order item
394      * @param preq                       - payment request document
395      * @param expiredOrClosedAccountList - list of expired or closed accounts to merge
396      */
397     public PaymentRequestItem(OleInvoiceItem poi, OlePaymentRequestDocument preq, HashMap<String, ExpiredOrClosedAccountEntry> expiredOrClosedAccountList) {
398 
399         // copy base attributes w/ extra array of fields not to be copied
400         PurApObjectUtils.populateFromBaseClass(PurApItemBase.class, poi, this, PurapConstants.PREQ_ITEM_UNCOPYABLE_FIELDS);
401 
402         setItemDescription(poi.getItemDescription());
403 
404         //New Source Line should be set for PaymentRequestItem
405         resetAccount();
406 
407         // set up accounts
408         List accounts = new ArrayList();
409         for (PurApAccountingLine account : poi.getSourceAccountingLines()) {
410             InvoiceAccount poa = (InvoiceAccount) account;
411 
412             // check if this account is expired/closed and replace as needed
413             SpringContext.getBean(AccountsPayableService.class).processExpiredOrClosedAccount(poa, expiredOrClosedAccountList);
414 
415             //KFSMI-4522 copy an accounting line with zero dollar amount if system parameter allows
416             if (poa.getAmount().isZero()) {
417                 if (SpringContext.getBean(AccountsPayableService.class).canCopyAccountingLinesWithZeroAmount()) {
418                     accounts.add(new PaymentRequestAccount(this, poa));
419                 }
420             } else {
421                 accounts.add(new PaymentRequestAccount(this, poa));
422             }
423         }
424 
425         this.setSourceAccountingLines(accounts);
426         this.getUseTaxItems().clear();
427         //List<PurApItemUseTax> newUseTaxItems = new ArrayList<PurApItemUseTax>();
428         /// this.setUseTaxItems(newUseTaxItems);
429         //copy use tax items over, and blank out keys (useTaxId and itemIdentifier)
430         /*
431         this.getUseTaxItems().clear();
432         for (PurApItemUseTax useTaxItem : poi.getUseTaxItems()) {
433             PaymentRequestItemUseTax newItemUseTax = new PaymentRequestItemUseTax(useTaxItem);
434             this.getUseTaxItems().add(newItemUseTax);
435 
436         }
437         */
438 
439         // clear amount and desc on below the line - we probably don't need that null
440         // itemType check but it's there just in case remove if it causes problems
441         // also do this if of type service
442         if ((ObjectUtils.isNotNull(this.getItemType()) && this.getItemType().isAmountBasedGeneralLedgerIndicator())) {
443             // setting unit price to be null to be more consistent with other below the line
444             // this.setItemUnitPrice(null);
445         }
446 
447         // copy custom
448         this.purchaseOrderItemUnitPrice = poi.getItemUnitPrice();
449 //        this.purchaseOrderCommodityCode = poi.getPurchaseOrderCommodityCd();
450 
451         // set doc fields
452         this.setPurapDocumentIdentifier(preq.getPurapDocumentIdentifier());
453         this.setPurapDocument(preq);
454     }
455 
456 }