2010-11-30 09:41:08 +11:00
< ? php defined ( 'SYSPATH' ) or die ( 'No direct access allowed.' );
/**
2011-12-20 16:46:10 +11:00
* This class supports OSB payments .
2010-11-30 09:41:08 +11:00
*
* @ package OSB
2011-12-20 16:46:10 +11:00
* @ subpackage Payment
2010-11-30 09:41:08 +11:00
* @ category Models
* @ author Deon George
* @ copyright ( c ) 2010 Open Source Billing
* @ license http :// dev . osbill . net / license . html
*/
2012-12-11 08:48:30 +11:00
class Model_Payment extends ORM_OSB {
2010-11-30 09:41:08 +11:00
// Relationships
protected $_has_many = array (
2011-07-22 11:04:20 +10:00
'payment_item' => array ( 'far_key' => 'id' ),
2010-11-30 09:41:08 +11:00
'invoice' => array ( 'through' => 'payment_item' ),
);
protected $_belongs_to = array (
'account' => array (),
'checkout' => array ( 'foreign_key' => 'checkout_plugin_id' ),
);
2011-08-02 16:20:11 +10:00
protected $_sorting = array (
'date_payment' => 'DESC'
);
2010-11-30 09:41:08 +11:00
2011-07-22 11:04:20 +10:00
protected $_display_filters = array (
'date_payment' => array (
array ( 'Config::date' , array ( ':value' )),
),
'total_amt' => array (
array ( 'Currency::display' , array ( ':value' )),
),
2010-11-30 09:41:08 +11:00
);
2011-12-20 16:46:10 +11:00
// Items belonging to an invoice
private $payment_items = array ();
2011-12-29 13:52:24 +11:00
public function __construct ( $id = NULL ) {
// Load our model.
parent :: __construct ( $id );
// Load our sub items
if ( $this -> loaded ())
$this -> payment_items = $this -> payment_item -> find_all () -> as_array ();
}
2011-12-20 16:46:10 +11:00
/**
* Return a list of invoice items for this payment .
*/
public function items () {
2011-12-29 13:52:24 +11:00
return $this -> payment_items ;
2011-12-20 16:46:10 +11:00
}
/**
* Add an item to an invoice
*/
2011-12-29 13:52:24 +11:00
public function add_item ( $iid ) {
// Find our id, if it exists
foreach ( $this -> payment_items as $pio )
if ( $pio -> invoice_id == $iid )
return $pio ;
2011-12-20 16:46:10 +11:00
2011-12-29 13:52:24 +11:00
// New Item
2011-12-20 16:46:10 +11:00
$c = count ( $this -> payment_items );
2012-11-10 10:13:57 +11:00
$this -> payment_items [ $c ] = ORM :: factory ( 'Payment_Item' );
2011-12-29 13:52:24 +11:00
$this -> payment_items [ $c ] -> invoice_id = $iid ;
2011-12-20 16:46:10 +11:00
return $this -> payment_items [ $c ];
}
2010-11-30 09:41:08 +11:00
/**
* Find all items that are exportable .
*
* @ param int $start List payments that were modified this many days ago
*/
public function export ( $start ) {
2012-11-10 10:13:57 +11:00
return ORM :: factory ( 'Payment' )
2010-11-30 09:41:08 +11:00
-> where ( 'date_payment' , '>=' , time () - $start * 86400 )
-> find_all ();
}
2011-08-02 16:20:11 +10:00
/**
* Calculate the remaining balance available for this payment
*/
public function balance ( $format = FALSE ) {
$t = 0 ;
foreach ( $this -> payment_item -> find_all () as $pio )
$t += $pio -> alloc_amt ;
return $format ? Currency :: display ( $this -> total_amt - $t ) : $this -> total_amt - $t ;
}
/**
* Return a list of invoices that this payment is applied to
*/
public function invoices () {
$invoices = array ();
foreach ( $this -> payment_item -> find_all () as $pio )
array_push ( $invoices , $pio -> invoice );
return $invoices ;
}
public function invoicelist () {
return join ( ',' , $this -> invoices ());
}
2011-08-16 12:27:19 +10:00
/** LIST FUNCTIONS **/
public function list_unapplied () {
$pi = array ();
// @todo database suffix needs to be dynamically calculated
foreach ( DB :: Query ( Database :: SELECT ,
2011-10-14 16:44:12 +11:00
sprintf ( 'SELECT A.id AS id,A.total_amt as total_amt FROM ab_%s A LEFT JOIN ab_%s B ON (A.site_id=B.site_id AND A.id=B.payment_id) WHERE (A.refund_status=0 OR A.refund_status IS NULL) GROUP BY A.id HAVING ROUND(SUM(IFNULL(B.alloc_amt,0)),2)!=A.total_amt ORDER BY account_id,payment_id' , 'payment' , 'payment_item' ))
2011-08-16 12:27:19 +10:00
-> execute () as $values ) {
array_push ( $pi , $values [ 'id' ]);
}
return $this -> where ( 'id' , 'IN' , $pi ) -> order_by ( 'account_id' ) -> find_all ();
}
2011-09-27 21:22:13 +10:00
public function list_recent_table () {
// @todo This should be in a config file.
$css = '<style type="text/css">' ;
$css .= 'table.box-left { border: 1px solid #AAAACC; margin-right: auto; }' ;
$css .= 'tr.head { font-weight: bold; }' ;
$css .= 'td.head { font-weight: bold; }' ;
$css .= 'td.right { text-align: right; }' ;
$css .= 'tr.odd { background-color: #FCFCFE; }' ;
$css .= 'tr.even { background-color: #F6F6F8; }' ;
$css .= '</style>' ;
return $css . Table :: display (
$this -> limit ( 10 ) -> find_all (),
25 ,
2011-12-20 16:46:10 +11:00
array (
2011-09-27 21:22:13 +10:00
'id' => array ( 'label' => 'ID' ),
'date_payment' => array ( 'label' => 'Date' ),
'checkout->display("name")' => array ( 'label' => 'Method' ),
'total_amt' => array ( 'label' => 'Total' , 'class' => 'right' ),
'balance(TRUE)' => array ( 'label' => 'Balance' , 'class' => 'right' ),
'invoicelist()' => array ( 'label' => 'Invoices' ),
),
2011-12-20 16:46:10 +11:00
array (
2011-09-27 21:22:13 +10:00
'type' => 'list' ,
));
}
2011-12-20 16:46:10 +11:00
public function save ( Validation $validation = NULL ) {
// Our items will be clobbered once we save the object, so we need to save it here.
$items = $this -> items ();
$this -> source_id = Auth :: instance () -> get_user () -> id ;
$this -> ip = Request :: $client_ip ;
2011-12-29 13:52:24 +11:00
// Make sure we dont over allocate
$t = 0 ;
$msg = '' ;
foreach ( $items as $pio ) {
// Only need to check items that ave actually changed.
if ( $pio -> changed ()) {
2012-11-10 10:13:57 +11:00
$old_pio = ORM :: factory ( 'Payment_Item' , $pio -> id );
2011-12-29 13:52:24 +11:00
2012-11-10 10:13:57 +11:00
if (( $it = $pio -> invoice -> due () + ORM :: factory ( 'Payment_Item' , $pio -> id ) -> alloc_amt - $pio -> alloc_amt ) < 0 )
2011-12-29 13:52:24 +11:00
$msg .= ( $msg ? ' ' : '' ) . sprintf ( 'Invoice %s over allocated by %3.2f.' , $pio -> invoice_id , $it );
}
$t += $pio -> alloc_amt ;
}
if ( $t > ( float ) $this -> total_amt )
$msg .= ( $msg ? ' ' : '' ) . sprintf ( 'Payment over allocated by %3.2f.' , $t - $this -> total_amt );
if ( $msg ) {
SystemMessage :: add ( array (
'title' => 'Payment NOT Recorded' ,
'type' => 'warning' ,
'body' => $msg ,
));
return FALSE ;
}
2011-12-20 16:46:10 +11:00
// Save the payment
parent :: save ( $validation );
// Need to save the associated items and their taxes
2013-02-02 17:35:21 +11:00
if ( ! $this -> changed () OR $this -> saved ()) {
2011-12-20 16:46:10 +11:00
foreach ( $items as $pio ) {
2013-02-02 17:35:21 +11:00
// Skip applying 0 payments to invoices.
if ( Currency :: round ( $pio -> alloc_amt ) == 0 AND ! $pio -> loaded ())
continue ;
2011-12-20 16:46:10 +11:00
$pio -> payment_id = $this -> id ;
if ( ! $pio -> check ()) {
// @todo Mark payment as cancelled and write a memo, then...
throw new Kohana_Exception ( 'Problem saving payment_item for invoice :invoice - Failed check()' , array ( ':invoice' => $invoice -> id ));
}
$pio -> save ();
if ( ! $pio -> saved ()) {
// @todo Mark payment as cancelled and write a memo, then...
throw new Kohana_Exception ( 'Problem saving payment_item for invoice :invoice - Failed save()' , array ( ':invoice' => $invoice -> id ));
}
}
2013-02-02 17:35:21 +11:00
} else {
2011-12-20 16:46:10 +11:00
throw new Kohana_Exception ( 'Couldnt save payment for some reason?' );
2013-02-02 17:35:21 +11:00
}
2011-12-20 16:46:10 +11:00
return TRUE ;
}
2010-11-30 09:41:08 +11:00
}
2011-08-16 12:27:19 +10:00
?>