2016-10-09 14:36:03 -05:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Transaction.php
|
|
|
|
* Copyright (C) 2016 thegrumpydictator@gmail.com
|
|
|
|
*
|
|
|
|
* This software may be modified and distributed under the terms of the
|
|
|
|
* Creative Commons Attribution-ShareAlike 4.0 International License.
|
|
|
|
*
|
|
|
|
* See the LICENSE file for details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
declare(strict_types = 1);
|
|
|
|
|
|
|
|
namespace FireflyIII\Support\Twig;
|
|
|
|
|
|
|
|
use Amount;
|
|
|
|
use FireflyIII\Models\AccountType;
|
|
|
|
use FireflyIII\Models\Transaction as TransactionModel;
|
2016-12-29 02:02:23 -06:00
|
|
|
use FireflyIII\Models\TransactionCurrency;
|
2016-10-09 14:36:03 -05:00
|
|
|
use FireflyIII\Models\TransactionType;
|
2017-02-19 00:34:39 -06:00
|
|
|
use Steam;
|
2016-10-09 14:36:03 -05:00
|
|
|
use Twig_Extension;
|
|
|
|
use Twig_SimpleFilter;
|
|
|
|
use Twig_SimpleFunction;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class Transaction
|
|
|
|
*
|
|
|
|
* @package FireflyIII\Support\Twig
|
|
|
|
*/
|
|
|
|
class Transaction extends Twig_Extension
|
|
|
|
{
|
2016-12-29 02:02:23 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function formatAnything(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'formatAnything', function (TransactionCurrency $currency, string $amount): string {
|
|
|
|
|
|
|
|
return Amount::formatAnything($currency, $amount, true);
|
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-10-15 05:39:34 -05:00
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
2016-12-29 02:02:23 -06:00
|
|
|
public function formatAnythingPlain(): Twig_SimpleFunction
|
2016-10-15 05:39:34 -05:00
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
2016-12-29 02:02:23 -06:00
|
|
|
'formatAnythingPlain', function (TransactionCurrency $currency, string $amount): string {
|
2016-10-15 05:39:34 -05:00
|
|
|
|
2016-12-29 02:02:23 -06:00
|
|
|
return Amount::formatAnything($currency, $amount, false);
|
2016-10-15 05:39:34 -05:00
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-10-09 14:36:03 -05:00
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
2016-12-29 02:02:23 -06:00
|
|
|
public function formatByCode(): Twig_SimpleFunction
|
2016-10-09 14:36:03 -05:00
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
2016-12-29 02:02:23 -06:00
|
|
|
'formatByCode', function (string $currencyCode, string $amount): string {
|
2016-10-09 14:36:03 -05:00
|
|
|
|
2016-12-29 02:02:23 -06:00
|
|
|
return Amount::formatByCode($currencyCode, $amount, true);
|
2016-10-09 14:36:03 -05:00
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getFilters(): array
|
|
|
|
{
|
|
|
|
$filters = [
|
|
|
|
$this->typeIconTransaction(),
|
|
|
|
];
|
|
|
|
|
|
|
|
return $filters;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getFunctions(): array
|
|
|
|
{
|
|
|
|
$functions = [
|
2016-12-29 02:02:23 -06:00
|
|
|
$this->formatAnything(),
|
|
|
|
$this->formatAnythingPlain(),
|
2016-10-09 14:36:03 -05:00
|
|
|
$this->transactionSourceAccount(),
|
|
|
|
$this->transactionDestinationAccount(),
|
|
|
|
$this->optionalJournalAmount(),
|
|
|
|
$this->transactionBudgets(),
|
2016-10-15 05:39:34 -05:00
|
|
|
$this->transactionIdBudgets(),
|
2016-10-09 14:36:03 -05:00
|
|
|
$this->transactionCategories(),
|
2016-10-15 05:39:34 -05:00
|
|
|
$this->transactionIdCategories(),
|
2016-10-09 14:36:03 -05:00
|
|
|
$this->splitJournalIndicator(),
|
2016-12-29 02:02:23 -06:00
|
|
|
$this->formatByCode(),
|
2016-10-09 14:36:03 -05:00
|
|
|
];
|
|
|
|
|
|
|
|
return $functions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the name of the extension.
|
|
|
|
*
|
|
|
|
* @return string The extension name
|
|
|
|
*/
|
|
|
|
public function getName(): string
|
|
|
|
{
|
|
|
|
return 'transaction';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function optionalJournalAmount(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'optionalJournalAmount', function (int $journalId, string $transactionAmount, string $code, string $type): string {
|
2016-12-29 02:02:23 -06:00
|
|
|
// get amount of journal:
|
|
|
|
$amount = strval(TransactionModel::where('transaction_journal_id', $journalId)->whereNull('deleted_at')->where('amount', '<', 0)->sum('amount'));
|
|
|
|
// display deposit and transfer positive
|
2016-10-09 14:36:03 -05:00
|
|
|
if ($type === TransactionType::DEPOSIT || $type === TransactionType::TRANSFER) {
|
|
|
|
$amount = bcmul($amount, '-1');
|
|
|
|
}
|
|
|
|
|
2016-12-29 02:02:23 -06:00
|
|
|
// not equal to transaction amount?
|
|
|
|
if (bccomp($amount, $transactionAmount) !== 0 && bccomp($amount, bcmul($transactionAmount, '-1')) !== 0) {
|
|
|
|
//$currency =
|
2016-12-29 02:05:02 -06:00
|
|
|
return sprintf(' (%s)', Amount::formatByCode($code, $amount, true));
|
2016-10-09 14:36:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
|
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-10-10 00:49:55 -05:00
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
2016-10-09 14:36:03 -05:00
|
|
|
public function splitJournalIndicator(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'splitJournalIndicator', function (int $journalId) {
|
|
|
|
$count = TransactionModel::where('transaction_journal_id', $journalId)->whereNull('deleted_at')->count();
|
|
|
|
if ($count > 2) {
|
2016-11-05 12:57:45 -05:00
|
|
|
return '<i class="fa fa-fw fa-share-alt" aria-hidden="true"></i>';
|
2016-10-09 14:36:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
|
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionBudgets(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionBudgets', function (TransactionModel $transaction): string {
|
2016-10-15 05:39:34 -05:00
|
|
|
return $this->getTransactionBudgets($transaction);
|
2016-10-09 14:36:03 -05:00
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionCategories(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionCategories', function (TransactionModel $transaction): string {
|
2016-10-15 05:39:34 -05:00
|
|
|
return $this->getTransactionCategories($transaction);
|
2016-10-09 14:36:03 -05:00
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionDestinationAccount(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionDestinationAccount', function (TransactionModel $transaction): string {
|
|
|
|
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->account_encrypted), $transaction->account_name);
|
2016-10-09 14:36:03 -05:00
|
|
|
$id = intval($transaction->account_id);
|
|
|
|
$type = $transaction->account_type;
|
|
|
|
|
2016-11-12 14:08:14 -06:00
|
|
|
// name is present in object, use that one:
|
|
|
|
if (bccomp($transaction->transaction_amount, '0') === -1 && !is_null($transaction->opposing_account_id)) {
|
|
|
|
|
2016-12-07 01:58:55 -06:00
|
|
|
$name = $transaction->opposing_account_name;
|
2016-11-12 14:08:14 -06:00
|
|
|
$id = intval($transaction->opposing_account_id);
|
2017-03-03 05:55:28 -06:00
|
|
|
$type = $transaction->opposing_account_type;
|
2016-11-12 14:08:14 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find the opposing account and use that one:
|
|
|
|
if (bccomp($transaction->transaction_amount, '0') === -1 && is_null($transaction->opposing_account_id)) {
|
2016-10-09 14:36:03 -05:00
|
|
|
// if the amount is negative, find the opposing account and use that one:
|
|
|
|
$journalId = $transaction->journal_id;
|
|
|
|
/** @var TransactionModel $other */
|
2016-11-28 13:38:03 -06:00
|
|
|
$other = TransactionModel::where('transaction_journal_id', $journalId)->where('transactions.id', '!=', $transaction->id)
|
2016-12-14 11:59:12 -06:00
|
|
|
->where('amount', '=', bcmul($transaction->transaction_amount, '-1'))->where('identifier', $transaction->identifier)
|
|
|
|
->leftJoin('accounts', 'accounts.id', '=', 'transactions.account_id')
|
|
|
|
->leftJoin('account_types', 'account_types.id', '=', 'accounts.account_type_id')
|
|
|
|
->first(['transactions.account_id', 'accounts.encrypted', 'accounts.name', 'account_types.type']);
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($other->encrypted), $other->name);
|
2016-10-09 14:36:03 -05:00
|
|
|
$id = $other->account_id;
|
|
|
|
$type = $other->type;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($type === AccountType::CASH) {
|
|
|
|
return '<span class="text-success">(cash)</span>';
|
|
|
|
}
|
|
|
|
|
2016-11-02 01:04:14 -05:00
|
|
|
return sprintf('<a title="%1$s" href="%2$s">%1$s</a>', e($name), route('accounts.show', [$id]));
|
2016-10-09 14:36:03 -05:00
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-10-15 05:39:34 -05:00
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionIdBudgets(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionIdBudgets', function (int $transactionId): string {
|
|
|
|
$transaction = TransactionModel::find($transactionId);
|
|
|
|
|
|
|
|
return $this->getTransactionBudgets($transaction);
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionIdCategories(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionIdCategories', function (int $transactionId): string {
|
|
|
|
$transaction = TransactionModel::find($transactionId);
|
|
|
|
|
|
|
|
return $this->getTransactionCategories($transaction);
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-10-09 14:36:03 -05:00
|
|
|
/**
|
|
|
|
* @return Twig_SimpleFunction
|
|
|
|
*/
|
|
|
|
public function transactionSourceAccount(): Twig_SimpleFunction
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFunction(
|
|
|
|
'transactionSourceAccount', function (TransactionModel $transaction): string {
|
|
|
|
|
2016-11-12 14:06:48 -06:00
|
|
|
// if the amount is negative, assume that the current account (the one in $transaction) is indeed the source account.
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->account_encrypted), $transaction->account_name);
|
2016-10-09 14:36:03 -05:00
|
|
|
$id = intval($transaction->account_id);
|
|
|
|
$type = $transaction->account_type;
|
|
|
|
|
2016-11-12 14:06:48 -06:00
|
|
|
// name is present in object, use that one:
|
|
|
|
if (bccomp($transaction->transaction_amount, '0') === 1 && !is_null($transaction->opposing_account_id)) {
|
|
|
|
|
2016-12-09 07:21:26 -06:00
|
|
|
$name = $transaction->opposing_account_name;
|
2016-11-12 14:06:48 -06:00
|
|
|
$id = intval($transaction->opposing_account_id);
|
2017-03-03 05:55:28 -06:00
|
|
|
$type = $transaction->opposing_account_type;
|
2016-11-12 14:06:48 -06:00
|
|
|
}
|
|
|
|
// Find the opposing account and use that one:
|
|
|
|
if (bccomp($transaction->transaction_amount, '0') === 1 && is_null($transaction->opposing_account_id)) {
|
2016-10-09 14:36:03 -05:00
|
|
|
$journalId = $transaction->journal_id;
|
|
|
|
/** @var TransactionModel $other */
|
2016-11-28 13:38:03 -06:00
|
|
|
$other = TransactionModel::where('transaction_journal_id', $journalId)->where('transactions.id', '!=', $transaction->id)
|
2016-12-14 11:59:12 -06:00
|
|
|
->where('amount', '=', bcmul($transaction->transaction_amount, '-1'))->where('identifier', $transaction->identifier)
|
|
|
|
->leftJoin('accounts', 'accounts.id', '=', 'transactions.account_id')
|
|
|
|
->leftJoin('account_types', 'account_types.id', '=', 'accounts.account_type_id')
|
|
|
|
->first(['transactions.account_id', 'accounts.encrypted', 'accounts.name', 'account_types.type']);
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($other->encrypted), $other->name);
|
2016-10-09 14:36:03 -05:00
|
|
|
$id = $other->account_id;
|
|
|
|
$type = $other->type;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($type === AccountType::CASH) {
|
|
|
|
return '<span class="text-success">(cash)</span>';
|
|
|
|
}
|
|
|
|
|
2016-11-02 01:04:14 -05:00
|
|
|
return sprintf('<a title="%1$s" href="%2$s">%1$s</a>', e($name), route('accounts.show', [$id]));
|
2016-10-09 14:36:03 -05:00
|
|
|
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @SuppressWarnings(PHPMD.CyclomaticComplexity) // it's 5.
|
|
|
|
*
|
|
|
|
* @return Twig_SimpleFilter
|
|
|
|
*/
|
|
|
|
public function typeIconTransaction(): Twig_SimpleFilter
|
|
|
|
{
|
|
|
|
return new Twig_SimpleFilter(
|
|
|
|
'typeIconTransaction', function (TransactionModel $transaction): string {
|
|
|
|
|
|
|
|
switch ($transaction->transaction_type_type) {
|
|
|
|
case TransactionType::WITHDRAWAL:
|
2016-11-02 14:08:11 -05:00
|
|
|
$txt = sprintf('<i class="fa fa-long-arrow-left fa-fw" title="%s"></i>', trans('firefly.withdrawal'));
|
2016-10-09 14:36:03 -05:00
|
|
|
break;
|
|
|
|
case TransactionType::DEPOSIT:
|
2016-11-02 01:04:14 -05:00
|
|
|
$txt = sprintf('<i class="fa fa-long-arrow-right fa-fw" title="%s"></i>', trans('firefly.deposit'));
|
2016-10-09 14:36:03 -05:00
|
|
|
break;
|
|
|
|
case TransactionType::TRANSFER:
|
2016-11-02 01:04:14 -05:00
|
|
|
$txt = sprintf('<i class="fa fa-fw fa-exchange" title="%s"></i>', trans('firefly.transfer'));
|
2016-10-09 14:36:03 -05:00
|
|
|
break;
|
|
|
|
case TransactionType::OPENING_BALANCE:
|
2016-11-02 01:04:14 -05:00
|
|
|
$txt = sprintf('<i class="fa-fw fa fa-ban" title="%s"></i>', trans('firefly.openingBalance'));
|
2016-10-09 14:36:03 -05:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$txt = '';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $txt;
|
|
|
|
}, ['is_safe' => ['html']]
|
|
|
|
);
|
|
|
|
}
|
2016-10-15 05:39:34 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param TransactionModel $transaction
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function getTransactionBudgets(TransactionModel $transaction): string
|
|
|
|
{
|
2016-12-27 23:09:48 -06:00
|
|
|
// journal has a budget:
|
|
|
|
if (isset($transaction->transaction_journal_budget_id)) {
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->transaction_journal_budget_encrypted), $transaction->transaction_journal_budget_name);
|
2016-12-27 23:09:48 -06:00
|
|
|
|
2016-12-28 11:26:44 -06:00
|
|
|
return sprintf('<a href="%s" title="%s">%s</a>', route('budgets.show', [$transaction->transaction_journal_budget_id]), $name, $name);
|
2016-12-27 23:09:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// transaction has a budget
|
|
|
|
if (isset($transaction->transaction_budget_id)) {
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->transaction_budget_encrypted), $transaction->transaction_budget_name);
|
2016-12-27 23:09:48 -06:00
|
|
|
|
2016-12-28 11:26:44 -06:00
|
|
|
return sprintf('<a href="%s" title="%s">%s</a>', route('budgets.show', [$transaction->transaction_budget_id]), $name, $name);
|
2016-12-27 23:09:48 -06:00
|
|
|
}
|
|
|
|
|
2016-10-15 05:39:34 -05:00
|
|
|
// see if the transaction has a budget:
|
|
|
|
$budgets = $transaction->budgets()->get();
|
|
|
|
if ($budgets->count() === 0) {
|
|
|
|
$budgets = $transaction->transactionJournal()->first()->budgets()->get();
|
|
|
|
}
|
|
|
|
if ($budgets->count() > 0) {
|
|
|
|
$str = [];
|
|
|
|
foreach ($budgets as $budget) {
|
2016-12-28 11:26:44 -06:00
|
|
|
$str[] = sprintf('<a href="%s" title="%s">%s</a>', route('budgets.show', [$budget->id]), $budget->name, $budget->name);
|
2016-10-15 05:39:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return join(', ', $str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param TransactionModel $transaction
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function getTransactionCategories(TransactionModel $transaction): string
|
|
|
|
{
|
2016-12-27 23:09:48 -06:00
|
|
|
// journal has a category:
|
|
|
|
if (isset($transaction->transaction_journal_category_id)) {
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->transaction_journal_category_encrypted), $transaction->transaction_journal_category_name);
|
2016-12-27 23:09:48 -06:00
|
|
|
|
2016-12-28 11:26:44 -06:00
|
|
|
return sprintf('<a href="%s" title="%s">%s</a>', route('categories.show', [$transaction->transaction_journal_category_id]), $name, $name);
|
2016-12-27 23:09:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// transaction has a category:
|
|
|
|
if (isset($transaction->transaction_category_id)) {
|
2017-02-19 00:34:39 -06:00
|
|
|
$name = Steam::decrypt(intval($transaction->transaction_category_encrypted), $transaction->transaction_category_name);
|
2016-12-27 23:09:48 -06:00
|
|
|
|
2016-12-28 11:26:44 -06:00
|
|
|
return sprintf('<a href="%s" title="%s">%s</a>', route('categories.show', [$transaction->transaction_category_id]), $name, $name);
|
2016-12-27 23:09:48 -06:00
|
|
|
}
|
|
|
|
|
2016-10-15 05:39:34 -05:00
|
|
|
// see if the transaction has a category:
|
|
|
|
$categories = $transaction->categories()->get();
|
|
|
|
if ($categories->count() === 0) {
|
|
|
|
$categories = $transaction->transactionJournal()->first()->categories()->get();
|
|
|
|
}
|
|
|
|
if ($categories->count() > 0) {
|
|
|
|
$str = [];
|
|
|
|
foreach ($categories as $category) {
|
2016-12-28 11:26:44 -06:00
|
|
|
$str[] = sprintf('<a href="%s" title="%s">%s</a>', route('categories.show', [$category->id]), $category->name, $category->name);
|
2016-10-15 05:39:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return join(', ', $str);
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|
2016-10-23 05:42:44 -05:00
|
|
|
}
|