| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * ConvertController.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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-09 07:44:22 +02:00
										 |  |  | declare(strict_types=1); | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Transaction; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use ExpandedForm; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  | use FireflyIII\Models\Account; | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | use FireflyIII\Models\AccountType; | 
					
						
							|  |  |  | use FireflyIII\Models\TransactionJournal; | 
					
						
							|  |  |  | use FireflyIII\Models\TransactionType; | 
					
						
							|  |  |  | use FireflyIII\Repositories\Account\AccountRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  | use FireflyIII\Repositories\Journal\JournalRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | use Illuminate\Http\Request; | 
					
						
							|  |  |  | use Session; | 
					
						
							|  |  |  | use View; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Class ConvertController | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package FireflyIII\Http\Controllers\Transaction | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class ConvertController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /** @var  AccountRepositoryInterface */ | 
					
						
							|  |  |  |     private $accounts; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * ConvertController constructor. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // some useful repositories:
 | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							|  |  |  |             function ($request, $next) { | 
					
						
							|  |  |  |                 $this->accounts = app(AccountRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 View::share('title', trans('firefly.transactions')); | 
					
						
							|  |  |  |                 View::share('mainTitleIcon', 'fa-exchange'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionType    $destinationType | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector|View | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-06 08:59:08 +01:00
										 |  |  |     public function index(TransactionType $destinationType, TransactionJournal $journal) | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         // @codeCoverageIgnoreStart
 | 
					
						
							| 
									
										
										
										
											2016-11-22 19:10:17 +01:00
										 |  |  |         if ($this->isOpeningBalance($journal)) { | 
					
						
							|  |  |  |             return $this->redirectToAccount($journal); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         // @codeCoverageIgnoreEnd
 | 
					
						
							| 
									
										
										
										
											2016-11-22 19:10:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         $positiveAmount = $journal->amountPositive(); | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |         $assetAccounts  = ExpandedForm::makeSelectList($this->accounts->getActiveAccountsByType([AccountType::DEFAULT, AccountType::ASSET])); | 
					
						
							|  |  |  |         $sourceType     = $journal->transactionType; | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |         $subTitle       = trans('firefly.convert_to_' . $destinationType->type, ['description' => $journal->description]); | 
					
						
							|  |  |  |         $subTitleIcon   = 'fa-exchange'; | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |         // cannot convert to its own type.
 | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |         if ($sourceType->type === $destinationType->type) { | 
					
						
							|  |  |  |             Session::flash('info', trans('firefly.convert_is_already_type_' . $destinationType->type)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return redirect(route('transactions.show', [$journal->id])); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // cannot convert split.
 | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |         if ($journal->transactions()->count() > 2) { | 
					
						
							|  |  |  |             Session::flash('error', trans('firefly.cannot_convert_split_journl')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return redirect(route('transactions.show', [$journal->id])); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // get source and destination account:
 | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         $sourceAccount      = $journal->sourceAccountList()->first(); | 
					
						
							|  |  |  |         $destinationAccount = $journal->destinationAccountList()->first(); | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return view( | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |             'transactions.convert', | 
					
						
							|  |  |  |             compact( | 
					
						
							|  |  |  |                 'sourceType', 'destinationType', 'journal', 'assetAccounts', | 
					
						
							|  |  |  |                 'positiveAmount', 'sourceAccount', 'destinationAccount', 'sourceType', | 
					
						
							|  |  |  |                 'subTitle', 'subTitleIcon' | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // convert withdrawal to deposit requires a new source account ()
 | 
					
						
							|  |  |  |         //  or to transfer requires
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |      * @param Request                    $request | 
					
						
							|  |  |  |      * @param JournalRepositoryInterface $repository | 
					
						
							|  |  |  |      * @param TransactionType            $destinationType | 
					
						
							|  |  |  |      * @param TransactionJournal         $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return \Illuminate\Http\RedirectResponse|\Illuminate\Routing\Redirector | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-06 08:59:08 +01:00
										 |  |  |     public function postIndex(Request $request, JournalRepositoryInterface $repository, TransactionType $destinationType, TransactionJournal $journal) | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |         // @codeCoverageIgnoreStart
 | 
					
						
							| 
									
										
										
										
											2016-11-22 19:10:17 +01:00
										 |  |  |         if ($this->isOpeningBalance($journal)) { | 
					
						
							|  |  |  |             return $this->redirectToAccount($journal); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |         // @codeCoverageIgnoreEnd
 | 
					
						
							| 
									
										
										
										
											2016-11-22 19:10:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         $data = $request->all(); | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         if ($journal->transactionType->type === $destinationType->type) { | 
					
						
							|  |  |  |             Session::flash('error', trans('firefly.convert_is_already_type_' . $destinationType->type)); | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return redirect(route('transactions.show', [$journal->id])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($journal->transactions()->count() > 2) { | 
					
						
							|  |  |  |             Session::flash('error', trans('firefly.cannot_convert_split_journl')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return redirect(route('transactions.show', [$journal->id])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         // get the new source and destination account:
 | 
					
						
							|  |  |  |         $source      = $this->getSourceAccount($journal, $destinationType, $data); | 
					
						
							|  |  |  |         $destination = $this->getDestinationAccount($journal, $destinationType, $data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // update the journal:
 | 
					
						
							|  |  |  |         $errors = $repository->convert($journal, $destinationType, $source, $destination); | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         if ($errors->count() > 0) { | 
					
						
							| 
									
										
										
										
											2017-02-15 15:12:46 +01:00
										 |  |  |             return redirect(route('transactions.convert.index', [strtolower($destinationType->type), $journal->id]))->withErrors($errors)->withInput(); | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         Session::flash('success', trans('firefly.converted_to_' . $destinationType->type)); | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         return redirect(route('transactions.show', [$journal->id])); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * @param TransactionType    $destinationType | 
					
						
							|  |  |  |      * @param array              $data | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Account | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function getDestinationAccount(TransactionJournal $journal, TransactionType $destinationType, array $data): Account | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var AccountRepositoryInterface $accountRepository */ | 
					
						
							|  |  |  |         $accountRepository  = app(AccountRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         $sourceAccount      = $journal->sourceAccountList()->first(); | 
					
						
							|  |  |  |         $destinationAccount = $journal->destinationAccountList()->first(); | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         $sourceType         = $journal->transactionType; | 
					
						
							|  |  |  |         $joined             = $sourceType->type . '-' . $destinationType->type; | 
					
						
							|  |  |  |         switch ($joined) { | 
					
						
							|  |  |  |             default: | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |                 throw new FireflyException('Cannot handle ' . $joined); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2017-06-07 07:38:58 +02:00
										 |  |  |             case TransactionType::WITHDRAWAL . '-' . TransactionType::DEPOSIT: | 
					
						
							|  |  |  |                 // one
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $destination = $sourceAccount; | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-07 07:38:58 +02:00
										 |  |  |             case TransactionType::WITHDRAWAL . '-' . TransactionType::TRANSFER: | 
					
						
							|  |  |  |                 // two
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $destination = $accountRepository->find(intval($data['destination_account_asset'])); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-07 07:38:58 +02:00
										 |  |  |             case TransactionType::DEPOSIT . '-' . TransactionType::WITHDRAWAL: | 
					
						
							|  |  |  |             case TransactionType::TRANSFER . '-' . TransactionType::WITHDRAWAL: | 
					
						
							|  |  |  |                 // three and five
 | 
					
						
							| 
									
										
										
										
											2017-05-07 19:45:40 +02:00
										 |  |  |                 if ($data['destination_account_expense'] === '') { | 
					
						
							|  |  |  |                     // destination is a cash account.
 | 
					
						
							|  |  |  |                     $destination = $accountRepository->getCashAccount(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     return $destination; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $data        = [ | 
					
						
							|  |  |  |                     'name'           => $data['destination_account_expense'], | 
					
						
							|  |  |  |                     'accountType'    => 'expense', | 
					
						
							|  |  |  |                     'virtualBalance' => 0, | 
					
						
							|  |  |  |                     'active'         => true, | 
					
						
							|  |  |  |                     'iban'           => null, | 
					
						
							|  |  |  |                 ]; | 
					
						
							|  |  |  |                 $destination = $accountRepository->store($data); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-07 07:38:58 +02:00
										 |  |  |             case TransactionType::DEPOSIT . '-' . TransactionType::TRANSFER: | 
					
						
							|  |  |  |             case TransactionType::TRANSFER . '-' . TransactionType::DEPOSIT: | 
					
						
							|  |  |  |                 // four and six
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $destination = $destinationAccount; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-29 17:30:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         return $destination; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * @param TransactionType    $destinationType | 
					
						
							|  |  |  |      * @param array              $data | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Account | 
					
						
							|  |  |  |      * @throws FireflyException | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private function getSourceAccount(TransactionJournal $journal, TransactionType $destinationType, array $data): Account | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var AccountRepositoryInterface $accountRepository */ | 
					
						
							|  |  |  |         $accountRepository  = app(AccountRepositoryInterface::class); | 
					
						
							| 
									
										
										
										
											2017-03-04 07:18:35 +01:00
										 |  |  |         $sourceAccount      = $journal->sourceAccountList()->first(); | 
					
						
							|  |  |  |         $destinationAccount = $journal->destinationAccountList()->first(); | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         $sourceType         = $journal->transactionType; | 
					
						
							|  |  |  |         $joined             = $sourceType->type . '-' . $destinationType->type; | 
					
						
							|  |  |  |         switch ($joined) { | 
					
						
							|  |  |  |             default: | 
					
						
							| 
									
										
										
										
											2017-03-04 11:19:44 +01:00
										 |  |  |                 throw new FireflyException('Cannot handle ' . $joined); // @codeCoverageIgnore
 | 
					
						
							| 
									
										
										
										
											2017-06-24 13:04:41 +02:00
										 |  |  |             case TransactionType::WITHDRAWAL . '-' . TransactionType::DEPOSIT: | 
					
						
							|  |  |  |             case TransactionType::TRANSFER . '-' . TransactionType::DEPOSIT: | 
					
						
							| 
									
										
										
										
											2017-05-07 19:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if ($data['source_account_revenue'] === '') { | 
					
						
							|  |  |  |                     // destination is a cash account.
 | 
					
						
							|  |  |  |                     $destination = $accountRepository->getCashAccount(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     return $destination; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $data   = [ | 
					
						
							|  |  |  |                     'name'           => $data['source_account_revenue'], | 
					
						
							|  |  |  |                     'accountType'    => 'revenue', | 
					
						
							|  |  |  |                     'virtualBalance' => 0, | 
					
						
							|  |  |  |                     'active'         => true, | 
					
						
							|  |  |  |                     'iban'           => null, | 
					
						
							|  |  |  |                 ]; | 
					
						
							|  |  |  |                 $source = $accountRepository->store($data); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-24 13:04:41 +02:00
										 |  |  |             case TransactionType::WITHDRAWAL . '-' . TransactionType::TRANSFER: | 
					
						
							|  |  |  |             case TransactionType::TRANSFER . '-' . TransactionType::WITHDRAWAL: | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $source = $sourceAccount; | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-24 13:04:41 +02:00
										 |  |  |             case TransactionType::DEPOSIT . '-' . TransactionType::WITHDRAWAL: | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $source = $destinationAccount; | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2017-06-24 13:04:41 +02:00
										 |  |  |             case TransactionType::DEPOSIT . '-' . TransactionType::TRANSFER: | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |                 $source = $accountRepository->find(intval($data['source_account_asset'])); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         return $source; | 
					
						
							| 
									
										
										
										
											2016-10-29 15:14:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-22 19:42:45 +01:00
										 |  |  | } |