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) | * [Support Forum](https://forum.invoiceninja.com) | ||||||
| * [StackOverflow](https://stackoverflow.com/tags/invoice-ninja/) | * [StackOverflow](https://stackoverflow.com/tags/invoice-ninja/) | ||||||
| 
 | 
 | ||||||
| ## Mobile App | ## Mobile Apps | ||||||
| * [iPhone](https://apps.apple.com/us/app/invoice-ninja-v5/id1503970375#?platform=iphone) | * [iPhone](https://apps.apple.com/app/id1503970375?platform=iphone) | ||||||
| * [Android](https://play.google.com/store/apps/details?id=com.invoiceninja.app) | * [Android](https://play.google.com/store/apps/details?id=com.invoiceninja.app) | ||||||
| * [Linux](https://github.com/invoiceninja/flutter-mobile) |  | ||||||
| 
 | 
 | ||||||
| ## Desktop App | ## Desktop Apps | ||||||
| * [MacOS](https://apps.apple.com/app/id1503970375) | * [macOS](https://apps.apple.com/app/id1503970375?platform=mac) | ||||||
| * [Windows](https://microsoft.com/en-us/p/invoice-ninja/9n3f2bbcfdr6) | * [Windows](https://microsoft.com/en-us/p/invoice-ninja/9n3f2bbcfdr6) | ||||||
| * [MacOS Desktop](https://snapcraft.io/invoiceninja) | * [Linux](https://snapcraft.io/invoiceninja) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Installation Options | ## Installation Options | ||||||
|  | |||||||
| @ -175,7 +175,7 @@ class SendRemindersCron extends Command | |||||||
|         /**Refresh Invoice values*/ |         /**Refresh Invoice values*/ | ||||||
|         $invoice->calc()->getInvoice()->save(); |         $invoice->calc()->getInvoice()->save(); | ||||||
|         $invoice->fresh(); |         $invoice->fresh(); | ||||||
|         $invoice->service()->deletePdf(); |         $invoice->service()->deletePdf()->save(); | ||||||
| 
 | 
 | ||||||
|         /* Refresh the client here to ensure the balance is fresh */ |         /* Refresh the client here to ensure the balance is fresh */ | ||||||
|         $client = $invoice->client; |         $client = $invoice->client; | ||||||
|  | |||||||
| @ -1000,42 +1000,6 @@ class BaseController extends Controller | |||||||
|         return redirect('/setup'); |         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() |     private function setBuild() | ||||||
|     { |     { | ||||||
|         $build = ''; |         $build = ''; | ||||||
|  | |||||||
| @ -31,12 +31,12 @@ class MatchBankTransactionRequest extends Request | |||||||
| 
 | 
 | ||||||
|         $rules = [ |         $rules = [ | ||||||
|             'transactions' => 'bail|array', |             'transactions' => 'bail|array', | ||||||
|             'transactions.*.id' => 'bail|required', |  | ||||||
|             'transactions.*.invoice_ids' => 'nullable|string|sometimes', |             '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.*.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.*.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; |         return $rules; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -160,6 +160,9 @@ class MatchBankTransactions implements ShouldQueue | |||||||
|     {  |     {  | ||||||
|         $this->bt = BankTransaction::find($input['id']); |         $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'])); |         $_invoices = Invoice::withTrashed()->find($this->getInvoices($input['invoice_ids'])); | ||||||
|          |          | ||||||
|         $amount = $this->bt->amount; |         $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!!
 |         //if there is a category id, pull it from Yodlee and insert - or just reuse!!
 | ||||||
|         $this->bt = BankTransaction::find($input['id']); |         $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 = ExpenseFactory::create($this->bt->company_id, $this->bt->user_id); | ||||||
|         $expense->category_id = $this->resolveCategory($input); |         $expense->category_id = $this->resolveCategory($input); | ||||||
|         $expense->amount = $this->bt->amount; |         $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.due_date'] = ['value' => $this->formatInvoiceField('due_date'), 'label' => ctrans('texts.invoices')]; | ||||||
|         $data['$invoices.po_number'] = ['value' => $this->formatInvoiceField('po_number'), '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)); |         $arrKeysLength = array_map('strlen', array_keys($data)); | ||||||
|         array_multisort($arrKeysLength, SORT_DESC, $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;
 | ||||||
|  | 
 | ||||||
|  | //           }
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | //         }
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | |||||||
| @ -98,6 +98,8 @@ class InstantBankPay implements MethodInterface | |||||||
|             $request->getPaymentHash() |             $request->getPaymentHash() | ||||||
|         ); |         ); | ||||||
|          |          | ||||||
|  |         $this->go_cardless->init(); | ||||||
|  | 
 | ||||||
|         try { |         try { | ||||||
|             $billing_request = $this->go_cardless->gateway->billingRequests()->get( |             $billing_request = $this->go_cardless->gateway->billingRequests()->get( | ||||||
|                 $this->go_cardless->payment_hash->data->billing_request |                 $this->go_cardless->payment_hash->data->billing_request | ||||||
|  | |||||||
| @ -291,13 +291,13 @@ class GoCardlessPaymentDriver extends BaseDriver | |||||||
|                     return response()->json([], 200); |                     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'] |                     $event['links']['billing_request'] | ||||||
|                 ); |                 ); | ||||||
| 
 | 
 | ||||||
|                 $payment = $this->go_cardless->gateway->payments()->get( |                 $payment = $this->gateway->payments()->get( | ||||||
|                     $billing_request->payment_request->links->payment |                     $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(); |                     $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){ |                     $invoices->each(function ($invoice){ | ||||||
| 
 | 
 | ||||||
|                         //if payments exist already, they just need to be confirmed.
 |                         //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){ |                             $invoice->payments()->where('status_id', 1)->cursor()->each(function ($payment){ | ||||||
|                                 $payment->status_id = 4; |                                 $payment->status_id = 4; | ||||||
| @ -347,12 +347,12 @@ class GoCardlessPaymentDriver extends BaseDriver | |||||||
|         $data = [ |         $data = [ | ||||||
|             'payment_method' => $payment->links->mandate, |             'payment_method' => $payment->links->mandate, | ||||||
|             'payment_type' => PaymentType::INSTANT_BANK_PAY, |             '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, |             'transaction_reference' => $payment->id, | ||||||
|             'gateway_type_id' => GatewayType::INSTANT_BANK_PAY, |             '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->status_id = Payment::STATUS_COMPLETED; | ||||||
|         $payment->save(); |         $payment->save(); | ||||||
| 
 | 
 | ||||||
| @ -361,8 +361,8 @@ class GoCardlessPaymentDriver extends BaseDriver | |||||||
|             SystemLog::CATEGORY_GATEWAY_RESPONSE, |             SystemLog::CATEGORY_GATEWAY_RESPONSE, | ||||||
|             SystemLog::EVENT_GATEWAY_SUCCESS, |             SystemLog::EVENT_GATEWAY_SUCCESS, | ||||||
|             SystemLog::TYPE_GOCARDLESS, |             SystemLog::TYPE_GOCARDLESS, | ||||||
|             $this->go_cardless->client, |             $this->client, | ||||||
|             $this->go_cardless->client->company, |             $this->client->company, | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -187,7 +187,7 @@ class BaseRepository | |||||||
|         if(!$model->id){ |         if(!$model->id){ | ||||||
|             $this->new_model = true; |             $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) { |                 $model->line_items = (collect($model->line_items))->map(function ($item) use($model,$client) { | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -73,6 +73,7 @@ class Design extends BaseDesign | |||||||
|     const PLAIN = 'plain'; |     const PLAIN = 'plain'; | ||||||
|     const PLAYFUL = 'playful'; |     const PLAYFUL = 'playful'; | ||||||
|     const CUSTOM = 'custom'; |     const CUSTOM = 'custom'; | ||||||
|  |     const CALM = 'calm'; | ||||||
|      |      | ||||||
|     const DELIVERY_NOTE = 'delivery_note'; |     const DELIVERY_NOTE = 'delivery_note'; | ||||||
|     const STATEMENT = 'statement'; |     const STATEMENT = 'statement'; | ||||||
|  | |||||||
| @ -114,7 +114,7 @@ class Helpers | |||||||
|             return ''; |             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; |         $string_hit = false; | ||||||
| 
 | 
 | ||||||
|         foreach ( [':MONTH',':YEAR',':QUARTER',':WEEK'] as $string )  |         foreach ( [':MONTH',':YEAR',':QUARTER',':WEEK'] as $string )  | ||||||
| @ -144,21 +144,21 @@ class Helpers | |||||||
|                 ':QUARTER' => 'Q'.now()->quarter, |                 ':QUARTER' => 'Q'.now()->quarter, | ||||||
|                 ':WEEK_BEFORE' => \sprintf( |                 ':WEEK_BEFORE' => \sprintf( | ||||||
|                     '%s %s %s', |                     '%s %s %s', | ||||||
|                     Carbon::now()->subDays(7)->translatedFormat($entity->date_format()), |                     Carbon::now()->subDays(6)->translatedFormat($entity->date_format()), | ||||||
|                     ctrans('texts.to'), |                     ctrans('texts.to'), | ||||||
|                     Carbon::now()->translatedFormat($entity->date_format()) |                     Carbon::now()->translatedFormat($entity->date_format()) | ||||||
|                 ), |                 ), | ||||||
|                 ':WEEK_AHEAD' => \sprintf( |                 ':WEEK_AHEAD' => \sprintf( | ||||||
|                     '%s %s %s', |                     '%s %s %s', | ||||||
|                     Carbon::now()->addDays(7)->translatedFormat($entity->date_format()), |                     Carbon::now()->addDays(6)->translatedFormat($entity->date_format()), | ||||||
|                     ctrans('texts.to'), |                     ctrans('texts.to'), | ||||||
|                     Carbon::now()->addDays(14)->translatedFormat($entity->date_format()) |                     Carbon::now()->addDays(13)->translatedFormat($entity->date_format()) | ||||||
|                 ), |                 ), | ||||||
|                 ':WEEK' => \sprintf( |                 ':WEEK' => \sprintf( | ||||||
|                     '%s %s %s', |                     '%s %s %s', | ||||||
|                     Carbon::now()->translatedFormat($entity->date_format()), |                     Carbon::now()->translatedFormat($entity->date_format()), | ||||||
|                     ctrans('texts.to'), |                     ctrans('texts.to'), | ||||||
|                     Carbon::now()->addDays(7)->translatedFormat($entity->date_format()) |                     Carbon::now()->addDays(6)->translatedFormat($entity->date_format()) | ||||||
|                 ), |                 ), | ||||||
|             ], |             ], | ||||||
|             'raw' => [ |             'raw' => [ | ||||||
|  | |||||||
| @ -124,6 +124,7 @@ class HtmlEngine | |||||||
|         $data['$line_tax_labels'] = ['value' => $this->lineTaxLabels(), 'label' => ctrans('texts.taxes')]; |         $data['$line_tax_labels'] = ['value' => $this->lineTaxLabels(), 'label' => ctrans('texts.taxes')]; | ||||||
|         $data['$line_tax_values'] = ['value' => $this->lineTaxValues(), '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['$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['$invoice.date'] = &$data['$date']; | ||||||
|         $data['$invoiceDate'] = &$data['$date']; |         $data['$invoiceDate'] = &$data['$date']; | ||||||
| @ -167,6 +168,10 @@ class HtmlEngine | |||||||
|                 $data['$invoice.project'] = &$data['$project.name']; |                 $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) { |             if($this->entity->vendor) { | ||||||
|                 $data['$invoice.vendor'] = ['value' => $this->entity->vendor->present()->name(), 'label' => ctrans('texts.vendor_name')]; |                 $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' => 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' => 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' => 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) { |         foreach ($designs as $design) { | ||||||
|  | |||||||
| @ -101,6 +101,36 @@ | |||||||
|         #content .center {
 |         #content .center {
 | ||||||
|             text-align: 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> |     </style> | ||||||
| </head> | </head> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -80,6 +80,33 @@ | |||||||
|         #content .left {
 |         #content .left {
 | ||||||
|             text-align: left !important; |             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> |     </style> | ||||||
| 
 | 
 | ||||||
|     <!--[if gte mso 9]> |     <!--[if gte mso 9]> | ||||||
|  | |||||||
| @ -143,6 +143,36 @@ | |||||||
|             color: {{ $design == 'dark' ? '#ffffff' : '#000000' }} !important; |             color: {{ $design == 'dark' ? '#ffffff' : '#000000' }} !important; | ||||||
|             opacity: {{ $design == 'dark' ? '87%': '100%' }} !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> |     </style> | ||||||
| </head> | </head> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -47,6 +47,7 @@ | |||||||
|         line-height: var(--line-height); |         line-height: var(--line-height); | ||||||
|         position: fixed; |         position: fixed; | ||||||
|         top: 0; |         top: 0; | ||||||
|  |         width: 100%; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -224,7 +225,7 @@ | |||||||
|         position: fixed; |         position: fixed; | ||||||
|         bottom: 0; |         bottom: 0; | ||||||
|         display: grid; |         display: grid; | ||||||
|         grid-template-columns: 1fr 1fr 1fr; |         grid-template-columns: 1fr; | ||||||
|         gap: 15px; |         gap: 15px; | ||||||
|         color: white; |         color: white; | ||||||
|     } |     } | ||||||
| @ -239,6 +240,11 @@ | |||||||
|         padding-top: 0.5rem |         padding-top: 0.5rem | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     [data-ref="footer_content"]{ | ||||||
|  |         padding-right: 2rem; | ||||||
|  |         margin-right: 2rem; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     table { |     table { | ||||||
|         width: 100%; |         width: 100%; | ||||||
|     } |     } | ||||||
| @ -285,6 +291,36 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
| @ -380,7 +416,7 @@ $entity_images | |||||||
| 
 | 
 | ||||||
| <div id="footer"> | <div id="footer"> | ||||||
|     <div style="width: 100%;"> |     <div style="width: 100%;"> | ||||||
|         <p data-ref="total_table-footer">$entity_footer</p> |         <p data-ref="footer_content">$entity_footer</p> | ||||||
| 
 | 
 | ||||||
|         <script> |         <script> | ||||||
|             // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. |             // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. | ||||||
| @ -402,6 +438,4 @@ $entity_images | |||||||
|             }); |             }); | ||||||
|         </script> |         </script> | ||||||
|     </div> |     </div> | ||||||
|     <div> <!-- #2 column --> </div> |  | ||||||
|     <div> <!-- #3 column --> </div> |  | ||||||
| </div> | </div> | ||||||
|  | |||||||
| @ -276,6 +276,36 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -262,7 +262,37 @@ | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [data-ref="total_table-public_notes"] { font-weight: normal; }  |     [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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
| @ -290,6 +320,7 @@ | |||||||
|     /** To find out selectors on your own: https://invoiceninja.github.io/docs/custom-fields/#snippets **/ |     /** To find out selectors on your own: https://invoiceninja.github.io/docs/custom-fields/#snippets **/ | ||||||
| </style> | </style> | ||||||
| 
 | 
 | ||||||
|  | <div id="body"> | ||||||
| <table style="min-width: 100%"> | <table style="min-width: 100%"> | ||||||
|    <thead> |    <thead> | ||||||
|       <tr> |       <tr> | ||||||
| @ -301,7 +332,7 @@ | |||||||
|    <tbody> |    <tbody> | ||||||
|       <tr> |       <tr> | ||||||
|          <td> |          <td> | ||||||
|             <div id="body"> |             <div id=""> | ||||||
|                <div class="header-wrapper"> |                <div class="header-wrapper"> | ||||||
|                   <div> |                   <div> | ||||||
|                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> |                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> | ||||||
| @ -316,7 +347,7 @@ | |||||||
|                <div id="vendor-details"></div> |                <div id="vendor-details"></div> | ||||||
| 
 | 
 | ||||||
|                <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> |                <table id="entity-details" cellspacing="0" dir="ltr"></table> | ||||||
|                </div> |                </div> | ||||||
|                </div> |                </div> | ||||||
| @ -343,6 +374,7 @@ | |||||||
|       </tr> |       </tr> | ||||||
|    </tfoot> |    </tfoot> | ||||||
| </table> | </table> | ||||||
|  | </div> | ||||||
| 
 | 
 | ||||||
| <div class="repeating-header" id="header"></div> | <div class="repeating-header" id="header"></div> | ||||||
| 
 | 
 | ||||||
| @ -371,24 +403,23 @@ | |||||||
|     </div> |     </div> | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
|         <script> |     <script> | ||||||
|             // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. |         // Clear up space a bit, if [product-table, tasks-table, delivery-note-table] isn't present. | ||||||
|             document.addEventListener('DOMContentLoaded', () => { |         document.addEventListener('DOMContentLoaded', () => { | ||||||
|                 let tables = [ |             let tables = [ | ||||||
|                     'product-table', 'task-table', 'delivery-note-table', |                 'product-table', 'task-table', 'delivery-note-table', | ||||||
|                     'statement-invoice-table', 'statement-payment-table', 'statement-aging-table-totals', |                 'statement-invoice-table', 'statement-payment-table', 'statement-aging-table-totals', | ||||||
|                     'statement-invoice-table-totals', 'statement-payment-table-totals', 'statement-aging-table', |                 'statement-invoice-table-totals', 'statement-payment-table-totals', 'statement-aging-table', | ||||||
|                     'client-details','vendor-details', 'swiss-qr' |                 'client-details','vendor-details', 'swiss-qr' | ||||||
|                 ]; |             ]; | ||||||
| 
 | 
 | ||||||
|                 tables.forEach((tableIdentifier) => { |             tables.forEach((tableIdentifier) => { | ||||||
|                     console.log(document.getElementById(tableIdentifier)); |                 console.log(document.getElementById(tableIdentifier)); | ||||||
| 
 | 
 | ||||||
|                     document.getElementById(tableIdentifier)?.childElementCount === 0 |                 document.getElementById(tableIdentifier)?.childElementCount === 0 | ||||||
|                         ? document.getElementById(tableIdentifier).style.setProperty('display', 'none', 'important') |                     ? document.getElementById(tableIdentifier).style.setProperty('display', 'none', 'important') | ||||||
|                         : ''; |                     : ''; | ||||||
|                 }); |  | ||||||
|             }); |             }); | ||||||
|         </script> |         }); | ||||||
|  |     </script> | ||||||
|        |        | ||||||
| </div> |  | ||||||
| @ -257,6 +257,36 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -229,6 +229,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -234,6 +234,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -251,6 +251,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -278,6 +278,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |       /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|       /** Hide company logo **/ |       /** Hide company logo **/ | ||||||
|  | |||||||
| @ -221,6 +221,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -294,6 +294,36 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
| @ -323,7 +353,7 @@ | |||||||
|     /** For more info, please check our docs: https://invoiceninja.github.io **/ |     /** 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 **/ |     /** To find out selectors on your own: https://invoiceninja.github.io/docs/custom-fields/#snippets **/ | ||||||
| </style> | </style> | ||||||
| 
 | <div id="body"> | ||||||
| <table style="min-width: 100%"> | <table style="min-width: 100%"> | ||||||
|    <thead> |    <thead> | ||||||
|       <tr> |       <tr> | ||||||
| @ -335,7 +365,6 @@ | |||||||
|    <tbody> |    <tbody> | ||||||
|       <tr> |       <tr> | ||||||
|          <td> |          <td> | ||||||
|             <div id="body"> |  | ||||||
|                <div class="header-wrapper"> |                <div class="header-wrapper"> | ||||||
|                   <div> |                   <div> | ||||||
|                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> |                      <img class="company-logo" src="$company.logo" alt="$company.name logo"> | ||||||
| @ -368,7 +397,6 @@ | |||||||
|                <table id="statement-aging-table" cellspacing="0" data-ref="table"></table> |                <table id="statement-aging-table" cellspacing="0" data-ref="table"></table> | ||||||
|                <div id="statement-aging-table-totals" data-ref="statement-totals"></div> |                <div id="statement-aging-table-totals" data-ref="statement-totals"></div> | ||||||
|                <div id="table-totals" cellspacing="0"></div>       |                <div id="table-totals" cellspacing="0"></div>       | ||||||
|             </div> |  | ||||||
|          </td> |          </td> | ||||||
|       </tr> |       </tr> | ||||||
|    </tbody> |    </tbody> | ||||||
| @ -380,7 +408,7 @@ | |||||||
|       </tr> |       </tr> | ||||||
|    </tfoot> |    </tfoot> | ||||||
| </table> | </table> | ||||||
| 
 | </div> | ||||||
| <div class="repeating-header"> | <div class="repeating-header"> | ||||||
|     <div id="header"> |     <div id="header"> | ||||||
|         <div style="background-color: #00968B"><!-- 1 --></div> |         <div style="background-color: #00968B"><!-- 1 --></div> | ||||||
|  | |||||||
| @ -258,6 +258,35 @@ | |||||||
|         overflow-wrap: break-word;  |         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. **/ |     /** Useful snippets, uncomment to enable. **/ | ||||||
| 
 | 
 | ||||||
|     /** Hide company logo **/ |     /** Hide company logo **/ | ||||||
|  | |||||||
| @ -59,5 +59,3 @@ Route::get('checkout/3ds_redirect/{company_key}/{company_gateway_id}/{hash}', [C | |||||||
| Route::get('mollie/3ds_redirect/{company_key}/{company_gateway_id}/{hash}', [Mollie3dsController::class, 'index'])->middleware('domain_db')->name('mollie.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('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::get('.well-known/apple-developer-merchantid-domain-association', [ApplePayDomainController::class, 'showAppleMerchantId']); | ||||||
| 
 |  | ||||||
| Route::fallback([BaseController::class, 'reactCatch']); |  | ||||||
| @ -11,6 +11,7 @@ | |||||||
| 
 | 
 | ||||||
| namespace Tests\Feature; | namespace Tests\Feature; | ||||||
| 
 | 
 | ||||||
|  | use App\Factory\InvoiceItemFactory; | ||||||
| use App\Factory\InvoiceToRecurringInvoiceFactory; | use App\Factory\InvoiceToRecurringInvoiceFactory; | ||||||
| use App\Factory\RecurringInvoiceToInvoiceFactory; | use App\Factory\RecurringInvoiceToInvoiceFactory; | ||||||
| use App\Models\Client; | use App\Models\Client; | ||||||
| @ -51,6 +52,53 @@ class RecurringInvoiceTest extends TestCase | |||||||
|         $this->makeTestData(); |         $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() |     public function testPostRecurringInvoice() | ||||||
|     { |     { | ||||||
|         $data = [ |         $data = [ | ||||||
|  | |||||||
| @ -48,27 +48,4 @@ class AutoBillInvoiceTest extends TestCase | |||||||
|         $this->assertEquals($this->client->fresh()->credit_balance, 0); |         $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