mirror of
				https://github.com/crater-invoice/crater.git
				synced 2025-11-03 22:13:18 -05:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			dark-invoi
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 59f433dd00 | 
							
								
								
									
										162
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										162
									
								
								.github/workflows/uffizzi-build.yml
									
									
									
									
										vendored
									
									
								
							@ -1,162 +0,0 @@
 | 
				
			|||||||
name: Build PR Image
 | 
					 | 
				
			||||||
on:
 | 
					 | 
				
			||||||
  pull_request:
 | 
					 | 
				
			||||||
    types: [opened,synchronize,reopened,closed]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
jobs:
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  build-application:
 | 
					 | 
				
			||||||
    name: Build and Push `application`
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }}
 | 
					 | 
				
			||||||
    outputs:
 | 
					 | 
				
			||||||
      tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: Checkout git repo
 | 
					 | 
				
			||||||
        uses: actions/checkout@v3
 | 
					 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					 | 
				
			||||||
        uses: docker/setup-buildx-action@v2        
 | 
					 | 
				
			||||||
      - name: Generate UUID image name
 | 
					 | 
				
			||||||
        id: uuid
 | 
					 | 
				
			||||||
        run: echo "UUID_TAG_APP=$(uuidgen)" >> $GITHUB_ENV
 | 
					 | 
				
			||||||
      - name: Docker metadata
 | 
					 | 
				
			||||||
        id: meta
 | 
					 | 
				
			||||||
        uses: docker/metadata-action@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          images: registry.uffizzi.com/${{ env.UUID_TAG_APP }}
 | 
					 | 
				
			||||||
          tags: type=raw,value=60d
 | 
					 | 
				
			||||||
      - name: Build and Push Image to registry.uffizzi.com ephemeral registry
 | 
					 | 
				
			||||||
        uses: docker/build-push-action@v2
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          push: true
 | 
					 | 
				
			||||||
          context: ./
 | 
					 | 
				
			||||||
          tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
					 | 
				
			||||||
          file: ./uffizzi/Dockerfile
 | 
					 | 
				
			||||||
          cache-from: type=gha
 | 
					 | 
				
			||||||
          cache-to: type=gha,mode=max
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  build-nginx:
 | 
					 | 
				
			||||||
    name: Build and Push `nginx`
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }}
 | 
					 | 
				
			||||||
    outputs:
 | 
					 | 
				
			||||||
      tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: Checkout git repo
 | 
					 | 
				
			||||||
        uses: actions/checkout@v3
 | 
					 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					 | 
				
			||||||
        uses: docker/setup-buildx-action@v2        
 | 
					 | 
				
			||||||
      - name: Generate UUID image name
 | 
					 | 
				
			||||||
        id: uuid
 | 
					 | 
				
			||||||
        run: echo "UUID_TAG_NGINX=$(uuidgen)" >> $GITHUB_ENV
 | 
					 | 
				
			||||||
      - name: Docker metadata
 | 
					 | 
				
			||||||
        id: meta
 | 
					 | 
				
			||||||
        uses: docker/metadata-action@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          images: registry.uffizzi.com/${{ env.UUID_TAG_NGINX }}
 | 
					 | 
				
			||||||
          tags: type=raw,value=60d
 | 
					 | 
				
			||||||
      - name: Build and Push Image to Uffizzi ephemeral registry
 | 
					 | 
				
			||||||
        uses: docker/build-push-action@v2
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          push: true
 | 
					 | 
				
			||||||
          context: ./
 | 
					 | 
				
			||||||
          tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
					 | 
				
			||||||
          file: ./uffizzi/nginx/Dockerfile
 | 
					 | 
				
			||||||
          cache-from: type=gha
 | 
					 | 
				
			||||||
          cache-to: type=gha,mode=max
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  build-crond:
 | 
					 | 
				
			||||||
    name: Build and Push `crond`
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    if: ${{ github.event_name != 'pull_request' || github.event.action != 'closed' }}
 | 
					 | 
				
			||||||
    outputs:
 | 
					 | 
				
			||||||
      tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: Checkout git repo
 | 
					 | 
				
			||||||
        uses: actions/checkout@v3
 | 
					 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					 | 
				
			||||||
        uses: docker/setup-buildx-action@v2        
 | 
					 | 
				
			||||||
      - name: Generate UUID image name
 | 
					 | 
				
			||||||
        id: uuid
 | 
					 | 
				
			||||||
        run: echo "UUID_TAG_CROND=$(uuidgen)" >> $GITHUB_ENV
 | 
					 | 
				
			||||||
      - name: Docker metadata
 | 
					 | 
				
			||||||
        id: meta
 | 
					 | 
				
			||||||
        uses: docker/metadata-action@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          images: registry.uffizzi.com/${{ env.UUID_TAG_CROND }}
 | 
					 | 
				
			||||||
          tags: type=raw,value=60d
 | 
					 | 
				
			||||||
      - name: Build and Push Image to registry.uffizzi.com ephemeral registry
 | 
					 | 
				
			||||||
        uses: docker/build-push-action@v2
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          push: true
 | 
					 | 
				
			||||||
          context: ./
 | 
					 | 
				
			||||||
          tags: ${{ steps.meta.outputs.tags }}
 | 
					 | 
				
			||||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
					 | 
				
			||||||
          file: ./uffizzi/crond/Dockerfile      
 | 
					 | 
				
			||||||
          cache-from: type=gha
 | 
					 | 
				
			||||||
          cache-to: type=gha,mode=max
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  render-compose-file:
 | 
					 | 
				
			||||||
    name: Render Docker Compose File
 | 
					 | 
				
			||||||
    # Pass output of this workflow to another triggered by `workflow_run` event.
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    outputs:
 | 
					 | 
				
			||||||
      compose-file-cache-key: ${{ steps.hash.outputs.hash }}
 | 
					 | 
				
			||||||
    needs: 
 | 
					 | 
				
			||||||
      - build-application
 | 
					 | 
				
			||||||
      - build-nginx
 | 
					 | 
				
			||||||
      - build-crond
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: Checkout git repo
 | 
					 | 
				
			||||||
        uses: actions/checkout@v3
 | 
					 | 
				
			||||||
      - name: Render Compose File
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          APP_IMAGE=$(echo ${{ needs.build-application.outputs.tags }})
 | 
					 | 
				
			||||||
          export APP_IMAGE
 | 
					 | 
				
			||||||
          NGINX_IMAGE=$(echo ${{ needs.build-nginx.outputs.tags }})
 | 
					 | 
				
			||||||
          export NGINX_IMAGE
 | 
					 | 
				
			||||||
          CROND_IMAGE=$(echo ${{ needs.build-crond.outputs.tags }})
 | 
					 | 
				
			||||||
          export CROND_IMAGE
 | 
					 | 
				
			||||||
          # Render simple template from environment variables.
 | 
					 | 
				
			||||||
          envsubst < ./uffizzi/docker-compose.uffizzi.yml > docker-compose.rendered.yml
 | 
					 | 
				
			||||||
          cat docker-compose.rendered.yml
 | 
					 | 
				
			||||||
      - name: Upload Rendered Compose File as Artifact
 | 
					 | 
				
			||||||
        uses: actions/upload-artifact@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          name: preview-spec
 | 
					 | 
				
			||||||
          path: docker-compose.rendered.yml
 | 
					 | 
				
			||||||
          retention-days: 2
 | 
					 | 
				
			||||||
      - name: Serialize PR Event to File
 | 
					 | 
				
			||||||
        run:  |
 | 
					 | 
				
			||||||
          cat << EOF > event.json
 | 
					 | 
				
			||||||
          ${{ toJSON(github.event) }} 
 | 
					 | 
				
			||||||
          
 | 
					 | 
				
			||||||
          EOF
 | 
					 | 
				
			||||||
      - name: Upload PR Event as Artifact
 | 
					 | 
				
			||||||
        uses: actions/upload-artifact@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          name: preview-spec
 | 
					 | 
				
			||||||
          path: event.json
 | 
					 | 
				
			||||||
          retention-days: 2
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  delete-preview:
 | 
					 | 
				
			||||||
    name: Call for Preview Deletion
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    if: ${{ github.event.action == 'closed' }}
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      # If this PR is closing, we will not render a compose file nor pass it to the next workflow.
 | 
					 | 
				
			||||||
      - name: Serialize PR Event to File
 | 
					 | 
				
			||||||
        run: echo '${{ toJSON(github.event) }}' > event.json
 | 
					 | 
				
			||||||
      - name: Upload PR Event as Artifact
 | 
					 | 
				
			||||||
        uses: actions/upload-artifact@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          name: preview-spec
 | 
					 | 
				
			||||||
          path: event.json
 | 
					 | 
				
			||||||
          retention-days: 2
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
							
								
								
									
										84
									
								
								.github/workflows/uffizzi-preview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										84
									
								
								.github/workflows/uffizzi-preview.yml
									
									
									
									
										vendored
									
									
								
							@ -1,84 +0,0 @@
 | 
				
			|||||||
name: Deploy Uffizzi Preview
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
on:
 | 
					 | 
				
			||||||
  workflow_run:
 | 
					 | 
				
			||||||
    workflows:
 | 
					 | 
				
			||||||
      - "Build PR Image"
 | 
					 | 
				
			||||||
    types:
 | 
					 | 
				
			||||||
      - completed
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
jobs:
 | 
					 | 
				
			||||||
  cache-compose-file:
 | 
					 | 
				
			||||||
    name: Cache Compose File
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    outputs:
 | 
					 | 
				
			||||||
      compose-file-cache-key: ${{ env.COMPOSE_FILE_HASH }}
 | 
					 | 
				
			||||||
      pr-number: ${{ env.PR_NUMBER }}
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: 'Download artifacts'
 | 
					 | 
				
			||||||
        # Fetch output (zip archive) from the workflow run that triggered this workflow.
 | 
					 | 
				
			||||||
        uses: actions/github-script@v6
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          script: |
 | 
					 | 
				
			||||||
            let allArtifacts = await github.rest.actions.listWorkflowRunArtifacts({
 | 
					 | 
				
			||||||
               owner: context.repo.owner,
 | 
					 | 
				
			||||||
               repo: context.repo.repo,
 | 
					 | 
				
			||||||
               run_id: context.payload.workflow_run.id,
 | 
					 | 
				
			||||||
            });
 | 
					 | 
				
			||||||
            let matchArtifact = allArtifacts.data.artifacts.filter((artifact) => {
 | 
					 | 
				
			||||||
              return artifact.name == "preview-spec"
 | 
					 | 
				
			||||||
            })[0];
 | 
					 | 
				
			||||||
            let download = await github.rest.actions.downloadArtifact({
 | 
					 | 
				
			||||||
               owner: context.repo.owner,
 | 
					 | 
				
			||||||
               repo: context.repo.repo,
 | 
					 | 
				
			||||||
               artifact_id: matchArtifact.id,
 | 
					 | 
				
			||||||
               archive_format: 'zip',
 | 
					 | 
				
			||||||
            });
 | 
					 | 
				
			||||||
            let fs = require('fs');
 | 
					 | 
				
			||||||
            fs.writeFileSync(`${process.env.GITHUB_WORKSPACE}/preview-spec.zip`, Buffer.from(download.data));
 | 
					 | 
				
			||||||
      - name: 'Unzip artifact'
 | 
					 | 
				
			||||||
        run: unzip preview-spec.zip
 | 
					 | 
				
			||||||
      - name: Read Event into ENV
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          echo 'EVENT_JSON<<EOF' >> $GITHUB_ENV
 | 
					 | 
				
			||||||
          cat event.json >> $GITHUB_ENV
 | 
					 | 
				
			||||||
          echo 'EOF' >> $GITHUB_ENV
 | 
					 | 
				
			||||||
      - name: Hash Rendered Compose File
 | 
					 | 
				
			||||||
        id: hash
 | 
					 | 
				
			||||||
        # If the previous workflow was triggered by a PR close event, we will not have a compose file artifact.
 | 
					 | 
				
			||||||
        if: ${{ fromJSON(env.EVENT_JSON).action != 'closed' }}
 | 
					 | 
				
			||||||
        run: echo "COMPOSE_FILE_HASH=$(md5sum docker-compose.rendered.yml | awk '{ print $1 }')" >> $GITHUB_ENV
 | 
					 | 
				
			||||||
      - name: Cache Rendered Compose File
 | 
					 | 
				
			||||||
        if: ${{ fromJSON(env.EVENT_JSON).action != 'closed' }}
 | 
					 | 
				
			||||||
        uses: actions/cache@v3
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          path: docker-compose.rendered.yml
 | 
					 | 
				
			||||||
          key: ${{ env.COMPOSE_FILE_HASH }}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      - name: Read PR Number From Event Object
 | 
					 | 
				
			||||||
        id: pr
 | 
					 | 
				
			||||||
        run: echo "PR_NUMBER=${{ fromJSON(env.EVENT_JSON).number }}" >> $GITHUB_ENV
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      - name: DEBUG - Print Job Outputs
 | 
					 | 
				
			||||||
        if: ${{ runner.debug }}
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          echo "PR number: ${{ env.PR_NUMBER }}"
 | 
					 | 
				
			||||||
          echo "Compose file hash: ${{ env.COMPOSE_FILE_HASH }}"
 | 
					 | 
				
			||||||
          cat event.json
 | 
					 | 
				
			||||||
  deploy-uffizzi-preview:
 | 
					 | 
				
			||||||
    name: Use Remote Workflow to Preview on Uffizzi
 | 
					 | 
				
			||||||
    needs:
 | 
					 | 
				
			||||||
      - cache-compose-file
 | 
					 | 
				
			||||||
    uses: UffizziCloud/preview-action/.github/workflows/reusable.yaml@v2.6.1
 | 
					 | 
				
			||||||
    with:
 | 
					 | 
				
			||||||
      # If this workflow was triggered by a PR close event, cache-key will be an empty string
 | 
					 | 
				
			||||||
      # and this reusable workflow will delete the preview deployment.
 | 
					 | 
				
			||||||
      compose-file-cache-key: ${{ needs.cache-compose-file.outputs.compose-file-cache-key }}
 | 
					 | 
				
			||||||
      compose-file-cache-path: docker-compose.rendered.yml
 | 
					 | 
				
			||||||
      server: https://app.uffizzi.com/
 | 
					 | 
				
			||||||
      pr-number: ${{ needs.cache-compose-file.outputs.pr-number }}
 | 
					 | 
				
			||||||
    permissions:
 | 
					 | 
				
			||||||
      contents: read
 | 
					 | 
				
			||||||
      pull-requests: write
 | 
					 | 
				
			||||||
      id-token: write
 | 
					 | 
				
			||||||
@ -1,4 +1,4 @@
 | 
				
			|||||||
FROM php:8.1-fpm
 | 
					FROM php:7.4-fpm
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Arguments defined in docker-compose.yml
 | 
					# Arguments defined in docker-compose.yml
 | 
				
			||||||
ARG user
 | 
					ARG user
 | 
				
			||||||
 | 
				
			|||||||
@ -103,7 +103,6 @@ class CustomerStatsController extends Controller
 | 
				
			|||||||
        )
 | 
					        )
 | 
				
			||||||
            ->whereCompany()
 | 
					            ->whereCompany()
 | 
				
			||||||
            ->whereCustomer($customer->id)
 | 
					            ->whereCustomer($customer->id)
 | 
				
			||||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
					 | 
				
			||||||
            ->sum('total');
 | 
					            ->sum('total');
 | 
				
			||||||
        $totalReceipts = Payment::whereBetween(
 | 
					        $totalReceipts = Payment::whereBetween(
 | 
				
			||||||
            'payment_date',
 | 
					            'payment_date',
 | 
				
			||||||
 | 
				
			|||||||
@ -104,7 +104,6 @@ class DashboardController extends Controller
 | 
				
			|||||||
            'invoice_date',
 | 
					            'invoice_date',
 | 
				
			||||||
            [$startDate->format('Y-m-d'), $start->format('Y-m-d')]
 | 
					            [$startDate->format('Y-m-d'), $start->format('Y-m-d')]
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
					 | 
				
			||||||
            ->whereCompany()
 | 
					            ->whereCompany()
 | 
				
			||||||
            ->sum('base_total');
 | 
					            ->sum('base_total');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -142,7 +141,6 @@ class DashboardController extends Controller
 | 
				
			|||||||
        $recent_due_invoices = Invoice::with('customer')
 | 
					        $recent_due_invoices = Invoice::with('customer')
 | 
				
			||||||
            ->whereCompany()
 | 
					            ->whereCompany()
 | 
				
			||||||
            ->where('base_due_amount', '>', 0)
 | 
					            ->where('base_due_amount', '>', 0)
 | 
				
			||||||
            ->where('status', '<>', Invoice::STATUS_DRAFT)
 | 
					 | 
				
			||||||
            ->take(5)
 | 
					            ->take(5)
 | 
				
			||||||
            ->latest()
 | 
					            ->latest()
 | 
				
			||||||
            ->get();
 | 
					            ->get();
 | 
				
			||||||
 | 
				
			|||||||
@ -24,7 +24,6 @@ class InvoicesController extends Controller
 | 
				
			|||||||
        $limit = $request->has('limit') ? $request->limit : 10;
 | 
					        $limit = $request->has('limit') ? $request->limit : 10;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $invoices = Invoice::whereCompany()
 | 
					        $invoices = Invoice::whereCompany()
 | 
				
			||||||
            ->whereTabFilters($request->tab_status)
 | 
					 | 
				
			||||||
            ->join('customers', 'customers.id', '=', 'invoices.customer_id')
 | 
					            ->join('customers', 'customers.id', '=', 'invoices.customer_id')
 | 
				
			||||||
            ->applyFilters($request->all())
 | 
					            ->applyFilters($request->all())
 | 
				
			||||||
            ->select('invoices.*', 'customers.name')
 | 
					            ->select('invoices.*', 'customers.name')
 | 
				
			||||||
 | 
				
			|||||||
@ -2,15 +2,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
					namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use PDF;
 | 
					 | 
				
			||||||
use Carbon\Carbon;
 | 
					use Carbon\Carbon;
 | 
				
			||||||
 | 
					use Crater\Http\Controllers\Controller;
 | 
				
			||||||
use Crater\Models\Company;
 | 
					use Crater\Models\Company;
 | 
				
			||||||
use Crater\Models\Currency;
 | 
					use Crater\Models\CompanySetting;
 | 
				
			||||||
use Crater\Models\Customer;
 | 
					use Crater\Models\Customer;
 | 
				
			||||||
use Illuminate\Http\Request;
 | 
					use Illuminate\Http\Request;
 | 
				
			||||||
use Crater\Models\CompanySetting;
 | 
					 | 
				
			||||||
use Illuminate\Support\Facades\App;
 | 
					use Illuminate\Support\Facades\App;
 | 
				
			||||||
use Crater\Http\Controllers\Controller;
 | 
					use PDF;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class CustomerSalesReportController extends Controller
 | 
					class CustomerSalesReportController extends Controller
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
@ -57,7 +56,6 @@ class CustomerSalesReportController extends Controller
 | 
				
			|||||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
					        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
				
			||||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
					        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
				
			||||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
					        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
				
			||||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $colors = [
 | 
					        $colors = [
 | 
				
			||||||
            'primary_text_color',
 | 
					            'primary_text_color',
 | 
				
			||||||
@ -82,7 +80,6 @@ class CustomerSalesReportController extends Controller
 | 
				
			|||||||
            'company' => $company,
 | 
					            'company' => $company,
 | 
				
			||||||
            'from_date' => $from_date,
 | 
					            'from_date' => $from_date,
 | 
				
			||||||
            'to_date' => $to_date,
 | 
					            'to_date' => $to_date,
 | 
				
			||||||
            'currency' => $currency,
 | 
					 | 
				
			||||||
        ]);
 | 
					        ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $pdf = PDF::loadView('app.pdf.reports.sales-customers');
 | 
					        $pdf = PDF::loadView('app.pdf.reports.sales-customers');
 | 
				
			||||||
 | 
				
			|||||||
@ -2,15 +2,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
					namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use PDF;
 | 
					 | 
				
			||||||
use Carbon\Carbon;
 | 
					use Carbon\Carbon;
 | 
				
			||||||
use Crater\Models\Company;
 | 
					 | 
				
			||||||
use Crater\Models\Expense;
 | 
					 | 
				
			||||||
use Crater\Models\Currency;
 | 
					 | 
				
			||||||
use Illuminate\Http\Request;
 | 
					 | 
				
			||||||
use Crater\Models\CompanySetting;
 | 
					 | 
				
			||||||
use Illuminate\Support\Facades\App;
 | 
					 | 
				
			||||||
use Crater\Http\Controllers\Controller;
 | 
					use Crater\Http\Controllers\Controller;
 | 
				
			||||||
 | 
					use Crater\Models\Company;
 | 
				
			||||||
 | 
					use Crater\Models\CompanySetting;
 | 
				
			||||||
 | 
					use Crater\Models\Expense;
 | 
				
			||||||
 | 
					use Illuminate\Http\Request;
 | 
				
			||||||
 | 
					use Illuminate\Support\Facades\App;
 | 
				
			||||||
 | 
					use PDF;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ExpensesReportController extends Controller
 | 
					class ExpensesReportController extends Controller
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
@ -44,7 +43,6 @@ class ExpensesReportController extends Controller
 | 
				
			|||||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
					        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
				
			||||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
					        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
				
			||||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
					        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
				
			||||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $colors = [
 | 
					        $colors = [
 | 
				
			||||||
            'primary_text_color',
 | 
					            'primary_text_color',
 | 
				
			||||||
@ -68,7 +66,6 @@ class ExpensesReportController extends Controller
 | 
				
			|||||||
            'company' => $company,
 | 
					            'company' => $company,
 | 
				
			||||||
            'from_date' => $from_date,
 | 
					            'from_date' => $from_date,
 | 
				
			||||||
            'to_date' => $to_date,
 | 
					            'to_date' => $to_date,
 | 
				
			||||||
            'currency' => $currency,
 | 
					 | 
				
			||||||
        ]);
 | 
					        ]);
 | 
				
			||||||
        $pdf = PDF::loadView('app.pdf.reports.expenses');
 | 
					        $pdf = PDF::loadView('app.pdf.reports.expenses');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -2,15 +2,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
					namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use PDF;
 | 
					 | 
				
			||||||
use Carbon\Carbon;
 | 
					use Carbon\Carbon;
 | 
				
			||||||
use Crater\Models\Company;
 | 
					 | 
				
			||||||
use Crater\Models\Currency;
 | 
					 | 
				
			||||||
use Illuminate\Http\Request;
 | 
					 | 
				
			||||||
use Crater\Models\InvoiceItem;
 | 
					 | 
				
			||||||
use Crater\Models\CompanySetting;
 | 
					 | 
				
			||||||
use Illuminate\Support\Facades\App;
 | 
					 | 
				
			||||||
use Crater\Http\Controllers\Controller;
 | 
					use Crater\Http\Controllers\Controller;
 | 
				
			||||||
 | 
					use Crater\Models\Company;
 | 
				
			||||||
 | 
					use Crater\Models\CompanySetting;
 | 
				
			||||||
 | 
					use Crater\Models\InvoiceItem;
 | 
				
			||||||
 | 
					use Illuminate\Http\Request;
 | 
				
			||||||
 | 
					use Illuminate\Support\Facades\App;
 | 
				
			||||||
 | 
					use PDF;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ItemSalesReportController extends Controller
 | 
					class ItemSalesReportController extends Controller
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
@ -44,7 +43,6 @@ class ItemSalesReportController extends Controller
 | 
				
			|||||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
					        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
				
			||||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
					        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
				
			||||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
					        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
				
			||||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $colors = [
 | 
					        $colors = [
 | 
				
			||||||
            'primary_text_color',
 | 
					            'primary_text_color',
 | 
				
			||||||
@ -68,7 +66,6 @@ class ItemSalesReportController extends Controller
 | 
				
			|||||||
            'company' => $company,
 | 
					            'company' => $company,
 | 
				
			||||||
            'from_date' => $from_date,
 | 
					            'from_date' => $from_date,
 | 
				
			||||||
            'to_date' => $to_date,
 | 
					            'to_date' => $to_date,
 | 
				
			||||||
            'currency' => $currency,
 | 
					 | 
				
			||||||
        ]);
 | 
					        ]);
 | 
				
			||||||
        $pdf = PDF::loadView('app.pdf.reports.sales-items');
 | 
					        $pdf = PDF::loadView('app.pdf.reports.sales-items');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -2,16 +2,15 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
					namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use PDF;
 | 
					 | 
				
			||||||
use Carbon\Carbon;
 | 
					use Carbon\Carbon;
 | 
				
			||||||
 | 
					use Crater\Http\Controllers\Controller;
 | 
				
			||||||
use Crater\Models\Company;
 | 
					use Crater\Models\Company;
 | 
				
			||||||
 | 
					use Crater\Models\CompanySetting;
 | 
				
			||||||
use Crater\Models\Expense;
 | 
					use Crater\Models\Expense;
 | 
				
			||||||
use Crater\Models\Payment;
 | 
					use Crater\Models\Payment;
 | 
				
			||||||
use Crater\Models\Currency;
 | 
					 | 
				
			||||||
use Illuminate\Http\Request;
 | 
					use Illuminate\Http\Request;
 | 
				
			||||||
use Crater\Models\CompanySetting;
 | 
					 | 
				
			||||||
use Illuminate\Support\Facades\App;
 | 
					use Illuminate\Support\Facades\App;
 | 
				
			||||||
use Crater\Http\Controllers\Controller;
 | 
					use PDF;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ProfitLossReportController extends Controller
 | 
					class ProfitLossReportController extends Controller
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
@ -50,8 +49,6 @@ class ProfitLossReportController extends Controller
 | 
				
			|||||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
					        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
				
			||||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
					        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
				
			||||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
					        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
				
			||||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $colors = [
 | 
					        $colors = [
 | 
				
			||||||
            'primary_text_color',
 | 
					            'primary_text_color',
 | 
				
			||||||
@ -77,7 +74,6 @@ class ProfitLossReportController extends Controller
 | 
				
			|||||||
            'company' => $company,
 | 
					            'company' => $company,
 | 
				
			||||||
            'from_date' => $from_date,
 | 
					            'from_date' => $from_date,
 | 
				
			||||||
            'to_date' => $to_date,
 | 
					            'to_date' => $to_date,
 | 
				
			||||||
            'currency' => $currency,
 | 
					 | 
				
			||||||
        ]);
 | 
					        ]);
 | 
				
			||||||
        $pdf = PDF::loadView('app.pdf.reports.profit-loss');
 | 
					        $pdf = PDF::loadView('app.pdf.reports.profit-loss');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -2,15 +2,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
					namespace Crater\Http\Controllers\V1\Admin\Report;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use PDF;
 | 
					 | 
				
			||||||
use Carbon\Carbon;
 | 
					use Carbon\Carbon;
 | 
				
			||||||
use Crater\Models\Tax;
 | 
					 | 
				
			||||||
use Crater\Models\Company;
 | 
					 | 
				
			||||||
use Crater\Models\Currency;
 | 
					 | 
				
			||||||
use Illuminate\Http\Request;
 | 
					 | 
				
			||||||
use Crater\Models\CompanySetting;
 | 
					 | 
				
			||||||
use Illuminate\Support\Facades\App;
 | 
					 | 
				
			||||||
use Crater\Http\Controllers\Controller;
 | 
					use Crater\Http\Controllers\Controller;
 | 
				
			||||||
 | 
					use Crater\Models\Company;
 | 
				
			||||||
 | 
					use Crater\Models\CompanySetting;
 | 
				
			||||||
 | 
					use Crater\Models\Tax;
 | 
				
			||||||
 | 
					use Illuminate\Http\Request;
 | 
				
			||||||
 | 
					use Illuminate\Support\Facades\App;
 | 
				
			||||||
 | 
					use PDF;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class TaxSummaryReportController extends Controller
 | 
					class TaxSummaryReportController extends Controller
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
@ -45,8 +44,6 @@ class TaxSummaryReportController extends Controller
 | 
				
			|||||||
        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
					        $dateFormat = CompanySetting::getSetting('carbon_date_format', $company->id);
 | 
				
			||||||
        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
					        $from_date = Carbon::createFromFormat('Y-m-d', $request->from_date)->format($dateFormat);
 | 
				
			||||||
        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
					        $to_date = Carbon::createFromFormat('Y-m-d', $request->to_date)->format($dateFormat);
 | 
				
			||||||
        $currency = Currency::findOrFail(CompanySetting::getSetting('currency', $company->id));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $colors = [
 | 
					        $colors = [
 | 
				
			||||||
            'primary_text_color',
 | 
					            'primary_text_color',
 | 
				
			||||||
@ -71,7 +68,6 @@ class TaxSummaryReportController extends Controller
 | 
				
			|||||||
            'company' => $company,
 | 
					            'company' => $company,
 | 
				
			||||||
            'from_date' => $from_date,
 | 
					            'from_date' => $from_date,
 | 
				
			||||||
            'to_date' => $to_date,
 | 
					            'to_date' => $to_date,
 | 
				
			||||||
            'currency' => $currency,
 | 
					 | 
				
			||||||
        ]);
 | 
					        ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        $pdf = PDF::loadView('app.pdf.reports.tax-summary');
 | 
					        $pdf = PDF::loadView('app.pdf.reports.tax-summary');
 | 
				
			||||||
 | 
				
			|||||||
@ -3,6 +3,7 @@
 | 
				
			|||||||
namespace Crater\Http\Requests;
 | 
					namespace Crater\Http\Requests;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use Illuminate\Foundation\Http\FormRequest;
 | 
					use Illuminate\Foundation\Http\FormRequest;
 | 
				
			||||||
 | 
					use Illuminate\Support\Str;
 | 
				
			||||||
use Illuminate\Validation\Rule;
 | 
					use Illuminate\Validation\Rule;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class CompaniesRequest extends FormRequest
 | 
					class CompaniesRequest extends FormRequest
 | 
				
			||||||
@ -33,10 +34,6 @@ class CompaniesRequest extends FormRequest
 | 
				
			|||||||
            'currency' => [
 | 
					            'currency' => [
 | 
				
			||||||
                'required'
 | 
					                'required'
 | 
				
			||||||
            ],
 | 
					            ],
 | 
				
			||||||
            'slug' => [
 | 
					 | 
				
			||||||
                'required',
 | 
					 | 
				
			||||||
                Rule::unique('companies')
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            'address.name' => [
 | 
					            'address.name' => [
 | 
				
			||||||
                'nullable',
 | 
					                'nullable',
 | 
				
			||||||
            ],
 | 
					            ],
 | 
				
			||||||
@ -71,11 +68,11 @@ class CompaniesRequest extends FormRequest
 | 
				
			|||||||
    {
 | 
					    {
 | 
				
			||||||
        return collect($this->validated())
 | 
					        return collect($this->validated())
 | 
				
			||||||
            ->only([
 | 
					            ->only([
 | 
				
			||||||
                'name',
 | 
					                'name'
 | 
				
			||||||
                'slug'
 | 
					 | 
				
			||||||
            ])
 | 
					            ])
 | 
				
			||||||
            ->merge([
 | 
					            ->merge([
 | 
				
			||||||
                'owner_id' => $this->user()->id
 | 
					                'owner_id' => $this->user()->id,
 | 
				
			||||||
 | 
					                'slug' => Str::slug($this->name)
 | 
				
			||||||
            ])
 | 
					            ])
 | 
				
			||||||
            ->toArray();
 | 
					            ->toArray();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
				
			|||||||
@ -30,8 +30,7 @@ class CompanyRequest extends FormRequest
 | 
				
			|||||||
                Rule::unique('companies')->ignore($this->header('company'), 'id'),
 | 
					                Rule::unique('companies')->ignore($this->header('company'), 'id'),
 | 
				
			||||||
            ],
 | 
					            ],
 | 
				
			||||||
            'slug' => [
 | 
					            'slug' => [
 | 
				
			||||||
                'required',
 | 
					                'nullable'
 | 
				
			||||||
                Rule::unique('companies')->ignore($this->header('company'), 'id'),
 | 
					 | 
				
			||||||
            ],
 | 
					            ],
 | 
				
			||||||
            'address.country_id' => [
 | 
					            'address.country_id' => [
 | 
				
			||||||
                'required',
 | 
					                'required',
 | 
				
			||||||
 | 
				
			|||||||
@ -23,7 +23,7 @@ class EstimateResource extends JsonResource
 | 
				
			|||||||
            'reference_number' => $this->reference_number,
 | 
					            'reference_number' => $this->reference_number,
 | 
				
			||||||
            'tax_per_item' => $this->tax_per_item,
 | 
					            'tax_per_item' => $this->tax_per_item,
 | 
				
			||||||
            'discount_per_item' => $this->discount_per_item,
 | 
					            'discount_per_item' => $this->discount_per_item,
 | 
				
			||||||
            'notes' => $this->notes,
 | 
					            'notes' => $this->getNotes(),
 | 
				
			||||||
            'discount' => $this->discount,
 | 
					            'discount' => $this->discount,
 | 
				
			||||||
            'discount_type' => $this->discount_type,
 | 
					            'discount_type' => $this->discount_type,
 | 
				
			||||||
            'discount_val' => $this->discount_val,
 | 
					            'discount_val' => $this->discount_val,
 | 
				
			||||||
 | 
				
			|||||||
@ -18,7 +18,7 @@ class PaymentResource extends JsonResource
 | 
				
			|||||||
            'id' => $this->id,
 | 
					            'id' => $this->id,
 | 
				
			||||||
            'payment_number' => $this->payment_number,
 | 
					            'payment_number' => $this->payment_number,
 | 
				
			||||||
            'payment_date' => $this->payment_date,
 | 
					            'payment_date' => $this->payment_date,
 | 
				
			||||||
            'notes' => $this->notes,
 | 
					            'notes' => $this->getNotes(),
 | 
				
			||||||
            'amount' => $this->amount,
 | 
					            'amount' => $this->amount,
 | 
				
			||||||
            'unique_hash' => $this->unique_hash,
 | 
					            'unique_hash' => $this->unique_hash,
 | 
				
			||||||
            'invoice_id' => $this->invoice_id,
 | 
					            'invoice_id' => $this->invoice_id,
 | 
				
			||||||
 | 
				
			|||||||
@ -217,7 +217,7 @@ class Company extends Model implements HasMedia
 | 
				
			|||||||
            'estimate_billing_address_format' => $billingAddressFormat,
 | 
					            'estimate_billing_address_format' => $billingAddressFormat,
 | 
				
			||||||
            'payment_company_address_format' => $companyAddressFormat,
 | 
					            'payment_company_address_format' => $companyAddressFormat,
 | 
				
			||||||
            'payment_from_customer_address_format' => $paymentFromCustomerAddress,
 | 
					            'payment_from_customer_address_format' => $paymentFromCustomerAddress,
 | 
				
			||||||
            'currency' => request()->currency ?? 1,
 | 
					            'currency' => request()->currency ?? 13,
 | 
				
			||||||
            'time_zone' => 'Asia/Kolkata',
 | 
					            'time_zone' => 'Asia/Kolkata',
 | 
				
			||||||
            'language' => 'en',
 | 
					            'language' => 'en',
 | 
				
			||||||
            'fiscal_year' => '1-12',
 | 
					            'fiscal_year' => '1-12',
 | 
				
			||||||
 | 
				
			|||||||
@ -483,8 +483,7 @@ class Estimate extends Model implements HasMedia
 | 
				
			|||||||
            '{ESTIMATE_DATE}' => $this->formattedEstimateDate,
 | 
					            '{ESTIMATE_DATE}' => $this->formattedEstimateDate,
 | 
				
			||||||
            '{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate,
 | 
					            '{ESTIMATE_EXPIRY_DATE}' => $this->formattedExpiryDate,
 | 
				
			||||||
            '{ESTIMATE_NUMBER}' => $this->estimate_number,
 | 
					            '{ESTIMATE_NUMBER}' => $this->estimate_number,
 | 
				
			||||||
            '{PDF_LINK}' => $this->estimatePdfUrl,
 | 
					            '{ESTIMATE_REF_NUMBER}' => $this->reference_number,
 | 
				
			||||||
            '{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
 | 
					 | 
				
			||||||
        ];
 | 
					        ];
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -240,7 +240,7 @@ class Expense extends Model implements HasMedia
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if ($request->hasFile('attachment_receipt')) {
 | 
					        if ($request->hasFile('attachment_receipt')) {
 | 
				
			||||||
            $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
 | 
					            $expense->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if ($request->customFields) {
 | 
					        if ($request->customFields) {
 | 
				
			||||||
@ -262,12 +262,12 @@ class Expense extends Model implements HasMedia
 | 
				
			|||||||
            ExchangeRateLog::addExchangeRateLog($this);
 | 
					            ExchangeRateLog::addExchangeRateLog($this);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (isset($request->is_attachment_receipt_removed) && $request->is_attachment_receipt_removed == "true") {
 | 
					        if (isset($request->is_attachment_receipt_removed) && (bool) $request->is_attachment_receipt_removed) {
 | 
				
			||||||
            $this->clearMediaCollection('receipts');
 | 
					            $this->clearMediaCollection('receipts');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if ($request->hasFile('attachment_receipt')) {
 | 
					        if ($request->hasFile('attachment_receipt')) {
 | 
				
			||||||
            $this->clearMediaCollection('receipts');
 | 
					            $this->clearMediaCollection('receipts');
 | 
				
			||||||
            $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts', 'local');
 | 
					            $this->addMediaFromRequest('attachment_receipt')->toMediaCollection('receipts');
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if ($request->customFields) {
 | 
					        if ($request->customFields) {
 | 
				
			||||||
 | 
				
			|||||||
@ -187,6 +187,16 @@ class Invoice extends Model implements HasMedia
 | 
				
			|||||||
        return Carbon::parse($this->invoice_date)->format($dateFormat);
 | 
					        return Carbon::parse($this->invoice_date)->format($dateFormat);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public function scopeWhereStatus($query, $status)
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        return $query->where('invoices.status', $status);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    public function scopeWherePaidStatus($query, $status)
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        return $query->where('invoices.paid_status', $status);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public function scopeWhereDueStatus($query, $status)
 | 
					    public function scopeWhereDueStatus($query, $status)
 | 
				
			||||||
    {
 | 
					    {
 | 
				
			||||||
        return $query->whereIn('invoices.paid_status', [
 | 
					        return $query->whereIn('invoices.paid_status', [
 | 
				
			||||||
@ -224,40 +234,6 @@ class Invoice extends Model implements HasMedia
 | 
				
			|||||||
        $query->orderBy($orderByField, $orderBy);
 | 
					        $query->orderBy($orderByField, $orderBy);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    public function scopeWhereStatus($query, $status)
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
        return $query->where('invoices.status', $status);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    public function scopeWherePaidStatus($query, $status)
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
        return $query->where('invoices.paid_status', $status);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    public function scopeWhereTabFilters($query, $status)
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
        if ($status == "DRAFT") {
 | 
					 | 
				
			||||||
            return $query->where('invoices.status', $status);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if ($status == "SENT") {
 | 
					 | 
				
			||||||
            return $query->whereIn('invoices.status', [
 | 
					 | 
				
			||||||
                self::STATUS_SENT,
 | 
					 | 
				
			||||||
                self::STATUS_VIEWED,
 | 
					 | 
				
			||||||
                self::STATUS_COMPLETED
 | 
					 | 
				
			||||||
            ]);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if ($status == 'DUE') {
 | 
					 | 
				
			||||||
            return $query->whereIn('invoices.paid_status', [
 | 
					 | 
				
			||||||
                self::STATUS_UNPAID,
 | 
					 | 
				
			||||||
                self::STATUS_PARTIALLY_PAID,
 | 
					 | 
				
			||||||
            ]);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return ;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    public function scopeApplyFilters($query, array $filters)
 | 
					    public function scopeApplyFilters($query, array $filters)
 | 
				
			||||||
    {
 | 
					    {
 | 
				
			||||||
        $filters = collect($filters);
 | 
					        $filters = collect($filters);
 | 
				
			||||||
@ -273,11 +249,17 @@ class Invoice extends Model implements HasMedia
 | 
				
			|||||||
                $filters->get('status') == self::STATUS_PAID
 | 
					                $filters->get('status') == self::STATUS_PAID
 | 
				
			||||||
            ) {
 | 
					            ) {
 | 
				
			||||||
                $query->wherePaidStatus($filters->get('status'));
 | 
					                $query->wherePaidStatus($filters->get('status'));
 | 
				
			||||||
 | 
					            } elseif ($filters->get('status') == 'DUE') {
 | 
				
			||||||
 | 
					                $query->whereDueStatus($filters->get('status'));
 | 
				
			||||||
            } else {
 | 
					            } else {
 | 
				
			||||||
                $query->whereStatus($filters->get('status'));
 | 
					                $query->whereStatus($filters->get('status'));
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if ($filters->get('paid_status')) {
 | 
				
			||||||
 | 
					            $query->wherePaidStatus($filters->get('status'));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if ($filters->get('invoice_id')) {
 | 
					        if ($filters->get('invoice_id')) {
 | 
				
			||||||
            $query->whereInvoice($filters->get('invoice_id'));
 | 
					            $query->whereInvoice($filters->get('invoice_id'));
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
@ -669,9 +651,7 @@ class Invoice extends Model implements HasMedia
 | 
				
			|||||||
            '{INVOICE_DATE}' => $this->formattedInvoiceDate,
 | 
					            '{INVOICE_DATE}' => $this->formattedInvoiceDate,
 | 
				
			||||||
            '{INVOICE_DUE_DATE}' => $this->formattedDueDate,
 | 
					            '{INVOICE_DUE_DATE}' => $this->formattedDueDate,
 | 
				
			||||||
            '{INVOICE_NUMBER}' => $this->invoice_number,
 | 
					            '{INVOICE_NUMBER}' => $this->invoice_number,
 | 
				
			||||||
            '{PDF_LINK}' => $this->invoicePdfUrl,
 | 
					            '{INVOICE_REF_NUMBER}' => $this->reference_number,
 | 
				
			||||||
            '{DUE_AMOUNT}' => format_money_pdf($this->due_amount, $this->customer->currency),
 | 
					 | 
				
			||||||
            '{TOTAL_AMOUNT}' => format_money_pdf($this->total, $this->customer->currency)
 | 
					 | 
				
			||||||
        ];
 | 
					        ];
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -435,8 +435,7 @@ class Payment extends Model implements HasMedia
 | 
				
			|||||||
            '{PAYMENT_DATE}' => $this->formattedPaymentDate,
 | 
					            '{PAYMENT_DATE}' => $this->formattedPaymentDate,
 | 
				
			||||||
            '{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null,
 | 
					            '{PAYMENT_MODE}' => $this->paymentMethod ? $this->paymentMethod->name : null,
 | 
				
			||||||
            '{PAYMENT_NUMBER}' => $this->payment_number,
 | 
					            '{PAYMENT_NUMBER}' => $this->payment_number,
 | 
				
			||||||
            '{PDF_LINK}' => $this->paymentPdfUrl,
 | 
					            '{PAYMENT_AMOUNT}' => $this->reference_number,
 | 
				
			||||||
            '{PAYMENT_AMOUNT}' => format_money_pdf($this->amount, $this->customer->currency)
 | 
					 | 
				
			||||||
        ];
 | 
					        ];
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -38,7 +38,7 @@
 | 
				
			|||||||
    "barryvdh/laravel-ide-helper": "^2.6",
 | 
					    "barryvdh/laravel-ide-helper": "^2.6",
 | 
				
			||||||
    "beyondcode/laravel-dump-server": "^1.0",
 | 
					    "beyondcode/laravel-dump-server": "^1.0",
 | 
				
			||||||
    "facade/ignition": "^2.3.6",
 | 
					    "facade/ignition": "^2.3.6",
 | 
				
			||||||
    "friendsofphp/php-cs-fixer": "^3.8",
 | 
					    "friendsofphp/php-cs-fixer": "^3.0",
 | 
				
			||||||
    "fakerphp/faker": "^1.9.1",
 | 
					    "fakerphp/faker": "^1.9.1",
 | 
				
			||||||
    "mockery/mockery": "^1.3.1",
 | 
					    "mockery/mockery": "^1.3.1",
 | 
				
			||||||
    "nunomaduro/collision": "^5.0",
 | 
					    "nunomaduro/collision": "^5.0",
 | 
				
			||||||
@ -81,10 +81,7 @@
 | 
				
			|||||||
  "config": {
 | 
					  "config": {
 | 
				
			||||||
    "optimize-autoloader": true,
 | 
					    "optimize-autoloader": true,
 | 
				
			||||||
    "preferred-install": "dist",
 | 
					    "preferred-install": "dist",
 | 
				
			||||||
    "sort-packages": true,
 | 
					    "sort-packages": true
 | 
				
			||||||
    "allow-plugins": {
 | 
					 | 
				
			||||||
      "pestphp/pest-plugin": true
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "extra": {
 | 
					  "extra": {
 | 
				
			||||||
    "laravel": {
 | 
					    "laravel": {
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										2345
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2345
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										483
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										483
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							@ -65,6 +65,13 @@
 | 
				
			|||||||
        "to-fast-properties": "^2.0.0"
 | 
					        "to-fast-properties": "^2.0.0"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@esbuild/linux-loong64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-bZBrLAIX1kpWelV0XemxBZllyRmM6vgFQQG2GdNb+r3Fkp0FOh1NJSvekXDs7jq70k4euu1cryLMfU+mTXlEpw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@eslint/eslintrc": {
 | 
					    "@eslint/eslintrc": {
 | 
				
			||||||
      "version": "0.4.3",
 | 
					      "version": "0.4.3",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.3.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.4.3.tgz",
 | 
				
			||||||
@ -214,6 +221,23 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.9.3.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.9.3.tgz",
 | 
				
			||||||
      "integrity": "sha512-xDu17cEfh7Kid/d95kB6tZsLOmSWKCZKtprnhVepjsSaCij+lM3mItSJDuuHDMbCWTh8Ejmebwb+KONcCJ0eXQ=="
 | 
					      "integrity": "sha512-xDu17cEfh7Kid/d95kB6tZsLOmSWKCZKtprnhVepjsSaCij+lM3mItSJDuuHDMbCWTh8Ejmebwb+KONcCJ0eXQ=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@rvxlab/tailwind-plugin-ios-full-height": {
 | 
				
			||||||
 | 
					      "version": "1.1.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@rvxlab/tailwind-plugin-ios-full-height/-/tailwind-plugin-ios-full-height-1.1.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-jPIxXn0raN/YTk8nXesqM+JbS2WWd5XaUk/MbaAgVDDPyYtsPfeN3B26xIhSa2oE2+JB66tegPUMSOmixzroXg==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "@stripe/stripe-js": {
 | 
				
			||||||
 | 
					      "version": "1.35.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@stripe/stripe-js/-/stripe-js-1.35.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-UIuzpbJqgXCTvJhY/aZYvBtaKdMfQgnIv6kkLlfRJ9smZcC4zoPvq3j7k9wobYI+idHAWP4BRiPnqA8lvzJCtg=="
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "@tailwindcss/aspect-ratio": {
 | 
				
			||||||
 | 
					      "version": "0.4.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@tailwindcss/aspect-ratio/-/aspect-ratio-0.4.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-WJu0I4PpqNPuutpaA9zDUq2JXR+lorZ7PbLcKNLmb6GL9/HLfC7w3CRsMhJF4BbYd/lkY6CfXOvkYpuGnZfkpQ==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@tailwindcss/forms": {
 | 
					    "@tailwindcss/forms": {
 | 
				
			||||||
      "version": "0.4.0",
 | 
					      "version": "0.4.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@tailwindcss/forms/-/forms-0.4.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@tailwindcss/forms/-/forms-0.4.0.tgz",
 | 
				
			||||||
@ -223,6 +247,22 @@
 | 
				
			|||||||
        "mini-svg-data-uri": "^1.2.3"
 | 
					        "mini-svg-data-uri": "^1.2.3"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@tailwindcss/line-clamp": {
 | 
				
			||||||
 | 
					      "version": "0.3.1",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@tailwindcss/line-clamp/-/line-clamp-0.3.1.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-pNr0T8LAc3TUx/gxCfQZRe9NB2dPEo/cedPHzUGIPxqDMhgjwNm6jYxww4W5l0zAsAddxr+XfZcqttGiFDgrGg=="
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "@tailwindcss/typography": {
 | 
				
			||||||
 | 
					      "version": "0.5.4",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@tailwindcss/typography/-/typography-0.5.4.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-QEdg40EmGvE7kKoDei8zr5sf4D1pIayHj4R31bH3lX8x2BtTiR+jNejYPOkhbmy3DXgkMF9jC8xqNiGFAuL9Sg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "lodash.castarray": "^4.4.0",
 | 
				
			||||||
 | 
					        "lodash.isplainobject": "^4.0.6",
 | 
				
			||||||
 | 
					        "lodash.merge": "^4.6.2"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@tiptap/core": {
 | 
					    "@tiptap/core": {
 | 
				
			||||||
      "version": "2.0.0-beta.99",
 | 
					      "version": "2.0.0-beta.99",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@tiptap/core/-/core-2.0.0-beta.99.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@tiptap/core/-/core-2.0.0-beta.99.tgz",
 | 
				
			||||||
@ -386,6 +426,11 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/@tiptap/extension-text/-/extension-text-2.0.0-beta.13.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@tiptap/extension-text/-/extension-text-2.0.0-beta.13.tgz",
 | 
				
			||||||
      "integrity": "sha512-0EtAwuRldCAoFaL/iXgkRepEeOd55rPg5N4FQUN1xTwZT7PDofukP0DG/2jff/Uj17x4uTaJAa9qlFWuNnDvjw=="
 | 
					      "integrity": "sha512-0EtAwuRldCAoFaL/iXgkRepEeOd55rPg5N4FQUN1xTwZT7PDofukP0DG/2jff/Uj17x4uTaJAa9qlFWuNnDvjw=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@tiptap/extension-text-align": {
 | 
				
			||||||
 | 
					      "version": "2.0.0-beta.31",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@tiptap/extension-text-align/-/extension-text-align-2.0.0-beta.31.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-gSJqi57piiMPc2r6WEkXv7ZgQIogigsRUhmlnZC/7s3zzOvjXrexWnV0Ctt/9A7BKcM7OHMykpZyoewvk6QRTw=="
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@tiptap/starter-kit": {
 | 
					    "@tiptap/starter-kit": {
 | 
				
			||||||
      "version": "2.0.0-beta.97",
 | 
					      "version": "2.0.0-beta.97",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@tiptap/starter-kit/-/starter-kit-2.0.0-beta.97.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@tiptap/starter-kit/-/starter-kit-2.0.0-beta.97.tgz",
 | 
				
			||||||
@ -537,6 +582,12 @@
 | 
				
			|||||||
        "@types/prosemirror-transform": "*"
 | 
					        "@types/prosemirror-transform": "*"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@vitejs/plugin-vue": {
 | 
				
			||||||
 | 
					      "version": "1.10.2",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vitejs/plugin-vue/-/plugin-vue-1.10.2.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-/QJ0Z9qfhAFtKRY+r57ziY4BSbGUTGsPRMpB/Ron3QPwBZM4OZAZHdTa4a8PafCwU5DTatXG8TMDoP8z+oDqJw==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@vue/compiler-core": {
 | 
					    "@vue/compiler-core": {
 | 
				
			||||||
      "version": "3.2.4",
 | 
					      "version": "3.2.4",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.2.4.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.2.4.tgz",
 | 
				
			||||||
@ -558,6 +609,70 @@
 | 
				
			|||||||
        "@vue/shared": "3.2.4"
 | 
					        "@vue/shared": "3.2.4"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@vue/compiler-sfc": {
 | 
				
			||||||
 | 
					      "version": "3.2.38",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vue/compiler-sfc/-/compiler-sfc-3.2.38.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-KZjrW32KloMYtTcHAFuw3CqsyWc5X6seb8KbkANSWt3Cz9p2qA8c1GJpSkksFP9ABb6an0FLCFl46ZFXx3kKpg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@babel/parser": "^7.16.4",
 | 
				
			||||||
 | 
					        "@vue/compiler-core": "3.2.38",
 | 
				
			||||||
 | 
					        "@vue/compiler-dom": "3.2.38",
 | 
				
			||||||
 | 
					        "@vue/compiler-ssr": "3.2.38",
 | 
				
			||||||
 | 
					        "@vue/reactivity-transform": "3.2.38",
 | 
				
			||||||
 | 
					        "@vue/shared": "3.2.38",
 | 
				
			||||||
 | 
					        "estree-walker": "^2.0.2",
 | 
				
			||||||
 | 
					        "magic-string": "^0.25.7",
 | 
				
			||||||
 | 
					        "postcss": "^8.1.10",
 | 
				
			||||||
 | 
					        "source-map": "^0.6.1"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "@babel/parser": {
 | 
				
			||||||
 | 
					          "version": "7.18.13",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.18.13.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-dgXcIfMuQ0kgzLB2b9tRZs7TTFFaGM2AbtA4fJgUUYukzGH4jwsS7hzQHEGs67jdehpm22vkgKwvbU+aEflgwg==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/compiler-core": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-/FsvnSu7Z+lkd/8KXMa4yYNUiqQrI22135gfsQYVGuh5tqEgOB0XqrUdb/KnCLa5+TmQLPwvyUnKMyCpu+SX3Q==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "@babel/parser": "^7.16.4",
 | 
				
			||||||
 | 
					            "@vue/shared": "3.2.38",
 | 
				
			||||||
 | 
					            "estree-walker": "^2.0.2",
 | 
				
			||||||
 | 
					            "source-map": "^0.6.1"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/compiler-dom": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/compiler-dom/-/compiler-dom-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-zqX4FgUbw56kzHlgYuEEJR8mefFiiyR3u96498+zWPsLeh1WKvgIReoNE+U7gG8bCUdvsrJ0JRmev0Ky6n2O0g==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "@vue/compiler-core": "3.2.38",
 | 
				
			||||||
 | 
					            "@vue/shared": "3.2.38"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/compiler-ssr": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/compiler-ssr/-/compiler-ssr-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-bm9jOeyv1H3UskNm4S6IfueKjUNFmi2kRweFIGnqaGkkRePjwEcfCVqyS3roe7HvF4ugsEkhf4+kIvDhip6XzQ==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "@vue/compiler-dom": "3.2.38",
 | 
				
			||||||
 | 
					            "@vue/shared": "3.2.38"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/shared": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-dTyhTIRmGXBjxJE+skC8tTWCGLCVc4wQgRRLt8+O9p5ewBAjoBwtCAkLPrtToSr1xltoe3st21Pv953aOZ7alg==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@vue/compiler-ssr": {
 | 
					    "@vue/compiler-ssr": {
 | 
				
			||||||
      "version": "3.2.19",
 | 
					      "version": "3.2.19",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@vue/compiler-ssr/-/compiler-ssr-3.2.19.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@vue/compiler-ssr/-/compiler-ssr-3.2.19.tgz",
 | 
				
			||||||
@ -607,6 +722,45 @@
 | 
				
			|||||||
        "@vue/shared": "3.2.4"
 | 
					        "@vue/shared": "3.2.4"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@vue/reactivity-transform": {
 | 
				
			||||||
 | 
					      "version": "3.2.38",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vue/reactivity-transform/-/reactivity-transform-3.2.38.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-3SD3Jmi1yXrDwiNJqQ6fs1x61WsDLqVk4NyKVz78mkaIRh6d3IqtRnptgRfXn+Fzf+m6B1KxBYWq1APj6h4qeA==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@babel/parser": "^7.16.4",
 | 
				
			||||||
 | 
					        "@vue/compiler-core": "3.2.38",
 | 
				
			||||||
 | 
					        "@vue/shared": "3.2.38",
 | 
				
			||||||
 | 
					        "estree-walker": "^2.0.2",
 | 
				
			||||||
 | 
					        "magic-string": "^0.25.7"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "@babel/parser": {
 | 
				
			||||||
 | 
					          "version": "7.18.13",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.18.13.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-dgXcIfMuQ0kgzLB2b9tRZs7TTFFaGM2AbtA4fJgUUYukzGH4jwsS7hzQHEGs67jdehpm22vkgKwvbU+aEflgwg==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/compiler-core": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-/FsvnSu7Z+lkd/8KXMa4yYNUiqQrI22135gfsQYVGuh5tqEgOB0XqrUdb/KnCLa5+TmQLPwvyUnKMyCpu+SX3Q==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "@babel/parser": "^7.16.4",
 | 
				
			||||||
 | 
					            "@vue/shared": "3.2.38",
 | 
				
			||||||
 | 
					            "estree-walker": "^2.0.2",
 | 
				
			||||||
 | 
					            "source-map": "^0.6.1"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "@vue/shared": {
 | 
				
			||||||
 | 
					          "version": "3.2.38",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.2.38.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-dTyhTIRmGXBjxJE+skC8tTWCGLCVc4wQgRRLt8+O9p5ewBAjoBwtCAkLPrtToSr1xltoe3st21Pv953aOZ7alg==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@vue/ref-transform": {
 | 
					    "@vue/ref-transform": {
 | 
				
			||||||
      "version": "3.2.19",
 | 
					      "version": "3.2.19",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@vue/ref-transform/-/ref-transform-3.2.19.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@vue/ref-transform/-/ref-transform-3.2.19.tgz",
 | 
				
			||||||
@ -677,6 +831,44 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.2.4.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.2.4.tgz",
 | 
				
			||||||
      "integrity": "sha512-j2j1MRmjalVKr3YBTxl/BClSIc8UQ8NnPpLYclxerK65JIowI4O7n8O8lElveEtEoHxy1d7BelPUDI0Q4bumqg=="
 | 
					      "integrity": "sha512-j2j1MRmjalVKr3YBTxl/BClSIc8UQ8NnPpLYclxerK65JIowI4O7n8O8lElveEtEoHxy1d7BelPUDI0Q4bumqg=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "@vuelidate/components": {
 | 
				
			||||||
 | 
					      "version": "1.2.1",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vuelidate/components/-/components-1.2.1.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-xaFGcKVQbST0l7yQufuLAbRUwC/5SR4Z8+s7fwF3B1BtQQwlttftAZg1lTm9I30EYsBdslk38XC4z3sDC8Nm4w==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@vuelidate/core": "^2.0.0-alpha.44"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "@vuelidate/core": {
 | 
				
			||||||
 | 
					      "version": "2.0.0-alpha.44",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vuelidate/core/-/core-2.0.0-alpha.44.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-3DlCe3E0RRXbB+OfPacUetKhLmXzmnjeHkzjnbkc03p06mKm6h9pXR5pd6Mv4s4tus4sieuKDb2YWNmKK6rQeA==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "vue-demi": "^0.13.4"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "vue-demi": {
 | 
				
			||||||
 | 
					          "version": "0.13.11",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/vue-demi/-/vue-demi-0.13.11.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-IR8HoEEGM65YY3ZJYAjMlKygDQn25D5ajNFNoKh9RSDMQtlzCxtfQjdQgv9jjK+m3377SsJXY8ysq8kLCZL25A=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "@vuelidate/validators": {
 | 
				
			||||||
 | 
					      "version": "2.0.0-alpha.31",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/@vuelidate/validators/-/validators-2.0.0-alpha.31.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-+MFA9nZ7Y9zCpq383/voPDk/hiAmu6KqiJJhLOYB/FmrUPVoyKnuKnI9Bwiq8ok9GZlVkI8BnIrKPKGj9QpwiQ==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "vue-demi": "^0.13.4"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "vue-demi": {
 | 
				
			||||||
 | 
					          "version": "0.13.11",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/vue-demi/-/vue-demi-0.13.11.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-IR8HoEEGM65YY3ZJYAjMlKygDQn25D5ajNFNoKh9RSDMQtlzCxtfQjdQgv9jjK+m3377SsJXY8ysq8kLCZL25A=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "@vueuse/core": {
 | 
					    "@vueuse/core": {
 | 
				
			||||||
      "version": "6.0.0",
 | 
					      "version": "6.0.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/@vueuse/core/-/core-6.0.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/@vueuse/core/-/core-6.0.0.tgz",
 | 
				
			||||||
@ -1240,6 +1432,175 @@
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-Cy9llcy8DvET5uznocPyqL3BFRrFXSVqbgpMJ9Wz8oVjZlh/zUSNbPRbov0VX7VxN2JH1Oa0uNxZ7eLRb62pJA==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@esbuild/linux-loong64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-android-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-android-arm64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-darwin-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-darwin-arm64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-freebsd-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-freebsd-arm64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-32": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-arm": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-arm64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-mips64le": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-ppc64le": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-riscv64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-linux-s390x": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-netbsd-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-openbsd-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-sunos-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-windows-32": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-windows-64": "0.14.54",
 | 
				
			||||||
 | 
					        "esbuild-windows-arm64": "0.14.54"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-android-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-android-64/-/esbuild-android-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-Tz2++Aqqz0rJ7kYBfz+iqyE3QMycD4vk7LBRyWaAVFgFtQ/O8EJOnVmTOiDWYZ/uYzB4kvP+bqejYdVKzE5lAQ==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-android-arm64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-android-arm64/-/esbuild-android-arm64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-F9E+/QDi9sSkLaClO8SOV6etqPd+5DgJje1F9lOWoNncDdOBL2YF59IhsWATSt0TLZbYCf3pNlTHvVV5VfHdvg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-darwin-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-darwin-64/-/esbuild-darwin-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-jtdKWV3nBviOd5v4hOpkVmpxsBy90CGzebpbO9beiqUYVMBtSc0AL9zGftFuBon7PNDcdvNCEuQqw2x0wP9yug==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-darwin-arm64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-darwin-arm64/-/esbuild-darwin-arm64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-OPafJHD2oUPyvJMrsCvDGkRrVCar5aVyHfWGQzY1dWnzErjrDuSETxwA2HSsyg2jORLY8yBfzc1MIpUkXlctmw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-freebsd-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-freebsd-64/-/esbuild-freebsd-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-OKwd4gmwHqOTp4mOGZKe/XUlbDJ4Q9TjX0hMPIDBUWWu/kwhBAudJdBoxnjNf9ocIB6GN6CPowYpR/hRCbSYAg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-freebsd-arm64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-freebsd-arm64/-/esbuild-freebsd-arm64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-sFwueGr7OvIFiQT6WeG0jRLjkjdqWWSrfbVwZp8iMP+8UHEHRBvlaxL6IuKNDwAozNUmbb8nIMXa7oAOARGs1Q==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-32": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-32/-/esbuild-linux-32-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-1ZuY+JDI//WmklKlBgJnglpUL1owm2OX+8E1syCD6UAxcMM/XoWd76OHSjl/0MR0LisSAXDqgjT3uJqT67O3qw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-64/-/esbuild-linux-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-EgjAgH5HwTbtNsTqQOXWApBaPVdDn7XcK+/PtJwZLT1UmpLoznPd8c5CxqsH2dQK3j05YsB3L17T8vE7cp4cCg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-arm": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-arm/-/esbuild-linux-arm-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-qqz/SjemQhVMTnvcLGoLOdFpCYbz4v4fUo+TfsWG+1aOu70/80RV6bgNpR2JCrppV2moUQkww+6bWxXRL9YMGw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-arm64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-arm64/-/esbuild-linux-arm64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-WL71L+0Rwv+Gv/HTmxTEmpv0UgmxYa5ftZILVi2QmZBgX3q7+tDeOQNqGtdXSdsL8TQi1vIaVFHUPDe0O0kdig==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-mips64le": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-mips64le/-/esbuild-linux-mips64le-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-qTHGQB8D1etd0u1+sB6p0ikLKRVuCWhYQhAHRPkO+OF3I/iSlTKNNS0Lh2Oc0g0UFGguaFZZiPJdJey3AGpAlw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-ppc64le": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-ppc64le/-/esbuild-linux-ppc64le-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-j3OMlzHiqwZBDPRCDFKcx595XVfOfOnv68Ax3U4UKZ3MTYQB5Yz3X1mn5GnodEVYzhtZgxEBidLWeIs8FDSfrQ==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-riscv64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-riscv64/-/esbuild-linux-riscv64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-y7Vt7Wl9dkOGZjxQZnDAqqn+XOqFD7IMWiewY5SPlNlzMX39ocPQlOaoxvT4FllA5viyV26/QzHtvTjVNOxHZg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-linux-s390x": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-linux-s390x/-/esbuild-linux-s390x-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-zaHpW9dziAsi7lRcyV4r8dhfG1qBidQWUXweUjnw+lliChJqQr+6XD71K41oEIC3Mx1KStovEmlzm+MkGZHnHA==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-netbsd-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-netbsd-64/-/esbuild-netbsd-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-PR01lmIMnfJTgeU9VJTDY9ZerDWVFIUzAtJuDHwwceppW7cQWjBBqP48NdeRtoP04/AtO9a7w3viI+PIDr6d+w==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-openbsd-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-openbsd-64/-/esbuild-openbsd-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-Qyk7ikT2o7Wu76UsvvDS5q0amJvmRzDyVlL0qf5VLsLchjCa1+IAvd8kTBgUxD7VBUUVgItLkk609ZHUc1oCaw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-sunos-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-sunos-64/-/esbuild-sunos-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-28GZ24KmMSeKi5ueWzMcco6EBHStL3B6ubM7M51RmPwXQGLe0teBGJocmWhgwccA1GeFXqxzILIxXpHbl9Q/Kw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-windows-32": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-windows-32/-/esbuild-windows-32-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-T+rdZW19ql9MjS7pixmZYVObd9G7kcaZo+sETqNH4RCkuuYSuv9AGHUVnPoP9hhuE1WM1ZimHz1CIBHBboLU7w==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-windows-64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-windows-64/-/esbuild-windows-64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-AoHTRBUuYwXtZhjXZbA1pGfTo8cJo3vZIcWGLiUcTNgHpJJMC1rVA44ZereBHMJtotyN71S8Qw0npiCIkW96cQ==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "esbuild-windows-arm64": {
 | 
				
			||||||
 | 
					      "version": "0.14.54",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/esbuild-windows-arm64/-/esbuild-windows-arm64-0.14.54.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-M0kuUvXhot1zOISQGXwWn6YtS+Y/1RT9WrVIOywZnJHo3jCDyewAc79aKNQWFCQm+xNHVTq9h8dZKvygoXQQRg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "optional": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "escalade": {
 | 
					    "escalade": {
 | 
				
			||||||
      "version": "3.1.1",
 | 
					      "version": "3.1.1",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz",
 | 
				
			||||||
@ -1947,12 +2308,24 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz",
 | 
				
			||||||
      "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw=="
 | 
					      "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "lodash.castarray": {
 | 
				
			||||||
 | 
					      "version": "4.4.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/lodash.castarray/-/lodash.castarray-4.4.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-aVx8ztPv7/2ULbArGJ2Y42bG1mEQ5mGjpdvrbJcJFU3TbYybe+QlLS4pst9zV52ymy2in1KpFPiZnAOATxD4+Q==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "lodash.clonedeep": {
 | 
					    "lodash.clonedeep": {
 | 
				
			||||||
      "version": "4.5.0",
 | 
					      "version": "4.5.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz",
 | 
				
			||||||
      "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=",
 | 
					      "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=",
 | 
				
			||||||
      "dev": true
 | 
					      "dev": true
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "lodash.isplainobject": {
 | 
				
			||||||
 | 
					      "version": "4.0.6",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "lodash.merge": {
 | 
					    "lodash.merge": {
 | 
				
			||||||
      "version": "4.6.2",
 | 
					      "version": "4.6.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
 | 
				
			||||||
@ -2205,6 +2578,22 @@
 | 
				
			|||||||
      "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==",
 | 
					      "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==",
 | 
				
			||||||
      "dev": true
 | 
					      "dev": true
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "pinia": {
 | 
				
			||||||
 | 
					      "version": "2.0.21",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/pinia/-/pinia-2.0.21.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-6ol04PtL29O0Z6JHI47O3JUSoyOJ7Og0rstXrHVMZSP4zAldsQBXJCNF0i/H7m8vp/Hjd/CSmuPl7C5QAwpeWQ==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@vue/devtools-api": "^6.2.1",
 | 
				
			||||||
 | 
					        "vue-demi": "*"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "@vue/devtools-api": {
 | 
				
			||||||
 | 
					          "version": "6.2.1",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-6.2.1.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-OEgAMeQXvCoJ+1x8WyQuVZzFo0wcyCmUR3baRVLmKBo1LmYZWMlRiXlux5jd0fqVJu6PfDbOrZItVqUEzLobeQ=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "postcss": {
 | 
					    "postcss": {
 | 
				
			||||||
      "version": "8.4.5",
 | 
					      "version": "8.4.5",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.5.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.5.tgz",
 | 
				
			||||||
@ -2508,6 +2897,15 @@
 | 
				
			|||||||
        "glob": "^7.1.3"
 | 
					        "glob": "^7.1.3"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "rollup": {
 | 
				
			||||||
 | 
					      "version": "2.78.1",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.78.1.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-VeeCgtGi4P+o9hIg+xz4qQpRl6R401LWEXBmxYKOV4zlF82lyhgh2hTZnheFUbANE8l2A41F458iwj2vEYaXJg==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "fsevents": "~2.3.2"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "rope-sequence": {
 | 
					    "rope-sequence": {
 | 
				
			||||||
      "version": "1.3.2",
 | 
					      "version": "1.3.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/rope-sequence/-/rope-sequence-1.3.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/rope-sequence/-/rope-sequence-1.3.2.tgz",
 | 
				
			||||||
@ -2668,6 +3066,12 @@
 | 
				
			|||||||
        "has-flag": "^3.0.0"
 | 
					        "has-flag": "^3.0.0"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "supports-preserve-symlinks-flag": {
 | 
				
			||||||
 | 
					      "version": "1.0.0",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
 | 
				
			||||||
 | 
					      "dev": true
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "table": {
 | 
					    "table": {
 | 
				
			||||||
      "version": "6.7.2",
 | 
					      "version": "6.7.2",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/table/-/table-6.7.2.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/table/-/table-6.7.2.tgz",
 | 
				
			||||||
@ -2934,12 +3338,86 @@
 | 
				
			|||||||
      "resolved": "https://registry.npmjs.org/v-money3/-/v-money3-3.16.1.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/v-money3/-/v-money3-3.16.1.tgz",
 | 
				
			||||||
      "integrity": "sha512-U0GjmdybvEwfxCpZiTUbKugSglJbX6wxlyMeg0YJdLTAKlnjMRDph3hpNJlTlg5Gs8MQRpDVdaLysBjV749HLg=="
 | 
					      "integrity": "sha512-U0GjmdybvEwfxCpZiTUbKugSglJbX6wxlyMeg0YJdLTAKlnjMRDph3hpNJlTlg5Gs8MQRpDVdaLysBjV749HLg=="
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "v-tooltip": {
 | 
				
			||||||
 | 
					      "version": "4.0.0-beta.17",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/v-tooltip/-/v-tooltip-4.0.0-beta.17.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-d7v/6KEXQOtcj3NT3Z1LpbDv8SBh8JgbsD+3s/zGIGCxiXC2SoVW6wGV4X0MlCo97PiosibcSe+VKbFiy4AKnQ==",
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "@popperjs/core": "^2.11.0",
 | 
				
			||||||
 | 
					        "vue-resize": "^2.0.0-alpha.1"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "@popperjs/core": {
 | 
				
			||||||
 | 
					          "version": "2.11.6",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.6.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-50/17A98tWUfQ176raKiOGXuYpLyyVMkxxG6oylzL3BPOlA6ADGdK7EYunSa4I064xerltq9TGXs8HmOk5E+vw=="
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "v8-compile-cache": {
 | 
					    "v8-compile-cache": {
 | 
				
			||||||
      "version": "2.3.0",
 | 
					      "version": "2.3.0",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz",
 | 
				
			||||||
      "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==",
 | 
					      "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==",
 | 
				
			||||||
      "dev": true
 | 
					      "dev": true
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "vite": {
 | 
				
			||||||
 | 
					      "version": "2.9.13",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/vite/-/vite-2.9.13.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-AsOBAaT0AD7Mhe8DuK+/kE4aWYFMx/i0ZNi98hJclxb4e0OhQcZYUrvLjIaQ8e59Ui7txcvKMiJC1yftqpQoDw==",
 | 
				
			||||||
 | 
					      "dev": true,
 | 
				
			||||||
 | 
					      "requires": {
 | 
				
			||||||
 | 
					        "esbuild": "^0.14.27",
 | 
				
			||||||
 | 
					        "fsevents": "~2.3.2",
 | 
				
			||||||
 | 
					        "postcss": "^8.4.13",
 | 
				
			||||||
 | 
					        "resolve": "^1.22.0",
 | 
				
			||||||
 | 
					        "rollup": "^2.59.0"
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "dependencies": {
 | 
				
			||||||
 | 
					        "is-core-module": {
 | 
				
			||||||
 | 
					          "version": "2.10.0",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.10.0.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-Erxj2n/LDAZ7H8WNJXd9tw38GYM3dv8rk8Zcs+jJuxYTW7sozH+SS8NtrSjVL1/vpLvWi1hxy96IzjJ3EHTJJg==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "has": "^1.0.3"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "nanoid": {
 | 
				
			||||||
 | 
					          "version": "3.3.4",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "postcss": {
 | 
				
			||||||
 | 
					          "version": "8.4.16",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.16.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-ipHE1XBvKzm5xI7hiHCZJCSugxvsdq2mPnsq5+UF+VHCjiBvtDrlxJfMBToWaP9D5XlgNmcFGqoHmUn0EYEaRQ==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "nanoid": "^3.3.4",
 | 
				
			||||||
 | 
					            "picocolors": "^1.0.0",
 | 
				
			||||||
 | 
					            "source-map-js": "^1.0.2"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "resolve": {
 | 
				
			||||||
 | 
					          "version": "1.22.1",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==",
 | 
				
			||||||
 | 
					          "dev": true,
 | 
				
			||||||
 | 
					          "requires": {
 | 
				
			||||||
 | 
					            "is-core-module": "^2.9.0",
 | 
				
			||||||
 | 
					            "path-parse": "^1.0.7",
 | 
				
			||||||
 | 
					            "supports-preserve-symlinks-flag": "^1.0.0"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "source-map-js": {
 | 
				
			||||||
 | 
					          "version": "1.0.2",
 | 
				
			||||||
 | 
					          "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz",
 | 
				
			||||||
 | 
					          "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==",
 | 
				
			||||||
 | 
					          "dev": true
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "vue": {
 | 
					    "vue": {
 | 
				
			||||||
      "version": "3.2.4",
 | 
					      "version": "3.2.4",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/vue/-/vue-3.2.4.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/vue/-/vue-3.2.4.tgz",
 | 
				
			||||||
@ -3002,6 +3480,11 @@
 | 
				
			|||||||
        "@vue/devtools-api": "^6.0.0-beta.7"
 | 
					        "@vue/devtools-api": "^6.0.0-beta.7"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "vue-resize": {
 | 
				
			||||||
 | 
					      "version": "2.0.0-alpha.1",
 | 
				
			||||||
 | 
					      "resolved": "https://registry.npmjs.org/vue-resize/-/vue-resize-2.0.0-alpha.1.tgz",
 | 
				
			||||||
 | 
					      "integrity": "sha512-7+iqOueLU7uc9NrMfrzbG8hwMqchfVfSzpVlCMeJQe4pyibqyoifDNbKTZvwxZKDvGkB+PdFeKvnGZMoEb8esg=="
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "vue-router": {
 | 
					    "vue-router": {
 | 
				
			||||||
      "version": "4.0.11",
 | 
					      "version": "4.0.11",
 | 
				
			||||||
      "resolved": "https://registry.npmjs.org/vue-router/-/vue-router-4.0.11.tgz",
 | 
					      "resolved": "https://registry.npmjs.org/vue-router/-/vue-router-4.0.11.tgz",
 | 
				
			||||||
 | 
				
			|||||||
@ -24,10 +24,10 @@
 | 
				
			|||||||
    "sass": "^1.32.12",
 | 
					    "sass": "^1.32.12",
 | 
				
			||||||
    "tailwind-scrollbar": "^1.3.1",
 | 
					    "tailwind-scrollbar": "^1.3.1",
 | 
				
			||||||
    "tailwindcss": "^3.0.6",
 | 
					    "tailwindcss": "^3.0.6",
 | 
				
			||||||
    "vite": "^2.6.1"
 | 
					    "vite": "^2.9.13"
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "dependencies": {
 | 
					  "dependencies": {
 | 
				
			||||||
    "@headlessui/vue": "^1.5.0",
 | 
					    "@headlessui/vue": "^1.4.0",
 | 
				
			||||||
    "@heroicons/vue": "^1.0.1",
 | 
					    "@heroicons/vue": "^1.0.1",
 | 
				
			||||||
    "@popperjs/core": "^2.9.2",
 | 
					    "@popperjs/core": "^2.9.2",
 | 
				
			||||||
    "@stripe/stripe-js": "^1.21.2",
 | 
					    "@stripe/stripe-js": "^1.21.2",
 | 
				
			||||||
@ -48,8 +48,7 @@
 | 
				
			|||||||
    "mini-svg-data-uri": "^1.3.3",
 | 
					    "mini-svg-data-uri": "^1.3.3",
 | 
				
			||||||
    "moment": "^2.29.1",
 | 
					    "moment": "^2.29.1",
 | 
				
			||||||
    "pinia": "^2.0.4",
 | 
					    "pinia": "^2.0.4",
 | 
				
			||||||
    "v-calendar": "3.0.0-alpha.8",
 | 
					    "v-money3": "^3.13.5",
 | 
				
			||||||
    "v-money3": "3.16.1",
 | 
					 | 
				
			||||||
    "v-tooltip": "^4.0.0-alpha.1",
 | 
					    "v-tooltip": "^4.0.0-alpha.1",
 | 
				
			||||||
    "vue": "^3.2.0-beta.5",
 | 
					    "vue": "^3.2.0-beta.5",
 | 
				
			||||||
    "vue-flatpickr-component": "^9.0.3",
 | 
					    "vue-flatpickr-component": "^9.0.3",
 | 
				
			||||||
 | 
				
			|||||||
@ -64,7 +64,7 @@ function mergeExistingValues() {
 | 
				
			|||||||
  if (props.isEdit) {
 | 
					  if (props.isEdit) {
 | 
				
			||||||
    props.store[props.storeProp].fields.forEach((field) => {
 | 
					    props.store[props.storeProp].fields.forEach((field) => {
 | 
				
			||||||
      const existingIndex = props.store[props.storeProp].customFields.findIndex(
 | 
					      const existingIndex = props.store[props.storeProp].customFields.findIndex(
 | 
				
			||||||
        (f) => f.id == field.custom_field_id
 | 
					        (f) => f.id === field.custom_field_id
 | 
				
			||||||
      )
 | 
					      )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      if (existingIndex > -1) {
 | 
					      if (existingIndex > -1) {
 | 
				
			||||||
 | 
				
			|||||||
@ -9,7 +9,7 @@ import { computed } from 'vue'
 | 
				
			|||||||
const props = defineProps({
 | 
					const props = defineProps({
 | 
				
			||||||
  modelValue: {
 | 
					  modelValue: {
 | 
				
			||||||
    type: String,
 | 
					    type: String,
 | 
				
			||||||
    default: moment().format('YYYY-MM-DD HH:mm'),
 | 
					    default: moment().format('YYYY-MM-DD hh:MM'),
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -17,7 +17,18 @@
 | 
				
			|||||||
            <td class="px-5 py-4 text-left align-top">
 | 
					            <td class="px-5 py-4 text-left align-top">
 | 
				
			||||||
              <div class="flex justify-start">
 | 
					              <div class="flex justify-start">
 | 
				
			||||||
                <div
 | 
					                <div
 | 
				
			||||||
                  class="flex items-center justify-center w-5 h-5 mt-2 mr-2 text-gray-300 cursor-move  handle"
 | 
					                  class="
 | 
				
			||||||
 | 
					                    flex
 | 
				
			||||||
 | 
					                    items-center
 | 
				
			||||||
 | 
					                    justify-center
 | 
				
			||||||
 | 
					                    w-5
 | 
				
			||||||
 | 
					                    h-5
 | 
				
			||||||
 | 
					                    mt-2
 | 
				
			||||||
 | 
					                    text-gray-300
 | 
				
			||||||
 | 
					                    cursor-move
 | 
				
			||||||
 | 
					                    handle
 | 
				
			||||||
 | 
					                    mr-2
 | 
				
			||||||
 | 
					                  "
 | 
				
			||||||
                >
 | 
					                >
 | 
				
			||||||
                  <DragIcon />
 | 
					                  <DragIcon />
 | 
				
			||||||
                </div>
 | 
					                </div>
 | 
				
			||||||
@ -97,7 +108,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
                          <BaseIcon
 | 
					                          <BaseIcon
 | 
				
			||||||
                            name="ChevronDownIcon"
 | 
					                            name="ChevronDownIcon"
 | 
				
			||||||
                            class="w-4 h-4 ml-1 text-gray-500"
 | 
					                            class="w-4 h-4 text-gray-500 ml-1"
 | 
				
			||||||
                          />
 | 
					                          />
 | 
				
			||||||
                        </span>
 | 
					                        </span>
 | 
				
			||||||
                      </BaseButton>
 | 
					                      </BaseButton>
 | 
				
			||||||
@ -144,7 +155,7 @@
 | 
				
			|||||||
              <BaseContentPlaceholders v-if="loading">
 | 
					              <BaseContentPlaceholders v-if="loading">
 | 
				
			||||||
                <BaseContentPlaceholdersText
 | 
					                <BaseContentPlaceholdersText
 | 
				
			||||||
                  :lines="1"
 | 
					                  :lines="1"
 | 
				
			||||||
                  class="w-24 h-8 border rounded-md"
 | 
					                  class="w-24 h-8 rounded-md border"
 | 
				
			||||||
                />
 | 
					                />
 | 
				
			||||||
              </BaseContentPlaceholders>
 | 
					              </BaseContentPlaceholders>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -164,7 +175,6 @@
 | 
				
			|||||||
                :ability="abilities.CREATE_INVOICE"
 | 
					                :ability="abilities.CREATE_INVOICE"
 | 
				
			||||||
                :store="store"
 | 
					                :store="store"
 | 
				
			||||||
                :store-prop="storeProp"
 | 
					                :store-prop="storeProp"
 | 
				
			||||||
                :discount="discount"
 | 
					 | 
				
			||||||
                @update="updateTax"
 | 
					                @update="updateTax"
 | 
				
			||||||
              />
 | 
					              />
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
 | 
				
			|||||||
@ -30,13 +30,24 @@
 | 
				
			|||||||
        <template v-if="userStore.hasAbilities(ability)" #action>
 | 
					        <template v-if="userStore.hasAbilities(ability)" #action>
 | 
				
			||||||
          <button
 | 
					          <button
 | 
				
			||||||
            type="button"
 | 
					            type="button"
 | 
				
			||||||
            class="flex items-center justify-center w-full px-2 py-2 bg-gray-200 border-none outline-none cursor-pointer "
 | 
					            class="
 | 
				
			||||||
 | 
					              flex
 | 
				
			||||||
 | 
					              items-center
 | 
				
			||||||
 | 
					              justify-center
 | 
				
			||||||
 | 
					              w-full
 | 
				
			||||||
 | 
					              px-2
 | 
				
			||||||
 | 
					              cursor-pointer
 | 
				
			||||||
 | 
					              py-2
 | 
				
			||||||
 | 
					              bg-gray-200
 | 
				
			||||||
 | 
					              border-none
 | 
				
			||||||
 | 
					              outline-none
 | 
				
			||||||
 | 
					            "
 | 
				
			||||||
            @click="openTaxModal"
 | 
					            @click="openTaxModal"
 | 
				
			||||||
          >
 | 
					          >
 | 
				
			||||||
            <BaseIcon name="CheckCircleIcon" class="h-5 text-primary-400" />
 | 
					            <BaseIcon name="CheckCircleIcon" class="h-5 text-primary-400" />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            <label
 | 
					            <label
 | 
				
			||||||
              class="ml-2 text-sm leading-none cursor-pointer text-primary-400"
 | 
					              class="ml-2 text-sm leading-none text-primary-400 cursor-pointer"
 | 
				
			||||||
              >{{ $t('invoices.add_new_tax') }}</label
 | 
					              >{{ $t('invoices.add_new_tax') }}</label
 | 
				
			||||||
            >
 | 
					            >
 | 
				
			||||||
          </button>
 | 
					          </button>
 | 
				
			||||||
@ -104,10 +115,6 @@ const props = defineProps({
 | 
				
			|||||||
    type: Number,
 | 
					    type: Number,
 | 
				
			||||||
    default: 0,
 | 
					    default: 0,
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  discountedTotal: {
 | 
					 | 
				
			||||||
    type: Number,
 | 
					 | 
				
			||||||
    default: 0,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  currency: {
 | 
					  currency: {
 | 
				
			||||||
    type: [Object, String],
 | 
					    type: [Object, String],
 | 
				
			||||||
    required: true,
 | 
					    required: true,
 | 
				
			||||||
@ -146,19 +153,19 @@ const filteredTypes = computed(() => {
 | 
				
			|||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const taxAmount = computed(() => {
 | 
					const taxAmount = computed(() => {
 | 
				
			||||||
  if (localTax.compound_tax && props.discountedTotal) {
 | 
					  if (localTax.compound_tax && props.total) {
 | 
				
			||||||
    return ((props.discountedTotal + props.totalTax) * localTax.percent) / 100
 | 
					    return ((props.total + props.totalTax) * localTax.percent) / 100
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (props.discountedTotal && localTax.percent) {
 | 
					  if (props.total && localTax.percent) {
 | 
				
			||||||
    return (props.discountedTotal * localTax.percent) / 100
 | 
					    return (props.total * localTax.percent) / 100
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return 0
 | 
					  return 0
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
watch(
 | 
					watch(
 | 
				
			||||||
  () => props.discountedTotal,
 | 
					  () => props.total,
 | 
				
			||||||
  () => {
 | 
					  () => {
 | 
				
			||||||
    updateRowTax()
 | 
					    updateRowTax()
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
				
			|||||||
@ -29,7 +29,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
      <label
 | 
					      <label
 | 
				
			||||||
        v-else
 | 
					        v-else
 | 
				
			||||||
        class="flex items-center justify-center m-0 text-lg text-black uppercase "
 | 
					        class="
 | 
				
			||||||
 | 
					          flex
 | 
				
			||||||
 | 
					          items-center
 | 
				
			||||||
 | 
					          justify-center
 | 
				
			||||||
 | 
					          m-0
 | 
				
			||||||
 | 
					          text-lg text-black
 | 
				
			||||||
 | 
					          uppercase
 | 
				
			||||||
 | 
					        "
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <BaseFormatMoney
 | 
					        <BaseFormatMoney
 | 
				
			||||||
          :amount="store.getSubTotal"
 | 
					          :amount="store.getSubTotal"
 | 
				
			||||||
@ -59,7 +66,14 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
      <label
 | 
					      <label
 | 
				
			||||||
        v-else-if="store[storeProp].tax_per_item === 'YES'"
 | 
					        v-else-if="store[storeProp].tax_per_item === 'YES'"
 | 
				
			||||||
        class="flex items-center justify-center m-0 text-lg text-black uppercase "
 | 
					        class="
 | 
				
			||||||
 | 
					          flex
 | 
				
			||||||
 | 
					          items-center
 | 
				
			||||||
 | 
					          justify-center
 | 
				
			||||||
 | 
					          m-0
 | 
				
			||||||
 | 
					          text-lg text-black
 | 
				
			||||||
 | 
					          uppercase
 | 
				
			||||||
 | 
					        "
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <BaseFormatMoney :amount="tax.amount" :currency="defaultCurrency" />
 | 
					        <BaseFormatMoney :amount="tax.amount" :currency="defaultCurrency" />
 | 
				
			||||||
      </label>
 | 
					      </label>
 | 
				
			||||||
@ -84,7 +98,7 @@
 | 
				
			|||||||
      <BaseContentPlaceholders v-if="isLoading">
 | 
					      <BaseContentPlaceholders v-if="isLoading">
 | 
				
			||||||
        <BaseContentPlaceholdersText
 | 
					        <BaseContentPlaceholdersText
 | 
				
			||||||
          :lines="1"
 | 
					          :lines="1"
 | 
				
			||||||
          class="w-24 h-8 border rounded-md"
 | 
					          class="w-24 h-8 rounded-md border"
 | 
				
			||||||
        />
 | 
					        />
 | 
				
			||||||
      </BaseContentPlaceholders>
 | 
					      </BaseContentPlaceholders>
 | 
				
			||||||
      <div v-else class="flex" style="width: 140px" role="group">
 | 
					      <div v-else class="flex" style="width: 140px" role="group">
 | 
				
			||||||
@ -100,7 +114,7 @@
 | 
				
			|||||||
        <BaseDropdown position="bottom-end">
 | 
					        <BaseDropdown position="bottom-end">
 | 
				
			||||||
          <template #activator>
 | 
					          <template #activator>
 | 
				
			||||||
            <BaseButton
 | 
					            <BaseButton
 | 
				
			||||||
              class="p-2 rounded-none rounded-tr-md rounded-br-md"
 | 
					              class="rounded-tr-md rounded-br-md p-2 rounded-none"
 | 
				
			||||||
              type="button"
 | 
					              type="button"
 | 
				
			||||||
              variant="white"
 | 
					              variant="white"
 | 
				
			||||||
            >
 | 
					            >
 | 
				
			||||||
@ -113,7 +127,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
                <BaseIcon
 | 
					                <BaseIcon
 | 
				
			||||||
                  name="ChevronDownIcon"
 | 
					                  name="ChevronDownIcon"
 | 
				
			||||||
                  class="w-4 h-4 ml-1 text-gray-500"
 | 
					                  class="w-4 h-4 text-gray-500 ml-1"
 | 
				
			||||||
                />
 | 
					                />
 | 
				
			||||||
              </span>
 | 
					              </span>
 | 
				
			||||||
            </BaseButton>
 | 
					            </BaseButton>
 | 
				
			||||||
@ -166,7 +180,15 @@
 | 
				
			|||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <div
 | 
					    <div
 | 
				
			||||||
      class="flex items-center justify-between w-full pt-2 mt-5 border-t border-gray-200 border-solid "
 | 
					      class="
 | 
				
			||||||
 | 
					        flex
 | 
				
			||||||
 | 
					        items-center
 | 
				
			||||||
 | 
					        justify-between
 | 
				
			||||||
 | 
					        w-full
 | 
				
			||||||
 | 
					        pt-2
 | 
				
			||||||
 | 
					        mt-5
 | 
				
			||||||
 | 
					        border-t border-gray-200 border-solid
 | 
				
			||||||
 | 
					      "
 | 
				
			||||||
    >
 | 
					    >
 | 
				
			||||||
      <BaseContentPlaceholders v-if="isLoading">
 | 
					      <BaseContentPlaceholders v-if="isLoading">
 | 
				
			||||||
        <BaseContentPlaceholdersText :lines="1" class="w-16 h-5" />
 | 
					        <BaseContentPlaceholdersText :lines="1" class="w-16 h-5" />
 | 
				
			||||||
@ -182,7 +204,14 @@
 | 
				
			|||||||
      </BaseContentPlaceholders>
 | 
					      </BaseContentPlaceholders>
 | 
				
			||||||
      <label
 | 
					      <label
 | 
				
			||||||
        v-else
 | 
					        v-else
 | 
				
			||||||
        class="flex items-center justify-center text-lg uppercase  text-primary-400"
 | 
					        class="
 | 
				
			||||||
 | 
					          flex
 | 
				
			||||||
 | 
					          items-center
 | 
				
			||||||
 | 
					          justify-center
 | 
				
			||||||
 | 
					          text-lg
 | 
				
			||||||
 | 
					          uppercase
 | 
				
			||||||
 | 
					          text-primary-400
 | 
				
			||||||
 | 
					        "
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <BaseFormatMoney :amount="store.getTotal" :currency="defaultCurrency" />
 | 
					        <BaseFormatMoney :amount="store.getTotal" :currency="defaultCurrency" />
 | 
				
			||||||
      </label>
 | 
					      </label>
 | 
				
			||||||
@ -305,7 +334,6 @@ function selectPercentage() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
function onSelectTax(selectedTax) {
 | 
					function onSelectTax(selectedTax) {
 | 
				
			||||||
  let amount = 0
 | 
					  let amount = 0
 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) {
 | 
					  if (selectedTax.compound_tax && props.store.getSubtotalWithDiscount) {
 | 
				
			||||||
    amount = Math.round(
 | 
					    amount = Math.round(
 | 
				
			||||||
      ((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) *
 | 
					      ((props.store.getSubtotalWithDiscount + props.store.getTotalSimpleTax) *
 | 
				
			||||||
 | 
				
			|||||||
@ -48,24 +48,6 @@
 | 
				
			|||||||
            />
 | 
					            />
 | 
				
			||||||
          </BaseInputGroup>
 | 
					          </BaseInputGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
          <BaseInputGroup
 | 
					 | 
				
			||||||
            :label="$tc('settings.company_info.company_slug')"
 | 
					 | 
				
			||||||
            :help-text="$t('settings.company_info.company_slug_help_text')"
 | 
					 | 
				
			||||||
            :error="
 | 
					 | 
				
			||||||
              v$.newCompanyForm.slug.$error &&
 | 
					 | 
				
			||||||
              v$.newCompanyForm.slug.$errors[0].$message
 | 
					 | 
				
			||||||
            "
 | 
					 | 
				
			||||||
            :content-loading="isFetchingInitialData"
 | 
					 | 
				
			||||||
            required
 | 
					 | 
				
			||||||
          >
 | 
					 | 
				
			||||||
            <BaseInput
 | 
					 | 
				
			||||||
              v-model="newCompanyForm.slug"
 | 
					 | 
				
			||||||
              :invalid="v$.newCompanyForm.slug.$error"
 | 
					 | 
				
			||||||
              :content-loading="isFetchingInitialData"
 | 
					 | 
				
			||||||
              @input="v$.newCompanyForm.slug.$touch()"
 | 
					 | 
				
			||||||
            />
 | 
					 | 
				
			||||||
          </BaseInputGroup>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <BaseInputGroup
 | 
					          <BaseInputGroup
 | 
				
			||||||
            :content-loading="isFetchingInitialData"
 | 
					            :content-loading="isFetchingInitialData"
 | 
				
			||||||
            :label="$tc('settings.company_info.country')"
 | 
					            :label="$tc('settings.company_info.country')"
 | 
				
			||||||
@ -148,7 +130,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
<script setup>
 | 
					<script setup>
 | 
				
			||||||
import { useModalStore } from '@/scripts/stores/modal'
 | 
					import { useModalStore } from '@/scripts/stores/modal'
 | 
				
			||||||
import { computed, onMounted, ref, reactive, watch } from 'vue'
 | 
					import { computed, onMounted, ref, reactive } from 'vue'
 | 
				
			||||||
import { useI18n } from 'vue-i18n'
 | 
					import { useI18n } from 'vue-i18n'
 | 
				
			||||||
import { required, minLength, helpers } from '@vuelidate/validators'
 | 
					import { required, minLength, helpers } from '@vuelidate/validators'
 | 
				
			||||||
import { useVuelidate } from '@vuelidate/core'
 | 
					import { useVuelidate } from '@vuelidate/core'
 | 
				
			||||||
@ -170,7 +152,6 @@ let companyLogoName = ref(null)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const newCompanyForm = reactive({
 | 
					const newCompanyForm = reactive({
 | 
				
			||||||
  name: null,
 | 
					  name: null,
 | 
				
			||||||
  slug: null,
 | 
					 | 
				
			||||||
  currency: '',
 | 
					  currency: '',
 | 
				
			||||||
  address: {
 | 
					  address: {
 | 
				
			||||||
    country_id: null,
 | 
					    country_id: null,
 | 
				
			||||||
@ -181,9 +162,6 @@ const modalActive = computed(() => {
 | 
				
			|||||||
  return modalStore.active && modalStore.componentName === 'CompanyModal'
 | 
					  return modalStore.active && modalStore.componentName === 'CompanyModal'
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const slugValidator = (value) => {
 | 
					 | 
				
			||||||
  return value == slugify(value)
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
const rules = {
 | 
					const rules = {
 | 
				
			||||||
  newCompanyForm: {
 | 
					  newCompanyForm: {
 | 
				
			||||||
    name: {
 | 
					    name: {
 | 
				
			||||||
@ -193,17 +171,6 @@ const rules = {
 | 
				
			|||||||
        minLength(3)
 | 
					        minLength(3)
 | 
				
			||||||
      ),
 | 
					      ),
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    slug: {
 | 
					 | 
				
			||||||
      required: helpers.withMessage(t('validation.required'), required),
 | 
					 | 
				
			||||||
      minLength: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.name_min_length', { count: 3 }),
 | 
					 | 
				
			||||||
        minLength(3)
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
      slugValidator: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.invalid_slug'),
 | 
					 | 
				
			||||||
        slugValidator
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    address: {
 | 
					    address: {
 | 
				
			||||||
      country_id: {
 | 
					      country_id: {
 | 
				
			||||||
        required: helpers.withMessage(t('validation.required'), required),
 | 
					        required: helpers.withMessage(t('validation.required'), required),
 | 
				
			||||||
@ -276,7 +243,6 @@ async function submitCompanyData() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
function resetNewCompanyForm() {
 | 
					function resetNewCompanyForm() {
 | 
				
			||||||
  newCompanyForm.name = ''
 | 
					  newCompanyForm.name = ''
 | 
				
			||||||
  newCompanyForm.slug = ''
 | 
					 | 
				
			||||||
  newCompanyForm.currency = ''
 | 
					  newCompanyForm.currency = ''
 | 
				
			||||||
  newCompanyForm.address.country_id = ''
 | 
					  newCompanyForm.address.country_id = ''
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -291,24 +257,4 @@ function closeCompanyModal() {
 | 
				
			|||||||
    v$.value.$reset()
 | 
					    v$.value.$reset()
 | 
				
			||||||
  }, 300)
 | 
					  }, 300)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
// watcher for if change company name then auto fill company slug value
 | 
					 | 
				
			||||||
watch(
 | 
					 | 
				
			||||||
  () => newCompanyForm.name,
 | 
					 | 
				
			||||||
  (currentValue) => {
 | 
					 | 
				
			||||||
    newCompanyForm.slug = slugify(currentValue)
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function slugify(string) {
 | 
					 | 
				
			||||||
  return string
 | 
					 | 
				
			||||||
    .toString()
 | 
					 | 
				
			||||||
    .trim()
 | 
					 | 
				
			||||||
    .toLowerCase()
 | 
					 | 
				
			||||||
    .replace(/\s+/g, '-')
 | 
					 | 
				
			||||||
    .replace(/[^\w\-]+/g, '')
 | 
					 | 
				
			||||||
    .replace(/\-\-+/g, '-')
 | 
					 | 
				
			||||||
    .replace(/^-+/, '')
 | 
					 | 
				
			||||||
    .replace(/-+$/, '')
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
				
			|||||||
@ -15,13 +15,6 @@
 | 
				
			|||||||
      bg-gradient-to-r
 | 
					      bg-gradient-to-r
 | 
				
			||||||
      from-primary-500
 | 
					      from-primary-500
 | 
				
			||||||
      to-primary-400
 | 
					      to-primary-400
 | 
				
			||||||
      dark:from-gray-700/70 dark:to-gray-800/70
 | 
					 | 
				
			||||||
      bg-primary-500
 | 
					 | 
				
			||||||
      dark:bg-transparent
 | 
					 | 
				
			||||||
      dark:backdrop-blur-xl
 | 
					 | 
				
			||||||
      dark:shadow-glass
 | 
					 | 
				
			||||||
      dark:border
 | 
					 | 
				
			||||||
      dark:border-white/10
 | 
					 | 
				
			||||||
    "
 | 
					    "
 | 
				
			||||||
  >
 | 
					  >
 | 
				
			||||||
    <router-link
 | 
					    <router-link
 | 
				
			||||||
@ -60,7 +53,6 @@
 | 
				
			|||||||
        cursor-pointer
 | 
					        cursor-pointer
 | 
				
			||||||
        md:hidden md:ml-0
 | 
					        md:hidden md:ml-0
 | 
				
			||||||
        hover:bg-gray-100
 | 
					        hover:bg-gray-100
 | 
				
			||||||
        dark:bg-gray-800 dark:border-gray-500 dark:border
 | 
					 | 
				
			||||||
      "
 | 
					      "
 | 
				
			||||||
      @click.prevent="onToggle"
 | 
					      @click.prevent="onToggle"
 | 
				
			||||||
    >
 | 
					    >
 | 
				
			||||||
@ -151,7 +143,7 @@
 | 
				
			|||||||
          <template #activator>
 | 
					          <template #activator>
 | 
				
			||||||
            <img
 | 
					            <img
 | 
				
			||||||
              :src="previewAvatar"
 | 
					              :src="previewAvatar"
 | 
				
			||||||
              class="block w-8 h-8 rounded md:h-9 md:w-9 object-cover"
 | 
					              class="block w-8 h-8 rounded md:h-9 md:w-9"
 | 
				
			||||||
            />
 | 
					            />
 | 
				
			||||||
          </template>
 | 
					          </template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -15,9 +15,7 @@
 | 
				
			|||||||
        leave-from="opacity-100"
 | 
					        leave-from="opacity-100"
 | 
				
			||||||
        leave-to="opacity-0"
 | 
					        leave-to="opacity-0"
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <DialogOverlay
 | 
					        <DialogOverlay class="fixed inset-0 bg-gray-600 bg-opacity-75" />
 | 
				
			||||||
          class="fixed inset-0 bg-gray-600 bg-opacity-75 dark:bg-gray-900/90"
 | 
					 | 
				
			||||||
        />
 | 
					 | 
				
			||||||
      </TransitionChild>
 | 
					      </TransitionChild>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      <TransitionChild
 | 
					      <TransitionChild
 | 
				
			||||||
@ -29,9 +27,7 @@
 | 
				
			|||||||
        leave-from="translate-x-0"
 | 
					        leave-from="translate-x-0"
 | 
				
			||||||
        leave-to="-translate-x-full"
 | 
					        leave-to="-translate-x-full"
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <div
 | 
					        <div class="relative flex flex-col flex-1 w-full max-w-xs bg-white">
 | 
				
			||||||
          class="relative flex flex-col flex-1 w-full max-w-xs bg-white dark:bg-gray-800"
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          <TransitionChild
 | 
					          <TransitionChild
 | 
				
			||||||
            as="template"
 | 
					            as="template"
 | 
				
			||||||
            enter="ease-in-out duration-300"
 | 
					            enter="ease-in-out duration-300"
 | 
				
			||||||
@ -54,7 +50,8 @@
 | 
				
			|||||||
                  focus:outline-none
 | 
					                  focus:outline-none
 | 
				
			||||||
                  focus:ring-2
 | 
					                  focus:ring-2
 | 
				
			||||||
                  focus:ring-inset
 | 
					                  focus:ring-inset
 | 
				
			||||||
                focus:ring-white"
 | 
					                  focus:ring-white
 | 
				
			||||||
 | 
					                "
 | 
				
			||||||
                @click="globalStore.setSidebarVisibility(false)"
 | 
					                @click="globalStore.setSidebarVisibility(false)"
 | 
				
			||||||
              >
 | 
					              >
 | 
				
			||||||
                <span class="sr-only">Close sidebar</span>
 | 
					                <span class="sr-only">Close sidebar</span>
 | 
				
			||||||
@ -85,8 +82,8 @@
 | 
				
			|||||||
                :to="item.link"
 | 
					                :to="item.link"
 | 
				
			||||||
                :class="[
 | 
					                :class="[
 | 
				
			||||||
                  hasActiveUrl(item.link)
 | 
					                  hasActiveUrl(item.link)
 | 
				
			||||||
                    ? 'text-primary-500 border-primary-500 bg-gray-100 dark:shadow-glass dark:backdrop-blur-xl dark:hover:bg-gray-700  dark:bg-gray-700/50 dark:text-primary-400 dark:font-medium'
 | 
					                    ? 'text-primary-500 border-primary-500 bg-gray-100 '
 | 
				
			||||||
                    : 'text-black dark:text-gray-300',
 | 
					                    : 'text-black',
 | 
				
			||||||
                  'cursor-pointer px-0 pl-4 py-3 border-transparent flex items-center border-l-4 border-solid text-sm not-italic font-medium',
 | 
					                  'cursor-pointer px-0 pl-4 py-3 border-transparent flex items-center border-l-4 border-solid text-sm not-italic font-medium',
 | 
				
			||||||
                ]"
 | 
					                ]"
 | 
				
			||||||
                @click="globalStore.setSidebarVisibility(false)"
 | 
					                @click="globalStore.setSidebarVisibility(false)"
 | 
				
			||||||
@ -103,10 +100,6 @@
 | 
				
			|||||||
                />
 | 
					                />
 | 
				
			||||||
                {{ $t(item.title) }}
 | 
					                {{ $t(item.title) }}
 | 
				
			||||||
              </router-link>
 | 
					              </router-link>
 | 
				
			||||||
              <LightDarkSwitch
 | 
					 | 
				
			||||||
                :show-label="false"
 | 
					 | 
				
			||||||
                class="absolute right-6 top-6 !w-auto"
 | 
					 | 
				
			||||||
              />
 | 
					 | 
				
			||||||
            </nav>
 | 
					            </nav>
 | 
				
			||||||
          </div>
 | 
					          </div>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
@ -123,13 +116,14 @@
 | 
				
			|||||||
      hidden
 | 
					      hidden
 | 
				
			||||||
      w-56
 | 
					      w-56
 | 
				
			||||||
      h-screen
 | 
					      h-screen
 | 
				
			||||||
 | 
					      pb-32
 | 
				
			||||||
 | 
					      overflow-y-auto
 | 
				
			||||||
      bg-white
 | 
					      bg-white
 | 
				
			||||||
      border-r border-gray-200 border-solid
 | 
					      border-r border-gray-200 border-solid
 | 
				
			||||||
      xl:w-64
 | 
					      xl:w-64
 | 
				
			||||||
      md:fixed md:flex md:flex-col md:inset-y-0
 | 
					      md:fixed md:flex md:flex-col md:inset-y-0
 | 
				
			||||||
      pt-16
 | 
					      pt-16
 | 
				
			||||||
    dark:border-gray-800
 | 
					    "
 | 
				
			||||||
    dark:bg-gray-800/80"
 | 
					 | 
				
			||||||
  >
 | 
					  >
 | 
				
			||||||
    <div
 | 
					    <div
 | 
				
			||||||
      v-for="menu in globalStore.menuGroups"
 | 
					      v-for="menu in globalStore.menuGroups"
 | 
				
			||||||
@ -142,8 +136,8 @@
 | 
				
			|||||||
        :to="item.link"
 | 
					        :to="item.link"
 | 
				
			||||||
        :class="[
 | 
					        :class="[
 | 
				
			||||||
          hasActiveUrl(item.link)
 | 
					          hasActiveUrl(item.link)
 | 
				
			||||||
            ? 'text-primary-500 border-primary-500 bg-gray-100 dark:border-primary-400 dark:shadow-glass dark:backdrop-blur-xl dark:hover:bg-gray-700 dark:bg-gray-700/50 dark:text-primary-400 dark:font-medium'
 | 
					            ? 'text-primary-500 border-primary-500 bg-gray-100 '
 | 
				
			||||||
            : 'text-black dark:hover:bg-transparent dark:hover:text-white dark:text-gray-300',
 | 
					            : 'text-black',
 | 
				
			||||||
          'cursor-pointer px-0 pl-6 hover:bg-gray-50 py-3 group flex items-center border-l-4 border-solid border-transparent text-sm not-italic font-medium',
 | 
					          'cursor-pointer px-0 pl-6 hover:bg-gray-50 py-3 group flex items-center border-l-4 border-solid border-transparent text-sm not-italic font-medium',
 | 
				
			||||||
        ]"
 | 
					        ]"
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
@ -151,8 +145,8 @@
 | 
				
			|||||||
          :name="item.icon"
 | 
					          :name="item.icon"
 | 
				
			||||||
          :class="[
 | 
					          :class="[
 | 
				
			||||||
            hasActiveUrl(item.link)
 | 
					            hasActiveUrl(item.link)
 | 
				
			||||||
              ? 'text-primary-500 group-hover:text-primary-500 dark:text-primary-400 dark:group-hover:text-primary-500 '
 | 
					              ? 'text-primary-500 group-hover:text-primary-500 '
 | 
				
			||||||
              : 'text-gray-400 group-hover:text-black dark:text-gray-400 dark:group-hover:text-white',
 | 
					              : 'text-gray-400 group-hover:text-black',
 | 
				
			||||||
            'mr-4 shrink-0 h-5 w-5 ',
 | 
					            'mr-4 shrink-0 h-5 w-5 ',
 | 
				
			||||||
          ]"
 | 
					          ]"
 | 
				
			||||||
        />
 | 
					        />
 | 
				
			||||||
@ -160,9 +154,6 @@
 | 
				
			|||||||
        {{ $t(item.title) }}
 | 
					        {{ $t(item.title) }}
 | 
				
			||||||
      </router-link>
 | 
					      </router-link>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
    <LightDarkSwitch
 | 
					 | 
				
			||||||
      class="absolute bottom-0 py-4 border-t border-gray-200 dark:border-gray-700"
 | 
					 | 
				
			||||||
    />
 | 
					 | 
				
			||||||
  </div>
 | 
					  </div>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -178,7 +169,6 @@ import {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import { useRoute } from 'vue-router'
 | 
					import { useRoute } from 'vue-router'
 | 
				
			||||||
import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
					import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
				
			||||||
import LightDarkSwitch from '@/scripts/components/LightDarkSwitcher.vue'
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
const route = useRoute()
 | 
					const route = useRoute()
 | 
				
			||||||
const globalStore = useGlobalStore()
 | 
					const globalStore = useGlobalStore()
 | 
				
			||||||
 | 
				
			|||||||
@ -184,20 +184,6 @@ export const useCompanyStore = (useWindow = false) => {
 | 
				
			|||||||
      setDefaultCurrency(data) {
 | 
					      setDefaultCurrency(data) {
 | 
				
			||||||
        this.defaultCurrency = data.currency
 | 
					        this.defaultCurrency = data.currency
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
 | 
					 | 
				
			||||||
      checkCompanyHasCurrencyTransactions() {
 | 
					 | 
				
			||||||
        return new Promise((resolve, reject) => {
 | 
					 | 
				
			||||||
          axios
 | 
					 | 
				
			||||||
            .get(`/api/v1/company/has-transactions`)
 | 
					 | 
				
			||||||
            .then((response) => {
 | 
					 | 
				
			||||||
              resolve(response)
 | 
					 | 
				
			||||||
            })
 | 
					 | 
				
			||||||
            .catch((err) => {
 | 
					 | 
				
			||||||
              handleError(err)
 | 
					 | 
				
			||||||
              reject(err)
 | 
					 | 
				
			||||||
            })
 | 
					 | 
				
			||||||
        })
 | 
					 | 
				
			||||||
      },
 | 
					 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
  })()
 | 
					  })()
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -34,7 +34,6 @@ export const useGlobalStore = (useWindow = false) => {
 | 
				
			|||||||
      isAppLoaded: false,
 | 
					      isAppLoaded: false,
 | 
				
			||||||
      isSidebarOpen: false,
 | 
					      isSidebarOpen: false,
 | 
				
			||||||
      areCurrenciesLoading: false,
 | 
					      areCurrenciesLoading: false,
 | 
				
			||||||
      isDarkModeOn: false,
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
      downloadReport: null,
 | 
					      downloadReport: null,
 | 
				
			||||||
    }),
 | 
					    }),
 | 
				
			||||||
 | 
				
			|||||||
@ -32,8 +32,6 @@
 | 
				
			|||||||
          :content-loading="isLoading"
 | 
					          :content-loading="isLoading"
 | 
				
			||||||
          :calendar-button="true"
 | 
					          :calendar-button="true"
 | 
				
			||||||
          calendar-button-icon="calendar"
 | 
					          calendar-button-icon="calendar"
 | 
				
			||||||
          :show-extra-options="true"
 | 
					 | 
				
			||||||
          :source-date="estimateStore.newEstimate.estimate_date"
 | 
					 | 
				
			||||||
        />
 | 
					        />
 | 
				
			||||||
      </BaseInputGroup>
 | 
					      </BaseInputGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -34,24 +34,6 @@
 | 
				
			|||||||
          />
 | 
					          />
 | 
				
			||||||
        </BaseInputGroup>
 | 
					        </BaseInputGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <BaseInputGroup
 | 
					 | 
				
			||||||
          :label="$tc('wizard.company_slug')"
 | 
					 | 
				
			||||||
          :help-text="$t('wizard.company_slug_help_text')"
 | 
					 | 
				
			||||||
          :error="
 | 
					 | 
				
			||||||
            v$.companyForm.slug.$error &&
 | 
					 | 
				
			||||||
            v$.companyForm.slug.$errors[0].$message
 | 
					 | 
				
			||||||
          "
 | 
					 | 
				
			||||||
          required
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          <BaseInput
 | 
					 | 
				
			||||||
            v-model="companyForm.slug"
 | 
					 | 
				
			||||||
            :invalid="v$.companyForm.slug.$error"
 | 
					 | 
				
			||||||
            type="text"
 | 
					 | 
				
			||||||
            name="slug"
 | 
					 | 
				
			||||||
            @input="v$.companyForm.slug.$touch()"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
        </BaseInputGroup>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        <BaseInputGroup
 | 
					        <BaseInputGroup
 | 
				
			||||||
          :label="$t('wizard.country')"
 | 
					          :label="$t('wizard.country')"
 | 
				
			||||||
          :error="
 | 
					          :error="
 | 
				
			||||||
@ -75,7 +57,9 @@
 | 
				
			|||||||
            track-by="name"
 | 
					            track-by="name"
 | 
				
			||||||
          />
 | 
					          />
 | 
				
			||||||
        </BaseInputGroup>
 | 
					        </BaseInputGroup>
 | 
				
			||||||
 | 
					      </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      <div class="grid grid-cols-1 gap-4 mb-4 md:grid-cols-2 md:mb-6">
 | 
				
			||||||
        <BaseInputGroup :label="$t('wizard.state')">
 | 
					        <BaseInputGroup :label="$t('wizard.state')">
 | 
				
			||||||
          <BaseInput
 | 
					          <BaseInput
 | 
				
			||||||
            v-model="companyForm.address.state"
 | 
					            v-model="companyForm.address.state"
 | 
				
			||||||
@ -160,9 +144,9 @@
 | 
				
			|||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<script setup>
 | 
					<script setup>
 | 
				
			||||||
import { ref, computed, onMounted, reactive, watch } from 'vue'
 | 
					import { ref, computed, onMounted, reactive } from 'vue'
 | 
				
			||||||
import { useI18n } from 'vue-i18n'
 | 
					import { useI18n } from 'vue-i18n'
 | 
				
			||||||
import { required, minLength, maxLength, helpers } from '@vuelidate/validators'
 | 
					import { required, maxLength, helpers } from '@vuelidate/validators'
 | 
				
			||||||
import { useVuelidate } from '@vuelidate/core'
 | 
					import { useVuelidate } from '@vuelidate/core'
 | 
				
			||||||
import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
					import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
				
			||||||
import { useCompanyStore } from '@/scripts/admin/stores/company'
 | 
					import { useCompanyStore } from '@/scripts/admin/stores/company'
 | 
				
			||||||
@ -178,7 +162,6 @@ let logoFileName = ref(null)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const companyForm = reactive({
 | 
					const companyForm = reactive({
 | 
				
			||||||
  name: null,
 | 
					  name: null,
 | 
				
			||||||
  slug: null,
 | 
					 | 
				
			||||||
  address: {
 | 
					  address: {
 | 
				
			||||||
    address_street_1: '',
 | 
					    address_street_1: '',
 | 
				
			||||||
    address_street_2: '',
 | 
					    address_street_2: '',
 | 
				
			||||||
@ -205,28 +188,10 @@ onMounted(async () => {
 | 
				
			|||||||
  })?.id
 | 
					  })?.id
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const slugValidator = (value) => {
 | 
					 | 
				
			||||||
  return value == slugify(value)
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
const rules = {
 | 
					const rules = {
 | 
				
			||||||
  companyForm: {
 | 
					  companyForm: {
 | 
				
			||||||
    name: {
 | 
					    name: {
 | 
				
			||||||
      required: helpers.withMessage(t('validation.required'), required),
 | 
					      required: helpers.withMessage(t('validation.required'), required),
 | 
				
			||||||
      minLength: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.name_min_length', { count: 3 }),
 | 
					 | 
				
			||||||
        minLength(3)
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    slug: {
 | 
					 | 
				
			||||||
      required: helpers.withMessage(t('validation.required'), required),
 | 
					 | 
				
			||||||
      minLength: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.name_min_length', { count: 3 }),
 | 
					 | 
				
			||||||
        minLength(3)
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
      slugValidator: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.invalid_slug'),
 | 
					 | 
				
			||||||
        slugValidator
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    address: {
 | 
					    address: {
 | 
				
			||||||
      country_id: {
 | 
					      country_id: {
 | 
				
			||||||
@ -284,24 +249,4 @@ async function next() {
 | 
				
			|||||||
    emit('next', 7)
 | 
					    emit('next', 7)
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
// watcher for if change company name then auto fill company slug value
 | 
					 | 
				
			||||||
watch(
 | 
					 | 
				
			||||||
  () => companyForm.name,
 | 
					 | 
				
			||||||
  (currentValue) => {
 | 
					 | 
				
			||||||
    companyForm.slug = slugify(currentValue)
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function slugify(string) {
 | 
					 | 
				
			||||||
  return string
 | 
					 | 
				
			||||||
    .toString()
 | 
					 | 
				
			||||||
    .trim()
 | 
					 | 
				
			||||||
    .toLowerCase()
 | 
					 | 
				
			||||||
    .replace(/\s+/g, '-')
 | 
					 | 
				
			||||||
    .replace(/[^\w\-]+/g, '')
 | 
					 | 
				
			||||||
    .replace(/\-\-+/g, '-')
 | 
					 | 
				
			||||||
    .replace(/^-+/, '')
 | 
					 | 
				
			||||||
    .replace(/-+$/, '')
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
				
			|||||||
@ -56,7 +56,7 @@
 | 
				
			|||||||
        <BaseMultiselect
 | 
					        <BaseMultiselect
 | 
				
			||||||
          v-model="filters.status"
 | 
					          v-model="filters.status"
 | 
				
			||||||
          :groups="true"
 | 
					          :groups="true"
 | 
				
			||||||
          :options="invoiceStatus"
 | 
					          :options="status"
 | 
				
			||||||
          searchable
 | 
					          searchable
 | 
				
			||||||
          :placeholder="$t('general.select_a_status')"
 | 
					          :placeholder="$t('general.select_a_status')"
 | 
				
			||||||
          @update:modelValue="setActiveTab"
 | 
					          @update:modelValue="setActiveTab"
 | 
				
			||||||
@ -130,27 +130,11 @@
 | 
				
			|||||||
        "
 | 
					        "
 | 
				
			||||||
      >
 | 
					      >
 | 
				
			||||||
        <!-- Tabs -->
 | 
					        <!-- Tabs -->
 | 
				
			||||||
        <BaseTabGroup
 | 
					        <BaseTabGroup class="-mb-5" @change="setStatusFilter">
 | 
				
			||||||
          class="-mb-5"
 | 
					          <BaseTab :title="$t('general.all')" filter="" />
 | 
				
			||||||
          :selected-index="selectedIndex"
 | 
					          <BaseTab :title="$t('general.draft')" filter="DRAFT" />
 | 
				
			||||||
          @change="changeTabStatus"
 | 
					          <BaseTab :title="$t('general.sent')" filter="SENT" />
 | 
				
			||||||
        >
 | 
					          <BaseTab :title="$t('general.due')" filter="DUE" />
 | 
				
			||||||
          <BaseTab
 | 
					 | 
				
			||||||
            :title="invoiceTabStatus[0].title"
 | 
					 | 
				
			||||||
            :tab-status="invoiceTabStatus[0].value"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
          <BaseTab
 | 
					 | 
				
			||||||
            :title="invoiceTabStatus[1].title"
 | 
					 | 
				
			||||||
            :tab-status="invoiceTabStatus[1].value"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
          <BaseTab
 | 
					 | 
				
			||||||
            :title="invoiceTabStatus[2].title"
 | 
					 | 
				
			||||||
            :tab-status="invoiceTabStatus[2].value"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
          <BaseTab
 | 
					 | 
				
			||||||
            :title="invoiceTabStatus[3].title"
 | 
					 | 
				
			||||||
            :tab-status="invoiceTabStatus[3].value"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
        </BaseTabGroup>
 | 
					        </BaseTabGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <BaseDropdown
 | 
					        <BaseDropdown
 | 
				
			||||||
@ -305,10 +289,10 @@ const utils = inject('$utils')
 | 
				
			|||||||
const table = ref(null)
 | 
					const table = ref(null)
 | 
				
			||||||
const showFilters = ref(false)
 | 
					const showFilters = ref(false)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const invoiceStatus = ref([
 | 
					const status = ref([
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
    label: 'Status',
 | 
					    label: 'Status',
 | 
				
			||||||
    options: ['DRAFT', 'SENT', 'VIEWED', 'COMPLETED'],
 | 
					    options: ['DRAFT', 'DUE', 'SENT', 'VIEWED', 'COMPLETED'],
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
    label: 'Paid Status',
 | 
					    label: 'Paid Status',
 | 
				
			||||||
@ -316,29 +300,10 @@ const invoiceStatus = ref([
 | 
				
			|||||||
  },
 | 
					  },
 | 
				
			||||||
  ,
 | 
					  ,
 | 
				
			||||||
])
 | 
					])
 | 
				
			||||||
 | 
					 | 
				
			||||||
const invoiceTabStatus = {
 | 
					 | 
				
			||||||
  0: {
 | 
					 | 
				
			||||||
    title: t('general.all'),
 | 
					 | 
				
			||||||
    value: '',
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  1: {
 | 
					 | 
				
			||||||
    title: t('general.draft'),
 | 
					 | 
				
			||||||
    value: 'DRAFT',
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  2: {
 | 
					 | 
				
			||||||
    title: t('general.sent'),
 | 
					 | 
				
			||||||
    value: 'SENT',
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  3: {
 | 
					 | 
				
			||||||
    title: t('general.due'),
 | 
					 | 
				
			||||||
    value: 'DUE',
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
const isRequestOngoing = ref(true)
 | 
					const isRequestOngoing = ref(true)
 | 
				
			||||||
 | 
					const activeTab = ref('general.draft')
 | 
				
			||||||
const router = useRouter()
 | 
					const router = useRouter()
 | 
				
			||||||
const userStore = useUserStore()
 | 
					const userStore = useUserStore()
 | 
				
			||||||
const selectedIndex = ref(0)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
let filters = reactive({
 | 
					let filters = reactive({
 | 
				
			||||||
  customer_id: '',
 | 
					  customer_id: '',
 | 
				
			||||||
@ -346,7 +311,6 @@ let filters = reactive({
 | 
				
			|||||||
  from_date: '',
 | 
					  from_date: '',
 | 
				
			||||||
  to_date: '',
 | 
					  to_date: '',
 | 
				
			||||||
  invoice_number: '',
 | 
					  invoice_number: '',
 | 
				
			||||||
  tab_status: '',
 | 
					 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const showEmptyScreen = computed(
 | 
					const showEmptyScreen = computed(
 | 
				
			||||||
@ -437,7 +401,6 @@ async function fetchData({ page, filter, sort }) {
 | 
				
			|||||||
    from_date: filters.from_date,
 | 
					    from_date: filters.from_date,
 | 
				
			||||||
    to_date: filters.to_date,
 | 
					    to_date: filters.to_date,
 | 
				
			||||||
    invoice_number: filters.invoice_number,
 | 
					    invoice_number: filters.invoice_number,
 | 
				
			||||||
    tab_status: filters.tab_status,
 | 
					 | 
				
			||||||
    orderByField: sort.fieldName || 'created_at',
 | 
					    orderByField: sort.fieldName || 'created_at',
 | 
				
			||||||
    orderBy: sort.order || 'desc',
 | 
					    orderBy: sort.order || 'desc',
 | 
				
			||||||
    page,
 | 
					    page,
 | 
				
			||||||
@ -460,9 +423,29 @@ async function fetchData({ page, filter, sort }) {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function changeTabStatus(val, index) {
 | 
					function setStatusFilter(val) {
 | 
				
			||||||
  filters.tab_status = val['tab-status']
 | 
					  if (activeTab.value == val.title) {
 | 
				
			||||||
  selectedIndex.value = index
 | 
					    return true
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  activeTab.value = val.title
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  switch (val.title) {
 | 
				
			||||||
 | 
					    case t('general.draft'):
 | 
				
			||||||
 | 
					      filters.status = 'DRAFT'
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					    case t('general.sent'):
 | 
				
			||||||
 | 
					      filters.status = 'SENT'
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    case t('general.due'):
 | 
				
			||||||
 | 
					      filters.status = 'DUE'
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    default:
 | 
				
			||||||
 | 
					      filters.status = ''
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function setFilters() {
 | 
					function setFilters() {
 | 
				
			||||||
@ -480,6 +463,8 @@ function clearFilter() {
 | 
				
			|||||||
  filters.from_date = ''
 | 
					  filters.from_date = ''
 | 
				
			||||||
  filters.to_date = ''
 | 
					  filters.to_date = ''
 | 
				
			||||||
  filters.invoice_number = ''
 | 
					  filters.invoice_number = ''
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  activeTab.value = t('general.all')
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function removeMultipleInvoices() {
 | 
					async function removeMultipleInvoices() {
 | 
				
			||||||
@ -520,21 +505,39 @@ function toggleFilter() {
 | 
				
			|||||||
function setActiveTab(val) {
 | 
					function setActiveTab(val) {
 | 
				
			||||||
  switch (val) {
 | 
					  switch (val) {
 | 
				
			||||||
    case 'DRAFT':
 | 
					    case 'DRAFT':
 | 
				
			||||||
      selectedIndex.value = 1
 | 
					      activeTab.value = t('general.draft')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					    case 'SENT':
 | 
				
			||||||
 | 
					      activeTab.value = t('general.sent')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    case 'DUE':
 | 
				
			||||||
 | 
					      activeTab.value = t('general.due')
 | 
				
			||||||
      break
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    case 'SENT':
 | 
					 | 
				
			||||||
    case 'VIEWED':
 | 
					 | 
				
			||||||
    case 'COMPLETED':
 | 
					    case 'COMPLETED':
 | 
				
			||||||
 | 
					      activeTab.value = t('invoices.completed')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    case 'PAID':
 | 
					    case 'PAID':
 | 
				
			||||||
      selectedIndex.value = 2
 | 
					      activeTab.value = t('invoices.paid')
 | 
				
			||||||
      break
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    case 'UNPAID':
 | 
					    case 'UNPAID':
 | 
				
			||||||
 | 
					      activeTab.value = t('invoices.unpaid')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    case 'PARTIALLY_PAID':
 | 
					    case 'PARTIALLY_PAID':
 | 
				
			||||||
      selectedIndex.value = 3
 | 
					      activeTab.value = t('invoices.partially_paid')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    case 'VIEWED':
 | 
				
			||||||
 | 
					      activeTab.value = t('invoices.viewed')
 | 
				
			||||||
 | 
					      break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    default:
 | 
				
			||||||
 | 
					      activeTab.value = t('general.all')
 | 
				
			||||||
      break
 | 
					      break
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  filters.tab_status = invoiceTabStatus[selectedIndex.value].value
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
				
			|||||||
@ -32,8 +32,6 @@
 | 
				
			|||||||
          :content-loading="isLoading"
 | 
					          :content-loading="isLoading"
 | 
				
			||||||
          :calendar-button="true"
 | 
					          :calendar-button="true"
 | 
				
			||||||
          calendar-button-icon="calendar"
 | 
					          calendar-button-icon="calendar"
 | 
				
			||||||
          :show-extra-options="true"
 | 
					 | 
				
			||||||
          :source-date="invoiceStore.newInvoice.invoice_date"
 | 
					 | 
				
			||||||
        />
 | 
					        />
 | 
				
			||||||
      </BaseInputGroup>
 | 
					      </BaseInputGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -82,9 +82,9 @@
 | 
				
			|||||||
            required
 | 
					            required
 | 
				
			||||||
          >
 | 
					          >
 | 
				
			||||||
            <BaseCustomerSelectInput
 | 
					            <BaseCustomerSelectInput
 | 
				
			||||||
              v-if="!isLoadingContent"
 | 
					 | 
				
			||||||
              v-model="paymentStore.currentPayment.customer_id"
 | 
					              v-model="paymentStore.currentPayment.customer_id"
 | 
				
			||||||
              :content-loading="isLoadingContent"
 | 
					              :content-loading="isLoadingContent"
 | 
				
			||||||
 | 
					              v-if="!isLoadingContent"
 | 
				
			||||||
              :invalid="v$.currentPayment.customer_id.$error"
 | 
					              :invalid="v$.currentPayment.customer_id.$error"
 | 
				
			||||||
              :placeholder="$t('customers.select_a_customer')"
 | 
					              :placeholder="$t('customers.select_a_customer')"
 | 
				
			||||||
              show-action
 | 
					              show-action
 | 
				
			||||||
@ -423,7 +423,7 @@ function onCustomerChange(customer_id) {
 | 
				
			|||||||
  if (customer_id) {
 | 
					  if (customer_id) {
 | 
				
			||||||
    let data = {
 | 
					    let data = {
 | 
				
			||||||
      customer_id: customer_id,
 | 
					      customer_id: customer_id,
 | 
				
			||||||
      tab_status: 'DUE',
 | 
					      status: 'DUE',
 | 
				
			||||||
      limit: 'all',
 | 
					      limit: 'all',
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -446,11 +446,7 @@ function onCustomerChange(customer_id) {
 | 
				
			|||||||
          paymentStore.currentPayment.selectedCustomer = res2.data.data
 | 
					          paymentStore.currentPayment.selectedCustomer = res2.data.data
 | 
				
			||||||
          paymentStore.currentPayment.customer = res2.data.data
 | 
					          paymentStore.currentPayment.customer = res2.data.data
 | 
				
			||||||
          paymentStore.currentPayment.currency = res2.data.data.currency
 | 
					          paymentStore.currentPayment.currency = res2.data.data.currency
 | 
				
			||||||
          if (
 | 
					          if(isEdit.value && !customerStore.editCustomer && paymentStore.currentPayment.customer_id) {
 | 
				
			||||||
            isEdit.value &&
 | 
					 | 
				
			||||||
            !customerStore.editCustomer &&
 | 
					 | 
				
			||||||
            paymentStore.currentPayment.customer_id
 | 
					 | 
				
			||||||
          ) {
 | 
					 | 
				
			||||||
            customerStore.editCustomer = res2.data.data
 | 
					            customerStore.editCustomer = res2.data.data
 | 
				
			||||||
          }
 | 
					          }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
				
			|||||||
@ -28,19 +28,6 @@
 | 
				
			|||||||
          />
 | 
					          />
 | 
				
			||||||
        </BaseInputGroup>
 | 
					        </BaseInputGroup>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <BaseInputGroup
 | 
					 | 
				
			||||||
          :label="$tc('settings.company_info.company_slug')"
 | 
					 | 
				
			||||||
          :help-text="$t('settings.company_info.company_slug_help_text')"
 | 
					 | 
				
			||||||
          :error="v$.slug.$error && v$.slug.$errors[0].$message"
 | 
					 | 
				
			||||||
          required
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          <BaseInput
 | 
					 | 
				
			||||||
            v-model="companyForm.slug"
 | 
					 | 
				
			||||||
            :invalid="v$.slug.$error"
 | 
					 | 
				
			||||||
            @blur="v$.slug.$touch()"
 | 
					 | 
				
			||||||
          />
 | 
					 | 
				
			||||||
        </BaseInputGroup>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        <BaseInputGroup :label="$tc('settings.company_info.phone')">
 | 
					        <BaseInputGroup :label="$tc('settings.company_info.phone')">
 | 
				
			||||||
          <BaseInput v-model="companyForm.address.phone" />
 | 
					          <BaseInput v-model="companyForm.address.phone" />
 | 
				
			||||||
        </BaseInputGroup>
 | 
					        </BaseInputGroup>
 | 
				
			||||||
@ -173,7 +160,6 @@ let isSaving = ref(false)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const companyForm = reactive({
 | 
					const companyForm = reactive({
 | 
				
			||||||
  name: null,
 | 
					  name: null,
 | 
				
			||||||
  slug: null,
 | 
					 | 
				
			||||||
  logo: null,
 | 
					  logo: null,
 | 
				
			||||||
  address: {
 | 
					  address: {
 | 
				
			||||||
    address_street_1: '',
 | 
					    address_street_1: '',
 | 
				
			||||||
@ -207,14 +193,7 @@ const rules = computed(() => {
 | 
				
			|||||||
    name: {
 | 
					    name: {
 | 
				
			||||||
      required: helpers.withMessage(t('validation.required'), required),
 | 
					      required: helpers.withMessage(t('validation.required'), required),
 | 
				
			||||||
      minLength: helpers.withMessage(
 | 
					      minLength: helpers.withMessage(
 | 
				
			||||||
        t('validation.name_min_length', { count: 3 }),
 | 
					        t('validation.name_min_length'),
 | 
				
			||||||
        minLength(3)
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    slug: {
 | 
					 | 
				
			||||||
      required: helpers.withMessage(t('validation.required'), required),
 | 
					 | 
				
			||||||
      minLength: helpers.withMessage(
 | 
					 | 
				
			||||||
        t('validation.name_min_length', { count: 3 }),
 | 
					 | 
				
			||||||
        minLength(3)
 | 
					        minLength(3)
 | 
				
			||||||
      ),
 | 
					      ),
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
				
			|||||||
@ -8,11 +8,7 @@
 | 
				
			|||||||
        <BaseInputGroup
 | 
					        <BaseInputGroup
 | 
				
			||||||
          :content-loading="isFetchingInitialData"
 | 
					          :content-loading="isFetchingInitialData"
 | 
				
			||||||
          :label="$tc('settings.preferences.currency')"
 | 
					          :label="$tc('settings.preferences.currency')"
 | 
				
			||||||
          :help-text="
 | 
					          :help-text="$t('settings.preferences.company_currency_unchangeable')"
 | 
				
			||||||
            isCurrencyDisabled
 | 
					 | 
				
			||||||
              ? $t('settings.preferences.company_currency_unchangeable')
 | 
					 | 
				
			||||||
              : ''
 | 
					 | 
				
			||||||
          "
 | 
					 | 
				
			||||||
          :error="v$.currency.$error && v$.currency.$errors[0].$message"
 | 
					          :error="v$.currency.$error && v$.currency.$errors[0].$message"
 | 
				
			||||||
          required
 | 
					          required
 | 
				
			||||||
        >
 | 
					        >
 | 
				
			||||||
@ -25,7 +21,7 @@
 | 
				
			|||||||
            :searchable="true"
 | 
					            :searchable="true"
 | 
				
			||||||
            track-by="name"
 | 
					            track-by="name"
 | 
				
			||||||
            :invalid="v$.currency.$error"
 | 
					            :invalid="v$.currency.$error"
 | 
				
			||||||
            :disabled="isCurrencyDisabled"
 | 
					            disabled
 | 
				
			||||||
            class="w-full"
 | 
					            class="w-full"
 | 
				
			||||||
          >
 | 
					          >
 | 
				
			||||||
          </BaseMultiselect>
 | 
					          </BaseMultiselect>
 | 
				
			||||||
@ -191,7 +187,6 @@ const { t, tm } = useI18n()
 | 
				
			|||||||
let isSaving = ref(false)
 | 
					let isSaving = ref(false)
 | 
				
			||||||
let isDataSaving = ref(false)
 | 
					let isDataSaving = ref(false)
 | 
				
			||||||
let isFetchingInitialData = ref(false)
 | 
					let isFetchingInitialData = ref(false)
 | 
				
			||||||
let isCurrencyDisabled = ref(true)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
const settingsForm = reactive({ ...companyStore.selectedCompanySettings })
 | 
					const settingsForm = reactive({ ...companyStore.selectedCompanySettings })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -287,14 +282,10 @@ setInitialData()
 | 
				
			|||||||
async function setInitialData() {
 | 
					async function setInitialData() {
 | 
				
			||||||
  isFetchingInitialData.value = true
 | 
					  isFetchingInitialData.value = true
 | 
				
			||||||
  Promise.all([
 | 
					  Promise.all([
 | 
				
			||||||
    companyStore.checkCompanyHasCurrencyTransactions(),
 | 
					 | 
				
			||||||
    globalStore.fetchCurrencies(),
 | 
					    globalStore.fetchCurrencies(),
 | 
				
			||||||
    globalStore.fetchDateFormats(),
 | 
					    globalStore.fetchDateFormats(),
 | 
				
			||||||
    globalStore.fetchTimeZones(),
 | 
					    globalStore.fetchTimeZones(),
 | 
				
			||||||
  ]).then(([res1]) => {
 | 
					  ]).then(([res1]) => {
 | 
				
			||||||
    if (res1.data?.has_transactions == false) {
 | 
					 | 
				
			||||||
      isCurrencyDisabled.value = false
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    isFetchingInitialData.value = false
 | 
					    isFetchingInitialData.value = false
 | 
				
			||||||
  })
 | 
					  })
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -1,46 +1,46 @@
 | 
				
			|||||||
<template>
 | 
					<template>
 | 
				
			||||||
  <div class="bg-white shadow overflow-hidden rounded-lg mt-6 dark:bg-gray-800">
 | 
					  <div class="bg-white shadow overflow-hidden rounded-lg mt-6">
 | 
				
			||||||
    <div class="px-4 py-5 sm:px-6">
 | 
					    <div class="px-4 py-5 sm:px-6">
 | 
				
			||||||
      <h3 class="text-lg leading-6 font-medium text-gray-900 dark:text-gray-400">
 | 
					      <h3 class="text-lg leading-6 font-medium text-gray-900">
 | 
				
			||||||
        {{ $t('invoices.invoice_information') }}
 | 
					        {{ $t('invoices.invoice_information') }}
 | 
				
			||||||
      </h3>
 | 
					      </h3>
 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
    <div v-if="invoice" class="border-t border-gray-200 px-4 py-5 sm:p-0">
 | 
					    <div v-if="invoice" class="border-t border-gray-200 px-4 py-5 sm:p-0">
 | 
				
			||||||
      <dl class="sm:divide-y sm:divide-gray-200 dark:sm:divide-gray-500">
 | 
					      <dl class="sm:divide-y sm:divide-gray-200">
 | 
				
			||||||
        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
					        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
				
			||||||
          <dt class="text-sm font-medium text-gray-500 dark:text-gray-400">
 | 
					          <dt class="text-sm font-medium text-gray-500">
 | 
				
			||||||
            {{ $t('general.from') }}
 | 
					            {{ $t('general.from') }}
 | 
				
			||||||
          </dt>
 | 
					          </dt>
 | 
				
			||||||
          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2   dark:text-gray-100">
 | 
					          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
				
			||||||
            {{ invoice.company.name }}
 | 
					            {{ invoice.company.name }}
 | 
				
			||||||
          </dd>
 | 
					          </dd>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
					        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
				
			||||||
          <dt class="text-sm font-medium text-gray-500 dark:text-gray-400">
 | 
					          <dt class="text-sm font-medium text-gray-500">
 | 
				
			||||||
            {{ $t('general.to') }}
 | 
					            {{ $t('general.to') }}
 | 
				
			||||||
          </dt>
 | 
					          </dt>
 | 
				
			||||||
          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2   dark:text-gray-100">
 | 
					          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
				
			||||||
            {{ invoice.customer.name }}
 | 
					            {{ invoice.customer.name }}
 | 
				
			||||||
          </dd>
 | 
					          </dd>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
					        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
				
			||||||
          <dt class="text-sm font-medium text-gray-500 capitalize dark:text-gray-400 ">
 | 
					          <dt class="text-sm font-medium text-gray-500 capitalize">
 | 
				
			||||||
            {{ $t('invoices.paid_status').toLowerCase() }}
 | 
					            {{ $t('invoices.paid_status').toLowerCase() }}
 | 
				
			||||||
          </dt>
 | 
					          </dt>
 | 
				
			||||||
          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
					          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
				
			||||||
            <BaseInvoiceStatusBadge
 | 
					            <BaseInvoiceStatusBadge
 | 
				
			||||||
              :status="invoice.paid_status"
 | 
					              :status="invoice.paid_status"
 | 
				
			||||||
              class="px-3 py-1 dark:text-gray-400"
 | 
					              class="px-3 py-1"
 | 
				
			||||||
            >
 | 
					            >
 | 
				
			||||||
              {{ invoice.paid_status }}
 | 
					              {{ invoice.paid_status }}
 | 
				
			||||||
            </BaseInvoiceStatusBadge>
 | 
					            </BaseInvoiceStatusBadge>
 | 
				
			||||||
          </dd>
 | 
					          </dd>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
					        <div class="py-4 sm:py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
 | 
				
			||||||
          <dt class="text-sm font-medium text-gray-500 dark:text-gray-400">
 | 
					          <dt class="text-sm font-medium text-gray-500">
 | 
				
			||||||
            {{ $t('invoices.total') }}
 | 
					            {{ $t('invoices.total') }}
 | 
				
			||||||
          </dt>
 | 
					          </dt>
 | 
				
			||||||
          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2 dark:border-gray-600 dark:bg-gray-700 dark:text-white">
 | 
					          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
				
			||||||
            <BaseFormatMoney
 | 
					            <BaseFormatMoney
 | 
				
			||||||
              :currency="invoice.currency"
 | 
					              :currency="invoice.currency"
 | 
				
			||||||
              :amount="invoice.total"
 | 
					              :amount="invoice.total"
 | 
				
			||||||
@ -54,7 +54,7 @@
 | 
				
			|||||||
          <dt class="text-sm font-medium text-gray-500">
 | 
					          <dt class="text-sm font-medium text-gray-500">
 | 
				
			||||||
            {{ $t('invoices.notes') }}
 | 
					            {{ $t('invoices.notes') }}
 | 
				
			||||||
          </dt>
 | 
					          </dt>
 | 
				
			||||||
          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2 dark:text-gray-100" >
 | 
					          <dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
 | 
				
			||||||
            <span v-html="invoice.formatted_notes"></span>
 | 
					            <span v-html="invoice.formatted_notes"></span>
 | 
				
			||||||
          </dd>
 | 
					          </dd>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
 | 
				
			|||||||
@ -1,101 +0,0 @@
 | 
				
			|||||||
<!-- This example requires Tailwind CSS v2.0+ -->
 | 
					 | 
				
			||||||
<script lang="ts" setup>
 | 
					 | 
				
			||||||
import { Switch, SwitchGroup, SwitchLabel } from '@headlessui/vue'
 | 
					 | 
				
			||||||
import { useGlobalStore } from '@/scripts/admin/stores/global'
 | 
					 | 
				
			||||||
import { computed, ref } from 'vue'
 | 
					 | 
				
			||||||
defineProps({
 | 
					 | 
				
			||||||
  showLabel: {
 | 
					 | 
				
			||||||
    type: Boolean,
 | 
					 | 
				
			||||||
    default: true,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  vertical: {
 | 
					 | 
				
			||||||
    type: Boolean,
 | 
					 | 
				
			||||||
    default: false,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const globalStore = useGlobalStore()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const enabled = ref(
 | 
					 | 
				
			||||||
  localStorage.getItem('theme') === 'dark' ||
 | 
					 | 
				
			||||||
    document.documentElement.classList.contains('dark')
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
globalStore.isDarkModeOn = enabled
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
function onChange(val) {
 | 
					 | 
				
			||||||
  if (val) {
 | 
					 | 
				
			||||||
    localStorage.theme = 'dark'
 | 
					 | 
				
			||||||
    document.documentElement.classList.add('dark')
 | 
					 | 
				
			||||||
    document.documentElement.style.setProperty('color-scheme', 'dark')
 | 
					 | 
				
			||||||
    globalStore.isDarkModeOn = true
 | 
					 | 
				
			||||||
  } else {
 | 
					 | 
				
			||||||
    localStorage.theme = 'light'
 | 
					 | 
				
			||||||
    document.documentElement.classList.remove('dark')
 | 
					 | 
				
			||||||
    document.documentElement.style.setProperty('color-scheme', 'light')
 | 
					 | 
				
			||||||
    globalStore.isDarkModeOn = false
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
</script>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
<template>
 | 
					 | 
				
			||||||
  <div class="w-full flex justify-center">
 | 
					 | 
				
			||||||
    <SwitchGroup
 | 
					 | 
				
			||||||
      as="div"
 | 
					 | 
				
			||||||
      class="flex items-center"
 | 
					 | 
				
			||||||
      :class="vertical ? 'flex-col justify-center' : 'flex-row'"
 | 
					 | 
				
			||||||
    >
 | 
					 | 
				
			||||||
      <Switch
 | 
					 | 
				
			||||||
        v-model="enabled"
 | 
					 | 
				
			||||||
        class="relative inline-flex flex-shrink-0 h-6 w-11 border-2 border-transparent rounded-full cursor-pointer transition-colors ease-in-out duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary-500 dark:ring-offset-gray-700"
 | 
					 | 
				
			||||||
        :class="[enabled ? 'bg-primary-600' : 'bg-gray-200']"
 | 
					 | 
				
			||||||
        @update:modelValue="onChange"
 | 
					 | 
				
			||||||
      >
 | 
					 | 
				
			||||||
        <span class="sr-only">Use setting</span>
 | 
					 | 
				
			||||||
        <span
 | 
					 | 
				
			||||||
          class="pointer-events-none relative inline-block h-5 w-5 rounded-full bg-white shadow transform ring-0 transition ease-in-out duration-200"
 | 
					 | 
				
			||||||
          :class="[enabled ? 'translate-x-5' : 'translate-x-0']"
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          <span
 | 
					 | 
				
			||||||
            class="absolute inset-0 h-full w-full flex items-center justify-center transition-opacity"
 | 
					 | 
				
			||||||
            :class="[
 | 
					 | 
				
			||||||
              enabled
 | 
					 | 
				
			||||||
                ? 'opacity-0 ease-out duration-100'
 | 
					 | 
				
			||||||
                : 'opacity-100 ease-in duration-200',
 | 
					 | 
				
			||||||
            ]"
 | 
					 | 
				
			||||||
            aria-hidden="true"
 | 
					 | 
				
			||||||
          >
 | 
					 | 
				
			||||||
            <BaseIcon class="h-3 w-3 text-yellow-500" name="SunIcon" />
 | 
					 | 
				
			||||||
          </span>
 | 
					 | 
				
			||||||
          <span
 | 
					 | 
				
			||||||
            class="absolute inset-0 h-full w-full flex items-center justify-center transition-opacity"
 | 
					 | 
				
			||||||
            :class="[
 | 
					 | 
				
			||||||
              enabled
 | 
					 | 
				
			||||||
                ? 'opacity-100 ease-in duration-200'
 | 
					 | 
				
			||||||
                : 'opacity-0 ease-out duration-100',
 | 
					 | 
				
			||||||
            ]"
 | 
					 | 
				
			||||||
            aria-hidden="true"
 | 
					 | 
				
			||||||
          >
 | 
					 | 
				
			||||||
            <BaseIcon class="h-3 w-3 text-primary-500" name="MoonIcon" />
 | 
					 | 
				
			||||||
          </span>
 | 
					 | 
				
			||||||
        </span>
 | 
					 | 
				
			||||||
      </Switch>
 | 
					 | 
				
			||||||
      <SwitchLabel
 | 
					 | 
				
			||||||
        v-if="showLabel"
 | 
					 | 
				
			||||||
        as="span"
 | 
					 | 
				
			||||||
        class="cursor-pointer"
 | 
					 | 
				
			||||||
        :class="vertical ? 'px-1 text-center mt-2' : 'ml-3'"
 | 
					 | 
				
			||||||
      >
 | 
					 | 
				
			||||||
        <span
 | 
					 | 
				
			||||||
          v-if="enabled"
 | 
					 | 
				
			||||||
          class="text-sm font-medium text-gray-500 dark:text-gray-400"
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          Dark Mode
 | 
					 | 
				
			||||||
        </span>
 | 
					 | 
				
			||||||
        <span v-else class="text-sm font-medium text-gray-500">
 | 
					 | 
				
			||||||
          Light Mode
 | 
					 | 
				
			||||||
        </span>
 | 
					 | 
				
			||||||
      </SwitchLabel>
 | 
					 | 
				
			||||||
    </SwitchGroup>
 | 
					 | 
				
			||||||
  </div>
 | 
					 | 
				
			||||||
</template>
 | 
					 | 
				
			||||||
@ -1,9 +1,5 @@
 | 
				
			|||||||
<template>
 | 
					<template>
 | 
				
			||||||
  <div
 | 
					  <div class="bg-white rounded-lg shadow">
 | 
				
			||||||
    class="bg-white rounded-lg shadow dark:bg-gray-800 dark:text-white dark:shadow-glass dark:border dark:border-white/10 dark:bg-gray-800/70 relative"
 | 
					 | 
				
			||||||
  >
 | 
					 | 
				
			||||||
    <BaseDarkHighlight class="z-[-1] mt-10" />
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    <div
 | 
					    <div
 | 
				
			||||||
      v-if="hasHeaderSlot"
 | 
					      v-if="hasHeaderSlot"
 | 
				
			||||||
      class="px-5 py-4 text-black border-b border-gray-100 border-solid"
 | 
					      class="px-5 py-4 text-black border-b border-gray-100 border-solid"
 | 
				
			||||||
 | 
				
			|||||||
@ -126,7 +126,7 @@ onMounted(() => {
 | 
				
			|||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const value = computed({
 | 
					const value = computed({
 | 
				
			||||||
  get: () => (props.modelValue ? props.modelValue : ''),
 | 
					  get: () => props.modelValue,
 | 
				
			||||||
  set: (value) => {
 | 
					  set: (value) => {
 | 
				
			||||||
    emit('update:modelValue', value)
 | 
					    emit('update:modelValue', value)
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
@ -195,9 +195,7 @@ async function getFields() {
 | 
				
			|||||||
          { label: 'Date', value: 'INVOICE_DATE' },
 | 
					          { label: 'Date', value: 'INVOICE_DATE' },
 | 
				
			||||||
          { label: 'Due Date', value: 'INVOICE_DUE_DATE' },
 | 
					          { label: 'Due Date', value: 'INVOICE_DUE_DATE' },
 | 
				
			||||||
          { label: 'Number', value: 'INVOICE_NUMBER' },
 | 
					          { label: 'Number', value: 'INVOICE_NUMBER' },
 | 
				
			||||||
          { label: 'PDF Link', value: 'PDF_LINK' },
 | 
					          { label: 'Ref Number', value: 'INVOICE_REF_NUMBER' },
 | 
				
			||||||
          { label: 'Due Amount', value: 'DUE_AMOUNT' },
 | 
					 | 
				
			||||||
          { label: 'Total Amount', value: 'TOTAL_AMOUNT' },
 | 
					 | 
				
			||||||
          ...invoiceFields.value.map((i) => ({
 | 
					          ...invoiceFields.value.map((i) => ({
 | 
				
			||||||
            label: i.label,
 | 
					            label: i.label,
 | 
				
			||||||
            value: i.slug,
 | 
					            value: i.slug,
 | 
				
			||||||
@ -213,8 +211,7 @@ async function getFields() {
 | 
				
			|||||||
          { label: 'Date', value: 'ESTIMATE_DATE' },
 | 
					          { label: 'Date', value: 'ESTIMATE_DATE' },
 | 
				
			||||||
          { label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' },
 | 
					          { label: 'Expiry Date', value: 'ESTIMATE_EXPIRY_DATE' },
 | 
				
			||||||
          { label: 'Number', value: 'ESTIMATE_NUMBER' },
 | 
					          { label: 'Number', value: 'ESTIMATE_NUMBER' },
 | 
				
			||||||
          { label: 'PDF Link', value: 'PDF_LINK' },
 | 
					          { label: 'Ref Number', value: 'ESTIMATE_REF_NUMBER' },
 | 
				
			||||||
          { label: 'Total Amount', value: 'TOTAL_AMOUNT' },
 | 
					 | 
				
			||||||
          ...estimateFields.value.map((i) => ({
 | 
					          ...estimateFields.value.map((i) => ({
 | 
				
			||||||
            label: i.label,
 | 
					            label: i.label,
 | 
				
			||||||
            value: i.slug,
 | 
					            value: i.slug,
 | 
				
			||||||
@ -231,7 +228,6 @@ async function getFields() {
 | 
				
			|||||||
          { label: 'Number', value: 'PAYMENT_NUMBER' },
 | 
					          { label: 'Number', value: 'PAYMENT_NUMBER' },
 | 
				
			||||||
          { label: 'Mode', value: 'PAYMENT_MODE' },
 | 
					          { label: 'Mode', value: 'PAYMENT_MODE' },
 | 
				
			||||||
          { label: 'Amount', value: 'PAYMENT_AMOUNT' },
 | 
					          { label: 'Amount', value: 'PAYMENT_AMOUNT' },
 | 
				
			||||||
          { label: 'PDF Link', value: 'PDF_LINK' },
 | 
					 | 
				
			||||||
          ...paymentFields.value.map((i) => ({
 | 
					          ...paymentFields.value.map((i) => ({
 | 
				
			||||||
            label: i.label,
 | 
					            label: i.label,
 | 
				
			||||||
            value: i.slug,
 | 
					            value: i.slug,
 | 
				
			||||||
 | 
				
			|||||||
@ -1,21 +0,0 @@
 | 
				
			|||||||
<template>
 | 
					 | 
				
			||||||
  <div
 | 
					 | 
				
			||||||
    class="
 | 
					 | 
				
			||||||
      hidden
 | 
					 | 
				
			||||||
      top-0
 | 
					 | 
				
			||||||
      w-full
 | 
					 | 
				
			||||||
      absolute
 | 
					 | 
				
			||||||
      ml-auto
 | 
					 | 
				
			||||||
      mr-auto
 | 
					 | 
				
			||||||
      left-0
 | 
					 | 
				
			||||||
      right-0
 | 
					 | 
				
			||||||
      text-center
 | 
					 | 
				
			||||||
      h-full
 | 
					 | 
				
			||||||
      rounded-full
 | 
					 | 
				
			||||||
      bg-highlight/[.10]
 | 
					 | 
				
			||||||
      blur-2xl
 | 
					 | 
				
			||||||
      dark:block
 | 
					 | 
				
			||||||
      z-[-1]
 | 
					 | 
				
			||||||
    "
 | 
					 | 
				
			||||||
  />
 | 
					 | 
				
			||||||
</template>
 | 
					 | 
				
			||||||
@ -7,33 +7,7 @@
 | 
				
			|||||||
    />
 | 
					    />
 | 
				
			||||||
  </BaseContentPlaceholders>
 | 
					  </BaseContentPlaceholders>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  <div v-else :class="computedContainerClass">
 | 
					  <div v-else :class="computedContainerClass" class="relative flex flex-row">
 | 
				
			||||||
    <date-picker
 | 
					 | 
				
			||||||
      ref="vCalendar"
 | 
					 | 
				
			||||||
      v-model="date"
 | 
					 | 
				
			||||||
      :mode="mode"
 | 
					 | 
				
			||||||
      :is24hr="time24hr"
 | 
					 | 
				
			||||||
      class="w-full"
 | 
					 | 
				
			||||||
      color="indigo"
 | 
					 | 
				
			||||||
      :input-debounce="500"
 | 
					 | 
				
			||||||
      :update-on-input="false"
 | 
					 | 
				
			||||||
      :is-range="false"
 | 
					 | 
				
			||||||
      trim-weeks
 | 
					 | 
				
			||||||
      :is-required="isRequired"
 | 
					 | 
				
			||||||
      :popover="{
 | 
					 | 
				
			||||||
        visibility: disabled ? 'hidden' : 'focus',
 | 
					 | 
				
			||||||
        showDelay: 0,
 | 
					 | 
				
			||||||
        hideDelay: 1,
 | 
					 | 
				
			||||||
      }"
 | 
					 | 
				
			||||||
      :attributes="attrs"
 | 
					 | 
				
			||||||
      :model-config="config"
 | 
					 | 
				
			||||||
      :masks="masks"
 | 
					 | 
				
			||||||
      :locale="global.locale"
 | 
					 | 
				
			||||||
    >
 | 
					 | 
				
			||||||
      <template
 | 
					 | 
				
			||||||
        #default="{ inputValue, inputEvents, togglePopover, hidePopover }"
 | 
					 | 
				
			||||||
      >
 | 
					 | 
				
			||||||
        <!-- calendar icon -->
 | 
					 | 
				
			||||||
    <svg
 | 
					    <svg
 | 
				
			||||||
      v-if="showCalendarIcon && !hasIconSlot"
 | 
					      v-if="showCalendarIcon && !hasIconSlot"
 | 
				
			||||||
      viewBox="0 0 20 20"
 | 
					      viewBox="0 0 20 20"
 | 
				
			||||||
@ -50,7 +24,7 @@
 | 
				
			|||||||
        text-gray-400
 | 
					        text-gray-400
 | 
				
			||||||
        cursor-pointer
 | 
					        cursor-pointer
 | 
				
			||||||
      "
 | 
					      "
 | 
				
			||||||
          @click="togglePopover()"
 | 
					      @click="onClickDp"
 | 
				
			||||||
    >
 | 
					    >
 | 
				
			||||||
      <path
 | 
					      <path
 | 
				
			||||||
        fill-rule="evenodd"
 | 
					        fill-rule="evenodd"
 | 
				
			||||||
@ -61,54 +35,24 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    <slot v-if="showCalendarIcon && hasIconSlot" name="icon" />
 | 
					    <slot v-if="showCalendarIcon && hasIconSlot" name="icon" />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        <input
 | 
					    <FlatPickr
 | 
				
			||||||
          :value="inputValue"
 | 
					      ref="dp"
 | 
				
			||||||
 | 
					      v-model="date"
 | 
				
			||||||
 | 
					      v-bind="$attrs"
 | 
				
			||||||
 | 
					      :disabled="disabled"
 | 
				
			||||||
 | 
					      :config="config"
 | 
				
			||||||
      :class="[defaultInputClass, inputInvalidClass, inputDisabledClass]"
 | 
					      :class="[defaultInputClass, inputInvalidClass, inputDisabledClass]"
 | 
				
			||||||
          readonly
 | 
					 | 
				
			||||||
          v-on="inputEvents"
 | 
					 | 
				
			||||||
          @blur="hidePopover()"
 | 
					 | 
				
			||||||
    />
 | 
					    />
 | 
				
			||||||
      </template>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      <template v-if="showExtraOptions" #footer>
 | 
					 | 
				
			||||||
        <div
 | 
					 | 
				
			||||||
          class="bg-gray-100 grid grid-cols-3 gap-2 p-2 border-t rounded-b-lg"
 | 
					 | 
				
			||||||
        >
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="moveToDate(sourceDate)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.same_day') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="withInDays(7)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.within_7_days') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="withInDays(15)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.within_15_days') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="withInDays(30)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.within_30_days') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="withInDays(45)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.within_45_days') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          <button type="button" class="extra-button" @click="withInDays(60)">
 | 
					 | 
				
			||||||
            {{ global.t('date_picker.within_60_days') }}
 | 
					 | 
				
			||||||
          </button>
 | 
					 | 
				
			||||||
        </div>
 | 
					 | 
				
			||||||
      </template>
 | 
					 | 
				
			||||||
    </date-picker>
 | 
					 | 
				
			||||||
  </div>
 | 
					  </div>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<script type="text/babel" setup>
 | 
					<script type="text/babel" setup>
 | 
				
			||||||
import { Calendar, DatePicker } from 'v-calendar'
 | 
					import FlatPickr from 'vue-flatpickr-component'
 | 
				
			||||||
import 'v-calendar/dist/style.css'
 | 
					import 'flatpickr/dist/flatpickr.css'
 | 
				
			||||||
import { computed, reactive, watch, ref, useSlots } from 'vue'
 | 
					import { computed, reactive, watch, ref, useSlots } from 'vue'
 | 
				
			||||||
import { useCompanyStore } from '@/scripts/admin/stores/company'
 | 
					import { useCompanyStore } from '@/scripts/admin/stores/company'
 | 
				
			||||||
import moment from 'moment'
 | 
					
 | 
				
			||||||
 | 
					const dp = ref(null)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const props = defineProps({
 | 
					const props = defineProps({
 | 
				
			||||||
  modelValue: {
 | 
					  modelValue: {
 | 
				
			||||||
@ -146,31 +90,36 @@ const props = defineProps({
 | 
				
			|||||||
  defaultInputClass: {
 | 
					  defaultInputClass: {
 | 
				
			||||||
    type: String,
 | 
					    type: String,
 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
      'border-2 font-base pl-8 py-2 outline-none focus:ring-primary-400 focus:outline-none focus:border-primary-400 block w-full sm:text-sm border-gray-200 rounded-md text-black',
 | 
					      'font-base pl-8 py-2 outline-none focus:ring-primary-400 focus:outline-none focus:border-primary-400 block w-full sm:text-sm border-gray-200 rounded-md text-black',
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  time24hr: {
 | 
					  time24hr: {
 | 
				
			||||||
    type: Boolean,
 | 
					    type: Boolean,
 | 
				
			||||||
    default: false,
 | 
					    default: false,
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  isRequired: {
 | 
					 | 
				
			||||||
    type: Boolean,
 | 
					 | 
				
			||||||
    default: false,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  showExtraOptions: {
 | 
					 | 
				
			||||||
    type: Boolean,
 | 
					 | 
				
			||||||
    default: false,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  sourceDate: {
 | 
					 | 
				
			||||||
    type: [String, Date],
 | 
					 | 
				
			||||||
    default: () => new Date(),
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const emit = defineEmits(['update:modelValue'])
 | 
					const emit = defineEmits(['update:modelValue'])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const slots = useSlots()
 | 
					const slots = useSlots()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const companyStore = useCompanyStore()
 | 
					const companyStore = useCompanyStore()
 | 
				
			||||||
const { global } = window.i18n
 | 
					
 | 
				
			||||||
const vCalendar = ref(null)
 | 
					let config = reactive({
 | 
				
			||||||
 | 
					  altInput: true,
 | 
				
			||||||
 | 
					  enableTime: props.enableTime,
 | 
				
			||||||
 | 
					  time_24hr: props.time24hr,
 | 
				
			||||||
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const date = computed({
 | 
				
			||||||
 | 
					  get: () => props.modelValue,
 | 
				
			||||||
 | 
					  set: (value) => {
 | 
				
			||||||
 | 
					    emit('update:modelValue', value)
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const carbonFormat = computed(() => {
 | 
				
			||||||
 | 
					  return companyStore.selectedCompanySettings?.carbon_date_format
 | 
				
			||||||
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const hasIconSlot = computed(() => {
 | 
					const hasIconSlot = computed(() => {
 | 
				
			||||||
  return !!slots.icon
 | 
					  return !!slots.icon
 | 
				
			||||||
@ -186,6 +135,7 @@ const inputInvalidClass = computed(() => {
 | 
				
			|||||||
  if (props.invalid) {
 | 
					  if (props.invalid) {
 | 
				
			||||||
    return 'border-red-400 ring-red-400 focus:ring-red-400 focus:border-red-400'
 | 
					    return 'border-red-400 ring-red-400 focus:ring-red-400 focus:border-red-400'
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return ''
 | 
					  return ''
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -193,97 +143,35 @@ const inputDisabledClass = computed(() => {
 | 
				
			|||||||
  if (props.disabled) {
 | 
					  if (props.disabled) {
 | 
				
			||||||
    return 'border border-solid rounded-md outline-none input-field box-border-2 base-date-picker-input placeholder-gray-400 bg-gray-200 text-gray-600 border-gray-200'
 | 
					    return 'border border-solid rounded-md outline-none input-field box-border-2 base-date-picker-input placeholder-gray-400 bg-gray-200 text-gray-600 border-gray-200'
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return ''
 | 
					  return ''
 | 
				
			||||||
})
 | 
					})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// to convert YYYY-MM-DD | YYYY-MM-DD HH:mm format
 | 
					function onClickDp(params) {
 | 
				
			||||||
function convertYMDFormat(date) {
 | 
					  dp.value.fp.open()
 | 
				
			||||||
  let format = props.enableTime ? 'YYYY-MM-DD HH:mm' : 'YYYY-MM-DD'
 | 
					 | 
				
			||||||
  return date ? moment(date).format(format) : date
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const date = computed({
 | 
					 | 
				
			||||||
  get: () => props.modelValue,
 | 
					 | 
				
			||||||
  set: (value) => {
 | 
					 | 
				
			||||||
    emit('update:modelValue', value)
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const mode = computed(() => {
 | 
					 | 
				
			||||||
  return props.enableTime ? 'dateTime' : 'date'
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const config = reactive({
 | 
					 | 
				
			||||||
  type: 'string',
 | 
					 | 
				
			||||||
  mask: 'YYYY-MM-DD', // Uses 'iso' if missing
 | 
					 | 
				
			||||||
  //timeAdjust: '00:00:00',
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const masks = reactive({
 | 
					 | 
				
			||||||
  input: null,
 | 
					 | 
				
			||||||
  inputDateTime: null,
 | 
					 | 
				
			||||||
  inputDateTime24hr: null,
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const attrs = reactive([
 | 
					 | 
				
			||||||
  {
 | 
					 | 
				
			||||||
    dates: new Date(),
 | 
					 | 
				
			||||||
    highlight: {
 | 
					 | 
				
			||||||
      fillMode: 'outline',
 | 
					 | 
				
			||||||
    },
 | 
					 | 
				
			||||||
    /* popover: {
 | 
					 | 
				
			||||||
      label: 'Today Date',
 | 
					 | 
				
			||||||
      visibility: 'hover',
 | 
					 | 
				
			||||||
    }, */
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
])
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
const carbonFormat = computed(() => {
 | 
					 | 
				
			||||||
  return companyStore.selectedCompanySettings?.moment_date_format
 | 
					 | 
				
			||||||
})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
watch(
 | 
					watch(
 | 
				
			||||||
  () => carbonFormat,
 | 
					  () => props.enableTime,
 | 
				
			||||||
  () => {
 | 
					  (val) => {
 | 
				
			||||||
    if (!props.enableTime) {
 | 
					    if (props.enableTime) {
 | 
				
			||||||
      masks.input = carbonFormat.value ? carbonFormat.value : 'DD MMM YYYY'
 | 
					      config.enableTime = props.enableTime
 | 
				
			||||||
      config.mask = 'YYYY-MM-DD'
 | 
					 | 
				
			||||||
    } else {
 | 
					 | 
				
			||||||
      let timeFormat = 'HH:mm'
 | 
					 | 
				
			||||||
      if (props.time24hr) {
 | 
					 | 
				
			||||||
        masks.inputDateTime24hr = carbonFormat.value
 | 
					 | 
				
			||||||
          ? `${carbonFormat.value} ${timeFormat}`
 | 
					 | 
				
			||||||
          : `DD MMM YYYY ${timeFormat}`
 | 
					 | 
				
			||||||
      } else {
 | 
					 | 
				
			||||||
        masks.inputDateTime = carbonFormat.value
 | 
					 | 
				
			||||||
          ? `${carbonFormat.value} ${timeFormat}`
 | 
					 | 
				
			||||||
          : `DD MMM YYYY ${timeFormat}`
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      config.mask = `YYYY-MM-DD ${timeFormat}`
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  { immediate: true }
 | 
					  { immediate: true }
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async function moveToDate(_date) {
 | 
					watch(
 | 
				
			||||||
  const calendar = vCalendar.value
 | 
					  () => carbonFormat,
 | 
				
			||||||
  _date = _date ? _date : convertYMDFormat(new Date())
 | 
					  () => {
 | 
				
			||||||
  date.value = _date
 | 
					    if (!props.enableTime) {
 | 
				
			||||||
  // await calendar.move(_date)
 | 
					      config.altFormat = carbonFormat.value ? carbonFormat.value : 'd M Y'
 | 
				
			||||||
  calendar.hidePopover()
 | 
					    } else {
 | 
				
			||||||
}
 | 
					      config.altFormat = carbonFormat.value
 | 
				
			||||||
 | 
					        ? `${carbonFormat.value} H:i `
 | 
				
			||||||
async function withInDays(noOfDays) {
 | 
					        : 'd M Y H:i'
 | 
				
			||||||
  if (!noOfDays) return false
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  let newDate = moment(props.sourceDate).add(noOfDays, 'days').toDate()
 | 
					 | 
				
			||||||
  newDate = convertYMDFormat(newDate)
 | 
					 | 
				
			||||||
  moveToDate(newDate)
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					  },
 | 
				
			||||||
 | 
					  { immediate: true }
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
					 | 
				
			||||||
<style scoped>
 | 
					 | 
				
			||||||
.extra-button {
 | 
					 | 
				
			||||||
  @apply bg-primary-500 text-white text-sm font-semibold px-2 py-1 rounded hover:bg-primary-700;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
</style>
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1,10 +1,6 @@
 | 
				
			|||||||
<template>
 | 
					<template>
 | 
				
			||||||
  <div>
 | 
					  <div>
 | 
				
			||||||
    <TabGroup
 | 
					    <TabGroup :default-index="defaultIndex" @change="onChange">
 | 
				
			||||||
      :selected-index="selectedIndex"
 | 
					 | 
				
			||||||
      :default-index="defaultIndex"
 | 
					 | 
				
			||||||
      @change="onChange"
 | 
					 | 
				
			||||||
    >
 | 
					 | 
				
			||||||
      <TabList
 | 
					      <TabList
 | 
				
			||||||
        :class="[
 | 
					        :class="[
 | 
				
			||||||
          'flex border-b border-grey-light',
 | 
					          'flex border-b border-grey-light',
 | 
				
			||||||
@ -58,10 +54,6 @@ const props = defineProps({
 | 
				
			|||||||
    type: Number,
 | 
					    type: Number,
 | 
				
			||||||
    default: 0,
 | 
					    default: 0,
 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  selectedIndex: {
 | 
					 | 
				
			||||||
    type: Number,
 | 
					 | 
				
			||||||
    default: 0,
 | 
					 | 
				
			||||||
  },
 | 
					 | 
				
			||||||
  filter: {
 | 
					  filter: {
 | 
				
			||||||
    type: String,
 | 
					    type: String,
 | 
				
			||||||
    default: null,
 | 
					    default: null,
 | 
				
			||||||
@ -75,6 +67,6 @@ const slots = useSlots()
 | 
				
			|||||||
const tabs = computed(() => slots.default().map((tab) => tab.props))
 | 
					const tabs = computed(() => slots.default().map((tab) => tab.props))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function onChange(d) {
 | 
					function onChange(d) {
 | 
				
			||||||
  emit('change', tabs.value[d], d)
 | 
					  emit('change', tabs.value[d])
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
				
			|||||||
@ -7,12 +7,12 @@ export function usePopper(options) {
 | 
				
			|||||||
  let popper = ref(null)
 | 
					  let popper = ref(null)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  onMounted(() => {
 | 
					  onMounted(() => {
 | 
				
			||||||
    watchEffect((onInvalidate) => {
 | 
					    watchEffect(onInvalidate => {
 | 
				
			||||||
      if (!container.value) return
 | 
					      if (!container.value) return
 | 
				
			||||||
      if (!activator.value) return
 | 
					      if (!activator.value) return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      let containerEl = container.value.el || container.value
 | 
					      let containerEl = container.value.el || container.value
 | 
				
			||||||
      let activatorEl = activator.value.$el || activator.value
 | 
					      let activatorEl = activator.value.el || activator.value
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      if (!(activatorEl instanceof HTMLElement)) return
 | 
					      if (!(activatorEl instanceof HTMLElement)) return
 | 
				
			||||||
      if (!(containerEl instanceof HTMLElement)) return
 | 
					      if (!(containerEl instanceof HTMLElement)) return
 | 
				
			||||||
 | 
				
			|||||||
@ -863,8 +863,6 @@
 | 
				
			|||||||
    "company_info": {
 | 
					    "company_info": {
 | 
				
			||||||
      "company_info": "Company info",
 | 
					      "company_info": "Company info",
 | 
				
			||||||
      "company_name": "Company Name",
 | 
					      "company_name": "Company Name",
 | 
				
			||||||
      "company_slug": "Company Slug",
 | 
					 | 
				
			||||||
      "company_slug_help_text": "A unique URL friendly name for your company (It will appear on Customer Portal URL)",
 | 
					 | 
				
			||||||
      "company_logo": "Company Logo",
 | 
					      "company_logo": "Company Logo",
 | 
				
			||||||
      "section_description": "Information about your company that will be displayed on invoices, estimates and other documents created by Crater.",
 | 
					      "section_description": "Information about your company that will be displayed on invoices, estimates and other documents created by Crater.",
 | 
				
			||||||
      "phone": "Phone",
 | 
					      "phone": "Phone",
 | 
				
			||||||
@ -1326,8 +1324,6 @@
 | 
				
			|||||||
    "company_info": "Company Information",
 | 
					    "company_info": "Company Information",
 | 
				
			||||||
    "company_info_desc": "This information will be displayed on invoices. Note that you can edit this later on settings page.",
 | 
					    "company_info_desc": "This information will be displayed on invoices. Note that you can edit this later on settings page.",
 | 
				
			||||||
    "company_name": "Company Name",
 | 
					    "company_name": "Company Name",
 | 
				
			||||||
    "company_slug": "Company Slug",
 | 
					 | 
				
			||||||
    "company_slug_help_text": "A unique URL friendly name for your company (It will appear on Customer Portal URL)",
 | 
					 | 
				
			||||||
    "company_logo": "Company Logo",
 | 
					    "company_logo": "Company Logo",
 | 
				
			||||||
    "logo_preview": "Logo Preview",
 | 
					    "logo_preview": "Logo Preview",
 | 
				
			||||||
    "preferences": "Company Preferences",
 | 
					    "preferences": "Company Preferences",
 | 
				
			||||||
@ -1458,8 +1454,7 @@
 | 
				
			|||||||
    "at_least_one_ability": "Please select atleast one Permission.",
 | 
					    "at_least_one_ability": "Please select atleast one Permission.",
 | 
				
			||||||
    "valid_driver_key": "Please enter a valid {driver} key.",
 | 
					    "valid_driver_key": "Please enter a valid {driver} key.",
 | 
				
			||||||
    "valid_exchange_rate": "Please enter a valid exchange rate.",
 | 
					    "valid_exchange_rate": "Please enter a valid exchange rate.",
 | 
				
			||||||
    "company_name_not_same": "Company name must match with given name.",
 | 
					    "company_name_not_same": "Company name must match with given name."
 | 
				
			||||||
    "invalid_slug": "Invalid Slug"
 | 
					 | 
				
			||||||
  },
 | 
					  },
 | 
				
			||||||
  "errors": {
 | 
					  "errors": {
 | 
				
			||||||
    "starter_plan": "This feature is available on Starter plan and onwards!",
 | 
					    "starter_plan": "This feature is available on Starter plan and onwards!",
 | 
				
			||||||
@ -1527,13 +1522,5 @@
 | 
				
			|||||||
  "pdf_bill_to": "Bill to,",
 | 
					  "pdf_bill_to": "Bill to,",
 | 
				
			||||||
  "pdf_ship_to": "Ship to,",
 | 
					  "pdf_ship_to": "Ship to,",
 | 
				
			||||||
  "pdf_received_from": "Received from:",
 | 
					  "pdf_received_from": "Received from:",
 | 
				
			||||||
  "pdf_tax_label": "Tax",
 | 
					  "pdf_tax_label": "Tax"
 | 
				
			||||||
  "date_picker": {
 | 
					 | 
				
			||||||
    "same_day": "Same Day",
 | 
					 | 
				
			||||||
    "within_7_days": "Within 7 Days",
 | 
					 | 
				
			||||||
    "within_15_days": "Within 15 Days",
 | 
					 | 
				
			||||||
    "within_30_days": "Within 30 Days",
 | 
					 | 
				
			||||||
    "within_45_days": "Within 45 Days",
 | 
					 | 
				
			||||||
    "within_60_days": "Within 60 Days"
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -25,7 +25,7 @@
 | 
				
			|||||||
</head>
 | 
					</head>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<body
 | 
					<body
 | 
				
			||||||
    class="h-full overflow-hidden bg-gray-100 dark:bg-gray-900 dark:text-white font-base
 | 
					    class="h-full overflow-hidden bg-gray-100 font-base
 | 
				
			||||||
    @if(isset($current_theme)) theme-{{ $current_theme }} @else theme-{{get_app_setting('admin_portal_theme') ?? 'crater'}} @endif ">
 | 
					    @if(isset($current_theme)) theme-{{ $current_theme }} @else theme-{{get_app_setting('admin_portal_theme') ?? 'crater'}} @endif ">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <!-- Module Scripts -->
 | 
					    <!-- Module Scripts -->
 | 
				
			||||||
@ -38,14 +38,6 @@
 | 
				
			|||||||
    @endforeach
 | 
					    @endforeach
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <script type="module">
 | 
					    <script type="module">
 | 
				
			||||||
        if (localStorage.theme === 'dark' || (!('theme' in localStorage) && window.matchMedia('(prefers-color-scheme: dark)').matches)) {
 | 
					 | 
				
			||||||
            document.documentElement.classList.add('dark')
 | 
					 | 
				
			||||||
            document.documentElement.style.setProperty('color-scheme', 'dark');
 | 
					 | 
				
			||||||
        } else {
 | 
					 | 
				
			||||||
            document.documentElement.classList.remove('dark')
 | 
					 | 
				
			||||||
            document.documentElement.style.setProperty('color-scheme', 'light')
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        @if(isset($customer_logo))
 | 
					        @if(isset($customer_logo))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        window.customer_logo = "/storage/{{$customer_logo}}"
 | 
					        window.customer_logo = "/storage/{{$customer_logo}}"
 | 
				
			||||||
 | 
				
			|||||||
@ -133,6 +133,7 @@
 | 
				
			|||||||
            line-height: 21px;
 | 
					            line-height: 21px;
 | 
				
			||||||
            color: #5851D8;
 | 
					            color: #5851D8;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    </style>
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @if (App::isLocale('th'))
 | 
					    @if (App::isLocale('th'))
 | 
				
			||||||
@ -169,7 +170,7 @@
 | 
				
			|||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                        <td>
 | 
					                        <td>
 | 
				
			||||||
                            <p class="expense-amount">
 | 
					                            <p class="expense-amount">
 | 
				
			||||||
                            {!! format_money_pdf($expenseCategory->total_amount, $currency) !!}
 | 
					                                {!! format_money_pdf($expenseCategory->total_amount) !!}
 | 
				
			||||||
                            </p>
 | 
					                            </p>
 | 
				
			||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                    </tr>
 | 
					                    </tr>
 | 
				
			||||||
@ -181,7 +182,7 @@
 | 
				
			|||||||
    <table class="expense-total-table">
 | 
					    <table class="expense-total-table">
 | 
				
			||||||
        <tr>
 | 
					        <tr>
 | 
				
			||||||
            <td class="expense-total-cell">
 | 
					            <td class="expense-total-cell">
 | 
				
			||||||
                <p class="expense-total">{!! format_money_pdf($totalExpense, $currency) !!}</p>
 | 
					                <p class="expense-total">{!! format_money_pdf($totalExpense) !!}</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
    </table>
 | 
					    </table>
 | 
				
			||||||
@ -191,7 +192,7 @@
 | 
				
			|||||||
                <p class="report-footer-label">@lang('pdf_total_expenses_label')</p>
 | 
					                <p class="report-footer-label">@lang('pdf_total_expenses_label')</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
            <td>
 | 
					            <td>
 | 
				
			||||||
                <p class="report-footer-value">{!! format_money_pdf($totalExpense, $currency) !!}</p>
 | 
					                <p class="report-footer-value">{!! format_money_pdf($totalExpense) !!}</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
    </table>
 | 
					    </table>
 | 
				
			||||||
 | 
				
			|||||||
@ -158,6 +158,7 @@
 | 
				
			|||||||
            line-height: 21px;
 | 
					            line-height: 21px;
 | 
				
			||||||
            color: #5851D8;
 | 
					            color: #5851D8;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    </style>
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @if (App::isLocale('th'))
 | 
					    @if (App::isLocale('th'))
 | 
				
			||||||
@ -189,7 +190,7 @@
 | 
				
			|||||||
                    <p class="income-title">@lang("pdf_income_label")</p>
 | 
					                    <p class="income-title">@lang("pdf_income_label")</p>
 | 
				
			||||||
                </td>
 | 
					                </td>
 | 
				
			||||||
                <td>
 | 
					                <td>
 | 
				
			||||||
                    <p class="income-amount">{!! format_money_pdf($income, $currency) !!}</p>
 | 
					                    <p class="income-amount">{!! format_money_pdf($income) !!}</p>
 | 
				
			||||||
                </td>
 | 
					                </td>
 | 
				
			||||||
            </tr>
 | 
					            </tr>
 | 
				
			||||||
        </table>
 | 
					        </table>
 | 
				
			||||||
@ -205,7 +206,7 @@
 | 
				
			|||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                        <td>
 | 
					                        <td>
 | 
				
			||||||
                            <p class="expense-amount">
 | 
					                            <p class="expense-amount">
 | 
				
			||||||
                            {!! format_money_pdf($expenseCategory->total_amount, $currency) !!}
 | 
					                                {!! format_money_pdf($expenseCategory->total_amount) !!}
 | 
				
			||||||
                            </p>
 | 
					                            </p>
 | 
				
			||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                    </tr>
 | 
					                    </tr>
 | 
				
			||||||
@ -218,7 +219,7 @@
 | 
				
			|||||||
    <table class="expense-total-indicator-table">
 | 
					    <table class="expense-total-indicator-table">
 | 
				
			||||||
        <tr>
 | 
					        <tr>
 | 
				
			||||||
            <td class="expense-total-cell">
 | 
					            <td class="expense-total-cell">
 | 
				
			||||||
                <p class="expense-total">{!! format_money_pdf($totalExpense, $currency) !!}</p>
 | 
					                <p class="expense-total">{!! format_money_pdf($totalExpense) !!}</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
    </table>
 | 
					    </table>
 | 
				
			||||||
@ -228,7 +229,7 @@
 | 
				
			|||||||
                <p class="report-footer-label">@lang("pdf_net_profit_label")</p>
 | 
					                <p class="report-footer-label">@lang("pdf_net_profit_label")</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
            <td>
 | 
					            <td>
 | 
				
			||||||
                <p class="report-footer-value">{!! format_money_pdf($income - $totalExpense, $currency) !!}</p>
 | 
					                <p class="report-footer-value">{!! format_money_pdf($income - $totalExpense) !!}</p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
    </table>
 | 
					    </table>
 | 
				
			||||||
 | 
				
			|||||||
@ -137,6 +137,7 @@
 | 
				
			|||||||
        .text-center {
 | 
					        .text-center {
 | 
				
			||||||
            text-align: center;
 | 
					            text-align: center;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    </style>
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @if (App::isLocale('th'))
 | 
					    @if (App::isLocale('th'))
 | 
				
			||||||
@ -175,7 +176,7 @@
 | 
				
			|||||||
                            </td>
 | 
					                            </td>
 | 
				
			||||||
                            <td>
 | 
					                            <td>
 | 
				
			||||||
                                <p class="sales-amount">
 | 
					                                <p class="sales-amount">
 | 
				
			||||||
                            {!! format_money_pdf($invoice->base_total, $currency) !!}
 | 
					                                    {!! format_money_pdf($invoice->base_total) !!}
 | 
				
			||||||
                                </p>
 | 
					                                </p>
 | 
				
			||||||
                            </td>
 | 
					                            </td>
 | 
				
			||||||
                        </tr>
 | 
					                        </tr>
 | 
				
			||||||
@ -186,7 +187,7 @@
 | 
				
			|||||||
                <tr>
 | 
					                <tr>
 | 
				
			||||||
                    <td class="sales-total-cell">
 | 
					                    <td class="sales-total-cell">
 | 
				
			||||||
                        <p class="sales-total-amount">
 | 
					                        <p class="sales-total-amount">
 | 
				
			||||||
                        {!! format_money_pdf($customer->totalAmount, $currency) !!}
 | 
					                            {!! format_money_pdf($customer->totalAmount) !!}
 | 
				
			||||||
                        </p>
 | 
					                        </p>
 | 
				
			||||||
                    </td>
 | 
					                    </td>
 | 
				
			||||||
                </tr>
 | 
					                </tr>
 | 
				
			||||||
@ -202,7 +203,7 @@
 | 
				
			|||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
            <td>
 | 
					            <td>
 | 
				
			||||||
                <p class="report-footer-value">
 | 
					                <p class="report-footer-value">
 | 
				
			||||||
                    {!! format_money_pdf($totalAmount, $currency) !!}
 | 
					                    {!! format_money_pdf($totalAmount) !!}
 | 
				
			||||||
                </p>
 | 
					                </p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
 | 
				
			|||||||
@ -137,6 +137,7 @@
 | 
				
			|||||||
        .text-center {
 | 
					        .text-center {
 | 
				
			||||||
            text-align: center;
 | 
					            text-align: center;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    </style>
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @if (App::isLocale('th'))
 | 
					    @if (App::isLocale('th'))
 | 
				
			||||||
@ -174,7 +175,7 @@
 | 
				
			|||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                        <td>
 | 
					                        <td>
 | 
				
			||||||
                            <p class="item-sales-amount">
 | 
					                            <p class="item-sales-amount">
 | 
				
			||||||
                            {!! format_money_pdf($item->total_amount, $currency) !!}
 | 
					                                {!! format_money_pdf($item->total_amount) !!}
 | 
				
			||||||
                            </p>
 | 
					                            </p>
 | 
				
			||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                    </tr>
 | 
					                    </tr>
 | 
				
			||||||
@ -186,7 +187,7 @@
 | 
				
			|||||||
            <tr>
 | 
					            <tr>
 | 
				
			||||||
                <td class="sales-total-cell">
 | 
					                <td class="sales-total-cell">
 | 
				
			||||||
                    <p class="sales-total-amount">
 | 
					                    <p class="sales-total-amount">
 | 
				
			||||||
                        {!! format_money_pdf($totalAmount, $currency) !!}
 | 
					                        {!! format_money_pdf($totalAmount) !!}
 | 
				
			||||||
                    </p>
 | 
					                    </p>
 | 
				
			||||||
                </td>
 | 
					                </td>
 | 
				
			||||||
            </tr>
 | 
					            </tr>
 | 
				
			||||||
@ -201,7 +202,7 @@
 | 
				
			|||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
            <td>
 | 
					            <td>
 | 
				
			||||||
                <p class="report-footer-value">
 | 
					                <p class="report-footer-value">
 | 
				
			||||||
                    {!! format_money_pdf($totalAmount, $currency) !!}
 | 
					                    {!! format_money_pdf($totalAmount) !!}
 | 
				
			||||||
                </p>
 | 
					                </p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
 | 
				
			|||||||
@ -134,6 +134,7 @@
 | 
				
			|||||||
            line-height: 21px;
 | 
					            line-height: 21px;
 | 
				
			||||||
            color: #5851D8;
 | 
					            color: #5851D8;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    </style>
 | 
					    </style>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @if (App::isLocale('th'))
 | 
					    @if (App::isLocale('th'))
 | 
				
			||||||
@ -174,7 +175,7 @@
 | 
				
			|||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                        <td>
 | 
					                        <td>
 | 
				
			||||||
                            <p class="tax-amount">
 | 
					                            <p class="tax-amount">
 | 
				
			||||||
                            {!! format_money_pdf($tax->total_tax_amount, $currency) !!}
 | 
					                                {!! format_money_pdf($tax->total_tax_amount) !!}
 | 
				
			||||||
                            </p>
 | 
					                            </p>
 | 
				
			||||||
                        </td>
 | 
					                        </td>
 | 
				
			||||||
                    </tr>
 | 
					                    </tr>
 | 
				
			||||||
@ -188,7 +189,7 @@
 | 
				
			|||||||
        <tr>
 | 
					        <tr>
 | 
				
			||||||
            <td class="tax-total-cell">
 | 
					            <td class="tax-total-cell">
 | 
				
			||||||
                <p class="tax-total">
 | 
					                <p class="tax-total">
 | 
				
			||||||
                    {!! format_money_pdf($totalTaxAmount, $currency) !!}
 | 
					                    {!! format_money_pdf($totalTaxAmount) !!}
 | 
				
			||||||
                </p>
 | 
					                </p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
@ -200,7 +201,7 @@
 | 
				
			|||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
            <td>
 | 
					            <td>
 | 
				
			||||||
                <p class="report-footer-value">
 | 
					                <p class="report-footer-value">
 | 
				
			||||||
                    {!! format_money_pdf($totalTaxAmount, $currency) !!}
 | 
					                    {!! format_money_pdf($totalTaxAmount) !!}
 | 
				
			||||||
                </p>
 | 
					                </p>
 | 
				
			||||||
            </td>
 | 
					            </td>
 | 
				
			||||||
        </tr>
 | 
					        </tr>
 | 
				
			||||||
 | 
				
			|||||||
@ -19,7 +19,6 @@ module.exports = {
 | 
				
			|||||||
    './resources/scripts/**/*.js',
 | 
					    './resources/scripts/**/*.js',
 | 
				
			||||||
    './resources/scripts/**/*.vue',
 | 
					    './resources/scripts/**/*.vue',
 | 
				
			||||||
  ],
 | 
					  ],
 | 
				
			||||||
  darkMode: 'class',
 | 
					 | 
				
			||||||
  theme: {
 | 
					  theme: {
 | 
				
			||||||
    extend: {
 | 
					    extend: {
 | 
				
			||||||
      colors: {
 | 
					      colors: {
 | 
				
			||||||
@ -36,7 +35,6 @@ module.exports = {
 | 
				
			|||||||
          900: withOpacityValue('--color-primary-900'),
 | 
					          900: withOpacityValue('--color-primary-900'),
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        black: '#040405',
 | 
					        black: '#040405',
 | 
				
			||||||
        highlight: 'rgb(56, 189, 248)',
 | 
					 | 
				
			||||||
        red: colors.red,
 | 
					        red: colors.red,
 | 
				
			||||||
        teal: colors.teal,
 | 
					        teal: colors.teal,
 | 
				
			||||||
        gray: colors.slate,
 | 
					        gray: colors.slate,
 | 
				
			||||||
 | 
				
			|||||||
@ -415,31 +415,32 @@ test('update estimate with EUR currency', function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    $response = putJson('api/v1/estimates/'.$estimate->id, $estimate2);
 | 
					    $response = putJson('api/v1/estimates/'.$estimate->id, $estimate2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $estimate_assert = collect($estimate2)
 | 
					    $this->assertDatabaseHas('estimates', [
 | 
				
			||||||
        ->only([
 | 
					        'id' => $estimate['id'],
 | 
				
			||||||
            'id',
 | 
					        'template_name' => $estimate2['template_name'],
 | 
				
			||||||
            'template_name',
 | 
					        'estimate_number' => $estimate2['estimate_number'],
 | 
				
			||||||
            'estimate_number',
 | 
					        'discount_type' => $estimate2['discount_type'],
 | 
				
			||||||
            'discount_type',
 | 
					        'discount_val' => $estimate2['discount_val'],
 | 
				
			||||||
            'discount_val',
 | 
					        'sub_total' => $estimate2['sub_total'],
 | 
				
			||||||
            'sub_total',
 | 
					        'discount' => $estimate2['discount'],
 | 
				
			||||||
            'discount',
 | 
					        'customer_id' => $estimate2['customer_id'],
 | 
				
			||||||
            'customer_id',
 | 
					        'total' => $estimate2['total'],
 | 
				
			||||||
            'total',
 | 
					        'tax' => $estimate2['tax'],
 | 
				
			||||||
            'tax'
 | 
					        'exchange_rate' => $estimate2['exchange_rate'],
 | 
				
			||||||
        ])
 | 
					        'base_discount_val' => $estimate2['base_discount_val'],
 | 
				
			||||||
        ->toArray();
 | 
					        'base_sub_total' => $estimate2['base_sub_total'],
 | 
				
			||||||
 | 
					        'base_total' => $estimate2['base_total'],
 | 
				
			||||||
 | 
					        'base_tax' => $estimate2['base_tax'],
 | 
				
			||||||
 | 
					    ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $this->assertDatabaseHas('estimates', $estimate_assert);
 | 
					    $this->assertDatabaseHas('estimate_items', [
 | 
				
			||||||
 | 
					        'estimate_id' => $estimate2['items'][0]['estimate_id'],
 | 
				
			||||||
    $estimate_item_assert = collect($estimate2['items'][0])
 | 
					        'exchange_rate' => $estimate2['items'][0]['exchange_rate'],
 | 
				
			||||||
        ->only([
 | 
					        'base_price' => $estimate2['items'][0]['base_price'],
 | 
				
			||||||
            'estimate_id',
 | 
					        'base_discount_val' => $estimate2['items'][0]['base_discount_val'],
 | 
				
			||||||
            'amount'
 | 
					        'base_tax' => $estimate2['items'][0]['base_tax'],
 | 
				
			||||||
        ])
 | 
					        'base_total' => $estimate2['items'][0]['base_total'],
 | 
				
			||||||
        ->toArray();
 | 
					    ]);
 | 
				
			||||||
 | 
					 | 
				
			||||||
    $this->assertDatabaseHas('estimate_items', $estimate_item_assert);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $response->assertStatus(200);
 | 
					    $response->assertStatus(200);
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
				
			|||||||
@ -37,15 +37,13 @@ test('create expense', function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    postJson('api/v1/expenses', $expense)->assertStatus(201);
 | 
					    postJson('api/v1/expenses', $expense)->assertStatus(201);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $expense = collect($expense)
 | 
					    $this->assertDatabaseHas('expenses', [
 | 
				
			||||||
        ->only([
 | 
					        'notes' => $expense['notes'],
 | 
				
			||||||
            'notes',
 | 
					        'expense_category_id' => $expense['expense_category_id'],
 | 
				
			||||||
            'expense_category_id',
 | 
					        'amount' => $expense['amount'],
 | 
				
			||||||
            'amount'
 | 
					        'exchange_rate' => $expense['exchange_rate'],
 | 
				
			||||||
        ])
 | 
					        'base_amount' => $expense['base_amount'],
 | 
				
			||||||
        ->toArray();
 | 
					    ]);
 | 
				
			||||||
 | 
					 | 
				
			||||||
    $this->assertDatabaseHas('expenses', $expense);
 | 
					 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
test('store validates using a form request', function () {
 | 
					test('store validates using a form request', function () {
 | 
				
			||||||
@ -148,13 +146,11 @@ test('update expense with EUR currency', function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    putJson('api/v1/expenses/'.$expense->id, $expense2)->assertOk();
 | 
					    putJson('api/v1/expenses/'.$expense->id, $expense2)->assertOk();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $expense2 = collect($expense2)
 | 
					    $this->assertDatabaseHas('expenses', [
 | 
				
			||||||
        ->only([
 | 
					        'id' => $expense->id,
 | 
				
			||||||
            'id',
 | 
					        'expense_category_id' => $expense2['expense_category_id'],
 | 
				
			||||||
            'expense_category_id',
 | 
					        'amount' => $expense2['amount'],
 | 
				
			||||||
            'amount'
 | 
					        'exchange_rate' => $expense2['exchange_rate'],
 | 
				
			||||||
        ])
 | 
					        'base_amount' => $expense2['base_amount'],
 | 
				
			||||||
        ->toArray();
 | 
					    ]);
 | 
				
			||||||
 | 
					 | 
				
			||||||
    $this->assertDatabaseHas('expenses', $expense2);
 | 
					 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
				
			|||||||
@ -9,6 +9,7 @@ use Crater\Models\Tax;
 | 
				
			|||||||
use Crater\Models\User;
 | 
					use Crater\Models\User;
 | 
				
			||||||
use Illuminate\Support\Facades\Artisan;
 | 
					use Illuminate\Support\Facades\Artisan;
 | 
				
			||||||
use Laravel\Sanctum\Sanctum;
 | 
					use Laravel\Sanctum\Sanctum;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use function Pest\Laravel\getJson;
 | 
					use function Pest\Laravel\getJson;
 | 
				
			||||||
use function Pest\Laravel\postJson;
 | 
					use function Pest\Laravel\postJson;
 | 
				
			||||||
use function Pest\Laravel\putJson;
 | 
					use function Pest\Laravel\putJson;
 | 
				
			||||||
@ -430,36 +431,31 @@ test('update invoice with EUR currency', function () {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk();
 | 
					    putJson('api/v1/invoices/'.$invoice->id, $invoice2)->assertOk();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $invoice_assert = collect($invoice2)
 | 
					    $this->assertDatabaseHas('invoices', [
 | 
				
			||||||
        ->only([
 | 
					        'id' => $invoice['id'],
 | 
				
			||||||
            'invoice_number',
 | 
					        'invoice_number' => $invoice2['invoice_number'],
 | 
				
			||||||
            'template_name',
 | 
					        'sub_total' => $invoice2['sub_total'],
 | 
				
			||||||
            'sub_total',
 | 
					        'total' => $invoice2['total'],
 | 
				
			||||||
            'total',
 | 
					        'tax' => $invoice2['tax'],
 | 
				
			||||||
            'tax',
 | 
					        'discount' => $invoice2['discount'],
 | 
				
			||||||
            'discount',
 | 
					        'customer_id' => $invoice2['customer_id'],
 | 
				
			||||||
            'customer_id',
 | 
					        'template_name' => $invoice2['template_name'],
 | 
				
			||||||
        ])
 | 
					        'exchange_rate' => $invoice2['exchange_rate'],
 | 
				
			||||||
        ->toArray();
 | 
					        'base_total' => $invoice2['base_total'],
 | 
				
			||||||
 | 
					    ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $this->assertDatabaseHas('invoices', $invoice_assert);
 | 
					    $this->assertDatabaseHas('invoice_items', [
 | 
				
			||||||
 | 
					        'invoice_id' => $invoice2['items'][0]['invoice_id'],
 | 
				
			||||||
 | 
					        'item_id' => $invoice2['items'][0]['item_id'],
 | 
				
			||||||
 | 
					        'name' => $invoice2['items'][0]['name'],
 | 
				
			||||||
 | 
					        'exchange_rate' => $invoice2['items'][0]['exchange_rate'],
 | 
				
			||||||
 | 
					        'base_price' => $invoice2['items'][0]['base_price'],
 | 
				
			||||||
 | 
					        'base_total' => $invoice2['items'][0]['base_total'],
 | 
				
			||||||
 | 
					    ]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    $invoice_item_assert = collect($invoice2['items'][0])
 | 
					    $this->assertDatabaseHas('taxes', [
 | 
				
			||||||
        ->only([
 | 
					        'amount' => $invoice2['taxes'][0]['amount'],
 | 
				
			||||||
            'invoice_id',
 | 
					        'name' => $invoice2['taxes'][0]['name'],
 | 
				
			||||||
            'item_id',
 | 
					        'base_amount' => $invoice2['taxes'][0]['base_amount'],
 | 
				
			||||||
            'name',
 | 
					    ]);
 | 
				
			||||||
        ])
 | 
					 | 
				
			||||||
        ->toArray();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $this->assertDatabaseHas('invoice_items', $invoice_item_assert);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $invoice_tax_assert = collect($invoice2['taxes'][0])
 | 
					 | 
				
			||||||
        ->only([
 | 
					 | 
				
			||||||
            'name',
 | 
					 | 
				
			||||||
            'amount'
 | 
					 | 
				
			||||||
        ])
 | 
					 | 
				
			||||||
        ->toArray();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    $this->assertDatabaseHas('taxes', $invoice_tax_assert);
 | 
					 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
				
			|||||||
@ -1,40 +0,0 @@
 | 
				
			|||||||
APP_ENV=production
 | 
					 | 
				
			||||||
APP_KEY=base64:kgk/4DW1vEVy7aEvet5FPp5un6PIGe/so8H0mvoUtW0=
 | 
					 | 
				
			||||||
APP_DEBUG=true
 | 
					 | 
				
			||||||
APP_LOG_LEVEL=debug
 | 
					 | 
				
			||||||
APP_URL=http://crater.test
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
DB_CONNECTION=mysql
 | 
					 | 
				
			||||||
DB_HOST=127.0.0.1
 | 
					 | 
				
			||||||
DB_PORT=3306
 | 
					 | 
				
			||||||
DB_DATABASE=crater
 | 
					 | 
				
			||||||
DB_USERNAME=crater
 | 
					 | 
				
			||||||
DB_PASSWORD=crater
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
BROADCAST_DRIVER=log
 | 
					 | 
				
			||||||
CACHE_DRIVER=file
 | 
					 | 
				
			||||||
QUEUE_DRIVER=sync
 | 
					 | 
				
			||||||
SESSION_DRIVER=cookie
 | 
					 | 
				
			||||||
SESSION_LIFETIME=1440
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
REDIS_HOST=127.0.0.1
 | 
					 | 
				
			||||||
REDIS_PASSWORD=null
 | 
					 | 
				
			||||||
REDIS_PORT=6379
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
MAIL_DRIVER=smtp
 | 
					 | 
				
			||||||
MAIL_HOST=
 | 
					 | 
				
			||||||
MAIL_PORT=
 | 
					 | 
				
			||||||
MAIL_USERNAME=
 | 
					 | 
				
			||||||
MAIL_PASSWORD=
 | 
					 | 
				
			||||||
MAIL_ENCRYPTION=
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
PUSHER_APP_ID=
 | 
					 | 
				
			||||||
PUSHER_KEY=
 | 
					 | 
				
			||||||
PUSHER_SECRET=
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
SANCTUM_STATEFUL_DOMAINS=crater.test
 | 
					 | 
				
			||||||
SESSION_DOMAIN=crater.test
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
TRUSTED_PROXIES="*"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CRON_JOB_AUTH_TOKEN=""
 | 
					 | 
				
			||||||
@ -1,48 +0,0 @@
 | 
				
			|||||||
FROM php:8.1-fpm
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Install system dependencies
 | 
					 | 
				
			||||||
RUN apt-get update && apt-get install -y \
 | 
					 | 
				
			||||||
    git \
 | 
					 | 
				
			||||||
    curl \
 | 
					 | 
				
			||||||
    libpng-dev \
 | 
					 | 
				
			||||||
    libonig-dev \
 | 
					 | 
				
			||||||
    libxml2-dev \
 | 
					 | 
				
			||||||
    zip \
 | 
					 | 
				
			||||||
    unzip \
 | 
					 | 
				
			||||||
    libzip-dev \
 | 
					 | 
				
			||||||
    libmagickwand-dev \
 | 
					 | 
				
			||||||
    mariadb-client
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Clear cache
 | 
					 | 
				
			||||||
RUN apt-get clean && rm -rf /var/lib/apt/lists/*
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN pecl install imagick \
 | 
					 | 
				
			||||||
    && docker-php-ext-enable imagick
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Install PHP extensions
 | 
					 | 
				
			||||||
RUN docker-php-ext-install pdo_mysql mbstring zip exif pcntl bcmath gd
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Get latest Composer
 | 
					 | 
				
			||||||
COPY --from=composer:latest /usr/bin/composer /usr/bin/composer
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Create system user to run Composer and Artisan Commands
 | 
					 | 
				
			||||||
RUN useradd -G www-data,root -u 1000 -d /home/crater-user crater-user
 | 
					 | 
				
			||||||
RUN mkdir -p /home/crater-user/.composer && \
 | 
					 | 
				
			||||||
    chown -R crater-user:crater-user /home/crater-user
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Mounted volumes
 | 
					 | 
				
			||||||
COPY ./ /var/www
 | 
					 | 
				
			||||||
COPY ./docker-compose/php/uploads.ini /usr/local/etc/php/conf.d/uploads.ini
 | 
					 | 
				
			||||||
COPY ./uffizzi/.env.example /var/www/.env
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Set working directory
 | 
					 | 
				
			||||||
WORKDIR /var/www
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN chown -R crater-user:crater-user ./
 | 
					 | 
				
			||||||
RUN chmod -R 775 composer.json composer.lock \ 
 | 
					 | 
				
			||||||
        composer.lock storage/framework/ \
 | 
					 | 
				
			||||||
        storage/logs/ bootstrap/cache/ /home/crater-user/.composer
 | 
					 | 
				
			||||||
RUN chown -R $(whoami):$(whoami) /var/log/
 | 
					 | 
				
			||||||
RUN chmod -R 775 /var/log
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
USER crater-user
 | 
					 | 
				
			||||||
@ -1,68 +0,0 @@
 | 
				
			|||||||
FROM php:8.1-fpm as build
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Install system dependencies
 | 
					 | 
				
			||||||
RUN apt-get update && apt-get install -y \
 | 
					 | 
				
			||||||
    git \
 | 
					 | 
				
			||||||
    curl \
 | 
					 | 
				
			||||||
    libpng-dev \
 | 
					 | 
				
			||||||
    libonig-dev \
 | 
					 | 
				
			||||||
    libxml2-dev \
 | 
					 | 
				
			||||||
    zip \
 | 
					 | 
				
			||||||
    unzip \
 | 
					 | 
				
			||||||
    libzip-dev \
 | 
					 | 
				
			||||||
    libmagickwand-dev \
 | 
					 | 
				
			||||||
    mariadb-client
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Clear cache
 | 
					 | 
				
			||||||
RUN apt-get clean && rm -rf /var/lib/apt/lists/*
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN pecl install imagick \
 | 
					 | 
				
			||||||
    && docker-php-ext-enable imagick
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Install PHP extensions
 | 
					 | 
				
			||||||
RUN docker-php-ext-install pdo_mysql mbstring zip exif pcntl bcmath gd
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Get latest Composer
 | 
					 | 
				
			||||||
COPY --from=composer:latest /usr/bin/composer /usr/bin/composer
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Create system user to run Composer and Artisan Commands
 | 
					 | 
				
			||||||
RUN useradd -G www-data,root -u 1000 -d /home/crater-user crater-user
 | 
					 | 
				
			||||||
RUN mkdir -p /home/crater-user/.composer && \
 | 
					 | 
				
			||||||
    chown -R crater-user:crater-user /home/crater-user
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Mounted volumes
 | 
					 | 
				
			||||||
COPY ./ /var/www
 | 
					 | 
				
			||||||
COPY ./docker-compose/php/uploads.ini /usr/local/etc/php/conf.d/uploads.ini
 | 
					 | 
				
			||||||
COPY ./uffizzi/.env.example /var/www/.env
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Set working directory
 | 
					 | 
				
			||||||
WORKDIR /var/www
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN chown -R crater-user:crater-user ./
 | 
					 | 
				
			||||||
RUN chmod -R 775 composer.json composer.lock \ 
 | 
					 | 
				
			||||||
        composer.lock storage/framework/ \
 | 
					 | 
				
			||||||
        storage/logs/ bootstrap/cache/ /home/crater-user/.composer
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN composer config --no-plugins allow-plugins.pestphp/pest-plugin true && \
 | 
					 | 
				
			||||||
    composer install --no-interaction --prefer-dist --optimize-autoloader && \
 | 
					 | 
				
			||||||
    php artisan storage:link || true && \
 | 
					 | 
				
			||||||
    php artisan key:generate
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
FROM php:8.0-fpm-alpine
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN apk add --no-cache \
 | 
					 | 
				
			||||||
    php8-bcmath
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN docker-php-ext-install pdo pdo_mysql bcmath
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
COPY docker-compose/crontab /etc/crontabs/root
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Mounted volumes
 | 
					 | 
				
			||||||
COPY --from=build /var/www /var/www
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN chown -R $(whoami):$(whoami) /var/www/
 | 
					 | 
				
			||||||
RUN chmod -R 775 /var/www/
 | 
					 | 
				
			||||||
RUN chown -R $(whoami):$(whoami) /var/log/
 | 
					 | 
				
			||||||
RUN chmod -R 775 /var/log/
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CMD ["crond", "-f"]
 | 
					 | 
				
			||||||
@ -1,58 +0,0 @@
 | 
				
			|||||||
version: '3'
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
x-uffizzi:
 | 
					 | 
				
			||||||
  ingress:
 | 
					 | 
				
			||||||
    service: nginx
 | 
					 | 
				
			||||||
    port: 80
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
services:
 | 
					 | 
				
			||||||
  app:
 | 
					 | 
				
			||||||
    image: "${APP_IMAGE}"
 | 
					 | 
				
			||||||
    restart: unless-stopped
 | 
					 | 
				
			||||||
    working_dir: /var/www/
 | 
					 | 
				
			||||||
    command: ["-c","
 | 
					 | 
				
			||||||
            composer config --no-plugins allow-plugins.pestphp/pest-plugin true && 
 | 
					 | 
				
			||||||
            composer install --no-interaction --prefer-dist --optimize-autoloader && 
 | 
					 | 
				
			||||||
            php artisan storage:link || true && 
 | 
					 | 
				
			||||||
            php artisan key:generate --force && 
 | 
					 | 
				
			||||||
            php-fpm",
 | 
					 | 
				
			||||||
            ]
 | 
					 | 
				
			||||||
    entrypoint: /bin/sh
 | 
					 | 
				
			||||||
    depends_on:
 | 
					 | 
				
			||||||
      - db
 | 
					 | 
				
			||||||
    deploy:
 | 
					 | 
				
			||||||
      resources:
 | 
					 | 
				
			||||||
        limits:
 | 
					 | 
				
			||||||
          memory: 1000m
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  db:
 | 
					 | 
				
			||||||
    image: mariadb
 | 
					 | 
				
			||||||
    restart: always
 | 
					 | 
				
			||||||
    environment:
 | 
					 | 
				
			||||||
      MYSQL_USER: crater
 | 
					 | 
				
			||||||
      MYSQL_PASSWORD: crater
 | 
					 | 
				
			||||||
      MYSQL_DATABASE: crater
 | 
					 | 
				
			||||||
      MYSQL_ROOT_PASSWORD: crater
 | 
					 | 
				
			||||||
    ports:
 | 
					 | 
				
			||||||
      - '33006:3306'
 | 
					 | 
				
			||||||
    deploy:
 | 
					 | 
				
			||||||
      resources:
 | 
					 | 
				
			||||||
        limits:
 | 
					 | 
				
			||||||
          memory: 500m
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  nginx:
 | 
					 | 
				
			||||||
    image: "${NGINX_IMAGE}"
 | 
					 | 
				
			||||||
    restart: unless-stopped
 | 
					 | 
				
			||||||
    ports:
 | 
					 | 
				
			||||||
      - 80:80
 | 
					 | 
				
			||||||
    depends_on:
 | 
					 | 
				
			||||||
      - app
 | 
					 | 
				
			||||||
    resources:
 | 
					 | 
				
			||||||
      limits:
 | 
					 | 
				
			||||||
        memory: 500m
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  cron:
 | 
					 | 
				
			||||||
    image: "${CROND_IMAGE}"
 | 
					 | 
				
			||||||
    restart: always
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
@ -1,7 +0,0 @@
 | 
				
			|||||||
FROM nginx:1.17-alpine
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
RUN rm /etc/nginx/conf.d/default.conf
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
COPY ./ /var/www
 | 
					 | 
				
			||||||
COPY ./uffizzi/nginx/nginx /etc/nginx/conf.d/
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
@ -1,22 +0,0 @@
 | 
				
			|||||||
server {
 | 
					 | 
				
			||||||
    client_max_body_size 64M;
 | 
					 | 
				
			||||||
    listen 80;
 | 
					 | 
				
			||||||
    index index.php index.html;
 | 
					 | 
				
			||||||
    error_log  /var/log/nginx/error.log;
 | 
					 | 
				
			||||||
    access_log /var/log/nginx/access.log;
 | 
					 | 
				
			||||||
    root /var/www/public;
 | 
					 | 
				
			||||||
    location ~ \.php$ {
 | 
					 | 
				
			||||||
        try_files $uri =404;
 | 
					 | 
				
			||||||
        fastcgi_split_path_info ^(.+\.php)(/.+)$;
 | 
					 | 
				
			||||||
        fastcgi_pass localhost:9000;
 | 
					 | 
				
			||||||
        fastcgi_index index.php;
 | 
					 | 
				
			||||||
        include fastcgi_params;
 | 
					 | 
				
			||||||
        fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
 | 
					 | 
				
			||||||
        fastcgi_param PATH_INFO $fastcgi_path_info;
 | 
					 | 
				
			||||||
        fastcgi_read_timeout 300;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    location / {
 | 
					 | 
				
			||||||
        try_files $uri $uri/ /index.php?$query_string;
 | 
					 | 
				
			||||||
        gzip_static on;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
		Reference in New Issue
	
	Block a user