2010-11-30 09:41:08 +11:00
|
|
|
<?php defined('SYSPATH') or die('No direct access allowed.');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class provides OSB exporting capabilities for Quickbooks.
|
|
|
|
*
|
2013-03-20 09:35:19 +11:00
|
|
|
* @package Export
|
|
|
|
* @category Helpers
|
2010-11-30 09:41:08 +11:00
|
|
|
* @author Deon George
|
2013-03-20 09:35:19 +11:00
|
|
|
* @copyright (c) 2009-2013 Open Source Billing
|
2010-11-30 09:41:08 +11:00
|
|
|
* @license http://dev.osbill.net/license.html
|
|
|
|
*/
|
|
|
|
class Export_Quicken extends Export {
|
|
|
|
public function export() {
|
2011-10-14 16:44:12 +11:00
|
|
|
if (! empty($_POST['id']) AND count($_POST['id'])) {
|
2010-11-30 09:41:08 +11:00
|
|
|
$qo = new Quicken;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
foreach ($_POST['id'] as $pid) {
|
2012-11-10 10:13:57 +11:00
|
|
|
$po = ORM::factory('Payment',$pid);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($po->loaded()) {
|
2010-11-30 09:41:08 +11:00
|
|
|
$invoice_ids = array();
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
foreach ($po->payment_item->find_all() as $pio) {
|
2010-11-30 09:41:08 +11:00
|
|
|
// If our invoice ID is not blank, then the payment was applied to an invoice
|
|
|
|
if ($pio->invoice->id) {
|
|
|
|
// Record our invoice IDs for the summary
|
|
|
|
array_push($invoice_ids,$pio->invoice->id);
|
|
|
|
|
|
|
|
$qio = new Quicken_Invoice;
|
|
|
|
$qio->TRNSID = sprintf('%06s',$pio->invoice->id);
|
|
|
|
$qio->DATE = date('m/d/Y',$pio->invoice->date_orig);
|
|
|
|
$qio->MEMO = 'Import from OSB';
|
|
|
|
$qio->CLEAR = 'N';
|
|
|
|
$qio->TOPRINT = 'N';
|
|
|
|
$qio->PAID = 'N';
|
2011-10-14 16:44:12 +11:00
|
|
|
$qio->ADDR1 = $po->account->address1;
|
|
|
|
$qio->ADDR2 = $po->account->address2;
|
|
|
|
$qio->ADDR3 = sprintf('%s, %s %s',$po->account->city,$po->account->state,$po->account->zip);
|
2010-11-30 09:41:08 +11:00
|
|
|
// @todo - should be configurable
|
|
|
|
$qio->TERMS = '7 Days';
|
|
|
|
// @todo - should be configurable
|
2013-02-12 22:14:59 +11:00
|
|
|
$qio->INVTITLE = Company::instance()->name().' Invoice';
|
2010-11-30 09:41:08 +11:00
|
|
|
// @todo - should be configurable
|
2013-02-12 22:14:59 +11:00
|
|
|
$qio->INVMEMO = 'Thank you for using '.Company::instance()->name();
|
2010-11-30 09:41:08 +11:00
|
|
|
$qio->DOCNUM = sprintf('%06s',$pio->invoice->id);
|
|
|
|
$qio->DUEDATE = date('m/d/Y',$pio->invoice->due_date);
|
2011-10-14 16:44:12 +11:00
|
|
|
$qio->AMOUNT = sprintf('%3.2f',$pio->invoice->total());
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($po->account->company)
|
|
|
|
$qio->NAME = $po->account->company;
|
2010-11-30 09:41:08 +11:00
|
|
|
else
|
2011-10-14 16:44:12 +11:00
|
|
|
$qio->NAME = sprintf('%s %s',$po->account->last_name,$po->account->first_name);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
// Other Quicken fields not used.
|
|
|
|
#$qio->CLASS = '';
|
|
|
|
#$qio->SHIPVIA = '';
|
|
|
|
#$qio->SHIPDATE = '';
|
|
|
|
#$qio->OTHER1 = '';
|
|
|
|
#$qio->REP = '';
|
|
|
|
#$qio->FOB = '';
|
|
|
|
#$qio->PONUM = '';
|
|
|
|
#$qio->SADDR1 = '';
|
|
|
|
#$qio->SADDR2 = '';
|
|
|
|
#$qio->SADDR3 = '';
|
|
|
|
#$qio->SADDR4 = '';
|
|
|
|
#$qio->SADDR5 = '';
|
|
|
|
|
|
|
|
// Add the items to the invoice
|
|
|
|
foreach ($pio->invoice->invoice_item->find_all() as $iio) {
|
|
|
|
$qto = new Quicken_InvoiceItem;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($iio->period())
|
|
|
|
$daterange = $iio->period();
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// @todo This should go.
|
2010-11-30 09:41:08 +11:00
|
|
|
elseif ($iio->product_attr && preg_match('/^a/',$iio->product_attr)) {
|
|
|
|
echo 'Uncaptured';die();
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// @todo This should go.
|
2010-11-30 09:41:08 +11:00
|
|
|
} elseif ($iio->product_attr && preg_match('/^s/',$iio->product_attr))
|
|
|
|
$daterange = preg_replace("/\r?\n/",' ',unserialize($iio->product_attr));
|
|
|
|
|
|
|
|
else
|
|
|
|
$daterange = '';
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($iio->product_id) {
|
2012-11-10 10:13:57 +11:00
|
|
|
$mo = ORM::factory('Module',array('name'=>'product'));
|
|
|
|
$eo = ORM::factory('Export')
|
|
|
|
->where('plugin_name','=',strtolower($this->plugin))
|
2011-10-14 16:44:12 +11:00
|
|
|
->and_where('module_id','=',$mo->id)
|
2010-11-30 09:41:08 +11:00
|
|
|
->and_where('item_id','=',$iio->product_id)
|
|
|
|
->find();
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($eo->loaded()) {
|
2012-08-01 22:34:59 +10:00
|
|
|
$qto->ACCNT = $eo->map_data['account'];
|
|
|
|
$qto->INVITEM = $eo->map_data['item'];
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
} else {
|
2011-10-14 16:44:12 +11:00
|
|
|
throw new Kohana_Exception('Missing product map data for :product (:id)',
|
2013-04-26 11:42:09 +10:00
|
|
|
array(':product'=>$iio->product->title(),':id'=>$iio->product_id));
|
2011-10-14 16:44:12 +11:00
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
$qto->ACCNT = 'Other Income';
|
|
|
|
$qto->INVITEM = 'Product:Unknown';
|
|
|
|
}
|
|
|
|
|
|
|
|
$qto->MEMO = sprintf('%s (%s)',
|
|
|
|
$iio->product->product_translate->find()->name,$daterange);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$qto->ACCNT = 'Other Income';
|
|
|
|
$qto->INVITEM = 'Unknown';
|
|
|
|
$qto->MEMO = sprintf('%s (%s)',
|
2012-08-01 22:34:59 +10:00
|
|
|
$iio->product->product_translate->find()->name,$daterange);
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
$qto->CLEAR = 'N';
|
|
|
|
$qto->QNTY = -1;
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($pio->invoice->tax()) {
|
2010-11-30 09:41:08 +11:00
|
|
|
$qto->TAXABLE = 'Y';
|
|
|
|
# @todo, get this from OSB
|
|
|
|
$qto->TAXCODE = 'GST';
|
|
|
|
$qto->TAXRATE = sprintf('%3.2f%%','0.10');
|
2011-10-14 16:44:12 +11:00
|
|
|
$qto->TAXAMOUNT = sprintf('%3.2f',$iio->tax()*-1);
|
2010-11-30 09:41:08 +11:00
|
|
|
} else {
|
|
|
|
$qto->TAXAMOUNT = 0;
|
|
|
|
}
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
// @todo This rounding should be a system config.
|
|
|
|
$qto->PRICE = sprintf('%3.2f',round($iio->subtotal()-$iio->discount(),2));
|
|
|
|
$qto->AMOUNT = sprintf('%3.2f',round($iio->subtotal()-$iio->discount(),2)*-1);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
$qio->addInvoiceItem($qto);
|
|
|
|
}
|
|
|
|
|
2012-08-01 22:34:59 +10:00
|
|
|
// Add credits as a other item
|
|
|
|
$qto = new Quicken_InvoiceItem;
|
|
|
|
$qto->ACCNT = 'Other Income';
|
|
|
|
$qto->INVITEM = 'Product:Unknown';
|
|
|
|
$qto->CLEAR = 'N';
|
|
|
|
$qto->QNTY = 1;
|
|
|
|
$qto->MEMO = 'Credit Item';
|
|
|
|
|
|
|
|
if ($pio->invoice->tax()) {
|
|
|
|
$qto->TAXABLE = 'Y';
|
|
|
|
# @todo, get this from OSB
|
|
|
|
$qto->TAXCODE = 'GST';
|
|
|
|
$qto->TAXRATE = sprintf('%3.2f%%','0.10');
|
2013-02-26 14:19:32 +11:00
|
|
|
$tax = round($pio->invoice->total_credits()/11,2);
|
2012-08-01 22:34:59 +10:00
|
|
|
$qto->TAXAMOUNT = sprintf('%3.2f',$tax);
|
|
|
|
} else {
|
|
|
|
$qto->TAXAMOUNT = 0;
|
|
|
|
}
|
|
|
|
|
2013-02-26 14:19:32 +11:00
|
|
|
$qto->PRICE = sprintf('%3.2f',round(($pio->invoice->total_credits()-$tax)*-1,2));
|
|
|
|
$qto->AMOUNT = sprintf('%3.2f',round(($pio->invoice->total_credits()-$tax),2));
|
2012-08-01 22:34:59 +10:00
|
|
|
$qio->addInvoiceItem($qto);
|
|
|
|
|
2010-11-30 09:41:08 +11:00
|
|
|
$qo->addInvoice($qio);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$qpo = new Quicken_Payment;
|
2011-10-14 16:44:12 +11:00
|
|
|
$qpo->AMOUNT = sprintf('%3.2f',$po->total_amt);
|
|
|
|
$qpo->TRNSID = sprintf('P%06s',$po->id);
|
|
|
|
$qpo->DATE = date('m/d/Y',$po->date_payment);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
// @todo this should be from a function - when no invoice is paid we cant use $qio
|
2011-10-14 16:44:12 +11:00
|
|
|
if ($po->account->company)
|
|
|
|
$qpo->NAME = $po->account->company;
|
2010-11-30 09:41:08 +11:00
|
|
|
else
|
2011-10-14 16:44:12 +11:00
|
|
|
$qpo->NAME = sprintf('%s %s',$po->account->last_name,$po->account->first_name);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
$qpo->CLEAR = 'N';
|
2011-10-14 16:44:12 +11:00
|
|
|
$qpo->MEMO = sprintf('Payment for invoice(s) %s (%s)',implode(':',$invoice_ids),$po->checkout->name);
|
2010-11-30 09:41:08 +11:00
|
|
|
|
|
|
|
// @todo Accounts/Payment should be configurable
|
2013-04-20 11:40:44 +10:00
|
|
|
switch ($po->checkout->plugin) {
|
2010-11-30 09:41:08 +11:00
|
|
|
// @todo this is direct debit
|
2013-04-20 11:40:44 +10:00
|
|
|
case 'DD_EZYPAY':
|
2010-11-30 09:41:08 +11:00
|
|
|
$qpo->PAYMETH = 'DirectDebit';
|
|
|
|
$qpo->ACCNT = 'Ezypay';
|
|
|
|
break;
|
|
|
|
|
2013-04-20 11:40:44 +10:00
|
|
|
case 'REMIT_CHEQUE':
|
2010-11-30 09:41:08 +11:00
|
|
|
$qpo->PAYMETH = 'Cheque';
|
|
|
|
$qpo->ACCNT = 'Undeposited Funds';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'REMIT_BANK_WIRE':
|
|
|
|
$qpo->PAYMETH = 'DirectCredit';
|
|
|
|
$qpo->ACCNT = 'Bendigo Bank';
|
|
|
|
break;
|
|
|
|
|
2013-04-20 11:40:44 +10:00
|
|
|
case 'PAYPAL_CART':
|
2010-11-30 09:41:08 +11:00
|
|
|
$qpo->PAYMETH = 'Paypal';
|
|
|
|
$qpo->ACCNT = 'Paypal';
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$qpo->PAYMETH = 'TBA';
|
|
|
|
$qpo->ACCNT = 'Undeposited Funds';
|
|
|
|
}
|
|
|
|
|
2011-10-14 16:44:12 +11:00
|
|
|
if (isset($qio))
|
|
|
|
$qio->addPayment($qpo);
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! empty($qo))
|
2011-07-22 11:04:20 +10:00
|
|
|
$this->response->body($qo->export());
|
2010-11-30 09:41:08 +11:00
|
|
|
|
2011-07-22 11:04:20 +10:00
|
|
|
$this->response->send_file(TRUE,'quicken-import.iif',array('mime_type'=>'text/plain'));
|
2010-11-30 09:41:08 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
?>
|