mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-26 10:02:50 -04:00 
			
		
		
		
	Merge pull request #7954 from turbo124/v5-develop
Fixes for variables resolving in Recurring Invoices
This commit is contained in:
		
						commit
						b0a8d97e24
					
				
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
								
							| @ -26,15 +26,14 @@ All Pro and Enterprise features from the hosted app are included in the open-cod | ||||
| * [Support Forum](https://forum.invoiceninja.com) | ||||
| * [StackOverflow](https://stackoverflow.com/tags/invoice-ninja/) | ||||
| 
 | ||||
| ## Mobile App | ||||
| * [iPhone](https://apps.apple.com/us/app/invoice-ninja-v5/id1503970375#?platform=iphone) | ||||
| ## Mobile Apps | ||||
| * [iPhone](https://apps.apple.com/app/id1503970375?platform=iphone) | ||||
| * [Android](https://play.google.com/store/apps/details?id=com.invoiceninja.app) | ||||
| * [Linux](https://github.com/invoiceninja/flutter-mobile) | ||||
| 
 | ||||
| ## Desktop App | ||||
| * [MacOS](https://apps.apple.com/app/id1503970375) | ||||
| ## Desktop Apps | ||||
| * [macOS](https://apps.apple.com/app/id1503970375?platform=mac) | ||||
| * [Windows](https://microsoft.com/en-us/p/invoice-ninja/9n3f2bbcfdr6) | ||||
| * [MacOS Desktop](https://snapcraft.io/invoiceninja) | ||||
| * [Linux](https://snapcraft.io/invoiceninja) | ||||
| 
 | ||||
| 
 | ||||
| ## Installation Options | ||||
|  | ||||
| @ -175,7 +175,7 @@ class SendRemindersCron extends Command | ||||
|         /**Refresh Invoice values*/ | ||||
|         $invoice->calc()->getInvoice()->save(); | ||||
|         $invoice->fresh(); | ||||
|         $invoice->service()->deletePdf(); | ||||
|         $invoice->service()->deletePdf()->save(); | ||||
| 
 | ||||
|         /* Refresh the client here to ensure the balance is fresh */ | ||||
|         $client = $invoice->client; | ||||
|  | ||||
| @ -1000,42 +1000,6 @@ class BaseController extends Controller | ||||
|         return redirect('/setup'); | ||||
|     } | ||||
| 
 | ||||
|     public function reactCatch() | ||||
|     {  | ||||
| 
 | ||||
|         if ((bool) $this->checkAppSetup() !== false && $account = Account::first()) { | ||||
|             if (config('ninja.require_https') && ! request()->isSecure()) { | ||||
|                 return redirect()->secure(request()->getRequestUri()); | ||||
|             } | ||||
|   | ||||
|             $data = []; | ||||
| 
 | ||||
|             //pass report errors bool to front end
 | ||||
|             $data['report_errors'] = Ninja::isSelfHost() ? $account->report_errors : true; | ||||
| 
 | ||||
|             //pass referral code to front end
 | ||||
|             $data['rc'] = request()->has('rc') ? request()->input('rc') : ''; | ||||
|             $data['build'] = request()->has('build') ? request()->input('build') : ''; | ||||
|             $data['login'] = request()->has('login') ? request()->input('login') : 'false'; | ||||
|             $data['signup'] = request()->has('signup') ? request()->input('signup') : 'false'; | ||||
| 
 | ||||
|             $data['user_agent'] = request()->server('HTTP_USER_AGENT'); | ||||
| 
 | ||||
|             $data['path'] = $this->setBuild(); | ||||
| 
 | ||||
|             $this->buildCache(); | ||||
| 
 | ||||
|             if (Ninja::isSelfHost() && $account->set_react_as_default_ap) { | ||||
|                 return view('react.index', $data); | ||||
|             } else { | ||||
|                 abort('page not found', 404); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         return redirect('/setup'); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     private function setBuild() | ||||
|     { | ||||
|         $build = ''; | ||||
|  | ||||
| @ -31,12 +31,12 @@ class MatchBankTransactionRequest extends Request | ||||
| 
 | ||||
|         $rules = [ | ||||
|             'transactions' => 'bail|array', | ||||
|             'transactions.*.id' => 'bail|required', | ||||
|             'transactions.*.invoice_ids' => 'nullable|string|sometimes', | ||||
|         ]; | ||||
| 
 | ||||
|         $rules['transactions.*.ninja_category_id'] = 'bail|nullable|sometimes|exists:expense_categories,id,company_id,'.auth()->user()->company()->id.',is_deleted,0'; | ||||
|         $rules['transactions.*.vendor_id'] = 'bail|sometimes|exists:vendors,id,company_id,'.auth()->user()->company()->id.',is_deleted,0'; | ||||
|         $rules['transactions.*.id'] = 'bail|required|exists:bank_transactions,id,company_id,'.auth()->user()->company()->id.',is_deleted,0'; | ||||
| 
 | ||||
|         return $rules; | ||||
| 
 | ||||
|  | ||||
| @ -160,6 +160,9 @@ class MatchBankTransactions implements ShouldQueue | ||||
|     {  | ||||
|         $this->bt = BankTransaction::find($input['id']); | ||||
| 
 | ||||
|             if(!$this->bt || $this->bt->status_id == BankTransaction::STATUS_CONVERTED) | ||||
|                 return $this; | ||||
| 
 | ||||
|         $_invoices = Invoice::withTrashed()->find($this->getInvoices($input['invoice_ids'])); | ||||
|          | ||||
|         $amount = $this->bt->amount; | ||||
| @ -180,6 +183,10 @@ class MatchBankTransactions implements ShouldQueue | ||||
|         //if there is a category id, pull it from Yodlee and insert - or just reuse!!
 | ||||
|         $this->bt = BankTransaction::find($input['id']); | ||||
| 
 | ||||
|             if(!$this->bt || $this->bt->status_id == BankTransaction::STATUS_CONVERTED) | ||||
|                 return $this; | ||||
| 
 | ||||
| 
 | ||||
|         $expense = ExpenseFactory::create($this->bt->company_id, $this->bt->user_id); | ||||
|         $expense->category_id = $this->resolveCategory($input); | ||||
|         $expense->amount = $this->bt->amount; | ||||
|  | ||||
| @ -243,6 +243,14 @@ class PaymentEmailEngine extends BaseEmailEngine | ||||
|         $data['$invoices.due_date'] = ['value' => $this->formatInvoiceField('due_date'), 'label' => ctrans('texts.invoices')]; | ||||
|         $data['$invoices.po_number'] = ['value' => $this->formatInvoiceField('po_number'), 'label' => ctrans('texts.invoices')]; | ||||
| 
 | ||||
| 
 | ||||
|         if($this->payment->status_id == 4) { | ||||
|             $data['$status_logo'] = ['value' => '<div class="stamp is-paid"> ' . ctrans('texts.paid') .'</div>', 'label' => '']; | ||||
|         } | ||||
|         else | ||||
|             $data['$status_logo'] = ['value' => '', 'label' => '']; | ||||
| 
 | ||||
| 
 | ||||
|         $arrKeysLength = array_map('strlen', array_keys($data)); | ||||
|         array_multisort($arrKeysLength, SORT_DESC, $data); | ||||
| 
 | ||||
|  | ||||
| @ -136,3 +136,36 @@ class AuthorizeCreateCustomer | ||||
| //   }
 | ||||
| // }
 | ||||
| } | ||||
| 
 | ||||
| // $request = new net\authorize\api\contract\v1\GetCustomerProfileIdsRequest();
 | ||||
| // $request->setMerchantAuthentication($auth->merchant_authentication);
 | ||||
| // $controller = new net\authorize\api\controller\GetCustomerProfileIdsController($request);
 | ||||
| // $response = $controller->executeWithApiResponse($auth->mode());
 | ||||
| 
 | ||||
| // // $customer_profile_id = end($response->getIds());
 | ||||
| 
 | ||||
| //         foreach($response->getIds() as $customer_profile_id)
 | ||||
| //         {
 | ||||
| //         $request = new net\authorize\api\contract\v1\GetCustomerProfileRequest();
 | ||||
| //         $request->setMerchantAuthentication($auth->merchant_authentication);
 | ||||
| //         $request->setCustomerProfileId($customer_profile_id);
 | ||||
| //         $controller = new net\authorize\api\controller\GetCustomerProfileController($request);
 | ||||
| //         $response = $controller->executeWithApiResponse($auth->mode());
 | ||||
|          | ||||
| //         $profileSelected = $response->getProfile();
 | ||||
| 
 | ||||
| //           if($profileSelected->getEmail() == 'katnandan@gmail.com')
 | ||||
| //           {
 | ||||
| 
 | ||||
| //             $profileSelected;
 | ||||
| //             break;
 | ||||
| 
 | ||||
| //           }
 | ||||
| 
 | ||||
| 
 | ||||
| //         }
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -97,6 +97,8 @@ class InstantBankPay implements MethodInterface | ||||
|         $this->go_cardless->setPaymentHash( | ||||
|             $request->getPaymentHash() | ||||
|         ); | ||||
|          | ||||
|         $this->go_cardless->init(); | ||||
| 
 | ||||
|         try { | ||||
|             $billing_request = $this->go_cardless->gateway->billingRequests()->get( | ||||
|  | ||||
| @ -291,13 +291,13 @@ class GoCardlessPaymentDriver extends BaseDriver | ||||
|                     return response()->json([], 200); | ||||
|                 } | ||||
| 
 | ||||
|                 $this->go_cardless->setPaymentHash($hash); | ||||
|                 $this->setPaymentHash($hash); | ||||
| 
 | ||||
|                 $billing_request = $this->go_cardless->gateway->billingRequests()->get( | ||||
|                 $billing_request = $this->gateway->billingRequests()->get( | ||||
|                     $event['links']['billing_request'] | ||||
|                 ); | ||||
| 
 | ||||
|                 $payment = $this->go_cardless->gateway->payments()->get( | ||||
|                 $payment = $this->gateway->payments()->get( | ||||
|                     $billing_request->payment_request->links->payment | ||||
|                 ); | ||||
| 
 | ||||
| @ -305,12 +305,12 @@ class GoCardlessPaymentDriver extends BaseDriver | ||||
| 
 | ||||
|                     $invoices = Invoice::whereIn('id', $this->transformKeys(array_column($hash->invoices(), 'invoice_id')))->withTrashed()->get(); | ||||
| 
 | ||||
|                     $this->go_cardless->client = $invoices->first()->client; | ||||
|                     $this->client = $invoices->first()->client; | ||||
| 
 | ||||
|                     $invoices->each(function ($invoice){ | ||||
| 
 | ||||
|                         //if payments exist already, they just need to be confirmed.
 | ||||
|                         if($invoice->payments()->exists){ | ||||
|                         if($invoice->payments()->exists()){ | ||||
|                              | ||||
|                             $invoice->payments()->where('status_id', 1)->cursor()->each(function ($payment){ | ||||
|                                 $payment->status_id = 4; | ||||
| @ -347,12 +347,12 @@ class GoCardlessPaymentDriver extends BaseDriver | ||||
|         $data = [ | ||||
|             'payment_method' => $payment->links->mandate, | ||||
|             'payment_type' => PaymentType::INSTANT_BANK_PAY, | ||||
|             'amount' => $this->go_cardless->payment_hash->data->amount_with_fee, | ||||
|             'amount' => $this->payment_hash->data->amount_with_fee, | ||||
|             'transaction_reference' => $payment->id, | ||||
|             'gateway_type_id' => GatewayType::INSTANT_BANK_PAY, | ||||
|         ]; | ||||
| 
 | ||||
|         $payment = $this->go_cardless->createPayment($data, Payment::STATUS_COMPLETED); | ||||
|         $payment = $this->createPayment($data, Payment::STATUS_COMPLETED); | ||||
|         $payment->status_id = Payment::STATUS_COMPLETED; | ||||
|         $payment->save(); | ||||
| 
 | ||||
| @ -361,8 +361,8 @@ class GoCardlessPaymentDriver extends BaseDriver | ||||
|             SystemLog::CATEGORY_GATEWAY_RESPONSE, | ||||
|             SystemLog::EVENT_GATEWAY_SUCCESS, | ||||
|             SystemLog::TYPE_GOCARDLESS, | ||||
|             $this->go_cardless->client, | ||||
|             $this->go_cardless->client->company, | ||||
|             $this->client, | ||||
|             $this->client->company, | ||||
|         ); | ||||
| 
 | ||||
|     } | ||||
|  | ||||
| @ -187,7 +187,7 @@ class BaseRepository | ||||
|         if(!$model->id){ | ||||
|             $this->new_model = true; | ||||
| 
 | ||||
|             if(is_array($model->line_items)) | ||||
|             if(is_array($model->line_items) && !($model instanceof RecurringInvoice)) | ||||
|             {                 | ||||
|                 $model->line_items = (collect($model->line_items))->map(function ($item) use($model,$client) { | ||||
| 
 | ||||
|  | ||||
| @ -73,7 +73,8 @@ class Design extends BaseDesign | ||||
|     const PLAIN = 'plain'; | ||||
|     const PLAYFUL = 'playful'; | ||||
|     const CUSTOM = 'custom'; | ||||
| 
 | ||||
|     const CALM = 'calm'; | ||||
|      | ||||
|     const DELIVERY_NOTE = 'delivery_note'; | ||||
|     const STATEMENT = 'statement'; | ||||
|     const PURCHASE_ORDER = 'purchase_order'; | ||||
|  | ||||
| @ -114,7 +114,7 @@ class Helpers | ||||
|             return ''; | ||||
|         } | ||||
| 
 | ||||
|         // 04-10-2022 Return Early if no reserved keywords are present, this is a very expenseive process
 | ||||
|         // 04-10-2022 Return Early if no reserved keywords are present, this is a very expensive process
 | ||||
|         $string_hit = false; | ||||
| 
 | ||||
|         foreach ( [':MONTH',':YEAR',':QUARTER',':WEEK'] as $string )  | ||||
| @ -144,21 +144,21 @@ class Helpers | ||||
|                 ':QUARTER' => 'Q'.now()->quarter, | ||||
|                 ':WEEK_BEFORE' => \sprintf( | ||||
|                     '%s %s %s', | ||||
|                     Carbon::now()->subDays(7)->translatedFormat($entity->date_format()), | ||||
|                     Carbon::now()->subDays(6)->translatedFormat($entity->date_format()), | ||||
|                     ctrans('texts.to'), | ||||
|                     Carbon::now()->translatedFormat($entity->date_format()) | ||||
|                 ), | ||||
|                 ':WEEK_AHEAD' => \sprintf( | ||||
|                     '%s %s %s', | ||||
|                     Carbon::now()->addDays(7)->translatedFormat($entity->date_format()), | ||||
|                     Carbon::now()->addDays(6)->translatedFormat($entity->date_format()), | ||||
|                     ctrans('texts.to'), | ||||
|                     Carbon::now()->addDays(14)->translatedFormat($entity->date_format()) | ||||
|                     Carbon::now()->addDays(13)->translatedFormat($entity->date_format()) | ||||
|                 ), | ||||
|                 ':WEEK' => \sprintf( | ||||
|                     '%s %s %s', | ||||
|                     Carbon::now()->translatedFormat($entity->date_format()), | ||||
|                     ctrans('texts.to'), | ||||
|                     Carbon::now()->addDays(7)->translatedFormat($entity->date_format()) | ||||
|                     Carbon::now()->addDays(6)->translatedFormat($entity->date_format()) | ||||
|                 ), | ||||
|             ], | ||||
|             'raw' => [ | ||||
|  | ||||
| @ -124,6 +124,7 @@ class HtmlEngine | ||||
|         $data['$line_tax_labels'] = ['value' => $this->lineTaxLabels(), 'label' => ctrans('texts.taxes')]; | ||||
|         $data['$line_tax_values'] = ['value' => $this->lineTaxValues(), 'label' => ctrans('texts.taxes')]; | ||||
|         $data['$date'] = ['value' => $this->translateDate($this->entity->date, $this->client->date_format(), $this->client->locale()) ?: ' ', 'label' => ctrans('texts.date')]; | ||||
|         $data['$status_logo'] = ['value' => '', 'label' => '']; | ||||
| 
 | ||||
|         $data['$invoice.date'] = &$data['$date']; | ||||
|         $data['$invoiceDate'] = &$data['$date']; | ||||
| @ -167,6 +168,10 @@ class HtmlEngine | ||||
|                 $data['$invoice.project'] = &$data['$project.name']; | ||||
|             } | ||||
| 
 | ||||
|             if($this->entity->status_id == 4) { | ||||
|                 $data['$status_logo'] = ['value' => '<div class="stamp is-paid"> ' . ctrans('texts.paid') .'</div>', 'label' => '']; | ||||
|             } | ||||
| 
 | ||||
|             if($this->entity->vendor) { | ||||
|                 $data['$invoice.vendor'] = ['value' => $this->entity->vendor->present()->name(), 'label' => ctrans('texts.vendor_name')]; | ||||
|             } | ||||
|  | ||||
							
								
								
									
										53
									
								
								database/migrations/2022_11_16_093535_calmness_design.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								database/migrations/2022_11_16_093535_calmness_design.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | ||||
| <?php | ||||
| 
 | ||||
| use App\Models\Design; | ||||
| use App\Services\PdfMaker\Design as PdfMakerDesign; | ||||
| use App\Utils\Ninja; | ||||
| use Illuminate\Database\Migrations\Migration; | ||||
| use Illuminate\Database\Schema\Blueprint; | ||||
| use Illuminate\Support\Facades\Schema; | ||||
| 
 | ||||
| return new class extends Migration | ||||
| { | ||||
|     /** | ||||
|      * Run the migrations. | ||||
|      * | ||||
|      * @return void | ||||
|      */ | ||||
|     public function up() | ||||
|     { | ||||
|          | ||||
|         if (Ninja::isHosted()) { | ||||
|             $design = new Design(); | ||||
| 
 | ||||
|             $design->name = 'Calm'; | ||||
|             $design->is_custom = false; | ||||
|             $design->design = ''; | ||||
|             $design->is_active = true; | ||||
| 
 | ||||
|             $design->save(); | ||||
|         } elseif (Design::count() !== 0) { | ||||
|             $design = new Design(); | ||||
| 
 | ||||
|             $design->name = 'Calm'; | ||||
|             $design->is_custom = false; | ||||
|             $design->design = ''; | ||||
|             $design->is_active = true; | ||||
| 
 | ||||
|             $design->save(); | ||||
|         } | ||||
| 
 | ||||
|         \Illuminate\Support\Facades\Artisan::call('ninja:design-update'); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Reverse the migrations. | ||||
|      * | ||||
|      * @return void | ||||
|      */ | ||||
|     public function down() | ||||
|     { | ||||
|         //
 | ||||
|     } | ||||
| }; | ||||
| @ -38,6 +38,7 @@ class DesignSeeder extends Seeder | ||||
|             ['id' => 8, 'name' => 'Hipster', 'user_id' => null, 'company_id' => null, 'is_custom' => false, 'design' => '', 'is_active' => true], | ||||
|             ['id' => 9, 'name' => 'Playful', 'user_id' => null, 'company_id' => null, 'is_custom' => false, 'design' => '', 'is_active' => true], | ||||
|             ['id' => 10, 'name' => 'Tech', 'user_id' => null, 'company_id' => null, 'is_custom' => false, 'design' => '', 'is_active' => true], | ||||
|             ['id' => 11, 'name' => 'Calm', 'user_id' => null, 'company_id' => null, 'is_custom' => false, 'design' => '', 'is_active' => true], | ||||
|         ]; | ||||
| 
 | ||||
|         foreach ($designs as $design) { | ||||
|  | ||||
| @ -101,6 +101,36 @@ | ||||
|         #content .center {
 | ||||
|             text-align: center; | ||||
|         } | ||||
| 
 | ||||
|         .stamp { | ||||
|             transform: rotate(12deg); | ||||
|             color: #555;
 | ||||
|             font-size: 3rem; | ||||
|             font-weight: 700; | ||||
|             border: 0.25rem solid #555;
 | ||||
|             display: inline-block; | ||||
|             padding: 0.25rem 1rem; | ||||
|             text-transform: uppercase; | ||||
|             border-radius: 1rem; | ||||
|             font-family: 'Courier'; | ||||
|             mix-blend-mode: multiply; | ||||
|             z-index:200 !important; | ||||
|             position:  fixed; | ||||
|             text-align: center; | ||||
|         } | ||||
| 
 | ||||
|         .is-paid { | ||||
|             color:  #D23;
 | ||||
|             border: 1rem double  #D23;
 | ||||
|             transform: rotate(-5deg); | ||||
|             font-size: 6rem; | ||||
|             font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|             border-radius: 0; | ||||
|             padding: 0.5rem; | ||||
|             opacity: 0.2; | ||||
|             z-index:200 !important; | ||||
|             position:  fixed; | ||||
|         }  | ||||
|     </style> | ||||
| </head> | ||||
| 
 | ||||
|  | ||||
| @ -80,6 +80,33 @@ | ||||
|         #content .left {
 | ||||
|             text-align: left !important; | ||||
|         } | ||||
| 
 | ||||
|         .stamp { | ||||
|             transform: rotate(12deg); | ||||
|             color: #555;
 | ||||
|             font-size: 3rem; | ||||
|             font-weight: 700; | ||||
|             border: 0.25rem solid #555;
 | ||||
|             text-transform: uppercase; | ||||
|             border-radius: 1rem; | ||||
|             font-family: 'Courier'; | ||||
|             mix-blend-mode: multiply; | ||||
|             z-index:200 !important; | ||||
|             position: relative; | ||||
|         } | ||||
| 
 | ||||
|         .is-paid { | ||||
|             color:  #D23;
 | ||||
|             border: 1rem double  #D23;
 | ||||
|             transform: rotate(-5deg); | ||||
|             font-size: 6rem; | ||||
|             font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|             border-radius: 0; | ||||
|             padding: 0.5rem; | ||||
|             opacity: 0.2; | ||||
|             z-index:200 !important; | ||||
|             position: relative; | ||||
|         }  | ||||
|     </style> | ||||
| 
 | ||||
|     <!--[if gte mso 9]> | ||||
|  | ||||
| @ -143,6 +143,36 @@ | ||||
|             color: {{ $design == 'dark' ? '#ffffff' : '#000000' }} !important; | ||||
|             opacity: {{ $design == 'dark' ? '87%': '100%' }} !important; | ||||
|         } | ||||
| 
 | ||||
|         .stamp { | ||||
|           transform: rotate(12deg); | ||||
|             color: #555;
 | ||||
|             font-size: 3rem; | ||||
|             font-weight: 700; | ||||
|             border: 0.25rem solid #555;
 | ||||
|             display: inline-block; | ||||
|             padding: 0.25rem 1rem; | ||||
|             text-transform: uppercase; | ||||
|             border-radius: 1rem; | ||||
|             font-family: 'Courier'; | ||||
|             mix-blend-mode: multiply; | ||||
|             z-index:200 !important; | ||||
|             position:  fixed; | ||||
|             text-align: center; | ||||
|         } | ||||
| 
 | ||||
|         .is-paid { | ||||
|             color:  #D23;
 | ||||
|             border: 1rem double  #D23;
 | ||||
|             transform: rotate(-5deg); | ||||
|             font-size: 6rem; | ||||
|             font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|             border-radius: 0; | ||||
|             padding: 0.5rem; | ||||
|             opacity: 0.2; | ||||
|             z-index:200 !important; | ||||
|             position:  fixed; | ||||
|         }  | ||||
|     </style> | ||||
| </head> | ||||
| 
 | ||||
|  | ||||
| @ -47,6 +47,7 @@ | ||||
|         line-height: var(--line-height); | ||||
|         position: fixed; | ||||
|         top: 0; | ||||
|         width: 100%; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
| @ -224,7 +225,7 @@ | ||||
|         position: fixed; | ||||
|         bottom: 0; | ||||
|         display: grid; | ||||
|         grid-template-columns: 1fr 1fr 1fr; | ||||
|         grid-template-columns: 1fr; | ||||
|         gap: 15px; | ||||
|         color: white; | ||||
|     } | ||||
| @ -239,6 +240,11 @@ | ||||
|         padding-top: 0.5rem | ||||
|     } | ||||
| 
 | ||||
|     [data-ref="footer_content"]{ | ||||
|         padding-right: 2rem; | ||||
|         margin-right: 2rem; | ||||
|     } | ||||
| 
 | ||||
|     table { | ||||
|         width: 100%; | ||||
|     } | ||||
| @ -285,6 +291,36 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
| 
 | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
| @ -380,7 +416,7 @@ $entity_images | ||||
| 
 | ||||
| <div id="footer"> | ||||
|     <div style="width: 100%;"> | ||||
|         <p data-ref="total_table-footer">$entity_footer</p> | ||||
|         <p data-ref="footer_content">$entity_footer</p> | ||||
| 
 | ||||
|         <script> | ||||
|             // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. | ||||
| @ -402,6 +438,4 @@ $entity_images | ||||
|             }); | ||||
|         </script> | ||||
|     </div> | ||||
|     <div> <!-- #2 column --> </div> | ||||
|     <div> <!-- #3 column --> </div> | ||||
| </div> | ||||
|  | ||||
| @ -276,6 +276,36 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|      | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -262,7 +262,37 @@ | ||||
|     } | ||||
| 
 | ||||
|     [data-ref="total_table-public_notes"] { font-weight: normal; }  | ||||
|     [data-ref="total_table-terms"] { font-weight: normal; }  | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
| @ -290,6 +320,7 @@ | ||||
|     /** To find out selectors on your own: https://invoiceninja.github.io/docs/custom-fields/#snippets **/ | ||||
| </style> | ||||
| 
 | ||||
| <div id="body"> | ||||
| <table style="min-width: 100%"> | ||||
|    <thead> | ||||
|       <tr> | ||||
| @ -301,7 +332,7 @@ | ||||
|    <tbody> | ||||
|       <tr> | ||||
|          <td> | ||||
|             <div id="body"> | ||||
|             <div id=""> | ||||
|                <div class="header-wrapper"> | ||||
|                   <div> | ||||
|                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> | ||||
| @ -316,7 +347,7 @@ | ||||
|                <div id="vendor-details"></div> | ||||
| 
 | ||||
|                <div> | ||||
|                <p class="entity-label" style="font-size:32px; color:$primary_color;">$entity_label</p> | ||||
|                <p class="entity-label" style="font-size:32px; font-weight: bold; color:$primary_color;">$entity_label</p> | ||||
|                <table id="entity-details" cellspacing="0" dir="ltr"></table> | ||||
|                </div> | ||||
|                </div> | ||||
| @ -343,6 +374,7 @@ | ||||
|       </tr> | ||||
|    </tfoot> | ||||
| </table> | ||||
| </div> | ||||
| 
 | ||||
| <div class="repeating-header" id="header"></div> | ||||
| 
 | ||||
| @ -371,24 +403,23 @@ | ||||
|     </div> | ||||
| </div> | ||||
| 
 | ||||
|         <script> | ||||
|             // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. | ||||
|             document.addEventListener('DOMContentLoaded', () => { | ||||
|                 let tables = [ | ||||
|                     'product-table', 'task-table', 'delivery-note-table', | ||||
|                     'statement-invoice-table', 'statement-payment-table', 'statement-aging-table-totals', | ||||
|                     'statement-invoice-table-totals', 'statement-payment-table-totals', 'statement-aging-table', | ||||
|                     'client-details','vendor-details', 'swiss-qr' | ||||
|                 ]; | ||||
|     <script> | ||||
|         // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. | ||||
|         document.addEventListener('DOMContentLoaded', () => { | ||||
|             let tables = [ | ||||
|                 'product-table', 'task-table', 'delivery-note-table', | ||||
|                 'statement-invoice-table', 'statement-payment-table', 'statement-aging-table-totals', | ||||
|                 'statement-invoice-table-totals', 'statement-payment-table-totals', 'statement-aging-table', | ||||
|                 'client-details','vendor-details', 'swiss-qr' | ||||
|             ]; | ||||
| 
 | ||||
|                 tables.forEach((tableIdentifier) => { | ||||
|                     console.log(document.getElementById(tableIdentifier)); | ||||
|             tables.forEach((tableIdentifier) => { | ||||
|                 console.log(document.getElementById(tableIdentifier)); | ||||
| 
 | ||||
|                     document.getElementById(tableIdentifier)?.childElementCount === 0 | ||||
|                         ? document.getElementById(tableIdentifier).style.setProperty('display', 'none', 'important') | ||||
|                         : ''; | ||||
|                 }); | ||||
|                 document.getElementById(tableIdentifier)?.childElementCount === 0 | ||||
|                     ? document.getElementById(tableIdentifier).style.setProperty('display', 'none', 'important') | ||||
|                     : ''; | ||||
|             }); | ||||
|         </script> | ||||
|          | ||||
| </div> | ||||
|         }); | ||||
|     </script> | ||||
|        | ||||
| @ -257,6 +257,36 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
|      | ||||
|         .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|      | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -229,6 +229,35 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -233,7 +233,36 @@ | ||||
|         max-width: 300px; | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }      | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -251,6 +251,35 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
|      | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -278,6 +278,35 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|       } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|       /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|       /** Hide company logo **/ | ||||
|  | ||||
| @ -221,6 +221,35 @@ | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -293,6 +293,36 @@ | ||||
|         max-width: 300px; | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|      | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
| @ -323,7 +353,7 @@ | ||||
|     /** For more info, please check our docs: https://invoiceninja.github.io **/ | ||||
|     /** To find out selectors on your own: https://invoiceninja.github.io/docs/custom-fields/#snippets **/ | ||||
| </style> | ||||
| 
 | ||||
| <div id="body"> | ||||
| <table style="min-width: 100%"> | ||||
|    <thead> | ||||
|       <tr> | ||||
| @ -335,7 +365,6 @@ | ||||
|    <tbody> | ||||
|       <tr> | ||||
|          <td> | ||||
|             <div id="body"> | ||||
|                <div class="header-wrapper"> | ||||
|                   <div> | ||||
|                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> | ||||
| @ -367,8 +396,7 @@ | ||||
|                <div id="statement-payment-table-totals" data-ref="statement-totals"></div> | ||||
|                <table id="statement-aging-table" cellspacing="0" data-ref="table"></table> | ||||
|                <div id="statement-aging-table-totals" data-ref="statement-totals"></div> | ||||
|                <div id="table-totals" cellspacing="0"></div> | ||||
|             </div> | ||||
|                <div id="table-totals" cellspacing="0"></div>       | ||||
|          </td> | ||||
|       </tr> | ||||
|    </tbody> | ||||
| @ -380,7 +408,7 @@ | ||||
|       </tr> | ||||
|    </tfoot> | ||||
| </table> | ||||
| 
 | ||||
| </div> | ||||
| <div class="repeating-header"> | ||||
|     <div id="header"> | ||||
|         <div style="background-color: #00968B"><!-- 1 --></div> | ||||
|  | ||||
| @ -257,7 +257,36 @@ | ||||
|         max-width: 300px; | ||||
|         overflow-wrap: break-word;  | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     .stamp { | ||||
|       transform: rotate(12deg); | ||||
|         color: #555; | ||||
|         font-size: 3rem; | ||||
|         font-weight: 700; | ||||
|         border: 0.25rem solid #555; | ||||
|         display: inline-block; | ||||
|         padding: 0.25rem 1rem; | ||||
|         text-transform: uppercase; | ||||
|         border-radius: 1rem; | ||||
|         font-family: 'Courier'; | ||||
|         mix-blend-mode: multiply; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|         text-align: center; | ||||
|     } | ||||
| 
 | ||||
|     .is-paid { | ||||
|         color:  #D23; | ||||
|         border: 1rem double  #D23; | ||||
|         transform: rotate(-5deg); | ||||
|         font-size: 6rem; | ||||
|         font-family: "Open sans", Helvetica, Arial, sans-serif; | ||||
|         border-radius: 0; | ||||
|         padding: 0.5rem; | ||||
|         opacity: 0.2; | ||||
|         z-index:200 !important; | ||||
|         position:  fixed; | ||||
|     }  | ||||
|     /** Useful snippets, uncomment to enable. **/ | ||||
| 
 | ||||
|     /** Hide company logo **/ | ||||
|  | ||||
| @ -58,6 +58,4 @@ Route::get('yodlee/onboard/{token}', [YodleeController::class, 'auth'])->name('y | ||||
| Route::get('checkout/3ds_redirect/{company_key}/{company_gateway_id}/{hash}', [Checkout3dsController::class, 'index'])->middleware('domain_db')->name('checkout.3ds_redirect'); | ||||
| Route::get('mollie/3ds_redirect/{company_key}/{company_gateway_id}/{hash}', [Mollie3dsController::class, 'index'])->middleware('domain_db')->name('mollie.3ds_redirect'); | ||||
| Route::get('gocardless/ibp_redirect/{company_key}/{company_gateway_id}/{hash}', [GoCardlessController::class, 'ibpRedirect'])->middleware('domain_db')->name('gocardless.ibp_redirect'); | ||||
| Route::get('.well-known/apple-developer-merchantid-domain-association', [ApplePayDomainController::class, 'showAppleMerchantId']); | ||||
| 
 | ||||
| Route::fallback([BaseController::class, 'reactCatch']); | ||||
| Route::get('.well-known/apple-developer-merchantid-domain-association', [ApplePayDomainController::class, 'showAppleMerchantId']); | ||||
| @ -11,6 +11,7 @@ | ||||
| 
 | ||||
| namespace Tests\Feature; | ||||
| 
 | ||||
| use App\Factory\InvoiceItemFactory; | ||||
| use App\Factory\InvoiceToRecurringInvoiceFactory; | ||||
| use App\Factory\RecurringInvoiceToInvoiceFactory; | ||||
| use App\Models\Client; | ||||
| @ -51,6 +52,53 @@ class RecurringInvoiceTest extends TestCase | ||||
|         $this->makeTestData(); | ||||
|     } | ||||
| 
 | ||||
|     public function testPostRecurringInvoiceWithPlaceholderVariables() | ||||
|     { | ||||
|         $line_items = []; | ||||
| 
 | ||||
|         $item = InvoiceItemFactory::create(); | ||||
|         $item->quantity = 1; | ||||
|         $item->cost = 10; | ||||
|         $item->task_id = $this->encodePrimaryKey($this->task->id); | ||||
|         $item->expense_id = $this->encodePrimaryKey($this->expense->id); | ||||
|         $item->notes = "Hello this is the month of :MONTH"; | ||||
| 
 | ||||
|         $line_items[] = $item; | ||||
| 
 | ||||
| 
 | ||||
|         $data = [ | ||||
|             'frequency_id' => 1, | ||||
|             'status_id' => 1, | ||||
|             'discount' => 0, | ||||
|             'is_amount_discount' => 1, | ||||
|             'po_number' => '3434343', | ||||
|             'public_notes' => 'notes', | ||||
|             'is_deleted' => 0, | ||||
|             'custom_value1' => 0, | ||||
|             'custom_value2' => 0, | ||||
|             'custom_value3' => 0, | ||||
|             'custom_value4' => 0, | ||||
|             'status' => 1, | ||||
|             'client_id' => $this->encodePrimaryKey($this->client->id), | ||||
|             'line_items' => $line_items, | ||||
|             'remaining_cycles' => -1, | ||||
|         ]; | ||||
| 
 | ||||
|         $response = $this->withHeaders([ | ||||
|             'X-API-SECRET' => config('ninja.api_secret'), | ||||
|             'X-API-TOKEN' => $this->token, | ||||
|         ])->post('/api/v1/recurring_invoices/', $data) | ||||
|             ->assertStatus(200); | ||||
| 
 | ||||
|         $arr = $response->json(); | ||||
|         $this->assertEquals(RecurringInvoice::STATUS_DRAFT, $arr['data']['status_id']); | ||||
| 
 | ||||
|         $notes = end($arr['data']['line_items'])['notes']; | ||||
| 
 | ||||
|         $this->assertTrue(str_contains($notes, ':MONTH')); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     public function testPostRecurringInvoice() | ||||
|     { | ||||
|         $data = [ | ||||
|  | ||||
| @ -48,27 +48,4 @@ class AutoBillInvoiceTest extends TestCase | ||||
|         $this->assertEquals($this->client->fresh()->credit_balance, 0); | ||||
|     } | ||||
| 
 | ||||
|     // public function testAutoBillSetOffFunctionality()
 | ||||
|     // {
 | ||||
| 
 | ||||
|     //     $settings = $this->company->settings;
 | ||||
|     //     $settings->use_credits_payment = 'off';
 | ||||
| 
 | ||||
|     //     $this->company->settings = $settings;
 | ||||
|     //     $this->company->save();
 | ||||
| 
 | ||||
|     //     $this->assertEquals($this->client->balance, 10);
 | ||||
|     //     $this->assertEquals($this->client->paid_to_date, 0);
 | ||||
|     //     $this->assertEquals($this->client->credit_balance, 10);
 | ||||
| 
 | ||||
|     //     $this->invoice->service()->markSent()->autoBill()->save();
 | ||||
| 
 | ||||
|     //     $this->assertNotNull($this->invoice->payments());
 | ||||
|     //     $this->assertEquals(0, $this->invoice->payments()->sum('payments.amount'));
 | ||||
| 
 | ||||
|     //     $this->assertEquals($this->client->balance, 10);
 | ||||
|     //     $this->assertEquals($this->client->paid_to_date, 0);
 | ||||
|     //     $this->assertEquals($this->client->credit_balance, 10);
 | ||||
| 
 | ||||
|     // }
 | ||||
| } | ||||
|  | ||||
							
								
								
									
										86
									
								
								tests/Unit/LateFeeTest.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								tests/Unit/LateFeeTest.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,86 @@ | ||||
| <?php | ||||
| /** | ||||
|  * Invoice Ninja (https://invoiceninja.com). | ||||
|  * | ||||
|  * @link https://github.com/invoiceninja/invoiceninja source repository | ||||
|  * | ||||
|  * @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com) | ||||
|  * | ||||
|  * @license https://www.elastic.co/licensing/elastic-license | ||||
|  */ | ||||
| 
 | ||||
| namespace Tests\Unit; | ||||
| 
 | ||||
| use App\DataMapper\InvoiceItem; | ||||
| use App\Models\Invoice; | ||||
| use Illuminate\Foundation\Testing\DatabaseTransactions; | ||||
| use Tests\MockAccountData; | ||||
| use Tests\TestCase; | ||||
| 
 | ||||
| /** | ||||
|  * @test | ||||
|  */ | ||||
| class LateFeeTest extends TestCase | ||||
| { | ||||
|     use DatabaseTransactions; | ||||
|     use MockAccountData; | ||||
| 
 | ||||
|     protected function setUp() :void | ||||
|     { | ||||
|         parent::setUp(); | ||||
| 
 | ||||
|         $this->makeTestData(); | ||||
|     } | ||||
| 
 | ||||
|     public function testLateFeeBalances() | ||||
|     { | ||||
| 
 | ||||
|         $this->assertEquals(10, $this->client->balance); | ||||
|         $this->assertEquals(10, $this->invoice->balance); | ||||
| 
 | ||||
|         $this->invoice = $this->setLateFee($this->invoice, 5, 0); | ||||
| 
 | ||||
|         $this->assertEquals(15, $this->client->fresh()->balance); | ||||
|         $this->assertEquals(15, $this->invoice->fresh()->balance); | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     private function setLateFee($invoice, $amount, $percent) :Invoice | ||||
|     { | ||||
| 
 | ||||
|         $temp_invoice_balance = $invoice->balance; | ||||
| 
 | ||||
|         if ($amount <= 0 && $percent <= 0) { | ||||
|             return $invoice; | ||||
|         } | ||||
| 
 | ||||
|         $fee = $amount; | ||||
| 
 | ||||
|         if ($invoice->partial > 0) { | ||||
|             $fee += round($invoice->partial * $percent / 100, 2); | ||||
|         } else { | ||||
|             $fee += round($invoice->balance * $percent / 100, 2); | ||||
|         } | ||||
| 
 | ||||
|         $invoice_item = new InvoiceItem; | ||||
|         $invoice_item->type_id = '5'; | ||||
|         $invoice_item->product_key = trans('texts.fee'); | ||||
|         $invoice_item->notes = ctrans('texts.late_fee_added', ['date' => now()]); | ||||
|         $invoice_item->quantity = 1; | ||||
|         $invoice_item->cost = $fee; | ||||
| 
 | ||||
|         $invoice_items = $invoice->line_items; | ||||
|         $invoice_items[] = $invoice_item; | ||||
| 
 | ||||
|         $invoice->line_items = $invoice_items; | ||||
| 
 | ||||
|         /**Refresh Invoice values*/ | ||||
|         $invoice = $invoice->calc()->getInvoice(); | ||||
| 
 | ||||
|         $invoice->client->service()->updateBalance($invoice->balance - $temp_invoice_balance)->save(); | ||||
|         $invoice->ledger()->updateInvoiceBalance($invoice->balance - $temp_invoice_balance, "Late Fee Adjustment for invoice {$invoice->number}"); | ||||
| 
 | ||||
|         return $invoice; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user