349 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			349 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace Database\Factories;
 | |
| 
 | |
| use App\Abstracts\Factory as AbstractFactory;
 | |
| use App\Events\Document\DocumentCancelled;
 | |
| use App\Events\Document\DocumentCreated;
 | |
| use App\Events\Document\DocumentReceived;
 | |
| use App\Events\Document\DocumentSent;
 | |
| use App\Events\Document\DocumentViewed;
 | |
| use App\Events\Document\PaymentReceived;
 | |
| use App\Jobs\Document\UpdateDocument;
 | |
| use App\Models\Common\Contact;
 | |
| use App\Models\Common\Item;
 | |
| use App\Models\Document\Document as Model;
 | |
| use App\Models\Setting\Tax;
 | |
| use App\Utilities\Date;
 | |
| use App\Utilities\Overrider;
 | |
| use Illuminate\Database\Eloquent\Factories\Factory;
 | |
| 
 | |
| class Document extends AbstractFactory
 | |
| {
 | |
|     /**
 | |
|      * The name of the factory's corresponding model.
 | |
|      *
 | |
|      * @var string
 | |
|      */
 | |
|     protected $model = Model::class;
 | |
| 
 | |
|     /**
 | |
|      * Define the model's default state.
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function definition()
 | |
|     {
 | |
|         $issued_at = $this->faker->dateTimeBetween(now()->startOfYear(), now()->endOfYear())->format('Y-m-d H:i:s');
 | |
|         $due_at = Date::parse($issued_at)->addDays($this->faker->randomNumber(3))->format('Y-m-d H:i:s');
 | |
| 
 | |
|         return [
 | |
|             'company_id' => $this->company->id,
 | |
|             'issued_at' => $issued_at,
 | |
|             'due_at' => $due_at,
 | |
|             'currency_code' => setting('default.currency'),
 | |
|             'currency_rate' => '1',
 | |
|             'notes' => $this->faker->text(5),
 | |
|             'amount' => '0',
 | |
|             'created_from' => 'core::factory',
 | |
|         ];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model type is invoice.
 | |
|      */
 | |
|     public function invoice(): Factory
 | |
|     {
 | |
|         return $this->state(function (array $attributes): array {
 | |
|             $contacts = Contact::customer()->enabled()->get();
 | |
| 
 | |
|             if ($contacts->count()) {
 | |
|                 $contact = $contacts->random(1)->first();
 | |
|             } else {
 | |
|                 $contact = Contact::factory()->customer()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $statuses = ['draft', 'sent', 'viewed', 'partial', 'paid', 'cancelled'];
 | |
| 
 | |
|             return [
 | |
|                 'type' => Model::INVOICE_TYPE,
 | |
|                 'document_number' => setting('invoice.number_prefix') . $this->faker->randomNumber(setting('invoice.number_digit')),
 | |
|                 'category_id' => $this->company->categories()->income()->get()->random(1)->pluck('id')->first(),
 | |
|                 'contact_id' => $contact->id,
 | |
|                 'contact_name' => $contact->name,
 | |
|                 'contact_email' => $contact->email,
 | |
|                 'contact_tax_number' => $contact->tax_number,
 | |
|                 'contact_phone' => $contact->phone,
 | |
|                 'contact_address' => $contact->address,
 | |
|                 'status' => $this->faker->randomElement($statuses),
 | |
|             ];
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model type is bill.
 | |
|      */
 | |
|     public function bill(): Factory
 | |
|     {
 | |
|         return $this->state(function (array $attributes): array {
 | |
|             $contacts = Contact::vendor()->enabled()->get();
 | |
| 
 | |
|             if ($contacts->count()) {
 | |
|                 $contact = $contacts->random(1)->first();
 | |
|             } else {
 | |
|                 $contact = Contact::factory()->vendor()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $statuses = ['draft', 'received', 'partial', 'paid', 'cancelled'];
 | |
| 
 | |
|             return [
 | |
|                 'type' => Model::BILL_TYPE,
 | |
|                 'document_number' => setting('bill.number_prefix') . $this->faker->randomNumber(setting('bill.number_digit')),
 | |
|                 'category_id' => $this->company->categories()->expense()->get()->random(1)->pluck('id')->first(),
 | |
|                 'contact_id' => $contact->id,
 | |
|                 'contact_name' => $contact->name,
 | |
|                 'contact_email' => $contact->email,
 | |
|                 'contact_tax_number' => $contact->tax_number,
 | |
|                 'contact_phone' => $contact->phone,
 | |
|                 'contact_address' => $contact->address,
 | |
|                 'status' => $this->faker->randomElement($statuses),
 | |
|             ];
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is draft.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function draft()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'draft',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is received.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function received()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'received',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is sent.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function sent()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'sent',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is viewed.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function viewed()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'viewed',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is partial.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function partial()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'partial',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is paid.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function paid()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'paid',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model status is cancelled.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function cancelled()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'status' => 'cancelled',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model is recurring.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function recurring()
 | |
|     {
 | |
|         return $this->state([
 | |
|             'recurring_frequency' => 'yes',
 | |
|             'recurring_interval' => '1',
 | |
|             'recurring_custom_frequency' => $this->faker->randomElement(['monthly', 'weekly']),
 | |
|             'recurring_count' => '1',
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Indicate that the model has items.
 | |
|      *
 | |
|      * @return \Illuminate\Database\Eloquent\Factories\Factory
 | |
|      */
 | |
|     public function items()
 | |
|     {
 | |
|         return $this->state(function (array $attributes) {
 | |
|             $amount = $this->faker->randomFloat(2, 1, 1000);
 | |
| 
 | |
|             $taxes = Tax::enabled()->get();
 | |
| 
 | |
|             if ($taxes->count()) {
 | |
|                 $tax = $taxes->random(1)->first();
 | |
|             } else {
 | |
|                 $tax = Tax::factory()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $items = Item::enabled()->get();
 | |
| 
 | |
|             if ($items->count()) {
 | |
|                 $item = $items->random(1)->first();
 | |
|             } else {
 | |
|                 $item = Item::factory()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $items = [
 | |
|                 [
 | |
|                     'type' => $attributes['type'],
 | |
|                     'name' => $item->name,
 | |
|                     'description' => $this->faker->text,
 | |
|                     'item_id' => $item->id,
 | |
|                     'tax_ids' => [$tax->id],
 | |
|                     'quantity' => '1',
 | |
|                     'price' => $amount,
 | |
|                     'currency' => setting('default.currency'),
 | |
|                 ],
 | |
|             ];
 | |
| 
 | |
|             return [
 | |
|                 'items' => $items,
 | |
|                 'recurring_frequency' => 'no',
 | |
|             ];
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Configure the model factory.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function configure()
 | |
|     {
 | |
|         return $this->afterCreating(function (Model $document) {
 | |
|             Overrider::load('currencies');
 | |
| 
 | |
|             $init_status = $document->status;
 | |
| 
 | |
|             $document->status = 'draft';
 | |
|             event(new DocumentCreated($document, request()));
 | |
|             $document->status = $init_status;
 | |
| 
 | |
|             $amount = $this->faker->randomFloat(2, 1, 1000);
 | |
| 
 | |
|             $taxes = Tax::enabled()->get();
 | |
| 
 | |
|             if ($taxes->count()) {
 | |
|                 $tax = $taxes->random(1)->first();
 | |
|             } else {
 | |
|                 $tax = Tax::factory()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $items = Item::enabled()->get();
 | |
| 
 | |
|             if ($items->count()) {
 | |
|                 $item = $items->random(1)->first();
 | |
|             } else {
 | |
|                 $item = Item::factory()->enabled()->create();
 | |
|             }
 | |
| 
 | |
|             $items = [
 | |
|                 [
 | |
|                     'name' => $item->name,
 | |
|                     'description' => $this->faker->text,
 | |
|                     'item_id' => $item->id,
 | |
|                     'tax_ids' => [$tax->id],
 | |
|                     'quantity' => '1',
 | |
|                     'price' => $amount,
 | |
|                     'currency' => $document->currency_code,
 | |
|                 ],
 | |
|             ];
 | |
| 
 | |
|             $request = [
 | |
|                 'items' => $items,
 | |
|                 'recurring_frequency' => 'no',
 | |
|             ];
 | |
| 
 | |
|             $updated_document = $this->dispatch(new UpdateDocument($document, $request));
 | |
| 
 | |
|             switch ($init_status) {
 | |
|                 case 'received':
 | |
|                     event(new DocumentReceived($updated_document));
 | |
| 
 | |
|                     break;
 | |
|                 case 'sent':
 | |
|                     event(new DocumentSent($updated_document));
 | |
| 
 | |
|                     break;
 | |
|                 case 'viewed':
 | |
|                     $updated_document->status = 'sent';
 | |
|                     event(new DocumentViewed($updated_document));
 | |
|                     $updated_document->status = $init_status;
 | |
| 
 | |
|                     break;
 | |
|                 case 'partial':
 | |
|                 case 'paid':
 | |
|                     $payment_request = [
 | |
|                         'paid_at' => $updated_document->due_at,
 | |
|                         'type' => config('type.' . $document->type . '.transaction_type'),
 | |
|                     ];
 | |
| 
 | |
|                     if ($init_status === 'partial') {
 | |
|                         $payment_request['amount'] = (int) round($amount / 3, $document->currency->precision);
 | |
|                     }
 | |
| 
 | |
|                     event(new PaymentReceived($updated_document, $payment_request));
 | |
| 
 | |
|                     break;
 | |
|                 case 'cancelled':
 | |
|                     event(new DocumentCancelled($updated_document));
 | |
| 
 | |
|                     break;
 | |
|             }
 | |
|         });
 | |
|     }
 | |
| }
 |